mirror of
https://github.com/element-plus/element-plus.git
synced 2026-03-13 07:51:17 +08:00
* feat(components): [image-viewer] add custom failed content * Update packages/components/image-viewer/src/image-viewer.vue Co-authored-by: Noblet Ouways <91417411+Dsaquel@users.noreply.github.com> * feat: rename error slot * test: add custom load failed slot tests for Image and ImageViewer * docs: perf dome * fix: update v * feat: add activeIndex and src properties * fix: add key binding to img element for better reactivity * fix: keep original structure * fix: restore error source in image load-failed example * feat: add image preview * refactor: remove unused var * fix: update demo * chore: better contrast for dark mode --------- Co-authored-by: Noblet Ouways <91417411+Dsaquel@users.noreply.github.com> Co-authored-by: Dsaquel <291874700n@gmail.com>
338 lines
9.5 KiB
TypeScript
338 lines
9.5 KiB
TypeScript
import { nextTick } from 'vue'
|
|
import { flushPromises, mount } from '@vue/test-utils'
|
|
import { describe, expect, test, vi } from 'vitest'
|
|
import {
|
|
IMAGE_FAIL,
|
|
IMAGE_SUCCESS,
|
|
mockImageEvent,
|
|
} from '@element-plus/test-utils/mock'
|
|
import Image from '../src/image.vue'
|
|
import triggerEvent from '@element-plus/test-utils/trigger-event'
|
|
import { EVENT_CODE } from '@element-plus/constants'
|
|
|
|
import type { AnchorHTMLAttributes, ImgHTMLAttributes } from 'vue'
|
|
import type { ImageProps } from '../src/image'
|
|
|
|
type ElImageProps = ImgHTMLAttributes &
|
|
AnchorHTMLAttributes &
|
|
Partial<ImageProps>
|
|
|
|
// firstly wait for image event
|
|
// secondly wait for vue render
|
|
async function doubleWait() {
|
|
await nextTick()
|
|
await nextTick()
|
|
}
|
|
|
|
const _mount = (template: string, data: Record<string, any>) =>
|
|
mount({
|
|
components: {
|
|
[Image.name!]: Image,
|
|
},
|
|
template,
|
|
data,
|
|
})
|
|
|
|
describe('Image.vue', () => {
|
|
test('render test', () => {
|
|
const wrapper = mount(Image)
|
|
expect(wrapper.find('.el-image').exists()).toBe(true)
|
|
})
|
|
|
|
test('imageStyle fit test', async () => {
|
|
const fits = ['fill', 'contain', 'cover', 'none', 'scale-down'] as const
|
|
for (const fit of fits) {
|
|
const wrapper = mount(() => <Image src={IMAGE_SUCCESS} fit={fit} />)
|
|
await doubleWait()
|
|
expect(wrapper.find('img').attributes('style')).toContain(
|
|
`object-fit: ${fit};`
|
|
)
|
|
}
|
|
})
|
|
|
|
test('preview classname test', async () => {
|
|
const props: ElImageProps = {
|
|
fit: 'cover',
|
|
src: IMAGE_SUCCESS,
|
|
previewSrcList: Array.from<string>({ length: 3 }).fill(IMAGE_SUCCESS),
|
|
}
|
|
const wrapper = mount(() => <Image {...props} />)
|
|
await doubleWait()
|
|
expect(wrapper.find('img').classes()).toContain('el-image__preview')
|
|
})
|
|
|
|
test('preview initial index test', async () => {
|
|
const props: ElImageProps = {
|
|
src: IMAGE_SUCCESS,
|
|
previewSrcList: Array.from<string>({ length: 3 }).map(
|
|
(_, idx) => IMAGE_FAIL + idx
|
|
),
|
|
initialIndex: 1,
|
|
}
|
|
const wrapper = mount(() => <Image {...props} />)
|
|
await doubleWait()
|
|
await wrapper.find('.el-image__inner').trigger('click')
|
|
expect(wrapper.find('.el-image-viewer__img').attributes('src')).toBe(
|
|
IMAGE_FAIL + 1
|
|
)
|
|
})
|
|
|
|
test('native loading attributes', async () => {
|
|
const wrapper = mount(Image, {
|
|
props: {
|
|
src: IMAGE_SUCCESS,
|
|
loading: 'eager',
|
|
} as ElImageProps,
|
|
})
|
|
|
|
await doubleWait()
|
|
expect(wrapper.find('img').exists()).toBe(true)
|
|
expect(wrapper.find('img').attributes('loading')).toBe('eager')
|
|
|
|
await wrapper.setProps({ loading: undefined })
|
|
expect(wrapper.find('img').attributes('loading')).toBe(undefined)
|
|
})
|
|
|
|
test('$attrs', async () => {
|
|
const alt = 'this ia alt'
|
|
const props: ElImageProps = {
|
|
alt,
|
|
src: IMAGE_SUCCESS,
|
|
referrerpolicy: 'origin',
|
|
}
|
|
const wrapper = mount(() => <Image {...props} />)
|
|
await doubleWait()
|
|
expect(wrapper.find('img').attributes('alt')).toBe(alt)
|
|
expect(wrapper.find('img').attributes('referrerpolicy')).toBe('origin')
|
|
})
|
|
|
|
test('pass event listeners', async () => {
|
|
let result = false
|
|
const props: ElImageProps = {
|
|
src: IMAGE_SUCCESS,
|
|
onClick: () => (result = true),
|
|
}
|
|
const wrapper = mount(() => <Image {...props} />)
|
|
await doubleWait()
|
|
await wrapper.find('.el-image__inner').trigger('click')
|
|
expect(result).toBeTruthy()
|
|
})
|
|
|
|
test('image preview close-on-press-escape', async () => {
|
|
const onClose = vi.fn()
|
|
const wrapper = mount(
|
|
<Image
|
|
previewSrcList={Array.from<string>({ length: 3 }).fill(IMAGE_SUCCESS)}
|
|
onClose={onClose}
|
|
closeOnPressEscape={false}
|
|
/>
|
|
)
|
|
|
|
await doubleWait()
|
|
wrapper.getCurrentComponent().exposed!.showPreview()
|
|
|
|
triggerEvent(document.body, 'keydown', EVENT_CODE.esc)
|
|
await nextTick()
|
|
expect(wrapper.vm.showViewer).toBeTruthy()
|
|
|
|
await wrapper.setProps({ closeOnPressEscape: true })
|
|
triggerEvent(document.body, 'keydown', EVENT_CODE.esc)
|
|
await nextTick()
|
|
|
|
expect(wrapper.vm.showViewer).toBeFalsy()
|
|
expect(onClose).toHaveBeenCalledTimes(1)
|
|
})
|
|
|
|
test('manually open preview', async () => {
|
|
const url = IMAGE_SUCCESS
|
|
const srcList = Array.from<string>({ length: 3 }).map(
|
|
(_, idx) => IMAGE_FAIL + idx
|
|
)
|
|
const wrapper = _mount(
|
|
`
|
|
<el-image
|
|
ref="imageRef"
|
|
style="width: 100px; height: 100px"
|
|
:src="url"
|
|
:preview-src-list="srcList"
|
|
:initial-index="1"
|
|
fit="cover"
|
|
/>`,
|
|
() => ({
|
|
url,
|
|
srcList,
|
|
})
|
|
)
|
|
await doubleWait()
|
|
wrapper.vm.$refs.imageRef.showPreview()
|
|
await doubleWait()
|
|
expect(wrapper.find('.el-image-viewer__img').attributes('src')).toBe(
|
|
IMAGE_FAIL + 1
|
|
)
|
|
})
|
|
//@todo lazy image test
|
|
|
|
test('`show-progress` prop to control whether to display progress', async () => {
|
|
const url = IMAGE_SUCCESS
|
|
const srcList = Array.from<string>({ length: 3 }).fill(IMAGE_SUCCESS)
|
|
const wrapper = _mount(
|
|
`
|
|
<el-image
|
|
ref="imageRef"
|
|
:src="url"
|
|
:preview-src-list="srcList"
|
|
:show-progress="false"
|
|
/>`,
|
|
() => ({
|
|
url,
|
|
srcList,
|
|
})
|
|
)
|
|
await doubleWait()
|
|
wrapper.vm.$refs.imageRef.showPreview()
|
|
await doubleWait()
|
|
expect(wrapper.find('.el-image-viewer__progress').exists()).toBe(false)
|
|
|
|
wrapper.setProps({ showProgress: true })
|
|
await doubleWait()
|
|
expect(wrapper.find('.el-image-viewer__progress').exists()).toBe(true)
|
|
})
|
|
|
|
test('progress slot', async () => {
|
|
const url = IMAGE_SUCCESS
|
|
const srcList = Array.from<string>({ length: 3 }).fill(IMAGE_SUCCESS)
|
|
const wrapper = _mount(
|
|
`
|
|
<el-image
|
|
ref="imageRef"
|
|
:src="url"
|
|
:preview-src-list="srcList"
|
|
>
|
|
<template #progress="{ activeIndex, total }">
|
|
<div>{{ activeIndex + 1 }} - {{ total }}</div>
|
|
</template>
|
|
</el-image>`,
|
|
() => ({
|
|
url,
|
|
srcList,
|
|
})
|
|
)
|
|
await doubleWait()
|
|
wrapper.vm.$refs.imageRef.showPreview()
|
|
await doubleWait()
|
|
expect(wrapper.find('.el-image-viewer__progress').exists()).toBe(true)
|
|
expect(wrapper.find('.el-image-viewer__progress').text()).toBe('1 - 3')
|
|
|
|
// progress slot's priority is higher than `show-progress` prop
|
|
wrapper.setProps({ showProgress: false })
|
|
await doubleWait()
|
|
expect(wrapper.find('.el-image-viewer__progress').exists()).toBe(true)
|
|
})
|
|
|
|
test('custom viewer load failed slot', async () => {
|
|
const url = IMAGE_SUCCESS
|
|
const srcList = ['error']
|
|
const wrapper = _mount(
|
|
`
|
|
<el-image
|
|
ref="imageRef"
|
|
:src="url"
|
|
:preview-src-list="srcList"
|
|
>
|
|
<template #viewer-error>
|
|
<div class="load-failed-slot">
|
|
load failed slot
|
|
</div>
|
|
</template>
|
|
</el-image>`,
|
|
() => ({
|
|
url,
|
|
srcList,
|
|
})
|
|
)
|
|
|
|
await doubleWait()
|
|
wrapper.vm.$refs.imageRef.showPreview()
|
|
await doubleWait()
|
|
|
|
const img = wrapper.find('.el-image-viewer__canvas img')
|
|
await img.trigger('error')
|
|
await doubleWait()
|
|
expect(wrapper.find('.load-failed-slot').exists()).toBe(true)
|
|
})
|
|
|
|
describe('load', () => {
|
|
mockImageEvent()
|
|
|
|
test('image load success test', async () => {
|
|
const alt = 'this ia alt'
|
|
const wrapper = mount({
|
|
setup() {
|
|
const props: ElImageProps = {
|
|
alt,
|
|
src: IMAGE_SUCCESS,
|
|
}
|
|
return () => <Image {...props} />
|
|
},
|
|
})
|
|
expect(wrapper.find('.el-image__placeholder').exists()).toBe(true)
|
|
await flushPromises()
|
|
expect(wrapper.find('.el-image__inner').exists()).toBe(true)
|
|
expect(wrapper.find('img').exists()).toBe(true)
|
|
await flushPromises()
|
|
expect(wrapper.find('.el-image__placeholder').exists()).toBe(false)
|
|
expect(wrapper.find('.el-image__error').exists()).toBe(false)
|
|
})
|
|
|
|
test('image load error test', async () => {
|
|
const wrapper = mount(Image, {
|
|
props: {
|
|
src: IMAGE_FAIL,
|
|
},
|
|
})
|
|
await doubleWait()
|
|
wrapper.emitted('error') && expect(wrapper.emitted('error')).toBeDefined()
|
|
expect(wrapper.find('.el-image__inner').exists()).toBe(false)
|
|
expect(wrapper.find('img').exists()).toBe(false)
|
|
expect(wrapper.find('.el-image__error').exists()).toBe(true)
|
|
})
|
|
|
|
test('image load sequence success test', async () => {
|
|
const wrapper = mount(Image, {
|
|
props: {
|
|
src: IMAGE_FAIL,
|
|
},
|
|
})
|
|
await flushPromises()
|
|
const errorCountBefore = wrapper.emitted('error')?.length || 0
|
|
|
|
await wrapper.setProps({ src: IMAGE_SUCCESS })
|
|
await nextTick()
|
|
const img = wrapper.find('img')
|
|
if (img.exists()) {
|
|
await img.trigger('load')
|
|
}
|
|
await flushPromises()
|
|
// expect no new error event to be emitted
|
|
expect(wrapper.emitted('error')?.length).toBe(errorCountBefore)
|
|
|
|
expect(wrapper.find('.el-image__inner').exists()).toBe(true)
|
|
expect(wrapper.find('img').exists()).toBe(true)
|
|
expect(wrapper.find('.el-image__placeholder').exists()).toBe(false)
|
|
expect(wrapper.find('.el-image__error').exists()).toBe(false)
|
|
})
|
|
|
|
test('emit load event', async () => {
|
|
const handleLoad = vi.fn()
|
|
const props: ElImageProps = {
|
|
src: IMAGE_SUCCESS,
|
|
onLoad: handleLoad,
|
|
}
|
|
const wrapper = mount(() => <Image {...props} />)
|
|
await flushPromises()
|
|
expect(wrapper.find('.el-image__inner').exists()).toBe(true)
|
|
expect(handleLoad).toBeCalled()
|
|
})
|
|
})
|
|
})
|