| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838 | import Pen from './lib/pen';import Downloader from './lib/downloader';const util = require('./lib/util');const downloader = new Downloader();// 最大尝试的绘制次数const MAX_PAINT_COUNT = 5;const ACTION_DEFAULT_SIZE = 24;const ACTION_OFFSET = '2rpx';Component({  canvasWidthInPx: 0,  canvasHeightInPx: 0,  paintCount: 0,  currentPalette: {},  movingCache: {},  outterDisabled: false,  isDisabled: false,  needClear: false,  /**   * 组件的属性列表   */  properties: {    customStyle: {      type: String,    },    // 运行自定义选择框和删除缩放按钮    customActionStyle: {      type: Object,    },    palette: {      type: Object,      observer: function (newVal, oldVal) {        if (this.isNeedRefresh(newVal, oldVal)) {          this.paintCount = 0;          this.startPaint();        }      },    },    dancePalette: {      type: Object,      observer: function (newVal, oldVal) {        if (!this.isEmpty(newVal)) {          this.initDancePalette(newVal);        }      },    },    // 缩放比,会在传入的 palette 中统一乘以该缩放比    scaleRatio: {      type: Number,      value: 1    },    widthPixels: {      type: Number,      value: 0    },    // 启用脏检查,默认 false    dirty: {      type: Boolean,      value: false,    },    LRU: {      type: Boolean,      value: true,    },    action: {      type: Object,      observer: function (newVal, oldVal) {        if (newVal && !this.isEmpty(newVal)) {          this.doAction(newVal, (callbackInfo) => {            this.movingCache = callbackInfo          }, false, true)        }      },    },    disableAction: {      type: Boolean,      observer: function (isDisabled) {        this.outterDisabled = isDisabled        this.isDisabled = isDisabled      }    },    clearActionBox: {      type: Boolean,      observer: function (needClear) {        if (needClear && !this.needClear) {          if (this.frontContext) {            setTimeout(() => {              this.frontContext.draw();            }, 100);            this.touchedView = {};            this.prevFindedIndex = this.findedIndex            this.findedIndex = -1;          }        }        this.needClear = needClear      }    },  },  data: {    picURL: '',    showCanvas: true,    painterStyle: '',  },  methods: {    /**     * 判断一个 object 是否为 空     * @param {object} object     */    isEmpty(object) {      for (const i in object) {        return false;      }      return true;    },    isNeedRefresh(newVal, oldVal) {      if (!newVal || this.isEmpty(newVal) || (this.data.dirty && util.equal(newVal, oldVal))) {        return false;      }      return true;    },    getBox(rect, type) {      const boxArea = {        type: 'rect',        css: {          height: `${rect.bottom - rect.top}px`,          width: `${rect.right - rect.left}px`,          left: `${rect.left}px`,          top: `${rect.top}px`,          borderWidth: '4rpx',          borderColor: '#1A7AF8',          color: 'transparent'        }      }      if (type === 'text') {        boxArea.css = Object.assign({}, boxArea.css, {          borderStyle: 'dashed'        })      }      if (this.properties.customActionStyle && this.properties.customActionStyle.border) {        boxArea.css = Object.assign({}, boxArea.css, this.properties.customActionStyle.border)      }      Object.assign(boxArea, {        id: 'box'      })      return boxArea    },    getScaleIcon(rect, type) {      let scaleArea = {}      const {        customActionStyle      } = this.properties      if (customActionStyle && customActionStyle.scale) {        scaleArea = {          type: 'image',          url: type === 'text' ? customActionStyle.scale.textIcon : customActionStyle.scale.imageIcon,          css: {            height: `${2 * ACTION_DEFAULT_SIZE}rpx`,            width: `${2 * ACTION_DEFAULT_SIZE}rpx`,            borderRadius: `${ACTION_DEFAULT_SIZE}rpx`,          }        }      } else {        scaleArea = {          type: 'rect',          css: {            height: `${2 * ACTION_DEFAULT_SIZE}rpx`,            width: `${2 * ACTION_DEFAULT_SIZE}rpx`,            borderRadius: `${ACTION_DEFAULT_SIZE}rpx`,            color: '#0000ff',          }        }      }      scaleArea.css = Object.assign({}, scaleArea.css, {        align: 'center',        left: `${rect.right + ACTION_OFFSET.toPx()}px`,        top: type === 'text' ? `${rect.top - ACTION_OFFSET.toPx() - scaleArea.css.height.toPx() / 2}px` : `${rect.bottom - ACTION_OFFSET.toPx() - scaleArea.css.height.toPx() / 2}px`      })      Object.assign(scaleArea, {        id: 'scale'      })      return scaleArea    },    getDeleteIcon(rect) {      let deleteArea = {}      const {        customActionStyle      } = this.properties      if (customActionStyle && customActionStyle.scale) {        deleteArea = {          type: 'image',          url: customActionStyle.delete.icon,          css: {            height: `${2 * ACTION_DEFAULT_SIZE}rpx`,            width: `${2 * ACTION_DEFAULT_SIZE}rpx`,            borderRadius: `${ACTION_DEFAULT_SIZE}rpx`,          }        }      } else {        deleteArea = {          type: 'rect',          css: {            height: `${2 * ACTION_DEFAULT_SIZE}rpx`,            width: `${2 * ACTION_DEFAULT_SIZE}rpx`,            borderRadius: `${ACTION_DEFAULT_SIZE}rpx`,            color: '#0000ff',          }        }      }      deleteArea.css = Object.assign({}, deleteArea.css, {        align: 'center',        left: `${rect.left - ACTION_OFFSET.toPx()}px`,        top: `${rect.top - ACTION_OFFSET.toPx() - deleteArea.css.height.toPx() / 2}px`      })      Object.assign(deleteArea, {        id: 'delete'      })      return deleteArea    },    doAction(action, callback, isMoving, overwrite) {      let newVal = null      if (action) {        newVal = action.view      }      if (newVal && newVal.id && this.touchedView.id !== newVal.id) {        // 带 id 的动作给撤回时使用,不带 id,表示对当前选中对象进行操作        const {          views        } = this.currentPalette;        for (let i = 0; i < views.length; i++) {          if (views[i].id === newVal.id) {            // 跨层回撤,需要重新构建三层关系            this.touchedView = views[i];            this.findedIndex = i;            this.sliceLayers();            break          }        }      }      const doView = this.touchedView      if (!doView || this.isEmpty(doView)) {        return      }      if (newVal && newVal.css) {        if (overwrite) {          doView.css = newVal.css        } else if (Array.isArray(doView.css) && Array.isArray(newVal.css)) {          doView.css = Object.assign({}, ...doView.css, ...newVal.css)        } else if (Array.isArray(doView.css)) {          doView.css = Object.assign({}, ...doView.css, newVal.css)        } else if (Array.isArray(newVal.css)) {          doView.css = Object.assign({}, doView.css, ...newVal.css)        } else {          doView.css = Object.assign({}, doView.css, newVal.css)        }      }      if (newVal && newVal.rect) {        doView.rect = newVal.rect;      }      if (newVal && newVal.url && doView.url && newVal.url !== doView.url) {        downloader.download(newVal.url, this.properties.LRU).then((path) => {          if (newVal.url.startsWith('https')) {            doView.originUrl = newVal.url          }          doView.url = path;          wx.getImageInfo({            src: path,            success: (res) => {              doView.sHeight = res.height              doView.sWidth = res.width              this.reDraw(doView, callback, isMoving)            },            fail: () => {              this.reDraw(doView, callback, isMoving)            }          })        }).catch((error) => {          // 未下载成功,直接绘制          console.error(error)          this.reDraw(doView, callback, isMoving)        })      } else {        (newVal && newVal.text && doView.text && newVal.text !== doView.text) && (doView.text = newVal.text);        (newVal && newVal.content && doView.content && newVal.content !== doView.content) && (doView.content = newVal.content);        this.reDraw(doView, callback, isMoving)      }    },    reDraw(doView, callback, isMoving) {      const draw = {        width: this.currentPalette.width,        height: this.currentPalette.height,        views: this.isEmpty(doView) ? [] : [doView]      }      const pen = new Pen(this.globalContext, draw);      if (isMoving && doView.type === 'text') {        pen.paint((callbackInfo) => {          callback && callback(callbackInfo);          this.triggerEvent('viewUpdate', {            view: this.touchedView          });        }, true, this.movingCache);      } else {        // 某些机型(华为 P20)非移动和缩放场景下,只绘制一遍会偶然性图片绘制失败        if (!isMoving && !this.isScale) {          pen.paint()        }        pen.paint((callbackInfo) => {          callback && callback(callbackInfo);          this.triggerEvent('viewUpdate', {            view: this.touchedView          });        })      }      const {        rect,        css,        type      } = doView      this.block = {        width: this.currentPalette.width,        height: this.currentPalette.height,        views: this.isEmpty(doView) ? [] : [this.getBox(rect, doView.type)]      }      if (css && css.scalable) {        this.block.views.push(this.getScaleIcon(rect, type))      }      if (css && css.deletable) {        this.block.views.push(this.getDeleteIcon(rect))      }      const topBlock = new Pen(this.frontContext, this.block)      topBlock.paint();    },    isInView(x, y, rect) {      return (x > rect.left &&        y > rect.top &&        x < rect.right &&        y < rect.bottom      )    },    isInDelete(x, y) {      for (const view of this.block.views) {        if (view.id === 'delete') {          return (x > view.rect.left &&            y > view.rect.top &&            x < view.rect.right &&            y < view.rect.bottom)        }      }      return false    },    isInScale(x, y) {      for (const view of this.block.views) {        if (view.id === 'scale') {          return (x > view.rect.left &&            y > view.rect.top &&            x < view.rect.right &&            y < view.rect.bottom)        }      }      return false    },    touchedView: {},    findedIndex: -1,    onClick() {      const x = this.startX      const y = this.startY      const totalLayerCount = this.currentPalette.views.length      let canBeTouched = []      let isDelete = false      let deleteIndex = -1      for (let i = totalLayerCount - 1; i >= 0; i--) {        const view = this.currentPalette.views[i]        const {          rect        } = view        if (this.touchedView && this.touchedView.id && this.touchedView.id === view.id && this.isInDelete(x, y, rect)) {          canBeTouched.length = 0          deleteIndex = i          isDelete = true          break        }        if (this.isInView(x, y, rect)) {          canBeTouched.push({            view,            index: i          })        }      }      this.touchedView = {}      if (canBeTouched.length === 0) {        this.findedIndex = -1      } else {        let i = 0        const touchAble = canBeTouched.filter(item => Boolean(item.view.id))        if (touchAble.length === 0) {          this.findedIndex = canBeTouched[0].index        } else {          for (i = 0; i < touchAble.length; i++) {            if (this.findedIndex === touchAble[i].index) {              i++              break            }          }          if (i === touchAble.length) {            i = 0          }          this.touchedView = touchAble[i].view          this.findedIndex = touchAble[i].index          this.triggerEvent('viewClicked', {            view: this.touchedView          })        }      }      if (this.findedIndex < 0 || (this.touchedView && !this.touchedView.id)) {        // 证明点击了背景 或无法移动的view        this.frontContext.draw();        if (isDelete) {          this.triggerEvent('touchEnd', {            view: this.currentPalette.views[deleteIndex],            index: deleteIndex,            type: 'delete'          })          this.doAction()        } else if (this.findedIndex < 0) {          this.triggerEvent('viewClicked', {})        }        this.findedIndex = -1        this.prevFindedIndex = -1      } else if (this.touchedView && this.touchedView.id) {        this.sliceLayers();      }    },    sliceLayers() {      const bottomLayers = this.currentPalette.views.slice(0, this.findedIndex)      const topLayers = this.currentPalette.views.slice(this.findedIndex + 1)      const bottomDraw = {        width: this.currentPalette.width,        height: this.currentPalette.height,        background: this.currentPalette.background,        views: bottomLayers      }      const topDraw = {        width: this.currentPalette.width,        height: this.currentPalette.height,        views: topLayers      }      if (this.prevFindedIndex < this.findedIndex) {        new Pen(this.bottomContext, bottomDraw).paint();        this.doAction(null, (callbackInfo) => {          this.movingCache = callbackInfo        })        new Pen(this.topContext, topDraw).paint();      } else {        new Pen(this.topContext, topDraw).paint();        this.doAction(null, (callbackInfo) => {          this.movingCache = callbackInfo        })        new Pen(this.bottomContext, bottomDraw).paint();      }      this.prevFindedIndex = this.findedIndex    },    startX: 0,    startY: 0,    startH: 0,    startW: 0,    isScale: false,    startTimeStamp: 0,    onTouchStart(event) {      if (this.isDisabled) {        return      }      const {        x,        y      } = event.touches[0]      this.startX = x      this.startY = y      this.startTimeStamp = new Date().getTime()      if (this.touchedView && !this.isEmpty(this.touchedView)) {        const {          rect        } = this.touchedView        if (this.isInScale(x, y, rect)) {          this.isScale = true          this.movingCache = {}          this.startH = rect.bottom - rect.top          this.startW = rect.right - rect.left        } else {          this.isScale = false        }      } else {        this.isScale = false      }    },    onTouchEnd(e) {      if (this.isDisabled) {        return      }      const current = new Date().getTime()      if ((current - this.startTimeStamp) <= 500 && !this.hasMove) {        !this.isScale && this.onClick(e)      } else if (this.touchedView && !this.isEmpty(this.touchedView)) {        this.triggerEvent('touchEnd', {          view: this.touchedView,        })      }      this.hasMove = false    },    onTouchCancel(e) {      if (this.isDisabled) {        return      }      this.onTouchEnd(e)    },    hasMove: false,    onTouchMove(event) {      if (this.isDisabled) {        return      }      this.hasMove = true      if (!this.touchedView || (this.touchedView && !this.touchedView.id)) {        return      }      const {        x,        y      } = event.touches[0]      const offsetX = x - this.startX      const offsetY = y - this.startY      const {        rect,        type      } = this.touchedView      let css = {}      if (this.isScale) {        const newW = this.startW + offsetX > 1 ? this.startW + offsetX : 1        if (this.touchedView.css && this.touchedView.css.minWidth) {          if (newW < this.touchedView.css.minWidth.toPx()) {            return          }        }        if (this.touchedView.rect && this.touchedView.rect.minWidth) {          if (newW < this.touchedView.rect.minWidth) {            return          }        }        const newH = this.startH + offsetY > 1 ? this.startH + offsetY : 1        css = {          width: `${newW}px`,        }        if (type !== 'text') {          if (type === 'image') {            css.height = `${(newW) * this.startH / this.startW }px`          } else {            css.height = `${newH}px`          }        }      } else {        this.startX = x        this.startY = y        css = {          left: `${rect.x + offsetX}px`,          top: `${rect.y + offsetY}px`,          right: undefined,          bottom: undefined        }      }      this.doAction({        view: {          css        }      }, (callbackInfo) => {        if (this.isScale) {          this.movingCache = callbackInfo        }      }, !this.isScale)    },    initScreenK() {      if (!(getApp() && getApp().systemInfo && getApp().systemInfo.screenWidth)) {        try {          getApp().systemInfo = wx.getSystemInfoSync();        } catch (e) {          console.error(`Painter get system info failed, ${JSON.stringify(e)}`);          return;        }      }      this.screenK = 0.5;      if (getApp() && getApp().systemInfo && getApp().systemInfo.screenWidth) {        this.screenK = getApp().systemInfo.screenWidth / 750;      }      setStringPrototype(this.screenK, this.properties.scaleRatio);    },    initDancePalette() {      this.isDisabled = true;      this.initScreenK();      this.downloadImages(this.properties.dancePalette).then((palette) => {        this.currentPalette = palette        const {          width,          height        } = palette;        if (!width || !height) {          console.error(`You should set width and height correctly for painter, width: ${width}, height: ${height}`);          return;        }        this.setData({          painterStyle: `width:${width.toPx()}px;height:${height.toPx()}px;`,        });        this.frontContext || (this.frontContext = wx.createCanvasContext('front', this));        this.bottomContext || (this.bottomContext = wx.createCanvasContext('bottom', this));        this.topContext || (this.topContext = wx.createCanvasContext('top', this));        this.globalContext || (this.globalContext = wx.createCanvasContext('k-canvas', this));        new Pen(this.bottomContext, palette).paint(() => {          this.isDisabled = false;          this.isDisabled = this.outterDisabled;          this.triggerEvent('didShow');        });        this.globalContext.draw();        this.frontContext.draw();        this.topContext.draw();      });      this.touchedView = {};    },    startPaint() {      this.initScreenK();      this.downloadImages(this.properties.palette).then((palette) => {        const {          width,          height        } = palette;        if (!width || !height) {          console.error(`You should set width and height correctly for painter, width: ${width}, height: ${height}`);          return;        }        // 生成图片时,根据设置的像素值重新绘制        this.canvasWidthInPx = width.toPx();        if (this.properties.widthPixels) {          setStringPrototype(this.screenK, this.properties.widthPixels / this.canvasWidthInPx)          this.canvasWidthInPx = this.properties.widthPixels        }        this.canvasHeightInPx = height.toPx();        this.setData({          photoStyle: `width:${this.canvasWidthInPx}px;height:${this.canvasHeightInPx}px;`,        });        this.photoContext || (this.photoContext = wx.createCanvasContext('photo', this));        new Pen(this.photoContext, palette).paint(() => {          this.saveImgToLocal();        });        setStringPrototype(this.screenK, this.properties.scaleRatio);      });    },    downloadImages(palette) {      return new Promise((resolve, reject) => {        let preCount = 0;        let completeCount = 0;        const paletteCopy = JSON.parse(JSON.stringify(palette));        if (paletteCopy.background) {          preCount++;          downloader.download(paletteCopy.background, this.properties.LRU).then((path) => {            paletteCopy.background = path;            completeCount++;            if (preCount === completeCount) {              resolve(paletteCopy);            }          }, () => {            completeCount++;            if (preCount === completeCount) {              resolve(paletteCopy);            }          });        }        if (paletteCopy.views) {          for (const view of paletteCopy.views) {            if (view && view.type === 'image' && view.url) {              preCount++;              /* eslint-disable no-loop-func */              downloader.download(view.url, this.properties.LRU).then((path) => {                view.originUrl = view.url;                view.url = path;                wx.getImageInfo({                  src: path,                  success: (res) => {                    // 获得一下图片信息,供后续裁减使用                    view.sWidth = res.width;                    view.sHeight = res.height;                  },                  fail: (error) => {                    // 如果图片坏了,则直接置空,防止坑爹的 canvas 画崩溃了                    view.url = "";                    console.error(`getImageInfo ${view.url} failed, ${JSON.stringify(error)}`);                  },                  complete: () => {                    completeCount++;                    if (preCount === completeCount) {                      resolve(paletteCopy);                    }                  },                });              }, () => {                completeCount++;                if (preCount === completeCount) {                  resolve(paletteCopy);                }              });            }          }        }        if (preCount === 0) {          resolve(paletteCopy);        }      });    },    saveImgToLocal() {      const that = this;      setTimeout(() => {        wx.canvasToTempFilePath({          canvasId: 'photo',          destWidth: that.canvasWidthInPx,          destHeight: that.canvasHeightInPx,          success: function (res) {            that.getImageInfo(res.tempFilePath);          },          fail: function (error) {            console.error(`canvasToTempFilePath failed, ${JSON.stringify(error)}`);            that.triggerEvent('imgErr', {              error: error            });          },        }, this);      }, 300);    },    getImageInfo(filePath) {      const that = this;      wx.getImageInfo({        src: filePath,        success: (infoRes) => {          if (that.paintCount > MAX_PAINT_COUNT) {            const error = `The result is always fault, even we tried ${MAX_PAINT_COUNT} times`;            console.error(error);            that.triggerEvent('imgErr', {              error: error            });            return;          }          // 比例相符时才证明绘制成功,否则进行强制重绘制          if (Math.abs((infoRes.width * that.canvasHeightInPx - that.canvasWidthInPx * infoRes.height) / (infoRes.height * that.canvasHeightInPx)) < 0.01) {            that.triggerEvent('imgOK', {              path: filePath            });          } else {            that.startPaint();          }          that.paintCount++;        },        fail: (error) => {          console.error(`getImageInfo failed, ${JSON.stringify(error)}`);          that.triggerEvent('imgErr', {            error: error          });        },      });    },  },});function setStringPrototype(screenK, scale) {  /* eslint-disable no-extend-native */  /**   * 是否支持负数   * @param {Boolean} minus 是否支持负数   * @param {Number} baseSize 当设置了 % 号时,设置的基准值   */  String.prototype.toPx = function toPx(minus, baseSize) {    if (this === '0') {      return 0    }    let reg;    if (minus) {      reg = /^-?[0-9]+([.]{1}[0-9]+){0,1}(rpx|px|%)$/g;    } else {      reg = /^[0-9]+([.]{1}[0-9]+){0,1}(rpx|px|%)$/g;    }    const results = reg.exec(this);    if (!this || !results) {      console.error(`The size: ${this} is illegal`);      return 0;    }    const unit = results[2];    const value = parseFloat(this);    let res = 0;    if (unit === 'rpx') {      res = Math.round(value * (screenK || 0.5) * (scale || 1));    } else if (unit === 'px') {      res = Math.round(value * (scale || 1));    } else if (unit === '%') {      res = Math.round(value * baseSize / 100);    }    return res;  };}
 |