types(ImagePreview): use tsx (#8176)

This commit is contained in:
neverland 2021-02-19 17:16:07 +08:00 committed by GitHub
parent d6bfb8fadc
commit d67031a98d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 102 additions and 57 deletions

View File

@ -1,9 +1,8 @@
import { nextTick, onMounted, reactive, ref, watch } from 'vue'; import { nextTick, onMounted, PropType, reactive, ref, watch } from 'vue';
// Utils // Utils
import { UnknownProp } from '../utils'; import { ComponentInstance, UnknownProp, createNamespace } from '../utils';
import { bem, createComponent } from './shared'; import { callInterceptor, Interceptor } from '../utils/interceptor';
import { callInterceptor } from '../utils/interceptor';
// Composition // Composition
import { useWindowSize } from '@vant/use'; import { useWindowSize } from '@vant/use';
@ -11,19 +10,21 @@ import { useExpose } from '../composables/use-expose';
// Components // Components
import Icon from '../icon'; import Icon from '../icon';
import Swipe from '../swipe'; import Swipe, { SwipeToOptions } from '../swipe';
import Popup from '../popup'; import Popup, { PopupCloseIconPosition } from '../popup';
import ImagePreviewItem from './ImagePreviewItem'; import ImagePreviewItem from './ImagePreviewItem';
const [createComponent, bem] = createNamespace('image-preview');
export default createComponent({ export default createComponent({
props: { props: {
show: Boolean, show: Boolean,
closeable: Boolean, closeable: Boolean,
className: UnknownProp, className: UnknownProp,
beforeClose: Function, beforeClose: Function as PropType<Interceptor>,
showIndicators: Boolean, showIndicators: Boolean,
images: { images: {
type: Array, type: Array as PropType<string[]>,
default: () => [], default: () => [],
}, },
loop: { loop: {
@ -63,7 +64,7 @@ export default createComponent({
default: true, default: true,
}, },
closeIconPosition: { closeIconPosition: {
type: String, type: String as PropType<PopupCloseIconPosition>,
default: 'top-right', default: 'top-right',
}, },
}, },
@ -71,7 +72,7 @@ export default createComponent({
emits: ['scale', 'close', 'closed', 'change', 'update:show'], emits: ['scale', 'close', 'closed', 'change', 'update:show'],
setup(props, { emit, slots }) { setup(props, { emit, slots }) {
const swipeRef = ref(); const swipeRef = ref<ComponentInstance>();
const windowSize = useWindowSize(); const windowSize = useWindowSize();
const state = reactive({ const state = reactive({
@ -89,11 +90,11 @@ export default createComponent({
} }
}; };
const emitScale = (args) => { const emitScale = (args: { scale: number; index: number }) => {
emit('scale', args); emit('scale', args);
}; };
const toggle = (show) => { const toggle = (show: boolean) => {
emit('update:show', show); emit('update:show', show);
}; };
@ -107,7 +108,7 @@ export default createComponent({
}); });
}; };
const setActive = (active) => { const setActive = (active: number) => {
if (active !== state.active) { if (active !== state.active) {
state.active = active; state.active = active;
emit('change', active); emit('change', active);
@ -177,7 +178,7 @@ export default createComponent({
emit('closed'); emit('closed');
}; };
const swipeTo = (index, options) => { const swipeTo = (index: number, options?: SwipeToOptions) => {
if (swipeRef.value) { if (swipeRef.value) {
swipeRef.value.swipeTo(index, options); swipeRef.value.swipeTo(index, options);
} }
@ -189,7 +190,12 @@ export default createComponent({
watch([windowSize.width, windowSize.height], resize); watch([windowSize.width, windowSize.height], resize);
watch(() => props.startPosition, setActive); watch(
() => props.startPosition,
(value) => {
setActive(+value);
}
);
watch( watch(
() => props.show, () => props.show,

View File

@ -1,8 +1,7 @@
import { watch, computed, reactive } from 'vue'; import { watch, computed, reactive, CSSProperties, defineComponent } from 'vue';
// Utils // Utils
import { bem } from './shared'; import { range, preventDefault, createNamespace } from '../utils';
import { range, preventDefault } from '../utils';
// Composition // Composition
import { useTouch } from '../composables/use-touch'; import { useTouch } from '../composables/use-touch';
@ -12,22 +11,36 @@ import Image from '../image';
import Loading from '../loading'; import Loading from '../loading';
import SwipeItem from '../swipe-item'; import SwipeItem from '../swipe-item';
function getDistance(touches) { function getDistance(touches: TouchList) {
return Math.sqrt( return Math.sqrt(
(touches[0].clientX - touches[1].clientX) ** 2 + (touches[0].clientX - touches[1].clientX) ** 2 +
(touches[0].clientY - touches[1].clientY) ** 2 (touches[0].clientY - touches[1].clientY) ** 2
); );
} }
export default { const bem = createNamespace('image-preview')[1];
export default defineComponent({
props: { props: {
src: String, src: String,
show: Boolean, show: Boolean,
active: Number, active: Number,
minZoom: [Number, String], minZoom: {
maxZoom: [Number, String], type: [Number, String],
rootWidth: Number, required: true,
rootHeight: Number, },
maxZoom: {
type: [Number, String],
required: true,
},
rootWidth: {
type: Number,
required: true,
},
rootHeight: {
type: Number,
required: true,
},
}, },
emits: ['scale', 'close'], emits: ['scale', 'close'],
@ -54,7 +67,7 @@ export default {
const imageStyle = computed(() => { const imageStyle = computed(() => {
const { scale, moveX, moveY, moving, zooming } = state; const { scale, moveX, moveY, moving, zooming } = state;
const style = { const style: CSSProperties = {
transitionDuration: zooming || moving ? '0s' : '.3s', transitionDuration: zooming || moving ? '0s' : '.3s',
}; };
@ -93,7 +106,7 @@ export default {
return 0; return 0;
}); });
const setScale = (scale) => { const setScale = (scale: number) => {
scale = range(scale, +props.minZoom, +props.maxZoom); scale = range(scale, +props.minZoom, +props.maxZoom);
if (scale !== state.scale) { if (scale !== state.scale) {
@ -119,14 +132,14 @@ export default {
state.moveY = 0; state.moveY = 0;
}; };
let startMoveX; let startMoveX: number;
let startMoveY; let startMoveY: number;
let startScale; let startScale: number;
let startDistance; let startDistance: number;
let doubleTapTimer; let doubleTapTimer: NodeJS.Timeout | null;
let touchStartTime; let touchStartTime: number;
const onTouchStart = (event) => { const onTouchStart = (event: TouchEvent) => {
const { touches } = event; const { touches } = event;
const { offsetX } = touch; const { offsetX } = touch;
@ -134,7 +147,7 @@ export default {
startMoveX = state.moveX; startMoveX = state.moveX;
startMoveY = state.moveY; startMoveY = state.moveY;
touchStartTime = new Date(); touchStartTime = Date.now();
state.moving = touches.length === 1 && state.scale !== 1; state.moving = touches.length === 1 && state.scale !== 1;
state.zooming = touches.length === 2 && !offsetX.value; state.zooming = touches.length === 2 && !offsetX.value;
@ -145,7 +158,7 @@ export default {
} }
}; };
const onTouchMove = (event) => { const onTouchMove = (event: TouchEvent) => {
const { touches } = event; const { touches } = event;
touch.move(event); touch.move(event);
@ -172,7 +185,7 @@ export default {
const checkTap = () => { const checkTap = () => {
const { offsetX, offsetY } = touch; const { offsetX, offsetY } = touch;
const deltaTime = new Date() - touchStartTime; const deltaTime = Date.now() - touchStartTime;
const TAP_TIME = 250; const TAP_TIME = 250;
const TAP_OFFSET = 10; const TAP_OFFSET = 10;
@ -194,7 +207,7 @@ export default {
} }
}; };
const onTouchEnd = (event) => { const onTouchEnd = (event: TouchEvent) => {
let stopPropagation = false; let stopPropagation = false;
/* istanbul ignore else */ /* istanbul ignore else */
@ -234,8 +247,8 @@ export default {
touch.reset(); touch.reset();
}; };
const onLoad = (event) => { const onLoad = (event: Event) => {
const { naturalWidth, naturalHeight } = event.target; const { naturalWidth, naturalHeight } = event.target as HTMLImageElement;
state.imageRatio = naturalHeight / naturalWidth; state.imageRatio = naturalHeight / naturalWidth;
}; };
@ -274,4 +287,4 @@ export default {
); );
}; };
}, },
}; });

View File

@ -1,23 +1,47 @@
import { inBrowser } from '../utils'; import { App, TeleportProps } from 'vue';
import { ComponentInstance, inBrowser } from '../utils';
import { mountComponent, usePopupState } from '../utils/mount-component'; import { mountComponent, usePopupState } from '../utils/mount-component';
import { Interceptor } from '../utils/interceptor';
import { PopupCloseIconPosition } from '../popup';
import VanImagePreview from './ImagePreview'; import VanImagePreview from './ImagePreview';
let instance; let instance: ComponentInstance;
const defaultConfig = { export type ImagePreviewOptions = {
loop?: boolean;
images: string[];
maxZoom?: number;
minZoom?: number;
teleport?: TeleportProps['to'];
className?: unknown;
showIndex?: boolean;
closeable?: boolean;
closeIcon?: string;
beforeClose?: Interceptor;
swipeDuration?: number;
startPosition?: number;
showIndicators?: boolean;
closeOnPopstate?: boolean;
closeIconPosition?: PopupCloseIconPosition;
onClose?(): void;
onScale?(args: { scale: number; index: number }): void;
onChange?(index: number): void;
};
const defaultConfig: ImagePreviewOptions = {
loop: true, loop: true,
images: [], images: [],
maxZoom: 3, maxZoom: 3,
minZoom: 1 / 3, minZoom: 1 / 3,
onScale: null, onScale: undefined,
onClose: null, onClose: undefined,
onChange: null, onChange: undefined,
teleport: 'body', teleport: 'body',
className: '', className: '',
showIndex: true, showIndex: true,
closeable: false, closeable: false,
closeIcon: 'clear', closeIcon: 'clear',
beforeClose: null, beforeClose: undefined,
startPosition: 0, startPosition: 0,
swipeDuration: 300, swipeDuration: 300,
showIndicators: false, showIndicators: false,
@ -29,9 +53,8 @@ function initInstance() {
({ instance } = mountComponent({ ({ instance } = mountComponent({
setup() { setup() {
const { state, toggle } = usePopupState(); const { state, toggle } = usePopupState();
const onClosed = () => { const onClosed = () => {
state.images = []; (state as any).images = [];
}; };
return () => ( return () => (
@ -47,7 +70,10 @@ function initInstance() {
})); }));
} }
const ImagePreview = (images, startPosition = 0) => { const ImagePreview = (
images: string[] | ImagePreviewOptions,
startPosition = 0
) => {
/* istanbul ignore if */ /* istanbul ignore if */
if (!inBrowser) { if (!inBrowser) {
return; return;
@ -69,8 +95,8 @@ const ImagePreview = (images, startPosition = 0) => {
ImagePreview.Component = VanImagePreview; ImagePreview.Component = VanImagePreview;
ImagePreview.install = (app) => { ImagePreview.install = (app: App) => {
app.use(VanImagePreview); app.use(VanImagePreview as any);
}; };
export default ImagePreview; export default ImagePreview;

View File

@ -1,5 +0,0 @@
import { createNamespace } from '../utils';
const [createComponent, bem] = createNamespace('image-preview');
export { createComponent, bem };

View File

@ -12,11 +12,13 @@ declare module 'vue' {
onBlur?: EventHandler; onBlur?: EventHandler;
onOpen?: EventHandler; onOpen?: EventHandler;
onEdit?: EventHandler; onEdit?: EventHandler;
onLoad?: EventHandler;
onClose?: EventHandler; onClose?: EventHandler;
onFocus?: EventHandler; onFocus?: EventHandler;
onInput?: EventHandler; onInput?: EventHandler;
onClick?: EventHandler; onClick?: EventHandler;
onPress?: EventHandler; onPress?: EventHandler;
onScale?: EventHandler;
onCancel?: EventHandler; onCancel?: EventHandler;
onOpened?: EventHandler; onOpened?: EventHandler;
onClosed?: EventHandler; onClosed?: EventHandler;
@ -26,7 +28,10 @@ declare module 'vue' {
onToggle?: EventHandler; onToggle?: EventHandler;
onConfirm?: EventHandler; onConfirm?: EventHandler;
onKeypress?: EventHandler; onKeypress?: EventHandler;
onTouchend?: EventHandler;
onClickStep?: EventHandler; onClickStep?: EventHandler;
onTouchmove?: EventHandler;
onTouchstart?: EventHandler; onTouchstart?: EventHandler;
onTouchcancel?: EventHandler;
} }
} }