Rengi gölgelere ayırma işlevi.

 

Bir ay boyunca, rengi gölgelere ayırma sorununu inatla çözdüm. Görevin zor olduğu ortaya çıktı. Çalışmanın temeli olarak Windows renk paletini aldım ( grafik özelliklerindeki ). Uzun bir çalışmadan sonra hala kalıplar buldum. Ve işte nasıl:

  • İlk olarak, rengi en yüksek, orta ve en düşük olanı belirleyerek üç ana bileşene ayırdım.
  • Ayrıca, bileşenlerin değerleri üzerinden grafikler oluşturmaya ve çizgiler çizmeye başladım.
  • Kaydırıcıyı sürüklerken paletteki sayıların değişimini gözlemleyerek, belli bir anda sayıların değişim hızı değiştiği için çizgilerin çıkış açısında bir kırılma olduğunu fark ettim.
  • Çizgilerin kırılma eksenini grafiğin merkezine ayarladım ve her çizginin, her birinin kendi yükselme açısına sahip iki parçadan oluştuğunu gördüm.
  • Ayrıca, renk paleti ile deneyler yaparak, en yüksek bileşenin maksimum çıkış açısı olduğunu fark ettim. İlk başta 67.5 derece olduğunu düşündüm. Ancak, uygulama 63,5 dereceye eşit olduğunu göstermiştir.
  • Uzun bir süre, renk bileşenlerinin doğru parçalarını nasıl doğru çizeceğimi çözemedim. Birçok bilinmeyen vardı. Ama asıl mesele, grafikteki orijinal rengin koordinatını nasıl bulacağınızdır?
  • Paleti denemeye devam ederken, renk değerini belirli bir sayı ile değiştirirken kaydırıcının belirli bir mesafe hareket ettiğini fark ettim. Yavaş yavaş, kaydırıcının kaydırma mesafesinin küçük bileşenin değerinin yarısına eşit olduğunu fark ettim.
  • Maksimum yükselme açısı doğrusu üzerinde en yüksek bileşenin koordinatını bulup bu noktaya en alttaki bileşenin değerinin yarısını eklersem, grafikte orijinal rengin koordinatını bulacağımı varsaydım. Uygulama, varsayımın doğru olduğunu kanıtladı.
  • Orijinal rengin koordinatı ve kırılma ekseni göz önüne alındığında, segmentlerin her birinin açılarını hesaplayabilir ve çizgisi boyunca her bileşen için değerler alabilirim. Bunu yapmak için okul trigonometrisini kullandım.
  • Kağıt üzerindeki hesaplamaları kontrol ettikten sonra fonksiyonu yazmaya başladım. Şimdi bunu herkesle paylaşmak istiyorum.
 //+------------------------------------------------------------------+
