@@ -4,6 +4,7 @@ import listenOnRootMixin from './listen-on-root'
4
4
import { from as arrayFrom } from '../utils/array'
5
5
import { assign } from '../utils/object'
6
6
import KeyCodes from '../utils/key-codes'
7
+ import BvEvent from '../utils/bv-event.class'
7
8
import warn from '../utils/warn'
8
9
import { isVisible , closest , selectAll , getAttr , eventOn , eventOff } from '../utils/dom'
9
10
@@ -68,7 +69,8 @@ export default {
68
69
data ( ) {
69
70
return {
70
71
visible : false ,
71
- inNavbar : null
72
+ inNavbar : null ,
73
+ visibleChangePrevented : false
72
74
}
73
75
} ,
74
76
created ( ) {
@@ -97,19 +99,36 @@ export default {
97
99
this . removePopper ( )
98
100
} ,
99
101
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
103
105
return
104
106
}
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
+ }
109
128
}
110
129
} ,
111
- disabled ( state , old ) {
112
- if ( state !== old && state && this . visible ) {
130
+ disabled ( newValue , oldValue ) {
131
+ if ( newValue !== oldValue && newValue && this . visible ) {
113
132
// Hide dropdown if disabled changes to true
114
133
this . visible = false
115
134
}
@@ -121,12 +140,16 @@ export default {
121
140
}
122
141
} ,
123
142
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
+ } ,
124
149
showMenu ( ) {
125
150
if ( this . disabled ) {
126
151
return
127
152
}
128
- // TODO: move emit show to visible watcher, to allow cancelling of show
129
- this . $emit ( 'show' )
130
153
// Ensure other menus are closed
131
154
this . emitOnRoot ( 'bv::dropdown::shown' , this )
132
155
@@ -157,8 +180,6 @@ export default {
157
180
this . $nextTick ( this . focusFirstItem )
158
181
} ,
159
182
hideMenu ( ) {
160
- // TODO: move emit hide to visible watcher, to allow cancelling of hide
161
- this . $emit ( 'hide' )
162
183
this . setTouchStart ( false )
163
184
this . emitOnRoot ( 'bv::dropdown::hidden' , this )
164
185
this . $emit ( 'hidden' )
@@ -258,12 +279,17 @@ export default {
258
279
// We only toggle on Click, Enter, Space, and Arrow Down
259
280
return
260
281
}
261
- evt . preventDefault ( )
262
- evt . stopPropagation ( )
263
282
if ( this . disabled ) {
264
283
this . visible = false
265
284
return
266
285
}
286
+ this . $emit ( 'toggle' , evt )
287
+ if ( evt . defaultPrevented ) {
288
+ // Exit if canceled
289
+ return
290
+ }
291
+ evt . preventDefault ( )
292
+ evt . stopPropagation ( )
267
293
// Toggle visibility
268
294
this . visible = ! this . visible
269
295
} ,
0 commit comments