ckgl/node_modules/vant/es/image-preview/ImagePreviewItem.mjs
2024-12-21 13:52:42 +08:00

339 lines
10 KiB
JavaScript

import { ref, watch, computed, reactive, defineComponent, createVNode as _createVNode } from "vue";
import { clamp, numericProp, preventDefault, createNamespace, makeRequiredProp, LONG_PRESS_START_TIME } from "../utils/index.mjs";
import { useExpose } from "../composables/use-expose.mjs";
import { useTouch } from "../composables/use-touch.mjs";
import { raf, useEventListener, useRect } from "@vant/use";
import { Image } from "../image/index.mjs";
import { Loading } from "../loading/index.mjs";
import { SwipeItem } from "../swipe-item/index.mjs";
const getDistance = (touches) => Math.sqrt((touches[0].clientX - touches[1].clientX) ** 2 + (touches[0].clientY - touches[1].clientY) ** 2);
const getCenter = (touches) => ({
x: (touches[0].clientX + touches[1].clientX) / 2,
y: (touches[0].clientY + touches[1].clientY) / 2
});
const bem = createNamespace("image-preview")[1];
const longImageRatio = 2.6;
const imagePreviewItemProps = {
src: String,
show: Boolean,
active: Number,
minZoom: makeRequiredProp(numericProp),
maxZoom: makeRequiredProp(numericProp),
rootWidth: makeRequiredProp(Number),
rootHeight: makeRequiredProp(Number),
disableZoom: Boolean,
doubleScale: Boolean,
closeOnClickImage: Boolean,
closeOnClickOverlay: Boolean,
vertical: Boolean
};
var stdin_default = defineComponent({
props: imagePreviewItemProps,
emits: ["scale", "close", "longPress"],
setup(props, {
emit,
slots
}) {
const state = reactive({
scale: 1,
moveX: 0,
moveY: 0,
moving: false,
zooming: false,
initializing: false,
imageRatio: 0
});
const touch = useTouch();
const imageRef = ref();
const swipeItem = ref();
const vertical = ref(false);
const isLongImage = ref(false);
let initialMoveY = 0;
const imageStyle = computed(() => {
const {
scale,
moveX,
moveY,
moving,
zooming,
initializing
} = state;
const style = {
transitionDuration: zooming || moving || initializing ? "0s" : ".3s"
};
if (scale !== 1 || isLongImage.value) {
style.transform = `matrix(${scale}, 0, 0, ${scale}, ${moveX}, ${moveY})`;
}
return style;
});
const maxMoveX = computed(() => {
if (state.imageRatio) {
const {
rootWidth,
rootHeight
} = props;
const displayWidth = vertical.value ? rootHeight / state.imageRatio : rootWidth;
return Math.max(0, (state.scale * displayWidth - rootWidth) / 2);
}
return 0;
});
const maxMoveY = computed(() => {
if (state.imageRatio) {
const {
rootWidth,
rootHeight
} = props;
const displayHeight = vertical.value ? rootHeight : rootWidth * state.imageRatio;
return Math.max(0, (state.scale * displayHeight - rootHeight) / 2);
}
return 0;
});
const setScale = (scale, center) => {
var _a;
scale = clamp(scale, +props.minZoom, +props.maxZoom + 1);
if (scale !== state.scale) {
const ratio = scale / state.scale;
state.scale = scale;
if (center) {
const imageRect = useRect((_a = imageRef.value) == null ? void 0 : _a.$el);
const origin = {
x: imageRect.width * 0.5,
y: imageRect.height * 0.5
};
const moveX = state.moveX - (center.x - imageRect.left - origin.x) * (ratio - 1);
const moveY = state.moveY - (center.y - imageRect.top - origin.y) * (ratio - 1);
state.moveX = clamp(moveX, -maxMoveX.value, maxMoveX.value);
state.moveY = clamp(moveY, -maxMoveY.value, maxMoveY.value);
} else {
state.moveX = 0;
state.moveY = isLongImage.value ? initialMoveY : 0;
}
emit("scale", {
scale,
index: props.active
});
}
};
const resetScale = () => {
setScale(1);
};
const toggleScale = () => {
const scale = state.scale > 1 ? 1 : 2;
setScale(scale, scale === 2 || isLongImage.value ? {
x: touch.startX.value,
y: touch.startY.value
} : void 0);
};
let fingerNum;
let startMoveX;
let startMoveY;
let startScale;
let startDistance;
let lastCenter;
let doubleTapTimer;
let touchStartTime;
let isImageMoved = false;
const onTouchStart = (event) => {
const {
touches
} = event;
fingerNum = touches.length;
if (fingerNum === 2 && props.disableZoom) {
return;
}
const {
offsetX
} = touch;
touch.start(event);
startMoveX = state.moveX;
startMoveY = state.moveY;
touchStartTime = Date.now();
isImageMoved = false;
state.moving = fingerNum === 1 && (state.scale !== 1 || isLongImage.value);
state.zooming = fingerNum === 2 && !offsetX.value;
if (state.zooming) {
startScale = state.scale;
startDistance = getDistance(touches);
}
};
const onTouchMove = (event) => {
const {
touches
} = event;
touch.move(event);
if (state.moving) {
const {
deltaX,
deltaY
} = touch;
const moveX = deltaX.value + startMoveX;
const moveY = deltaY.value + startMoveY;
if ((props.vertical ? touch.isVertical() && Math.abs(moveY) > maxMoveY.value : touch.isHorizontal() && Math.abs(moveX) > maxMoveX.value) && !isImageMoved) {
state.moving = false;
return;
}
isImageMoved = true;
preventDefault(event, true);
state.moveX = clamp(moveX, -maxMoveX.value, maxMoveX.value);
state.moveY = clamp(moveY, -maxMoveY.value, maxMoveY.value);
}
if (state.zooming) {
preventDefault(event, true);
if (touches.length === 2) {
const distance = getDistance(touches);
const scale = startScale * distance / startDistance;
lastCenter = getCenter(touches);
setScale(scale, lastCenter);
}
}
};
const checkClose = (event) => {
var _a;
const swipeItemEl = (_a = swipeItem.value) == null ? void 0 : _a.$el;
if (!swipeItemEl) return;
const imageEl = swipeItemEl.firstElementChild;
const isClickOverlay = event.target === swipeItemEl;
const isClickImage = imageEl == null ? void 0 : imageEl.contains(event.target);
if (!props.closeOnClickImage && isClickImage) return;
if (!props.closeOnClickOverlay && isClickOverlay) return;
emit("close");
};
const checkTap = (event) => {
if (fingerNum > 1) {
return;
}
const deltaTime = Date.now() - touchStartTime;
const TAP_TIME = 250;
if (touch.isTap.value) {
if (deltaTime < TAP_TIME) {
if (props.doubleScale) {
if (doubleTapTimer) {
clearTimeout(doubleTapTimer);
doubleTapTimer = null;
toggleScale();
} else {
doubleTapTimer = setTimeout(() => {
checkClose(event);
doubleTapTimer = null;
}, TAP_TIME);
}
} else {
checkClose(event);
}
} else if (deltaTime > LONG_PRESS_START_TIME) {
emit("longPress");
}
}
};
const onTouchEnd = (event) => {
let stopPropagation = false;
if (state.moving || state.zooming) {
stopPropagation = true;
if (state.moving && startMoveX === state.moveX && startMoveY === state.moveY) {
stopPropagation = false;
}
if (!event.touches.length) {
if (state.zooming) {
state.moveX = clamp(state.moveX, -maxMoveX.value, maxMoveX.value);
state.moveY = clamp(state.moveY, -maxMoveY.value, maxMoveY.value);
state.zooming = false;
}
state.moving = false;
startMoveX = 0;
startMoveY = 0;
startScale = 1;
if (state.scale < 1) {
resetScale();
}
const maxZoom = +props.maxZoom;
if (state.scale > maxZoom) {
setScale(maxZoom, lastCenter);
}
}
}
preventDefault(event, stopPropagation);
checkTap(event);
touch.reset();
};
const resize = () => {
const {
rootWidth,
rootHeight
} = props;
const rootRatio = rootHeight / rootWidth;
const {
imageRatio
} = state;
vertical.value = state.imageRatio > rootRatio && imageRatio < longImageRatio;
isLongImage.value = state.imageRatio > rootRatio && imageRatio >= longImageRatio;
if (isLongImage.value) {
initialMoveY = (imageRatio * rootWidth - rootHeight) / 2;
state.moveY = initialMoveY;
state.initializing = true;
raf(() => {
state.initializing = false;
});
}
resetScale();
};
const onLoad = (event) => {
const {
naturalWidth,
naturalHeight
} = event.target;
state.imageRatio = naturalHeight / naturalWidth;
resize();
};
watch(() => props.active, resetScale);
watch(() => props.show, (value) => {
if (!value) {
resetScale();
}
});
watch(() => [props.rootWidth, props.rootHeight], resize);
useEventListener("touchmove", onTouchMove, {
target: computed(() => {
var _a;
return (_a = swipeItem.value) == null ? void 0 : _a.$el;
})
});
useExpose({
resetScale
});
return () => {
const imageSlots = {
loading: () => _createVNode(Loading, {
"type": "spinner"
}, null)
};
return _createVNode(SwipeItem, {
"ref": swipeItem,
"class": bem("swipe-item"),
"onTouchstartPassive": onTouchStart,
"onTouchend": onTouchEnd,
"onTouchcancel": onTouchEnd
}, {
default: () => [slots.image ? _createVNode("div", {
"class": bem("image-wrap")
}, [slots.image({
src: props.src,
onLoad,
style: imageStyle.value
})]) : _createVNode(Image, {
"ref": imageRef,
"src": props.src,
"fit": "contain",
"class": bem("image", {
vertical: vertical.value
}),
"style": imageStyle.value,
"onLoad": onLoad
}, imageSlots)]
});
};
}
});
export {
stdin_default as default
};