grayscale.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. /*
  2. * -- grayscale.js --
  3. * Copyright (C) James Padolsey (http://james.padolsey.com)
  4. *
  5. */
  6. var grayscale = (function(){
  7. var config = {
  8. colorProps: ['color','backgroundColor','borderBottomColor','borderTopColor','borderLeftColor','borderRightColor','backgroundImage'],
  9. externalImageHandler : {
  10. /* Grayscaling externally hosted images does not work
  11. - Use these functions to handle those images as you so desire */
  12. /* Out of convenience these functions are also used for browsers
  13. like Chrome that do not support CanvasContext.getImageData */
  14. init : function(el, src) {
  15. if (el.nodeName.toLowerCase() === 'img') {
  16. // Is IMG element...
  17. } else {
  18. // Is background-image element:
  19. // Default - remove background images
  20. data(el).backgroundImageSRC = src;
  21. el.style.backgroundImage = '';
  22. }
  23. },
  24. reset : function(el) {
  25. if (el.nodeName.toLowerCase() === 'img') {
  26. // Is IMG element...
  27. } else {
  28. // Is background-image element:
  29. el.style.backgroundImage = 'url(' + (data(el).backgroundImageSRC || '') + ')';
  30. }
  31. }
  32. }
  33. },
  34. log = function(){
  35. try { window.console.log.apply(console, arguments); }
  36. catch(e) {};
  37. },
  38. isExternal = function(url) {
  39. // Checks whether URL is external: 'CanvasContext.getImageData'
  40. // only works if the image is on the current domain.
  41. return (new RegExp('https?://(?!' + window.location.hostname + ')')).test(url);
  42. },
  43. data = (function(){
  44. var cache = [0],
  45. expando = 'data' + (+new Date());
  46. return function(elem) {
  47. var cacheIndex = elem[expando],
  48. nextCacheIndex = cache.length;
  49. if(!cacheIndex) {
  50. cacheIndex = elem[expando] = nextCacheIndex;
  51. cache[cacheIndex] = {};
  52. }
  53. return cache[cacheIndex];
  54. };
  55. })(),
  56. desatIMG = function(img, prepare, realEl) {
  57. // realEl is only set when img is temp (for BG images)
  58. var canvas = document.createElement('canvas'),
  59. context = canvas.getContext('2d'),
  60. height = img.naturalHeight || img.offsetHeight || img.height,
  61. width = img.naturalWidth || img.offsetWidth || img.width,
  62. imgData;
  63. canvas.height = height;
  64. canvas.width = width;
  65. context.drawImage(img, 0, 0);
  66. try {
  67. imgData = context.getImageData(0, 0, width, height);
  68. } catch(e) {}
  69. if (prepare) {
  70. desatIMG.preparing = true;
  71. // Slowly recurse through pixels for prep,
  72. // :: only occurs on grayscale.prepare()
  73. var y = 0;
  74. (function(){
  75. if (!desatIMG.preparing) { return; }
  76. if (y === height) {
  77. // Finished!
  78. context.putImageData(imgData, 0, 0, 0, 0, width, height);
  79. realEl ? (data(realEl).BGdataURL = canvas.toDataURL())
  80. : (data(img).dataURL = canvas.toDataURL())
  81. }
  82. for (var x = 0; x < width; x++) {
  83. var i = (y * width + x) * 4;
  84. // Apply Monoschrome level across all channels:
  85. imgData.data[i] = imgData.data[i+1] = imgData.data[i+2] =
  86. RGBtoGRAYSCALE(imgData.data[i], imgData.data[i+1], imgData.data[i+2]);
  87. }
  88. y++;
  89. setTimeout(arguments.callee, 0);
  90. })();
  91. return;
  92. } else {
  93. // If desatIMG was called without 'prepare' flag
  94. // then cancel recursion and proceed with force! (below)
  95. desatIMG.preparing = false;
  96. }
  97. for (var y = 0; y < height; y++) {
  98. for (var x = 0; x < width; x++) {
  99. var i = (y * width + x) * 4;
  100. // Apply Monoschrome level across all channels:
  101. imgData.data[i] = imgData.data[i+1] = imgData.data[i+2] =
  102. RGBtoGRAYSCALE(imgData.data[i], imgData.data[i+1], imgData.data[i+2]);
  103. }
  104. }
  105. context.putImageData(imgData, 0, 0, 0, 0, width, height);
  106. return canvas;
  107. },
  108. getStyle = function(el, prop) {
  109. var style = document.defaultView && document.defaultView.getComputedStyle ?
  110. document.defaultView.getComputedStyle(el, null)[prop]
  111. : el.currentStyle[prop];
  112. // If format is #FFFFFF: (convert to RGB)
  113. if (style && /^#[A-F0-9]/i.test(style)) {
  114. var hex = style.match(/[A-F0-9]{2}/ig);
  115. style = 'rgb(' + parseInt(hex[0], 16) + ','
  116. + parseInt(hex[1], 16) + ','
  117. + parseInt(hex[2], 16) + ')';
  118. }
  119. return style;
  120. },
  121. RGBtoGRAYSCALE = function(r,g,b) {
  122. // Returns single monochrome figure:
  123. return parseInt( (0.2125 * r) + (0.7154 * g) + (0.0721 * b), 10 );
  124. },
  125. getAllNodes = function(context) {
  126. var all = Array.prototype.slice.call(context.getElementsByTagName('*'));
  127. all.unshift(context);
  128. return all;
  129. };
  130. var init = function(context) {
  131. // Handle if a DOM collection is passed instead of a single el:
  132. if (context && context[0] && context.length && context[0].nodeName) {
  133. // Is a DOM collection:
  134. var allContexts = Array.prototype.slice.call(context),
  135. cIndex = -1, cLen = allContexts.length;
  136. while (++cIndex<cLen) { init.call(this, allContexts[cIndex]); }
  137. return;
  138. }
  139. context = context || document.documentElement;
  140. if (!document.createElement('canvas').getContext) {
  141. context.style.filter = 'progid:DXImageTransform.Microsoft.BasicImage(grayscale=1)';
  142. context.style.zoom = 1;
  143. return;
  144. }
  145. var all = getAllNodes(context),
  146. i = -1, len = all.length;
  147. while (++i<len) {
  148. var cur = all[i];
  149. if (cur.nodeName.toLowerCase() === 'img') {
  150. var src = cur.getAttribute('src');
  151. if(!src) { continue; }
  152. if (isExternal(src)) {
  153. config.externalImageHandler.init(cur, src);
  154. } else {
  155. data(cur).realSRC = src;
  156. try {
  157. // Within try statement just encase there's no support....
  158. cur.src = data(cur).dataURL || desatIMG(cur).toDataURL();
  159. } catch(e) { config.externalImageHandler.init(cur, src); }
  160. }
  161. } else {
  162. for (var pIndex = 0, pLen = config.colorProps.length; pIndex < pLen; pIndex++) {
  163. var prop = config.colorProps[pIndex],
  164. style = getStyle(cur, prop);
  165. if (!style) {continue;}
  166. if (cur.style[prop]) {
  167. data(cur)[prop] = style;
  168. }
  169. // RGB color:
  170. if (style.substring(0,4) === 'rgb(') {
  171. var monoRGB = RGBtoGRAYSCALE.apply(null, style.match(/\d+/g));
  172. cur.style[prop] = style = 'rgb(' + monoRGB + ',' + monoRGB + ',' + monoRGB + ')';
  173. continue;
  174. }
  175. // Background Image:
  176. if (style.indexOf('url(') > -1) {
  177. var urlPatt = /\(['"]?(.+?)['"]?\)/,
  178. url = style.match(urlPatt)[1];
  179. if (isExternal(url)) {
  180. config.externalImageHandler.init(cur, url);
  181. data(cur).externalBG = true;
  182. continue;
  183. }
  184. // data(cur).BGdataURL refers to caches URL (from preparation)
  185. try {
  186. var imgSRC = data(cur).BGdataURL || (function(){
  187. var temp = document.createElement('img');
  188. temp.src = url;
  189. return desatIMG(temp).toDataURL();
  190. })();
  191. cur.style[prop] = style.replace(urlPatt, function(_, url){
  192. return '(' + imgSRC + ')';
  193. });
  194. } catch(e) { config.externalImageHandler.init(cur, url); }
  195. }
  196. }
  197. }
  198. }
  199. };
  200. init.reset = function(context) {
  201. // Handle if a DOM collection is passed instead of a single el:
  202. if (context && context[0] && context.length && context[0].nodeName) {
  203. // Is a DOM collection:
  204. var allContexts = Array.prototype.slice.call(context),
  205. cIndex = -1, cLen = allContexts.length;
  206. while (++cIndex<cLen) { init.reset.call(this, allContexts[cIndex]); }
  207. return;
  208. }
  209. context = context || document.documentElement;
  210. if (!document.createElement('canvas').getContext) {
  211. context.style.filter = 'progid:DXImageTransform.Microsoft.BasicImage(grayscale=0)';
  212. return;
  213. }
  214. var all = getAllNodes(context),
  215. i = -1, len = all.length;
  216. while (++i<len) {
  217. var cur = all[i];
  218. if (cur.nodeName.toLowerCase() === 'img') {
  219. var src = cur.getAttribute('src');
  220. if (isExternal(src)) {
  221. config.externalImageHandler.reset(cur, src);
  222. }
  223. cur.src = data(cur).realSRC || src;
  224. } else {
  225. for (var pIndex = 0, pLen = config.colorProps.length; pIndex < pLen; pIndex++) {
  226. if (data(cur).externalBG) {
  227. config.externalImageHandler.reset(cur);
  228. }
  229. var prop = config.colorProps[pIndex];
  230. cur.style[prop] = data(cur)[prop] || '';
  231. }
  232. }
  233. }
  234. };
  235. init.prepare = function(context) {
  236. // Handle if a DOM collection is passed instead of a single el:
  237. if (context && context[0] && context.length && context[0].nodeName) {
  238. // Is a DOM collection:
  239. var allContexts = Array.prototype.slice.call(context),
  240. cIndex = -1, cLen = allContexts.length;
  241. while (++cIndex<cLen) { init.prepare.call(null, allContexts[cIndex]); }
  242. return;
  243. }
  244. // Slowly recurses through all elements
  245. // so as not to lock up on the user.
  246. context = context || document.documentElement;
  247. if (!document.createElement('canvas').getContext) { return; }
  248. var all = getAllNodes(context),
  249. i = -1, len = all.length;
  250. while (++i<len) {
  251. var cur = all[i];
  252. if (data(cur).skip) { return; }
  253. if (cur.nodeName.toLowerCase() === 'img') {
  254. if (cur.getAttribute('src') && !isExternal(cur.src)) {
  255. desatIMG(cur, true);
  256. }
  257. } else {
  258. var style = getStyle(cur, 'backgroundImage');
  259. if (style.indexOf('url(') > -1) {
  260. var urlPatt = /\(['"]?(.+?)['"]?\)/,
  261. url = style.match(urlPatt)[1];
  262. if (!isExternal(url)) {
  263. var temp = document.createElement('img');
  264. temp.src = url;
  265. desatIMG(temp, true, cur);
  266. }
  267. }
  268. }
  269. }
  270. };
  271. return init;
  272. })();