compress.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466
  1. "use strict";
  2. var _interopRequireDefault = require("@babel/runtime-corejs2/helpers/interopRequireDefault");
  3. var _Object$defineProperty = require("@babel/runtime-corejs2/core-js/object/define-property");
  4. _Object$defineProperty(exports, "__esModule", {
  5. value: true
  6. });
  7. exports["default"] = void 0;
  8. var _keys = _interopRequireDefault(require("@babel/runtime-corejs2/core-js/object/keys"));
  9. var _iterator = _interopRequireDefault(require("@babel/runtime-corejs2/core-js/symbol/iterator"));
  10. var _symbol = _interopRequireDefault(require("@babel/runtime-corejs2/core-js/symbol"));
  11. var _promise = _interopRequireDefault(require("@babel/runtime-corejs2/core-js/promise"));
  12. var _assign = _interopRequireDefault(require("@babel/runtime-corejs2/core-js/object/assign"));
  13. var _exifJs = require("exif-js");
  14. var _utils = require("./utils");
  15. var __assign = void 0 && (void 0).__assign || function () {
  16. __assign = _assign["default"] || function (t) {
  17. for (var s, i = 1, n = arguments.length; i < n; i++) {
  18. s = arguments[i];
  19. for (var p in s) {
  20. if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];
  21. }
  22. }
  23. return t;
  24. };
  25. return __assign.apply(this, arguments);
  26. };
  27. var __awaiter = void 0 && (void 0).__awaiter || function (thisArg, _arguments, P, generator) {
  28. function adopt(value) {
  29. return value instanceof P ? value : new P(function (resolve) {
  30. resolve(value);
  31. });
  32. }
  33. return new (P || (P = _promise["default"]))(function (resolve, reject) {
  34. function fulfilled(value) {
  35. try {
  36. step(generator.next(value));
  37. } catch (e) {
  38. reject(e);
  39. }
  40. }
  41. function rejected(value) {
  42. try {
  43. step(generator["throw"](value));
  44. } catch (e) {
  45. reject(e);
  46. }
  47. }
  48. function step(result) {
  49. result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected);
  50. }
  51. step((generator = generator.apply(thisArg, _arguments || [])).next());
  52. });
  53. };
  54. var __generator = void 0 && (void 0).__generator || function (thisArg, body) {
  55. var _ = {
  56. label: 0,
  57. sent: function sent() {
  58. if (t[0] & 1) throw t[1];
  59. return t[1];
  60. },
  61. trys: [],
  62. ops: []
  63. },
  64. f,
  65. y,
  66. t,
  67. g;
  68. return g = {
  69. next: verb(0),
  70. "throw": verb(1),
  71. "return": verb(2)
  72. }, typeof _symbol["default"] === "function" && (g[_iterator["default"]] = function () {
  73. return this;
  74. }), g;
  75. function verb(n) {
  76. return function (v) {
  77. return step([n, v]);
  78. };
  79. }
  80. function step(op) {
  81. if (f) throw new TypeError("Generator is already executing.");
  82. while (_) {
  83. try {
  84. if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
  85. if (y = 0, t) op = [op[0] & 2, t.value];
  86. switch (op[0]) {
  87. case 0:
  88. case 1:
  89. t = op;
  90. break;
  91. case 4:
  92. _.label++;
  93. return {
  94. value: op[1],
  95. done: false
  96. };
  97. case 5:
  98. _.label++;
  99. y = op[1];
  100. op = [0];
  101. continue;
  102. case 7:
  103. op = _.ops.pop();
  104. _.trys.pop();
  105. continue;
  106. default:
  107. if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) {
  108. _ = 0;
  109. continue;
  110. }
  111. if (op[0] === 3 && (!t || op[1] > t[0] && op[1] < t[3])) {
  112. _.label = op[1];
  113. break;
  114. }
  115. if (op[0] === 6 && _.label < t[1]) {
  116. _.label = t[1];
  117. t = op;
  118. break;
  119. }
  120. if (t && _.label < t[2]) {
  121. _.label = t[2];
  122. _.ops.push(op);
  123. break;
  124. }
  125. if (t[2]) _.ops.pop();
  126. _.trys.pop();
  127. continue;
  128. }
  129. op = body.call(thisArg, _);
  130. } catch (e) {
  131. op = [6, e];
  132. y = 0;
  133. } finally {
  134. f = t = 0;
  135. }
  136. }
  137. if (op[0] & 5) throw op[1];
  138. return {
  139. value: op[0] ? op[1] : void 0,
  140. done: true
  141. };
  142. }
  143. };
  144. var __read = void 0 && (void 0).__read || function (o, n) {
  145. var m = typeof _symbol["default"] === "function" && o[_iterator["default"]];
  146. if (!m) return o;
  147. var i = m.call(o),
  148. r,
  149. ar = [],
  150. e;
  151. try {
  152. while ((n === void 0 || n-- > 0) && !(r = i.next()).done) {
  153. ar.push(r.value);
  154. }
  155. } catch (error) {
  156. e = {
  157. error: error
  158. };
  159. } finally {
  160. try {
  161. if (r && !r.done && (m = i["return"])) m.call(i);
  162. } finally {
  163. if (e) throw e.error;
  164. }
  165. }
  166. return ar;
  167. };
  168. var __spread = void 0 && (void 0).__spread || function () {
  169. for (var ar = [], i = 0; i < arguments.length; i++) {
  170. ar = ar.concat(__read(arguments[i]));
  171. }
  172. return ar;
  173. };
  174. var mimeTypes = {
  175. PNG: 'image/png',
  176. JPEG: 'image/jpeg',
  177. WEBP: 'image/webp',
  178. BMP: 'image/bmp'
  179. };
  180. var maxSteps = 4;
  181. var scaleFactor = Math.log(2);
  182. var supportMimeTypes = (0, _keys["default"])(mimeTypes).map(function (type) {
  183. return mimeTypes[type];
  184. });
  185. var defaultType = mimeTypes.JPEG;
  186. function isSupportedType(type) {
  187. return supportMimeTypes.includes(type);
  188. }
  189. var Compress =
  190. /** @class */
  191. function () {
  192. function Compress(file, config) {
  193. this.file = file;
  194. this.config = config;
  195. this.config = __assign({
  196. quality: 0.92,
  197. noCompressIfLarger: false
  198. }, this.config);
  199. }
  200. Compress.prototype.process = function () {
  201. return __awaiter(this, void 0, void 0, function () {
  202. var srcDimension, originImage, canvas, scale, scaleCanvas, distBlob;
  203. return __generator(this, function (_a) {
  204. switch (_a.label) {
  205. case 0:
  206. this.outputType = this.file.type;
  207. srcDimension = {};
  208. if (!isSupportedType(this.file.type)) {
  209. throw new Error("unsupported file type: " + this.file.type);
  210. }
  211. return [4
  212. /*yield*/
  213. , this.getOriginImage()];
  214. case 1:
  215. originImage = _a.sent();
  216. return [4
  217. /*yield*/
  218. , this.getCanvas(originImage)];
  219. case 2:
  220. canvas = _a.sent();
  221. scale = 1;
  222. if (this.config.maxWidth) {
  223. scale = Math.min(1, this.config.maxWidth / canvas.width);
  224. }
  225. if (this.config.maxHeight) {
  226. scale = Math.min(1, scale, this.config.maxHeight / canvas.height);
  227. }
  228. srcDimension.width = canvas.width;
  229. srcDimension.height = canvas.height;
  230. return [4
  231. /*yield*/
  232. , this.doScale(canvas, scale)];
  233. case 3:
  234. scaleCanvas = _a.sent();
  235. distBlob = this.toBlob(scaleCanvas);
  236. if (distBlob.size > this.file.size && this.config.noCompressIfLarger) {
  237. return [2
  238. /*return*/
  239. , {
  240. dist: this.file,
  241. width: srcDimension.width,
  242. height: srcDimension.height
  243. }];
  244. }
  245. return [2
  246. /*return*/
  247. , {
  248. dist: distBlob,
  249. width: scaleCanvas.width,
  250. height: scaleCanvas.height
  251. }];
  252. }
  253. });
  254. });
  255. };
  256. Compress.prototype.clear = function (ctx, width, height) {
  257. // jpeg 没有 alpha 通道,透明区间会被填充成黑色,这里把透明区间填充为白色
  258. if (this.outputType === defaultType) {
  259. ctx.fillStyle = '#fff';
  260. ctx.fillRect(0, 0, width, height);
  261. } else {
  262. ctx.clearRect(0, 0, width, height);
  263. }
  264. };
  265. /** 通过 file 初始化 image 对象 */
  266. Compress.prototype.getOriginImage = function () {
  267. var _this = this;
  268. return new _promise["default"](function (resolve, reject) {
  269. var url = (0, _utils.createObjectURL)(_this.file);
  270. var img = new Image();
  271. img.onload = function () {
  272. resolve(img);
  273. };
  274. img.onerror = function () {
  275. reject('image load error');
  276. };
  277. img.src = url;
  278. });
  279. };
  280. Compress.prototype.getCanvas = function (img) {
  281. var _this = this;
  282. return new _promise["default"](function (resolve, reject) {
  283. // 通过得到图片的信息来调整显示方向以正确显示图片,主要解决 ios 系统上的图片会有旋转的问题
  284. _exifJs.EXIF.getData(img, function () {
  285. var orientation = _exifJs.EXIF.getTag(img, 'Orientation') || 1;
  286. var _a = (0, _utils.getTransform)(img, orientation),
  287. width = _a.width,
  288. height = _a.height,
  289. matrix = _a.matrix;
  290. var canvas = document.createElement('canvas');
  291. var context = canvas.getContext('2d');
  292. canvas.width = width;
  293. canvas.height = height;
  294. if (!context) {
  295. reject(new Error('context is null'));
  296. return;
  297. }
  298. _this.clear(context, width, height);
  299. context.transform.apply(context, __spread(matrix));
  300. context.drawImage(img, 0, 0);
  301. resolve(canvas);
  302. });
  303. });
  304. };
  305. Compress.prototype.doScale = function (source, scale) {
  306. return __awaiter(this, void 0, void 0, function () {
  307. var sctx, steps, factor, mirror, mctx, width, height, originWidth, originHeight, src, context, i, dw, dh, canvas, data;
  308. return __generator(this, function (_a) {
  309. if (scale === 1) {
  310. return [2
  311. /*return*/
  312. , source];
  313. }
  314. sctx = source.getContext('2d');
  315. steps = Math.min(maxSteps, Math.ceil(1 / scale / scaleFactor));
  316. factor = Math.pow(scale, 1 / steps);
  317. mirror = document.createElement('canvas');
  318. mctx = mirror.getContext('2d');
  319. width = source.width, height = source.height;
  320. originWidth = width;
  321. originHeight = height;
  322. mirror.width = width;
  323. mirror.height = height;
  324. if (!mctx || !sctx) {
  325. throw new Error("mctx or sctx can't be null");
  326. }
  327. for (i = 0; i < steps; i++) {
  328. dw = width * factor | 0 // eslint-disable-line no-bitwise
  329. ;
  330. dh = height * factor | 0 // eslint-disable-line no-bitwise
  331. ; // 到最后一步的时候 dw, dh 用目标缩放尺寸,否则会出现最后尺寸偏小的情况
  332. if (i === steps - 1) {
  333. dw = originWidth * scale;
  334. dh = originHeight * scale;
  335. }
  336. if (i % 2 === 0) {
  337. src = source;
  338. context = mctx;
  339. } else {
  340. src = mirror;
  341. context = sctx;
  342. } // 每次画前都清空,避免图像重叠
  343. this.clear(context, width, height);
  344. context.drawImage(src, 0, 0, width, height, 0, 0, dw, dh);
  345. width = dw;
  346. height = dh;
  347. }
  348. canvas = src === source ? mirror : source;
  349. data = context.getImageData(0, 0, width, height); // resize
  350. canvas.width = width;
  351. canvas.height = height; // store image data
  352. context.putImageData(data, 0, 0);
  353. return [2
  354. /*return*/
  355. , canvas];
  356. });
  357. });
  358. };
  359. /** 这里把 base64 字符串转为 blob 对象 */
  360. Compress.prototype.toBlob = function (result) {
  361. var dataURL = result.toDataURL(this.outputType, this.config.quality);
  362. var buffer = atob(dataURL.split(',')[1]).split('').map(function (_char) {
  363. return _char.charCodeAt(0);
  364. });
  365. var blob = new Blob([new Uint8Array(buffer)], {
  366. type: this.outputType
  367. });
  368. return blob;
  369. };
  370. return Compress;
  371. }();
  372. var compressImage = function compressImage(file, options) {
  373. return new Compress(file, options).process();
  374. };
  375. var _default = compressImage;
  376. exports["default"] = _default;