123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338 |
- 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
- };
|