@@ -29,8 +29,13 @@ import {
29
29
} from './utils'
30
30
import { useCallbacks } from './utils/callbacks'
31
31
import { encodeParam , decode } from './utils/encoding'
32
- import { normalizeQuery , parseQuery , stringifyQuery } from './utils/query'
33
- import { ref , Ref , markNonReactive , nextTick , App } from 'vue'
32
+ import {
33
+ normalizeQuery ,
34
+ parseQuery ,
35
+ stringifyQuery ,
36
+ LocationQueryValue ,
37
+ } from './utils/query'
38
+ import { ref , Ref , markNonReactive , nextTick , App , warn } from 'vue'
34
39
import { RouteRecordNormalized } from './matcher/types'
35
40
import { Link } from './components/Link'
36
41
import { View } from './components/View'
@@ -58,6 +63,11 @@ export interface Router {
58
63
history : RouterHistory
59
64
currentRoute : Ref < Immutable < RouteLocationNormalized > >
60
65
66
+ addRoute ( parentName : string , route : RouteRecord ) : ( ) => void
67
+ addRoute ( route : RouteRecord ) : ( ) => void
68
+ removeRoute ( name : string ) : void
69
+ getRoutes ( ) : RouteRecordNormalized [ ]
70
+
61
71
resolve ( to : RouteLocation ) : RouteLocationNormalized
62
72
createHref ( to : RouteLocationNormalized ) : string
63
73
push ( to : RouteLocation ) : Promise < RouteLocationNormalized >
@@ -100,6 +110,33 @@ export function createRouter({
100
110
const encodeParams = applyToParams . bind ( null , encodeParam )
101
111
const decodeParams = applyToParams . bind ( null , decode )
102
112
113
+ function addRoute ( parentOrRoute : string | RouteRecord , route ?: RouteRecord ) {
114
+ let parent : Parameters < typeof matcher [ 'addRoute' ] > [ 1 ] | undefined
115
+ let record : RouteRecord
116
+ if ( typeof parentOrRoute === 'string' ) {
117
+ parent = matcher . getRecordMatcher ( parentOrRoute )
118
+ record = route !
119
+ } else {
120
+ record = parentOrRoute
121
+ }
122
+
123
+ return matcher . addRoute ( record , parent )
124
+ }
125
+
126
+ function removeRoute ( name : string ) {
127
+ let recordMatcher = matcher . getRecordMatcher ( name )
128
+ if ( recordMatcher ) {
129
+ matcher . removeRoute ( recordMatcher )
130
+ } else if ( __DEV__ ) {
131
+ // TODO: adapt if we allow Symbol as a name
132
+ warn ( `Cannot remove non-existant route "${ name } "` )
133
+ }
134
+ }
135
+
136
+ function getRoutes ( ) : RouteRecordNormalized [ ] {
137
+ return matcher . getRoutes ( ) . map ( routeMatcher => routeMatcher . record )
138
+ }
139
+
103
140
function resolve (
104
141
location : RouteLocation ,
105
142
currentLocation ?: RouteLocationNormalized
@@ -161,14 +198,12 @@ export function createRouter({
161
198
) : Promise < RouteLocationNormalized > {
162
199
const toLocation : RouteLocationNormalized = ( pendingLocation = resolve ( to ) )
163
200
const from : RouteLocationNormalized = currentRoute . value
201
+ // @ts -ignore: no need to check the string as force do not exist on a string
202
+ const force : boolean | undefined = to . force
164
203
165
204
// TODO: should we throw an error as the navigation was aborted
166
205
// TODO: needs a proper check because order in query could be different
167
- if (
168
- from !== START_LOCATION_NORMALIZED &&
169
- from . fullPath === toLocation . fullPath
170
- )
171
- return from
206
+ if ( ! force && isSameLocation ( from , toLocation ) ) return from
172
207
173
208
toLocation . redirectedFrom = redirectedFrom
174
209
@@ -427,12 +462,19 @@ export function createRouter({
427
462
428
463
const router : Router = {
429
464
currentRoute,
465
+
466
+ addRoute,
467
+ removeRoute,
468
+ getRoutes,
469
+
430
470
push,
431
471
replace,
432
472
resolve,
473
+
433
474
beforeEach : beforeGuards . add ,
434
475
afterEach : afterGuards . add ,
435
476
createHref,
477
+
436
478
onError : errorHandlers . add ,
437
479
isReady,
438
480
@@ -497,3 +539,44 @@ function extractChangingRecords(
497
539
498
540
return [ leavingRecords , updatingRecords , enteringRecords ]
499
541
}
542
+
543
+ function isSameLocation (
544
+ a : RouteLocationNormalized ,
545
+ b : RouteLocationNormalized
546
+ ) : boolean {
547
+ return (
548
+ a . name === b . name &&
549
+ a . path === b . path &&
550
+ a . hash === b . hash &&
551
+ isSameLocationQuery ( a . query , b . query )
552
+ )
553
+ }
554
+
555
+ function isSameLocationQuery (
556
+ a : RouteLocationNormalized [ 'query' ] ,
557
+ b : RouteLocationNormalized [ 'query' ]
558
+ ) : boolean {
559
+ const aKeys = Object . keys ( a )
560
+ const bKeys = Object . keys ( b )
561
+ if ( aKeys . length !== bKeys . length ) return false
562
+ let i = 0
563
+ let key : string
564
+ while ( i < aKeys . length ) {
565
+ key = aKeys [ i ]
566
+ if ( key !== bKeys [ i ] ) return false
567
+ if ( ! isSameLocationQueryValue ( a [ key ] , b [ key ] ) ) return false
568
+ i ++
569
+ }
570
+
571
+ return true
572
+ }
573
+
574
+ function isSameLocationQueryValue (
575
+ a : LocationQueryValue | LocationQueryValue [ ] ,
576
+ b : LocationQueryValue | LocationQueryValue [ ]
577
+ ) : boolean {
578
+ if ( typeof a !== typeof b ) return false
579
+ if ( Array . isArray ( a ) )
580
+ return a . every ( ( value , i ) => value === ( b as LocationQueryValue [ ] ) [ i ] )
581
+ return a === b
582
+ }
0 commit comments