dropdown.spec.js 55 KB
Newer Older
Johann-S's avatar
Johann-S committed
1
2
import Popper from 'popper.js'

3
4
import Dropdown from '../../src/dropdown'
import EventHandler from '../../src/dom/event-handler'
Johann-S's avatar
Johann-S committed
5
6

/** Test helpers */
7
import { getFixture, clearFixture, createEvent, jQueryMock } from '../helpers/fixture'
Johann-S's avatar
Johann-S committed
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41

describe('Dropdown', () => {
  let fixtureEl

  beforeAll(() => {
    fixtureEl = getFixture()
  })

  afterEach(() => {
    clearFixture()
  })

  describe('VERSION', () => {
    it('should return plugin version', () => {
      expect(Dropdown.VERSION).toEqual(jasmine.any(String))
    })
  })

  describe('Default', () => {
    it('should return plugin default config', () => {
      expect(Dropdown.Default).toEqual(jasmine.any(Object))
    })
  })

  describe('DefaultType', () => {
    it('should return plugin default type config', () => {
      expect(Dropdown.DefaultType).toEqual(jasmine.any(Object))
    })
  })

  describe('constructor', () => {
    it('should create offset modifier correctly when offset option is a function', () => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
42
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
43
44
45
46
47
48
49
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

      const getOffset = offsets => offsets
Rohit Sharma's avatar
Rohit Sharma committed
50
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
51
52
53
54
55
56
57
58
59
60
61
62
63
      const dropdown = new Dropdown(btnDropdown, {
        offset: getOffset
      })

      const offset = dropdown._getOffset()

      expect(offset.offset).toBeUndefined()
      expect(typeof offset.fn).toEqual('function')
    })

    it('should create offset modifier correctly when offset option is not a function', () => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
64
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
65
66
67
68
69
70
71
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

      const myOffset = 7
Rohit Sharma's avatar
Rohit Sharma committed
72
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
73
74
75
76
77
78
79
80
81
82
      const dropdown = new Dropdown(btnDropdown, {
        offset: myOffset
      })

      const offset = dropdown._getOffset()

      expect(offset.offset).toEqual(myOffset)
      expect(offset.fn).toBeUndefined()
    })

Rohit Sharma's avatar
Rohit Sharma committed
83
    it('should add a listener on trigger which do not have data-bs-toggle="dropdown"', () => {
Johann-S's avatar
Johann-S committed
84
85
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
86
        '  <button class="btn">Dropdown</button>',
Johann-S's avatar
Johann-S committed
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

      const btnDropdown = fixtureEl.querySelector('.btn')
      const dropdown = new Dropdown(btnDropdown)

      spyOn(dropdown, 'toggle')

      btnDropdown.click()

      expect(dropdown.toggle).toHaveBeenCalled()
    })
102

103
    it('should allow to pass config to Popper with `popperConfig`', () => {
104
105
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
106
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
107
108
109
110
111
112
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
113
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
114
115
116
117
118
119
120
121
122
123
      const dropdown = new Dropdown(btnDropdown, {
        popperConfig: {
          placement: 'left'
        }
      })

      const popperConfig = dropdown._getPopperConfig()

      expect(popperConfig.placement).toEqual('left')
    })
Johann-S's avatar
Johann-S committed
124
125
126
127
128
129
  })

  describe('toggle', () => {
    it('should toggle a dropdown', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
130
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
131
132
133
134
135
136
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
137
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
138
139
140
141
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
142
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
143
144
145
146
147
148
149
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        done()
      })

      dropdown.toggle()
    })

