index.vue 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. <template>
  2. <canvas v-if="use2dCanvas" :id="canvasId" type="2d" :style="style"></canvas>
  3. <canvas v-else :canvas-id="canvasId" :style="style" :id="canvasId" :width="boardWidth * dpr" :height="boardHeight * dpr"></canvas>
  4. </template>
  5. <script>
  6. import { toPx, base64ToPath, compareVersion} from './utils';
  7. import { Draw } from './draw';
  8. import { adaptor } from './canvas';
  9. export default {
  10. // version: '1.5.9.2',
  11. name: 'l-painter',
  12. props: {
  13. board: Object,
  14. fileType: {
  15. type: String,
  16. default: 'png'
  17. },
  18. width: [Number, String],
  19. height: [Number, String],
  20. pixelRatio: Number,
  21. customStyle: String,
  22. isRenderImage: Boolean,
  23. isBase64ToPath: Boolean,
  24. isH5PathToBase64: Boolean,
  25. sleep: {
  26. type: Number,
  27. default: 1000/30
  28. },
  29. type: {
  30. type: String,
  31. default: '2d',
  32. }
  33. },
  34. data() {
  35. return {
  36. // #ifndef MP-WEIXIN || MP-QQ
  37. canvasId: `l-painter_${this._uid}`,
  38. // #endif
  39. // #ifdef MP-WEIXIN || MP-QQ
  40. canvasId: `l-painter`,
  41. // #endif
  42. // #ifdef MP-WEIXIN
  43. use2dCanvas: true,
  44. // #endif
  45. // #ifndef MP-WEIXIN
  46. use2dCanvas: false,
  47. // #endif
  48. draw: null,
  49. ctx: null
  50. };
  51. },
  52. computed: {
  53. newboard() {
  54. return this.board && JSON.parse(JSON.stringify(this.board))
  55. },
  56. style() {
  57. return `width:${this.boardWidth}px; height: ${this.boardHeight}px; ${this.customStyle}`;
  58. },
  59. dpr() {
  60. return this.pixelRatio || uni.getSystemInfoSync().pixelRatio;
  61. },
  62. boardWidth() {
  63. const { width = 200 } = this.board || {};
  64. return toPx(this.width || width);
  65. },
  66. boardHeight() {
  67. const { height = 200 } = this.board || {};
  68. return toPx(this.height || height);
  69. }
  70. },
  71. mounted() {
  72. // #ifdef MP-WEIXIN
  73. const {SDKVersion} = wx.getSystemInfoSync()
  74. this.use2dCanvas = this.type === '2d' && compareVersion(SDKVersion, '2.9.2') >= 0;
  75. // #endif
  76. this.$watch('newboard', async (val, old) => {
  77. if (JSON.stringify(val) === '{}' || !val) return;
  78. const {width: w, height: h} = val || {};
  79. const {width: ow, height: oh} = old || {};
  80. if(w !== ow || h !== oh) {
  81. this.inited = false;
  82. }
  83. this.render();
  84. }, {
  85. deep: true,
  86. immediate: true,
  87. })
  88. },
  89. methods: {
  90. async render(args = {}, single = false) {
  91. const ctx = await this.getContext()
  92. const { use2dCanvas, boardWidth, boardHeight, board, canvas, isBase64ToPath, isH5PathToBase64, sleep } = this;
  93. if (use2dCanvas && !canvas) {
  94. return Promise.reject(new Error('render: fail canvas has not been created'));
  95. }
  96. if(!this.boundary) {
  97. this.boundary = {
  98. top: 0,
  99. left: 0,
  100. width: boardWidth,
  101. height: boardHeight,
  102. sleep
  103. }
  104. }
  105. if(!single) {
  106. ctx.clearRect(0, 0, boardWidth, boardHeight);
  107. }
  108. if(!this.draw) {
  109. this.draw = new Draw(ctx, canvas, use2dCanvas, isH5PathToBase64, this.boundary);
  110. }
  111. if(JSON.stringify(args) != '{}' || board && JSON.stringify(board) != '{}') {
  112. await this.draw.drawBoard(JSON.stringify(args) != '{}' ? args : board);
  113. }
  114. await new Promise(resolve => this.$nextTick(resolve))
  115. if (!use2dCanvas && !single) {
  116. await this.canvasDraw(ctx);
  117. }
  118. this.$emit('done')
  119. if(this.isRenderImage && !single) {
  120. this.canvasToTempFilePath()
  121. .then(async res => {
  122. if(/^data:image\/(\w+);base64/.test(res.tempFilePath) && isBase64ToPath) {
  123. const img = await base64ToPath(res.tempFilePath)
  124. this.$emit('success', img)
  125. } else {
  126. this.$emit('success', res.tempFilePath)
  127. }
  128. })
  129. .catch(err => {
  130. this.$emit('fail', err)
  131. new Error(JSON.stringify(err))
  132. console.error(JSON.stringify(err))
  133. })
  134. }
  135. return Promise.resolve({ctx, draw: this.draw});
  136. },
  137. async custom(cb) {
  138. const {ctx, draw} = await this.render({}, true)
  139. ctx.save()
  140. await cb(ctx, draw)
  141. ctx.restore()
  142. return Promise.resolve(true);
  143. },
  144. async single(args = {}) {
  145. await this.render(args, true)
  146. return Promise.resolve(true);
  147. },
  148. canvasDraw(flag = false) {
  149. const {ctx} = this
  150. return new Promise(resolve => {
  151. ctx.draw(flag, () => {
  152. resolve(true);
  153. });
  154. });
  155. },
  156. async getContext() {
  157. if(this.ctx && this.inited) {
  158. return Promise.resolve(this.ctx)
  159. };
  160. const { type, use2dCanvas, dpr, boardWidth, boardHeight } = this;
  161. const _getContext = () => {
  162. return new Promise(resolve => {
  163. uni.createSelectorQuery()
  164. .in(this)
  165. .select('#' + this.canvasId)
  166. .boundingClientRect()
  167. .exec(res => {
  168. if(res) {
  169. const ctx = uni.createCanvasContext(this.canvasId, this);
  170. if (!this.inited) {
  171. this.inited = true;
  172. this.use2dCanvas = false;
  173. this.canvas = res
  174. }
  175. // #ifdef MP-ALIPAY
  176. ctx.scale(dpr, dpr);
  177. // #endif
  178. this.ctx = ctx
  179. resolve(ctx);
  180. }
  181. })
  182. })
  183. }
  184. // #ifndef MP-WEIXIN
  185. return _getContext()
  186. // #endif
  187. if(!use2dCanvas) {
  188. return _getContext()
  189. }
  190. return new Promise(resolve => {
  191. uni.createSelectorQuery()
  192. .in(this)
  193. .select('#l-painter')
  194. .node()
  195. .exec(res => {
  196. const canvas = res[0].node;
  197. if(!canvas) {
  198. this.use2dCanvas = false;
  199. return this.getContext()
  200. }
  201. const ctx = canvas.getContext(type);
  202. if (!this.inited) {
  203. this.inited = true;
  204. canvas.width = boardWidth * dpr;
  205. canvas.height = boardHeight * dpr;
  206. this.use2dCanvas = true;
  207. this.canvas = canvas
  208. ctx.scale(dpr, dpr);
  209. }
  210. this.ctx = adaptor(ctx)
  211. resolve(adaptor(ctx));
  212. });
  213. });
  214. },
  215. canvasToTempFilePath(args = {}) {
  216. const {use2dCanvas, canvasId} = this
  217. return new Promise((resolve, reject) => {
  218. let { top = 0, left = 0, width, height } = this.boundary || this
  219. let destWidth = width * this.dpr
  220. let destHeight = height * this.dpr
  221. // #ifdef MP-ALIPAY
  222. width = width * this.dpr
  223. height = height * this.dpr
  224. // #endif
  225. const copyArgs = {
  226. x: left,
  227. y: top,
  228. width,
  229. height,
  230. destWidth,
  231. destHeight,
  232. canvasId,
  233. fileType: args.fileType || this.fileType,
  234. quality: args.quality || 1,
  235. success: resolve,
  236. fail: reject
  237. }
  238. if (use2dCanvas) {
  239. delete copyArgs.canvasId
  240. copyArgs.canvas = this.canvas
  241. }
  242. uni.canvasToTempFilePath(copyArgs, this)
  243. })
  244. }
  245. }
  246. };
  247. </script>
  248. <style></style>