2019-10-02 11:43:54 +02:00
|
|
|
import EventHandler from '../../../src/dom/event-handler'
|
|
|
|
import { getFixture, clearFixture } from '../../helpers/fixture'
|
2019-03-13 16:23:50 +02:00
|
|
|
|
|
|
|
describe('EventHandler', () => {
|
|
|
|
let fixtureEl
|
|
|
|
|
|
|
|
beforeAll(() => {
|
|
|
|
fixtureEl = getFixture()
|
|
|
|
})
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
clearFixture()
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('on', () => {
|
|
|
|
it('should not add event listener if the event is not a string', () => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
EventHandler.on(div, null, () => {})
|
|
|
|
EventHandler.on(null, 'click', () => {})
|
|
|
|
|
|
|
|
expect().nothing()
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should add event listener', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
EventHandler.on(div, 'click', () => {
|
|
|
|
expect().nothing()
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
|
|
|
|
div.click()
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should add namespaced event listener', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
EventHandler.on(div, 'bs.namespace', () => {
|
|
|
|
expect().nothing()
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
|
|
|
|
EventHandler.trigger(div, 'bs.namespace')
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should add native namespaced event listener', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
EventHandler.on(div, 'click.namespace', () => {
|
|
|
|
expect().nothing()
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
|
|
|
|
EventHandler.trigger(div, 'click')
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should handle event delegation', done => {
|
|
|
|
EventHandler.on(document, 'click', '.test', () => {
|
|
|
|
expect().nothing()
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
|
|
|
|
fixtureEl.innerHTML = '<div class="test"></div>'
|
|
|
|
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
div.click()
|
|
|
|
})
|
2021-04-13 05:25:58 +02:00
|
|
|
|
|
|
|
it('should handle mouseenter/mouseleave like the native counterpart', done => {
|
|
|
|
fixtureEl.innerHTML = [
|
|
|
|
'<div class="outer">',
|
|
|
|
'<div class="inner">',
|
|
|
|
'<div class="nested">',
|
|
|
|
'<div class="deep"></div>',
|
|
|
|
'</div>',
|
|
|
|
'</div>',
|
2021-04-19 07:30:33 +02:00
|
|
|
'<div class="sibling"></div>',
|
2021-04-13 05:25:58 +02:00
|
|
|
'</div>'
|
2021-12-09 16:01:29 +02:00
|
|
|
].join('')
|
2021-04-13 05:25:58 +02:00
|
|
|
|
|
|
|
const outer = fixtureEl.querySelector('.outer')
|
|
|
|
const inner = fixtureEl.querySelector('.inner')
|
|
|
|
const nested = fixtureEl.querySelector('.nested')
|
|
|
|
const deep = fixtureEl.querySelector('.deep')
|
2021-04-19 07:30:33 +02:00
|
|
|
const sibling = fixtureEl.querySelector('.sibling')
|
2021-04-13 05:25:58 +02:00
|
|
|
|
|
|
|
const enterSpy = jasmine.createSpy('mouseenter')
|
|
|
|
const leaveSpy = jasmine.createSpy('mouseleave')
|
|
|
|
const delegateEnterSpy = jasmine.createSpy('mouseenter')
|
|
|
|
const delegateLeaveSpy = jasmine.createSpy('mouseleave')
|
|
|
|
|
|
|
|
EventHandler.on(inner, 'mouseenter', enterSpy)
|
|
|
|
EventHandler.on(inner, 'mouseleave', leaveSpy)
|
|
|
|
EventHandler.on(outer, 'mouseenter', '.inner', delegateEnterSpy)
|
|
|
|
EventHandler.on(outer, 'mouseleave', '.inner', delegateLeaveSpy)
|
|
|
|
|
2021-04-19 07:30:33 +02:00
|
|
|
EventHandler.on(sibling, 'mouseenter', () => {
|
2021-10-14 18:16:54 +03:00
|
|
|
expect(enterSpy.calls.count()).toEqual(2)
|
|
|
|
expect(leaveSpy.calls.count()).toEqual(2)
|
|
|
|
expect(delegateEnterSpy.calls.count()).toEqual(2)
|
|
|
|
expect(delegateLeaveSpy.calls.count()).toEqual(2)
|
2021-04-19 07:30:33 +02:00
|
|
|
done()
|
|
|
|
})
|
|
|
|
|
2021-04-13 05:25:58 +02:00
|
|
|
const moveMouse = (from, to) => {
|
|
|
|
from.dispatchEvent(new MouseEvent('mouseout', {
|
|
|
|
bubbles: true,
|
|
|
|
relatedTarget: to
|
|
|
|
}))
|
|
|
|
|
|
|
|
to.dispatchEvent(new MouseEvent('mouseover', {
|
|
|
|
bubbles: true,
|
|
|
|
relatedTarget: from
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
2021-04-19 07:30:33 +02:00
|
|
|
// from outer to deep and back to outer (nested)
|
2021-04-13 05:25:58 +02:00
|
|
|
moveMouse(outer, inner)
|
|
|
|
moveMouse(inner, nested)
|
|
|
|
moveMouse(nested, deep)
|
|
|
|
moveMouse(deep, nested)
|
|
|
|
moveMouse(nested, inner)
|
|
|
|
moveMouse(inner, outer)
|
|
|
|
|
|
|
|
setTimeout(() => {
|
2021-10-14 18:16:54 +03:00
|
|
|
expect(enterSpy.calls.count()).toEqual(1)
|
|
|
|
expect(leaveSpy.calls.count()).toEqual(1)
|
|
|
|
expect(delegateEnterSpy.calls.count()).toEqual(1)
|
|
|
|
expect(delegateLeaveSpy.calls.count()).toEqual(1)
|
2021-04-19 07:30:33 +02:00
|
|
|
|
|
|
|
// from outer to inner to sibling (adjacent)
|
|
|
|
moveMouse(outer, inner)
|
|
|
|
moveMouse(inner, sibling)
|
2021-04-13 05:25:58 +02:00
|
|
|
}, 20)
|
|
|
|
})
|
2019-03-13 16:23:50 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
describe('one', () => {
|
2021-04-13 05:25:58 +02:00
|
|
|
it('should call listener just once', done => {
|
2019-03-13 16:23:50 +02:00
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
|
|
|
|
let called = 0
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
const obj = {
|
|
|
|
oneListener() {
|
|
|
|
called++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
EventHandler.one(div, 'bootstrap', obj.oneListener)
|
|
|
|
|
|
|
|
EventHandler.trigger(div, 'bootstrap')
|
|
|
|
EventHandler.trigger(div, 'bootstrap')
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(called).toEqual(1)
|
|
|
|
done()
|
|
|
|
}, 20)
|
|
|
|
})
|
2021-04-13 05:25:58 +02:00
|
|
|
|
|
|
|
it('should call delegated listener just once', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
|
|
|
|
let called = 0
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
const obj = {
|
|
|
|
oneListener() {
|
|
|
|
called++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
EventHandler.one(fixtureEl, 'bootstrap', 'div', obj.oneListener)
|
|
|
|
|
|
|
|
EventHandler.trigger(div, 'bootstrap')
|
|
|
|
EventHandler.trigger(div, 'bootstrap')
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(called).toEqual(1)
|
|
|
|
done()
|
|
|
|
}, 20)
|
|
|
|
})
|
2019-03-13 16:23:50 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
describe('off', () => {
|
|
|
|
it('should not remove a listener', () => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
EventHandler.off(div, null, () => {})
|
|
|
|
EventHandler.off(null, 'click', () => {})
|
|
|
|
expect().nothing()
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove a listener', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
let called = 0
|
|
|
|
const handler = () => {
|
|
|
|
called++
|
|
|
|
}
|
|
|
|
|
|
|
|
EventHandler.on(div, 'foobar', handler)
|
|
|
|
EventHandler.trigger(div, 'foobar')
|
|
|
|
|
|
|
|
EventHandler.off(div, 'foobar', handler)
|
|
|
|
EventHandler.trigger(div, 'foobar')
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(called).toEqual(1)
|
|
|
|
done()
|
|
|
|
}, 20)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove all the events', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
let called = 0
|
|
|
|
|
|
|
|
EventHandler.on(div, 'foobar', () => {
|
|
|
|
called++
|
|
|
|
})
|
|
|
|
EventHandler.on(div, 'foobar', () => {
|
|
|
|
called++
|
|
|
|
})
|
|
|
|
EventHandler.trigger(div, 'foobar')
|
|
|
|
|
|
|
|
EventHandler.off(div, 'foobar')
|
|
|
|
EventHandler.trigger(div, 'foobar')
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(called).toEqual(2)
|
|
|
|
done()
|
|
|
|
}, 20)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove all the namespaced listeners if namespace is passed', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
let called = 0
|
|
|
|
|
|
|
|
EventHandler.on(div, 'foobar.namespace', () => {
|
|
|
|
called++
|
|
|
|
})
|
|
|
|
EventHandler.on(div, 'foofoo.namespace', () => {
|
|
|
|
called++
|
|
|
|
})
|
|
|
|
EventHandler.trigger(div, 'foobar.namespace')
|
|
|
|
EventHandler.trigger(div, 'foofoo.namespace')
|
|
|
|
|
|
|
|
EventHandler.off(div, '.namespace')
|
|
|
|
EventHandler.trigger(div, 'foobar.namespace')
|
|
|
|
EventHandler.trigger(div, 'foofoo.namespace')
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(called).toEqual(2)
|
|
|
|
done()
|
|
|
|
}, 20)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove the namespaced listeners', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
let calledCallback1 = 0
|
|
|
|
let calledCallback2 = 0
|
|
|
|
|
|
|
|
EventHandler.on(div, 'foobar.namespace', () => {
|
|
|
|
calledCallback1++
|
|
|
|
})
|
|
|
|
EventHandler.on(div, 'foofoo.namespace', () => {
|
|
|
|
calledCallback2++
|
|
|
|
})
|
|
|
|
|
|
|
|
EventHandler.trigger(div, 'foobar.namespace')
|
|
|
|
EventHandler.off(div, 'foobar.namespace')
|
|
|
|
EventHandler.trigger(div, 'foobar.namespace')
|
|
|
|
|
|
|
|
EventHandler.trigger(div, 'foofoo.namespace')
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(calledCallback1).toEqual(1)
|
|
|
|
expect(calledCallback2).toEqual(1)
|
|
|
|
done()
|
|
|
|
}, 20)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove the all the namespaced listeners for native events', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
let called = 0
|
|
|
|
|
|
|
|
EventHandler.on(div, 'click.namespace', () => {
|
|
|
|
called++
|
|
|
|
})
|
|
|
|
EventHandler.on(div, 'click.namespace2', () => {
|
|
|
|
called++
|
|
|
|
})
|
|
|
|
|
|
|
|
EventHandler.trigger(div, 'click')
|
|
|
|
EventHandler.off(div, 'click')
|
|
|
|
EventHandler.trigger(div, 'click')
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(called).toEqual(2)
|
|
|
|
done()
|
|
|
|
}, 20)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove the specified namespaced listeners for native events', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
|
|
|
|
let called1 = 0
|
|
|
|
let called2 = 0
|
|
|
|
|
|
|
|
EventHandler.on(div, 'click.namespace', () => {
|
|
|
|
called1++
|
|
|
|
})
|
|
|
|
EventHandler.on(div, 'click.namespace2', () => {
|
|
|
|
called2++
|
|
|
|
})
|
|
|
|
EventHandler.trigger(div, 'click')
|
|
|
|
|
|
|
|
EventHandler.off(div, 'click.namespace')
|
|
|
|
EventHandler.trigger(div, 'click')
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(called1).toEqual(1)
|
|
|
|
expect(called2).toEqual(2)
|
|
|
|
done()
|
|
|
|
}, 20)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove a listener registered by .one', done => {
|
|
|
|
fixtureEl.innerHTML = '<div></div>'
|
|
|
|
|
|
|
|
const div = fixtureEl.querySelector('div')
|
|
|
|
const handler = () => {
|
|
|
|
throw new Error('called')
|
|
|
|
}
|
|
|
|
|
|
|
|
EventHandler.one(div, 'foobar', handler)
|
|
|
|
EventHandler.off(div, 'foobar', handler)
|
|
|
|
|
|
|
|
EventHandler.trigger(div, 'foobar')
|
|
|
|
setTimeout(() => {
|
|
|
|
expect().nothing()
|
|
|
|
done()
|
|
|
|
}, 20)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove the correct delegated event listener', () => {
|
|
|
|
const element = document.createElement('div')
|
|
|
|
const subelement = document.createElement('span')
|
2021-07-30 01:23:00 +03:00
|
|
|
element.append(subelement)
|
2019-03-13 16:23:50 +02:00
|
|
|
|
|
|
|
const anchor = document.createElement('a')
|
2021-07-30 01:23:00 +03:00
|
|
|
element.append(anchor)
|
2019-03-13 16:23:50 +02:00
|
|
|
|
|
|
|
let i = 0
|
|
|
|
const handler = () => {
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
|
|
|
EventHandler.on(element, 'click', 'a', handler)
|
|
|
|
EventHandler.on(element, 'click', 'span', handler)
|
|
|
|
|
2021-07-30 01:23:00 +03:00
|
|
|
fixtureEl.append(element)
|
2019-03-13 16:23:50 +02:00
|
|
|
|
|
|
|
EventHandler.trigger(anchor, 'click')
|
|
|
|
EventHandler.trigger(subelement, 'click')
|
|
|
|
|
|
|
|
// first listeners called
|
2020-11-02 16:13:24 +02:00
|
|
|
expect(i).toEqual(2)
|
2019-03-13 16:23:50 +02:00
|
|
|
|
|
|
|
EventHandler.off(element, 'click', 'span', handler)
|
|
|
|
EventHandler.trigger(subelement, 'click')
|
|
|
|
|
|
|
|
// removed listener not called
|
2020-11-02 16:13:24 +02:00
|
|
|
expect(i).toEqual(2)
|
2019-03-13 16:23:50 +02:00
|
|
|
|
|
|
|
EventHandler.trigger(anchor, 'click')
|
|
|
|
|
|
|
|
// not removed listener called
|
2020-11-02 16:13:24 +02:00
|
|
|
expect(i).toEqual(3)
|
2019-03-13 16:23:50 +02:00
|
|
|
|
|
|
|
EventHandler.on(element, 'click', 'span', handler)
|
|
|
|
EventHandler.trigger(anchor, 'click')
|
|
|
|
EventHandler.trigger(subelement, 'click')
|
|
|
|
|
|
|
|
// listener re-registered
|
2020-11-02 16:13:24 +02:00
|
|
|
expect(i).toEqual(5)
|
2019-03-13 16:23:50 +02:00
|
|
|
|
|
|
|
EventHandler.off(element, 'click', 'span')
|
|
|
|
EventHandler.trigger(subelement, 'click')
|
|
|
|
|
|
|
|
// listener removed again
|
2020-11-02 16:13:24 +02:00
|
|
|
expect(i).toEqual(5)
|
2019-03-13 16:23:50 +02:00
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|