150
151
152
    it('should destroy old popper references on toggle', done => {
      fixtureEl.innerHTML = [
        '<div class="first dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
153
        '  <button class="firstBtn btn" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
154
155
156
157
158
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>',
        '<div class="second dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
159
        '  <button class="secondBtn btn" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
160
161
162
163
164
165
166
167
168
169
170
171
172
173
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

      const btnDropdown1 = fixtureEl.querySelector('.firstBtn')
      const btnDropdown2 = fixtureEl.querySelector('.secondBtn')
      const firstDropdownEl = fixtureEl.querySelector('.first')
      const secondDropdownEl = fixtureEl.querySelector('.second')
      const dropdown1 = new Dropdown(btnDropdown1)
      const dropdown2 = new Dropdown(btnDropdown2)

      firstDropdownEl.addEventListener('shown.bs.dropdown', () => {
174
        expect(btnDropdown1.classList.contains('show')).toEqual(true)
175
176
177
178
179
180
181
182
183
184
185
186
        spyOn(dropdown1._popper, 'destroy')
        dropdown2.toggle()
      })

      secondDropdownEl.addEventListener('shown.bs.dropdown', () => {
        expect(dropdown1._popper.destroy).toHaveBeenCalled()
        done()
      })

      dropdown1.toggle()
    })

Johann-S's avatar
Johann-S committed
187
188
189
    it('should toggle a dropdown and add/remove event listener on mobile', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
190
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
191
192
193
194
195
196
197
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

      const defaultValueOnTouchStart = document.documentElement.ontouchstart
Rohit Sharma's avatar
Rohit Sharma committed
198
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
199
200
201
202
203
204
205
206
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      document.documentElement.ontouchstart = () => {}
      spyOn(EventHandler, 'on')
      spyOn(EventHandler, 'off')

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
207
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
208
209
210
211
212
213
214
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        expect(EventHandler.on).toHaveBeenCalled()

        dropdown.toggle()
      })

      dropdownEl.addEventListener('hidden.bs.dropdown', () => {
215
        expect(btnDropdown.classList.contains('show')).toEqual(false)
Johann-S's avatar
Johann-S committed
216
217
218
219
220
221
222
223
224
225
226
227
228
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('false')
        expect(EventHandler.off).toHaveBeenCalled()

        document.documentElement.ontouchstart = defaultValueOnTouchStart
        done()
      })

      dropdown.toggle()
    })

    it('should toggle a dropdown at the right', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
229
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
230
231
232
233
234
235
        '  <div class="dropdown-menu dropdown-menu-right">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
236
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
237
238
239
240
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
241
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
242
243
244
245
246
247
248
249
250
251
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        done()
      })

      dropdown.toggle()
    })

    it('should toggle a dropup', done => {
      fixtureEl.innerHTML = [
        '<div class="dropup">',
Rohit Sharma's avatar
Rohit Sharma committed
252
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
253
254
255
256
257
258
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
259
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
260
261
262
263
      const dropupEl = fixtureEl.querySelector('.dropup')
      const dropdown = new Dropdown(btnDropdown)

      dropupEl.addEventListener('shown.bs.dropdown', () => {
264
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
265
266
267
268
269
270
271
272
273
274
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        done()
      })

      dropdown.toggle()
    })

    it('should toggle a dropup at the right', done => {
      fixtureEl.innerHTML = [
        '<div class="dropup">',
Rohit Sharma's avatar
Rohit Sharma committed
275
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
276
277
278
279
280
281
        '  <div class="dropdown-menu dropdown-menu-right">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
282
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
283
284
285
286
      const dropupEl = fixtureEl.querySelector('.dropup')
      const dropdown = new Dropdown(btnDropdown)

      dropupEl.addEventListener('shown.bs.dropdown', () => {
287
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
288
289
290
291
292
293
294
295
296
297
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        done()
      })

      dropdown.toggle()
    })

    it('should toggle a dropright', done => {
      fixtureEl.innerHTML = [
        '<div class="dropright">',
Rohit Sharma's avatar
Rohit Sharma committed
298
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
299
300
301
302
303
304
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
305
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
306
307
308
309
      const droprightEl = fixtureEl.querySelector('.dropright')
      const dropdown = new Dropdown(btnDropdown)

      droprightEl.addEventListener('shown.bs.dropdown', () => {
310
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
311
312
313
314
315
316
317
318
319
320
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        done()
      })

      dropdown.toggle()
    })

    it('should toggle a dropleft', done => {
      fixtureEl.innerHTML = [
        '<div class="dropleft">',
Rohit Sharma's avatar
Rohit Sharma committed
321
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
322
323
324
325
326
327
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
328
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
329
330
331
332
      const dropleftEl = fixtureEl.querySelector('.dropleft')
      const dropdown = new Dropdown(btnDropdown)

      dropleftEl.addEventListener('shown.bs.dropdown', () => {
333
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
334
335
336
337
338
339
340
341
342
343
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        done()
      })

      dropdown.toggle()
    })

    it('should toggle a dropdown with parent reference', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
344
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
345
346
347
348
349
350
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
351
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
352
353
354
355
356
357
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown, {
        reference: 'parent'
      })

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
358
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
359
360
361
362
363
364
365
366
367
368
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        done()
      })

      dropdown.toggle()
    })

    it('should toggle a dropdown with a dom node reference', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
369
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
370
371
372
373
374
375
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
376
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
377
378
379
380
381
382
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown, {
        reference: fixtureEl
      })

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
383
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
384
385
386
387
388
389
390
391
392
393
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        done()
      })

      dropdown.toggle()
    })

    it('should toggle a dropdown with a jquery object reference', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
394
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
395
396
397
398
399
400
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
401
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
402
403
404
405
406
407
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown, {
        reference: { 0: fixtureEl, jquery: 'jQuery' }
      })

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
408
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
409
410
411
412
413
414
415
416
417
418
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        done()
      })

      dropdown.toggle()
    })

    it('should not toggle a dropdown if the element is disabled', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
419
        '  <button disabled class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
420
421
422
423
424
425
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
426
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        throw new Error('should not throw shown.bs.dropdown event')
      })

      dropdown.toggle()

      setTimeout(() => {
        expect().nothing()
        done()
      })
    })

    it('should not toggle a dropdown if the element contains .disabled', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
445
        '  <button class="btn dropdown-toggle disabled" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
446
447
448
449
450
451
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
452
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        throw new Error('should not throw shown.bs.dropdown event')
      })

      dropdown.toggle()

      setTimeout(() => {
        expect().nothing()
        done()
      })
    })

    it('should not toggle a dropdown if the menu is shown', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
471
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
472
473
474
475
476
477
        '  <div class="dropdown-menu show">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
478
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        throw new Error('should not throw shown.bs.dropdown event')
      })

      dropdown.toggle()

      setTimeout(() => {
        expect().nothing()
        done()
      })
    })

    it('should not toggle a dropdown if show event is prevented', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
497
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
498
499
500
501
502
503
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
504
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('show.bs.dropdown', e => {
        e.preventDefault()
      })

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        throw new Error('should not throw shown.bs.dropdown event')
      })

      dropdown.toggle()

      setTimeout(() => {
        expect().nothing()
        done()
      })
    })
  })

  describe('show', () => {
    it('should show a dropdown', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
529
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
530
531
532
533
534
535
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
536
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
537
538
539
540
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
541
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
542
543
544
545
546
547
548
549
550
        done()
      })

      dropdown.show()
    })

    it('should not show a dropdown if the element is disabled', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
551
        '  <button disabled class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
552
553
554
555
556
557
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
558
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        throw new Error('should not throw shown.bs.dropdown event')
      })

      dropdown.show()

      setTimeout(() => {
        expect().nothing()
        done()
      }, 10)
    })

    it('should not show a dropdown if the element contains .disabled', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
577
        '  <button class="btn dropdown-toggle disabled" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
578
579
580
581
582
583
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
584
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        throw new Error('should not throw shown.bs.dropdown event')
      })

      dropdown.show()

      setTimeout(() => {
        expect().nothing()
        done()
      }, 10)
    })

    it('should not show a dropdown if the menu is shown', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
603
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
604
605
606
607
608
609
        '  <div class="dropdown-menu show">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
610
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        throw new Error('should not throw shown.bs.dropdown event')
      })

      dropdown.show()

      setTimeout(() => {
        expect().nothing()
        done()
      }, 10)
    })

    it('should not show a dropdown if show event is prevented', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
629
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
630
631
632
633
634
635
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
636
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('show.bs.dropdown', e => {
        e.preventDefault()
      })

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        throw new Error('should not throw shown.bs.dropdown event')
      })

      dropdown.show()

      setTimeout(() => {
        expect().nothing()
        done()
      }, 10)
    })
  })

  describe('hide', () => {
    it('should hide a dropdown', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
661
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
662
663
664
665
666
667
        '  <div class="dropdown-menu show">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
668
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
669
670
671
672
673
674
675
676
677
678
679
680
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdownMenu = fixtureEl.querySelector('.dropdown-menu')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('hidden.bs.dropdown', () => {
        expect(dropdownMenu.classList.contains('show')).toEqual(false)
        done()
      })

      dropdown.hide()
    })

681
682
683
    it('should hide a dropdown and destroy popper', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
684
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
685
686
687
688
689
690
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
691
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        spyOn(dropdown._popper, 'destroy')
        dropdown.hide()
      })

      dropdownEl.addEventListener('hidden.bs.dropdown', () => {
        expect(dropdown._popper.destroy).toHaveBeenCalled()
        done()
      })

      dropdown.show()
    })

