123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983 |
- <template>
- <view :style="'position: relative;' + customStyle + ';' + painterStyle">
- <canvas canvas-id="photo" :style="photoStyle + ';position: absolute; left: -9999px; top: -9999rpx;'" />
- <canvas canvas-id="bottom" :style="painterStyle + ';position: absolute;'" />
- <canvas canvas-id="k-canvas" :style="painterStyle + ';position: absolute;'" />
- <canvas canvas-id="top" :style="painterStyle + ';position: absolute;'" />
- <canvas
- canvas-id="front"
- :style="painterStyle + ';position: absolute;'"
- @touchstart="onTouchStart"
- @touchmove="onTouchMove"
- @touchend="onTouchEnd"
- @touchcancel="onTouchCancel"
- :disable-scroll="true"
- />
- </view>
- </template>
- <script>
- 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';
- export default {
- data() {
- return {
- picURL: '',
- showCanvas: true,
- painterStyle: '',
- photoStyle: ''
- };
- },
- canvasWidthInPx: 0,
- canvasHeightInPx: 0,
- paintCount: 0,
- currentPalette: {},
- movingCache: {},
- outterDisabled: false,
- isDisabled: false,
- needClear: false,
- /**
- * 组件的属性列表
- */
- props: {
- customStyle: {
- type: String
- },
- // 运行自定义选择框和删除缩放按钮
- customActionStyle: {
- type: Object
- },
- palette: {
- type: Object
- },
- dancePalette: {
- type: Object
- },
- // 缩放比,会在传入的 palette 中统一乘以该缩放比
- scaleRatio: {
- type: Number,
- default: 1
- },
- widthPixels: {
- type: Number,
- default: 1200
- },
- // 启用脏检查,默认 false
- dirty: {
- type: Boolean,
- default: false
- },
- LRU: {
- type: Boolean,
- default: true
- },
- action: {
- type: Object
- },
- disableAction: {
- type: Boolean
- },
- clearActionBox: {
- type: Boolean
- }
- },
- 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.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.customActionStyle && this.customActionStyle.border) {
- boxArea.css = Object.assign({}, boxArea.css, this.customActionStyle.border);
- }
- Object.assign(boxArea, {
- id: 'box'
- });
- return boxArea;
- },
- getScaleIcon(rect, type) {
- let scaleArea = {};
- const { customActionStyle } = this;
- 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;
- 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.LRU)
- .then((path) => {
- if (newVal.url.startsWith('https')) {
- doView.originUrl = newVal.url;
- }
- doView.url = path;
- uni.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 {
- if (newVal && newVal.text && doView.text && newVal.text !== doView.text) {
- doView.text = newVal.text;
- }
- if (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) => {
- if (callback) {
- callback(callbackInfo);
- }
- this.$emit('viewUpdate', {
- detail: {
- view: this.touchedView
- }
- });
- },
- true,
- this.movingCache
- );
- } else {
- // 某些机型(华为 P20)非移动和缩放场景下,只绘制一遍会偶然性图片绘制失败
- if (!isMoving && !this.isScale) {
- pen.paint();
- }
- pen.paint((callbackInfo) => {
- if (callback) {
- callback(callbackInfo);
- }
- this.$emit('viewUpdate', {
- detail: {
- 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.$emit('viewClicked', {
- detail: {
- view: this.touchedView
- }
- });
- }
- }
- if (this.findedIndex < 0 || (this.touchedView && !this.touchedView.id)) {
- // 证明点击了背景 或无法移动的view
- this.frontContext.draw();
- if (isDelete) {
- this.$emit('touchEnd', {
- detail: {
- view: this.currentPalette.views[deleteIndex],
- index: deleteIndex,
- type: 'delete'
- }
- });
- this.doAction();
- } else if (this.findedIndex < 0) {
- this.$emit('viewClicked', {
- detail: {}
- });
- }
- 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) {
- if (!this.isScale) {
- this.onClick(e);
- }
- } else if (this.touchedView && !this.isEmpty(this.touchedView)) {
- this.$emit('touchEnd', {
- detail: {
- 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().globalData.systemInfo && getApp().globalData.systemInfo.screenWidth)) {
- try {
- getApp().globalData.systemInfo = uni.getSystemInfoSync();
- } catch (e) {
- console.error(`Painter get system info failed, ${JSON.stringify(e)}`);
- return;
- }
- }
- this.screenK = 0.5;
- if (getApp() && getApp().globalData.systemInfo && getApp().globalData.systemInfo.screenWidth) {
- this.screenK = getApp().globalData.systemInfo.screenWidth / 750;
- }
- setStringPrototype(this.screenK, this.scaleRatio);
- },
- initDancePalette() {
- this.isDisabled = true;
- this.initScreenK();
- this.downloadImages(this.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 = uni.createCanvasContext('front', this));
- this.bottomContext || (this.bottomContext = uni.createCanvasContext('bottom', this));
- this.topContext || (this.topContext = uni.createCanvasContext('top', this));
- this.globalContext || (this.globalContext = uni.createCanvasContext('k-canvas', this));
- new Pen(this.bottomContext, palette).paint(() => {
- this.isDisabled = false;
- this.isDisabled = this.outterDisabled;
- this.$emit('didShow');
- });
- this.globalContext.draw();
- this.frontContext.draw();
- this.topContext.draw();
- });
- this.touchedView = {};
- },
- startPaint() {
- this.initScreenK();
- this.downloadImages(this.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.widthPixels) {
- setStringPrototype(this.screenK, this.widthPixels / this.canvasWidthInPx);
- this.canvasWidthInPx = this.widthPixels;
- }
- this.canvasHeightInPx = height.toPx();
- this.setData({
- photoStyle: `width:${this.canvasWidthInPx}px;height:${this.canvasHeightInPx}px;`
- });
- this.photoContext || (this.photoContext = uni.createCanvasContext('photo', this));
- new Pen(this.photoContext, palette).paint(() => {
- this.saveImgToLocal();
- });
- setStringPrototype(this.screenK, this.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.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.LRU).then(
- (path) => {
- view.originUrl = view.url;
- view.url = path;
- uni.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(() => {
- uni.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.$emit('imgErr', {
- detail: {
- error: error
- }
- });
- }
- },
- this
- );
- }, 300);
- },
- getImageInfo(filePath) {
- const that = this;
- uni.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.$emit('imgErr', {
- detail: {
- error: error
- }
- });
- return;
- } // 比例相符时才证明绘制成功,否则进行强制重绘制
- if (Math.abs((infoRes.width * that.canvasHeightInPx - that.canvasWidthInPx * infoRes.height) / (infoRes.height * that.canvasHeightInPx)) < 0.01) {
- that.$emit('imgOK', {
- detail: {
- path: filePath
- }
- });
- } else {
- that.startPaint();
- }
- that.paintCount++;
- },
- fail: (error) => {
- console.error(`getImageInfo failed, ${JSON.stringify(error)}`);
- that.$emit('imgErr', {
- detail: {
- error: error
- }
- });
- }
- });
- }
- },
- watch: {
- palette: {
- handler: function (newVal, oldVal) {
- if (this.isNeedRefresh(newVal, oldVal)) {
- this.paintCount = 0;
- this.startPaint();
- }
- },
- immediate: true,
- deep: true
- },
- dancePalette: {
- handler: function (newVal, oldVal) {
- if (!this.isEmpty(newVal)) {
- this.initDancePalette(newVal);
- }
- },
- immediate: true,
- deep: true
- },
- action: {
- handler: function (newVal, oldVal) {
- if (newVal && !this.isEmpty(newVal)) {
- this.doAction(
- newVal,
- (callbackInfo) => {
- this.movingCache = callbackInfo;
- },
- false,
- true
- );
- }
- },
- immediate: true,
- deep: true
- },
- disableAction: {
- handler: function (isDisabled) {
- this.outterDisabled = isDisabled;
- this.isDisabled = isDisabled;
- },
- immediate: true
- },
- clearActionBox: {
- handler: 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;
- },
- immediate: true
- }
- }
- };
- 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;
- };
- }
- </script>
- <style>
- @import './painter.css';
- </style>
|