feat(ImagePreview): expose resetScale method (#12426)

This commit is contained in:
inottn 2023-11-05 20:31:34 +08:00 committed by GitHub
parent 8079f7c24d
commit c50be595d4
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 74 additions and 16 deletions

View File

@ -38,7 +38,10 @@ import { Popup, PopupCloseIconPosition } from '../popup';
import ImagePreviewItem from './ImagePreviewItem'; import ImagePreviewItem from './ImagePreviewItem';
// Types // Types
import { ImagePreviewScaleEventParams } from './types'; import {
ImagePreviewScaleEventParams,
ImagePreviewItemInstance,
} from './types';
const [name, bem] = createNamespace('image-preview'); const [name, bem] = createNamespace('image-preview');
@ -86,6 +89,7 @@ export default defineComponent({
setup(props, { emit, slots }) { setup(props, { emit, slots }) {
const swipeRef = ref<SwipeInstance>(); const swipeRef = ref<SwipeInstance>();
const activedPreviewItemRef = ref<ImagePreviewItemInstance>();
const state = reactive({ const state = reactive({
active: 0, active: 0,
@ -167,6 +171,11 @@ export default defineComponent({
v-slots={{ v-slots={{
image: slots.image, image: slots.image,
}} }}
ref={(item) => {
if (index === state.active) {
activedPreviewItemRef.value = item as ImagePreviewItemInstance;
}
}}
src={image} src={image}
show={props.show} show={props.show}
active={state.active} active={state.active}
@ -206,7 +215,12 @@ export default defineComponent({
const swipeTo = (index: number, options?: SwipeToOptions) => const swipeTo = (index: number, options?: SwipeToOptions) =>
swipeRef.value?.swipeTo(index, options); swipeRef.value?.swipeTo(index, options);
useExpose({ swipeTo }); useExpose({
resetScale: () => {
activedPreviewItemRef.value?.resetScale();
},
swipeTo,
});
onMounted(resize); onMounted(resize);

View File

@ -5,6 +5,7 @@ import {
reactive, reactive,
defineComponent, defineComponent,
type CSSProperties, type CSSProperties,
type ExtractPropTypes,
} from 'vue'; } from 'vue';
// Utils // Utils
@ -20,6 +21,7 @@ import {
} from '../utils'; } from '../utils';
// Composables // Composables
import { useExpose } from '../composables/use-expose';
import { useTouch } from '../composables/use-touch'; import { useTouch } from '../composables/use-touch';
import { raf, useEventListener, useRect } from '@vant/use'; import { raf, useEventListener, useRect } from '@vant/use';
@ -43,19 +45,25 @@ const bem = createNamespace('image-preview')[1];
const longImageRatio = 2.6; 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,
closeOnClickOverlay: Boolean,
};
export type ImagePreviewItemProps = ExtractPropTypes<
typeof imagePreviewItemProps
>;
export default defineComponent({ export default defineComponent({
props: { props: imagePreviewItemProps,
src: String,
show: Boolean,
active: Number,
minZoom: makeRequiredProp(numericProp),
maxZoom: makeRequiredProp(numericProp),
rootWidth: makeRequiredProp(Number),
rootHeight: makeRequiredProp(Number),
disableZoom: Boolean,
doubleScale: Boolean,
closeOnClickOverlay: Boolean,
},
emits: ['scale', 'close', 'longPress'], emits: ['scale', 'close', 'longPress'],
@ -375,6 +383,8 @@ export default defineComponent({
target: computed(() => swipeItem.value?.$el), target: computed(() => swipeItem.value?.$el),
}); });
useExpose({ resetScale });
return () => { return () => {
const imageSlots = { const imageSlots = {
loading: () => <Loading type="spinner" />, loading: () => <Loading type="spinner" />,

View File

@ -261,7 +261,8 @@ Use [ref](https://vuejs.org/v2/api/#ref) to get ImagePreview instance and call i
| Name | Description | Attribute | Return value | | Name | Description | Attribute | Return value |
| --- | --- | --- | --- | | --- | --- | --- | --- |
| swipeTo `2.9.0` | Swipe to target index | _index: number, options?: SwipeToOptions_ | - | | resetScale `4.7.4` | Reset the current image's zoom ratio | - | - |
| swipeTo | Swipe to target index | _index: number, options?: SwipeToOptions_ | - |
### Types ### Types

View File

@ -268,7 +268,8 @@ Vant 中导出了以下 ImagePreview 相关的辅助函数:
| 方法名 | 说明 | 参数 | 返回值 | | 方法名 | 说明 | 参数 | 返回值 |
| --- | --- | --- | --- | | --- | --- | --- | --- |
| swipeTo `2.9.0` | 切换到指定位置 | _index: number, options?: SwipeToOptions_ | - | | resetScale `4.7.4` | 重置当前图片的缩放比 | - | - |
| swipeTo | 切换到指定位置 | _index: number, options?: SwipeToOptions_ | - |
### 类型定义 ### 类型定义

View File

@ -8,6 +8,7 @@ import {
import { LONG_PRESS_START_TIME } from '../../utils'; import { LONG_PRESS_START_TIME } from '../../utils';
import ImagePreview from '../ImagePreview'; import ImagePreview from '../ImagePreview';
import { images, triggerZoom } from './shared'; import { images, triggerZoom } from './shared';
import type { ImagePreviewInstance } from '../types';
test('should swipe to current index after calling the swipeTo method', async () => { test('should swipe to current index after calling the swipeTo method', async () => {
const wrapper = mount(ImagePreview, { const wrapper = mount(ImagePreview, {
@ -365,3 +366,23 @@ test('should emit long-press event after long press', async () => {
index: 0, index: 0,
}); });
}); });
test('should reset scale after calling the resetScale method', async () => {
const wrapper = mount(ImagePreview, {
props: {
show: true,
images,
},
});
await later();
const image = wrapper.find('.van-image');
triggerZoom(image, 300, 300);
await later();
expect(image.style.transform).toBeTruthy();
(wrapper.vm as ImagePreviewInstance).resetScale();
await later();
expect(image.style.transform).toBeFalsy();
});

View File

@ -7,6 +7,7 @@ import type { Interceptor } from '../utils';
import type { SwipeToOptions } from '../swipe'; import type { SwipeToOptions } from '../swipe';
import type { PopupCloseIconPosition } from '../popup'; import type { PopupCloseIconPosition } from '../popup';
import type { ImagePreviewProps } from './ImagePreview'; import type { ImagePreviewProps } from './ImagePreview';
import type { ImagePreviewItemProps } from './ImagePreviewItem';
export type ImagePreviewOptions = { export type ImagePreviewOptions = {
loop?: boolean; loop?: boolean;
@ -39,7 +40,17 @@ export type ImagePreviewScaleEventParams = {
index: number; index: number;
}; };
type ImagePreviewItemExpose = {
resetScale: () => void;
};
export type ImagePreviewItemInstance = ComponentPublicInstance<
ImagePreviewItemProps,
ImagePreviewItemExpose
>;
export type ImagePreviewExpose = { export type ImagePreviewExpose = {
resetScale: () => void;
swipeTo: (index: number, options?: SwipeToOptions) => void; swipeTo: (index: number, options?: SwipeToOptions) => void;
}; };