//|                                                Full Gradient.mqh |
//|                                                      Peter Konow |
//|                                             https://www.mql5.com |
//+------------------------------------------------------------------+
void Диапазон_оттенков( color _Цвет, string &Все_оттенки[ 256 ])
{
 color R,G,B,q;
 //------------------------------------------------------
 string Этот_цвет;
 double Тангенс_угла_старшего_треугольника_1,      
        Тангенс_угла_среднего_треугольника_1,                  
        Тангенс_угла_младшего_треугольника_1,                       
        Значение_в_точке_преломления_старшей_компоненты,  
        Значение_в_точке_преломления_средней_компоненты,
        Значение_в_точке_преломления_младшей_компоненты,    
        Тангенс_угла_старшего_треугольника_2,
        Тангенс_угла_среднего_треугольника_2,    
        Тангенс_угла_младшего_треугольника_2;
 //------------------------------------------------------
 double pi = 3.1415926536 ,
        Comp_1,Comp_2,Comp_3,
         //-----------------------------------------------
        Первая_компонента,
        Вторая_компонента,
        Третья_компонента,
         //-----------------------------------------------
        Исходный_R = GetR(_Цвет), 
        Исходный_G = GetG(_Цвет),  
        Исходный_B = GetB(_Цвет),  
         //-----------------------------------------------
        Старшая_компонента         = Нужная_компонента(Исходный_R,Исходный_G,Исходный_B, 0 ), 
        Средняя_компонента         = Нужная_компонента(Исходный_R,Исходный_G,Исходный_B, 1 ), 
        Младшая_компонента         = Нужная_компонента(Исходный_R,Исходный_G,Исходный_B, 2 ), 
         //-----------------------------------------------
        Координата_исходного_цвета = Старшая_компонента/ tan (( 63.43989 *pi)/ 180 ) + Младшая_компонента/ 2 ; 
         //-----------------------------------------------
        
 //-----------------------------------------------
 if (Старшая_компонента == Исходный_R)R = Старшая_компонента;
 if (Старшая_компонента == Исходный_G)G = Старшая_компонента; 
 if (Старшая_компонента == Исходный_B)B = Старшая_компонента;     
 //------------------------
 if (Средняя_компонента == Исходный_R)R = Средняя_компонента;
 if (Средняя_компонента == Исходный_G)G = Средняя_компонента; 
 if (Средняя_компонента == Исходный_B)B = Средняя_компонента; 
 //------------------------
 if (Младшая_компонента == Исходный_R)R = Младшая_компонента;
 if (Младшая_компонента == Исходный_G)G = Младшая_компонента; 
 if (Младшая_компонента == Исходный_B)B = Младшая_компонента;     
 //==========================================================================================
 if (Координата_исходного_цвета <= 127 )
   {
    Тангенс_угла_старшего_треугольника_1  = Старшая_компонента/Координата_исходного_цвета;
    Тангенс_угла_среднего_треугольника_1  = Средняя_компонента/Координата_исходного_цвета;     
    Тангенс_угла_младшего_треугольника_1  = Младшая_компонента/Координата_исходного_цвета;
     //-----------------------------------------------
    Значение_в_точке_преломления_старшей_компоненты  = Тангенс_угла_старшего_треугольника_1* 128 ;
    Значение_в_точке_преломления_средней_компоненты  = Тангенс_угла_среднего_треугольника_1* 128 ;
    Значение_в_точке_преломления_младшей_компоненты  = Тангенс_угла_младшего_треугольника_1* 128 ;    
     //-----------------------------------------------    
    Тангенс_угла_старшего_треугольника_2  = ( 255 - Значение_в_точке_преломления_старшей_компоненты)/ 128 ;
    Тангенс_угла_среднего_треугольника_2  = ( 255 - Значение_в_точке_преломления_средней_компоненты)/ 128 ;       
    Тангенс_угла_младшего_треугольника_2  = ( 255 - Значение_в_точке_преломления_младшей_компоненты)/ 128 ;
     //-----------------------------------------------    
     for ( int a1 = 0 ; a1 < 128 ; a1++)
      {
       Comp_1 = Тангенс_угла_старшего_треугольника_1*a1;
       Comp_2 = Тангенс_угла_среднего_треугольника_1*a1;
       Comp_3 = Тангенс_угла_младшего_треугольника_1*a1;
       //---------------------------------------------------
       if (Comp_1 > 255 )Comp_1 = 255 ;
       if (Comp_1 > 255 )Comp_2 = 255 ;
       if (Comp_1 > 255 )Comp_3 = 255 ;
       //---------------------------------------------------
       if (R == Старшая_компонента)Первая_компонента = Comp_1;
       if (R == Средняя_компонента)Первая_компонента = Comp_2;
       if (R == Младшая_компонента)Первая_компонента = Comp_3;
       //---------------------------------------------------
       if (G == Старшая_компонента)Вторая_компонента = Comp_1;
       if (G == Средняя_компонента)Вторая_компонента = Comp_2;
       if (G == Младшая_компонента)Вторая_компонента = Comp_3;
       //---------------------------------------------------
       if (B == Старшая_компонента)Третья_компонента = Comp_1;
       if (B == Средняя_компонента)Третья_компонента = Comp_2;
       if (B == Младшая_компонента)Третья_компонента = Comp_3;
       //---------------------------------------------------
       Этот_цвет = ( string ) MathRound (Первая_компонента) + "," + ( string ) MathRound (Вторая_компонента) + "," + ( string ) MathRound (Третья_компонента);
       //---------------------------------------------------------------------------    
       Все_оттенки[a1] = Этот_цвет; 
       //---------------------------------------------------------------------------
      }
     //------------------------------------------------------------------------------
     for ( int a2 = 255 ; a2 >= a1; a2--)
      {
       Comp_1 = 255 - Тангенс_угла_старшего_треугольника_2*q;
       Comp_2 = 255 - Тангенс_угла_среднего_треугольника_2*q;
       Comp_3 = 255 - Тангенс_угла_младшего_треугольника_2*q;
       //---------------------------------------------------
       if (Comp_1 > 255 )Comp_1 = 255 ;
       if (Comp_1 > 255 )Comp_2 = 255 ;
       if (Comp_1 > 255 )Comp_3 = 255 ;
       //---------------------------------------------------       
       if (R == Старшая_компонента)Первая_компонента = Comp_1;
       if (R == Средняя_компонента)Первая_компонента = Comp_2;
       if (R == Младшая_компонента)Первая_компонента = Comp_3;
       //---------------------------------------------------
       if (G == Старшая_компонента)Вторая_компонента = Comp_1;
       if (G == Средняя_компонента)Вторая_компонента = Comp_2;
       if (G == Младшая_компонента)Вторая_компонента = Comp_3;
       //---------------------------------------------------
       if (B == Старшая_компонента)Третья_компонента = Comp_1;
       if (B == Средняя_компонента)Третья_компонента = Comp_2;
       if (B == Младшая_компонента)Третья_компонента = Comp_3;
       //---------------------------------------------------
       Этот_цвет = ( string ) MathRound (Первая_компонента) + "," + ( string ) MathRound (Вторая_компонента) + "," + ( string ) MathRound (Третья_компонента);
       //---------------------------------------------------------------------------    
       Все_оттенки[a2] = Этот_цвет; 
       //---------------------------------------------------------------------------
       q++;
      }
  } 
 //------------------------------------------------------------------------------
 if (Координата_исходного_цвета > 127 )
   {
    Тангенс_угла_старшего_треугольника_1  = ( 255 - Старшая_компонента)/( 255 - Координата_исходного_цвета);
    Тангенс_угла_среднего_треугольника_1  = ( 255 - Средняя_компонента)/( 255 - Координата_исходного_цвета);       
    Тангенс_угла_младшего_треугольника_1  = ( 255 - Младшая_компонента)/( 255 - Координата_исходного_цвета);
     //-----------------------------------------------
    Значение_в_точке_преломления_старшей_компоненты  = 255 - (Тангенс_угла_старшего_треугольника_1* 128 );
    Значение_в_точке_преломления_средней_компоненты  = 255 - (Тангенс_угла_среднего_треугольника_1* 128 );
    Значение_в_точке_преломления_младшей_компоненты  = 255 - (Тангенс_угла_младшего_треугольника_1* 128 );    
     //-----------------------------------------------    
    Тангенс_угла_старшего_треугольника_2  = Значение_в_точке_преломления_старшей_компоненты/ 128 ;
    Тангенс_угла_среднего_треугольника_2  = Значение_в_точке_преломления_средней_компоненты/ 128 ;      
    Тангенс_угла_младшего_треугольника_2  = Значение_в_точке_преломления_младшей_компоненты/ 128 ;
     //-----------------------------------------------    
     for ( int b1 = 0 ; b1 < 128 ; b1++)
      {
       Comp_1 = Тангенс_угла_старшего_треугольника_2*b1;
       Comp_2 = Тангенс_угла_среднего_треугольника_2*b1;
       Comp_3 = Тангенс_угла_младшего_треугольника_2*b1;
       //---------------------------------------------------
       if (Comp_1 > 255 )Comp_1 = 255 ;
       if (Comp_1 > 255 )Comp_2 = 255 ;
       if (Comp_1 > 255 )Comp_3 = 255 ;
       //---------------------------------------------------
       if (R == Старшая_компонента)Первая_компонента = Comp_1;
       if (R == Средняя_компонента)Первая_компонента = Comp_2;
       if (R == Младшая_компонента)Первая_компонента = Comp_3;
       //---------------------------------------------------
       if (G == Старшая_компонента)Вторая_компонента = Comp_1;
       if (G == Средняя_компонента)Вторая_компонента = Comp_2;
       if (G == Младшая_компонента)Вторая_компонента = Comp_3;
       //---------------------------------------------------
       if (B == Старшая_компонента)Третья_компонента = Comp_1;
       if (B == Средняя_компонента)Третья_компонента = Comp_2;
       if (B == Младшая_компонента)Третья_компонента = Comp_3;
       //---------------------------------------------------
       Этот_цвет = ( string ) MathRound (Первая_компонента) + "," + ( string ) MathRound (Вторая_компонента) + "," + ( string ) MathRound (Третья_компонента);
       //---------------------------------------------------------------------------    
       Все_оттенки[b1] = Этот_цвет; 
       //---------------------------------------------------------------------------
      }
     //------------------------------------------------------------------------------
     for ( int b2 = 255 ; b2 >= b1; b2--)
      {
       Comp_1 = 255 - Тангенс_угла_старшего_треугольника_1*q;
       Comp_2 = 255 - Тангенс_угла_среднего_треугольника_1*q;
       Comp_3 = 255 - Тангенс_угла_младшего_треугольника_1*q;
       //---------------------------------------------------
       if (Comp_1 > 255 )Comp_1 = 255 ;
       if (Comp_1 > 255 )Comp_2 = 255 ;
       if (Comp_1 > 255 )Comp_3 = 255 ;
       //---------------------------------------------------       
       if (R == Старшая_компонента)Первая_компонента = Comp_1;
       if (R == Средняя_компонента)Первая_компонента = Comp_2;
       if (R == Младшая_компонента)Первая_компонента = Comp_3;
       //---------------------------------------------------
       if (G == Старшая_компонента)Вторая_компонента = Comp_1;
       if (G == Средняя_компонента)Вторая_компонента = Comp_2;
       if (G == Младшая_компонента)Вторая_компонента = Comp_3;
       //---------------------------------------------------
       if (B == Старшая_компонента)Третья_компонента = Comp_1;
       if (B == Средняя_компонента)Третья_компонента = Comp_2;
       if (B == Младшая_компонента)Третья_компонента = Comp_3;
       //---------------------------------------------------
       Этот_цвет = ( string ) MathRound (Первая_компонента) + "," + ( string ) MathRound (Вторая_компонента) + "," + ( string ) MathRound (Третья_компонента);
       //---------------------------------------------------------------------------    
       Все_оттенки[b2] = Этот_цвет; 
       //---------------------------------------------------------------------------
       q++;
      }
   }   
 //------------------------------------------------------------------------------  
}
//+------------------------------------------------------------------+
//| Получение значения компонента R                                  |
//+------------------------------------------------------------------+
double GetR( const color aColor)
  {
   return (aColor& 0xff );
  }
