@@ -3,11 +3,13 @@ import {
3
3
CustomInspectorNode ,
4
4
CustomInspectorNodeTag ,
5
5
CustomInspectorState ,
6
+ HookPayloads ,
6
7
setupDevtoolsPlugin ,
7
8
TimelineEvent ,
8
9
} from '@vue/devtools-api'
9
10
import { watch } from 'vue'
10
11
import { decode } from './encoding'
12
+ import { isSameRouteRecord } from './location'
11
13
import { RouterMatcher } from './matcher'
12
14
import { RouteRecordMatcher } from './matcher/pathMatcher'
13
15
import { PathParser } from './matcher/pathParserRanker'
@@ -75,8 +77,11 @@ export function addDevtools(app: App, router: Router, matcher: RouterMatcher) {
75
77
} )
76
78
77
79
watch ( router . currentRoute , ( ) => {
80
+ // refresh active state
81
+ refreshRoutesView ( )
78
82
// @ts -ignore
79
83
api . notifyComponentUpdate ( )
84
+ api . sendInspectorTree ( routerInspectorId )
80
85
} )
81
86
82
87
const navigationsLayerId = 'router:navigations:' + id
@@ -165,6 +170,10 @@ export function addDevtools(app: App, router: Router, matcher: RouterMatcher) {
165
170
} )
166
171
} )
167
172
173
+ /**
174
+ * Inspector of Existing routes
175
+ */
176
+
168
177
const routerInspectorId = 'router-inspector:' + id
169
178
170
179
api . addInspector ( {
@@ -174,32 +183,48 @@ export function addDevtools(app: App, router: Router, matcher: RouterMatcher) {
174
183
treeFilterPlaceholder : 'Search routes' ,
175
184
} )
176
185
186
+ function refreshRoutesView ( ) {
187
+ // the routes view isn't active
188
+ if ( ! activeRoutesPayload ) return
189
+ const payload = activeRoutesPayload
190
+
191
+ // children routes will appear as nested
192
+ let routes = matcher . getRoutes ( ) . filter ( route => ! route . parent )
193
+
194
+ // reset match state to false
195
+ routes . forEach ( resetMatchStateOnRouteRecord )
196
+
197
+ // apply a match state if there is a payload
198
+ if ( payload . filter ) {
199
+ routes = routes . filter ( route =>
200
+ // save matches state based on the payload
201
+ isRouteMatching ( route , payload . filter . toLowerCase ( ) )
202
+ )
203
+ }
204
+
205
+ // mark active routes
206
+ routes . forEach ( route =>
207
+ markRouteRecordActive ( route , router . currentRoute . value )
208
+ )
209
+ payload . rootNodes = routes . map ( formatRouteRecordForInspector )
210
+ }
211
+
212
+ let activeRoutesPayload : HookPayloads [ 'getInspectorTree' ] | undefined
177
213
api . on . getInspectorTree ( payload => {
214
+ activeRoutesPayload = payload
178
215
if ( payload . app === app && payload . inspectorId === routerInspectorId ) {
179
- let routes = matcher . getRoutes ( )
180
- if ( payload . filter ) {
181
- routes = routes . filter (
182
- route =>
183
- ! route . parent &&
184
- // save isActive state
185
- isRouteMatching ( route , payload . filter . toLowerCase ( ) )
186
- )
187
- }
188
- // reset match state if no filter is provided
189
- if ( ! payload . filter ) {
190
- routes . forEach ( route => {
191
- ; ( route as any ) . __vd_match = false
192
- } )
193
- }
194
- payload . rootNodes = routes . map ( formatRouteRecordForInspector )
216
+ refreshRoutesView ( )
195
217
}
196
218
} )
197
219
220
+ /**
221
+ * Display information about the currently selected route record
222
+ */
198
223
api . on . getInspectorState ( payload => {
199
224
if ( payload . app === app && payload . inspectorId === routerInspectorId ) {
200
225
const routes = matcher . getRoutes ( )
201
226
const route = routes . find (
202
- route => route . record . path === payload . nodeId
227
+ route => ( route . record as any ) . __vd_id === payload . nodeId
203
228
)
204
229
205
230
if ( route ) {
@@ -209,6 +234,9 @@ export function addDevtools(app: App, router: Router, matcher: RouterMatcher) {
209
234
}
210
235
}
211
236
} )
237
+
238
+ api . sendInspectorTree ( routerInspectorId )
239
+ api . sendInspectorState ( routerInspectorId )
212
240
}
213
241
)
214
242
}
@@ -229,16 +257,17 @@ function formatRouteRecordMatcherForStateInspector(
229
257
{ editable : false , key : 'path' , value : record . path } ,
230
258
]
231
259
232
- if ( record . name != null )
260
+ if ( record . name != null ) {
233
261
fields . push ( {
234
262
editable : false ,
235
263
key : 'name' ,
236
264
value : record . name ,
237
265
} )
266
+ }
238
267
239
268
fields . push ( { editable : false , key : 'regexp' , value : route . re } )
240
269
241
- if ( route . keys . length )
270
+ if ( route . keys . length ) {
242
271
fields . push ( {
243
272
editable : false ,
244
273
key : 'keys' ,
@@ -254,20 +283,23 @@ function formatRouteRecordMatcherForStateInspector(
254
283
} ,
255
284
} ,
256
285
} )
286
+ }
257
287
258
- if ( record . redirect != null )
288
+ if ( record . redirect != null ) {
259
289
fields . push ( {
260
290
editable : false ,
261
291
key : 'redirect' ,
262
292
value : record . redirect ,
263
293
} )
294
+ }
264
295
265
- if ( route . alias . length )
296
+ if ( route . alias . length ) {
266
297
fields . push ( {
267
298
editable : false ,
268
299
key : 'aliases' ,
269
300
value : route . alias . map ( alias => alias . record . path ) ,
270
301
} )
302
+ }
271
303
272
304
fields . push ( {
273
305
key : 'score' ,
@@ -286,6 +318,17 @@ function formatRouteRecordMatcherForStateInspector(
286
318
return fields
287
319
}
288
320
321
+ /**
322
+ * Extracted from tailwind palette
323
+ */
324
+ const PINK_500 = 0xec4899
325
+ const BLUE_600 = 0x2563eb
326
+ const LIME_500 = 0x84cc16
327
+ const CYAN_400 = 0x22d3ee
328
+ const ORANGE_400 = 0xfb923c
329
+ // const GRAY_100 = 0xf4f4f5
330
+ const DARK = 0x666666
331
+
289
332
function formatRouteRecordForInspector (
290
333
route : RouteRecordMatcher
291
334
) : CustomInspectorNode {
@@ -297,23 +340,39 @@ function formatRouteRecordForInspector(
297
340
tags . push ( {
298
341
label : String ( record . name ) ,
299
342
textColor : 0 ,
300
- backgroundColor : 0x00bcd4 ,
343
+ backgroundColor : CYAN_400 ,
301
344
} )
302
345
}
303
346
304
347
if ( record . aliasOf ) {
305
348
tags . push ( {
306
349
label : 'alias' ,
307
350
textColor : 0 ,
308
- backgroundColor : 0xff984f ,
351
+ backgroundColor : ORANGE_400 ,
309
352
} )
310
353
}
311
354
312
355
if ( ( route as any ) . __vd_match ) {
313
356
tags . push ( {
314
357
label : 'matches' ,
315
358
textColor : 0 ,
316
- backgroundColor : 0xf4f4f4 ,
359
+ backgroundColor : PINK_500 ,
360
+ } )
361
+ }
362
+
363
+ if ( ( route as any ) . __vd_exactActive ) {
364
+ tags . push ( {
365
+ label : 'exact' ,
366
+ textColor : 0 ,
367
+ backgroundColor : LIME_500 ,
368
+ } )
369
+ }
370
+
371
+ if ( ( route as any ) . __vd_active ) {
372
+ tags . push ( {
373
+ label : 'active' ,
374
+ textColor : 0 ,
375
+ backgroundColor : BLUE_600 ,
317
376
} )
318
377
}
319
378
@@ -323,32 +382,72 @@ function formatRouteRecordForInspector(
323
382
'redirect: ' +
324
383
( typeof record . redirect === 'string' ? record . redirect : 'Object' ) ,
325
384
textColor : 0xffffff ,
326
- backgroundColor : 0x666666 ,
385
+ backgroundColor : DARK ,
327
386
} )
328
387
}
329
388
389
+ // add an id to be able to select it. Using the `path` is not possible because
390
+ // empty path children would collide with their parents
391
+ let id = String ( routeRecordId ++ )
392
+ ; ( record as any ) . __vd_id = id
393
+
330
394
return {
331
- id : record . path ,
395
+ id,
332
396
label : record . path ,
333
397
tags,
334
398
// @ts -ignore
335
399
children : route . children . map ( formatRouteRecordForInspector ) ,
336
400
}
337
401
}
338
402
403
+ // incremental id for route records and inspector state
404
+ let routeRecordId = 0
405
+
339
406
const EXTRACT_REGEXP_RE = / ^ \/ ( .* ) \/ ( [ a - z ] * ) $ /
340
407
408
+ function markRouteRecordActive (
409
+ route : RouteRecordMatcher ,
410
+ currentRoute : RouteLocationNormalized
411
+ ) {
412
+ // no route will be active if matched is empty
413
+ // reset the matching state
414
+ const isExactActive =
415
+ currentRoute . matched . length &&
416
+ isSameRouteRecord (
417
+ currentRoute . matched [ currentRoute . matched . length - 1 ] ,
418
+ route . record
419
+ )
420
+ ; ( route as any ) . __vd_exactActive = ( route as any ) . __vd_active = isExactActive
421
+
422
+ if ( ! isExactActive ) {
423
+ ; ( route as any ) . __vd_active = currentRoute . matched . some ( match =>
424
+ isSameRouteRecord ( match , route . record )
425
+ )
426
+ }
427
+
428
+ route . children . forEach ( childRoute =>
429
+ markRouteRecordActive ( childRoute , currentRoute )
430
+ )
431
+ }
432
+
433
+ function resetMatchStateOnRouteRecord ( route : RouteRecordMatcher ) {
434
+ ; ( route as any ) . __vd_match = false
435
+ route . children . forEach ( resetMatchStateOnRouteRecord )
436
+ }
437
+
341
438
function isRouteMatching ( route : RouteRecordMatcher , filter : string ) : boolean {
342
439
const found = String ( route . re ) . match ( EXTRACT_REGEXP_RE )
343
440
// reset the matching state
344
441
; ( route as any ) . __vd_match = false
345
- if ( ! found || found . length < 3 ) return false
442
+ if ( ! found || found . length < 3 ) {
443
+ return false
444
+ }
346
445
347
446
// use a regexp without $ at the end to match nested routes better
348
447
const nonEndingRE = new RegExp ( found [ 1 ] . replace ( / \$ $ / , '' ) , found [ 2 ] )
349
448
if ( nonEndingRE . test ( filter ) ) {
350
449
// mark children as matches
351
- route . children . some ( child => isRouteMatching ( child , filter ) )
450
+ route . children . forEach ( child => isRouteMatching ( child , filter ) )
352
451
// exception case: `/`
353
452
if ( route . record . path !== '/' || filter === '/' ) {
354
453
; ( route as any ) . __vd_match = route . re . test ( filter )
0 commit comments