Skip to content

Commit 4136bd9

Browse files
jacobmllr95pi0
authored andcommitted
feat(dropdown): make show/hide events cancelable . also adds toggle event (#1807)
1 parent 5a7a290 commit 4136bd9

File tree

2 files changed

+70
-20
lines changed

2 files changed

+70
-20
lines changed

src/components/dropdown/package.json

+28-4
Original file line numberDiff line numberDiff line change
@@ -11,23 +11,47 @@
1111
"bDropdownDivider"
1212
],
1313
"events": [
14+
{
15+
"event": "show",
16+
"description": "Emitted just before dropdown is shown. Cancelable.",
17+
"args": [
18+
{
19+
"arg": "bvEvt",
20+
"description": "BvEvent object. Call bvEvt.preventDefault() to cancel show."
21+
}
22+
]
23+
},
1424
{
1525
"event": "shown",
16-
"description": "Emitted When dropdown is shown"
26+
"description": "Emitted when dropdown is shown."
27+
},
28+
{
29+
"event": "hide",
30+
"description": "Emitted just before dropdown is hidden. Cancelable.",
31+
"args": [
32+
{
33+
"arg": "bvEvt",
34+
"description": "BvEvent object. Call bvEvt.preventDefault() to cancel hide."
35+
}
36+
]
1737
},
1838
{
1939
"event": "hidden",
20-
"description": "Emitted When dropdown is hidden"
40+
"description": "Emitted when dropdown is hidden."
41+
},
42+
{
43+
"event": "toggle",
44+
"description": "Emitted when toggle button is clicked."
2145
},
2246
{
2347
"event": "click",
24-
"description": "Emitted when split button clicked in split mode."
48+
"description": "Emitted when split button is clicked in split mode."
2549
}
2650
],
2751
"slots": [
2852
{
2953
"name": "button-content",
30-
"description": "Can be used to implement custom text with icons and more styling"
54+
"description": "Can be used to implement custom text with icons and more styling."
3155
},
3256
{
3357
"name": "text",

src/mixins/dropdown.js

+42-16
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import listenOnRootMixin from './listen-on-root'
44
import { from as arrayFrom } from '../utils/array'
55
import { assign } from '../utils/object'
66
import KeyCodes from '../utils/key-codes'
7+
import BvEvent from '../utils/bv-event.class'
78
import warn from '../utils/warn'
89
import { isVisible, closest, selectAll, getAttr, eventOn, eventOff } from '../utils/dom'
910

@@ -68,7 +69,8 @@ export default {
6869
data () {
6970
return {
7071
visible: false,
71-
inNavbar: null
72+
inNavbar: null,
73+
visibleChangePrevented: false
7274
}
7375
},
7476
created () {
@@ -97,19 +99,36 @@ export default {
9799
this.removePopper()
98100
},
99101
watch: {
100-
visible (state, old) {
101-
if (state === old) {
102-
// Avoid duplicated emits
102+
visible (newValue, oldValue) {
103+
if (this.visibleChangePrevented) {
104+
this.visibleChangePrevented = false
103105
return
104106
}
105-
if (state) {
106-
this.showMenu()
107-
} else {
108-
this.hideMenu()
107+
108+
if (newValue !== oldValue) {
109+
const evtName = newValue ? 'show' : 'hide'
110+
let bvEvt = new BvEvent(evtName, {
111+
cancelable: true,
112+
vueTarget: this,
113+
target: this.$refs.menu,
114+
relatedTarget: null
115+
})
116+
this.emitEvent(bvEvt)
117+
if (bvEvt.defaultPrevented) {
118+
// Reset value and exit if canceled
119+
this.visibleChangePrevented = true
120+
this.visible = oldValue
121+
return
122+
}
123+
if (evtName === 'show') {
124+
this.showMenu()
125+
} else {
126+
this.hideMenu()
127+
}
109128
}
110129
},
111-
disabled (state, old) {
112-
if (state !== old && state && this.visible) {
130+
disabled (newValue, oldValue) {
131+
if (newValue !== oldValue && newValue && this.visible) {
113132
// Hide dropdown if disabled changes to true
114133
this.visible = false
115134
}
@@ -121,12 +140,16 @@ export default {
121140
}
122141
},
123142
methods: {
143+
// Event emitter
144+
emitEvent (bvEvt) {
145+
const type = bvEvt.type
146+
this.$emit(type, bvEvt)
147+
this.emitOnRoot(`bv::dropdown::${type}`, bvEvt)
148+
},
124149
showMenu () {
125150
if (this.disabled) {
126151
return
127152
}
128-
// TODO: move emit show to visible watcher, to allow cancelling of show
129-
this.$emit('show')
130153
// Ensure other menus are closed
131154
this.emitOnRoot('bv::dropdown::shown', this)
132155

@@ -157,8 +180,6 @@ export default {
157180
this.$nextTick(this.focusFirstItem)
158181
},
159182
hideMenu () {
160-
// TODO: move emit hide to visible watcher, to allow cancelling of hide
161-
this.$emit('hide')
162183
this.setTouchStart(false)
163184
this.emitOnRoot('bv::dropdown::hidden', this)
164185
this.$emit('hidden')
@@ -258,12 +279,17 @@ export default {
258279
// We only toggle on Click, Enter, Space, and Arrow Down
259280
return
260281
}
261-
evt.preventDefault()
262-
evt.stopPropagation()
263282
if (this.disabled) {
264283
this.visible = false
265284
return
266285
}
286+
this.$emit('toggle', evt)
287+
if (evt.defaultPrevented) {
288+
// Exit if canceled
289+
return
290+
}
291+
evt.preventDefault()
292+
evt.stopPropagation()
267293
// Toggle visibility
268294
this.visible = !this.visible
269295
},

0 commit comments

Comments
 (0)