mirror of
https://iceshrimp.dev/iceshrimp/iceshrimp
synced 2025-01-03 04:02:52 +09:00
300 lines
7.0 KiB
Vue
300 lines
7.0 KiB
Vue
<template>
|
|
<div class="mk-admin" :class="{ isMobile }">
|
|
<header v-show="isMobile">
|
|
<button class="nav" @click="navOpend = true"><fa icon="bars"/></button>
|
|
<span>MisskeyMyAdmin</span>
|
|
</header>
|
|
<div class="nav-backdrop"
|
|
v-if="navOpend && isMobile"
|
|
@click="navOpend = false"
|
|
@touchstart="navOpend = false"
|
|
></div>
|
|
<nav v-show="navOpend">
|
|
<div class="mi">
|
|
<img svg-inline src="../assets/header-icon.svg"/>
|
|
</div>
|
|
<div class="me">
|
|
<img class="avatar" :src="$store.state.i.avatarUrl" alt="avatar"/>
|
|
<p class="name"><mk-user-name :user="$store.state.i"/></p>
|
|
</div>
|
|
<ul>
|
|
<li @click="nav('dashboard')" :class="{ active: page == 'dashboard' }"><fa icon="home" fixed-width/>{{ $t('dashboard') }}</li>
|
|
<li @click="nav('instance')" :class="{ active: page == 'instance' }"><fa icon="cog" fixed-width/>{{ $t('instance') }}</li>
|
|
<li @click="nav('queue')" :class="{ active: page == 'queue' }"><fa :icon="faTasks" fixed-width/>{{ $t('queue') }}</li>
|
|
<li @click="nav('logs')" :class="{ active: page == 'logs' }"><fa :icon="faStream" fixed-width/>{{ $t('logs') }}</li>
|
|
<li @click="nav('db')" :class="{ active: page == 'db' }"><fa :icon="faDatabase" fixed-width/>{{ $t('db') }}</li>
|
|
<li @click="nav('moderators')" :class="{ active: page == 'moderators' }"><fa :icon="faHeadset" fixed-width/>{{ $t('moderators') }}</li>
|
|
<li @click="nav('users')" :class="{ active: page == 'users' }"><fa icon="users" fixed-width/>{{ $t('users') }}</li>
|
|
<li @click="nav('drive')" :class="{ active: page == 'drive' }"><fa icon="cloud" fixed-width/>{{ $t('@.drive') }}</li>
|
|
<li @click="nav('federation')" :class="{ active: page == 'federation' }"><fa :icon="faGlobe" fixed-width/>{{ $t('federation') }}</li>
|
|
<li @click="nav('emoji')" :class="{ active: page == 'emoji' }"><fa :icon="faGrin" fixed-width/>{{ $t('emoji') }}</li>
|
|
<li @click="nav('announcements')" :class="{ active: page == 'announcements' }"><fa icon="broadcast-tower" fixed-width/>{{ $t('announcements') }}</li>
|
|
<li @click="nav('abuse')" :class="{ active: page == 'abuse' }"><fa :icon="faExclamationCircle" fixed-width/>{{ $t('abuse') }}</li>
|
|
</ul>
|
|
<div class="back-to-misskey">
|
|
<a href="/"><fa :icon="faArrowLeft"/> {{ $t('back-to-misskey') }}</a>
|
|
</div>
|
|
<div class="version">
|
|
<small>Misskey {{ version }}</small>
|
|
</div>
|
|
</nav>
|
|
<main>
|
|
<div class="page">
|
|
<div v-if="page == 'dashboard'"><x-dashboard/></div>
|
|
<div v-if="page == 'instance'"><x-instance/></div>
|
|
<div v-if="page == 'queue'"><x-queue/></div>
|
|
<div v-if="page == 'logs'"><x-logs/></div>
|
|
<div v-if="page == 'db'"><x-db/></div>
|
|
<div v-if="page == 'moderators'"><x-moderators/></div>
|
|
<div v-if="page == 'users'"><x-users/></div>
|
|
<div v-if="page == 'emoji'"><x-emoji/></div>
|
|
<div v-if="page == 'announcements'"><x-announcements/></div>
|
|
<div v-if="page == 'drive'"><x-drive/></div>
|
|
<div v-if="page == 'federation'"><x-federation/></div>
|
|
<div v-if="page == 'abuse'"><x-abuse/></div>
|
|
</div>
|
|
</main>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts">
|
|
import Vue from 'vue';
|
|
import i18n from '../../i18n';
|
|
import { version } from '../../config';
|
|
import XDashboard from './dashboard.vue';
|
|
import XInstance from './instance.vue';
|
|
import XQueue from './queue.vue';
|
|
import XLogs from './logs.vue';
|
|
import XDb from './db.vue';
|
|
import XModerators from './moderators.vue';
|
|
import XEmoji from './emoji.vue';
|
|
import XAnnouncements from './announcements.vue';
|
|
import XUsers from './users.vue';
|
|
import XDrive from './drive.vue';
|
|
import XAbuse from './abuse.vue';
|
|
import XFederation from './federation.vue';
|
|
|
|
import { faHeadset, faArrowLeft, faGlobe, faExclamationCircle, faTasks, faStream, faDatabase } from '@fortawesome/free-solid-svg-icons';
|
|
import { faGrin } from '@fortawesome/free-regular-svg-icons';
|
|
|
|
// Detect the user agent
|
|
const ua = navigator.userAgent.toLowerCase();
|
|
const isMobile = /mobile|iphone|ipad|android/.test(ua);
|
|
|
|
export default Vue.extend({
|
|
i18n: i18n('admin/views/index.vue'),
|
|
components: {
|
|
XDashboard,
|
|
XInstance,
|
|
XQueue,
|
|
XLogs,
|
|
XDb,
|
|
XModerators,
|
|
XEmoji,
|
|
XAnnouncements,
|
|
XUsers,
|
|
XDrive,
|
|
XAbuse,
|
|
XFederation,
|
|
},
|
|
provide: {
|
|
isMobile
|
|
},
|
|
data() {
|
|
return {
|
|
page: 'dashboard',
|
|
version,
|
|
isMobile,
|
|
navOpend: !isMobile,
|
|
faGrin,
|
|
faArrowLeft,
|
|
faHeadset,
|
|
faGlobe,
|
|
faExclamationCircle,
|
|
faTasks,
|
|
faStream,
|
|
faDatabase,
|
|
};
|
|
},
|
|
methods: {
|
|
nav(page: string) {
|
|
this.page = page;
|
|
}
|
|
}
|
|
});
|
|
</script>
|
|
|
|
<style lang="stylus" scoped>
|
|
.mk-admin
|
|
$headerHeight = 48px
|
|
|
|
display flex
|
|
height 100%
|
|
|
|
> header
|
|
position fixed
|
|
top 0
|
|
z-index 10000
|
|
width 100%
|
|
color var(--mobileHeaderFg)
|
|
background-color var(--mobileHeaderBg)
|
|
box-shadow 0 1px 0 rgba(#000, 0.075)
|
|
|
|
&, *
|
|
user-select none
|
|
|
|
> span
|
|
display block
|
|
line-height $headerHeight
|
|
text-align center
|
|
|
|
> .nav
|
|
display block
|
|
position absolute
|
|
top 0
|
|
left 0
|
|
z-index 10001
|
|
padding 0
|
|
width $headerHeight
|
|
font-size 1.4em
|
|
line-height $headerHeight
|
|
border-right solid 1px rgba(#000, 0.1)
|
|
|
|
> [data-icon]
|
|
transition all 0.2s ease
|
|
|
|
> nav
|
|
position fixed
|
|
z-index 20001
|
|
top 0
|
|
left 0
|
|
width 250px
|
|
height 100vh
|
|
overflow auto
|
|
background #333
|
|
color #fff
|
|
|
|
> .mi
|
|
text-align center
|
|
|
|
> svg
|
|
width 24px
|
|
height 82px
|
|
vertical-align top
|
|
fill #fff
|
|
opacity 0.7
|
|
|
|
> .me
|
|
display flex
|
|
margin 0 16px 16px 16px
|
|
padding 16px 0
|
|
align-items center
|
|
border-top solid 1px #555
|
|
border-bottom solid 1px #555
|
|
|
|
> .avatar
|
|
height 48px
|
|
border-radius 100%
|
|
vertical-align middle
|
|
|
|
> .name
|
|
margin 0 16px
|
|
padding 0
|
|
color #fff
|
|
overflow hidden
|
|
text-overflow ellipsis
|
|
white-space nowrap
|
|
font-size 15px
|
|
|
|
> .back-to-misskey
|
|
margin 16px 16px 0 16px
|
|
padding 0
|
|
border-top solid 1px #555
|
|
|
|
> a
|
|
display block
|
|
padding 16px 4px
|
|
color inherit
|
|
text-decoration none
|
|
color #eee
|
|
font-size 15px
|
|
|
|
&:hover
|
|
color #fff
|
|
|
|
> [data-icon]
|
|
margin-right 6px
|
|
|
|
> .version
|
|
margin 0 16px 16px 16px
|
|
padding-top 16px
|
|
border-top solid 1px #555
|
|
text-align center
|
|
|
|
> small
|
|
opacity 0.7
|
|
|
|
> ul
|
|
margin 0
|
|
padding 0
|
|
list-style none
|
|
font-size 15px
|
|
|
|
> li
|
|
display block
|
|
padding 10px 16px
|
|
margin 0
|
|
cursor pointer
|
|
user-select none
|
|
color #eee
|
|
transition margin-left 0.2s ease
|
|
|
|
&:hover
|
|
color #fff
|
|
|
|
> [data-icon]
|
|
margin-right 6px
|
|
|
|
&.active
|
|
margin-left 8px
|
|
color var(--primary) !important
|
|
|
|
&:after
|
|
content ""
|
|
display block
|
|
position absolute
|
|
top 0
|
|
right 0
|
|
bottom 0
|
|
margin auto 0
|
|
height 0
|
|
border-top solid 16px transparent
|
|
border-right solid 16px var(--bg)
|
|
border-bottom solid 16px transparent
|
|
border-left solid 16px transparent
|
|
|
|
> .nav-backdrop
|
|
position fixed
|
|
top 0
|
|
left 0
|
|
z-index 20000
|
|
width 100%
|
|
height 100%
|
|
background var(--mobileNavBackdrop)
|
|
|
|
> main
|
|
width 100%
|
|
padding 0 0 0 250px
|
|
|
|
> .page
|
|
max-width 1150px
|
|
|
|
@media (min-width 500px)
|
|
padding 16px
|
|
|
|
&.isMobile
|
|
> main
|
|
padding $headerHeight 0 0 0
|
|
|
|
</style>
|