ae5d052274
to keep things manageable i merged a lot of one off values into just a handful of common sizes, so some parts of the ui will look different than upstream even with the "Misskey" rounding mode
292 lines
7.4 KiB
Vue
292 lines
7.4 KiB
Vue
<!--
|
|
SPDX-FileCopyrightText: syuilo and other misskey contributors
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
-->
|
|
|
|
<template>
|
|
<div class="mk-app">
|
|
<div v-if="!narrow && !root" class="side">
|
|
<div class="banner" :style="{ backgroundImage: instance.backgroundImageUrl ? `url(${ instance.backgroundImageUrl })` : 'none' }"></div>
|
|
<div class="dashboard">
|
|
<MkVisitorDashboard/>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="main">
|
|
<div v-if="!root" class="header">
|
|
<div v-if="narrow === false" class="wide">
|
|
<MkA to="/" class="link" activeClass="active"><i class="ph-house ph-bold ph-lg icon"></i> {{ i18n.ts.home }}</MkA>
|
|
<MkA v-if="isTimelineAvailable" to="/timeline" class="link" activeClass="active"><i class="ph-chat-text ph-bold ph-lg icon"></i> {{ i18n.ts.timeline }}</MkA>
|
|
<MkA to="/explore" class="link" activeClass="active"><i class="ph-hash ph-bold ph-lg icon"></i> {{ i18n.ts.explore }}</MkA>
|
|
<MkA to="/channels" class="link" activeClass="active"><i class="ph-television ph-bold ph-lg icon"></i> {{ i18n.ts.channel }}</MkA>
|
|
</div>
|
|
<div v-else-if="narrow === true" class="narrow">
|
|
<button class="menu _button" @click="showMenu = true">
|
|
<i class="ph-list ph-bold ph-lg-2 icon"></i>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<div class="contents">
|
|
<main v-if="!root" style="container-type: inline-size;">
|
|
<RouterView/>
|
|
</main>
|
|
<main v-else>
|
|
<RouterView/>
|
|
</main>
|
|
</div>
|
|
</div>
|
|
|
|
<Transition :name="'tray-back'">
|
|
<div
|
|
v-if="showMenu"
|
|
class="menu-back _modalBg"
|
|
@click="showMenu = false"
|
|
@touchstart.passive="showMenu = false"
|
|
></div>
|
|
</Transition>
|
|
|
|
<Transition :name="'tray'">
|
|
<div v-if="showMenu" class="menu">
|
|
<MkA to="/" class="link" activeClass="active"><i class="ph-house ph-bold ph-lg icon"></i>{{ i18n.ts.home }}</MkA>
|
|
<MkA v-if="isTimelineAvailable" to="/timeline" class="link" activeClass="active"><i class="ph-chat-text ph-bold ph-lg icon"></i>{{ i18n.ts.timeline }}</MkA>
|
|
<MkA to="/explore" class="link" activeClass="active"><i class="ph-hash ph-bold ph-lg icon"></i>{{ i18n.ts.explore }}</MkA>
|
|
<MkA to="/announcements" class="link" activeClass="active"><i class="ph-megaphone ph-bold ph-lg icon"></i>{{ i18n.ts.announcements }}</MkA>
|
|
<MkA to="/channels" class="link" activeClass="active"><i class="ph-television ph-bold ph-lg icon"></i>{{ i18n.ts.channel }}</MkA>
|
|
<div class="divider"></div>
|
|
<MkA to="/pages" class="link" activeClass="active"><i class="ph-newspaper ph-bold ph-lg icon"></i>{{ i18n.ts.pages }}</MkA>
|
|
<MkA to="/play" class="link" activeClass="active"><i class="ph-play ph-bold pg-lg icon"></i>Play</MkA>
|
|
<MkA to="/gallery" class="link" activeClass="active"><i class="ph-images-square ph-bold pg-lgs icon"></i>{{ i18n.ts.gallery }}</MkA>
|
|
<div class="action">
|
|
<button class="_buttonPrimary" @click="signup()">{{ i18n.ts.signup }}</button>
|
|
<button class="_button" @click="signin()">{{ i18n.ts.login }}</button>
|
|
</div>
|
|
</div>
|
|
</Transition>
|
|
</div>
|
|
<XCommon/>
|
|
</template>
|
|
|
|
<script lang="ts" setup>
|
|
import { ComputedRef, onMounted, provide } from 'vue';
|
|
import XCommon from './_common_/common.vue';
|
|
import { host, instanceName } from '@/config.js';
|
|
import * as os from '@/os.js';
|
|
import { instance } from '@/instance.js';
|
|
import XSigninDialog from '@/components/MkSigninDialog.vue';
|
|
import XSignupDialog from '@/components/MkSignupDialog.vue';
|
|
import { ColdDeviceStorage, defaultStore } from '@/store.js';
|
|
import { mainRouter } from '@/router.js';
|
|
import { PageMetadata, provideMetadataReceiver } from '@/scripts/page-metadata.js';
|
|
import { i18n } from '@/i18n.js';
|
|
import MkVisitorDashboard from '@/components/MkVisitorDashboard.vue';
|
|
|
|
const DESKTOP_THRESHOLD = 1100;
|
|
|
|
let pageMetadata = $ref<null | ComputedRef<PageMetadata>>();
|
|
|
|
provide('router', mainRouter);
|
|
provideMetadataReceiver((info) => {
|
|
pageMetadata = info;
|
|
if (pageMetadata.value) {
|
|
document.title = `${pageMetadata.value.title} | ${instanceName}`;
|
|
}
|
|
});
|
|
|
|
const announcements = {
|
|
endpoint: 'announcements',
|
|
limit: 10,
|
|
};
|
|
|
|
const isTimelineAvailable = $ref(instance.policies?.ltlAvailable || instance.policies?.gtlAvailable);
|
|
|
|
let showMenu = $ref(false);
|
|
let isDesktop = $ref(window.innerWidth >= DESKTOP_THRESHOLD);
|
|
let narrow = $ref(window.innerWidth < 1280);
|
|
let meta = $ref();
|
|
|
|
const keymap = $computed(() => {
|
|
return {
|
|
'd': () => {
|
|
if (ColdDeviceStorage.get('syncDeviceDarkMode')) return;
|
|
defaultStore.set('darkMode', !defaultStore.state.darkMode);
|
|
},
|
|
's': () => {
|
|
mainRouter.push('/search');
|
|
},
|
|
};
|
|
});
|
|
|
|
const root = $computed(() => mainRouter.currentRoute.value.name === 'index');
|
|
|
|
os.api('meta', { detail: true }).then(res => {
|
|
meta = res;
|
|
});
|
|
|
|
function signin() {
|
|
os.popup(XSigninDialog, {
|
|
autoSet: true,
|
|
}, {}, 'closed');
|
|
}
|
|
|
|
function signup() {
|
|
os.popup(XSignupDialog, {
|
|
autoSet: true,
|
|
}, {}, 'closed');
|
|
}
|
|
|
|
onMounted(() => {
|
|
if (!isDesktop) {
|
|
window.addEventListener('resize', () => {
|
|
if (window.innerWidth >= DESKTOP_THRESHOLD) isDesktop = true;
|
|
}, { passive: true });
|
|
}
|
|
});
|
|
|
|
defineExpose({
|
|
showMenu: $$(showMenu),
|
|
});
|
|
</script>
|
|
|
|
<style>
|
|
.github-corner:hover .octo-arm{animation:octocat-wave 560ms ease-in-out}@keyframes octocat-wave{0%,100%{transform:rotate(0)}20%,60%{transform:rotate(-25deg)}40%,80%{transform:rotate(10deg)}}@media (max-width:500px){.github-corner:hover .octo-arm{animation:none}.github-corner .octo-arm{animation:octocat-wave 560ms ease-in-out}}
|
|
</style>
|
|
|
|
<style lang="scss" scoped>
|
|
.tray-enter-active,
|
|
.tray-leave-active {
|
|
opacity: 1;
|
|
transform: translateX(0);
|
|
transition: transform 300ms cubic-bezier(0.23, 1, 0.32, 1), opacity 300ms cubic-bezier(0.23, 1, 0.32, 1);
|
|
}
|
|
.tray-enter-from,
|
|
.tray-leave-active {
|
|
opacity: 0;
|
|
transform: translateX(-240px);
|
|
}
|
|
|
|
.tray-back-enter-active,
|
|
.tray-back-leave-active {
|
|
opacity: 1;
|
|
transition: opacity 300ms cubic-bezier(0.23, 1, 0.32, 1);
|
|
}
|
|
.tray-back-enter-from,
|
|
.tray-back-leave-active {
|
|
opacity: 0;
|
|
}
|
|
|
|
.mk-app {
|
|
display: flex;
|
|
min-height: 100vh;
|
|
|
|
> .side {
|
|
position: sticky;
|
|
top: 0;
|
|
left: 0;
|
|
width: 500px;
|
|
height: 100vh;
|
|
background: var(--accent);
|
|
z-index: 1;
|
|
|
|
> .banner {
|
|
position: absolute;
|
|
top: 0;
|
|
left: 0;
|
|
width: 100%;
|
|
aspect-ratio: 1.5;
|
|
background-position: center;
|
|
background-size: cover;
|
|
-webkit-mask-image: linear-gradient(rgba(0, 0, 0, 1.0), transparent);
|
|
mask-image: linear-gradient(rgba(0, 0, 0, 1.0), transparent);
|
|
}
|
|
|
|
> .dashboard {
|
|
position: relative;
|
|
padding: 32px;
|
|
box-sizing: border-box;
|
|
max-height: 100%;
|
|
overflow: auto;
|
|
}
|
|
}
|
|
|
|
> .main {
|
|
flex: 1;
|
|
min-width: 0;
|
|
|
|
> .header {
|
|
background: var(--panel);
|
|
position: relative;
|
|
z-index: 1;
|
|
|
|
> .wide {
|
|
line-height: 50px;
|
|
padding: 0 16px;
|
|
|
|
> .link {
|
|
padding: 0 16px;
|
|
}
|
|
}
|
|
|
|
> .narrow {
|
|
> .menu {
|
|
padding: 16px;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
> .menu-back {
|
|
position: fixed;
|
|
z-index: 1001;
|
|
top: 0;
|
|
left: 0;
|
|
width: 100vw;
|
|
height: 100vh;
|
|
}
|
|
|
|
> .menu {
|
|
position: fixed;
|
|
z-index: 1001;
|
|
top: 0;
|
|
left: 0;
|
|
width: 240px;
|
|
height: 100vh;
|
|
background: var(--panel);
|
|
|
|
> .link {
|
|
display: block;
|
|
padding: 16px;
|
|
|
|
> .icon {
|
|
margin-right: 1em;
|
|
}
|
|
}
|
|
|
|
> .divider {
|
|
margin: 8px auto;
|
|
width: calc(100% - 32px);
|
|
border-top: solid 0.5px var(--divider);
|
|
}
|
|
|
|
> .action {
|
|
padding: 16px;
|
|
|
|
> button {
|
|
display: block;
|
|
width: 100%;
|
|
padding: 10px;
|
|
box-sizing: border-box;
|
|
text-align: center;
|
|
border-radius: var(--radius-ellipse);
|
|
|
|
&._button {
|
|
background: var(--panel);
|
|
}
|
|
|
|
&:first-child {
|
|
margin-bottom: 16px;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
</style>
|