|
1 |
| -import { RouteLocationNormalized, RouteLocationNormalizedLoaded } from './types' |
| 1 | +import { |
| 2 | + RouteLocationNormalized, |
| 3 | + RouteLocationNormalizedLoaded, |
| 4 | + _RouteLocationBase, |
| 5 | +} from './types' |
2 | 6 | import { warn } from './warning'
|
3 | 7 |
|
| 8 | +// we use types instead of interfaces to make it work with HistoryStateValue type |
| 9 | + |
| 10 | +/** |
| 11 | + * Scroll position similar to |
| 12 | + * {@link https://developer.mozilla.org/en-US/docs/Web/API/ScrollToOptions | `ScrollToOptions`}. |
| 13 | + * Note that not all browsers support `behavior`. |
| 14 | + */ |
4 | 15 | export type ScrollPositionCoordinates = {
|
5 |
| - /** |
6 |
| - * x position. 0 if not provided |
7 |
| - */ |
8 |
| - x?: number |
9 |
| - /** |
10 |
| - * y position. 0 if not provided |
11 |
| - */ |
12 |
| - y?: number |
| 16 | + behavior?: ScrollOptions['behavior'] |
| 17 | + left?: number |
| 18 | + top?: number |
| 19 | +} |
| 20 | + |
| 21 | +/** |
| 22 | + * Internal normalized version of {@link ScrollPositionCoordinates} that always |
| 23 | + * has `left` and `top` coordinates. |
| 24 | + * |
| 25 | + * @internal |
| 26 | + */ |
| 27 | +export type _ScrollPositionNormalized = { |
| 28 | + behavior?: ScrollOptions['behavior'] |
| 29 | + left: number |
| 30 | + top: number |
13 | 31 | }
|
14 | 32 |
|
15 | 33 | export interface ScrollPositionElement {
|
@@ -47,24 +65,25 @@ export interface ScrollBehaviorHandler<T> {
|
47 | 65 | function getElementPosition(
|
48 | 66 | el: Element,
|
49 | 67 | offset: ScrollPositionCoordinates
|
50 |
| -): Required<ScrollPositionCoordinates> { |
| 68 | +): _ScrollPositionNormalized { |
51 | 69 | const docRect = document.documentElement.getBoundingClientRect()
|
52 | 70 | const elRect = el.getBoundingClientRect()
|
53 | 71 |
|
54 | 72 | return {
|
55 |
| - x: elRect.left - docRect.left - (offset.x || 0), |
56 |
| - y: elRect.top - docRect.top - (offset.y || 0), |
| 73 | + behavior: offset.behavior, |
| 74 | + left: elRect.left - docRect.left - (offset.left || 0), |
| 75 | + top: elRect.top - docRect.top - (offset.top || 0), |
57 | 76 | }
|
58 | 77 | }
|
59 | 78 |
|
60 | 79 | export const computeScrollPosition = () =>
|
61 | 80 | ({
|
62 |
| - x: window.pageXOffset, |
63 |
| - y: window.pageYOffset, |
64 |
| - } as Required<ScrollPositionCoordinates>) |
| 81 | + left: window.pageXOffset, |
| 82 | + top: window.pageYOffset, |
| 83 | + } as _ScrollPositionNormalized) |
65 | 84 |
|
66 | 85 | export function scrollToPosition(position: ScrollPosition): void {
|
67 |
| - let normalizedPosition: ScrollPositionCoordinates |
| 86 | + let scrollToOptions: ScrollPositionCoordinates |
68 | 87 |
|
69 | 88 | if ('selector' in position) {
|
70 | 89 | /**
|
@@ -105,27 +124,26 @@ export function scrollToPosition(position: ScrollPosition): void {
|
105 | 124 | warn(`Couldn't find element with selector "${position.selector}"`)
|
106 | 125 | return
|
107 | 126 | }
|
108 |
| - normalizedPosition = getElementPosition(el, position.offset || {}) |
| 127 | + scrollToOptions = getElementPosition(el, position.offset || {}) |
109 | 128 | } else {
|
110 |
| - normalizedPosition = position |
| 129 | + scrollToOptions = position |
111 | 130 | }
|
112 | 131 |
|
113 |
| - window.scrollTo(normalizedPosition.x || 0, normalizedPosition.y || 0) |
| 132 | + if ('scrollBehavior' in document.documentElement.style) |
| 133 | + window.scrollTo(scrollToOptions) |
| 134 | + else window.scrollTo(scrollToOptions.left || 0, scrollToOptions.top || 0) |
114 | 135 | }
|
115 | 136 |
|
116 | 137 | export function getScrollKey(path: string, delta: number): string {
|
117 | 138 | const position: number = history.state ? history.state.position - delta : -1
|
118 | 139 | return position + path
|
119 | 140 | }
|
120 | 141 |
|
121 |
| -export const scrollPositions = new Map< |
122 |
| - string, |
123 |
| - Required<ScrollPositionCoordinates> |
124 |
| ->() |
| 142 | +export const scrollPositions = new Map<string, _ScrollPositionNormalized>() |
125 | 143 |
|
126 | 144 | export function saveScrollPosition(
|
127 | 145 | key: string,
|
128 |
| - scrollPosition: Required<ScrollPositionCoordinates> |
| 146 | + scrollPosition: _ScrollPositionNormalized |
129 | 147 | ) {
|
130 | 148 | scrollPositions.set(key, scrollPosition)
|
131 | 149 | }
|
|
0 commit comments