Skip to content

Commit 894d50d

Browse files
committed
feat(devtools): add devtools plugin
1 parent a821ec5 commit 894d50d

File tree

3 files changed

+282
-34
lines changed

3 files changed

+282
-34
lines changed

src/devtools.ts

+280-32
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,52 @@
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'
213
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
349

4-
export function addDevtools(app: App, router: Router) {
550
setupDevtoolsPlugin(
651
{
752
id: 'Router',
@@ -11,53 +56,256 @@ export function addDevtools(app: App, router: Router) {
1156
api => {
1257
api.on.inspectComponent((payload, ctx) => {
1358
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-
2259
payload.instanceData.state.push({
23-
type: stateType,
24-
key: 'time',
60+
type: 'Routing',
61+
key: '$route',
2562
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+
),
3567
})
3668
}
3769
})
3870

71+
watch(router.currentRoute, () => {
72+
// @ts-ignore
73+
api.notifyComponentUpdate()
74+
})
75+
76+
const navigationsLayerId = 'router:navigations'
77+
3978
api.addTimelineLayer({
40-
id: 'router:navigations',
79+
id: navigationsLayerId,
4180
label: 'Router Navigations',
42-
color: 0x92a2bf,
81+
color: 0x40a8c4,
4382
})
4483

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')
48115
api.addTimelineEvent({
49-
layerId: 'router:navigations',
116+
layerId: navigationsLayerId,
50117
event: {
51118
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,
56138
},
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: {},
58160
},
59161
})
60162
})
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+
})
61194
}
62195
)
63196
}
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+
}

src/matcher/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ import {
1818
import { warn } from '../warning'
1919
import { assign, noop } from '../utils'
2020

21-
interface RouterMatcher {
21+
export interface RouterMatcher {
2222
addRoute: (record: RouteRecordRaw, parent?: RouteRecordMatcher) => () => void
2323
removeRoute: {
2424
(matcher: RouteRecordMatcher): void

src/router.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1106,7 +1106,7 @@ export function createRouter(options: RouterOptions): Router {
11061106
}
11071107

11081108
if (__DEV__) {
1109-
addDevtools(app, router)
1109+
addDevtools(app, router, matcher)
11101110
}
11111111
},
11121112
}

0 commit comments

Comments
 (0)