hippofish/packages/client/src/components/MkNoteDetailed.vue

713 lines
15 KiB
Vue
Raw Normal View History

<template>
2023-04-08 02:01:42 +02:00
<div
v-if="!muted.muted"
v-show="!isDeleted"
ref="el"
v-hotkey="keymap"
2023-06-13 06:04:36 +02:00
v-size="{ max: [500, 350, 300] }"
2023-04-08 02:01:42 +02:00
class="lxwezrsl _block"
:tabindex="!isDeleted ? '-1' : null"
:class="{ renote: isRenote }"
>
<MkNoteSub
v-for="note in conversation"
2023-09-02 01:27:33 +02:00
v-if="conversation"
2023-04-08 02:01:42 +02:00
:key="note.id"
class="reply-to"
2023-04-08 02:01:42 +02:00
:note="note"
2023-09-02 01:27:33 +02:00
:detailed-view="true"
2023-04-08 02:01:42 +02:00
/>
2023-05-29 02:20:53 +02:00
<MkLoading v-else-if="note.reply" mini />
2023-04-08 02:01:42 +02:00
<MkNoteSub
2023-05-29 02:20:53 +02:00
v-if="note.reply"
:note="note.reply"
2023-04-08 02:01:42 +02:00
class="reply-to"
2023-09-02 01:27:33 +02:00
:detailed-view="true"
2023-04-08 02:01:42 +02:00
/>
2023-04-23 02:22:53 +02:00
2023-05-19 19:02:41 +02:00
<MkNote
ref="noteEl"
tabindex="-1"
2023-05-29 02:20:53 +02:00
:note="note"
2023-09-02 01:27:33 +02:00
detailed-view
@contextmenu.stop="onContextmenu"
2023-05-19 19:02:41 +02:00
></MkNote>
2023-07-17 00:32:32 +02:00
<MkTab v-model="tab" :style="'underline'" @update:modelValue="loadTab">
<option value="replies">
<!-- <i :class="icon('ph-arrow-u-up-left')"></i> -->
{{
wordWithCount(
note.repliesCount,
i18n.ts.reply,
i18n.ts.replies,
)
}}
2023-05-19 19:02:41 +02:00
</option>
2023-09-02 01:27:33 +02:00
<option v-if="note.renoteCount > 0" value="renotes">
<!-- <i :class="icon('ph-rocket-launch')"></i> -->
{{
wordWithCount(
note.renoteCount,
i18n.ts.renote,
i18n.ts.renotes,
)
}}
2023-05-19 19:02:41 +02:00
</option>
2023-09-02 01:27:33 +02:00
<option v-if="reactionsCount > 0" value="reactions">
<!-- <i :class="icon('ph-smiley')"></i> -->
{{
wordWithCount(
reactionsCount,
i18n.ts.reaction,
i18n.ts.reactions,
)
}}
2023-05-20 06:14:14 +02:00
</option>
2023-09-02 01:27:33 +02:00
<option v-if="directQuotes?.length > 0" value="quotes">
<!-- <i :class="icon('ph-quotes')"></i> -->
{{
wordWithCount(
directQuotes.length,
i18n.ts.quote,
i18n.ts.quotes,
)
}}
2023-05-21 03:20:17 +02:00
</option>
2023-09-02 01:27:33 +02:00
<option v-if="clips?.length > 0" value="clips">
<!-- <i :class="icon('ph-paperclip')"></i> -->
{{ wordWithCount(clips.length, i18n.ts.clip, i18n.ts.clips) }}
2023-05-19 19:02:41 +02:00
</option>
</MkTab>
2023-04-08 02:01:42 +02:00
<MkNoteSub
v-for="note in directReplies"
2023-09-02 01:27:33 +02:00
v-if="directReplies && tab === 'replies'"
2023-04-08 02:01:42 +02:00
:key="note.id"
:note="note"
class="reply"
:conversation="replies"
2023-09-02 01:27:33 +02:00
:detailed-view="true"
:parent-id="note.id"
2023-04-08 02:01:42 +02:00
/>
<MkLoading v-else-if="tab === 'replies' && note.repliesCount > 0" />
2023-05-20 00:41:59 +02:00
2023-05-19 19:02:41 +02:00
<MkNoteSub
v-for="note in directQuotes"
2023-09-02 01:27:33 +02:00
v-if="directQuotes && tab === 'quotes'"
2023-05-19 19:02:41 +02:00
:key="note.id"
:note="note"
class="reply"
2023-05-24 23:40:16 +02:00
:conversation="replies"
2023-09-02 01:27:33 +02:00
:detailed-view="true"
:parent-id="note.id"
2023-05-19 19:02:41 +02:00
/>
<MkLoading v-else-if="tab === 'quotes' && directQuotes.length > 0" />
2023-05-20 00:41:59 +02:00
2023-05-19 19:02:41 +02:00
<!-- <MkPagination
v-if="tab === 'renotes'"
v-slot="{ items }"
ref="pagingComponent"
:pagination="pagination"
> -->
2023-05-20 00:41:59 +02:00
<MkUserCardMini
v-for="item in renotes"
2023-09-02 01:27:33 +02:00
v-if="tab === 'renotes' && renotes"
2023-05-20 00:41:59 +02:00
:key="item.user.id"
:user="item.user"
/>
2023-05-19 19:02:41 +02:00
<!-- </MkPagination> -->
<MkLoading v-else-if="tab === 'renotes' && note.renoteCount > 0" />
2023-05-19 19:02:41 +02:00
2023-05-20 00:41:59 +02:00
<div v-if="tab === 'clips' && clips.length > 0" class="_content clips">
2023-05-19 19:02:41 +02:00
<MkA
v-for="item in clips"
:key="item.id"
:to="`/clips/${item.id}`"
class="item _panel"
>
<b>{{ item.name }}</b>
2023-05-20 00:41:59 +02:00
<div v-if="item.description" class="description">
2023-05-19 19:02:41 +02:00
{{ item.description }}
</div>
<div class="user">
<MkAvatar
:user="item.user"
class="avatar"
:show-indicator="true"
/>
2023-05-20 00:41:59 +02:00
<MkUserName :user="item.user" :nowrap="false" />
2023-05-19 19:02:41 +02:00
</div>
</MkA>
</div>
<MkLoading v-else-if="tab === 'clips' && clips.length > 0" />
2023-05-20 06:14:14 +02:00
<MkReactedUsers
2023-05-20 06:41:42 +02:00
v-if="tab === 'reactions' && reactionsCount > 0"
2023-05-29 02:20:53 +02:00
:note-id="note.id"
2023-05-20 06:14:14 +02:00
></MkReactedUsers>
2023-04-08 02:01:42 +02:00
</div>
<div v-else class="_panel muted" @click="muted.muted = false">
2023-05-04 07:13:13 +02:00
<I18n :src="softMuteReasonI18nSrc(muted.what)" tag="small">
2023-04-08 02:01:42 +02:00
<template #name>
<MkA
2023-05-05 03:38:56 +02:00
v-user-preview="note.userId"
2023-04-08 02:01:42 +02:00
class="name"
2023-05-05 03:38:56 +02:00
:to="userPage(note.user)"
2023-04-08 02:01:42 +02:00
>
2023-05-05 03:38:56 +02:00
<MkUserName :user="note.user" />
2023-04-08 02:01:42 +02:00
</MkA>
</template>
<template #reason>
<b class="_blur_text">{{ muted.matched.join(", ") }}</b>
</template>
</I18n>
</div>
</template>
<script lang="ts" setup>
2023-08-10 22:25:21 +02:00
import { onMounted, onUnmounted, onUpdated, ref } from "vue";
2024-03-02 06:24:05 +01:00
import type { StreamTypes, entities } from "firefish-js";
2023-05-19 19:02:41 +02:00
import MkTab from "@/components/MkTab.vue";
import MkNote from "@/components/MkNote.vue";
2023-04-08 02:01:42 +02:00
import MkNoteSub from "@/components/MkNoteSub.vue";
2023-09-02 01:27:33 +02:00
import type XRenoteButton from "@/components/MkRenoteButton.vue";
2023-05-19 19:02:41 +02:00
import MkUserCardMini from "@/components/MkUserCardMini.vue";
2023-05-20 06:14:14 +02:00
import MkReactedUsers from "@/components/MkReactedUsers.vue";
2023-04-08 02:01:42 +02:00
import { pleaseLogin } from "@/scripts/please-login";
import { getWordSoftMute } from "@/scripts/check-word-mute";
2023-04-08 02:01:42 +02:00
import { userPage } from "@/filters/user";
import * as os from "@/os";
import { defaultStore, noteViewInterruptors } from "@/store";
import { reactionPicker } from "@/scripts/reaction-picker";
import { $i } from "@/reactiveAccount";
2023-04-08 02:01:42 +02:00
import { i18n } from "@/i18n";
import { getNoteMenu } from "@/scripts/get-note-menu";
import { useNoteCapture } from "@/scripts/use-note-capture";
import { deepClone } from "@/scripts/clone";
import { useStream } from "@/stream";
// import icon from "@/scripts/icon";
const props = defineProps<{
note: entities.Note;
pinned?: boolean;
}>();
const stream = useStream();
2023-09-02 01:27:33 +02:00
const tab = ref("replies");
2023-05-19 19:02:41 +02:00
2023-09-02 01:27:33 +02:00
const note = ref(deepClone(props.note));
2022-03-04 17:23:34 +01:00
2023-05-04 07:13:13 +02:00
const softMuteReasonI18nSrc = (what?: string) => {
2023-05-04 07:41:18 +02:00
if (what === "note") return i18n.ts.userSaysSomethingReason;
if (what === "reply") return i18n.ts.userSaysSomethingReasonReply;
if (what === "renote") return i18n.ts.userSaysSomethingReasonRenote;
if (what === "quote") return i18n.ts.userSaysSomethingReasonQuote;
2023-05-04 07:13:13 +02:00
// I don't think here is reachable, but just in case
return i18n.ts.userSaysSomething;
2023-05-04 07:41:18 +02:00
};
2023-05-04 07:13:13 +02:00
const wordWithCount = (count: number, singular: string, plural: string) => {
if (count === 0) return plural;
return `${count} ${count === 1 ? singular : plural}`;
};
2022-03-04 17:23:34 +01:00
// plugin
if (noteViewInterruptors.length > 0) {
onMounted(async () => {
let result = deepClone(note.value);
2022-03-04 17:23:34 +01:00
for (const interruptor of noteViewInterruptors) {
result = await interruptor.handler(result);
}
note.value = result;
2022-03-04 17:23:34 +01:00
});
}
const el = ref<HTMLElement>();
const noteEl = ref();
const menuButton = ref<HTMLElement>();
const renoteButton = ref<InstanceType<typeof XRenoteButton>>();
const reactButton = ref<HTMLElement>();
const showContent = ref(false);
const isDeleted = ref(false);
const muted = ref(
getWordSoftMute(
note.value,
$i?.id,
defaultStore.state.mutedWords,
defaultStore.state.mutedLangs,
),
);
const translation = ref(null);
const translating = ref(false);
const conversation = ref<null | entities.Note[]>([]);
const replies = ref<entities.Note[]>([]);
const directReplies = ref<null | entities.Note[]>([]);
const directQuotes = ref<null | entities.Note[]>([]);
2023-09-02 01:27:33 +02:00
const clips = ref();
const renotes = ref();
let isScrolling;
2023-05-20 08:27:56 +02:00
const reactionsCount = Object.values(props.note.reactions).reduce(
(x, y) => x + y,
2023-07-06 03:28:27 +02:00
0,
2023-05-20 08:27:56 +02:00
);
2023-05-20 06:14:14 +02:00
const keymap = {
2023-04-08 02:01:42 +02:00
r: () => reply(true),
"e|a|plus": () => react(true),
q: () => renoteButton.value.renote(true),
esc: blur,
"m|o": () => menu(true),
s: () => showContent.value !== showContent.value,
};
useNoteCapture({
rootEl: el,
2023-09-02 01:27:33 +02:00
note,
isDeletedRef: isDeleted,
});
function reply(viaKeyboard = false): void {
pleaseLogin();
2023-04-30 18:29:50 +02:00
os.post({
reply: note.value,
2023-04-30 18:29:50 +02:00
animation: !viaKeyboard,
}).then(() => {
focus();
});
}
function react(viaKeyboard = false): void {
pleaseLogin();
blur();
2023-04-08 02:01:42 +02:00
reactionPicker.show(
reactButton.value,
(reaction) => {
os.api("notes/reactions/create", {
noteId: note.value.id,
2023-09-02 01:27:33 +02:00
reaction,
2023-04-08 02:01:42 +02:00
});
},
() => {
focus();
2023-07-06 03:28:27 +02:00
},
2023-04-08 02:01:42 +02:00
);
}
function undoReact(note): void {
const oldReaction = note.myReaction;
if (!oldReaction) return;
2023-04-08 02:01:42 +02:00
os.api("notes/reactions/delete", {
2022-06-05 05:26:36 +02:00
noteId: note.id,
});
}
2022-01-18 13:35:57 +01:00
function onContextmenu(ev: MouseEvent): void {
const isLink = (el: HTMLElement) => {
2023-04-08 02:01:42 +02:00
if (el.tagName === "A") return true;
if (el.parentElement) {
return isLink(el.parentElement);
}
};
2022-01-18 13:35:57 +01:00
if (isLink(ev.target)) return;
2023-04-08 02:01:42 +02:00
if (window.getSelection().toString() !== "") return;
if (defaultStore.state.useReactionPickerForContextMenu) {
2022-01-18 13:35:57 +01:00
ev.preventDefault();
react();
} else {
2023-04-08 02:01:42 +02:00
os.contextMenu(
getNoteMenu({
note: note.value,
2023-04-08 02:01:42 +02:00
translating,
translation,
menuButton,
isDeleted,
}),
2023-07-06 03:28:27 +02:00
ev,
2023-04-08 02:01:42 +02:00
).then(focus);
}
}
function menu(viaKeyboard = false): void {
2023-04-08 02:01:42 +02:00
os.popupMenu(
getNoteMenu({
note: note.value,
2023-04-08 02:01:42 +02:00
translating,
translation,
menuButton,
isDeleted,
}),
menuButton.value,
{
viaKeyboard,
2023-07-06 03:28:27 +02:00
},
2023-04-08 02:01:42 +02:00
).then(focus);
}
function focus() {
noteEl.value.focus();
}
function blur() {
noteEl.value.blur();
}
directReplies.value = null;
2023-04-08 02:01:42 +02:00
os.api("notes/children", {
noteId: note.value.id,
2022-07-25 22:41:45 +02:00
limit: 30,
2023-02-24 17:55:51 +01:00
depth: 12,
2023-04-08 02:01:42 +02:00
}).then((res) => {
2023-05-29 02:20:53 +02:00
res = res.reduce((acc, resNote) => {
if (resNote.userId == note.value.userId) {
2023-05-29 02:20:53 +02:00
return [...acc, resNote];
2023-05-24 23:40:16 +02:00
}
2023-05-29 02:20:53 +02:00
return [resNote, ...acc];
2023-05-24 23:40:16 +02:00
}, []);
replies.value = res;
directReplies.value = res
.filter((resNote) => resNote.replyId === note.value.id)
2023-04-08 02:01:42 +02:00
.reverse();
directQuotes.value = res.filter(
(resNote) => resNote.renoteId === note.value.id,
);
});
conversation.value = null;
if (note.value.replyId) {
2023-04-08 02:01:42 +02:00
os.api("notes/conversation", {
noteId: note.value.replyId,
limit: 30,
2023-04-08 02:01:42 +02:00
}).then((res) => {
conversation.value = res.reverse();
focus();
});
}
clips.value = null;
2023-05-19 19:02:41 +02:00
os.api("notes/clips", {
noteId: note.value.id,
2023-05-19 19:02:41 +02:00
}).then((res) => {
clips.value = res;
2023-05-19 19:02:41 +02:00
});
// const pagination = {
// endpoint: "notes/renotes",
2023-05-29 02:20:53 +02:00
// noteId: note.id,
2023-05-19 19:02:41 +02:00
// limit: 10,
// };
// const pagingComponent = $ref<InstanceType<typeof MkPagination>>();
renotes.value = null;
2023-05-19 19:02:41 +02:00
function loadTab() {
if (tab.value === "renotes" && !renotes.value) {
2023-05-19 19:02:41 +02:00
os.api("notes/renotes", {
noteId: note.value.id,
2023-05-19 19:02:41 +02:00
limit: 100,
}).then((res) => {
renotes.value = res;
2023-05-20 00:41:59 +02:00
});
2023-05-19 19:02:41 +02:00
}
}
async function onNoteUpdated(
noteData: StreamTypes.NoteUpdatedEvent,
): Promise<void> {
const { type, id, body } = noteData;
2023-04-30 18:29:50 +02:00
let found = -1;
if (id === note.value.id) {
2023-04-30 18:29:50 +02:00
found = 0;
} else {
for (let i = 0; i < replies.value.length; i++) {
const reply = replies.value[i];
if (reply.id === id) {
found = i + 1;
break;
}
}
}
if (found === -1) {
return;
}
switch (type) {
case "replied":
const { id: createdId } = body;
const replyNote = await os.api("notes/show", {
noteId: createdId,
});
replies.value.splice(found, 0, replyNote);
if (found === 0) {
directReplies.value.push(replyNote);
2023-04-30 18:29:50 +02:00
}
break;
case "deleted":
if (found === 0) {
isDeleted.value = true;
} else {
replies.value.splice(found - 1, 1);
}
2023-04-30 18:29:50 +02:00
break;
}
}
document.addEventListener("wheel", () => {
isScrolling = true;
2023-04-08 02:01:42 +02:00
});
onMounted(() => {
2023-04-30 18:29:50 +02:00
stream.on("noteUpdated", onNoteUpdated);
isScrolling = false;
noteEl.value.scrollIntoView();
});
onUpdated(() => {
if (!isScrolling) {
noteEl.value.scrollIntoView();
if (location.hash) {
location.replace(location.hash); // Jump to highlighted reply
}
}
2023-04-08 02:01:42 +02:00
});
onUnmounted(() => {
2023-04-30 18:29:50 +02:00
stream.off("noteUpdated", onNoteUpdated);
});
</script>
<style lang="scss" scoped>
2021-08-16 08:21:58 +02:00
.lxwezrsl {
font-size: 1.05em;
position: relative;
transition: box-shadow 0.1s ease;
contain: content;
&:focus-visible {
outline: none;
&:after {
content: "";
pointer-events: none;
display: block;
position: absolute;
z-index: 10;
top: 0;
left: 0;
right: 0;
bottom: 0;
margin: auto;
width: calc(100% - 8px);
height: calc(100% - 8px);
2022-07-28 18:25:11 +02:00
border: solid 1px var(--focus);
border-radius: var(--radius);
box-sizing: border-box;
}
}
> .reply-to {
2023-01-07 00:58:52 +01:00
margin-bottom: -16px;
padding-bottom: 16px;
}
2023-05-19 19:02:41 +02:00
> :deep(.note-container) {
padding-block: 28px 0;
padding-top: 12px;
font-size: 1.1rem;
overflow: clip;
outline: none;
scroll-margin-top: calc(var(--stickyTop) + 20vh);
2023-05-21 03:20:17 +02:00
&:not(:last-child) {
border-bottom: 1px solid var(--divider);
margin-bottom: 4px;
}
2023-05-19 19:02:41 +02:00
.article {
cursor: unset;
2023-05-19 19:02:41 +02:00
padding-bottom: 0;
}
2023-05-23 01:40:38 +02:00
&:first-child {
padding-top: 28px;
}
}
2023-05-20 08:27:56 +02:00
> :deep(.chips) {
2023-05-20 06:14:14 +02:00
padding-block: 6px 12px;
padding-left: 32px;
2023-05-20 06:53:15 +02:00
&:last-child {
margin-bottom: 12px;
}
2023-05-19 19:02:41 +02:00
}
2023-05-20 08:27:56 +02:00
> :deep(.user-card-mini),
2023-05-20 06:14:14 +02:00
> :deep(.reacted-users > *) {
2023-05-19 19:02:41 +02:00
padding-inline: 32px;
border-top: 1px solid var(--divider);
border-radius: 0;
}
2023-05-20 06:14:14 +02:00
> :deep(.reacted-users > div) {
padding-block: 12px;
}
2023-05-19 19:02:41 +02:00
> .reply {
border-top: solid 0.5px var(--divider);
2023-02-26 01:59:58 +01:00
padding-top: 24px;
padding-bottom: 10px;
2022-08-17 09:36:29 +02:00
}
2023-02-25 06:19:39 +01:00
// Hover
2023-05-24 03:48:54 +02:00
:deep(.reply > .main),
2023-04-08 02:01:42 +02:00
.reply-to,
:deep(.more) {
2023-02-25 06:19:39 +01:00
position: relative;
&::before {
content: "";
position: absolute;
inset: -12px -24px;
bottom: -0px;
background: var(--panelHighlight);
border-radius: var(--radius);
opacity: 0;
2023-07-06 03:28:27 +02:00
transition:
opacity 0.2s,
background 0.2s;
2023-02-25 06:19:39 +01:00
z-index: -1;
}
&.reply-to {
2023-02-25 06:19:39 +01:00
&::before {
inset: 0px 8px;
}
2023-06-13 06:04:36 +02:00
&:not(.max-width_500px)::before {
2023-06-05 19:11:14 +02:00
bottom: 16px;
}
2023-02-25 18:36:57 +01:00
&:first-of-type::before {
top: 12px;
}
&.reply.max-width_500px:first-of-type::before {
top: 4px;
}
}
// &::after {
// content: "";
// position: absolute;
// inset: -9999px;
// background: var(--modalBg);
// opacity: 0;
// z-index: -2;
// pointer-events: none;
// transition: opacity .2s;
// }
2023-02-25 06:19:39 +01:00
&.more::before {
inset: 0 !important;
}
2023-04-08 02:01:42 +02:00
&:hover,
&:focus-within {
2023-05-24 02:30:55 +02:00
--panel: var(--panelHighlight);
2023-02-25 06:19:39 +01:00
&::before {
opacity: 1;
2023-05-25 22:24:31 +02:00
background: var(--panelHighlight) !important;
2023-02-25 06:19:39 +01:00
}
}
// @media (pointer: coarse) {
// &:has(.button:focus-within) {
// z-index: 2;
// --X13: transparent;
// &::after {
// opacity: 1;
// backdrop-filter: var(--modalBgFilter);
// }
// }
// }
}
:deep(.reply:target > .main),
:deep(.reply-to:target) {
z-index: 2;
&::before {
outline: auto;
opacity: 1;
2023-05-25 22:24:31 +02:00
background: none;
}
}
2023-05-26 15:13:13 +02:00
&.max-width_500px {
2023-06-13 06:04:36 +02:00
font-size: 0.975em;
2023-06-05 19:11:14 +02:00
> .reply-to {
&::before {
inset-inline: -24px;
}
&:first-child {
padding-top: 14px;
&::before {
top: -24px;
}
}
2023-01-08 08:30:42 +01:00
}
2023-05-19 19:02:41 +02:00
> :deep(.note-container) {
2023-05-21 22:48:25 +02:00
padding: 12px 0 0 0;
2023-06-13 06:04:36 +02:00
font-size: 1.05rem;
2023-01-07 00:58:52 +01:00
> .header > .body {
padding-left: 10px;
}
}
2023-05-20 00:41:59 +02:00
> .clips,
2023-05-20 06:14:14 +02:00
> :deep(.user-card-mini),
> :deep(.reacted-users > *) {
2023-05-19 19:02:41 +02:00
padding-inline: 16px !important;
}
2023-05-21 03:46:33 +02:00
> :deep(.underline) {
2023-05-20 06:14:14 +02:00
padding-left: 16px !important;
}
}
&.max-width_300px {
font-size: 0.825em;
}
}
.muted {
padding: 8px;
text-align: center;
opacity: 0.7;
}
2023-05-19 19:02:41 +02:00
2023-05-20 00:41:59 +02:00
.clips {
// want to redesign at some point
2023-05-19 19:02:41 +02:00
padding: 24px 32px;
padding-top: 0;
> .item {
display: block;
padding: 16px;
// background: var(--buttonBg);
border: 1px solid var(--divider);
margin-bottom: var(--margin);
2023-05-20 00:41:59 +02:00
transition: background 0.2s;
&:hover,
&:focus-within {
2023-05-19 19:02:41 +02:00
background: var(--panelHighlight);
}
> .description {
padding: 8px 0;
}
> .user {
$height: 32px;
padding-top: 16px;
border-top: solid 0.5px var(--divider);
line-height: $height;
> .avatar {
width: $height;
height: $height;
}
}
}
}
</style>