Color.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615
  1. import { clamp, euclideanModulo, lerp } from './MathUtils.js';
  2. import { ColorManagement, SRGBToLinear, LinearToSRGB } from './ColorManagement.js';
  3. import { SRGBColorSpace, LinearSRGBColorSpace } from '../constants.js';
  4. const _colorKeywords = { 'aliceblue': 0xF0F8FF, 'antiquewhite': 0xFAEBD7, 'aqua': 0x00FFFF, 'aquamarine': 0x7FFFD4, 'azure': 0xF0FFFF,
  5. 'beige': 0xF5F5DC, 'bisque': 0xFFE4C4, 'black': 0x000000, 'blanchedalmond': 0xFFEBCD, 'blue': 0x0000FF, 'blueviolet': 0x8A2BE2,
  6. 'brown': 0xA52A2A, 'burlywood': 0xDEB887, 'cadetblue': 0x5F9EA0, 'chartreuse': 0x7FFF00, 'chocolate': 0xD2691E, 'coral': 0xFF7F50,
  7. 'cornflowerblue': 0x6495ED, 'cornsilk': 0xFFF8DC, 'crimson': 0xDC143C, 'cyan': 0x00FFFF, 'darkblue': 0x00008B, 'darkcyan': 0x008B8B,
  8. 'darkgoldenrod': 0xB8860B, 'darkgray': 0xA9A9A9, 'darkgreen': 0x006400, 'darkgrey': 0xA9A9A9, 'darkkhaki': 0xBDB76B, 'darkmagenta': 0x8B008B,
  9. 'darkolivegreen': 0x556B2F, 'darkorange': 0xFF8C00, 'darkorchid': 0x9932CC, 'darkred': 0x8B0000, 'darksalmon': 0xE9967A, 'darkseagreen': 0x8FBC8F,
  10. 'darkslateblue': 0x483D8B, 'darkslategray': 0x2F4F4F, 'darkslategrey': 0x2F4F4F, 'darkturquoise': 0x00CED1, 'darkviolet': 0x9400D3,
  11. 'deeppink': 0xFF1493, 'deepskyblue': 0x00BFFF, 'dimgray': 0x696969, 'dimgrey': 0x696969, 'dodgerblue': 0x1E90FF, 'firebrick': 0xB22222,
  12. 'floralwhite': 0xFFFAF0, 'forestgreen': 0x228B22, 'fuchsia': 0xFF00FF, 'gainsboro': 0xDCDCDC, 'ghostwhite': 0xF8F8FF, 'gold': 0xFFD700,
  13. 'goldenrod': 0xDAA520, 'gray': 0x808080, 'green': 0x008000, 'greenyellow': 0xADFF2F, 'grey': 0x808080, 'honeydew': 0xF0FFF0, 'hotpink': 0xFF69B4,
  14. 'indianred': 0xCD5C5C, 'indigo': 0x4B0082, 'ivory': 0xFFFFF0, 'khaki': 0xF0E68C, 'lavender': 0xE6E6FA, 'lavenderblush': 0xFFF0F5, 'lawngreen': 0x7CFC00,
  15. 'lemonchiffon': 0xFFFACD, 'lightblue': 0xADD8E6, 'lightcoral': 0xF08080, 'lightcyan': 0xE0FFFF, 'lightgoldenrodyellow': 0xFAFAD2, 'lightgray': 0xD3D3D3,
  16. 'lightgreen': 0x90EE90, 'lightgrey': 0xD3D3D3, 'lightpink': 0xFFB6C1, 'lightsalmon': 0xFFA07A, 'lightseagreen': 0x20B2AA, 'lightskyblue': 0x87CEFA,
  17. 'lightslategray': 0x778899, 'lightslategrey': 0x778899, 'lightsteelblue': 0xB0C4DE, 'lightyellow': 0xFFFFE0, 'lime': 0x00FF00, 'limegreen': 0x32CD32,
  18. 'linen': 0xFAF0E6, 'magenta': 0xFF00FF, 'maroon': 0x800000, 'mediumaquamarine': 0x66CDAA, 'mediumblue': 0x0000CD, 'mediumorchid': 0xBA55D3,
  19. 'mediumpurple': 0x9370DB, 'mediumseagreen': 0x3CB371, 'mediumslateblue': 0x7B68EE, 'mediumspringgreen': 0x00FA9A, 'mediumturquoise': 0x48D1CC,
  20. 'mediumvioletred': 0xC71585, 'midnightblue': 0x191970, 'mintcream': 0xF5FFFA, 'mistyrose': 0xFFE4E1, 'moccasin': 0xFFE4B5, 'navajowhite': 0xFFDEAD,
  21. 'navy': 0x000080, 'oldlace': 0xFDF5E6, 'olive': 0x808000, 'olivedrab': 0x6B8E23, 'orange': 0xFFA500, 'orangered': 0xFF4500, 'orchid': 0xDA70D6,
  22. 'palegoldenrod': 0xEEE8AA, 'palegreen': 0x98FB98, 'paleturquoise': 0xAFEEEE, 'palevioletred': 0xDB7093, 'papayawhip': 0xFFEFD5, 'peachpuff': 0xFFDAB9,
  23. 'peru': 0xCD853F, 'pink': 0xFFC0CB, 'plum': 0xDDA0DD, 'powderblue': 0xB0E0E6, 'purple': 0x800080, 'rebeccapurple': 0x663399, 'red': 0xFF0000, 'rosybrown': 0xBC8F8F,
  24. 'royalblue': 0x4169E1, 'saddlebrown': 0x8B4513, 'salmon': 0xFA8072, 'sandybrown': 0xF4A460, 'seagreen': 0x2E8B57, 'seashell': 0xFFF5EE,
  25. 'sienna': 0xA0522D, 'silver': 0xC0C0C0, 'skyblue': 0x87CEEB, 'slateblue': 0x6A5ACD, 'slategray': 0x708090, 'slategrey': 0x708090, 'snow': 0xFFFAFA,
  26. 'springgreen': 0x00FF7F, 'steelblue': 0x4682B4, 'tan': 0xD2B48C, 'teal': 0x008080, 'thistle': 0xD8BFD8, 'tomato': 0xFF6347, 'turquoise': 0x40E0D0,
  27. 'violet': 0xEE82EE, 'wheat': 0xF5DEB3, 'white': 0xFFFFFF, 'whitesmoke': 0xF5F5F5, 'yellow': 0xFFFF00, 'yellowgreen': 0x9ACD32 };
  28. const _rgb = { r: 0, g: 0, b: 0 };
  29. const _hslA = { h: 0, s: 0, l: 0 };
  30. const _hslB = { h: 0, s: 0, l: 0 };
  31. function hue2rgb( p, q, t ) {
  32. if ( t < 0 ) t += 1;
  33. if ( t > 1 ) t -= 1;
  34. if ( t < 1 / 6 ) return p + ( q - p ) * 6 * t;
  35. if ( t < 1 / 2 ) return q;
  36. if ( t < 2 / 3 ) return p + ( q - p ) * 6 * ( 2 / 3 - t );
  37. return p;
  38. }
  39. function toComponents( source, target ) {
  40. target.r = source.r;
  41. target.g = source.g;
  42. target.b = source.b;
  43. return target;
  44. }
  45. class Color {
  46. constructor( r, g, b ) {
  47. this.isColor = true;
  48. this.r = 1;
  49. this.g = 1;
  50. this.b = 1;
  51. if ( g === undefined && b === undefined ) {
  52. // r is THREE.Color, hex or string
  53. return this.set( r );
  54. }
  55. return this.setRGB( r, g, b );
  56. }
  57. set( value ) {
  58. if ( value && value.isColor ) {
  59. this.copy( value );
  60. } else if ( typeof value === 'number' ) {
  61. this.setHex( value );
  62. } else if ( typeof value === 'string' ) {
  63. this.setStyle( value );
  64. }
  65. return this;
  66. }
  67. setScalar( scalar ) {
  68. this.r = scalar;
  69. this.g = scalar;
  70. this.b = scalar;
  71. return this;
  72. }
  73. setHex( hex, colorSpace = SRGBColorSpace ) {
  74. hex = Math.floor( hex );
  75. this.r = ( hex >> 16 & 255 ) / 255;
  76. this.g = ( hex >> 8 & 255 ) / 255;
  77. this.b = ( hex & 255 ) / 255;
  78. ColorManagement.toWorkingColorSpace( this, colorSpace );
  79. return this;
  80. }
  81. setRGB( r, g, b, colorSpace = LinearSRGBColorSpace ) {
  82. this.r = r;
  83. this.g = g;
  84. this.b = b;
  85. ColorManagement.toWorkingColorSpace( this, colorSpace );
  86. return this;
  87. }
  88. setHSL( h, s, l, colorSpace = LinearSRGBColorSpace ) {
  89. // h,s,l ranges are in 0.0 - 1.0
  90. h = euclideanModulo( h, 1 );
  91. s = clamp( s, 0, 1 );
  92. l = clamp( l, 0, 1 );
  93. if ( s === 0 ) {
  94. this.r = this.g = this.b = l;
  95. } else {
  96. const p = l <= 0.5 ? l * ( 1 + s ) : l + s - ( l * s );
  97. const q = ( 2 * l ) - p;
  98. this.r = hue2rgb( q, p, h + 1 / 3 );
  99. this.g = hue2rgb( q, p, h );
  100. this.b = hue2rgb( q, p, h - 1 / 3 );
  101. }
  102. ColorManagement.toWorkingColorSpace( this, colorSpace );
  103. return this;
  104. }
  105. setStyle( style, colorSpace = SRGBColorSpace ) {
  106. function handleAlpha( string ) {
  107. if ( string === undefined ) return;
  108. if ( parseFloat( string ) < 1 ) {
  109. console.warn( 'THREE.Color: Alpha component of ' + style + ' will be ignored.' );
  110. }
  111. }
  112. let m;
  113. if ( m = /^((?:rgb|hsl)a?)\(([^\)]*)\)/.exec( style ) ) {
  114. // rgb / hsl
  115. let color;
  116. const name = m[ 1 ];
  117. const components = m[ 2 ];
  118. switch ( name ) {
  119. case 'rgb':
  120. case 'rgba':
  121. if ( color = /^\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*(?:,\s*(\d*\.?\d+)\s*)?$/.exec( components ) ) {
  122. // rgb(255,0,0) rgba(255,0,0,0.5)
  123. this.r = Math.min( 255, parseInt( color[ 1 ], 10 ) ) / 255;
  124. this.g = Math.min( 255, parseInt( color[ 2 ], 10 ) ) / 255;
  125. this.b = Math.min( 255, parseInt( color[ 3 ], 10 ) ) / 255;
  126. ColorManagement.toWorkingColorSpace( this, colorSpace );
  127. handleAlpha( color[ 4 ] );
  128. return this;
  129. }
  130. if ( color = /^\s*(\d+)\%\s*,\s*(\d+)\%\s*,\s*(\d+)\%\s*(?:,\s*(\d*\.?\d+)\s*)?$/.exec( components ) ) {
  131. // rgb(100%,0%,0%) rgba(100%,0%,0%,0.5)
  132. this.r = Math.min( 100, parseInt( color[ 1 ], 10 ) ) / 100;
  133. this.g = Math.min( 100, parseInt( color[ 2 ], 10 ) ) / 100;
  134. this.b = Math.min( 100, parseInt( color[ 3 ], 10 ) ) / 100;
  135. ColorManagement.toWorkingColorSpace( this, colorSpace );
  136. handleAlpha( color[ 4 ] );
  137. return this;
  138. }
  139. break;
  140. case 'hsl':
  141. case 'hsla':
  142. if ( color = /^\s*(\d*\.?\d+)\s*,\s*(\d+)\%\s*,\s*(\d+)\%\s*(?:,\s*(\d*\.?\d+)\s*)?$/.exec( components ) ) {
  143. // hsl(120,50%,50%) hsla(120,50%,50%,0.5)
  144. const h = parseFloat( color[ 1 ] ) / 360;
  145. const s = parseInt( color[ 2 ], 10 ) / 100;
  146. const l = parseInt( color[ 3 ], 10 ) / 100;
  147. handleAlpha( color[ 4 ] );
  148. return this.setHSL( h, s, l, colorSpace );
  149. }
  150. break;
  151. }
  152. } else if ( m = /^\#([A-Fa-f\d]+)$/.exec( style ) ) {
  153. // hex color
  154. const hex = m[ 1 ];
  155. const size = hex.length;
  156. if ( size === 3 ) {
  157. // #ff0
  158. this.r = parseInt( hex.charAt( 0 ) + hex.charAt( 0 ), 16 ) / 255;
  159. this.g = parseInt( hex.charAt( 1 ) + hex.charAt( 1 ), 16 ) / 255;
  160. this.b = parseInt( hex.charAt( 2 ) + hex.charAt( 2 ), 16 ) / 255;
  161. ColorManagement.toWorkingColorSpace( this, colorSpace );
  162. return this;
  163. } else if ( size === 6 ) {
  164. // #ff0000
  165. this.r = parseInt( hex.charAt( 0 ) + hex.charAt( 1 ), 16 ) / 255;
  166. this.g = parseInt( hex.charAt( 2 ) + hex.charAt( 3 ), 16 ) / 255;
  167. this.b = parseInt( hex.charAt( 4 ) + hex.charAt( 5 ), 16 ) / 255;
  168. ColorManagement.toWorkingColorSpace( this, colorSpace );
  169. return this;
  170. }
  171. }
  172. if ( style && style.length > 0 ) {
  173. return this.setColorName( style, colorSpace );
  174. }
  175. return this;
  176. }
  177. setColorName( style, colorSpace = SRGBColorSpace ) {
  178. // color keywords
  179. const hex = _colorKeywords[ style.toLowerCase() ];
  180. if ( hex !== undefined ) {
  181. // red
  182. this.setHex( hex, colorSpace );
  183. } else {
  184. // unknown color
  185. console.warn( 'THREE.Color: Unknown color ' + style );
  186. }
  187. return this;
  188. }
  189. clone() {
  190. return new this.constructor( this.r, this.g, this.b );
  191. }
  192. copy( color ) {
  193. this.r = color.r;
  194. this.g = color.g;
  195. this.b = color.b;
  196. return this;
  197. }
  198. copySRGBToLinear( color ) {
  199. this.r = SRGBToLinear( color.r );
  200. this.g = SRGBToLinear( color.g );
  201. this.b = SRGBToLinear( color.b );
  202. return this;
  203. }
  204. copyLinearToSRGB( color ) {
  205. this.r = LinearToSRGB( color.r );
  206. this.g = LinearToSRGB( color.g );
  207. this.b = LinearToSRGB( color.b );
  208. return this;
  209. }
  210. convertSRGBToLinear() {
  211. this.copySRGBToLinear( this );
  212. return this;
  213. }
  214. convertLinearToSRGB() {
  215. this.copyLinearToSRGB( this );
  216. return this;
  217. }
  218. getHex( colorSpace = SRGBColorSpace ) {
  219. ColorManagement.fromWorkingColorSpace( toComponents( this, _rgb ), colorSpace );
  220. return clamp( _rgb.r * 255, 0, 255 ) << 16 ^ clamp( _rgb.g * 255, 0, 255 ) << 8 ^ clamp( _rgb.b * 255, 0, 255 ) << 0;
  221. }
  222. getHexString( colorSpace = SRGBColorSpace ) {
  223. return ( '000000' + this.getHex( colorSpace ).toString( 16 ) ).slice( - 6 );
  224. }
  225. getHSL( target, colorSpace = LinearSRGBColorSpace ) {
  226. // h,s,l ranges are in 0.0 - 1.0
  227. ColorManagement.fromWorkingColorSpace( toComponents( this, _rgb ), colorSpace );
  228. const r = _rgb.r, g = _rgb.g, b = _rgb.b;
  229. const max = Math.max( r, g, b );
  230. const min = Math.min( r, g, b );
  231. let hue, saturation;
  232. const lightness = ( min + max ) / 2.0;
  233. if ( min === max ) {
  234. hue = 0;
  235. saturation = 0;
  236. } else {
  237. const delta = max - min;
  238. saturation = lightness <= 0.5 ? delta / ( max + min ) : delta / ( 2 - max - min );
  239. switch ( max ) {
  240. case r: hue = ( g - b ) / delta + ( g < b ? 6 : 0 ); break;
  241. case g: hue = ( b - r ) / delta + 2; break;
  242. case b: hue = ( r - g ) / delta + 4; break;
  243. }
  244. hue /= 6;
  245. }
  246. target.h = hue;
  247. target.s = saturation;
  248. target.l = lightness;
  249. return target;
  250. }
  251. getRGB( target, colorSpace = LinearSRGBColorSpace ) {
  252. ColorManagement.fromWorkingColorSpace( toComponents( this, _rgb ), colorSpace );
  253. target.r = _rgb.r;
  254. target.g = _rgb.g;
  255. target.b = _rgb.b;
  256. return target;
  257. }
  258. getStyle( colorSpace = SRGBColorSpace ) {
  259. ColorManagement.fromWorkingColorSpace( toComponents( this, _rgb ), colorSpace );
  260. if ( colorSpace !== SRGBColorSpace ) {
  261. // Requires CSS Color Module Level 4 (https://www.w3.org/TR/css-color-4/).
  262. return `color(${ colorSpace } ${ _rgb.r } ${ _rgb.g } ${ _rgb.b })`;
  263. }
  264. return `rgb(${( _rgb.r * 255 ) | 0},${( _rgb.g * 255 ) | 0},${( _rgb.b * 255 ) | 0})`;
  265. }
  266. offsetHSL( h, s, l ) {
  267. this.getHSL( _hslA );
  268. _hslA.h += h; _hslA.s += s; _hslA.l += l;
  269. this.setHSL( _hslA.h, _hslA.s, _hslA.l );
  270. return this;
  271. }
  272. add( color ) {
  273. this.r += color.r;
  274. this.g += color.g;
  275. this.b += color.b;
  276. return this;
  277. }
  278. addColors( color1, color2 ) {
  279. this.r = color1.r + color2.r;
  280. this.g = color1.g + color2.g;
  281. this.b = color1.b + color2.b;
  282. return this;
  283. }
  284. addScalar( s ) {
  285. this.r += s;
  286. this.g += s;
  287. this.b += s;
  288. return this;
  289. }
  290. sub( color ) {
  291. this.r = Math.max( 0, this.r - color.r );
  292. this.g = Math.max( 0, this.g - color.g );
  293. this.b = Math.max( 0, this.b - color.b );
  294. return this;
  295. }
  296. multiply( color ) {
  297. this.r *= color.r;
  298. this.g *= color.g;
  299. this.b *= color.b;
  300. return this;
  301. }
  302. multiplyScalar( s ) {
  303. this.r *= s;
  304. this.g *= s;
  305. this.b *= s;
  306. return this;
  307. }
  308. lerp( color, alpha ) {
  309. this.r += ( color.r - this.r ) * alpha;
  310. this.g += ( color.g - this.g ) * alpha;
  311. this.b += ( color.b - this.b ) * alpha;
  312. return this;
  313. }
  314. lerpColors( color1, color2, alpha ) {
  315. this.r = color1.r + ( color2.r - color1.r ) * alpha;
  316. this.g = color1.g + ( color2.g - color1.g ) * alpha;
  317. this.b = color1.b + ( color2.b - color1.b ) * alpha;
  318. return this;
  319. }
  320. lerpHSL( color, alpha ) {
  321. this.getHSL( _hslA );
  322. color.getHSL( _hslB );
  323. const h = lerp( _hslA.h, _hslB.h, alpha );
  324. const s = lerp( _hslA.s, _hslB.s, alpha );
  325. const l = lerp( _hslA.l, _hslB.l, alpha );
  326. this.setHSL( h, s, l );
  327. return this;
  328. }
  329. equals( c ) {
  330. return ( c.r === this.r ) && ( c.g === this.g ) && ( c.b === this.b );
  331. }
  332. fromArray( array, offset = 0 ) {
  333. this.r = array[ offset ];
  334. this.g = array[ offset + 1 ];
  335. this.b = array[ offset + 2 ];
  336. return this;
  337. }
  338. toArray( array = [], offset = 0 ) {
  339. array[ offset ] = this.r;
  340. array[ offset + 1 ] = this.g;
  341. array[ offset + 2 ] = this.b;
  342. return array;
  343. }
  344. fromBufferAttribute( attribute, index ) {
  345. this.r = attribute.getX( index );
  346. this.g = attribute.getY( index );
  347. this.b = attribute.getZ( index );
  348. if ( attribute.normalized === true ) {
  349. // assuming Uint8Array
  350. this.r /= 255;
  351. this.g /= 255;
  352. this.b /= 255;
  353. }
  354. return this;
  355. }
  356. toJSON() {
  357. return this.getHex();
  358. }
  359. *[ Symbol.iterator ]() {
  360. yield this.r;
  361. yield this.g;
  362. yield this.b;
  363. }
  364. }
  365. Color.NAMES = _colorKeywords;
  366. export { Color, SRGBToLinear };