refactor(ImagePreview): redesign function-call API (#10802)

This commit is contained in:
neverland 2022-07-09 15:32:02 +08:00 committed by GitHub
parent 5f526c9f0e
commit 1bc6cbdb69
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 99 additions and 119 deletions

View File

@ -172,6 +172,20 @@ declare module '@vue/runtime-core' {
} }
``` ```
### ImagePreview 调用方式调整
Vant 4 中,`ImagePreview` 组件的调用方式也进行了调整,与 `ImagePreview` 组件的改动一致:
```js
// Vant 3
ImagePreview(); // 函数调用
ImagePreview.Component; // 组件对象
// Vant 4
showImagePreview(); // 函数调用
ImagePreview; // 组件对象
```
### 事件命名调整 ### 事件命名调整
从 Vant 4 开始,所有的事件均采用 Vue 官方推荐的**驼峰格式**进行命名。 从 Vant 4 开始,所有的事件均采用 Vue 官方推荐的**驼峰格式**进行命名。

View File

@ -297,16 +297,3 @@ Dialog.alert({
// on close // on close
}); });
``` ```
### 在 JSX 中渲染 Dialog 组件无法展示?
请注意 `Dialog` 是一个函数,`Dialog.Component` 才是 Dialog 对应的组件。JSX 调用弹窗的正确姿势如下:
```jsx
export default {
setup() {
const show = ref(false);
return () => <Dialog.Component v-model={[show, 'show']} />;
},
};
```

View File