//+------------------------------------------------------------------+
//| Получение значения компонента G                                  |
//+------------------------------------------------------------------+
double GetG( const color aColor)
  {
   return ((aColor>> 8 )& 0xff );
  }
//+------------------------------------------------------------------+
//| Получение значения компонента B                                  |
//+------------------------------------------------------------------+
double GetB( const color aColor)
  {
   return ((aColor>> 16 )& 0xff );
  }
//--------------------------------------------------------------------
double Нужная_компонента( double C1, double C2, double C3, int Index)
{
 double Components[ 3 ]; 
 //----------------------------------------------
 Components[ 0 ] = C1;
 Components[ 1 ] = C2;
 Components[ 2 ] = C3;
 //----------------------------------------------
 ArraySort (Components, WHOLE_ARRAY , 0 , MODE_DESCEND );
 //----------------------------------------------
 return (Components[Index]);
}

 

İşte test için komut dosyası:

 void OnStart ()
  {
   color Main_color     = C'213,0,0' ; //здесь ставите нужный цвет. Компилируете. Тот же цвет ставите в цветовой палитре виндоус. Сравниваете цвета.
   string Gradient[ 256 ];
   //------------------------
   Диапазон_оттенков(Main_color,Gradient);
   //------------------------
   for ( int a1 = 0 ; a1 < 256 ; a1++) Alert ( __FUNCTION__ , "  Gradient[" ,a1, "]  " ,Gradient[a1]);
  }
