cc.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561
  1. #include<assert.h>
  2. #include<math.h>
  3. #include"cc.h"
  4. using namespace OBJREC;
  5. using namespace std;
  6. /*
  7. * Tristimulus reference values
  8. */
  9. const double _ccTristimulusValues[ccNTristimulus][3] =
  10. {
  11. /* 2 degrees */
  12. /*ccTA_2*/{109.850, 100.000, 35.585},
  13. /*ccTC_2*/{98.074, 100.000, 118.232},
  14. /* D50 */ {96.422, 100.000, 82.521},
  15. /* D55 */ {95.682, 100.000, 92.149},
  16. /* D65 */ {95.047, 100.000, 108.883},
  17. /* D75 */ {94.972, 100.000, 122.638},
  18. /* F2 */ {99.187, 100.000, 67.395},
  19. /* F7 */ {95.044, 100.000, 108.755},
  20. /* F11 */ {100.966, 100.000, 64.370},
  21. /* 10 degrees */
  22. /* A */ {111.144, 100.000, 35.200},
  23. /* C */ {97.285, 100.000, 116.145},
  24. /* D50 */ {96.720, 100.000, 81.427},
  25. /* D55 */ {95.799, 100.000, 90.926},
  26. /* D65 */ {94.811, 100.000, 107.304},
  27. /* D75 */ {94.416, 100.000, 120.641},
  28. /* F2 */ {103.280, 100.000, 69.026},
  29. /* F7 */ {95.792, 100.000, 107.687},
  30. /* F11 */ {103.866, 100.000, 65.627}
  31. };
  32. double ColorConversion::ccmin(double val1, double val2, double val3)
  33. {
  34. if (val1 < val2)
  35. {
  36. if(val1 < val3) return val1;
  37. else return val3;
  38. }
  39. else if (val2 < val3)
  40. return val2;
  41. return val3;
  42. }
  43. double ColorConversion::ccmax(double val1, double val2, double val3)
  44. {
  45. if (val1 > val2)
  46. {
  47. if(val1 > val3) return val1;
  48. else return val3;
  49. }
  50. else if (val2 > val3)
  51. return val2;
  52. return val3;
  53. }
  54. double ColorConversion::Hue_2_RGB( double v1, double v2, double vH )
  55. {
  56. if ( vH < 0 ) vH += 1;
  57. if ( vH > 1 ) vH -= 1;
  58. if ( ( 6 * vH ) < 1 ) return ( v1 + ( v2 - v1 ) * 6 * vH );
  59. if ( ( 2 * vH ) < 1 ) return ( v2 );
  60. if ( ( 3 * vH ) < 2 ) return ( v1 + ( v2 - v1 ) * ( ( 2./3 ) - vH ) * 6 );
  61. return ( v1 );
  62. }
  63. double ColorConversion::degree_2_radian(double val)
  64. {
  65. return (val*M_PI)/180;
  66. }
  67. //RGB values = From 0 to 1
  68. int ColorConversion::ccRGBtoHSL(double r, double g, double b, double* h, double* s, double* l)
  69. {
  70. double vmin, vmax, delta;
  71. double dr,dg,db;
  72. vmin = ccmin( r, g, b ); // Min. value of RGB
  73. vmax = ccmax( r, g, b ); // Max. value of RGB
  74. delta = vmax - vmin; // Delta RGB value
  75. *l = ( vmax + vmin ) / 2;
  76. if ( delta == 0 ) // This is a gray, no chroma...
  77. {
  78. *h = 0; // HSL results = From 0 to 1
  79. *s = 0;
  80. }
  81. else // Chromatic data...
  82. {
  83. if ( *l < 0.5 ) *s = delta / ( vmax + vmin );
  84. else *s = delta / ( 2 - vmax - vmin );
  85. dr = ( ( ( vmax - r ) / 6 ) + ( delta / 2 ) ) / delta;
  86. dg = ( ( ( vmax - g ) / 6 ) + ( delta / 2 ) ) / delta;
  87. db = ( ( ( vmax - b ) / 6 ) + ( delta / 2 ) ) / delta;
  88. if ( r == vmax ) *h = db - dg;
  89. else if ( g == vmax ) *h = ( 1./3 ) + dr - db;
  90. else if ( b == vmax ) *h = ( 2./3 ) + dg - dr;
  91. if ( *h < 0 ) *h += 1;
  92. if ( *h > 1 ) *h -= 1;
  93. }
  94. return 1;
  95. }
  96. int ColorConversion::ccHSLtoRGB(double h, double s, double l, double* r, double* g, double* b)
  97. {
  98. double v1, v2;
  99. if ( s == 0 ) // HSL values = From 0 to 1
  100. {
  101. *r = l * 255; // RGB results = From 0 to 255
  102. *g = l * 255;
  103. *b = l * 255;
  104. }
  105. else
  106. {
  107. if ( l < 0.5 )
  108. v2 = l * ( 1 + s );
  109. else
  110. v2 = ( l + s ) - ( s * l );
  111. v1 = 2 * l - v2;
  112. *r = 255 * Hue_2_RGB( v1, v2, h + ( 1./3 ) );
  113. *g = 255 * Hue_2_RGB( v1, v2, h );
  114. *b = 255 * Hue_2_RGB( v1, v2, h - ( 1./3 ) );
  115. }
  116. return 1;
  117. }
  118. // RGB values = From 0 to 1
  119. int ColorConversion::ccfRGBtoCMY(double r, double g, double b, double* c, double* m, double* y)
  120. {
  121. assert(c && m && y);
  122. *c = 1 - r;
  123. *m = 1 - g;
  124. *y = 1 - b;
  125. return 1;
  126. }
  127. // RGB values = From 0 to 255
  128. int ColorConversion::cciRGBtoCMY(int r, int g, int b, double* c, double* m, double* y)
  129. {
  130. assert(c && m && y);
  131. assert(r >= 0 && r < 256);
  132. assert(g >= 0 && g < 256);
  133. assert(b >= 0 && b < 256);
  134. *c = 1 - ( (double)r / 255 );
  135. *m = 1 - ( (double)g / 255 );
  136. *y = 1 - ( (double)b / 255 );
  137. return 1;
  138. }
  139. // CMY values = From 0 to 255
  140. int ColorConversion::ccCMYtofRGB(double c, double m, double y, double* r, double* g, double* b)
  141. {
  142. assert(r && g && b);
  143. assert(c >= 0 && c <= 1);
  144. assert(m >= 0 && m <= 1);
  145. assert(y >= 0 && y <= 1);
  146. *r = ( 1 - c );
  147. *g = ( 1 - m );
  148. *b = ( 1 - y );
  149. return 1;
  150. }
  151. // CMY values = From 0 to 1
  152. int ColorConversion::ccCMYtoiRGB(double c, double m, double y, int* r, int* g, int* b)
  153. {
  154. assert(r && g && b);
  155. assert(c >= 0 && c <= 1);
  156. assert(m >= 0 && m <= 1);
  157. assert(y >= 0 && y <= 1);
  158. *r = (int)( 1 - c ) * 255;
  159. *g = (int)( 1 - m ) * 255;
  160. *b = (int)( 1 - y ) * 255;
  161. return 1;
  162. }
  163. // CMY values = From 0 to 1
  164. int ColorConversion::ccCMYtoCMYK(double c, double m, double y, double* C, double* M, double* Y, double* K)
  165. {
  166. double var_K = 1;
  167. assert(C && M && Y && K);
  168. assert(c >= 0 && c <= 1);
  169. assert(m >= 0 && m <= 1);
  170. assert(y >= 0 && y <= 1);
  171. if ( c < var_K )
  172. var_K = c;
  173. if ( m < var_K )
  174. var_K = m;
  175. if ( y < var_K )
  176. var_K = y;
  177. *C = ( c - var_K ) / ( 1 - var_K );
  178. *M = ( m - var_K ) / ( 1 - var_K );
  179. *Y = ( y - var_K ) / ( 1 - var_K );
  180. *K = var_K;
  181. return 1;
  182. }
  183. // CMYK values = From 0 to 1
  184. int ColorConversion::ccCMYKtoCMY(double c, double m, double y, double k, double* C, double* M, double* Y)
  185. {
  186. assert(C && M && Y);
  187. assert(c >= 0 && c <= 1);
  188. assert(m >= 0 && m <= 1);
  189. assert(y >= 0 && y <= 1);
  190. assert(k >= 0 && k <= 1);
  191. *C = ( c * ( 1 - k ) + k );
  192. *M = ( m * ( 1 - k ) + k );
  193. *Y = ( y * ( 1 - k ) + k );
  194. return 1;
  195. }
  196. int ColorConversion::ccRGBtoHSV(double r, double g, double b, double* h, double* s, double* v)
  197. {
  198. double var_Min;
  199. double var_Max;
  200. double del_Max;
  201. var_Min = ccmin( r, g, b ); //Min. value of RGB
  202. var_Max = ccmax( r, g, g ); //Max. value of RGB
  203. del_Max = var_Max - var_Min; //Delta RGB value
  204. *v = var_Max;
  205. if ( del_Max == 0 ) // This is a gray, no chroma...
  206. {
  207. *h = 0; // HSV results = From 0 to 1
  208. *s = 0;
  209. }
  210. else // Chromatic data...
  211. {
  212. double del_R = ( ( ( var_Max - r ) / 6 ) + ( del_Max / 2 ) ) / del_Max;
  213. double del_G = ( ( ( var_Max - g ) / 6 ) + ( del_Max / 2 ) ) / del_Max;
  214. double del_B = ( ( ( var_Max - b ) / 6 ) + ( del_Max / 2 ) ) / del_Max;
  215. *s = del_Max / var_Max;
  216. if ( r == var_Max ) *h = del_B - del_G;
  217. else if ( g == var_Max ) *h = ( 1./3 ) + del_R - del_B;
  218. else if ( g == var_Max ) *h = ( 2./3 ) + del_G - del_R;
  219. if ( *h < 0 ) ; *h += 1;
  220. if ( *h > 1 ) ; *h -= 1;
  221. }
  222. return 1;
  223. }
  224. int ColorConversion::ccHSVtoRGB(double h, double s, double v, double* r, double* g, double* b)
  225. {
  226. if ( s == 0 ) // HSV values = From 0 to 1
  227. {
  228. *r = v;
  229. *g = v;
  230. *b = v;
  231. }
  232. else
  233. {
  234. double var_h = h * 6;
  235. double var_i = floor( var_h );
  236. double var_1 = v * ( 1 - s );
  237. double var_2 = v * ( 1 - s * ( var_h - var_i ) );
  238. double var_3 = v * ( 1 - s * ( 1 - ( var_h - var_i ) ) );
  239. if ( var_i == 0 ) { *r = v ; *g = var_3 ; *b = var_1; }
  240. else if ( var_i == 1 ) { *r = var_2 ; *g = v ; *b = var_1; }
  241. else if ( var_i == 2 ) { *r = var_1 ; *g = v ; *b = var_3; }
  242. else if ( var_i == 3 ) { *r = var_1 ; *g = var_2 ; *b = v; }
  243. else if ( var_i == 4 ) { *r = var_3 ; *g = var_1 ; *b = v; }
  244. else { *r = v ; *g = var_1 ; *b = var_2; }
  245. }
  246. return 1;
  247. }
  248. int ColorConversion::ccXYZtoHunterLab(double x, double y, double z, double* L, double* a, double* b)
  249. {
  250. *L = 10 * sqrt( y );
  251. *a = 17.5 * ( ( ( 1.02 * x ) - y ) / sqrt( y ) );
  252. *b = 7 * ( ( y - ( 0.847 * z ) ) / sqrt( y ) );
  253. return 1;
  254. }
  255. int ccHunterLabtoXYZ(double L, double a, double b, double* x, double* y, double* z)
  256. {
  257. double var_Y = L / 10;
  258. double var_X = a / 17.5 * L / 10;
  259. double var_Z = b / 7 * L / 10;
  260. *y = pow(var_Y,2);
  261. *x = ( var_X + *y ) / 1.02;
  262. *z = -( var_Z - *y ) / 0.847;
  263. return 1;
  264. }
  265. int ColorConversion::ccXYZtoYxy(double x, double y, double z, double* rY, double* rx, double* ry)
  266. {
  267. *rY = y;
  268. *rx = x / ( x + y + z );
  269. *ry = y / ( x + y + z );
  270. return 1;
  271. }
  272. int ColorConversion::ccXYZtoLMS(double x, double y, double z, double* l, double* m, double* s)
  273. {
  274. *l=0.7328*x+0.4296*y-0.1624*z;
  275. *m=-0.7036*x+1.6975*y+0.0061*z;
  276. *s=0.0030*x+0.0136*y+0.9834*z;
  277. return 1;
  278. }
  279. int ColorConversion::ccLMStoOPP(double l, double m, double s, double* lum, double* lm, double* slm)
  280. {
  281. *lum=l+m;
  282. if (fabs(*lum)>1.0e-8)
  283. {
  284. *lm=(l-m)/ *lum;
  285. }
  286. else
  287. {
  288. *lm=l-m; //linear model
  289. }
  290. if(fabs(s+ *lum)>1.0e-8)
  291. {
  292. *slm=(s- *lum)/(s+ *lum);
  293. }
  294. else
  295. {
  296. *slm=2*s- *lum;//linear model
  297. }
  298. return 1;
  299. }
  300. int ColorConversion::ccYxytoXYZ(double Y, double x, double y, double* rx, double* ry, double* rz)
  301. {
  302. //Y = From 0 to 100
  303. //x = From 0 to 1
  304. //y = From 0 to 1
  305. *rx = x * ( Y / y );
  306. *ry = Y;
  307. *rz = ( 1 - x - y ) * ( Y / y );
  308. return 1;
  309. }
  310. int ColorConversion::ccFixRGB(double* r, double* g, double* b)
  311. {
  312. double min,max;
  313. int mod=0;
  314. min=(*r>*g)?*g:*r;
  315. min=(min>*b)?*b:min;
  316. if (min<0) {*r+=-min; *g+=-min; *b+=-min; mod=1;}
  317. max=(*r>*g)?*r:*g;
  318. max=(max>*b)?max:*b;
  319. if (max>1){*r/=max; *g/=max; *b/=max; mod=1;}
  320. return mod;
  321. }
  322. //r,g,b from 0 to 1
  323. int ColorConversion::ccXYZtoRGB(double x, double y, double z, double* r, double* g, double* b, char ObsIll)
  324. {
  325. double ref_X = 100.0;//_ccTristimulusValues[ObsIll][ccX];
  326. double ref_Y = 100.0;//_ccTristimulusValues[ObsIll][ccY];
  327. double ref_Z = 100.0;//_ccTristimulusValues[ObsIll][ccZ];
  328. double var_X = x / ref_X; //X = From 0 to ref_X
  329. double var_Y = y / ref_Y; //Y = From 0 to ref_Y
  330. double var_Z = z / ref_Z; //Z = From 0 to ref_Y
  331. double var_R = var_X * 3.2406 + var_Y * -1.5372 + var_Z * -0.4986;
  332. double var_G = var_X * -0.9689 + var_Y * 1.8758 + var_Z * 0.0415;
  333. double var_B = var_X * 0.0557 + var_Y * -0.2040 + var_Z * 1.0570;
  334. if ( var_R > 0.0031308 ) var_R = 1.055 * ( pow(var_R,( 1 / 2.4 )) ) - 0.055;
  335. else var_R = 12.92 * var_R;
  336. if ( var_G > 0.0031308 ) var_G = 1.055 * ( pow(var_G,( 1 / 2.4 )) ) - 0.055;
  337. else var_G = 12.92 * var_G;
  338. if ( var_B > 0.0031308 ) var_B = 1.055 * ( pow(var_B,( 1 / 2.4 )) ) - 0.055;
  339. else var_B = 12.92 * var_B;
  340. *r = var_R;
  341. *g = var_G;
  342. *b = var_B;
  343. return 1;
  344. }
  345. int ColorConversion::ccRGBtoXYZ(double r, double g, double b, double* x, double* y, double* z, char ObsIll)
  346. {
  347. double var_R = r; //R = From 0 to 1
  348. double var_G = g; //G = From 0 to 1
  349. double var_B = b; //B = From 0 to 1
  350. if ( var_R > 0.04045 ) var_R = pow(( ( var_R + 0.055 ) / 1.055 ),2.4);
  351. else var_R = var_R / 12.92;
  352. if ( var_G > 0.04045 ) var_G = pow(( ( var_G + 0.055 ) / 1.055 ),2.4);
  353. else var_G = var_G / 12.92;
  354. if ( var_B > 0.04045 ) var_B = pow(( ( var_B + 0.055 ) / 1.055 ),2.4);
  355. else var_B = var_B / 12.92;
  356. var_R = var_R * 100;
  357. var_G = var_G * 100;
  358. var_B = var_B * 100;
  359. *x = var_R * 0.4124 + var_G * 0.3576 + var_B * 0.1805;
  360. *y = var_R * 0.2126 + var_G * 0.7152 + var_B * 0.0722;
  361. *z = var_R * 0.0193 + var_G * 0.1192 + var_B * 0.9505;
  362. return 1;
  363. }
  364. int ColorConversion::ccXYZtoCIE_Lab(double x, double y, double z, double* L, double* a, double* b, char ObsIll)
  365. {
  366. double var_X = x/_ccTristimulusValues[ObsIll][ccX];
  367. double var_Y = y/_ccTristimulusValues[ObsIll][ccY];
  368. double var_Z = z/_ccTristimulusValues[ObsIll][ccZ];
  369. if ( var_X > 0.008856 ) var_X = pow(var_X,( 1./3 ));
  370. else var_X = ( 7.787 * var_X ) + ( 16./ 116 );
  371. if ( var_Y > 0.008856 ) var_Y = pow(var_Y,( 1./3 ));
  372. else var_Y = ( 7.787 * var_Y ) + ( 16./ 116 );
  373. if ( var_Z > 0.008856 ) var_Z = pow(var_Z,( 1./3 ));
  374. else var_Z = ( 7.787 * var_Z ) + ( 16./ 116 );
  375. *L = ( 116 * var_Y ) - 16;
  376. *a = 500 * ( var_X - var_Y );
  377. *b = 200 * ( var_Y - var_Z );
  378. return 1;
  379. }
  380. int ColorConversion::ccCIE_LabtoXYZ(double L, double a, double b, double* x, double* y, double* z, char ObsIll)
  381. {
  382. double var_Y = ( L + 16 ) / 116;
  383. double var_X = a / 500 + var_Y;
  384. double var_Z = var_Y - b / 200;
  385. if ( pow(var_Y,3) > 0.008856 ) var_Y = pow(var_Y,3);
  386. else var_Y = ( var_Y - 16./ 116 ) / 7.787;
  387. if ( pow(var_X,3) > 0.008856 ) var_X = pow(var_X,3);
  388. else var_X = ( var_X - 16./ 116 ) / 7.787;
  389. if ( pow(var_Z,3) > 0.008856 ) var_Z = pow(var_Z,3);
  390. else var_Z = ( var_Z - 16./ 116 ) / 7.787;
  391. *x = _ccTristimulusValues[ObsIll][ccX] * var_X;
  392. *y = _ccTristimulusValues[ObsIll][ccY] * var_Y;
  393. *z = _ccTristimulusValues[ObsIll][ccZ] * var_Z;
  394. return 1;
  395. }
  396. int ColorConversion::ccCIE_LabtoCIE_LCH(double L, double a, double b, double* rL, double* rC, double* rH)
  397. {
  398. double var_H = atan2( b, a ); //Quadrant by signs
  399. if ( var_H > 0 ) var_H = ( var_H / M_PI ) * 180;
  400. else var_H = 360 - ( (int)var_H / M_PI ) * 180;
  401. *rL = L;
  402. *rC = sqrt( pow(a,2) + pow(b,2) );
  403. *rH = var_H;
  404. return 1;
  405. }
  406. int ColorConversion::ccCIE_LCHtoCIE_Lab(double L, double C, double H, double* rL, double* ra, double* rb)
  407. {
  408. //CIE-H = From 0 to 360
  409. *rL = L;
  410. *ra = cos( degree_2_radian(H) ) * C;
  411. *rb = sin( degree_2_radian(H) ) * C;
  412. return 1;
  413. }
  414. int ColorConversion::ccXYZtoCIE_Luv(double x, double y, double z, double* L, double* u, double* v, char ObsIll)
  415. {
  416. double ref_X = _ccTristimulusValues[ObsIll][ccX];
  417. double ref_Y = _ccTristimulusValues[ObsIll][ccY];
  418. double ref_Z = _ccTristimulusValues[ObsIll][ccZ];
  419. double ref_U = ( 4 * ref_X ) / ( ref_X + ( 15 * ref_Y ) + ( 3 * ref_Z ) );
  420. double ref_V = ( 9 * ref_Y ) / ( ref_X + ( 15 * ref_Y ) + ( 3 * ref_Z ) );
  421. double var_U = ( 4 * x ) / ( x + ( 15 * y ) + ( 3 * z ) );
  422. double var_V = ( 9 * y ) / ( x + ( 15 * y ) + ( 3 * z ) );
  423. double var_Y = y / 100;
  424. if ( var_Y > 0.008856 ) var_Y = pow(var_Y,( 1./3 ));
  425. else var_Y = ( 7.787 * var_Y ) + ( 16./ 116 );
  426. *L = ( 116 * var_Y ) - 16;
  427. *u = 13 * *L * ( var_U - ref_U );
  428. *v = 13 * *L * ( var_V - ref_V );
  429. return 1;
  430. }
  431. int ColorConversion::ccCIE_LuvtoXYZ(double L, double u, double v, double* x, double* y, double* z, char ObsIll)
  432. {
  433. double ref_X = _ccTristimulusValues[ObsIll][ccX];
  434. double ref_Y = _ccTristimulusValues[ObsIll][ccY];
  435. double ref_Z = _ccTristimulusValues[ObsIll][ccZ];
  436. double ref_U = ( 4 * ref_X ) / ( ref_X + ( 15 * ref_Y ) + ( 3 * ref_Z ) );
  437. double ref_V = ( 9 * ref_Y ) / ( ref_X + ( 15 * ref_Y ) + ( 3 * ref_Z ) );
  438. double var_U = u / ( 13 * L ) + ref_U;
  439. double var_V = v / ( 13 * L ) + ref_V;
  440. double var_Y = ( L + 16 ) / 116;
  441. if ( pow(var_Y,3) > 0.008856 ) var_Y = pow(var_Y,3);
  442. else var_Y = ( var_Y - 16./ 116 ) / 7.787;
  443. (*y) = var_Y * 100;
  444. (*x) = - ( 9 * *y * var_U ) / ( ( var_U - 4 ) * var_V - var_U * var_V );
  445. (*z) = ( 9 * *y - ( 15 * var_V * *y ) - ( var_V * *x ) ) / ( 3 * var_V );
  446. return 1;
  447. }