import Toast from '../../src/toast' /** Test helpers */ import { getFixture, clearFixture, createEvent, jQueryMock } from '../helpers/fixture' describe('Toast', () => { let fixtureEl beforeAll(() => { fixtureEl = getFixture() }) afterEach(() => { clearFixture() }) describe('VERSION', () => { it('should return plugin version', () => { expect(Toast.VERSION).toEqual(jasmine.any(String)) }) }) describe('DATA_KEY', () => { it('should return plugin data key', () => { expect(Toast.DATA_KEY).toEqual('bs.toast') }) }) describe('constructor', () => { it('should take care of element either passed as a CSS selector or DOM element', () => { fixtureEl.innerHTML = '
' const toastEl = fixtureEl.querySelector('.toast') const toastBySelector = new Toast('.toast') const toastByElement = new Toast(toastEl) expect(toastBySelector._element).toEqual(toastEl) expect(toastByElement._element).toEqual(toastEl) }) it('should allow to config in js', done => { fixtureEl.innerHTML = [ '
', '
', ' a simple toast', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('div') const toast = new Toast(toastEl, { delay: 1 }) toastEl.addEventListener('shown.bs.toast', () => { expect(toastEl.classList.contains('show')).toEqual(true) done() }) toast.show() }) it('should close toast when close element with data-bs-dismiss attribute is set', done => { fixtureEl.innerHTML = [ '
', ' ', '
' ].join('') const toastEl = fixtureEl.querySelector('div') const toast = new Toast(toastEl) toastEl.addEventListener('shown.bs.toast', () => { expect(toastEl.classList.contains('show')).toEqual(true) const button = toastEl.querySelector('.btn-close') button.click() }) toastEl.addEventListener('hidden.bs.toast', () => { expect(toastEl.classList.contains('show')).toEqual(false) done() }) toast.show() }) }) describe('Default', () => { it('should expose default setting to allow to override them', () => { const defaultDelay = 1000 Toast.Default.delay = defaultDelay fixtureEl.innerHTML = [ '
', ' ', '
' ].join('') const toastEl = fixtureEl.querySelector('div') const toast = new Toast(toastEl) expect(toast._config.delay).toEqual(defaultDelay) }) }) describe('DefaultType', () => { it('should expose default setting types for read', () => { expect(Toast.DefaultType).toEqual(jasmine.any(Object)) }) }) describe('show', () => { it('should auto hide', done => { fixtureEl.innerHTML = [ '
', '
', ' a simple toast', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) toastEl.addEventListener('hidden.bs.toast', () => { expect(toastEl.classList.contains('show')).toEqual(false) done() }) toast.show() }) it('should not add fade class', done => { fixtureEl.innerHTML = [ '
', '
', ' a simple toast', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) toastEl.addEventListener('shown.bs.toast', () => { expect(toastEl.classList.contains('fade')).toEqual(false) done() }) toast.show() }) it('should not trigger shown if show is prevented', done => { fixtureEl.innerHTML = [ '
', '
', ' a simple toast', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) const assertDone = () => { setTimeout(() => { expect(toastEl.classList.contains('show')).toEqual(false) done() }, 20) } toastEl.addEventListener('show.bs.toast', event => { event.preventDefault() assertDone() }) toastEl.addEventListener('shown.bs.toast', () => { throw new Error('shown event should not be triggered if show is prevented') }) toast.show() }) it('should clear timeout if toast is shown again before it is hidden', done => { fixtureEl.innerHTML = [ '
', '
', ' a simple toast', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) setTimeout(() => { toast._config.autohide = false toastEl.addEventListener('shown.bs.toast', () => { expect(toast._clearTimeout).toHaveBeenCalled() expect(toast._timeout).toBeNull() done() }) toast.show() }, toast._config.delay / 2) spyOn(toast, '_clearTimeout').and.callThrough() toast.show() }) it('should clear timeout if toast is interacted with mouse', done => { fixtureEl.innerHTML = [ '
', '
', ' a simple toast', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) const spy = spyOn(toast, '_clearTimeout').and.callThrough() setTimeout(() => { spy.calls.reset() toastEl.addEventListener('mouseover', () => { expect(toast._clearTimeout).toHaveBeenCalledTimes(1) expect(toast._timeout).toBeNull() done() }) const mouseOverEvent = createEvent('mouseover') toastEl.dispatchEvent(mouseOverEvent) }, toast._config.delay / 2) toast.show() }) it('should clear timeout if toast is interacted with keyboard', done => { fixtureEl.innerHTML = [ '', '
', '
', ' a simple toast', ' ', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) const spy = spyOn(toast, '_clearTimeout').and.callThrough() setTimeout(() => { spy.calls.reset() toastEl.addEventListener('focusin', () => { expect(toast._clearTimeout).toHaveBeenCalledTimes(1) expect(toast._timeout).toBeNull() done() }) const insideFocusable = toastEl.querySelector('button') insideFocusable.focus() }, toast._config.delay / 2) toast.show() }) it('should still auto hide after being interacted with mouse and keyboard', done => { fixtureEl.innerHTML = [ '', '
', '
', ' a simple toast', ' ', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) setTimeout(() => { toastEl.addEventListener('mouseover', () => { const insideFocusable = toastEl.querySelector('button') insideFocusable.focus() }) toastEl.addEventListener('focusin', () => { const mouseOutEvent = createEvent('mouseout') toastEl.dispatchEvent(mouseOutEvent) }) toastEl.addEventListener('mouseout', () => { const outsideFocusable = document.getElementById('outside-focusable') outsideFocusable.focus() }) toastEl.addEventListener('focusout', () => { expect(toast._timeout).not.toBeNull() done() }) const mouseOverEvent = createEvent('mouseover') toastEl.dispatchEvent(mouseOverEvent) }, toast._config.delay / 2) toast.show() }) it('should not auto hide if focus leaves but mouse pointer remains inside', done => { fixtureEl.innerHTML = [ '', '
', '
', ' a simple toast', ' ', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) setTimeout(() => { toastEl.addEventListener('mouseover', () => { const insideFocusable = toastEl.querySelector('button') insideFocusable.focus() }) toastEl.addEventListener('focusin', () => { const outsideFocusable = document.getElementById('outside-focusable') outsideFocusable.focus() }) toastEl.addEventListener('focusout', () => { expect(toast._timeout).toBeNull() done() }) const mouseOverEvent = createEvent('mouseover') toastEl.dispatchEvent(mouseOverEvent) }, toast._config.delay / 2) toast.show() }) it('should not auto hide if mouse pointer leaves but focus remains inside', done => { fixtureEl.innerHTML = [ '', '
', '
', ' a simple toast', ' ', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) setTimeout(() => { toastEl.addEventListener('mouseover', () => { const insideFocusable = toastEl.querySelector('button') insideFocusable.focus() }) toastEl.addEventListener('focusin', () => { const mouseOutEvent = createEvent('mouseout') toastEl.dispatchEvent(mouseOutEvent) }) toastEl.addEventListener('mouseout', () => { expect(toast._timeout).toBeNull() done() }) const mouseOverEvent = createEvent('mouseover') toastEl.dispatchEvent(mouseOverEvent) }, toast._config.delay / 2) toast.show() }) }) describe('hide', () => { it('should allow to hide toast manually', done => { fixtureEl.innerHTML = [ '
', '
', ' a simple toast', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) toastEl.addEventListener('shown.bs.toast', () => { toast.hide() }) toastEl.addEventListener('hidden.bs.toast', () => { expect(toastEl.classList.contains('show')).toEqual(false) done() }) toast.show() }) it('should do nothing when we call hide on a non shown toast', () => { fixtureEl.innerHTML = '
' const toastEl = fixtureEl.querySelector('div') const toast = new Toast(toastEl) spyOn(toastEl.classList, 'contains') toast.hide() expect(toastEl.classList.contains).toHaveBeenCalled() }) it('should not trigger hidden if hide is prevented', done => { fixtureEl.innerHTML = [ '
', '
', ' a simple toast', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('.toast') const toast = new Toast(toastEl) const assertDone = () => { setTimeout(() => { expect(toastEl.classList.contains('show')).toEqual(true) done() }, 20) } toastEl.addEventListener('shown.bs.toast', () => { toast.hide() }) toastEl.addEventListener('hide.bs.toast', event => { event.preventDefault() assertDone() }) toastEl.addEventListener('hidden.bs.toast', () => { throw new Error('hidden event should not be triggered if hide is prevented') }) toast.show() }) }) describe('dispose', () => { it('should allow to destroy toast', () => { fixtureEl.innerHTML = '
' const toastEl = fixtureEl.querySelector('div') spyOn(toastEl, 'addEventListener').and.callThrough() spyOn(toastEl, 'removeEventListener').and.callThrough() const toast = new Toast(toastEl) expect(Toast.getInstance(toastEl)).not.toBeNull() expect(toastEl.addEventListener).toHaveBeenCalledWith('click', jasmine.any(Function), jasmine.any(Boolean)) toast.dispose() expect(Toast.getInstance(toastEl)).toBeNull() expect(toastEl.removeEventListener).toHaveBeenCalledWith('click', jasmine.any(Function), jasmine.any(Boolean)) }) it('should allow to destroy toast and hide it before that', done => { fixtureEl.innerHTML = [ '
', '
', ' a simple toast', '
', '
' ].join('') const toastEl = fixtureEl.querySelector('div') const toast = new Toast(toastEl) const expected = () => { expect(toastEl.classList.contains('show')).toEqual(true) expect(Toast.getInstance(toastEl)).not.toBeNull() toast.dispose() expect(Toast.getInstance(toastEl)).toBeNull() expect(toastEl.classList.contains('show')).toEqual(false) done() } toastEl.addEventListener('shown.bs.toast', () => { setTimeout(expected, 1) }) toast.show() }) }) describe('jQueryInterface', () => { it('should create a toast', () => { fixtureEl.innerHTML = '
' const div = fixtureEl.querySelector('div') jQueryMock.fn.toast = Toast.jQueryInterface jQueryMock.elements = [div] jQueryMock.fn.toast.call(jQueryMock) expect(Toast.getInstance(div)).not.toBeNull() }) it('should not re create a toast', () => { fixtureEl.innerHTML = '
' const div = fixtureEl.querySelector('div') const toast = new Toast(div) jQueryMock.fn.toast = Toast.jQueryInterface jQueryMock.elements = [div] jQueryMock.fn.toast.call(jQueryMock) expect(Toast.getInstance(div)).toEqual(toast) }) it('should call a toast method', () => { fixtureEl.innerHTML = '
' const div = fixtureEl.querySelector('div') const toast = new Toast(div) spyOn(toast, 'show') jQueryMock.fn.toast = Toast.jQueryInterface jQueryMock.elements = [div] jQueryMock.fn.toast.call(jQueryMock, 'show') expect(Toast.getInstance(div)).toEqual(toast) expect(toast.show).toHaveBeenCalled() }) it('should throw error on undefined method', () => { fixtureEl.innerHTML = '
' const div = fixtureEl.querySelector('div') const action = 'undefinedMethod' jQueryMock.fn.toast = Toast.jQueryInterface jQueryMock.elements = [div] expect(() => { jQueryMock.fn.toast.call(jQueryMock, action) }).toThrowError(TypeError, `No method named "${action}"`) }) }) describe('getInstance', () => { it('should return a toast instance', () => { fixtureEl.innerHTML = '
' const div = fixtureEl.querySelector('div') const toast = new Toast(div) expect(Toast.getInstance(div)).toEqual(toast) expect(Toast.getInstance(div)).toBeInstanceOf(Toast) }) it('should return null when there is no toast instance', () => { fixtureEl.innerHTML = '
' const div = fixtureEl.querySelector('div') expect(Toast.getInstance(div)).toEqual(null) }) }) })