Skip to content

Commit 4b813b1

Browse files
committed
feat(link): make empty child active with adjacent children
1 parent 67a1426 commit 4b813b1

File tree

4 files changed

+92
-14
lines changed

4 files changed

+92
-14
lines changed

__tests__/RouterLink.spec.ts

+50-8
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ const records = {
1919
homeAlias: {} as RouteRecordNormalized,
2020
foo: {} as RouteRecordNormalized,
2121
parent: {} as RouteRecordNormalized,
22+
childEmpty: {} as RouteRecordNormalized,
2223
child: {} as RouteRecordNormalized,
2324
parentAlias: {} as RouteRecordNormalized,
2425
childAlias: {} as RouteRecordNormalized,
@@ -33,14 +34,20 @@ records.childAlias = { aliasOf: records.child } as RouteRecordNormalized
3334

3435
type RouteLocationResolved = RouteLocationNormalized & { href: string }
3536

36-
const locations: Record<
37-
string,
38-
{
39-
string: string
40-
normalized: RouteLocationResolved
41-
toResolve?: MatcherLocationRaw & Required<RouteQueryAndHash>
42-
}
43-
> = {
37+
function createLocations<
38+
T extends Record<
39+
string,
40+
{
41+
string: string
42+
normalized: RouteLocationResolved
43+
toResolve?: MatcherLocationRaw & Required<RouteQueryAndHash>
44+
}
45+
>
46+
>(locs: T) {
47+
return locs
48+
}
49+
50+
const locations = createLocations({
4451
basic: {
4552
string: '/home',
4653
// toResolve: { path: '/home', fullPath: '/home', undefined, query: {}, hash: '' },
@@ -167,6 +174,21 @@ const locations: Record<
167174
},
168175
},
169176

177+
childEmpty: {
178+
string: '/parent',
179+
normalized: {
180+
fullPath: '/parent',
181+
href: '/parent',
182+
path: '/parent',
183+
params: {},
184+
meta: {},
185+
query: {},
186+
hash: '',
187+
matched: [records.parent, records.childEmpty],
188+
redirectedFrom: undefined,
189+
name: undefined,
190+
},
191+
},
170192
child: {
171193
string: '/parent/child',
172194
normalized: {
@@ -257,6 +279,14 @@ const locations: Record<
257279
name: undefined,
258280
},
259281
},
282+
})
283+
284+
// add paths to records because they are used to check isActive
285+
for (let record in records) {
286+
let location = locations[record as keyof typeof locations]
287+
if (location) {
288+
records[record as keyof typeof records].path = location.normalized.path
289+
}
260290
}
261291

262292
async function factory(
@@ -461,6 +491,18 @@ describe('RouterLink', () => {
461491
)
462492
})
463493

494+
it('empty path child is active as if it was the parent when on adjacent child', async () => {
495+
const { wrapper } = await factory(
496+
locations.child.normalized,
497+
{ to: locations.childEmpty.string },
498+
locations.childEmpty.normalized
499+
)
500+
expect(wrapper.find('a')!.className).toContain('router-link-active')
501+
expect(wrapper.find('a')!.className).not.toContain(
502+
'router-link-exact-active'
503+
)
504+
})
505+
464506
it('alias parent is active if the child is an absolute path', async () => {
465507
const { wrapper } = await factory(
466508
locations.childAsAbsolute.normalized,

playground/App.vue

+19
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,25 @@
6969
<li>
7070
<router-link to="/always-redirect">/always-redirect</router-link>
7171
</li>
72+
<li>
73+
<router-link to="/children">/children</router-link>
74+
</li>
75+
<li>
76+
<router-link :to="{ name: 'default-child' }"
77+
>/children (child named)</router-link
78+
>
79+
</li>
80+
<li>
81+
<router-link :to="{ name: 'WithChildren' }"
82+
>/children (parent named)</router-link
83+
>
84+
</li>
85+
<li>
86+
<router-link to="/children/a">/children/a</router-link>
87+
</li>
88+
<li>
89+
<router-link to="/children/b">/children/b</router-link>
90+
</li>
7291
<li>
7392
<router-link to="/nested">/nested</router-link>
7493
</li>

playground/router.ts

+1
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,7 @@ export const router = createRouter({
5959
{ path: '/cant-leave', component: GuardedWithLeave },
6060
{
6161
path: '/children',
62+
name: 'WithChildren',
6263
component,
6364
children: [
6465
{ path: '', name: 'default-child', component },

src/RouterLink.ts

+22-6
Original file line numberDiff line numberDiff line change
@@ -30,13 +30,29 @@ export function useLink(props: UseLinkOptions) {
3030
const route = computed(() => router.resolve(unref(props.to)))
3131

3232
const activeRecordIndex = computed<number>(() => {
33-
// TODO: handle children with empty path: they should relate to their parent
34-
const currentMatched: RouteRecord | undefined =
35-
route.value.matched[route.value.matched.length - 1]
36-
if (!currentMatched) return -1
37-
return currentRoute.matched.findIndex(
38-
isSameRouteRecord.bind(null, currentMatched)
33+
let { matched } = route.value
34+
let { length } = matched
35+
const routeMatched: RouteRecord | undefined = matched[length - 1]
36+
let currentMatched = currentRoute.matched
37+
if (!routeMatched || !currentMatched.length) return -1
38+
let index = currentMatched.findIndex(
39+
isSameRouteRecord.bind(null, routeMatched)
3940
)
41+
if (index > -1) return index
42+
// possible parent record
43+
let parentRecord = matched[length - 2]
44+
if (
45+
length > 1 &&
46+
// if the have the same path, this link is referring to the empty child
47+
// are we currently are on a different child of the same parent
48+
routeMatched.path === parentRecord.path &&
49+
// avoid comparing the child with its parent
50+
currentMatched[currentMatched.length - 1].path !== parentRecord.path
51+
)
52+
return currentMatched.findIndex(
53+
isSameRouteRecord.bind(null, matched[length - 2])
54+
)
55+
return index
4056
})
4157

4258
const isActive = computed<boolean>(

0 commit comments

Comments
 (0)