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

893 lines
20 KiB
Vue
Raw Normal View History

<template>
2023-05-22 00:23:17 +02:00
<article
2023-05-04 07:13:13 +02:00
v-if="!muted.muted || muted.what === 'reply'"
2024-04-13 17:08:58 +02:00
:id="detailedView ? appearNote.id : undefined"
2023-04-08 02:01:42 +02:00
ref="el"
v-size="{ max: [450, 500] }"
class="wrpstxzv"
tabindex="-1"
2023-04-08 02:01:42 +02:00
:class="{
children: depth > 1,
singleStart: replies.length == 1,
firstColumn: depth == 1 && conversation,
}"
@contextmenu.stop="onContextmenu"
2023-04-08 02:01:42 +02:00
>
<div v-if="conversation && depth > 1" class="line"></div>
2023-06-02 23:53:28 +02:00
<div
class="main"
2023-06-01 05:35:25 +02:00
:style="{ cursor: expandOnNoteClick ? 'pointer' : '' }"
2023-09-02 01:27:33 +02:00
@click="noteClick"
2023-06-01 05:35:25 +02:00
>
2023-04-08 02:01:42 +02:00
<div class="avatar-container">
<MkAvatar class="avatar" :user="appearNote.user" />
<div
2024-04-20 14:55:47 +02:00
v-if="conversation == null || replies.length > 0"
2023-04-08 02:01:42 +02:00
class="line"
></div>
</div>
<div class="body">
2023-04-08 02:01:42 +02:00
<XNoteHeader class="header" :note="note" :mini="true" />
<div class="body">
<MkSubNoteContent
class="text"
:note="note"
2023-09-02 01:27:33 +02:00
:parent-id="parentId"
:conversation="conversation"
2023-09-02 01:27:33 +02:00
:detailed-view="detailedView"
2024-04-13 17:08:58 +02:00
@focusfooter="footerEl!.focus()"
/>
2023-04-08 02:01:42 +02:00
<div v-if="translating || translation" class="translation">
2024-04-13 17:08:58 +02:00
<MkLoading v-if="translating || translation == null" mini />
2023-04-08 02:01:42 +02:00
<div v-else class="translated">
<b
>{{
i18n.t("translatedFrom", {
x: translation.sourceLang,
})
}}:
</b>
<Mfm
:text="translation.text"
:author="appearNote.user"
2024-03-07 03:06:45 +01:00
:i="me"
:lang="targetLang"
2023-04-08 02:01:42 +02:00
:custom-emojis="appearNote.emojis"
/>
</div>
</div>
</div>
<footer ref="footerEl" class="footer" tabindex="-1">
2023-04-08 02:01:42 +02:00
<XReactionsViewer
v-if="enableEmojiReactions"
2023-04-08 02:01:42 +02:00
ref="reactionsViewer"
:note="appearNote"
/>
<button
v-tooltip.noDelay.bottom="i18n.ts.reply"
class="button _button"
@click.stop="reply()"
2023-04-08 02:01:42 +02:00
>
<i :class="icon('ph-arrow-u-up-left')"></i>
2023-04-08 02:01:42 +02:00
<template v-if="appearNote.repliesCount > 0">
<p class="count">{{ appearNote.repliesCount }}</p>
</template>
</button>
<XRenoteButton
ref="renoteButton"
class="button"
:note="appearNote"
:count="appearNote.renoteCount"
/>
<XStarButtonNoEmoji
v-if="!enableEmojiReactions"
class="button"
:note="appearNote"
:count="
Object.values(appearNote.reactions).reduce(
(partialSum, val) => partialSum + val,
2023-07-06 03:28:27 +02:00
0,
)
"
:reacted="appearNote.myReaction != null"
/>
2023-04-08 02:01:42 +02:00
<XStarButton
v-if="
enableEmojiReactions &&
appearNote.myReaction == null
"
2023-04-08 02:01:42 +02:00
ref="starButton"
class="button"
:note="appearNote"
/>
<button
v-if="
enableEmojiReactions &&
appearNote.myReaction == null
"
2023-04-08 02:01:42 +02:00
ref="reactButton"
v-tooltip.noDelay.bottom="i18n.ts.reaction"
class="button _button"
@click.stop="react()"
2023-04-08 02:01:42 +02:00
>
<i :class="icon('ph-smiley')"></i>
2023-04-08 02:01:42 +02:00
</button>
<button
v-if="
enableEmojiReactions &&
appearNote.myReaction != null
"
2023-04-08 02:01:42 +02:00
ref="reactButton"
2023-09-02 01:27:33 +02:00
v-tooltip.noDelay.bottom="i18n.ts.removeReaction"
2023-04-08 02:01:42 +02:00
class="button _button reacted"
@click.stop="undoReact(appearNote)"
2023-04-08 02:01:42 +02:00
>
<i :class="icon('ph-minus')"></i>
2023-04-08 02:01:42 +02:00
</button>
<XQuoteButton class="button" :note="appearNote" />
<button
v-if="
isSignedIn &&
isForeignLanguage &&
translation == null
"
2023-09-17 23:59:09 +02:00
v-tooltip.noDelay.bottom="i18n.ts.translate"
class="button _button"
@click.stop="translate"
>
<i :class="icon('ph-translate')"></i>
</button>
2023-04-08 02:01:42 +02:00
<button
ref="menuButton"
v-tooltip.noDelay.bottom="i18n.ts.more"
class="button _button"
@click.stop="menu()"
2023-04-08 02:01:42 +02:00
>
<i :class="icon('ph-dots-three-outline')"></i>
2023-04-08 02:01:42 +02:00
</button>
</footer>
</div>
</div>
2024-04-20 14:55:47 +02:00
<MkLoading v-if="conversationLoading" />
<template v-else-if="conversation">
<template
v-if="replyLevel < REPLY_LEVEL_UPPERBOUND && depth < DEPTH_UPPERBOUND"
>
<MkNoteSub
v-for="reply in replies.slice(0, REPLIES_LIMIT)"
:key="reply.id"
:note="reply"
class="reply"
:class="{ single: replies.length == 1 }"
:conversation="conversation"
:depth="replies.length == 1 ? depth : depth + 1"
:reply-level="replyLevel + 1"
:parent-id="appearNote.id"
:detailed-view="detailedView"
/>
<div v-if="hasMoreReplies" class="more">
<div class="line"></div>
<MkA class="text _link" :to="notePage(note)"
>{{ i18n.ts.continueThread }}
<i :class="icon('ph-caret-double-right')"></i
></MkA>
</div>
</template>
2023-04-08 02:01:42 +02:00
<div v-else-if="replies.length > 0" class="more">
<div class="line"></div>
<MkA class="text _link" :to="notePage(note)"
>{{ i18n.ts.continueThread }}
<i :class="icon('ph-caret-double-right')"></i
2023-04-08 02:01:42 +02:00
></MkA>
</div>
</template>
2023-05-22 00:23:17 +02:00
</article>
2023-05-04 07:13:13 +02:00
<div v-else class="muted" @click="muted.muted = false">
<I18n :src="softMuteReasonI18nSrc(muted.what)" tag="small">
<template #name>
<MkA
2023-05-05 03:38:56 +02:00
v-user-preview="note.userId"
2023-05-04 07:13:13 +02:00
class="name"
2023-05-05 03:38:56 +02:00
:to="userPage(note.user)"
2023-05-04 07:13:13 +02:00
>
2023-05-05 03:38:56 +02:00
<MkUserName :user="note.user" />
2023-05-04 07:13:13 +02:00
</MkA>
</template>
<template #reason>
<b class="_blur_text">{{ muted.matched.join(", ") }}</b>
</template>
</I18n>
</div>
</template>
<script lang="ts" setup>
2024-04-20 14:55:47 +02:00
import { computed, inject, ref, watch } from "vue";
2023-04-08 02:01:42 +02:00
import type { Ref } from "vue";
import type { entities } from "firefish-js";
2023-04-08 02:01:42 +02:00
import XNoteHeader from "@/components/MkNoteHeader.vue";
import MkSubNoteContent from "@/components/MkSubNoteContent.vue";
import XReactionsViewer from "@/components/MkReactionsViewer.vue";
import XStarButton from "@/components/MkStarButton.vue";
import XStarButtonNoEmoji from "@/components/MkStarButtonNoEmoji.vue";
2023-04-08 02:01:42 +02:00
import XRenoteButton from "@/components/MkRenoteButton.vue";
import XQuoteButton from "@/components/MkQuoteButton.vue";
import copyToClipboard from "@/scripts/copy-to-clipboard";
import { detectLanguage } from "@/scripts/language-utils";
import { url } from "@/config";
2023-04-08 02:01:42 +02:00
import { pleaseLogin } from "@/scripts/please-login";
import { getNoteMenu } from "@/scripts/get-note-menu";
import { getWordSoftMute } from "@/scripts/check-word-mute";
2023-04-08 02:01:42 +02:00
import { notePage } from "@/filters/note";
import { useRouter } from "@/router";
2023-05-05 01:14:16 +02:00
import { userPage } from "@/filters/user";
2023-04-08 02:01:42 +02:00
import * as os from "@/os";
import { reactionPicker } from "@/scripts/reaction-picker";
2024-03-16 16:49:12 +01:00
import { isSignedIn, me } from "@/me";
2023-04-08 02:01:42 +02:00
import { i18n } from "@/i18n";
import { useNoteCapture } from "@/scripts/use-note-capture";
import { defaultStore } from "@/store";
import { deepClone } from "@/scripts/clone";
import icon from "@/scripts/icon";
2024-04-13 17:08:58 +02:00
import type { NoteTranslation } from "@/types/note";
2022-09-15 23:43:57 +02:00
const router = useRouter();
2024-04-20 14:55:47 +02:00
const REPLIES_LIMIT = 10;
const REPLY_LEVEL_UPPERBOUND = 11;
const DEPTH_UPPERBOUND = 5;
2023-04-08 02:01:42 +02:00
const props = withDefaults(
defineProps<{
note: entities.Note;
conversation?: entities.Note[];
2024-04-20 14:55:47 +02:00
autoConversation?: boolean;
parentId?: string;
detailedView?: boolean;
2023-04-08 02:01:42 +02:00
// how many notes are in between this one and the note being viewed in detail
depth?: number;
// the actual reply level of this note within the conversation thread
replyLevel?: number;
2023-04-08 02:01:42 +02:00
}>(),
{
depth: 1,
replyLevel: 1,
2023-07-06 03:28:27 +02:00
},
2023-04-08 02:01:42 +02:00
);
2023-09-02 01:27:33 +02:00
const note = ref(deepClone(props.note));
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
2024-04-20 14:55:47 +02:00
const conversation = ref(props.conversation);
const conversationLoading = ref(false);
const replies = ref<entities.Note[]>([]);
const hasMoreReplies = ref(false);
function updateReplies() {
replies.value = (conversation.value ?? [])
.filter(
(item) =>
item.replyId === props.note.id || item.renoteId === props.note.id,
)
.reverse();
hasMoreReplies.value = replies.value.length >= REPLIES_LIMIT + 1;
}
watch(conversation, updateReplies, {
immediate: true,
});
if (props.autoConversation) {
if (note.value.repliesCount > 0 || note.value.renoteCount > 0) {
conversationLoading.value = true;
os.api("notes/children", {
noteId: note.value.id,
limit: REPLIES_LIMIT + 1,
depth: REPLY_LEVEL_UPPERBOUND + 1,
}).then((res) => {
conversation.value = res
.filter((n) => n.userId !== note.value.userId)
.reverse()
.concat(res.filter((n) => n.userId === note.value.userId));
conversationLoading.value = false;
});
} else {
conversation.value = [];
}
}
2023-04-08 02:01:42 +02:00
const isRenote =
note.value.renote != null &&
note.value.text == null &&
note.value.fileIds.length === 0 &&
note.value.poll == null;
2024-04-13 17:08:58 +02:00
const el = ref<HTMLElement | null>(null);
const footerEl = ref<HTMLElement | null>(null);
const menuButton = ref<HTMLElement>();
2024-04-13 17:08:58 +02:00
const starButton = ref<InstanceType<typeof XStarButton> | null>(null);
const renoteButton = ref<InstanceType<typeof XRenoteButton> | null>(null);
const reactButton = ref<HTMLElement | null>(null);
2023-09-02 01:27:33 +02:00
const appearNote = computed(() =>
isRenote ? (note.value.renote as entities.Note) : note.value,
2023-04-08 02:01:42 +02:00
);
const isDeleted = ref(false);
const muted = ref(
getWordSoftMute(
note.value,
2024-03-07 03:06:45 +01:00
me?.id,
defaultStore.state.mutedWords,
defaultStore.state.mutedLangs,
),
);
2024-04-13 17:08:58 +02:00
const translation = ref<NoteTranslation | null>(null);
const translating = ref(false);
const enableEmojiReactions = defaultStore.state.enableEmojiReactions;
2023-06-01 05:35:25 +02:00
const expandOnNoteClick = defaultStore.state.expandOnNoteClick;
const lang = localStorage.getItem("lang");
const translateLang = localStorage.getItem("translateLang");
const targetLang = (translateLang || lang || navigator.language)?.slice(0, 2);
const isForeignLanguage: boolean =
defaultStore.state.detectPostLanguage &&
appearNote.value.text != null &&
(() => {
const postLang = detectLanguage(appearNote.value.text);
return postLang !== "" && postLang !== targetLang;
})();
2023-09-06 19:38:42 +02:00
async function translate_(noteId, targetLang: string) {
return await os.api("notes/translate", {
2023-09-17 23:59:09 +02:00
noteId,
targetLang,
});
}
async function translate() {
if (translation.value != null) return;
translating.value = true;
translation.value = await translate_(
appearNote.value.id,
translateLang || lang || navigator.language,
);
// use UI language as the second translation language
if (
translateLang != null &&
lang != null &&
translateLang !== lang &&
(!translation.value ||
2024-03-28 17:26:52 +01:00
translation.value.sourceLang.toLowerCase() === translateLang.slice(0, 2))
)
translation.value = await translate_(appearNote.value.id, lang);
translating.value = false;
}
useNoteCapture({
rootEl: el,
note: appearNote,
isDeletedRef: isDeleted,
2024-04-20 14:55:47 +02:00
onReplied: (note) => {
if (hasMoreReplies.value === false) {
conversation.value = (conversation.value ?? []).concat([note]);
}
},
});
2024-04-13 17:08:58 +02:00
function reply(_viaKeyboard = false): void {
pleaseLogin();
2023-04-30 18:29:50 +02:00
os.post({
reply: appearNote.value,
2024-04-13 17:08:58 +02:00
// animation: !viaKeyboard,
2023-04-30 18:29:50 +02:00
}).then(() => {
focus();
});
}
2024-04-13 17:08:58 +02:00
function react(_viaKeyboard = false): void {
pleaseLogin();
blur();
2023-04-08 02:01:42 +02:00
reactionPicker.show(
2024-04-13 17:08:58 +02:00
reactButton.value!,
2023-04-08 02:01:42 +02:00
(reaction) => {
os.api("notes/reactions/create", {
noteId: appearNote.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", {
noteId: note.id,
});
}
const currentClipPage = inject<Ref<entities.Clip> | null>(
2023-04-08 02:01:42 +02:00
"currentClipPage",
2023-07-06 03:28:27 +02:00
null,
2023-04-08 02:01:42 +02:00
);
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,
currentClipPage,
}),
menuButton.value,
{
viaKeyboard,
2023-07-06 03:28:27 +02:00
},
2023-04-08 02:01:42 +02:00
).then(focus);
}
function onContextmenu(ev: MouseEvent): void {
2024-04-13 17:08:58 +02:00
const isLink = (el: HTMLElement | null) => {
if (el == null) return;
if (el.tagName === "A") return true;
if (el.parentElement) {
return isLink(el.parentElement);
}
};
2024-04-13 17:08:58 +02:00
if (isLink(ev.target as HTMLElement | null)) return;
if (window.getSelection()?.toString() !== "") return;
if (defaultStore.state.useReactionPickerForContextMenu) {
ev.preventDefault();
react();
} else {
os.contextMenu(
[
2023-05-26 04:47:10 +02:00
{
type: "label",
text: notePage(appearNote.value),
},
{
icon: `${icon("ph-browser")}`,
text: i18n.ts.openInWindow,
action: () => {
os.pageWindow(notePage(appearNote.value));
},
},
2024-04-13 09:37:23 +02:00
notePage(appearNote.value) !== location.pathname
2023-05-26 04:47:10 +02:00
? {
icon: `${icon("ph-arrows-out-simple")}`,
2023-05-26 04:47:10 +02:00
text: i18n.ts.showInPage,
action: () => {
2024-03-28 17:26:52 +01:00
router.push(notePage(appearNote.value), "forcePage");
2023-05-26 04:47:10 +02:00
},
2024-02-11 18:50:57 +01:00
}
2023-05-26 04:47:10 +02:00
: undefined,
null,
{
type: "a",
icon: `${icon("ph-arrow-square-out")}`,
text: i18n.ts.openInNewTab,
href: notePage(appearNote.value),
target: "_blank",
},
{
icon: `${icon("ph-link-simple")}`,
text: i18n.ts.copyLink,
action: () => {
copyToClipboard(`${url}${notePage(appearNote.value)}`);
},
},
note.value.user.host != null
2023-05-26 04:47:10 +02:00
? {
type: "a",
icon: `${icon("ph-arrow-square-up-right")}`,
2023-05-26 04:47:10 +02:00
text: i18n.ts.showOnRemote,
href: note.value.url ?? note.value.uri ?? "",
2023-05-26 04:47:10 +02:00
target: "_blank",
2024-02-11 18:50:57 +01:00
}
2023-05-26 04:47:10 +02:00
: undefined,
],
2023-07-06 03:28:27 +02:00
ev,
);
}
}
function focus() {
2024-04-13 17:08:58 +02:00
el.value!.focus();
}
function blur() {
2024-04-13 17:08:58 +02:00
el.value!.blur();
}
2024-04-13 17:08:58 +02:00
function noteClick(e: MouseEvent) {
if (document.getSelection()?.type === "Range" || !expandOnNoteClick) {
e.stopPropagation();
} else {
2023-04-08 02:01:42 +02:00
router.push(notePage(props.note));
}
}
</script>
<style lang="scss" scoped>
2020-03-20 11:19:28 +01:00
.wrpstxzv {
padding: 16px 32px;
outline: none;
&.children {
2023-02-20 02:12:01 +01:00
padding: 10px 0 0 var(--indent);
padding-left: var(--indent) !important;
font-size: 1em;
2022-11-24 22:55:23 +01:00
cursor: auto;
2023-06-13 06:04:36 +02:00
&.max-width_500px {
padding: 10px 0 0 8px;
}
}
> .main {
display: flex;
2023-01-07 00:58:52 +01:00
> .avatar-container {
2023-01-08 08:30:42 +01:00
margin-right: 8px;
2023-06-17 22:35:11 +02:00
z-index: 2;
2023-01-07 00:58:52 +01:00
> .avatar {
flex-shrink: 0;
display: block;
width: 38px;
height: 38px;
border-radius: 8px;
}
}
> .body {
2023-05-28 21:25:57 +02:00
position: relative;
flex: 1;
min-width: 0;
2023-02-26 02:22:17 +01:00
margin: 0 -200px;
padding: 0 200px;
2023-02-25 17:10:48 +01:00
overflow: clip;
2022-12-02 07:39:50 +01:00
@media (pointer: coarse) {
cursor: default;
}
2023-04-08 02:01:42 +02:00
> .header {
margin-bottom: 2px;
2022-11-24 22:55:23 +01:00
cursor: auto;
}
> .body {
> .translation {
border: solid 0.5px var(--divider);
border-radius: var(--radius);
padding: 12px;
margin-top: 8px;
}
}
> .footer {
position: relative;
z-index: 2;
display: flex;
flex-wrap: wrap;
2023-04-08 02:01:42 +02:00
2023-05-24 02:30:55 +02:00
> :deep(.button) {
position: relative;
margin: 0;
padding: 8px;
opacity: 0.7;
&:disabled {
opacity: 0.3 !important;
}
flex-grow: 1;
max-width: 3.5em;
width: max-content;
min-width: max-content;
2023-05-24 02:30:55 +02:00
height: auto;
2023-04-08 02:01:42 +02:00
transition: opacity 0.2s;
2023-05-24 02:30:55 +02:00
&::before {
content: "";
position: absolute;
inset: 0;
bottom: 2px;
background: var(--panel);
z-index: -1;
2023-05-24 03:02:52 +02:00
transition: background 0.2s;
2023-05-24 02:30:55 +02:00
}
&:first-of-type {
2023-04-08 02:01:42 +02:00
margin-left: -0.5em;
2023-05-24 02:30:55 +02:00
&::before {
border-radius: 100px 0 0 100px;
}
}
&:last-of-type {
&::before {
border-radius: 0 100px 100px 0;
}
}
&:hover {
color: var(--fgHighlighted);
}
2023-04-08 02:01:42 +02:00
2023-06-03 18:57:51 +02:00
> i {
display: inline !important;
}
> .count {
display: inline;
margin: 0 0 0 8px;
opacity: 0.7;
}
2023-04-08 02:01:42 +02:00
&.reacted {
color: var(--accent);
}
}
}
}
}
2023-05-24 02:30:55 +02:00
&.reply > .main {
margin-inline: -200px;
padding-inline: 200px;
&::before {
inset-inline: 176px !important;
}
}
2023-05-24 03:02:52 +02:00
&.reply,
&.reply-to {
2023-05-24 02:30:55 +02:00
> .main > .body {
margin-right: -24px;
padding-right: 24px;
margin-top: -12px;
padding-top: 12px;
margin-left: calc(0px - var(--avatarSize) - 32px);
padding-left: calc(var(--avatarSize) + 32px);
border-radius: var(--radius);
}
2023-02-26 02:22:17 +01:00
}
2023-06-05 19:11:14 +02:00
&.reply-to {
> .main > .body {
margin-left: calc(0px - var(--avatarSize) - 38px);
padding-left: calc(var(--avatarSize) + 38px);
margin-top: -16px;
padding-top: 16px;
}
}
2023-02-20 02:56:28 +01:00
&.reply {
--avatarSize: 38px;
.avatar-container {
margin-right: 8px !important;
}
2023-02-20 02:12:01 +01:00
}
2023-04-08 02:01:42 +02:00
> .reply,
> .more {
margin-top: 10px;
&.single {
padding: 0 !important;
2023-02-24 00:22:51 +01:00
> .line {
display: none;
}
}
}
> .more {
display: flex;
padding-block: 10px;
font-weight: 600;
> .line {
2023-05-24 02:30:55 +02:00
position: relative;
z-index: 2;
flex-grow: 0 !important;
margin-top: -10px !important;
margin-bottom: 10px !important;
margin-right: 10px !important;
&::before {
border-left-style: dashed !important;
border-bottom-left-radius: 100px !important;
}
}
i {
font-size: 1em !important;
vertical-align: middle !important;
}
2023-02-27 15:42:11 +01:00
a {
position: static;
&::before {
content: "";
position: absolute;
inset: 0;
}
&::after {
content: unset;
}
}
}
2023-01-07 00:58:52 +01:00
2023-04-08 02:01:42 +02:00
&.reply,
&.reply-to,
&.reply-to-more {
> .main:hover,
> .main:focus-within {
2023-02-25 06:39:43 +01:00
:deep(.footer .button) {
opacity: 1;
}
}
}
2023-04-08 02:01:42 +02:00
&.reply-to,
&.reply-to-more {
2023-01-28 22:10:45 +01:00
padding-bottom: 0;
&:first-child {
padding-top: 24px;
2023-01-28 22:10:45 +01:00
}
2023-02-20 02:12:01 +01:00
.line::before {
margin-bottom: -16px;
2023-02-20 00:31:16 +01:00
}
}
2023-04-08 02:01:42 +02:00
2023-02-20 00:31:16 +01:00
// Reply Lines
2023-04-08 02:01:42 +02:00
&.reply,
&.reply-to,
&.reply-to-more {
2023-02-24 00:22:51 +01:00
--indent: calc(var(--avatarSize) - 5px);
2023-02-20 00:31:16 +01:00
> .main {
> .avatar-container {
display: flex;
flex-direction: column;
align-items: center;
margin-right: 14px;
width: var(--avatarSize);
2023-02-20 00:31:16 +01:00
> .avatar {
width: var(--avatarSize);
height: var(--avatarSize);
margin: 0;
}
2023-01-07 00:58:52 +01:00
}
2023-02-20 00:31:16 +01:00
}
.line {
2023-02-24 00:22:51 +01:00
position: relative;
2023-05-24 02:30:55 +02:00
z-index: 2;
2023-02-20 00:31:16 +01:00
width: var(--avatarSize);
display: flex;
flex-grow: 1;
2023-02-24 00:22:51 +01:00
margin-bottom: -10px;
pointer-events: none;
2023-06-19 03:36:34 +02:00
opacity: 0.25;
2023-02-20 00:31:16 +01:00
&::before {
content: "";
2023-02-24 00:22:51 +01:00
position: absolute;
2023-06-17 22:35:11 +02:00
border-left: 2px solid currentColor;
2023-02-24 00:22:51 +01:00
margin-left: calc((var(--avatarSize) / 2) - 1px);
width: calc(var(--indent) / 2);
inset-block: 0;
min-height: 8px;
2023-01-07 00:58:52 +01:00
}
}
2023-02-20 00:31:16 +01:00
}
2023-04-08 02:01:42 +02:00
&.reply-to,
&.reply-to-more {
2023-02-24 00:22:51 +01:00
> .main > .avatar-container > .line {
margin-bottom: 0px !important;
}
}
2023-04-08 02:01:42 +02:00
&.single,
&.singleStart {
2023-02-24 00:22:51 +01:00
> .main > .avatar-container > .line {
margin-bottom: -10px !important;
2023-02-20 02:12:01 +01:00
}
}
2023-04-08 02:01:42 +02:00
.reply.children:not(:last-child) {
// Line that goes through multiple replies
2023-02-20 00:31:16 +01:00
position: relative;
> .line {
position: absolute;
top: 0;
left: 0;
bottom: 0;
}
2023-01-07 00:58:52 +01:00
}
2023-02-20 02:12:01 +01:00
// Reply line connectors
.reply.children:not(.single) {
position: relative;
> .line {
position: absolute;
2023-05-24 02:30:55 +02:00
z-index: 2;
2023-02-20 02:12:01 +01:00
left: 0;
2023-02-24 00:22:51 +01:00
top: 0;
2023-06-19 03:36:34 +02:00
opacity: 0.25;
2023-02-20 02:12:01 +01:00
&::after {
content: "";
position: absolute;
2023-06-17 22:35:11 +02:00
border-left: 2px solid currentColor;
border-bottom: 2px solid currentColor;
2023-02-20 19:18:37 +01:00
margin-left: calc((var(--avatarSize) / 2) - 1px);
width: calc(var(--indent) / 2);
2023-02-24 00:22:51 +01:00
height: calc((var(--avatarSize) / 2));
border-bottom-left-radius: calc(var(--indent) / 2);
top: 8px;
2023-02-20 02:12:01 +01:00
}
}
2023-02-24 00:22:51 +01:00
&:not(:last-child) > .line::after {
mask: linear-gradient(to right, transparent 2px, black 2px);
-webkit-mask: linear-gradient(to right, transparent 2px, black 2px);
}
2023-02-20 02:12:01 +01:00
}
// End Reply Divider
.children > .main:last-child {
padding-bottom: 1em;
&::before {
bottom: 1em;
}
// &::after {
// content: "";
// border-top: 1px solid var(--X13);
// position: absolute;
// bottom: 0;
// margin-left: calc(var(--avatarSize) + 12px);
// inset-inline: 0;
// }
}
&.firstColumn > .children:last-child > .main {
padding-bottom: 0 !important;
2023-04-08 02:01:42 +02:00
&::before {
bottom: 0 !important;
}
// &::after { content: unset }
}
2023-01-07 00:58:52 +01:00
2023-02-26 03:22:23 +01:00
&.max-width_500px {
2023-06-13 06:04:36 +02:00
padding: 14px 16px;
2023-02-26 03:22:23 +01:00
:not(.reply) > & {
.reply {
--avatarSize: 24px;
--indent: calc(var(--avatarSize) - 4px);
}
2023-03-16 00:53:22 +01:00
}
2023-03-16 01:26:13 +01:00
&.firstColumn {
2023-04-08 02:01:42 +02:00
> .main,
> .line,
> .children:not(.single) > .line {
2023-03-16 00:53:22 +01:00
--avatarSize: 35px;
--indent: 35px;
}
> .children:not(.single) {
padding-left: 28px !important;
2023-02-26 03:22:23 +01:00
}
}
2023-06-13 06:04:36 +02:00
&.reply-to {
--avatarSize: 46px;
2023-02-24 17:55:51 +01:00
padding: 14px 16px;
2023-01-28 22:10:45 +01:00
padding-top: 14px !important;
padding-bottom: 0 !important;
margin-bottom: 0 !important;
}
2023-01-07 00:58:52 +01:00
> .main > .avatar-container {
margin-right: 10px;
}
2023-05-24 02:49:56 +02:00
&:first-child > .main > .body {
margin-top: -20px;
padding-top: 22px;
}
2023-01-07 00:58:52 +01:00
}
}
2023-05-04 07:13:13 +02:00
.muted {
padding: 8px;
text-align: center;
opacity: 0.7;
}
</style>