graphics/webimage.js

  1. 'use strict';
  2. var Thing = require('./thing.js');
  3. var UNDEFINED = -1;
  4. var NOT_LOADED = 0;
  5. var NUM_CHANNELS = 4;
  6. var RED = 0;
  7. var GREEN = 1;
  8. var BLUE = 2;
  9. var ALPHA = 3;
  10. // Keep track of cross origin WebImage URLs that have already been loaded
  11. // so we can take advantage of loading cross origin images from the browser cache
  12. var cachedCrossOriginURLs = {};
  13. /**
  14. * @constructor
  15. * @augments Thing
  16. * @param {string} filename - Filepath to the image
  17. */
  18. function WebImage(filename) {
  19. if (typeof filename !== 'string') {
  20. throw new TypeError(
  21. 'You must pass a string to <span class="code">' +
  22. "new WebImage(filename)</span> that has the image's URL."
  23. );
  24. }
  25. Thing.call(this);
  26. var self = this;
  27. this.image = new Image();
  28. // If the image is from a different origin, we need to request the image using
  29. // crossOrigin 'Anonymous', which allows WebImage to treat the image as
  30. // same origin and manipulate pixel data
  31. // https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/crossorigin
  32. var urlParser = document.createElement('a');
  33. urlParser.href = filename;
  34. var src = filename;
  35. if (urlParser.origin != window.location.origin) {
  36. this.image.crossOrigin = 'Anonymous';
  37. // If we've loaded this cross origin URL before, keep using the same URL
  38. if (cachedCrossOriginURLs.hasOwnProperty(filename)) {
  39. src = cachedCrossOriginURLs[filename];
  40. } else {
  41. // Otherwise we need to avoid the browser cache
  42. // Browser may have the image cached without the proper
  43. // Access-Control-Allow-Origin header on the resource
  44. // Ensure that we initiate a new crossOrigin 'anonymous' request for this
  45. // image, rather than pulling from browser cache, by making filename unique
  46. // We'll keep using this unique filename next time
  47. src = filename + '?time=' + Date.now();
  48. cachedCrossOriginURLs[filename] = src;
  49. }
  50. }
  51. this.imageLoaded = false;
  52. this.image.src = src;
  53. this.filename = filename;
  54. this.width = NOT_LOADED;
  55. this.height = NOT_LOADED;
  56. this.image.onload = function() {
  57. self.imageLoaded = true;
  58. self.checkDimensions();
  59. self.loadPixelData();
  60. if (self.loadfn) {
  61. self.loadfn();
  62. }
  63. };
  64. this.set = 0;
  65. this.type = 'WebImage';
  66. this.displayFromData = false;
  67. this.dirtyHiddenCanvas = false;
  68. this.data = NOT_LOADED;
  69. }
  70. WebImage.prototype = new Thing();
  71. WebImage.prototype.constructor = WebImage;
  72. /**
  73. * Set a function to be called when the WebImage is loaded.
  74. *
  75. * @param {function} callback - A function
  76. */
  77. WebImage.prototype.loaded = function(callback) {
  78. this.loadfn = callback;
  79. };
  80. /**
  81. * Set the image of the WebImage.
  82. *
  83. * @param {string} filename - Filepath to the image
  84. */
  85. WebImage.prototype.setImage = function(filename) {
  86. if (typeof filename !== 'string') {
  87. throw new TypeError(
  88. 'You must pass a string to <span class="code">' +
  89. "new WebImage(filename)</span> that has the image's URL."
  90. );
  91. }
  92. var self = this;
  93. this.image = new Image();
  94. // If the image is from a different origin, we need to request the image using
  95. // crossOrigin 'Anonymous', which allows WebImage to treat the image as
  96. // same origin and manipulate pixel data
  97. // https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/crossorigin
  98. var urlParser = document.createElement('a');
  99. urlParser.href = filename;
  100. var src = filename;
  101. if (urlParser.origin != window.location.origin) {
  102. this.image.crossOrigin = 'Anonymous';
  103. // If we've loaded this cross origin URL before, keep using the same URL
  104. if (cachedCrossOriginURLs.hasOwnProperty(filename)) {
  105. src = cachedCrossOriginURLs[filename];
  106. } else {
  107. // Otherwise we need to avoid the browser cache
  108. // Browser may have the image cached without the proper
  109. // Access-Control-Allow-Origin header on the resource
  110. // Ensure that we initiate a new crossOrigin 'anonymous' request for this
  111. // image, rather than pulling from browser cache, by making filename unique
  112. // We'll keep using this unique filename next time
  113. src = filename + '?time=' + Date.now();
  114. cachedCrossOriginURLs[filename] = src;
  115. }
  116. }
  117. this.imageLoaded = false;
  118. this.image.src = src;
  119. this.filename = filename;
  120. this.width = NOT_LOADED;
  121. this.height = NOT_LOADED;
  122. this.image.onload = function() {
  123. self.imageLoaded = true;
  124. self.checkDimensions();
  125. self.loadPixelData();
  126. if (self.loadfn) {
  127. self.loadfn();
  128. }
  129. };
  130. this.set = 0;
  131. this.displayFromData = false;
  132. this.dirtyHiddenCanvas = false;
  133. this.data = NOT_LOADED;
  134. };
  135. /**
  136. * Reinforce the dimensions of the WebImage based on the image it displays.
  137. */
  138. WebImage.prototype.checkDimensions = function() {
  139. if (this.width == NOT_LOADED && this.imageLoaded) {
  140. this.width = this.image.width;
  141. this.height = this.image.height;
  142. }
  143. };
  144. /**
  145. * Draws the WebImage in the canvas.
  146. *
  147. * @param {CodeHSGraphics} __graphics__ - Instance of the __graphics__ module.
  148. */
  149. WebImage.prototype.draw = function(__graphics__) {
  150. this.checkDimensions();
  151. var context = __graphics__.getContext('2d');
  152. // Scale and translate
  153. // X scale, X scew, Y scew, Y scale, X position, Y position
  154. context.setTransform(1, 0, 0, 1, this.x + this.width / 2, this.y + this.height / 2);
  155. context.rotate(this.rotation);
  156. // If we should be displaying the underlying pixel data, display that
  157. // Otherwise display the image
  158. var elemToDraw = this.image;
  159. if (this.displayFromData && this.data !== NOT_LOADED && this.hiddenCanvas) {
  160. // Update the in memory canvas with the latest pixel data if necessary
  161. if (this.dirtyHiddenCanvas) {
  162. var ctx = this.hiddenCanvas.getContext('2d');
  163. ctx.clearRect(0, 0, this.hiddenCanvas.width, this.hiddenCanvas.height);
  164. ctx.putImageData(this.data, 0, 0);
  165. this.dirtyHiddenCanvas = false;
  166. }
  167. elemToDraw = this.hiddenCanvas;
  168. }
  169. try {
  170. context.drawImage(elemToDraw, -this.width / 2, -this.height / 2, this.width, this.height);
  171. } catch (err) {
  172. throw new TypeError(
  173. 'Unable to create a WebImage from <span class="code">' +
  174. this.filename +
  175. '</span> ' +
  176. 'Make sure you have a valid image URL. ' +
  177. 'Hint: You can use More > Upload to upload your image and create a valid image URL.'
  178. );
  179. } finally {
  180. // Reset transformation matrix
  181. // X scale, X scew, Y scew, Y scale, X position, Y position
  182. context.setTransform(1, 0, 0, 1, 0, 0);
  183. }
  184. };
  185. /**
  186. * Return the underlying ImageData for this image.
  187. * Read more at https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/getImageData
  188. */
  189. WebImage.prototype.loadPixelData = function() {
  190. if (this.data === NOT_LOADED && this.imageLoaded) {
  191. try {
  192. // get the ImageData for this image
  193. this.hiddenCanvas = document.createElement('canvas');
  194. this.hiddenCanvas.width = this.width;
  195. this.hiddenCanvas.height = this.height;
  196. var ctx = this.hiddenCanvas.getContext('2d');
  197. ctx.drawImage(this.image, 0, 0, this.width, this.height);
  198. this.data = ctx.getImageData(0, 0, this.width, this.height);
  199. this.dirtyHiddenCanvas = false;
  200. } catch (err) {
  201. // NOTE: This should never happen now that we request images using
  202. // image.crossOrigin = 'Anonymous'
  203. // If the image was loaded, that means the external domain gave us CORS
  204. // access to the image and the browser will treat it as if it is same origin,
  205. // meaning we should be allowed to call 'getImageData'
  206. //
  207. // Just in case 'getImageData' fails,
  208. // Fail silently so we can still display the image from cross origin,
  209. // we just don't access the underlying image data
  210. this.data = NOT_LOADED;
  211. }
  212. }
  213. return this.data;
  214. };
  215. /**
  216. * Checks if the passed point is contained in the WebImage.
  217. *
  218. * @param {number} x - The x coordinate of the point being tested.
  219. * @param {number} y - The y coordinate of the point being tested.
  220. * @returns {boolean} Whether the passed point is contained in the WebImage.
  221. */
  222. WebImage.prototype.containsPoint = function(x, y) {
  223. return x >= this.x && x <= this.x + this.width && y >= this.y && y <= this.y + this.height;
  224. };
  225. /**
  226. * Gets the width of the WebImage.
  227. *
  228. * @returns {number} Width of the WebImage.
  229. */
  230. WebImage.prototype.getWidth = function() {
  231. return this.width;
  232. };
  233. /**
  234. * Gets the height of the WebImage.
  235. *
  236. * @returns {number} Height of the WebImage.
  237. */
  238. WebImage.prototype.getHeight = function() {
  239. return this.height;
  240. };
  241. /**
  242. * Sets the size of the WebImage.
  243. *
  244. * @param {number} width - The desired width of the resulting WebImage.
  245. * @param {number} height - The desired height of the resulting WebImage.
  246. */
  247. WebImage.prototype.setSize = function(width, height) {
  248. if (arguments.length !== 2) {
  249. throw new Error(
  250. 'You should pass exactly 2 arguments to <span ' +
  251. 'class="code">setSize(width, height)</span>'
  252. );
  253. }
  254. if (typeof width !== 'number' || !isFinite(width)) {
  255. throw new TypeError(
  256. 'Invalid value for <span class="code">width' +
  257. '</span>. Make sure you are passing finite numbers to <span ' +
  258. 'class="code">setSize(width, height)</span>. Did you ' +
  259. 'forget the parentheses in <span class="code">getWidth()</span> ' +
  260. 'or <span class="code">getHeight()</span>? Or did you perform a ' +
  261. 'calculation on a variable that is not a number?'
  262. );
  263. }
  264. if (typeof height !== 'number' || !isFinite(height)) {
  265. throw new TypeError(
  266. 'Invalid value for <span class="code">height' +
  267. '</span>. Make sure you are passing finite numbers to <span ' +
  268. 'class="code">setSize(width, height)</span>. Did you ' +
  269. 'forget the parentheses in <span class="code">getWidth()</span> ' +
  270. 'or <span class="code">getHeight()</span>? Or did you perform a ' +
  271. 'calculation on a variable that is not a number?'
  272. );
  273. }
  274. this.width = Math.max(0, width);
  275. this.height = Math.max(0, height);
  276. };
  277. /* Get and set pixel functions */
  278. /**
  279. * Gets a pixel at the given x and y coordinates.
  280. * Read more here:
  281. * https://developer.mozilla.org/en-US/docs/Web/API/ImageData/data
  282. *
  283. * @param {number} x - The x coordinate of the point being tested.
  284. * @param {number} y - The y coordinate of the point being tested.
  285. * @returns {array} An array of 4 numbers representing the (r,g,b,a) values
  286. * of the pixel at that coordinate.
  287. */
  288. WebImage.prototype.getPixel = function(x, y) {
  289. if (this.data === NOT_LOADED || x > this.width || x < 0 || y > this.height || y < 0) {
  290. var noPixel = [UNDEFINED, UNDEFINED, UNDEFINED, UNDEFINED];
  291. return noPixel;
  292. } else {
  293. var index = NUM_CHANNELS * (y * this.width + x);
  294. var pixel = [
  295. this.data.data[index + RED],
  296. this.data.data[index + GREEN],
  297. this.data.data[index + BLUE],
  298. this.data.data[index + ALPHA],
  299. ];
  300. return pixel;
  301. }
  302. };
  303. /**
  304. * Get the red value at a given location in the image.
  305. *
  306. * @param {number} x - The x coordinate of the point being tested.
  307. * @param {number} y - The y coordinate of the point being tested.
  308. * @returns {integer} An integer between 0 and 255.
  309. */
  310. WebImage.prototype.getRed = function(x, y) {
  311. return this.getPixel(x, y)[RED];
  312. };
  313. /**
  314. * Get the green value at a given location in the image.
  315. *
  316. * @param {number} x - The x coordinate of the point being tested.
  317. * @param {number} y - The y coordinate of the point being tested.
  318. * @returns {integer} An integer between 0 and 255.
  319. */
  320. WebImage.prototype.getGreen = function(x, y) {
  321. return this.getPixel(x, y)[GREEN];
  322. };
  323. /**
  324. * Get the blue value at a given location in the image.
  325. *
  326. * @param {number} x - The x coordinate of the point being tested.
  327. * @param {number} y - The y coordinate of the point being tested.
  328. * @returns {integer} An integer between 0 and 255.
  329. */
  330. WebImage.prototype.getBlue = function(x, y) {
  331. return this.getPixel(x, y)[BLUE];
  332. };
  333. /**
  334. * Get the alpha value at a given location in the image.
  335. *
  336. * @param {number} x - The x coordinate of the point being tested.
  337. * @param {number} y - The y coordinate of the point being tested.
  338. * @returns {integer} An integer between 0 and 255.
  339. */
  340. WebImage.prototype.getAlpha = function(x, y) {
  341. return this.getPixel(x, y)[ALPHA];
  342. };
  343. /**
  344. * Set the `component` value at a given location in the image to `val`.
  345. *
  346. * @param {number} x - The x coordinate of the point being tested.
  347. * @param {number} y - The y coordinate of the point being tested.
  348. * @param {integer} component - Integer representing the color value to
  349. * be set. R, G, B = 0, 1, 2, respectively.
  350. * @param {integer} val - The desired value of the `component` at the pixel.
  351. * Must be between 0 and 255.
  352. */
  353. WebImage.prototype.setPixel = function(x, y, component, val) {
  354. if (this.data !== NOT_LOADED && !(x < 0 || y < 0 || x > this.width || y > this.height)) {
  355. // Update the pixel value
  356. var index = NUM_CHANNELS * (y * this.width + x);
  357. this.data.data[index + component] = val;
  358. // Now that we have modified the image data, we need to display
  359. // the image based on the underlying image data rather than the
  360. // image url
  361. this.displayFromData = true;
  362. this.dirtyHiddenCanvas = true;
  363. }
  364. };
  365. /**
  366. * Set the red value at a given location in the image to `val`.
  367. *
  368. * @param {number} x - The x coordinate of the point being tested.
  369. * @param {number} y - The y coordinate of the point being tested.
  370. * @param {integer} val - The desired value of the red component at the pixel.
  371. * Must be between 0 and 255.
  372. */
  373. WebImage.prototype.setRed = function(x, y, val) {
  374. this.setPixel(x, y, RED, val);
  375. };
  376. /**
  377. * Set the green value at a given location in the image to `val`.
  378. *
  379. * @param {number} x - The x coordinate of the point being tested.
  380. * @param {number} y - The y coordinate of the point being tested.
  381. * @param {integer} val - The desired value of the green component at the pixel.
  382. * Must be between 0 and 255.
  383. */
  384. WebImage.prototype.setGreen = function(x, y, val) {
  385. this.setPixel(x, y, GREEN, val);
  386. };
  387. /**
  388. * Set the blue value at a given location in the image to `val`.
  389. *
  390. * @param {number} x - The x coordinate of the point being tested.
  391. * @param {number} y - The y coordinate of the point being tested.
  392. * @param {integer} val - The desired value of the blue component at the pixel.
  393. * Must be between 0 and 255.
  394. */
  395. WebImage.prototype.setBlue = function(x, y, val) {
  396. this.setPixel(x, y, BLUE, val);
  397. };
  398. /**
  399. * Set the alpha value at a given location in the image to `val`.
  400. *
  401. * @param {number} x - The x coordinate of the point being tested.
  402. * @param {number} y - The y coordinate of the point being tested.
  403. * @param {integer} val - The desired value of the alpha component at the
  404. * pixel.
  405. * Must be between 0 and 255.
  406. */
  407. WebImage.prototype.setAlpha = function(x, y, val) {
  408. this.setPixel(x, y, ALPHA, val);
  409. };
  410. module.exports = WebImage;