mirror of
https://codeberg.org/yeentown/barkey.git
synced 2025-10-23 17:54:52 +00:00
75 lines
2.2 KiB
TypeScript
75 lines
2.2 KiB
TypeScript
/*
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
import { throttle } from 'throttle-debounce';
|
|
import { nextTick, onActivated, onDeactivated, onUnmounted, watch } from 'vue';
|
|
import type { Ref } from 'vue';
|
|
|
|
// note render skippingがオンだとズレるため、遷移直前にスクロール範囲に表示されているdata-scroll-anchor要素を特定して、復元時に当該要素までスクロールするようにする
|
|
|
|
export function useScrollPositionKeeper(scrollContainerRef: Ref<HTMLElement | null | undefined>): void {
|
|
let anchorId: string | null = null;
|
|
let ready = true;
|
|
|
|
watch(scrollContainerRef, (el) => {
|
|
if (!el) return;
|
|
|
|
const onScroll = () => {
|
|
if (!el) return;
|
|
if (!ready) return;
|
|
|
|
const scrollContainerRect = el.getBoundingClientRect();
|
|
const viewPosition = scrollContainerRect.height / 2;
|
|
|
|
const anchorEls = el.querySelectorAll('[data-scroll-anchor]');
|
|
for (let i = anchorEls.length - 1; i > -1; i--) { // 下から見た方が速い
|
|
const anchorEl = anchorEls[i] as HTMLElement;
|
|
const anchorRect = anchorEl.getBoundingClientRect();
|
|
const anchorTop = anchorRect.top;
|
|
const anchorBottom = anchorRect.bottom;
|
|
if (anchorTop <= viewPosition && anchorBottom >= viewPosition) {
|
|
anchorId = anchorEl.getAttribute('data-scroll-anchor');
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
|
|
// ほんとはscrollイベントじゃなくてonBeforeDeactivatedでやりたい
|
|
// https://github.com/vuejs/vue/issues/9454
|
|
// https://github.com/vuejs/rfcs/pull/284
|
|
el.addEventListener('scroll', throttle(1000, onScroll), { passive: true });
|
|
}, {
|
|
immediate: true,
|
|
});
|
|
|
|
const restore = () => {
|
|
if (!anchorId) return;
|
|
const scrollContainer = scrollContainerRef.value;
|
|
if (!scrollContainer) return;
|
|
const scrollAnchorEl = scrollContainer.querySelector(`[data-scroll-anchor="${anchorId}"]`);
|
|
if (!scrollAnchorEl) return;
|
|
scrollAnchorEl.scrollIntoView({
|
|
behavior: 'instant',
|
|
block: 'center',
|
|
inline: 'center',
|
|
});
|
|
};
|
|
|
|
onDeactivated(() => {
|
|
ready = false;
|
|
});
|
|
|
|
onActivated(() => {
|
|
restore();
|
|
nextTick(() => {
|
|
restore();
|
|
window.setTimeout(() => {
|
|
restore();
|
|
|
|
ready = true;
|
|
}, 100);
|
|
});
|
|
});
|
|
}
|