import Toast from '../../src/toast' /** Test helpers */ import { getFixture, clearFixture, 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('constructor', () => { 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() }) }) 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') const toast = new Toast(toastEl) expect(Toast.getInstance(toastEl)).toBeDefined() toast.dispose() expect(Toast.getInstance(toastEl)).toBeNull() }) 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)).toBeDefined() 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)).toBeDefined() }) 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] try { jQueryMock.fn.toast.call(jQueryMock, action) } catch (error) { expect(error.message).toEqual(`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) }) }) })