Johann-S's avatar
Johann-S committed
708
709
710
    it('should not hide a dropdown if the element is disabled', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
711
        '  <button disabled class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
712
713
714
715
716
717
        '  <div class="dropdown-menu show">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
718
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdownMenu = fixtureEl.querySelector('.dropdown-menu')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('hidden.bs.dropdown', () => {
        throw new Error('should not throw hidden.bs.dropdown event')
      })

      dropdown.hide()

      setTimeout(() => {
        expect(dropdownMenu.classList.contains('show')).toEqual(true)
        done()
      }, 10)
    })

    it('should not hide a dropdown if the element contains .disabled', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
738
        '  <button class="btn dropdown-toggle disabled" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
739
740
741
742
743
744
        '  <div class="dropdown-menu show">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
745
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdownMenu = fixtureEl.querySelector('.dropdown-menu')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('hidden.bs.dropdown', () => {
        throw new Error('should not throw hidden.bs.dropdown event')
      })

      dropdown.hide()

      setTimeout(() => {
        expect(dropdownMenu.classList.contains('show')).toEqual(true)
        done()
      }, 10)
    })

    it('should not hide a dropdown if the menu is not shown', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
765
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
766
767
768
769
770
771
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
772
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('hidden.bs.dropdown', () => {
        throw new Error('should not throw hidden.bs.dropdown event')
      })

      dropdown.hide()

      setTimeout(() => {
        expect().nothing()
        done()
      }, 10)
    })

    it('should not hide a dropdown if hide event is prevented', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
791
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
792
793
794
795
796
797
        '  <div class="dropdown-menu show">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
798
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      const dropdownMenu = fixtureEl.querySelector('.dropdown-menu')
      const dropdown = new Dropdown(btnDropdown)

      dropdownEl.addEventListener('hide.bs.dropdown', e => {
        e.preventDefault()
      })

      dropdownEl.addEventListener('hidden.bs.dropdown', () => {
        throw new Error('should not throw hidden.bs.dropdown event')
      })

      dropdown.hide()

      setTimeout(() => {
        expect(dropdownMenu.classList.contains('show')).toEqual(true)
        done()
      })
    })
  })

  describe('dispose', () => {
    it('should dispose dropdown', () => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
824
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
825
826
827
828
829
830
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
831
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
832
833
834
835
836
837
838
839
840
841
842
843
      const dropdown = new Dropdown(btnDropdown)

      expect(dropdown._popper).toBeNull()
      expect(dropdown._menu).toBeDefined()
      expect(dropdown._element).toBeDefined()

      dropdown.dispose()

      expect(dropdown._menu).toBeNull()
      expect(dropdown._element).toBeNull()
    })

844
    it('should dispose dropdown with Popper', () => {
Johann-S's avatar
Johann-S committed
845
846
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
847
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
848
849
850
851
852
853
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
854
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
      const dropdown = new Dropdown(btnDropdown)

      dropdown.toggle()

      expect(dropdown._popper).toBeDefined()
      expect(dropdown._menu).toBeDefined()
      expect(dropdown._element).toBeDefined()

      spyOn(Popper.prototype, 'destroy')

      dropdown.dispose()

      expect(dropdown._popper).toBeNull()
      expect(dropdown._menu).toBeNull()
      expect(dropdown._element).toBeNull()
      expect(Popper.prototype.destroy).toHaveBeenCalled()
    })
  })

  describe('update', () => {
875
    it('should call Popper and detect navbar on update', () => {
Johann-S's avatar
Johann-S committed
876
877
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
878
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
879
880
881
882
883
884
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
885
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
      const dropdown = new Dropdown(btnDropdown)

      dropdown.toggle()

      expect(dropdown._popper).toBeDefined()

      spyOn(dropdown._popper, 'scheduleUpdate')
      spyOn(dropdown, '_detectNavbar')

      dropdown.update()

      expect(dropdown._popper.scheduleUpdate).toHaveBeenCalled()
      expect(dropdown._detectNavbar).toHaveBeenCalled()
    })

    it('should just detect navbar on update', () => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
904
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
905
906
907
908
909
910
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
911
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
      const dropdown = new Dropdown(btnDropdown)

      spyOn(dropdown, '_detectNavbar')

      dropdown.update()

      expect(dropdown._popper).toBeNull()
      expect(dropdown._detectNavbar).toHaveBeenCalled()
    })
  })

  describe('data-api', () => {
    it('should not add class position-static to dropdown if boundary not set', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
927
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown">Dropdown</button>',
Johann-S's avatar
Johann-S committed
928
929
930
931
932
933
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
934
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
935
936
937
938
939
940
941
942
943
944
945
946
947
      const dropdownEl = fixtureEl.querySelector('.dropdown')

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        expect(dropdownEl.classList.contains('position-static')).toEqual(false)
        done()
      })

      btnDropdown.click()
    })

    it('should add class position-static to dropdown if boundary not scrollParent', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
948
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" data-bs-boundary="viewport">Dropdown</button>',
Johann-S's avatar
Johann-S committed
949
950
951
952
953
954
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
955
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
956
957
958
959
960
961
962
963
964
965
966
967
968
      const dropdownEl = fixtureEl.querySelector('.dropdown')

      dropdownEl.addEventListener('shown.bs.dropdown', () => {
        expect(dropdownEl.classList.contains('position-static')).toEqual(true)
        done()
      })

      btnDropdown.click()
    })

    it('should show and hide a dropdown', done => {
      fixtureEl.innerHTML = [
        '<div class="dropdown">',
Rohit Sharma's avatar
Rohit Sharma committed
969
        '  <button class="btn dropdown-toggle" data-bs-toggle="dropdown" aria-expanded="false">Dropdown</button>',
Johann-S's avatar
Johann-S committed
970
971
972
973
974
975
        '  <div class="dropdown-menu">',
        '    <a class="dropdown-item" href="#">Secondary link</a>',
        '  </div>',
        '</div>'
      ].join('')

Rohit Sharma's avatar
Rohit Sharma committed
976
      const btnDropdown = fixtureEl.querySelector('[data-bs-toggle="dropdown"]')
Johann-S's avatar
Johann-S committed
977
978
979
980
981
982
983
984
985
      const dropdownEl = fixtureEl.querySelector('.dropdown')
      let showEventTriggered = false
      let hideEventTriggered = false

      dropdownEl.addEventListener('show.bs.dropdown', () => {
        showEventTriggered = true
      })

      dropdownEl.addEventListener('shown.bs.dropdown', e => {
986
        expect(btnDropdown.classList.contains('show')).toEqual(true)
Johann-S's avatar
Johann-S committed
987
988
989
990
991
992
993
994
995
996
997
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('true')
        expect(showEventTriggered).toEqual(true)
        expect(e.relatedTarget).toEqual(btnDropdown)
        document.body.click()
      })

      dropdownEl.addEventListener('hide.bs.dropdown', () => {
        hideEventTriggered = true
      })

      dropdownEl.addEventListener('hidden.bs.dropdown', e => {
998
        expect(btnDropdown.classList.contains('show')).toEqual(false)
Johann-S's avatar
Johann-S committed
999
1000
        expect(btnDropdown.getAttribute('aria-expanded')).toEqual('false')
        expect(hideEventTriggered).toEqual(true)
For faster browsing, not all history is shown. View entire blame