refactor(Signature): avoid setting the canvas width and height in the next tick (#12347)

* refactor(Signature): avoid setting the canvas width and height in the next tick

* test: update snapshots
This commit is contained in:
inottn 2023-10-13 20:16:20 +08:00 committed by GitHub
parent 25bc5921e2
commit 1219e4386c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 36 additions and 57 deletions

View File

@ -1,8 +1,7 @@
import { import {
computed,
ref, ref,
reactive,
onMounted, onMounted,
nextTick,
defineComponent, defineComponent,
type ExtractPropTypes, type ExtractPropTypes,
} from 'vue'; } from 'vue';
@ -45,45 +44,45 @@ export default defineComponent({
setup(props, { emit }) { setup(props, { emit }) {
const canvasRef = ref<HTMLCanvasElement>(); const canvasRef = ref<HTMLCanvasElement>();
const wrapRef = ref<HTMLElement>(); const wrapRef = ref<HTMLElement>();
const ctx = computed(() => {
const state = reactive({ if (!canvasRef.value) return null;
width: 0, return canvasRef.value.getContext('2d');
height: 0,
ctx: null as CanvasRenderingContext2D | null | undefined,
ratio: inBrowser ? window.devicePixelRatio : 1,
}); });
let canvasRect: DOMRect;
const isRenderCanvas = inBrowser ? hasCanvasSupport() : true; const isRenderCanvas = inBrowser ? hasCanvasSupport() : true;
const ratio = inBrowser ? window.devicePixelRatio : 1;
let canvasWidth = 0;
let canvasHeight = 0;
let canvasRect: DOMRect;
const touchStart = () => { const touchStart = () => {
if (!state.ctx) { if (!ctx.value) {
return false; return false;
} }
state.ctx.beginPath(); ctx.value.beginPath();
state.ctx.lineWidth = props.lineWidth * state.ratio; ctx.value.lineWidth = props.lineWidth * ratio;
state.ctx.strokeStyle = props.penColor; ctx.value.strokeStyle = props.penColor;
canvasRect = useRect(canvasRef); canvasRect = useRect(canvasRef);
emit('start'); emit('start');
}; };
const touchMove = (event: TouchEvent) => { const touchMove = (event: TouchEvent) => {
if (!state.ctx) { if (!ctx.value) {
return false; return false;
} }
preventDefault(event); preventDefault(event);
const touch = event.touches[0]; const touch = event.touches[0];
const mouseX = (touch.clientX - (canvasRect?.left || 0)) * state.ratio; const mouseX = (touch.clientX - (canvasRect?.left || 0)) * ratio;
const mouseY = (touch.clientY - (canvasRect?.top || 0)) * state.ratio; const mouseY = (touch.clientY - (canvasRect?.top || 0)) * ratio;
state.ctx.lineCap = 'round'; ctx.value.lineCap = 'round';
state.ctx.lineJoin = 'round'; ctx.value.lineJoin = 'round';
state.ctx?.lineTo(mouseX, mouseY); ctx.value.lineTo(mouseX, mouseY);
state.ctx?.stroke(); ctx.value.stroke();
emit('signing', event); emit('signing', event);
}; };
@ -109,7 +108,7 @@ export default defineComponent({
) => { ) => {
if (ctx && props.backgroundColor) { if (ctx && props.backgroundColor) {
ctx.fillStyle = props.backgroundColor; ctx.fillStyle = props.backgroundColor;
ctx.fillRect(0, 0, state.width, state.height); ctx.fillRect(0, 0, canvasWidth, canvasHeight);
} }
}; };
@ -137,24 +136,21 @@ export default defineComponent({
}; };
const clear = () => { const clear = () => {
if (state.ctx) { if (ctx.value) {
state.ctx.clearRect(0, 0, state.width, state.height); ctx.value.clearRect(0, 0, canvasWidth, canvasHeight);
state.ctx.closePath(); ctx.value.closePath();
setCanvasBgColor(state.ctx); setCanvasBgColor(ctx.value);
} }
emit('clear'); emit('clear');
}; };
onMounted(() => { onMounted(() => {
if (isRenderCanvas) { if (isRenderCanvas && canvasRef.value) {
state.ctx = canvasRef.value?.getContext('2d'); const canvas = canvasRef.value;
state.width = (wrapRef.value?.offsetWidth || 0) * state.ratio; canvasWidth = canvas.width = (wrapRef.value?.offsetWidth || 0) * ratio;
state.height = (wrapRef.value?.offsetHeight || 0) * state.ratio; canvasHeight = canvas.height =
(wrapRef.value?.offsetHeight || 0) * ratio;
// ensure canvas is rendered setCanvasBgColor(ctx.value);
nextTick(() => {
setCanvasBgColor(state.ctx);
});
} }
}); });
@ -164,8 +160,6 @@ export default defineComponent({
{isRenderCanvas ? ( {isRenderCanvas ? (
<canvas <canvas
ref={canvasRef} ref={canvasRef}
width={state.width}
height={state.height}
onTouchstartPassive={touchStart} onTouchstartPassive={touchStart}
onTouchmove={touchMove} onTouchmove={touchMove}
onTouchend={touchEnd} onTouchend={touchEnd}

View File

@ -6,10 +6,7 @@ exports[`should render demo and match snapshot 1`] = `
<!--[--> <!--[-->
<div class="van-signature"> <div class="van-signature">
<div class="van-signature__content"> <div class="van-signature__content">
<canvas <canvas>
width="0"
height="0"
>
</canvas> </canvas>
</div> </div>
<div class="van-signature__footer"> <div class="van-signature__footer">
@ -44,10 +41,7 @@ exports[`should render demo and match snapshot 1`] = `
<!--[--> <!--[-->
<div class="van-signature"> <div class="van-signature">
<div class="van-signature__content"> <div class="van-signature__content">
<canvas <canvas>
width="0"
height="0"
>
</canvas> </canvas>
</div> </div>
<div class="van-signature__footer"> <div class="van-signature__footer">
@ -82,10 +76,7 @@ exports[`should render demo and match snapshot 1`] = `
<!--[--> <!--[-->
<div class="van-signature"> <div class="van-signature">
<div class="van-signature__content"> <div class="van-signature__content">
<canvas <canvas>
width="0"
height="0"
>
</canvas> </canvas>
</div> </div>
<div class="van-signature__footer"> <div class="van-signature__footer">
@ -120,10 +111,7 @@ exports[`should render demo and match snapshot 1`] = `
<!--[--> <!--[-->
<div class="van-signature"> <div class="van-signature">
<div class="van-signature__content"> <div class="van-signature__content">
<canvas <canvas>
width="0"
height="0"
>
</canvas> </canvas>
</div> </div>
<div class="van-signature__footer"> <div class="van-signature__footer">

View File

@ -3,10 +3,7 @@
exports[`should render correctly when SSR 1`] = ` exports[`should render correctly when SSR 1`] = `
<div class="van-signature"> <div class="van-signature">
<div class="van-signature__content"> <div class="van-signature__content">
<canvas <canvas>
width="0"
height="0"
>
</canvas> </canvas>
</div> </div>
<div class="van-signature__footer"> <div class="van-signature__footer">