forked from mirror/misskey

* fix * navhookをbootに移動 * サーバーサイドのbootも分けるように * 埋め込みページかどうかの判定は最初の一回だけに * tooltipは出せるように * fix design * 埋め込み独自のtooltipを削除 * ロジックの分岐が多かったMkNoteDetailedを分離 * fix indent * プレビュー用iframeにフォーカスが当たるのを修正 * popupの制御を出す側で行うように * パラメータが逆になっていたのを修正 * Update MkEmbedCodeGenDialog.vue * fix * eliminate misskey-js lint warns * fix * add appropriate attributes to embed html * enhance: サーバーサイドのembed系をさらに分離 * enhance: embed routerを分離(route定義をboot時に変更できるようにする改修を含む) * type * lint * fix indent * server-side styleを完全に分離 * Revert "refactor: 画面サイズのしきい値をconstにまとめる" This reverts commit05ca36f400
. * fix * revert all changes in base.pug * embedドメインをまとめた * embedドメインをまとめた * prevent calling contextmenu in embed page by stopping at the caller * fix import * fix import * improve directory structure * fix import * register timeline ui as a container * wa- * rename * wa- * Update EmMediaList.vue * Update EmMediaList.vue * Update EmMediaList.vue * Update EmMediaImage.vue * Update EmNote.vue * revert mkmedialist changes * 戻し漏れ * wip * tweak embed media ui * revert original media components * Update boot.embed.js * rename * wip * Update MkNote.vue * wip * Update MkSubNoteContent.vue * Update EmNote.vue * Update packages/frontend/src/router/definition.ts * Revert "Update packages/frontend/src/router/definition.ts" This reverts commit937ae44521
. * refactor EmMediaImage * fix import * remove unused imports * Update router.ts * wip * Update boot.ts * wip * wip * wip * wip * Update EmNote.vue * Update EmNote.vue * Create EmA.vue * Create EmAvatar.vue * Update EmAvatar.vue * wip * wip * wip * Create EmImgWithBlurhash.vue * Update EmImgWithBlurhash.vue * Create EmPagination.vue * wip * Update boot.ts * wip * wip * wi@p * wip * wip * wiop * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * Update boot.ts * wip * Update MkMisskeyFlavoredMarkdown.ts * wip * wip * wip * wip * wip * Update post-message.ts * wip * Update EmNoteDetailed.vue * Update EmNoteDetailed.vue * Create instance.ts * Update EmNoteDetailed.vue * wip * Update EmNoteDetailed.vue * wip * wip * wip * Update pnpm-lock.yaml * wip * wip * wp * wip * Update ClientServerService.ts * wip * Update boot.ts * Update vite.config.local-dev.ts * Update vite.config.ts * Create index.html * wa- * wip * Update boot.ts * wip * wip * wip * wip * wip * wip * wip * wip * wip * Create EmLink.vue * Create EmMention.vue * Update EmMfm.ts * wip * wip * wip * wip * Update vite.config.ts * Update boot.ts * Update EmA.vue * うぃp * wip * wip * Create EmError.vue * wip * Update MkEmbedCodeGenDialog.vue * Update EmNote.vue * wip * wip * Update user-timeline.vue * Update check-spdx-license-id.yml * wip * wip * style(frontend-shared): lint fixes on build.js * fix(frontend-shared): include `*.{js,json}` files in js-built * wip * use alias * refactor * refactor * Update scroll.ts * refactor * refactor * refactor * wip * wip * wip * wip * Update roles.vue * Update branding.vue * wip * wip * wip * Update page.vue * wip * fix import * add missing css variables * 絵文字をtwemojiに変更 クライアントデフォルトにあわせるため * force empoll readonly * fix compiler error * fix broken imports * tweak button style * run api extractor * fix storybook theme preloads * fix storybook instance imports * Update preview.ts * Update preview.ts * Update preview.ts * Revert "Update preview.ts" This reverts commit12bab1c6fb
. * Revert "Update preview.ts" This reverts commit5c0ce01dbd
. * Revert "Update preview.ts" This reverts commitf4863524d7
. * Revert "fix storybook instance imports" This reverts commited8eabb246
. * Revert "wip" This reverts commitd3c1926519
. * Revert "Update page.vue" This reverts commit27c7900b0c
. * Revert "Update branding.vue" This reverts commitc08ccb65ba
. * Revert "Update roles.vue" This reverts commit1488b67066
. * Revert "wip" This reverts commitaab1c76981
. * refactor: use common media proxy * fix imports * fix * fix: MediaProxyの初期化を保証する(storybook対策?) * enhance(frontend-embed): improve embedParams provide * fix(backend): MK_DEV_PREFER=backendのときにembed viteが読み込めないのを修正 * fix * embed-pageを共通化 * fix import * fix import * fix import * const.jsを共通化 (たぶんrevertしすぎた) * fix type error * fix duplicated import * fix lint * fix * コメントとして残す * sharedとembedをlint対象にする * lint * attempt to fix eslint (frontend-shared) * lint fixes --------- Co-authored-by: syuilo <4439005+syuilo@users.noreply.github.com> Co-authored-by: zyoshoka <107108195+zyoshoka@users.noreply.github.com>
269 lines
6.4 KiB
Vue
269 lines
6.4 KiB
Vue
<!--
|
|
SPDX-FileCopyrightText: syuilo and misskey-project
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
-->
|
|
|
|
<template>
|
|
<div ref="rootEl">
|
|
<div v-if="isPullStart" :class="$style.frame" :style="`--frame-min-height: ${pullDistance / (PULL_BRAKE_BASE + (pullDistance / PULL_BRAKE_FACTOR))}px;`">
|
|
<div :class="$style.frameContent">
|
|
<MkLoading v-if="isRefreshing" :class="$style.loader" :em="true"/>
|
|
<i v-else class="ti ti-arrow-bar-to-down" :class="[$style.icon, { [$style.refresh]: isPullEnd }]"></i>
|
|
<div :class="$style.text">
|
|
<template v-if="isPullEnd">{{ i18n.ts.releaseToRefresh }}</template>
|
|
<template v-else-if="isRefreshing">{{ i18n.ts.refreshing }}</template>
|
|
<template v-else>{{ i18n.ts.pullDownToRefresh }}</template>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div :class="{ [$style.slotClip]: isPullStart }">
|
|
<slot/>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts" setup>
|
|
import { onMounted, onUnmounted, ref, shallowRef } from 'vue';
|
|
import { i18n } from '@/i18n.js';
|
|
import { getScrollContainer } from '@@/js/scroll.js';
|
|
import { isHorizontalSwipeSwiping } from '@/scripts/touch.js';
|
|
|
|
const SCROLL_STOP = 10;
|
|
const MAX_PULL_DISTANCE = Infinity;
|
|
const FIRE_THRESHOLD = 230;
|
|
const RELEASE_TRANSITION_DURATION = 200;
|
|
const PULL_BRAKE_BASE = 1.5;
|
|
const PULL_BRAKE_FACTOR = 170;
|
|
|
|
const isPullStart = ref(false);
|
|
const isPullEnd = ref(false);
|
|
const isRefreshing = ref(false);
|
|
const pullDistance = ref(0);
|
|
|
|
let supportPointerDesktop = false;
|
|
let startScreenY: number | null = null;
|
|
|
|
const rootEl = shallowRef<HTMLDivElement>();
|
|
let scrollEl: HTMLElement | null = null;
|
|
|
|
let disabled = false;
|
|
|
|
const props = withDefaults(defineProps<{
|
|
refresher: () => Promise<void>;
|
|
}>(), {
|
|
refresher: () => Promise.resolve(),
|
|
});
|
|
|
|
const emit = defineEmits<{
|
|
(ev: 'refresh'): void;
|
|
}>();
|
|
|
|
function getScreenY(event) {
|
|
if (supportPointerDesktop) {
|
|
return event.screenY;
|
|
}
|
|
return event.touches[0].screenY;
|
|
}
|
|
|
|
function moveStart(event) {
|
|
if (!isPullStart.value && !isRefreshing.value && !disabled) {
|
|
isPullStart.value = true;
|
|
startScreenY = getScreenY(event);
|
|
pullDistance.value = 0;
|
|
}
|
|
}
|
|
|
|
function moveBySystem(to: number): Promise<void> {
|
|
return new Promise(r => {
|
|
const startHeight = pullDistance.value;
|
|
const overHeight = pullDistance.value - to;
|
|
if (overHeight < 1) {
|
|
r();
|
|
return;
|
|
}
|
|
const startTime = Date.now();
|
|
let intervalId = setInterval(() => {
|
|
const time = Date.now() - startTime;
|
|
if (time > RELEASE_TRANSITION_DURATION) {
|
|
pullDistance.value = to;
|
|
clearInterval(intervalId);
|
|
r();
|
|
return;
|
|
}
|
|
const nextHeight = startHeight - (overHeight / RELEASE_TRANSITION_DURATION) * time;
|
|
if (pullDistance.value < nextHeight) return;
|
|
pullDistance.value = nextHeight;
|
|
}, 1);
|
|
});
|
|
}
|
|
|
|
async function fixOverContent() {
|
|
if (pullDistance.value > FIRE_THRESHOLD) {
|
|
await moveBySystem(FIRE_THRESHOLD);
|
|
}
|
|
}
|
|
|
|
async function closeContent() {
|
|
if (pullDistance.value > 0) {
|
|
await moveBySystem(0);
|
|
}
|
|
}
|
|
|
|
function moveEnd() {
|
|
if (isPullStart.value && !isRefreshing.value) {
|
|
startScreenY = null;
|
|
if (isPullEnd.value) {
|
|
isPullEnd.value = false;
|
|
isRefreshing.value = true;
|
|
fixOverContent().then(() => {
|
|
emit('refresh');
|
|
props.refresher().then(() => {
|
|
refreshFinished();
|
|
});
|
|
});
|
|
} else {
|
|
closeContent().then(() => isPullStart.value = false);
|
|
}
|
|
}
|
|
}
|
|
|
|
function moving(event: TouchEvent | PointerEvent) {
|
|
if (!isPullStart.value || isRefreshing.value || disabled) return;
|
|
|
|
if ((scrollEl?.scrollTop ?? 0) > (supportPointerDesktop ? SCROLL_STOP : SCROLL_STOP + pullDistance.value) || isHorizontalSwipeSwiping.value) {
|
|
pullDistance.value = 0;
|
|
isPullEnd.value = false;
|
|
moveEnd();
|
|
return;
|
|
}
|
|
|
|
if (startScreenY === null) {
|
|
startScreenY = getScreenY(event);
|
|
}
|
|
const moveScreenY = getScreenY(event);
|
|
|
|
const moveHeight = moveScreenY - startScreenY!;
|
|
pullDistance.value = Math.min(Math.max(moveHeight, 0), MAX_PULL_DISTANCE);
|
|
|
|
if (pullDistance.value > 0) {
|
|
if (event.cancelable) event.preventDefault();
|
|
}
|
|
|
|
if (pullDistance.value > SCROLL_STOP) {
|
|
event.stopPropagation();
|
|
}
|
|
|
|
isPullEnd.value = pullDistance.value >= FIRE_THRESHOLD;
|
|
}
|
|
|
|
/**
|
|
* emit(refresh)が完了したことを知らせる関数
|
|
*
|
|
* タイムアウトがないのでこれを最終的に実行しないと出たままになる
|
|
*/
|
|
function refreshFinished() {
|
|
closeContent().then(() => {
|
|
isPullStart.value = false;
|
|
isRefreshing.value = false;
|
|
});
|
|
}
|
|
|
|
function setDisabled(value) {
|
|
disabled = value;
|
|
}
|
|
|
|
function onScrollContainerScroll() {
|
|
const scrollPos = scrollEl!.scrollTop;
|
|
|
|
// When at the top of the page, disable vertical overscroll so passive touch listeners can take over.
|
|
if (scrollPos === 0) {
|
|
scrollEl!.style.touchAction = 'pan-x pan-down pinch-zoom';
|
|
registerEventListenersForReadyToPull();
|
|
} else {
|
|
scrollEl!.style.touchAction = 'auto';
|
|
unregisterEventListenersForReadyToPull();
|
|
}
|
|
}
|
|
|
|
function registerEventListenersForReadyToPull() {
|
|
if (rootEl.value == null) return;
|
|
rootEl.value.addEventListener('touchstart', moveStart, { passive: true });
|
|
rootEl.value.addEventListener('touchmove', moving, { passive: false }); // passive: falseにしないとpreventDefaultが使えない
|
|
}
|
|
|
|
function unregisterEventListenersForReadyToPull() {
|
|
if (rootEl.value == null) return;
|
|
rootEl.value.removeEventListener('touchstart', moveStart);
|
|
rootEl.value.removeEventListener('touchmove', moving);
|
|
}
|
|
|
|
onMounted(() => {
|
|
if (rootEl.value == null) return;
|
|
|
|
scrollEl = getScrollContainer(rootEl.value);
|
|
if (scrollEl == null) return;
|
|
|
|
scrollEl.addEventListener('scroll', onScrollContainerScroll, { passive: true });
|
|
|
|
rootEl.value.addEventListener('touchend', moveEnd, { passive: true });
|
|
|
|
registerEventListenersForReadyToPull();
|
|
});
|
|
|
|
onUnmounted(() => {
|
|
if (scrollEl) scrollEl.removeEventListener('scroll', onScrollContainerScroll);
|
|
|
|
unregisterEventListenersForReadyToPull();
|
|
});
|
|
|
|
defineExpose({
|
|
setDisabled,
|
|
});
|
|
</script>
|
|
|
|
<style lang="scss" module>
|
|
.frame {
|
|
position: relative;
|
|
overflow: clip;
|
|
|
|
width: 100%;
|
|
min-height: var(--frame-min-height, 0px);
|
|
|
|
mask-image: linear-gradient(90deg, #000 0%, #000 80%, transparent);
|
|
-webkit-mask-image: -webkit-linear-gradient(90deg, #000 0%, #000 80%, transparent);
|
|
|
|
pointer-events: none;
|
|
}
|
|
|
|
.frameContent {
|
|
position: absolute;
|
|
bottom: 0;
|
|
width: 100%;
|
|
margin: 5px 0;
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: center;
|
|
font-size: 14px;
|
|
|
|
> .icon, > .loader {
|
|
margin: 6px 0;
|
|
}
|
|
|
|
> .icon {
|
|
transition: transform .25s;
|
|
|
|
&.refresh {
|
|
transform: rotate(180deg);
|
|
}
|
|
}
|
|
|
|
> .text {
|
|
margin: 5px 0;
|
|
}
|
|
}
|
|
|
|
.slotClip {
|
|
overflow-y: clip;
|
|
}
|
|
</style>
|