Skip to content

Commit d3b5dfb

Browse files
committed
feat(devtools): group navigations
1 parent 9dd9592 commit d3b5dfb

File tree

2 files changed

+69
-8
lines changed

2 files changed

+69
-8
lines changed

src/RouterLink.ts

+15
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,8 @@ import {
99
VNodeProps,
1010
AllowedComponentProps,
1111
ComponentCustomProps,
12+
getCurrentInstance,
13+
watchEffect,
1214
} from 'vue'
1315
import { RouteLocationRaw, VueUseOptions, RouteLocation } from './types'
1416
import { isSameRouteLocationParams, isSameRouteRecord } from './location'
@@ -165,6 +167,19 @@ export const RouterLinkImpl = /*#__PURE__*/ defineComponent({
165167
)]: link.isExactActive,
166168
}))
167169

170+
if ((__DEV__ || __FEATURE_PROD_DEVTOOLS__) && __BROWSER__) {
171+
const instance = getCurrentInstance()
172+
watchEffect(() => {
173+
if (!instance) return
174+
;(instance as any).__vrl_route = link.route
175+
})
176+
watchEffect(() => {
177+
if (!instance) return
178+
;(instance as any).__vrl_active = link.isActive
179+
;(instance as any).__vrl_exactActive = link.isExactActive
180+
})
181+
}
182+
168183
return () => {
169184
const children = slots.default && slots.default(link)
170185
return props.custom

src/devtools.ts

+54-8
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ import { RouterMatcher } from './matcher'
1414
import { RouteRecordMatcher } from './matcher/pathMatcher'
1515
import { PathParser } from './matcher/pathParserRanker'
1616
import { Router } from './router'
17-
import { RouteLocationNormalized } from './types'
17+
import { RouteLocation, RouteLocationNormalized } from './types'
1818
import { assign } from './utils'
1919

2020
function formatRouteLocation(
@@ -53,15 +53,24 @@ let routerId = 0
5353
export function addDevtools(app: App, router: Router, matcher: RouterMatcher) {
5454
// Take over router.beforeEach and afterEach
5555

56+
// make sure we are not registering the devtool twice
57+
if ((router as any).__hasDevtools) return
58+
;(router as any).__hasDevtools = true
59+
5660
// increment to support multiple router instances
5761
const id = routerId++
5862
setupDevtoolsPlugin(
5963
{
60-
id: 'Router' + (id ? ' ' + id : ''),
64+
id: 'org.vuejs.router' + (id ? '.' + id : ''),
6165
label: 'Vue Router',
66+
packageName: 'vue-router',
67+
homepage: 'https://next.router.vuejs.org/',
68+
logo: 'https://vuejs.org/images/icons/favicon-96x96.png',
69+
componentStateTypes: ['Routing'],
6270
app,
6371
},
6472
api => {
73+
// display state added by the router
6574
api.on.inspectComponent((payload, ctx) => {
6675
if (payload.instanceData) {
6776
payload.instanceData.state.push({
@@ -76,10 +85,38 @@ export function addDevtools(app: App, router: Router, matcher: RouterMatcher) {
7685
}
7786
})
7887

88+
// mark router-link as active
89+
api.on.visitComponentTree(({ treeNode: node, componentInstance }) => {
90+
if (node.name === 'RouterLink') {
91+
if (componentInstance.__vrl_route) {
92+
node.tags.push({
93+
label: (componentInstance.__vrl_route as RouteLocation).path,
94+
textColor: 0,
95+
backgroundColor: ORANGE_400,
96+
})
97+
}
98+
99+
if (componentInstance.__vrl_exactActive) {
100+
node.tags.push({
101+
label: 'exact',
102+
textColor: 0,
103+
backgroundColor: LIME_500,
104+
})
105+
}
106+
107+
if (componentInstance.__vrl_active) {
108+
node.tags.push({
109+
label: 'active',
110+
textColor: 0,
111+
backgroundColor: BLUE_600,
112+
})
113+
}
114+
}
115+
})
116+
79117
watch(router.currentRoute, () => {
80118
// refresh active state
81119
refreshRoutesView()
82-
// @ts-ignore
83120
api.notifyComponentUpdate()
84121
api.sendInspectorTree(routerInspectorId)
85122
})
@@ -103,14 +140,18 @@ export function addDevtools(app: App, router: Router, matcher: RouterMatcher) {
103140
api.addTimelineEvent({
104141
layerId: navigationsLayerId,
105142
event: {
106-
// @ts-ignore
143+
title: 'Error',
144+
subtitle: 'An uncaught error happened during navigation',
107145
logType: 'error',
108146
time: Date.now(),
109147
data: { error },
110148
},
111149
})
112150
})
113151

152+
// attached to `meta` and used to group events
153+
let navigationId = 0
154+
114155
router.beforeEach((to, from) => {
115156
const data: TimelineEvent<any, any>['data'] = {
116157
guard: formatDisplay('beforeEach'),
@@ -121,12 +162,18 @@ export function addDevtools(app: App, router: Router, matcher: RouterMatcher) {
121162
to: formatRouteLocation(to, 'Target location'),
122163
}
123164

165+
// Used to group navigations together, hide from devtools
166+
Object.defineProperty(to.meta, '__navigationId', {
167+
value: navigationId++,
168+
})
169+
124170
api.addTimelineEvent({
125171
layerId: navigationsLayerId,
126172
event: {
127173
time: Date.now(),
128-
meta: {},
174+
title: 'Start of navigation',
129175
data,
176+
groupId: (to.meta as any).__navigationId,
130177
},
131178
})
132179
})
@@ -161,11 +208,11 @@ export function addDevtools(app: App, router: Router, matcher: RouterMatcher) {
161208
api.addTimelineEvent({
162209
layerId: navigationsLayerId,
163210
event: {
211+
title: 'End of navigation',
164212
time: Date.now(),
165213
data,
166-
// @ts-ignore
167214
logType: failure ? 'warning' : 'default',
168-
meta: {},
215+
groupId: (to.meta as any).__navigationId,
169216
},
170217
})
171218
})
@@ -395,7 +442,6 @@ function formatRouteRecordForInspector(
395442
id,
396443
label: record.path,
397444
tags,
398-
// @ts-ignore
399445
children: route.children.map(formatRouteRecordForInspector),
400446
}
401447
}

0 commit comments

Comments
 (0)