Dosyalar:
 
Реter Konow :

İşte test için komut dosyası:

Sayesinde! Yorumlarla İngilizce kod yazıp bu kodları kütüphaneye göndermek güzel olurdu...

 
Bu görevin CCanvas sınıfı kullanılarak nasıl çözüldüğü de çok ilginç. Orijinal rengin tüm tonlarını nasıl aşabilirsin?
 
Vladimir Pastushak :

Sayesinde! Yorumlarla İngilizce kod yazıp bu kodları kütüphaneye göndermek güzel olurdu...

Kabul ediyorum. Boş bir zamanımda tercüme edeceğim. Kod tabanını mı kastediyorsunuz?

 
Реter Konow :

Kabul ediyorum. Boş bir zamanımda tercüme edeceğim. Kod tabanını mı kastediyorsunuz?

Kod tabanında bu tür kodların eklendiği ve güncellendiği bir kitaplık bölümü bulunmaktadır.

 
Vladimir Pastushak :

Kod tabanında bu tür kodların eklendiği ve güncellendiği bir kitaplık bölümü bulunmaktadır.

Apaçık.

 
Vladimir Pastushak :

Sayesinde! Yorumlarla İngilizce kod yazıp bu kodları kütüphaneye göndermek güzel olurdu...

CodeBade için elbette uluslararası bir dilde yazmanız gerekiyor. Ama neden Rusça konuşulan bir forumda bu konuda ısrar ???

