import { nextTick } from 'vue' import { mount } from '@vue/test-utils' import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, it, } from 'vitest' import { usePopperContainer } from '@element-plus/hooks' import { TOOLTIP_INJECTION_KEY } from '@element-plus/components/tooltip' import { genTooltipProvides } from '../test-helper/provides' import ElTooltipContent from '../src/content.vue' import type { VueWrapper } from '@vue/test-utils' const AXIOM = 'rem is the best girl' describe('', () => { const { controlled, id, open, trigger, onOpen, onClose, onToggle, onShow, onHide, onBeforeShow, onBeforeHide, } = genTooltipProvides() const defaultProvide = { [TOOLTIP_INJECTION_KEY as symbol]: { controlled, id, open, trigger, onOpen, onClose, onToggle, onShow, onHide, onBeforeShow, onBeforeHide, }, } let unmount: () => void const createComponent = (props = {}, provides = {}) => { const wrapper = mount( { setup() { usePopperContainer() return () => {AXIOM} }, }, { global: { provide: { ...defaultProvide, ...provides, }, stubs: ['ElPopperContent'], }, attachTo: document.body, } ) unmount = () => wrapper.unmount() return wrapper.findComponent(ElTooltipContent) } let wrapper: VueWrapper const OLD_ENV = process.env.NODE_ENV beforeAll(() => { process.env['NODE_ENV'] = 'test' }) afterAll(() => { process.env['NODE_ENV'] = OLD_ENV }) afterEach(() => { open.value = false controlled.value = false trigger.value = 'hover' unmount?.() document.body.innerHTML = '' }) describe('rendering', () => { it('should render nothing initially when not controlled', async () => { wrapper = createComponent() await nextTick() expect(wrapper.text()).toBe('') }) describe('persistent content', () => { it('should be able to inherit style', async () => { const customStyle = { position: 'absolute', } wrapper = createComponent({ style: customStyle, }) await nextTick() expect(wrapper.vm.contentStyle).toEqual(customStyle) }) }) describe('content rendering', () => { it('should not show the content when disabled', async () => { wrapper = createComponent({ disabled: true, }) await nextTick() expect(wrapper.vm.shouldShow).toBe(false) }) }) describe('events', () => { beforeEach(async () => { wrapper = createComponent() await nextTick() open.value = true await nextTick() await nextTick() }) it('should be able to enter trigger', async () => { const { vm } = wrapper expect(onOpen).not.toHaveBeenCalled() const enterEvent = new MouseEvent('mouseenter') vm.onContentEnter(enterEvent) expect(onOpen).toHaveBeenCalled() const leaveEvent = new MouseEvent('mouseleave') expect(onClose).not.toHaveBeenCalled() vm.onContentLeave(leaveEvent) expect(onClose).toHaveBeenCalled() }) it('should not trigger close event when the trigger is not hover', async () => { const { vm } = wrapper trigger.value = 'click' await nextTick() const leaveEvent = new MouseEvent('mouseleave') expect(onClose).not.toHaveBeenCalled() vm.onContentLeave(leaveEvent) expect(onClose).not.toHaveBeenCalled() }) it('should be able to stop triggering when controlled', async () => { controlled.value = true await nextTick() const { vm } = wrapper expect(onOpen).not.toHaveBeenCalled() const enterEvent = new MouseEvent('mouseenter') vm.onContentEnter(enterEvent) expect(onOpen).not.toHaveBeenCalled() const leaveEvent = new MouseEvent('mouseleave') expect(onClose).not.toHaveBeenCalled() vm.onContentLeave(leaveEvent) expect(onClose).not.toHaveBeenCalled() }) describe('onCloseOutside', () => { beforeEach(() => { // Have to mock this ref because we are not going to render the content in this component wrapper.vm.contentRef = { popperContentRef: document.createElement('div'), } }) it('should not close the content after click outside when trigger is hover', async () => { document.body.click() await nextTick() expect(onClose).not.toHaveBeenCalled() }) it('should not close the content after click outside when controlled', async () => { controlled.value = true trigger.value = 'click' await nextTick() document.body.click() await nextTick() expect(onClose).not.toHaveBeenCalled() }) it('should close component after click outside', async () => { trigger.value = 'click' wrapper.vm.onAfterShow() await nextTick() document.body.click() await nextTick() expect(onClose).toHaveBeenCalled() }) }) }) it('should append to', async () => { const el = document.createElement('div') const id = 'test_id' el.id = id document.body.appendChild(el) wrapper = createComponent({ appendTo: `#${id}`, }) await nextTick() expect(el.children).toHaveLength(1) expect(el.querySelector('[aria-hidden="true"]')).not.toBeNull() }) }) })