@@ -17,13 +17,14 @@ import {
17
17
} from './types'
18
18
import { RouterHistory , HistoryState } from './history/common'
19
19
import {
20
- ScrollToPosition ,
20
+ ScrollPositionCoordinates ,
21
21
ScrollPosition ,
22
- scrollToPosition ,
23
- saveScrollOnLeave ,
22
+ getSavedScrollPosition ,
24
23
getScrollKey ,
25
- getSavedScroll ,
26
- } from './utils/scroll'
24
+ saveScrollPosition ,
25
+ computeScrollPosition ,
26
+ scrollToPosition ,
27
+ } from './scrollBehavior'
27
28
import { createRouterMatcher } from './matcher'
28
29
import {
29
30
createRouterError ,
@@ -71,7 +72,7 @@ export interface ScrollBehavior {
71
72
(
72
73
to : RouteLocationNormalized ,
73
74
from : RouteLocationNormalizedLoaded ,
74
- savedPosition : ScrollToPosition | null
75
+ savedPosition : Required < ScrollPositionCoordinates > | null
75
76
) : // TODO: implement false nad refactor promise based type
76
77
Awaitable < ScrollPosition | false | void >
77
78
}
@@ -303,6 +304,7 @@ export function createRouter({
303
304
// to could be a string where `replace` is a function
304
305
const replace = ( to as RouteLocationOptions ) . replace === true
305
306
307
+ // TODO: create navigation failure
306
308
if ( ! force && isSameRouteLocation ( from , targetLocation ) ) return
307
309
308
310
const lastMatched =
@@ -515,25 +517,35 @@ export function createRouter({
515
517
516
518
// only consider as push if it's not the first navigation
517
519
const isFirstNavigation = from === START_LOCATION_NORMALIZED
520
+ const state = ! isBrowser ? { } : window . history . state
518
521
519
522
// change URL only if the user did a push/replace and if it's not the initial navigation because
520
523
// it's just reflecting the url
521
524
if ( isPush ) {
522
- if ( replace || isFirstNavigation ) history . replace ( toLocation , data )
525
+ // on the initial navigation, we want to reuse the scroll position from
526
+ // history state if it exists
527
+ if ( replace || isFirstNavigation )
528
+ history . replace ( toLocation , {
529
+ scroll : isFirstNavigation && state && state . scroll ,
530
+ ...data ,
531
+ } )
523
532
else history . push ( toLocation , data )
524
533
}
525
534
526
535
// accept current navigation
527
536
currentRoute . value = markRaw ( toLocation )
528
537
// TODO: this doesn't work on first load. Moving it to RouterView could allow automatically handling transitions too maybe
529
538
// TODO: refactor with a state getter
530
- const state = isPush || ! isBrowser ? { } : window . history . state
531
- const savedScroll = getSavedScroll ( getScrollKey ( toLocation . fullPath , 0 ) )
532
- handleScroll (
533
- toLocation ,
534
- from ,
535
- savedScroll || ( state && state . scroll )
536
- ) . catch ( err => triggerError ( err ) )
539
+ if ( isBrowser ) {
540
+ const savedScroll = getSavedScrollPosition (
541
+ getScrollKey ( toLocation . fullPath , 0 )
542
+ )
543
+ handleScroll (
544
+ toLocation ,
545
+ from ,
546
+ savedScroll || ( ( isFirstNavigation || ! isPush ) && state && state . scroll )
547
+ ) . catch ( err => triggerError ( err ) )
548
+ }
537
549
538
550
markAsReady ( )
539
551
}
@@ -547,7 +559,10 @@ export function createRouter({
547
559
pendingLocation = toLocation
548
560
const from = currentRoute . value
549
561
550
- saveScrollOnLeave ( getScrollKey ( from . fullPath , info . distance ) )
562
+ saveScrollPosition (
563
+ getScrollKey ( from . fullPath , info . distance ) ,
564
+ computeScrollPosition ( )
565
+ )
551
566
552
567
let failure : NavigationFailure | void
553
568
@@ -651,7 +666,7 @@ export function createRouter({
651
666
async function handleScroll (
652
667
to : RouteLocationNormalizedLoaded ,
653
668
from : RouteLocationNormalizedLoaded ,
654
- scrollPosition ?: ScrollToPosition
669
+ scrollPosition ?: Required < ScrollPositionCoordinates >
655
670
) {
656
671
if ( ! scrollBehavior ) return
657
672
@@ -753,22 +768,13 @@ function applyRouterPlugin(app: App, router: Router) {
753
768
get : ( ) => router . currentRoute . value ,
754
769
} )
755
770
756
- let started = false
757
- // TODO: can we use something that isn't a mixin? Like adding an onMount hook here
758
- if ( isBrowser ) {
759
- app . mixin ( {
760
- beforeCreate ( ) {
761
- if ( ! started ) {
762
- // this initial navigation is only necessary on client, on server it doesn't make sense
763
- // because it will create an extra unnecessary navigation and could lead to problems
764
- router . push ( router . history . location . fullPath ) . catch ( err => {
765
- if ( __DEV__ )
766
- console . error ( 'Unhandled error when starting the router' , err )
767
- else return err
768
- } )
769
- started = true
770
- }
771
- } ,
771
+ // this initial navigation is only necessary on client, on server it doesn't
772
+ // make sense because it will create an extra unnecessary navigation and could
773
+ // lead to problems
774
+ if ( isBrowser && router . currentRoute . value === START_LOCATION_NORMALIZED ) {
775
+ router . push ( router . history . location . fullPath ) . catch ( err => {
776
+ if ( __DEV__ )
777
+ console . error ( 'Unhandled error when starting the router' , err )
772
778
} )
773
779
}
774
780
0 commit comments