mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
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:
parent
25bc5921e2
commit
1219e4386c
@ -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}
|
||||||
|
@ -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">
|
||||||
|
@ -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">
|
||||||
|
Loading…
x
Reference in New Issue
Block a user