jquery.fileupload-image.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. /*
  2. * jQuery File Upload Image Preview & Resize Plugin 1.7.0
  3. * https://github.com/blueimp/jQuery-File-Upload
  4. *
  5. * Copyright 2013, Sebastian Tschan
  6. * https://blueimp.net
  7. *
  8. * Licensed under the MIT license:
  9. * http://www.opensource.org/licenses/MIT
  10. */
  11. /* jshint nomen:false */
  12. /* global define, window, Blob */
  13. (function (factory) {
  14. 'use strict';
  15. if (typeof define === 'function' && define.amd) {
  16. // Register as an anonymous AMD module:
  17. define([
  18. 'jquery',
  19. 'load-image',
  20. 'load-image-meta',
  21. 'load-image-exif',
  22. 'load-image-ios',
  23. 'canvas-to-blob',
  24. './jquery.fileupload-process'
  25. ], factory);
  26. } else {
  27. // Browser globals:
  28. factory(
  29. window.jQuery,
  30. window.loadImage
  31. );
  32. }
  33. }(function ($, loadImage) {
  34. 'use strict';
  35. // Prepend to the default processQueue:
  36. $.blueimp.fileupload.prototype.options.processQueue.unshift(
  37. {
  38. action: 'loadImageMetaData',
  39. disableImageHead: '@',
  40. disableExif: '@',
  41. disableExifThumbnail: '@',
  42. disableExifSub: '@',
  43. disableExifGps: '@',
  44. disabled: '@disableImageMetaDataLoad'
  45. },
  46. {
  47. action: 'loadImage',
  48. // Use the action as prefix for the "@" options:
  49. prefix: true,
  50. fileTypes: '@',
  51. maxFileSize: '@',
  52. noRevoke: '@',
  53. disabled: '@disableImageLoad'
  54. },
  55. {
  56. action: 'resizeImage',
  57. // Use "image" as prefix for the "@" options:
  58. prefix: 'image',
  59. maxWidth: '@',
  60. maxHeight: '@',
  61. minWidth: '@',
  62. minHeight: '@',
  63. crop: '@',
  64. orientation: '@',
  65. forceResize: '@',
  66. disabled: '@disableImageResize'
  67. },
  68. {
  69. action: 'saveImage',
  70. quality: '@imageQuality',
  71. type: '@imageType',
  72. disabled: '@disableImageResize'
  73. },
  74. {
  75. action: 'saveImageMetaData',
  76. disabled: '@disableImageMetaDataSave'
  77. },
  78. {
  79. action: 'resizeImage',
  80. // Use "preview" as prefix for the "@" options:
  81. prefix: 'preview',
  82. maxWidth: '@',
  83. maxHeight: '@',
  84. minWidth: '@',
  85. minHeight: '@',
  86. crop: '@',
  87. orientation: '@',
  88. thumbnail: '@',
  89. canvas: '@',
  90. disabled: '@disableImagePreview'
  91. },
  92. {
  93. action: 'setImage',
  94. name: '@imagePreviewName',
  95. disabled: '@disableImagePreview'
  96. },
  97. {
  98. action: 'deleteImageReferences',
  99. disabled: '@disableImageReferencesDeletion'
  100. }
  101. );
  102. // The File Upload Resize plugin extends the fileupload widget
  103. // with image resize functionality:
  104. $.widget('blueimp.fileupload', $.blueimp.fileupload, {
  105. options: {
  106. // The regular expression for the types of images to load:
  107. // matched against the file type:
  108. loadImageFileTypes: /^image\/(gif|jpeg|png)$/,
  109. // The maximum file size of images to load:
  110. loadImageMaxFileSize: 10000000, // 10MB
  111. // The maximum width of resized images:
  112. imageMaxWidth: 1920,
  113. // The maximum height of resized images:
  114. imageMaxHeight: 1080,
  115. // Defines the image orientation (1-8) or takes the orientation
  116. // value from Exif data if set to true:
  117. imageOrientation: false,
  118. // Define if resized images should be cropped or only scaled:
  119. imageCrop: false,
  120. // Disable the resize image functionality by default:
  121. disableImageResize: true,
  122. // The maximum width of the preview images:
  123. previewMaxWidth: 80,
  124. // The maximum height of the preview images:
  125. previewMaxHeight: 80,
  126. // Defines the preview orientation (1-8) or takes the orientation
  127. // value from Exif data if set to true:
  128. previewOrientation: true,
  129. // Create the preview using the Exif data thumbnail:
  130. previewThumbnail: true,
  131. // Define if preview images should be cropped or only scaled:
  132. previewCrop: false,
  133. // Define if preview images should be resized as canvas elements:
  134. previewCanvas: true
  135. },
  136. processActions: {
  137. // Loads the image given via data.files and data.index
  138. // as img element, if the browser supports the File API.
  139. // Accepts the options fileTypes (regular expression)
  140. // and maxFileSize (integer) to limit the files to load:
  141. loadImage: function (data, options) {
  142. if (options.disabled) {
  143. return data;
  144. }
  145. var that = this,
  146. file = data.files[data.index],
  147. dfd = $.Deferred();
  148. if (($.type(options.maxFileSize) === 'number' &&
  149. file.size > options.maxFileSize) ||
  150. (options.fileTypes &&
  151. !options.fileTypes.test(file.type)) ||
  152. !loadImage(
  153. file,
  154. function (img) {
  155. if (img.src) {
  156. data.img = img;
  157. }
  158. dfd.resolveWith(that, [data]);
  159. },
  160. options
  161. )) {
  162. return data;
  163. }
  164. return dfd.promise();
  165. },
  166. // Resizes the image given as data.canvas or data.img
  167. // and updates data.canvas or data.img with the resized image.
  168. // Also stores the resized image as preview property.
  169. // Accepts the options maxWidth, maxHeight, minWidth,
  170. // minHeight, canvas and crop:
  171. resizeImage: function (data, options) {
  172. if (options.disabled || !(data.canvas || data.img)) {
  173. return data;
  174. }
  175. options = $.extend({canvas: true}, options);
  176. var that = this,
  177. dfd = $.Deferred(),
  178. img = (options.canvas && data.canvas) || data.img,
  179. resolve = function (newImg) {
  180. if (newImg && (newImg.width !== img.width ||
  181. newImg.height !== img.height ||
  182. options.forceResize)) {
  183. data[newImg.getContext ? 'canvas' : 'img'] = newImg;
  184. }
  185. data.preview = newImg;
  186. dfd.resolveWith(that, [data]);
  187. },
  188. thumbnail;
  189. if (data.exif) {
  190. if (options.orientation === true) {
  191. options.orientation = data.exif.get('Orientation');
  192. }
  193. if (options.thumbnail) {
  194. thumbnail = data.exif.get('Thumbnail');
  195. if (thumbnail) {
  196. loadImage(thumbnail, resolve, options);
  197. return dfd.promise();
  198. }
  199. }
  200. }
  201. if (img) {
  202. resolve(loadImage.scale(img, options));
  203. return dfd.promise();
  204. }
  205. return data;
  206. },
  207. // Saves the processed image given as data.canvas
  208. // inplace at data.index of data.files:
  209. saveImage: function (data, options) {
  210. if (!data.canvas || options.disabled) {
  211. return data;
  212. }
  213. var that = this,
  214. file = data.files[data.index],
  215. dfd = $.Deferred();
  216. if (data.canvas.toBlob) {
  217. data.canvas.toBlob(
  218. function (blob) {
  219. if (!blob.name) {
  220. if (file.type === blob.type) {
  221. blob.name = file.name;
  222. } else if (file.name) {
  223. blob.name = file.name.replace(
  224. /\..+$/,
  225. '.' + blob.type.substr(6)
  226. );
  227. }
  228. }
  229. // Don't restore invalid meta data:
  230. if (file.type !== blob.type) {
  231. delete data.imageHead;
  232. }
  233. // Store the created blob at the position
  234. // of the original file in the files list:
  235. data.files[data.index] = blob;
  236. dfd.resolveWith(that, [data]);
  237. },
  238. options.type || file.type,
  239. options.quality
  240. );
  241. } else {
  242. return data;
  243. }
  244. return dfd.promise();
  245. },
  246. loadImageMetaData: function (data, options) {
  247. if (options.disabled) {
  248. return data;
  249. }
  250. var that = this,
  251. dfd = $.Deferred();
  252. loadImage.parseMetaData(data.files[data.index], function (result) {
  253. $.extend(data, result);
  254. dfd.resolveWith(that, [data]);
  255. }, options);
  256. return dfd.promise();
  257. },
  258. saveImageMetaData: function (data, options) {
  259. if (!(data.imageHead && data.canvas &&
  260. data.canvas.toBlob && !options.disabled)) {
  261. return data;
  262. }
  263. var file = data.files[data.index],
  264. blob = new Blob([
  265. data.imageHead,
  266. // Resized images always have a head size of 20 bytes,
  267. // including the JPEG marker and a minimal JFIF header:
  268. this._blobSlice.call(file, 20)
  269. ], {type: file.type});
  270. blob.name = file.name;
  271. data.files[data.index] = blob;
  272. return data;
  273. },
  274. // Sets the resized version of the image as a property of the
  275. // file object, must be called after "saveImage":
  276. setImage: function (data, options) {
  277. if (data.preview && !options.disabled) {
  278. data.files[data.index][options.name || 'preview'] = data.preview;
  279. }
  280. return data;
  281. },
  282. deleteImageReferences: function (data, options) {
  283. if (!options.disabled) {
  284. delete data.img;
  285. delete data.canvas;
  286. delete data.preview;
  287. delete data.imageHead;
  288. }
  289. return data;
  290. }
  291. }
  292. });
  293. }));