@ -16,12 +16,26 @@ const app = createApp();
app.use(ImagePreview); app.use(ImagePreview);
``` ```
### Function Call
Vant provides some utility functions that can quickly evoke global `ImagePreview` components.
For example, calling the `showImagePreview` function will render a Dialog directly in the page.
```js
import { showImagePreview } from 'vant';
showImagePreview(['https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg']);
```
## Usage ## Usage
### Basic Usage ### Basic Usage
```js ```js
ImagePreview([ import { showImagePreview } from 'vant';
showImagePreview([
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg',
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg',
]); ]);
@ -30,7 +44,9 @@ ImagePreview([
### Set Start Position ### Set Start Position
```js ```js
ImagePreview({ import { showImagePreview } from 'vant';
showImagePreview({
images: [ images: [
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg',
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg',
@ -44,7 +60,9 @@ ImagePreview({
After setting the `closeable` attribute, the close icon will be displayed in the upper right corner of the pop-up layer, and the icon can be customized through the `close-icon` attribute, and the icon location can be customized by using the `close-icon-position` attribute. After setting the `closeable` attribute, the close icon will be displayed in the upper right corner of the pop-up layer, and the icon can be customized through the `close-icon` attribute, and the icon location can be customized by using the `close-icon-position` attribute.
```js ```js
ImagePreview({ import { showImagePreview } from 'vant';
showImagePreview({
images: [ images: [
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg',
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg',
@ -56,9 +74,9 @@ ImagePreview({
### Close Event ### Close Event
```js ```js
import { Toast } from 'vant'; import { Toast, showImagePreview } from 'vant';
ImagePreview({ showImagePreview({
images: [ images: [
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg',
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg',
@ -72,7 +90,9 @@ ImagePreview({
### Before Close ### Before Close
```js ```js
const instance = ImagePreview({ import { showImagePreview } from 'vant';
const instance = showImagePreview({
images: [ images: [
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg',
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg',

View File

@ -2,61 +2,42 @@
### 介绍 ### 介绍
图片放大预览,支持函数调用和组件调用两种方式。 图片放大预览,支持组件调用和函数调用两种方式。
### 函数调用 ### 引入
`ImagePreview` 是一个函数,调用函数后会直接在页面中展示图片预览界面。 通过以下方式来全局注册组件,更多注册方式请参考[组件注册](#/zh-CN/advanced-usage#zu-jian-zhu-ce)。
```js
import { ImagePreview } from 'vant';
ImagePreview(['https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg']);
```
### 组件调用
通过组件调用 `ImagePreview` 时,可以通过下面的方式进行注册。
```js ```js
import { createApp } from 'vue'; import { createApp } from 'vue';
import { ImagePreview } from 'vant'; import { ImagePreview } from 'vant';
// 全局注册
const app = createApp(); const app = createApp();
app.use(ImagePreview); app.use(ImagePreview);
// 局部注册
export default {
components: {
[ImagePreview.Component.name]: ImagePreview.Component,
},
};
``` ```
`script setup` 中,可以通过以下方式使用: ### 函数调用
```html 为了便于使用 `ImagePreview`Vant 提供了一系列辅助函数,通过辅助函数可以快速唤起全局的弹窗组件。
<script setup>
const VanImagePreview = ImagePreview.Component;
</script>
<template> 比如使用 `showImagePreview` 函数,调用后会直接在页面中渲染对应的图片预览组件。
<!-- 中划线命名 -->
<van-image-preview /> ```js
<!-- 也支持大驼峰命名 --> import { showImagePreview } from 'vant';
<VanImagePreview>
</template> showImagePreview(['https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg']);
``` ```
## 代码演示 ## 代码演示
### 基础用法 ### 基础用法
直接传入图片数组,即可展示图片预览。 在调用 `showImagePreview` 时,直接传入图片数组,即可展示图片预览。
```js ```js
ImagePreview([ import { showImagePreview } from 'vant';
showImagePreview([
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg',
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg',
]); ]);
@ -64,10 +45,12 @@ ImagePreview([
### 指定初始位置 ### 指定初始位置
`ImagePreview` 支持传入配置对象,并通过 `startPosition` 选项指定图片的初始位置(索引值)。 `showImagePreview` 支持传入配置对象,并通过 `startPosition` 选项指定图片的初始位置(索引值)。
```js ```js
ImagePreview({ import { showImagePreview } from 'vant';
showImagePreview({
images: [ images: [
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg',
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg',
@ -81,7 +64,9 @@ ImagePreview({
设置 `closeable` 属性后,会在弹出层的右上角显示关闭图标,并且可以通过 `close-icon` 属性自定义图标,使用`close-icon-position` 属性可以自定义图标位置。 设置 `closeable` 属性后,会在弹出层的右上角显示关闭图标,并且可以通过 `close-icon` 属性自定义图标,使用`close-icon-position` 属性可以自定义图标位置。
```js ```js
ImagePreview({ import { showImagePreview } from 'vant';
showImagePreview({
images: [ images: [
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg',
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg',
@ -95,9 +80,9 @@ ImagePreview({
通过 `onClose` 选项监听图片预览的关闭事件。 通过 `onClose` 选项监听图片预览的关闭事件。
```js ```js
import { Toast } from 'vant'; import { Toast, showImagePreview } from 'vant';
ImagePreview({ showImagePreview({
images: [ images: [
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg',
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg',
@ -113,7 +98,9 @@ ImagePreview({
通过 `beforeClose` 属性可以拦截关闭行为。 通过 `beforeClose` 属性可以拦截关闭行为。
```js ```js
const instance = ImagePreview({ import { showImagePreview } from 'vant';
const instance = showImagePreview({
images: [ images: [
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-1.jpeg',
'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg', 'https://fastly.jsdelivr.net/npm/@vant/assets/apple-2.jpeg',
@ -166,7 +153,7 @@ export default {
### Options ### Options
通过函数调用 `ImagePreview` 时,支持传入以下选项: 通过函数调用 `showImagePreview` 时,支持传入以下选项:
| 参数名 | 说明 | 类型 | 默认值 | | 参数名 | 说明 | 类型 | 默认值 |
| --- | --- | --- | --- | | --- | --- | --- | --- |
@ -307,16 +294,3 @@ imagePreviewRef.value?.swipeTo(1);
### 在桌面端无法操作组件? ### 在桌面端无法操作组件?
参见[桌面端适配](#/zh-CN/advanced-usage#zhuo-mian-duan-gua-pei)。 参见[桌面端适配](#/zh-CN/advanced-usage#zhuo-mian-duan-gua-pei)。
### 在 JSX 中渲染 ImagePreview 组件无法展示?
请注意 `ImagePreview` 是一个函数,`ImagePreview.Component` 才是 ImagePreview 对应的组件。JSX 调用图片预览的正确姿势如下:
```jsx
export default {
setup() {
const show = ref(false);
return () => <ImagePreview.Component v-model={[show, 'show']} />;
},
};
```

View File

@ -1,12 +1,14 @@
<script setup lang="ts"> <script setup lang="ts">
import VanCell from '../../cell'; import VanCell from '../../cell';
import { ImagePreview, ImagePreviewOptions } from '..'; import {
showImagePreview,
ImagePreviewOptions,
ImagePreview as VanImagePreview,
} from '..';
import { ref } from 'vue'; import { ref } from 'vue';
import { cdnURL, useTranslate } from '../../../docs/site'; import { cdnURL, useTranslate } from '../../../docs/site';
import { Toast } from '../../toast'; import { Toast } from '../../toast';
const VanImagePreview = ImagePreview.Component;
const t = useTranslate({ const t = useTranslate({
'zh-CN': { 'zh-CN': {
closed: '关闭', closed: '关闭',
@ -51,16 +53,16 @@ const beforeClose = () =>
}, 1000); }, 1000);
}); });
const showComponentCall = () => {
show.value = true;
};
const onChange = (newIndex: number) => { const onChange = (newIndex: number) => {
index.value = newIndex; index.value = newIndex;
}; };
const showImagePreview = (options: Partial<ImagePreviewOptions> = {}) => { const showComponentCall = () => {
const instance = ImagePreview({ show.value = true;
};
const showFunctionCall = (options: Partial<ImagePreviewOptions> = {}) => {
const instance = showImagePreview({
images, images,
...options, ...options,
}); });
@ -75,24 +77,24 @@ const showImagePreview = (options: Partial<ImagePreviewOptions> = {}) => {
<template> <template>
<demo-block card :title="t('basicUsage')"> <demo-block card :title="t('basicUsage')">
<van-cell is-link :value="t('showImages')" @click="showImagePreview()" /> <van-cell is-link :value="t('showImages')" @click="showFunctionCall()" />
</demo-block> </demo-block>
<demo-block card :title="t('customConfig')"> <demo-block card :title="t('customConfig')">
<van-cell <van-cell
is-link is-link
:value="t('startPosition')" :value="t('startPosition')"
@click="showImagePreview({ startPosition: 1 })" @click="showFunctionCall({ startPosition: 1 })"
/> />
<van-cell <van-cell
is-link is-link
:value="t('showClose')" :value="t('showClose')"
@click="showImagePreview({ closeable: true })" @click="showFunctionCall({ closeable: true })"
/> />
<van-cell <van-cell
is-link is-link
:value="t('closeEvent')" :value="t('closeEvent')"
@click="showImagePreview({ onClose })" @click="showFunctionCall({ onClose })"
/> />
</demo-block> </demo-block>
@ -100,7 +102,7 @@ const showImagePreview = (options: Partial<ImagePreviewOptions> = {}) => {
<van-cell <van-cell
is-link is-link
:value="t('beforeClose')" :value="t('beforeClose')"
@click="showImagePreview({ beforeClose })" @click="showFunctionCall({ beforeClose })"
/> />
</demo-block> </demo-block>

View File

@ -1,7 +1,6 @@
import { extend, inBrowser, withInstall, ComponentInstance } from '../utils'; import { extend, inBrowser, ComponentInstance } from '../utils';
import { mountComponent, usePopupState } from '../utils/mount-component'; import { mountComponent, usePopupState } from '../utils/mount-component';
import VanImagePreview from './ImagePreview'; import VanImagePreview from './ImagePreview';
import type { App } from 'vue';
import type { ImagePreviewOptions } from './types'; import type { ImagePreviewOptions } from './types';
let instance: ComponentInstance; let instance: ComponentInstance;
@ -49,7 +48,7 @@ function initInstance() {
})); }));
} }
const ImagePreview = ( export const showImagePreview = (
options: string[] | ImagePreviewOptions, options: string[] | ImagePreviewOptions,
startPosition = 0 startPosition = 0
) => { ) => {
@ -70,11 +69,3 @@ const ImagePreview = (
return instance; return instance;
}; };
ImagePreview.Component = withInstall(VanImagePreview);
ImagePreview.install = (app: App) => {
app.use(ImagePreview.Component);
};
export { ImagePreview };

View File

@ -1,8 +1,11 @@
import { ImagePreview } from './function-call'; import { withInstall } from '../utils';
import _ImagePreview from './ImagePreview';
import type { ImagePreviewProps } from './ImagePreview'; import type { ImagePreviewProps } from './ImagePreview';
export const ImagePreview = withInstall(_ImagePreview);
export default ImagePreview; export default ImagePreview;
export { ImagePreview }; export { showImagePreview } from './function-call';
export type { ImagePreviewProps }; export type { ImagePreviewProps };
export type { export type {
ImagePreviewOptions, ImagePreviewOptions,

View File

@ -1,22 +1,11 @@
import { later, triggerDrag, mockGetBoundingClientRect } from '../../../test'; import { later, triggerDrag, mockGetBoundingClientRect } from '../../../test';
import { createApp, nextTick } from 'vue'; import { nextTick } from 'vue';
import { ImagePreview } from '../function-call'; import { showImagePreview } from '../function-call';
import ImagePreviewComponent from '../ImagePreview';
import { images, triggerZoom } from './shared'; import { images, triggerZoom } from './shared';
test('should expose ImagePreviewComponent in ImagePreview.Component', () => {
expect(ImagePreview.Component.name).toEqual('van-image-preview');
});
test('should register component to app', () => {
const app = createApp(document.body);
app.use(ImagePreview);
expect(app.component(ImagePreviewComponent.name)).toBeTruthy();
});
test('should allow to use the teleport option', async () => { test('should allow to use the teleport option', async () => {
const root = document.createElement('div'); const root = document.createElement('div');
ImagePreview({ images: [], teleport: root }); showImagePreview({ images: [], teleport: root });
await later(); await later();
expect(root.querySelector('.van-image-preview')).toBeTruthy(); expect(root.querySelector('.van-image-preview')).toBeTruthy();
@ -24,7 +13,7 @@ test('should allow to use the teleport option', async () => {
test('should trigger onClose option correctly', async () => { test('should trigger onClose option correctly', async () => {
const onClose = jest.fn(); const onClose = jest.fn();
const instance = ImagePreview({ const instance = showImagePreview({
images, images,
startPosition: 1, startPosition: 1,
onClose, onClose,
@ -41,7 +30,7 @@ test('should trigger onClose option correctly', async () => {
test('should trigger onChange option correctly', async () => { test('should trigger onChange option correctly', async () => {
const onChange = jest.fn(); const onChange = jest.fn();
ImagePreview({ showImagePreview({
images, images,
startPosition: 0, startPosition: 0,
onChange, onChange,
@ -55,7 +44,7 @@ test('should trigger onChange option correctly', async () => {
test('should trigger onScale option correctly', async () => { test('should trigger onScale option correctly', async () => {
const restore = mockGetBoundingClientRect({ width: 100 }); const restore = mockGetBoundingClientRect({ width: 100 });
ImagePreview({ showImagePreview({
images, images,
startPosition: 0, startPosition: 0,
onScale({ index, scale }) { onScale({ index, scale }) {

View File

@ -37,7 +37,7 @@ import { useExpose } from '../composables/use-expose';
// Components // Components
import { Icon } from '../icon'; import { Icon } from '../icon';
import { ImagePreview, type ImagePreviewOptions } from '../image-preview'; import { showImagePreview, type ImagePreviewOptions } from '../image-preview';
import UploaderPreviewItem from './UploaderPreviewItem'; import UploaderPreviewItem from './UploaderPreviewItem';
// Types // Types
@ -238,7 +238,7 @@ export default defineComponent({
}) })
.filter(Boolean) as string[]; .filter(Boolean) as string[];
imagePreview = ImagePreview( imagePreview = showImagePreview(
extend( extend(
{ {
images, images,