misskey/packages/frontend/src/pages/qr.read.vue

231 lines
6.0 KiB
Vue

<!--
SPDX-FileCopyrightText: syuilo and misskey-project
SPDX-License-Identifier: AGPL-3.0-only
-->
<template>
<div
ref="rootEl" :class="$style.root" :style="{
'--MI-QrReadScrollHeight': scrollContainer ? `${scrollHeight}px` : `calc( 100dvh - var(--MI-minBottomSpacing) )`,
'--MI-QrReadViewHeight': 'calc(var(--MI-QrReadScrollHeight) - var(--MI-stickyTop, 0px) - var(--MI-stickyBottom, 0px))',
'--MI-QrReadVideoHeight': 'min(calc(var(--MI-QrReadViewHeight) * 0.3), 512px)',
}"
>
<video ref="videoEl" :class="$style.video" autoplay muted playsinline></video>
<div
class="_spacer"
:style="{
'--MI-stickyTop': 'calc(var(--MI-stickyTop, 0px) + var(--MI-QrReadVideoHeight, 0px))',
}"
>
<MkTab v-model="tab" :class="$style.tab">
<option value="users">{{ i18n.ts.users }} ({{ users.length }})</option>
<option value="notes">{{ i18n.ts.notes }} ({{ notes.length }})</option>
</MkTab>
<div v-if="tab === 'users'" :class="['_gaps', $style.users]" style="padding-bottom: var(--MI-margin);">
<MkUserInfo v-for="user in users" :key="user.id" :user="user"/>
</div>
<div v-else-if="tab === 'notes'" class="_gaps" style="padding-bottom: var(--MI-margin);">
<MkNote v-for="note in notes" :key="note.id" :note="note"/>
</div>
</div>
</div>
</template>
<script lang="ts" setup>
import QrScanner from 'qr-scanner';
import { onMounted, onUnmounted, ref, shallowRef, watch } from 'vue';
import * as misskey from 'misskey-js';
import { getScrollContainer } from '@@/js/scroll.js';
import type { ApShowResponse } from 'misskey-js/entities.js';
import * as os from '@/os.js';
import { i18n } from '@/i18n.js';
import MkUserInfo from '@/components/MkUserInfo.vue';
import { misskeyApi } from '@/utility/misskey-api.js';
const LIST_RERENDER_INTERVAL = 1500;
const scrollContainer = shallowRef<HTMLElement | null>(null);
const rootEl = ref<HTMLDivElement | null>(null);
const scrollHeight = ref(window.innerHeight);
const tab = ref<'users' | 'notes'>('users');
const scannerInstance = shallowRef<QrScanner | null>(null);
const uris = ref<string[]>([]);
const sources = new Map<string, ApShowResponse | null>();
const usersSource = new Map<string, misskey.entities.UserDetailed | null>();
const notesSource = new Map<string, misskey.entities.Note | null>();
const users = ref<(misskey.entities.UserDetailed)[]>([]);
const notes = ref<misskey.entities.Note[]>([]);
const timer = ref<number | null>(null);
function updateLists() {
const results = uris.value.map(uri => sources.get(uri)).filter((r): r is ApShowResponse => !!r);
users.value = results.filter(r => r.type === 'User').map(r => r.object).filter((u): u is misskey.entities.UserDetailed => !!u);
notes.value = results.filter(r => r.type === 'Note').map(r => r.object).filter((n): n is misskey.entities.Note => !!n);
updateRequired.value = false;
}
const updateRequired = ref(false);
watch(uris, () => {
if (timer.value) {
updateRequired.value = true;
return;
}
updateLists();
timer.value = window.setTimeout(() => {
console.log('Update users after 3 seconds');
timer.value = null;
if (updateRequired.value) {
updateLists();
}
}, LIST_RERENDER_INTERVAL) as number;
});
watch(tab, () => {
if (timer.value) {
window.clearTimeout(timer.value);
timer.value = null;
}
});
async function processResult(result: QrScanner.ScanResult) {
if (!result) return;
const uri = result.data.trim();
try {
new URL(uri);
} catch {
return;
}
if (uris.value[0] !== uri) {
// 並べ替え
uris.value = [uri, ...uris.value.slice(0, 29).filter(u => u !== uri)];
}
if (usersSource.has(uri)) return;
// Start fetching user info
usersSource.set(uri, null);
notesSource.set(uri, null);
await misskeyApi('ap/show', { uri })
.then(data => {
if (data.type === 'User') {
usersSource.set(uri, data.object);
tab.value = 'users';
} else if (data.type === 'Note') {
notesSource.set(uri, data.object);
tab.value = 'notes';
}
updateLists();
})
.catch(err => {
console.error(err);
});
}
const alertLock = ref(false);
onMounted(() => {
const videoEl = window.document.querySelector('video');
if (!videoEl) {
os.alert({
type: 'error',
text: i18n.ts.somethingHappened,
});
return;
}
scannerInstance.value = new QrScanner(
videoEl,
processResult,
{
onDecodeError(err) {
if (err.toString().includes('No QR code found')) return;
if (alertLock.value) return;
alertLock.value = true;
os.alert({
type: 'error',
text: err.toString(),
}).finally(() => {
alertLock.value = false;
});
},
},
);
scannerInstance.value.start();
});
onUnmounted(() => {
if (timer.value) {
window.clearTimeout(timer.value);
timer.value = null;
}
scannerInstance.value?.destroy();
});
//#region scroll height
function checkScrollHeight() {
scrollHeight.value = scrollContainer.value ? scrollContainer.value.clientHeight : window.innerHeight;
}
let ro: ResizeObserver | undefined;
onMounted(() => {
scrollContainer.value = getScrollContainer(rootEl.value);
checkScrollHeight();
if (scrollContainer.value) {
ro = new ResizeObserver(checkScrollHeight);
ro.observe(scrollContainer.value);
}
});
onUnmounted(() => {
ro?.disconnect();
});
//#endregion
</script>
<style lang="scss" module>
.root {
position: relative;
}
.video {
position: sticky;
top: var(--MI-stickyTop, 0);
z-index: 1;
background: var(--MI_THEME-bg);
background-size: 16px 16px;
width: 100%;
height: var(--MI-QrReadVideoHeight);
object-fit: contain;
}
html[data-color-scheme=dark] .video {
--c: rgb(255 255 255 / 2%);
background-image: linear-gradient(45deg, var(--c) 16.67%, var(--MI_THEME-bg) 16.67%, var(--MI_THEME-bg) 50%, var(--c) 50%, var(--c) 66.67%, var(--MI_THEME-bg) 66.67%, var(--MI_THEME-bg) 100%);
}
html[data-color-scheme=light] .video {
--c: rgb(0 0 0 / 2%);
background-image: linear-gradient(45deg, var(--c) 16.67%, var(--MI_THEME-bg) 16.67%, var(--MI_THEME-bg) 50%, var(--c) 50%, var(--c) 66.67%, var(--MI_THEME-bg) 66.67%, var(--MI_THEME-bg) 100%);
}
.users {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(260px, 1fr));
grid-gap: var(--MI-margin);
}
</style>