<template> <header class="mvcprjjd sidebar" :class="{ iconOnly }"> <div class="body"> <div class="top"> <div class="banner" :user="$i" :style="{ backgroundImage: `url(${$i.bannerUrl})` }" ></div> <button v-click-anime v-tooltip.noDelay.right=" `${i18n.ts.account}: @${$i.username}` " class="item _button account" @click="openAccountMenu" > <MkAvatar :user="$i" class="icon" disableLink /><!-- <MkAcct class="text" :user="$i"/> --> </button> </div> <nav class="middle"> <MkA v-click-anime v-tooltip.noDelay.right="i18n.ts.timeline" class="item index" active-class="active" to="/" exact > <i class="icon ph-house ph-bold ph-fw ph-lg"></i ><span class="text">{{ i18n.ts.timeline }}</span> </MkA> <template v-for="item in menu"> <div v-if="item === '-'" class="divider"></div> <component :is="navbarItemDef[item].to ? 'MkA' : 'button'" v-else-if=" navbarItemDef[item] && navbarItemDef[item].show !== false " v-click-anime v-tooltip.noDelay.right=" i18n.ts[navbarItemDef[item].title] " class="item _button" :class="[item, { active: navbarItemDef[item].active }]" active-class="active" :to="navbarItemDef[item].to" v-on=" navbarItemDef[item].action ? { click: navbarItemDef[item].action } : {} " > <i class="icon ph-fw ph-lg" :class="navbarItemDef[item].icon" ></i ><span class="text">{{ i18n.ts[navbarItemDef[item].title] }}</span> <span v-if="navbarItemDef[item].indicated" class="indicator" ><i class="icon ph-circle ph-fill"></i ></span> </component> </template> <div class="divider"></div> <MkA v-if="$i.isAdmin || $i.isModerator" v-click-anime v-tooltip.noDelay.right="i18n.ts.controlPanel" class="item _button" active-class="active" to="/admin" > <span v-if=" thereIsUnresolvedAbuseReport || noMaintainerInformation || noBotProtection || noEmailServer || updateAvailable " class="indicator" ><i class="icon ph-circle ph-fill"></i> </span ><i class="icon ph-door ph-bold ph-fw ph-lg"></i ><span class="text">{{ i18n.ts.controlPanel }}</span> </MkA> <div class="divider" v-if="$i.isAdmin || $i.isModerator"></div> <button v-click-anime v-tooltip.noDelay.right="i18n.ts.more" class="item _button" @click="more" > <i class="icon ph-dots-three-outline ph-bold ph-fw ph-lg" ></i ><span class="text">{{ i18n.ts.more }}</span> <span v-if="otherMenuItemIndicated" class="indicator" ><i class="icon ph-circle ph-fill"></i ></span> </button> <MkA v-click-anime v-tooltip.noDelay.right="i18n.ts.settings" class="item _button" active-class="active" to="/settings" > <i class="icon ph-gear-six ph-bold ph-fw ph-lg"></i ><span class="text">{{ i18n.ts.settings }}</span> </MkA> </nav> <div class="bottom"> <button v-tooltip.noDelay.right="i18n.ts.note" class="item _button post" data-cy-open-post-form @click="os.post" > <i class="icon ph-pencil ph-bold ph-fw ph-lg"></i ><span class="text">{{ i18n.ts.note }}</span> </button> <!-- <button v-click-anime v-tooltip.noDelay.right="$instance.name ?? i18n.ts.instance" class="item _button instance" @click="openInstanceMenu"> <img :src="$instance.iconUrl || $instance.faviconUrl || '/favicon.ico'" alt="" class="icon"/> </button> --> <!-- <button v-click-anime v-tooltip.noDelay.right="`${i18n.ts.account}: @${$i.username}`" class="item _button account" @click="openAccountMenu"> <MkAvatar :user="$i" class="account"/><MkAcct class="text" :user="$i"/> </button> --> </div> </div> </header> </template> <script lang="ts" setup> import { computed, defineAsyncComponent, ref, watch } from "vue"; import * as os from "@/os"; import { navbarItemDef } from "@/navbar"; import { $i, openAccountMenu as openAccountMenu_ } from "@/account"; import { defaultStore } from "@/store"; import { i18n } from "@/i18n"; import { instance } from "@/instance"; import { version } from "@/config"; const isEmpty = (x: string | null) => x == null || x === ""; const iconOnly = ref(false); const menu = computed(() => defaultStore.state.menu); const otherMenuItemIndicated = computed(() => { for (const def in navbarItemDef) { if (menu.value.includes(def)) continue; if (navbarItemDef[def].indicated) return true; } return false; }); const calcViewState = () => { iconOnly.value = window.innerWidth <= 1279 || defaultStore.state.menuDisplay === "sideIcon"; }; calcViewState(); matchMedia("(max-width: 1279px)").onchange = (mql) => calcViewState(); watch(defaultStore.reactiveState.menuDisplay, () => { calcViewState(); }); let noMaintainerInformation = isEmpty(instance.maintainerName) || isEmpty(instance.maintainerEmail); let noBotProtection = !instance.disableRegistration && !instance.enableHcaptcha && !instance.enableRecaptcha; let noEmailServer = !instance.enableEmail; let thereIsUnresolvedAbuseReport = $ref(false); let updateAvailable = $ref(false); if ($i?.isAdmin) { os.api("admin/abuse-user-reports", { state: "unresolved", limit: 1, }).then((reports) => { if (reports?.length > 0) thereIsUnresolvedAbuseReport = true; }); if (defaultStore.state.showAdminUpdates) { os.api("latest-version").then((res) => { if (!res?.tag_name) { updateAvailable = false; return; } const tag = res.tag_name as string; if (tag === `v${version}`) { updateAvailable = false; return; } const tagDate = tag.includes('-') ? tag.substring(0, tag.indexOf('-')) : tag; const versionDate = version.includes('-') ? version.substring(0, version.indexOf('-')) : version; if (tagDate < versionDate) { updateAvailable = false; return; } updateAvailable = true; return; }); } } function openAccountMenu(ev: MouseEvent) { openAccountMenu_( { withExtraOperation: true, }, ev, ); } function more(ev: MouseEvent) { os.popup( defineAsyncComponent(() => import("@/components/MkLaunchPad.vue")), { src: ev.currentTarget ?? ev.target, }, {}, "closed", ); } </script> <style lang="scss" scoped> .mvcprjjd { $nav-width: 250px; $nav-icon-only-width: 80px; flex: 0 0 $nav-width; width: $nav-width; box-sizing: border-box; > .body { position: sticky; top: 0; width: $nav-icon-only-width; // ほんとは単に 100vh と書きたいところだが... https://css-tricks.com/the-trick-to-viewport-units-on-mobile/ height: calc(var(--vh, 1vh) * 100); box-sizing: border-box; overflow: auto; overflow-x: clip; #iceshrimp_app > :not(.wallpaper) & { background: var(--navBg); } #iceshrimp_app > .wallpaper:not(.centered) & { border-right: 1px solid var(--divider); } contain: strict; display: flex; flex-direction: column; } &:not(.iconOnly) { > .body { margin-left: -200px; padding-left: 200px; box-sizing: content-box; width: $nav-width; > .top { position: relative; z-index: 1; padding: 2rem 0; > .banner { position: absolute; top: 0; left: 0; width: 100%; height: 100%; background-size: cover; background-position: center center; -webkit-mask-image: linear-gradient(var(--gradient)); mask-image: linear-gradient(var(--gradient)); } > .account { position: relative; display: block; text-align: center; width: 100%; > .icon { display: inline-block; width: 55px; aspect-ratio: 1; } } } > .bottom { padding: 20px 0; > .post { position: relative; width: 100%; height: 40px; color: var(--fgOnAccent); font-weight: bold; text-align: left; display: flex; align-items: center; &:before { content: ""; display: block; width: calc(100% - 38px); height: 100%; margin: auto; position: absolute; top: 0; left: 0; right: 0; bottom: 0; border-radius: 999px; background: linear-gradient( -45deg, var(--buttonGradateA), var(--buttonGradateB) ); } &:hover, &:focus-within, &.active { &:before { background: var(--accentLighten); } } > .icon, > .text { position: relative; left: 3rem; color: var(--fgOnAccent); transform: translateY(0em); } > .text { margin-left: 1rem; } } > .instance { position: relative; display: block; text-align: center; width: 100%; > .icon { display: inline-block; width: 32px !important; aspect-ratio: 1; margin-top: 1rem; } } > .help { position: relative; display: block; text-align: center; width: 100%; margin-top: 1rem; color: var(--navFg); > .icon { display: inline-block; width: 38px; aspect-ratio: 1; } } } > .middle { flex: 0.1; > a { text-decoration: none; } > .divider { margin: 16px 16px; border-top: solid 0.5px var(--divider); } > .item { position: relative; display: flex; align-items: center; padding-left: 30px; line-height: 2.85rem; margin-bottom: 0.5rem; white-space: nowrap; width: 100%; text-align: left; box-sizing: border-box; color: var(--navFg); > .icon { position: relative; width: 32px; margin-right: 8px; } > .indicator { position: absolute; top: 0; left: 20px; color: var(--navIndicator); font-size: 8px; } > .text { position: relative; font-size: 0.9em; overflow: hidden; text-overflow: ellipsis; } &:hover, &:focus-within, &.active { &:before { content: ""; display: block; width: calc(100% - 34px); height: 100%; margin: auto; position: absolute; top: 0; left: 0; right: 0; bottom: 0; border-radius: 999px; } } &:hover, &:focus-within { &:before { background: var(--panelHighlight); } } &.active { color: var(--panelActiveFg); transition: all 0.4s ease; &:before { background: var(--panelActiveBg); } } } } } } &.iconOnly { flex: 0 0 $nav-icon-only-width; width: $nav-icon-only-width; > .body { width: $nav-icon-only-width; > .top { padding: 2rem 0; > .account { display: block; text-align: center; width: 100%; > .icon { display: inline-block; width: 40px; aspect-ratio: 1; transform: translateY(0em); } } } > .bottom { padding: 20px 0; > .post { display: block; position: relative; width: 100%; height: 52px; margin-bottom: 16px; text-align: center; &:before { content: ""; display: block; position: absolute; top: 0; left: 0; right: 0; bottom: 0; margin: auto; width: 52px; aspect-ratio: 1/1; border-radius: 100%; background: linear-gradient( -45deg, var(--buttonGradateA), var(--buttonGradateB) ); } &:hover, &:focus-within, &.active { &:before { background: var(--accentLighten); } } > .icon { position: relative; color: var(--fgOnAccent); } > .text { display: none; } } > .help { position: relative; display: block; text-align: center; width: 100%; margin-top: 1rem; color: var(--navFg); > .icon { display: inline-block; width: 38px; aspect-ratio: 1; } } > .instance { position: relative; display: block; text-align: center; width: 100%; > .icon { display: inline-block; width: 32px !important; aspect-ratio: 1; } } } > .middle { flex: 0.1; > a { text-decoration: none; } > .divider { margin: 8px auto; width: calc(100% - 32px); border-top: solid 0.5px var(--divider); } > .item { display: block; position: relative; padding: 1.1rem 0; margin-bottom: 0.2rem; width: 100%; text-align: center; > .icon { display: block; margin: 0 auto; opacity: 0.7; transform: translateY(0em); } > .text { display: none; } > .indicator { position: absolute; top: 6px; left: 24px; color: var(--navIndicator); font-size: 8px; } &:hover, &:focus-within, &.active { text-decoration: none; color: var(--panelActiveFg); transition: all 0.4s ease; &:before { content: ""; display: block; height: 100%; aspect-ratio: 1; margin: auto; position: absolute; top: 0; left: 0; right: 0; bottom: 0; border-radius: 999px; } > .icon, > .text { opacity: 1; } } &:hover, &:focus-within { &:before { background: var(--panelHighlight); } } &.active { &:before { background: var(--panelActiveBg); } } } } } } .item { outline: none; &:focus-visible:before { outline: auto; } } } </style>