Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
fix(useRouteHash,useRouteQuery,useRouteParams): re-evaluates the valu…
…e immediately (#3002)
- Loading branch information
1 parent
6b6701c
commit d525244
Showing
7 changed files
with
386 additions
and
40 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,65 @@ | ||
import { nextTick } from 'vue-demi' | ||
import { describe, expect, it } from 'vitest' | ||
import { useRouteHash } from '.' | ||
|
||
describe('useRouteHash', () => { | ||
const getRoute = (hash?: any) => ({ | ||
query: {}, | ||
fullPath: '', | ||
hash, | ||
matched: [], | ||
meta: {}, | ||
name: '', | ||
params: {}, | ||
path: '', | ||
redirectedFrom: undefined, | ||
}) | ||
|
||
it('should export', () => { | ||
expect(useRouteHash).toBeDefined() | ||
}) | ||
|
||
it('should return current value', () => { | ||
let route = getRoute('header') | ||
const router = { replace: (r: any) => route = r } as any | ||
|
||
const hash = useRouteHash(null, { route, router }) | ||
|
||
expect(hash.value).toBe(route.hash) | ||
}) | ||
|
||
it('should re-evaluate the value immediately', () => { | ||
let route = getRoute('header') | ||
const router = { replace: (r: any) => route = r } as any | ||
|
||
const hash = useRouteHash(null, { route, router }) | ||
|
||
hash.value = 'footer' | ||
|
||
expect(hash.value).toBe('footer') | ||
}) | ||
|
||
it('should update the route', async () => { | ||
let route = getRoute('foo') | ||
const router = { replace: (r: any) => route = r } as any | ||
|
||
const hash = useRouteHash(null, { route, router }) | ||
|
||
hash.value = 'footer' | ||
|
||
await nextTick() | ||
|
||
expect(hash.value).toBe('footer') | ||
expect(route.hash).toBe('footer') | ||
}) | ||
|
||
it('should return default value', () => { | ||
let route = getRoute() | ||
const router = { replace: (r: any) => route = r } as any | ||
|
||
const hash = useRouteHash('baz', { route, router }) | ||
|
||
expect(hash.value).toBe('baz') | ||
expect(route.hash).toBeUndefined() | ||
}) | ||
}) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,24 +1,38 @@ | ||
import { computed, nextTick } from 'vue-demi' | ||
import { customRef, nextTick } from 'vue-demi' | ||
import { useRoute, useRouter } from 'vue-router' | ||
import { toValue } from '@vueuse/shared' | ||
import type { ReactiveRouteOptions } from '../_types' | ||
import { toValue, tryOnScopeDispose } from '@vueuse/shared' | ||
import type { ReactiveRouteOptions, RouteHashValueRaw } from '../_types' | ||
|
||
let _hash: RouteHashValueRaw | ||
|
||
export function useRouteHash( | ||
defaultValue?: string, | ||
defaultValue?: RouteHashValueRaw, | ||
{ | ||
mode = 'replace', | ||
route = useRoute(), | ||
router = useRouter(), | ||
}: ReactiveRouteOptions = {}, | ||
) { | ||
return computed<string>({ | ||
_hash = route.hash | ||
|
||
tryOnScopeDispose(() => { | ||
_hash = undefined | ||
}) | ||
|
||
return customRef<RouteHashValueRaw>((track, trigger) => ({ | ||
get() { | ||
return route.hash ?? defaultValue | ||
track() | ||
|
||
return _hash || defaultValue | ||
}, | ||
set(v) { | ||
_hash = v === null ? undefined : v | ||
|
||
trigger() | ||
|
||
nextTick(() => { | ||
router[toValue(mode)]({ ...route, hash: v }) | ||
router[toValue(mode)]({ ...route, hash: _hash as string }) | ||
}) | ||
}, | ||
}) | ||
})) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,25 +1,146 @@ | ||
import { effectScope, nextTick, ref } from 'vue-demi' | ||
import { describe, expect, it } from 'vitest' | ||
import type { Ref } from 'vue-demi' | ||
import { useRouteParams } from '.' | ||
|
||
describe('useRouteQuery', () => { | ||
describe('useRouteParams', () => { | ||
const getRoute = (params: Record<string, any> = {}) => ({ | ||
params, | ||
query: {}, | ||
fullPath: '', | ||
hash: '', | ||
matched: [], | ||
meta: {}, | ||
name: '', | ||
params: { id: '1', ...params }, | ||
path: '', | ||
redirectedFrom: undefined, | ||
}) | ||
|
||
it('should return transformed value', () => { | ||
it('should export', () => { | ||
expect(useRouteParams).toBeDefined() | ||
}) | ||
|
||
it('should return current value', () => { | ||
const router = {} as any | ||
const route = getRoute({ | ||
id: '1', | ||
}) | ||
|
||
const id = useRouteParams('id', null, { route, router }) | ||
|
||
expect(id.value).toBe('1') | ||
}) | ||
|
||
it('should return transformed value', () => { | ||
const route = getRoute() | ||
const router = {} as any | ||
|
||
const id = useRouteParams('id', '1', { transform: Number, route, router }) | ||
|
||
expect(id.value).toBe(1) | ||
}) | ||
|
||
it('should re-evaluate the value immediately', () => { | ||
let route = getRoute() | ||
const router = { replace: (r: any) => route = r } as any | ||
|
||
const slug: Ref<any> = useRouteParams('slug', 'foo', { route, router }) | ||
const id: Ref<any> = useRouteParams('id', '123', { route, router }) | ||
const page: Ref<any> = useRouteParams('page', null, { route, router }) | ||
|
||
slug.value = 'bar' | ||
id.value = '456' | ||
page.value = '2' | ||
|
||
expect(slug.value).toBe('bar') | ||
expect(id.value).toBe('456') | ||
expect(page.value).toBe('2') | ||
}) | ||
|
||
it('should update the route', async () => { | ||
let route = getRoute() | ||
const router = { replace: (r: any) => route = r } as any | ||
|
||
const code: Ref<any> = useRouteParams('code', null, { route, router }) | ||
const page: Ref<any> = useRouteParams('page', null, { route, router }) | ||
const lang: Ref<any> = useRouteParams('lang', null, { route, router }) | ||
|
||
code.value = 'bar' | ||
page.value = '1' | ||
lang.value = 'en' | ||
|
||
await nextTick() | ||
|
||
expect(code.value).toBe('bar') | ||
expect(route.params.code).toBe('bar') | ||
expect(page.value).toBe('1') | ||
expect(route.params.page).toBe('1') | ||
expect(lang.value).toBe('en') | ||
expect(route.params.lang).toBe('en') | ||
}) | ||
|
||
it('should return default value', () => { | ||
let route = getRoute() | ||
const router = { replace: (r: any) => route = r } as any | ||
|
||
const page: Ref<any> = useRouteParams('page', 10, { route, router }) | ||
const lang: Ref<any> = useRouteParams('lang', 'pt-BR', { route, router }) | ||
|
||
expect(page.value).toBe(10) | ||
expect(lang.value).toBe('pt-BR') | ||
}) | ||
|
||
it('should reset state on scope dispose', async () => { | ||
let route = getRoute() | ||
const router = { replace: (r: any) => route = r } as any | ||
const scopeA = effectScope() | ||
const scopeB = effectScope() | ||
|
||
let page: Ref<any> = ref(null) | ||
let lang: Ref<any> = ref(null) | ||
let code: Ref<any> = ref(null) | ||
let slug: Ref<any> = ref(null) | ||
|
||
await scopeA.run(async () => { | ||
page = useRouteParams('page', null, { route, router }) | ||
lang = useRouteParams('lang', null, { route, router }) | ||
|
||
page.value = 2 | ||
lang.value = 'pt-BR' | ||
|
||
await nextTick() | ||
}) | ||
|
||
expect(page.value).toBe(2) | ||
expect(lang.value).toBe('pt-BR') | ||
expect(route.params.page).toBe(2) | ||
expect(route.params.lang).toBe('pt-BR') | ||
|
||
await scopeB.run(async () => { | ||
code = useRouteParams('code', null, { route, router }) | ||
slug = useRouteParams('slug', null, { route, router }) | ||
|
||
code.value = 'xyz' | ||
slug.value = 'vueuse' | ||
|
||
await nextTick() | ||
}) | ||
|
||
expect(code.value).toBe('xyz') | ||
expect(slug.value).toBe('vueuse') | ||
expect(route.params.code).toBe('xyz') | ||
expect(route.params.slug).toBe('vueuse') | ||
|
||
scopeB.stop() | ||
|
||
expect(page.value).toBe(2) | ||
expect(lang.value).toBe('pt-BR') | ||
expect(code.value).toBeNull() | ||
expect(slug.value).toBeNull() | ||
|
||
scopeA.stop() | ||
|
||
expect(page.value).toBeNull() | ||
expect(lang.value).toBeNull() | ||
}) | ||
}) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.