Şahsen, bu uluslararası benim için iğrenç ... çünkü onu hiç bilmiyorum ve anadili Rusça benim için çok daha açık.

 

Kolorimetride hatırladığım kadarıyla birçok farklı sistem var.

Belki ihtiyaçları karşılayan hazır bir sistem ve bunlar arasında geçiş formülleri vardır.

 
Georgiy Merts :

Kolorimetride hatırladığım kadarıyla birçok farklı sistem var.

Belki ihtiyaçları karşılayan hazır bir sistem ve bunlar arasında geçiş formülleri vardır.

Bir işlevim var. Bir renk gönderdi - tüm tonları aldım. Ayrıca, herhangi bir gradyan, dizi içinde basitçe döngü yapılarak çizilebilir.

 

Biraz sonra bir değişiklik yaptım. Başlangıçta, işlev geçersizdir. Bir değer döndürmez. Ama ondan sonra onu bir int türü yaptım ve aşağıya ekledim

 return (Координата_исходного_цвета);

Böylece, renk dizisine ek olarak, fonksiyon, kendisine gönderilen orijinal rengin bulunduğu hücrenin numarasını döndürmeye başladı. Bu, degradeyi çizmeyi kolaylaştırır. Orijinal renkten dizide sola veya sağa hareket edebilirsiniz.