painter.vue 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983
  1. <template>
  2. <view :style="'position: relative;' + customStyle + ';' + painterStyle">
  3. <canvas canvas-id="photo" :style="photoStyle + ';position: absolute; left: -9999px; top: -9999rpx;'" />
  4. <canvas canvas-id="bottom" :style="painterStyle + ';position: absolute;'" />
  5. <canvas canvas-id="k-canvas" :style="painterStyle + ';position: absolute;'" />
  6. <canvas canvas-id="top" :style="painterStyle + ';position: absolute;'" />
  7. <canvas
  8. canvas-id="front"
  9. :style="painterStyle + ';position: absolute;'"
  10. @touchstart="onTouchStart"
  11. @touchmove="onTouchMove"
  12. @touchend="onTouchEnd"
  13. @touchcancel="onTouchCancel"
  14. :disable-scroll="true"
  15. />
  16. </view>
  17. </template>
  18. <script>
  19. import Pen from './lib/pen';
  20. import Downloader from './lib/downloader';
  21. const util = require('./lib/util');
  22. const downloader = new Downloader(); // 最大尝试的绘制次数
  23. const MAX_PAINT_COUNT = 5;
  24. const ACTION_DEFAULT_SIZE = 24;
  25. const ACTION_OFFSET = '2rpx';
  26. export default {
  27. data() {
  28. return {
  29. picURL: '',
  30. showCanvas: true,
  31. painterStyle: '',
  32. photoStyle: ''
  33. };
  34. },
  35. canvasWidthInPx: 0,
  36. canvasHeightInPx: 0,
  37. paintCount: 0,
  38. currentPalette: {},
  39. movingCache: {},
  40. outterDisabled: false,
  41. isDisabled: false,
  42. needClear: false,
  43. /**
  44. * 组件的属性列表
  45. */
  46. props: {
  47. customStyle: {
  48. type: String
  49. },
  50. // 运行自定义选择框和删除缩放按钮
  51. customActionStyle: {
  52. type: Object
  53. },
  54. palette: {
  55. type: Object
  56. },
  57. dancePalette: {
  58. type: Object
  59. },
  60. // 缩放比,会在传入的 palette 中统一乘以该缩放比
  61. scaleRatio: {
  62. type: Number,
  63. default: 1
  64. },
  65. widthPixels: {
  66. type: Number,
  67. default: 1200
  68. },
  69. // 启用脏检查,默认 false
  70. dirty: {
  71. type: Boolean,
  72. default: false
  73. },
  74. LRU: {
  75. type: Boolean,
  76. default: true
  77. },
  78. action: {
  79. type: Object
  80. },
  81. disableAction: {
  82. type: Boolean
  83. },
  84. clearActionBox: {
  85. type: Boolean
  86. }
  87. },
  88. methods: {
  89. /**
  90. * 判断一个 object 是否为 空
  91. * @param {object} object
  92. */
  93. isEmpty(object) {
  94. for (const i in object) {
  95. return false;
  96. }
  97. return true;
  98. },
  99. isNeedRefresh(newVal, oldVal) {
  100. if (!newVal || this.isEmpty(newVal) || (this.dirty && util.equal(newVal, oldVal))) {
  101. return false;
  102. }
  103. return true;
  104. },
  105. getBox(rect, type) {
  106. const boxArea = {
  107. type: 'rect',
  108. css: {
  109. height: `${rect.bottom - rect.top}px`,
  110. width: `${rect.right - rect.left}px`,
  111. left: `${rect.left}px`,
  112. top: `${rect.top}px`,
  113. borderWidth: '4rpx',
  114. borderColor: '#1A7AF8',
  115. color: 'transparent'
  116. }
  117. };
  118. if (type === 'text') {
  119. boxArea.css = Object.assign({}, boxArea.css, {
  120. borderStyle: 'dashed'
  121. });
  122. }
  123. if (this.customActionStyle && this.customActionStyle.border) {
  124. boxArea.css = Object.assign({}, boxArea.css, this.customActionStyle.border);
  125. }
  126. Object.assign(boxArea, {
  127. id: 'box'
  128. });
  129. return boxArea;
  130. },
  131. getScaleIcon(rect, type) {
  132. let scaleArea = {};
  133. const { customActionStyle } = this;
  134. if (customActionStyle && customActionStyle.scale) {
  135. scaleArea = {
  136. type: 'image',
  137. url: type === 'text' ? customActionStyle.scale.textIcon : customActionStyle.scale.imageIcon,
  138. css: {
  139. height: `${2 * ACTION_DEFAULT_SIZE}rpx`,
  140. width: `${2 * ACTION_DEFAULT_SIZE}rpx`,
  141. borderRadius: `${ACTION_DEFAULT_SIZE}rpx`
  142. }
  143. };
  144. } else {
  145. scaleArea = {
  146. type: 'rect',
  147. css: {
  148. height: `${2 * ACTION_DEFAULT_SIZE}rpx`,
  149. width: `${2 * ACTION_DEFAULT_SIZE}rpx`,
  150. borderRadius: `${ACTION_DEFAULT_SIZE}rpx`,
  151. color: '#0000ff'
  152. }
  153. };
  154. }
  155. scaleArea.css = Object.assign({}, scaleArea.css, {
  156. align: 'center',
  157. left: `${rect.right + ACTION_OFFSET.toPx()}px`,
  158. top:
  159. type === 'text'
  160. ? `${rect.top - ACTION_OFFSET.toPx() - scaleArea.css.height.toPx() / 2}px`
  161. : `${rect.bottom - ACTION_OFFSET.toPx() - scaleArea.css.height.toPx() / 2}px`
  162. });
  163. Object.assign(scaleArea, {
  164. id: 'scale'
  165. });
  166. return scaleArea;
  167. },
  168. getDeleteIcon(rect) {
  169. let deleteArea = {};
  170. const { customActionStyle } = this;
  171. if (customActionStyle && customActionStyle.scale) {
  172. deleteArea = {
  173. type: 'image',
  174. url: customActionStyle.delete.icon,
  175. css: {
  176. height: `${2 * ACTION_DEFAULT_SIZE}rpx`,
  177. width: `${2 * ACTION_DEFAULT_SIZE}rpx`,
  178. borderRadius: `${ACTION_DEFAULT_SIZE}rpx`
  179. }
  180. };
  181. } else {
  182. deleteArea = {
  183. type: 'rect',
  184. css: {
  185. height: `${2 * ACTION_DEFAULT_SIZE}rpx`,
  186. width: `${2 * ACTION_DEFAULT_SIZE}rpx`,
  187. borderRadius: `${ACTION_DEFAULT_SIZE}rpx`,
  188. color: '#0000ff'
  189. }
  190. };
  191. }
  192. deleteArea.css = Object.assign({}, deleteArea.css, {
  193. align: 'center',
  194. left: `${rect.left - ACTION_OFFSET.toPx()}px`,
  195. top: `${rect.top - ACTION_OFFSET.toPx() - deleteArea.css.height.toPx() / 2}px`
  196. });
  197. Object.assign(deleteArea, {
  198. id: 'delete'
  199. });
  200. return deleteArea;
  201. },
  202. doAction(action, callback, isMoving, overwrite) {
  203. let newVal = null;
  204. if (action) {
  205. newVal = action.view;
  206. }
  207. if (newVal && newVal.id && this.touchedView.id !== newVal.id) {
  208. // 带 id 的动作给撤回时使用,不带 id,表示对当前选中对象进行操作
  209. const { views } = this.currentPalette;
  210. for (let i = 0; i < views.length; i++) {
  211. if (views[i].id === newVal.id) {
  212. // 跨层回撤,需要重新构建三层关系
  213. this.touchedView = views[i];
  214. this.findedIndex = i;
  215. this.sliceLayers();
  216. break;
  217. }
  218. }
  219. }
  220. const doView = this.touchedView;
  221. if (!doView || this.isEmpty(doView)) {
  222. return;
  223. }
  224. if (newVal && newVal.css) {
  225. if (overwrite) {
  226. doView.css = newVal.css;
  227. } else if (Array.isArray(doView.css) && Array.isArray(newVal.css)) {
  228. doView.css = Object.assign({}, ...doView.css, ...newVal.css);
  229. } else if (Array.isArray(doView.css)) {
  230. doView.css = Object.assign({}, ...doView.css, newVal.css);
  231. } else if (Array.isArray(newVal.css)) {
  232. doView.css = Object.assign({}, doView.css, ...newVal.css);
  233. } else {
  234. doView.css = Object.assign({}, doView.css, newVal.css);
  235. }
  236. }
  237. if (newVal && newVal.rect) {
  238. doView.rect = newVal.rect;
  239. }
  240. if (newVal && newVal.url && doView.url && newVal.url !== doView.url) {
  241. downloader
  242. .download(newVal.url, this.LRU)
  243. .then((path) => {
  244. if (newVal.url.startsWith('https')) {
  245. doView.originUrl = newVal.url;
  246. }
  247. doView.url = path;
  248. uni.getImageInfo({
  249. src: path,
  250. success: (res) => {
  251. doView.sHeight = res.height;
  252. doView.sWidth = res.width;
  253. this.reDraw(doView, callback, isMoving);
  254. },
  255. fail: () => {
  256. this.reDraw(doView, callback, isMoving);
  257. }
  258. });
  259. })
  260. .catch((error) => {
  261. // 未下载成功,直接绘制
  262. console.error(error);
  263. this.reDraw(doView, callback, isMoving);
  264. });
  265. } else {
  266. if (newVal && newVal.text && doView.text && newVal.text !== doView.text) {
  267. doView.text = newVal.text;
  268. }
  269. if (newVal && newVal.content && doView.content && newVal.content !== doView.content) {
  270. doView.content = newVal.content;
  271. }
  272. this.reDraw(doView, callback, isMoving);
  273. }
  274. },
  275. reDraw(doView, callback, isMoving) {
  276. const draw = {
  277. width: this.currentPalette.width,
  278. height: this.currentPalette.height,
  279. views: this.isEmpty(doView) ? [] : [doView]
  280. };
  281. const pen = new Pen(this.globalContext, draw);
  282. if (isMoving && doView.type === 'text') {
  283. pen.paint(
  284. (callbackInfo) => {
  285. if (callback) {
  286. callback(callbackInfo);
  287. }
  288. this.$emit('viewUpdate', {
  289. detail: {
  290. view: this.touchedView
  291. }
  292. });
  293. },
  294. true,
  295. this.movingCache
  296. );
  297. } else {
  298. // 某些机型(华为 P20)非移动和缩放场景下,只绘制一遍会偶然性图片绘制失败
  299. if (!isMoving && !this.isScale) {
  300. pen.paint();
  301. }
  302. pen.paint((callbackInfo) => {
  303. if (callback) {
  304. callback(callbackInfo);
  305. }
  306. this.$emit('viewUpdate', {
  307. detail: {
  308. view: this.touchedView
  309. }
  310. });
  311. });
  312. }
  313. const { rect, css, type } = doView;
  314. this.block = {
  315. width: this.currentPalette.width,
  316. height: this.currentPalette.height,
  317. views: this.isEmpty(doView) ? [] : [this.getBox(rect, doView.type)]
  318. };
  319. if (css && css.scalable) {
  320. this.block.views.push(this.getScaleIcon(rect, type));
  321. }
  322. if (css && css.deletable) {
  323. this.block.views.push(this.getDeleteIcon(rect));
  324. }
  325. const topBlock = new Pen(this.frontContext, this.block);
  326. topBlock.paint();
  327. },
  328. isInView(x, y, rect) {
  329. return x > rect.left && y > rect.top && x < rect.right && y < rect.bottom;
  330. },
  331. isInDelete(x, y) {
  332. for (const view of this.block.views) {
  333. if (view.id === 'delete') {
  334. return x > view.rect.left && y > view.rect.top && x < view.rect.right && y < view.rect.bottom;
  335. }
  336. }
  337. return false;
  338. },
  339. isInScale(x, y) {
  340. for (const view of this.block.views) {
  341. if (view.id === 'scale') {
  342. return x > view.rect.left && y > view.rect.top && x < view.rect.right && y < view.rect.bottom;
  343. }
  344. }
  345. return false;
  346. },
  347. touchedView: {},
  348. findedIndex: -1,
  349. onClick() {
  350. const x = this.startX;
  351. const y = this.startY;
  352. const totalLayerCount = this.currentPalette.views.length;
  353. let canBeTouched = [];
  354. let isDelete = false;
  355. let deleteIndex = -1;
  356. for (let i = totalLayerCount - 1; i >= 0; i--) {
  357. const view = this.currentPalette.views[i];
  358. const { rect } = view;
  359. if (this.touchedView && this.touchedView.id && this.touchedView.id === view.id && this.isInDelete(x, y, rect)) {
  360. canBeTouched.length = 0;
  361. deleteIndex = i;
  362. isDelete = true;
  363. break;
  364. }
  365. if (this.isInView(x, y, rect)) {
  366. canBeTouched.push({
  367. view,
  368. index: i
  369. });
  370. }
  371. }
  372. this.touchedView = {};
  373. if (canBeTouched.length === 0) {
  374. this.findedIndex = -1;
  375. } else {
  376. let i = 0;
  377. const touchAble = canBeTouched.filter((item) => Boolean(item.view.id));
  378. if (touchAble.length === 0) {
  379. this.findedIndex = canBeTouched[0].index;
  380. } else {
  381. for (i = 0; i < touchAble.length; i++) {
  382. if (this.findedIndex === touchAble[i].index) {
  383. i++;
  384. break;
  385. }
  386. }
  387. if (i === touchAble.length) {
  388. i = 0;
  389. }
  390. this.touchedView = touchAble[i].view;
  391. this.findedIndex = touchAble[i].index;
  392. this.$emit('viewClicked', {
  393. detail: {
  394. view: this.touchedView
  395. }
  396. });
  397. }
  398. }
  399. if (this.findedIndex < 0 || (this.touchedView && !this.touchedView.id)) {
  400. // 证明点击了背景 或无法移动的view
  401. this.frontContext.draw();
  402. if (isDelete) {
  403. this.$emit('touchEnd', {
  404. detail: {
  405. view: this.currentPalette.views[deleteIndex],
  406. index: deleteIndex,
  407. type: 'delete'
  408. }
  409. });
  410. this.doAction();
  411. } else if (this.findedIndex < 0) {
  412. this.$emit('viewClicked', {
  413. detail: {}
  414. });
  415. }
  416. this.findedIndex = -1;
  417. this.prevFindedIndex = -1;
  418. } else if (this.touchedView && this.touchedView.id) {
  419. this.sliceLayers();
  420. }
  421. },
  422. sliceLayers() {
  423. const bottomLayers = this.currentPalette.views.slice(0, this.findedIndex);
  424. const topLayers = this.currentPalette.views.slice(this.findedIndex + 1);
  425. const bottomDraw = {
  426. width: this.currentPalette.width,
  427. height: this.currentPalette.height,
  428. background: this.currentPalette.background,
  429. views: bottomLayers
  430. };
  431. const topDraw = {
  432. width: this.currentPalette.width,
  433. height: this.currentPalette.height,
  434. views: topLayers
  435. };
  436. if (this.prevFindedIndex < this.findedIndex) {
  437. new Pen(this.bottomContext, bottomDraw).paint();
  438. this.doAction(null, (callbackInfo) => {
  439. this.movingCache = callbackInfo;
  440. });
  441. new Pen(this.topContext, topDraw).paint();
  442. } else {
  443. new Pen(this.topContext, topDraw).paint();
  444. this.doAction(null, (callbackInfo) => {
  445. this.movingCache = callbackInfo;
  446. });
  447. new Pen(this.bottomContext, bottomDraw).paint();
  448. }
  449. this.prevFindedIndex = this.findedIndex;
  450. },
  451. startX: 0,
  452. startY: 0,
  453. startH: 0,
  454. startW: 0,
  455. isScale: false,
  456. startTimeStamp: 0,
  457. onTouchStart(event) {
  458. if (this.isDisabled) {
  459. return;
  460. }
  461. const { x, y } = event.touches[0];
  462. this.startX = x;
  463. this.startY = y;
  464. this.startTimeStamp = new Date().getTime();
  465. if (this.touchedView && !this.isEmpty(this.touchedView)) {
  466. const { rect } = this.touchedView;
  467. if (this.isInScale(x, y, rect)) {
  468. this.isScale = true;
  469. this.movingCache = {};
  470. this.startH = rect.bottom - rect.top;
  471. this.startW = rect.right - rect.left;
  472. } else {
  473. this.isScale = false;
  474. }
  475. } else {
  476. this.isScale = false;
  477. }
  478. },
  479. onTouchEnd(e) {
  480. if (this.isDisabled) {
  481. return;
  482. }
  483. const current = new Date().getTime();
  484. if (current - this.startTimeStamp <= 500 && !this.hasMove) {
  485. if (!this.isScale) {
  486. this.onClick(e);
  487. }
  488. } else if (this.touchedView && !this.isEmpty(this.touchedView)) {
  489. this.$emit('touchEnd', {
  490. detail: {
  491. view: this.touchedView
  492. }
  493. });
  494. }
  495. this.hasMove = false;
  496. },
  497. onTouchCancel(e) {
  498. if (this.isDisabled) {
  499. return;
  500. }
  501. this.onTouchEnd(e);
  502. },
  503. hasMove: false,
  504. onTouchMove(event) {
  505. if (this.isDisabled) {
  506. return;
  507. }
  508. this.hasMove = true;
  509. if (!this.touchedView || (this.touchedView && !this.touchedView.id)) {
  510. return;
  511. }
  512. const { x, y } = event.touches[0];
  513. const offsetX = x - this.startX;
  514. const offsetY = y - this.startY;
  515. const { rect, type } = this.touchedView;
  516. let css = {};
  517. if (this.isScale) {
  518. const newW = this.startW + offsetX > 1 ? this.startW + offsetX : 1;
  519. if (this.touchedView.css && this.touchedView.css.minWidth) {
  520. if (newW < this.touchedView.css.minWidth.toPx()) {
  521. return;
  522. }
  523. }
  524. if (this.touchedView.rect && this.touchedView.rect.minWidth) {
  525. if (newW < this.touchedView.rect.minWidth) {
  526. return;
  527. }
  528. }
  529. const newH = this.startH + offsetY > 1 ? this.startH + offsetY : 1;
  530. css = {
  531. width: `${newW}px`
  532. };
  533. if (type !== 'text') {
  534. if (type === 'image') {
  535. css.height = `${(newW * this.startH) / this.startW}px`;
  536. } else {
  537. css.height = `${newH}px`;
  538. }
  539. }
  540. } else {
  541. this.startX = x;
  542. this.startY = y;
  543. css = {
  544. left: `${rect.x + offsetX}px`,
  545. top: `${rect.y + offsetY}px`,
  546. right: undefined,
  547. bottom: undefined
  548. };
  549. }
  550. this.doAction(
  551. {
  552. view: {
  553. css
  554. }
  555. },
  556. (callbackInfo) => {
  557. if (this.isScale) {
  558. this.movingCache = callbackInfo;
  559. }
  560. },
  561. !this.isScale
  562. );
  563. },
  564. initScreenK() {
  565. if (!(getApp() && getApp().globalData.systemInfo && getApp().globalData.systemInfo.screenWidth)) {
  566. try {
  567. getApp().globalData.systemInfo = uni.getSystemInfoSync();
  568. } catch (e) {
  569. console.error(`Painter get system info failed, ${JSON.stringify(e)}`);
  570. return;
  571. }
  572. }
  573. this.screenK = 0.5;
  574. if (getApp() && getApp().globalData.systemInfo && getApp().globalData.systemInfo.screenWidth) {
  575. this.screenK = getApp().globalData.systemInfo.screenWidth / 750;
  576. }
  577. setStringPrototype(this.screenK, this.scaleRatio);
  578. },
  579. initDancePalette() {
  580. this.isDisabled = true;
  581. this.initScreenK();
  582. this.downloadImages(this.dancePalette).then((palette) => {
  583. this.currentPalette = palette;
  584. const { width, height } = palette;
  585. if (!width || !height) {
  586. console.error(`You should set width and height correctly for painter, width: ${width}, height: ${height}`);
  587. return;
  588. }
  589. this.setData({
  590. painterStyle: `width:${width.toPx()}px;height:${height.toPx()}px;`
  591. });
  592. this.frontContext || (this.frontContext = uni.createCanvasContext('front', this));
  593. this.bottomContext || (this.bottomContext = uni.createCanvasContext('bottom', this));
  594. this.topContext || (this.topContext = uni.createCanvasContext('top', this));
  595. this.globalContext || (this.globalContext = uni.createCanvasContext('k-canvas', this));
  596. new Pen(this.bottomContext, palette).paint(() => {
  597. this.isDisabled = false;
  598. this.isDisabled = this.outterDisabled;
  599. this.$emit('didShow');
  600. });
  601. this.globalContext.draw();
  602. this.frontContext.draw();
  603. this.topContext.draw();
  604. });
  605. this.touchedView = {};
  606. },
  607. startPaint() {
  608. this.initScreenK();
  609. this.downloadImages(this.palette).then((palette) => {
  610. const { width, height } = palette;
  611. if (!width || !height) {
  612. console.error(`You should set width and height correctly for painter, width: ${width}, height: ${height}`);
  613. return;
  614. } // 生成图片时,根据设置的像素值重新绘制
  615. this.canvasWidthInPx = width.toPx();
  616. if (this.widthPixels) {
  617. setStringPrototype(this.screenK, this.widthPixels / this.canvasWidthInPx);
  618. this.canvasWidthInPx = this.widthPixels;
  619. }
  620. this.canvasHeightInPx = height.toPx();
  621. this.setData({
  622. photoStyle: `width:${this.canvasWidthInPx}px;height:${this.canvasHeightInPx}px;`
  623. });
  624. this.photoContext || (this.photoContext = uni.createCanvasContext('photo', this));
  625. new Pen(this.photoContext, palette).paint(() => {
  626. this.saveImgToLocal();
  627. });
  628. setStringPrototype(this.screenK, this.scaleRatio);
  629. });
  630. },
  631. downloadImages(palette) {
  632. return new Promise((resolve, reject) => {
  633. let preCount = 0;
  634. let completeCount = 0;
  635. const paletteCopy = JSON.parse(JSON.stringify(palette));
  636. if (paletteCopy.background) {
  637. preCount++;
  638. downloader.download(paletteCopy.background, this.LRU).then(
  639. (path) => {
  640. paletteCopy.background = path;
  641. completeCount++;
  642. if (preCount === completeCount) {
  643. resolve(paletteCopy);
  644. }
  645. },
  646. () => {
  647. completeCount++;
  648. if (preCount === completeCount) {
  649. resolve(paletteCopy);
  650. }
  651. }
  652. );
  653. }
  654. if (paletteCopy.views) {
  655. for (const view of paletteCopy.views) {
  656. if (view && view.type === 'image' && view.url) {
  657. preCount++;
  658. /* eslint-disable no-loop-func */
  659. downloader.download(view.url, this.LRU).then(
  660. (path) => {
  661. view.originUrl = view.url;
  662. view.url = path;
  663. uni.getImageInfo({
  664. src: path,
  665. success: (res) => {
  666. // 获得一下图片信息,供后续裁减使用
  667. view.sWidth = res.width;
  668. view.sHeight = res.height;
  669. },
  670. fail: (error) => {
  671. // 如果图片坏了,则直接置空,防止坑爹的 canvas 画崩溃了
  672. view.url = '';
  673. console.error(`getImageInfo ${view.url} failed, ${JSON.stringify(error)}`);
  674. },
  675. complete: () => {
  676. completeCount++;
  677. if (preCount === completeCount) {
  678. resolve(paletteCopy);
  679. }
  680. }
  681. });
  682. },
  683. () => {
  684. completeCount++;
  685. if (preCount === completeCount) {
  686. resolve(paletteCopy);
  687. }
  688. }
  689. );
  690. }
  691. }
  692. }
  693. if (preCount === 0) {
  694. resolve(paletteCopy);
  695. }
  696. });
  697. },
  698. saveImgToLocal() {
  699. const that = this;
  700. setTimeout(() => {
  701. uni.canvasToTempFilePath(
  702. {
  703. canvasId: 'photo',
  704. destWidth: that.canvasWidthInPx,
  705. destHeight: that.canvasHeightInPx,
  706. success: function (res) {
  707. that.getImageInfo(res.tempFilePath);
  708. },
  709. fail: function (error) {
  710. console.error(`canvasToTempFilePath failed, ${JSON.stringify(error)}`);
  711. that.$emit('imgErr', {
  712. detail: {
  713. error: error
  714. }
  715. });
  716. }
  717. },
  718. this
  719. );
  720. }, 300);
  721. },
  722. getImageInfo(filePath) {
  723. const that = this;
  724. uni.getImageInfo({
  725. src: filePath,
  726. success: (infoRes) => {
  727. if (that.paintCount > MAX_PAINT_COUNT) {
  728. const error = `The result is always fault, even we tried ${MAX_PAINT_COUNT} times`;
  729. console.error(error);
  730. that.$emit('imgErr', {
  731. detail: {
  732. error: error
  733. }
  734. });
  735. return;
  736. } // 比例相符时才证明绘制成功,否则进行强制重绘制
  737. if (Math.abs((infoRes.width * that.canvasHeightInPx - that.canvasWidthInPx * infoRes.height) / (infoRes.height * that.canvasHeightInPx)) < 0.01) {
  738. that.$emit('imgOK', {
  739. detail: {
  740. path: filePath
  741. }
  742. });
  743. } else {
  744. that.startPaint();
  745. }
  746. that.paintCount++;
  747. },
  748. fail: (error) => {
  749. console.error(`getImageInfo failed, ${JSON.stringify(error)}`);
  750. that.$emit('imgErr', {
  751. detail: {
  752. error: error
  753. }
  754. });
  755. }
  756. });
  757. }
  758. },
  759. watch: {
  760. palette: {
  761. handler: function (newVal, oldVal) {
  762. if (this.isNeedRefresh(newVal, oldVal)) {
  763. this.paintCount = 0;
  764. this.startPaint();
  765. }
  766. },
  767. immediate: true,
  768. deep: true
  769. },
  770. dancePalette: {
  771. handler: function (newVal, oldVal) {
  772. if (!this.isEmpty(newVal)) {
  773. this.initDancePalette(newVal);
  774. }
  775. },
  776. immediate: true,
  777. deep: true
  778. },
  779. action: {
  780. handler: function (newVal, oldVal) {
  781. if (newVal && !this.isEmpty(newVal)) {
  782. this.doAction(
  783. newVal,
  784. (callbackInfo) => {
  785. this.movingCache = callbackInfo;
  786. },
  787. false,
  788. true
  789. );
  790. }
  791. },
  792. immediate: true,
  793. deep: true
  794. },
  795. disableAction: {
  796. handler: function (isDisabled) {
  797. this.outterDisabled = isDisabled;
  798. this.isDisabled = isDisabled;
  799. },
  800. immediate: true
  801. },
  802. clearActionBox: {
  803. handler: function (needClear) {
  804. if (needClear && !this.needClear) {
  805. if (this.frontContext) {
  806. setTimeout(() => {
  807. this.frontContext.draw();
  808. }, 100);
  809. this.touchedView = {};
  810. this.prevFindedIndex = this.findedIndex;
  811. this.findedIndex = -1;
  812. }
  813. }
  814. this.needClear = needClear;
  815. },
  816. immediate: true
  817. }
  818. }
  819. };
  820. function setStringPrototype(screenK, scale) {
  821. /* eslint-disable no-extend-native */
  822. /**
  823. * 是否支持负数
  824. * @param {Boolean} minus 是否支持负数
  825. * @param {Number} baseSize 当设置了 % 号时,设置的基准值
  826. */
  827. String.prototype.toPx = function toPx(minus, baseSize) {
  828. if (this === '0') {
  829. return 0;
  830. }
  831. let reg;
  832. if (minus) {
  833. reg = /^-?[0-9]+([.]{1}[0-9]+){0,1}(rpx|px|%)$/g;
  834. } else {
  835. reg = /^[0-9]+([.]{1}[0-9]+){0,1}(rpx|px|%)$/g;
  836. }
  837. const results = reg.exec(this);
  838. if (!this || !results) {
  839. console.error(`The size: ${this} is illegal`);
  840. return 0;
  841. }
  842. const unit = results[2];
  843. const value = parseFloat(this);
  844. let res = 0;
  845. if (unit === 'rpx') {
  846. res = Math.round(value * (screenK || 0.5) * (scale || 1));
  847. } else if (unit === 'px') {
  848. res = Math.round(value * (scale || 1));
  849. } else if (unit === '%') {
  850. res = Math.round((value * baseSize) / 100);
  851. }
  852. return res;
  853. };
  854. }
  855. </script>
  856. <style>
  857. @import './painter.css';
  858. </style>