1
- import { App , setupDevtoolsPlugin } from '@vue/devtools-api'
1
+ import {
2
+ App ,
3
+ CustomInspectorNode ,
4
+ CustomInspectorNodeTag ,
5
+ CustomInspectorState ,
6
+ setupDevtoolsPlugin ,
7
+ TimelineEvent ,
8
+ } from '@vue/devtools-api'
9
+ import { watch } from 'vue'
10
+ import { RouterMatcher } from './matcher'
11
+ import { RouteRecordMatcher } from './matcher/pathMatcher'
12
+ import { PathParser } from './matcher/pathParserRanker'
2
13
import { Router } from './router'
14
+ import { RouteLocationNormalized } from './types'
15
+
16
+ function formatRouteLocation (
17
+ routeLocation : RouteLocationNormalized ,
18
+ tooltip ?: string
19
+ ) {
20
+ const copy = {
21
+ ...routeLocation ,
22
+ // remove variables that can contain vue instances
23
+ matched : routeLocation . matched . map (
24
+ ( { instances, children, aliasOf, ...rest } ) => rest
25
+ ) ,
26
+ }
27
+
28
+ return {
29
+ _custom : {
30
+ type : null ,
31
+ readOnly : true ,
32
+ display : routeLocation . fullPath ,
33
+ tooltip,
34
+ value : copy ,
35
+ } ,
36
+ }
37
+ }
38
+
39
+ function formatDisplay ( display : string ) {
40
+ return {
41
+ _custom : {
42
+ display,
43
+ } ,
44
+ }
45
+ }
46
+
47
+ export function addDevtools ( app : App , router : Router , matcher : RouterMatcher ) {
48
+ // Take over router.beforeEach and afterEach
3
49
4
- export function addDevtools ( app : App , router : Router ) {
5
50
setupDevtoolsPlugin (
6
51
{
7
52
id : 'Router' ,
@@ -11,53 +56,256 @@ export function addDevtools(app: App, router: Router) {
11
56
api => {
12
57
api . on . inspectComponent ( ( payload , ctx ) => {
13
58
if ( payload . instanceData ) {
14
- const stateType = 'extra properties (test)'
15
- payload . instanceData . state . push ( {
16
- type : stateType ,
17
- key : 'foo' ,
18
- value : 'bar' ,
19
- editable : false ,
20
- } )
21
-
22
59
payload . instanceData . state . push ( {
23
- type : stateType ,
24
- key : 'time ' ,
60
+ type : 'Routing' ,
61
+ key : '$route ' ,
25
62
editable : false ,
26
- value : {
27
- _custom : {
28
- type : null ,
29
- readOnly : true ,
30
- display : `${ router . currentRoute . value . fullPath } s` ,
31
- tooltip : 'Current Route' ,
32
- value : router . currentRoute . value ,
33
- } ,
34
- } ,
63
+ value : formatRouteLocation (
64
+ router . currentRoute . value ,
65
+ 'Current Route'
66
+ ) ,
35
67
} )
36
68
}
37
69
} )
38
70
71
+ watch ( router . currentRoute , ( ) => {
72
+ // @ts -ignore
73
+ api . notifyComponentUpdate ( )
74
+ } )
75
+
76
+ const navigationsLayerId = 'router:navigations'
77
+
39
78
api . addTimelineLayer ( {
40
- id : 'router:navigations' ,
79
+ id : navigationsLayerId ,
41
80
label : 'Router Navigations' ,
42
- color : 0x92a2bf ,
81
+ color : 0x40a8c4 ,
43
82
} )
44
83
45
- router . afterEach ( ( from , to ) => {
46
- // @ts -ignore
47
- api . notifyComponentUpdate ( )
84
+ // const errorsLayerId = 'router:errors'
85
+ // api.addTimelineLayer({
86
+ // id: errorsLayerId,
87
+ // label: 'Router Errors',
88
+ // color: 0xea5455,
89
+ // })
90
+
91
+ router . onError ( error => {
92
+ api . addTimelineEvent ( {
93
+ layerId : navigationsLayerId ,
94
+ event : {
95
+ // @ts -ignore
96
+ logType : 'error' ,
97
+ time : Date . now ( ) ,
98
+ data : { error } ,
99
+ } ,
100
+ } )
101
+ } )
102
+
103
+ console . log ( 'adding devtools to timeline' )
104
+ router . beforeEach ( ( to , from ) => {
105
+ const data : TimelineEvent < any , any > [ 'data' ] = {
106
+ guard : formatDisplay ( 'beforEach' ) ,
107
+ from : formatRouteLocation (
108
+ from ,
109
+ 'Current Location during this navigation'
110
+ ) ,
111
+ to : formatRouteLocation ( to , 'Target location' ) ,
112
+ }
113
+
114
+ console . log ( 'adding to timeline' )
48
115
api . addTimelineEvent ( {
49
- layerId : 'router:navigations' ,
116
+ layerId : navigationsLayerId ,
50
117
event : {
51
118
time : Date . now ( ) ,
52
- data : {
53
- info : 'afterEach' ,
54
- from,
55
- to,
119
+ meta : { } ,
120
+ data,
121
+ } ,
122
+ } )
123
+ } )
124
+
125
+ router . afterEach ( ( to , from , failure ) => {
126
+ const data : TimelineEvent < any , any > [ 'data' ] = {
127
+ guard : formatDisplay ( 'afterEach' ) ,
128
+ }
129
+
130
+ if ( failure ) {
131
+ data . failure = {
132
+ _custom : {
133
+ type : Error ,
134
+ readOnly : true ,
135
+ display : failure ? failure . message : '' ,
136
+ tooltip : 'Navigation Failure' ,
137
+ value : failure ,
56
138
} ,
57
- meta : { foo : 'meta?' } ,
139
+ }
140
+ data . status = formatDisplay ( '❌' )
141
+ } else {
142
+ data . status = formatDisplay ( '✅' )
143
+ }
144
+
145
+ // we set here to have the right order
146
+ data . from = formatRouteLocation (
147
+ from ,
148
+ 'Current Location during this navigation'
149
+ )
150
+ data . to = formatRouteLocation ( to , 'Target location' )
151
+
152
+ api . addTimelineEvent ( {
153
+ layerId : navigationsLayerId ,
154
+ event : {
155
+ time : Date . now ( ) ,
156
+ data,
157
+ // @ts -ignore
158
+ logType : failure ? 'warning' : 'default' ,
159
+ meta : { } ,
58
160
} ,
59
161
} )
60
162
} )
163
+
164
+ const routerInspectorId = 'hahaha router-inspector'
165
+
166
+ api . addInspector ( {
167
+ id : routerInspectorId ,
168
+ label : 'Routes' ,
169
+ icon : 'book' ,
170
+ treeFilterPlaceholder : 'Filter routes' ,
171
+ } )
172
+
173
+ api . on . getInspectorTree ( payload => {
174
+ if ( payload . app === app && payload . inspectorId === routerInspectorId ) {
175
+ const routes = matcher . getRoutes ( ) . filter ( route => ! route . parent )
176
+ payload . rootNodes = routes . map ( formatRouteRecordForInspector )
177
+ }
178
+ } )
179
+
180
+ api . on . getInspectorState ( payload => {
181
+ if ( payload . app === app && payload . inspectorId === routerInspectorId ) {
182
+ const routes = matcher . getRoutes ( )
183
+ const route = routes . find (
184
+ route => route . record . path === payload . nodeId
185
+ )
186
+
187
+ if ( route ) {
188
+ payload . state = {
189
+ options : formatRouteRecordMatcherForStateInspector ( route ) ,
190
+ }
191
+ }
192
+ }
193
+ } )
61
194
}
62
195
)
63
196
}
197
+
198
+ function modifierForKey ( key : PathParser [ 'keys' ] [ number ] ) {
199
+ if ( key . optional ) {
200
+ return key . repeatable ? '*' : '?'
201
+ } else {
202
+ return key . repeatable ? '+' : ''
203
+ }
204
+ }
205
+
206
+ function formatRouteRecordMatcherForStateInspector (
207
+ route : RouteRecordMatcher
208
+ ) : CustomInspectorState [ string ] {
209
+ const { record } = route
210
+ const fields : CustomInspectorState [ string ] = [
211
+ { editable : false , key : 'path' , value : record . path } ,
212
+ ]
213
+
214
+ if ( record . name != null )
215
+ fields . push ( {
216
+ editable : false ,
217
+ key : 'name' ,
218
+ value : record . name ,
219
+ } )
220
+
221
+ fields . push ( { editable : false , key : 'regexp' , value : route . re } )
222
+
223
+ if ( route . keys . length )
224
+ fields . push ( {
225
+ editable : false ,
226
+ key : 'keys' ,
227
+ value : {
228
+ _custom : {
229
+ type : null ,
230
+ readOnly : true ,
231
+ display : route . keys
232
+ . map ( key => `${ key . name } ${ modifierForKey ( key ) } ` )
233
+ . join ( ' ' ) ,
234
+ tooltip : 'Param keys' ,
235
+ value : route . keys ,
236
+ } ,
237
+ } ,
238
+ } )
239
+
240
+ if ( record . redirect != null )
241
+ fields . push ( {
242
+ editable : false ,
243
+ key : 'redirect' ,
244
+ value : record . redirect ,
245
+ } )
246
+
247
+ if ( route . alias . length )
248
+ fields . push ( {
249
+ editable : false ,
250
+ key : 'aliases' ,
251
+ value : route . alias ,
252
+ } )
253
+
254
+ fields . push ( {
255
+ key : 'score' ,
256
+ editable : false ,
257
+ value : {
258
+ _custom : {
259
+ type : null ,
260
+ readOnly : true ,
261
+ display : route . score . map ( score => score . join ( ', ' ) ) . join ( ' | ' ) ,
262
+ tooltip : 'Score used to sort routes' ,
263
+ value : route . score ,
264
+ } ,
265
+ } ,
266
+ } )
267
+
268
+ return fields
269
+ }
270
+
271
+ function formatRouteRecordForInspector (
272
+ route : RouteRecordMatcher
273
+ ) : CustomInspectorNode {
274
+ const tags : CustomInspectorNodeTag [ ] = [ ]
275
+
276
+ const { record } = route
277
+
278
+ if ( record . name != null ) {
279
+ tags . push ( {
280
+ label : String ( record . name ) ,
281
+ textColor : 0 ,
282
+ backgroundColor : 0x00bcd4 ,
283
+ } )
284
+ }
285
+
286
+ if ( record . aliasOf ) {
287
+ tags . push ( {
288
+ label : 'alias' ,
289
+ textColor : 0 ,
290
+ backgroundColor : 0xff984f ,
291
+ } )
292
+ }
293
+
294
+ if ( record . redirect ) {
295
+ tags . push ( {
296
+ label :
297
+ 'redirect: ' +
298
+ ( typeof record . redirect === 'string' ? record . redirect : 'Object' ) ,
299
+ textColor : 0xffffff ,
300
+ backgroundColor : 0x666666 ,
301
+ } )
302
+ }
303
+
304
+ return {
305
+ id : record . path ,
306
+ label : record . path ,
307
+ tags,
308
+ // @ts -ignore
309
+ children : route . children . map ( formatRouteRecordForInspector ) ,
310
+ }
311
+ }
0 commit comments