Files
element-plus/packages/components/image/__tests__/image.test.tsx
jiaxiang 0ff86060ab feat(components): [image-viewer] add error slot (#21961)
* 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>
2025-09-17 17:46:35 +02:00

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()
})
})
})