csscolorparser.js 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. // (c) Dean McNamee <dean@gmail.com>, 2012.
  2. //
  3. // https://github.com/deanm/css-color-parser-js
  4. //
  5. // Permission is hereby granted, free of charge, to any person obtaining a copy
  6. // of this software and associated documentation files (the "Software"), to
  7. // deal in the Software without restriction, including without limitation the
  8. // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  9. // sell copies of the Software, and to permit persons to whom the Software is
  10. // furnished to do so, subject to the following conditions:
  11. //
  12. // The above copyright notice and this permission notice shall be included in
  13. // all copies or substantial portions of the Software.
  14. //
  15. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  16. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  17. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  18. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  19. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  20. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  21. // IN THE SOFTWARE.
  22. // http://www.w3.org/TR/css3-color/
  23. var kCSSColorTable = {
  24. "transparent": [0,0,0,0], "aliceblue": [240,248,255,1],
  25. "antiquewhite": [250,235,215,1], "aqua": [0,255,255,1],
  26. "aquamarine": [127,255,212,1], "azure": [240,255,255,1],
  27. "beige": [245,245,220,1], "bisque": [255,228,196,1],
  28. "black": [0,0,0,1], "blanchedalmond": [255,235,205,1],
  29. "blue": [0,0,255,1], "blueviolet": [138,43,226,1],
  30. "brown": [165,42,42,1], "burlywood": [222,184,135,1],
  31. "cadetblue": [95,158,160,1], "chartreuse": [127,255,0,1],
  32. "chocolate": [210,105,30,1], "coral": [255,127,80,1],
  33. "cornflowerblue": [100,149,237,1], "cornsilk": [255,248,220,1],
  34. "crimson": [220,20,60,1], "cyan": [0,255,255,1],
  35. "darkblue": [0,0,139,1], "darkcyan": [0,139,139,1],
  36. "darkgoldenrod": [184,134,11,1], "darkgray": [169,169,169,1],
  37. "darkgreen": [0,100,0,1], "darkgrey": [169,169,169,1],
  38. "darkkhaki": [189,183,107,1], "darkmagenta": [139,0,139,1],
  39. "darkolivegreen": [85,107,47,1], "darkorange": [255,140,0,1],
  40. "darkorchid": [153,50,204,1], "darkred": [139,0,0,1],
  41. "darksalmon": [233,150,122,1], "darkseagreen": [143,188,143,1],
  42. "darkslateblue": [72,61,139,1], "darkslategray": [47,79,79,1],
  43. "darkslategrey": [47,79,79,1], "darkturquoise": [0,206,209,1],
  44. "darkviolet": [148,0,211,1], "deeppink": [255,20,147,1],
  45. "deepskyblue": [0,191,255,1], "dimgray": [105,105,105,1],
  46. "dimgrey": [105,105,105,1], "dodgerblue": [30,144,255,1],
  47. "firebrick": [178,34,34,1], "floralwhite": [255,250,240,1],
  48. "forestgreen": [34,139,34,1], "fuchsia": [255,0,255,1],
  49. "gainsboro": [220,220,220,1], "ghostwhite": [248,248,255,1],
  50. "gold": [255,215,0,1], "goldenrod": [218,165,32,1],
  51. "gray": [128,128,128,1], "green": [0,128,0,1],
  52. "greenyellow": [173,255,47,1], "grey": [128,128,128,1],
  53. "honeydew": [240,255,240,1], "hotpink": [255,105,180,1],
  54. "indianred": [205,92,92,1], "indigo": [75,0,130,1],
  55. "ivory": [255,255,240,1], "khaki": [240,230,140,1],
  56. "lavender": [230,230,250,1], "lavenderblush": [255,240,245,1],
  57. "lawngreen": [124,252,0,1], "lemonchiffon": [255,250,205,1],
  58. "lightblue": [173,216,230,1], "lightcoral": [240,128,128,1],
  59. "lightcyan": [224,255,255,1], "lightgoldenrodyellow": [250,250,210,1],
  60. "lightgray": [211,211,211,1], "lightgreen": [144,238,144,1],
  61. "lightgrey": [211,211,211,1], "lightpink": [255,182,193,1],
  62. "lightsalmon": [255,160,122,1], "lightseagreen": [32,178,170,1],
  63. "lightskyblue": [135,206,250,1], "lightslategray": [119,136,153,1],
  64. "lightslategrey": [119,136,153,1], "lightsteelblue": [176,196,222,1],
  65. "lightyellow": [255,255,224,1], "lime": [0,255,0,1],
  66. "limegreen": [50,205,50,1], "linen": [250,240,230,1],
  67. "magenta": [255,0,255,1], "maroon": [128,0,0,1],
  68. "mediumaquamarine": [102,205,170,1], "mediumblue": [0,0,205,1],
  69. "mediumorchid": [186,85,211,1], "mediumpurple": [147,112,219,1],
  70. "mediumseagreen": [60,179,113,1], "mediumslateblue": [123,104,238,1],
  71. "mediumspringgreen": [0,250,154,1], "mediumturquoise": [72,209,204,1],
  72. "mediumvioletred": [199,21,133,1], "midnightblue": [25,25,112,1],
  73. "mintcream": [245,255,250,1], "mistyrose": [255,228,225,1],
  74. "moccasin": [255,228,181,1], "navajowhite": [255,222,173,1],
  75. "navy": [0,0,128,1], "oldlace": [253,245,230,1],
  76. "olive": [128,128,0,1], "olivedrab": [107,142,35,1],
  77. "orange": [255,165,0,1], "orangered": [255,69,0,1],
  78. "orchid": [218,112,214,1], "palegoldenrod": [238,232,170,1],
  79. "palegreen": [152,251,152,1], "paleturquoise": [175,238,238,1],
  80. "palevioletred": [219,112,147,1], "papayawhip": [255,239,213,1],
  81. "peachpuff": [255,218,185,1], "peru": [205,133,63,1],
  82. "pink": [255,192,203,1], "plum": [221,160,221,1],
  83. "powderblue": [176,224,230,1], "purple": [128,0,128,1],
  84. "red": [255,0,0,1], "rosybrown": [188,143,143,1],
  85. "royalblue": [65,105,225,1], "saddlebrown": [139,69,19,1],
  86. "salmon": [250,128,114,1], "sandybrown": [244,164,96,1],
  87. "seagreen": [46,139,87,1], "seashell": [255,245,238,1],
  88. "sienna": [160,82,45,1], "silver": [192,192,192,1],
  89. "skyblue": [135,206,235,1], "slateblue": [106,90,205,1],
  90. "slategray": [112,128,144,1], "slategrey": [112,128,144,1],
  91. "snow": [255,250,250,1], "springgreen": [0,255,127,1],
  92. "steelblue": [70,130,180,1], "tan": [210,180,140,1],
  93. "teal": [0,128,128,1], "thistle": [216,191,216,1],
  94. "tomato": [255,99,71,1], "turquoise": [64,224,208,1],
  95. "violet": [238,130,238,1], "wheat": [245,222,179,1],
  96. "white": [255,255,255,1], "whitesmoke": [245,245,245,1],
  97. "yellow": [255,255,0,1], "yellowgreen": [154,205,50,1]}
  98. function clamp_css_byte(i) { // Clamp to integer 0 .. 255.
  99. i = Math.round(i); // Seems to be what Chrome does (vs truncation).
  100. return i < 0 ? 0 : i > 255 ? 255 : i;
  101. }
  102. function clamp_css_float(f) { // Clamp to float 0.0 .. 1.0.
  103. return f < 0 ? 0 : f > 1 ? 1 : f;
  104. }
  105. function parse_css_int(str) { // int or percentage.
  106. if (str[str.length - 1] === '%')
  107. return clamp_css_byte(parseFloat(str) / 100 * 255);
  108. return clamp_css_byte(parseInt(str));
  109. }
  110. function parse_css_float(str) { // float or percentage.
  111. if (str[str.length - 1] === '%')
  112. return clamp_css_float(parseFloat(str) / 100);
  113. return clamp_css_float(parseFloat(str));
  114. }
  115. function css_hue_to_rgb(m1, m2, h) {
  116. if (h < 0) h += 1;
  117. else if (h > 1) h -= 1;
  118. if (h * 6 < 1) return m1 + (m2 - m1) * h * 6;
  119. if (h * 2 < 1) return m2;
  120. if (h * 3 < 2) return m1 + (m2 - m1) * (2/3 - h) * 6;
  121. return m1;
  122. }
  123. function parseCSSColor(css_str) {
  124. // Remove all whitespace, not compliant, but should just be more accepting.
  125. var str = css_str.replace(/ /g, '').toLowerCase();
  126. // Color keywords (and transparent) lookup.
  127. if (str in kCSSColorTable) return kCSSColorTable[str].slice(); // dup.
  128. // #abc and #abc123 syntax.
  129. if (str[0] === '#') {
  130. if (str.length === 4) {
  131. var iv = parseInt(str.substr(1), 16); // TODO(deanm): Stricter parsing.
  132. if (!(iv >= 0 && iv <= 0xfff)) return null; // Covers NaN.
  133. return [((iv & 0xf00) >> 4) | ((iv & 0xf00) >> 8),
  134. (iv & 0xf0) | ((iv & 0xf0) >> 4),
  135. (iv & 0xf) | ((iv & 0xf) << 4),
  136. 1];
  137. } else if (str.length === 7) {
  138. var iv = parseInt(str.substr(1), 16); // TODO(deanm): Stricter parsing.
  139. if (!(iv >= 0 && iv <= 0xffffff)) return null; // Covers NaN.
  140. return [(iv & 0xff0000) >> 16,
  141. (iv & 0xff00) >> 8,
  142. iv & 0xff,
  143. 1];
  144. }
  145. return null;
  146. }
  147. var op = str.indexOf('('), ep = str.indexOf(')');
  148. if (op !== -1 && ep + 1 === str.length) {
  149. var fname = str.substr(0, op);
  150. var params = str.substr(op+1, ep-(op+1)).split(',');
  151. var alpha = 1; // To allow case fallthrough.
  152. switch (fname) {
  153. case 'rgba':
  154. if (params.length !== 4) return null;
  155. alpha = parse_css_float(params.pop());
  156. // Fall through.
  157. case 'rgb':
  158. if (params.length !== 3) return null;
  159. return [parse_css_int(params[0]),
  160. parse_css_int(params[1]),
  161. parse_css_int(params[2]),
  162. alpha];
  163. case 'hsla':
  164. if (params.length !== 4) return null;
  165. alpha = parse_css_float(params.pop());
  166. // Fall through.
  167. case 'hsl':
  168. if (params.length !== 3) return null;
  169. var h = (((parseFloat(params[0]) % 360) + 360) % 360) / 360; // 0 .. 1
  170. // NOTE(deanm): According to the CSS spec s/l should only be
  171. // percentages, but we don't bother and let float or percentage.
  172. var s = parse_css_float(params[1]);
  173. var l = parse_css_float(params[2]);
  174. var m2 = l <= 0.5 ? l * (s + 1) : l + s - l * s;
  175. var m1 = l * 2 - m2;
  176. return [clamp_css_byte(css_hue_to_rgb(m1, m2, h+1/3) * 255),
  177. clamp_css_byte(css_hue_to_rgb(m1, m2, h) * 255),
  178. clamp_css_byte(css_hue_to_rgb(m1, m2, h-1/3) * 255),
  179. alpha];
  180. default:
  181. return null;
  182. }
  183. }
  184. return null;
  185. }
  186. try { exports.parseCSSColor = parseCSSColor } catch(e) { }