add app mode

This commit is contained in:
jxxghp
2024-06-13 17:30:50 +08:00
parent 7f2f7b100b
commit 1fc964ec16
14 changed files with 184 additions and 18 deletions

View File

@@ -33,11 +33,10 @@ defineProps<{
.nav-link a {
display: flex;
align-items: center;
cursor: pointer;
padding-left: 1.375rem;
padding-right: 1rem;
margin-right: 1.125em;
border-radius: 0 3.125rem 3.125rem 0 !important;
cursor: pointer;
margin-inline-end: 1.125em;
padding-inline: 1.375rem 1rem;
}
}
</style>

View File

@@ -19,6 +19,9 @@ const superUser = store.state.auth.superUser
const getMenuList = (header: string) => {
return SystemNavMenus.filter((item: NavMenu) => item.header === header && (!item.admin || superUser))
}
// APP模式
const appMode = computed(() => localStorage.getItem('MP_APPMODE') == '1')
</script>
<template>
@@ -27,7 +30,7 @@ const getMenuList = (header: string) => {
<template #navbar="{ toggleVerticalOverlayNavActive }">
<div class="d-flex h-100 align-center mx-1">
<!-- 👉 Vertical Nav Toggle -->
<IconBtn class="ms-n2 d-lg-none" @click="toggleVerticalOverlayNavActive(true)">
<IconBtn v-if="!appMode" class="ms-n2 d-lg-none" @click="toggleVerticalOverlayNavActive(true)">
<VIcon icon="mdi-menu" />
</IconBtn>
<!-- 👉 Search Bar -->

View File

@@ -1,3 +1,35 @@
<script setup lang="ts">
import { useDisplay } from 'vuetify'
const display = useDisplay()
const appMode = computed(() => {
return localStorage.getItem('MP_APPMODE') == '1' && display.mdAndDown.value
})
</script>
<template>
<div class="h-100 d-flex align-center justify-space-between" />
<div v-if="appMode" class="w-100" style="block-size: 3.5rem">
<VBottomNavigation grow horizontal color="primary">
<VBtn to="/dashboard">
<VIcon size="28">mdi-home-outline</VIcon>
<span>首页</span>
</VBtn>
<VBtn to="/ranking">
<VIcon size="28">mdi-star-check-outline</VIcon>
<span>推荐</span>
</VBtn>
<VBtn to="/subscribe-movie?tab=mysub">
<VIcon size="28">mdi-movie-roll</VIcon>
<span><span></span>电影</span>
</VBtn>
<VBtn to="/subscribe-tv?tab=mysub">
<VIcon size="28">mdi-television-classic</VIcon>
<span>电视剧</span>
</VBtn>
<VBtn to="/apps">
<VIcon size="28">mdi-dots-horizontal</VIcon>
<span>更多</span>
</VBtn>
</VBottomNavigation>
</div>
</template>

View File

@@ -6,6 +6,9 @@ import router from '@/router'
import avatar1 from '@images/avatars/avatar-1.png'
import api from '@/api'
import ProgressDialog from '@/components/dialog/ProgressDialog.vue'
import { useDisplay } from 'vuetify'
const display = useDisplay()
// Vuex Store
const store = useStore()
@@ -58,10 +61,20 @@ async function restart() {
}
}
// 是否精简模式
const isCompactMode = ref(localStorage.getItem('MP_APPMODE') == '1')
// 从Vuex Store中获取信息
const superUser = store.state.auth.superUser
const userName = store.state.auth.userName
const avatar = store.state.auth.avatar
// 监听精简模式切换
watch(isCompactMode, value => {
localStorage.setItem('MP_APPMODE', value ? '1' : '0')
//刷新页面
location.reload()
})
</script>
<template>
@@ -86,6 +99,12 @@ const avatar = store.state.auth.avatar
</VListItemTitle>
<VListItemSubtitle>{{ userName }}</VListItemSubtitle>
</VListItem>
<VListItem v-if="display.mdAndDown.value">
<template #prepend>
<VSwitch class="me-2" v-model="isCompactMode"></VSwitch>
</template>
<VListItemTitle>App模式</VListItemTitle>
</VListItem>
<VDivider class="my-2" />
<!-- 👉 Profile -->
@@ -105,7 +124,7 @@ const avatar = store.state.auth.avatar
</VListItem>
<!-- Divider -->
<VDivider class="my-2" />
<VDivider v-if="superUser" class="my-2" />
<!-- 👉 restart -->
<VListItem v-if="superUser" @click="restart">
@@ -115,9 +134,6 @@ const avatar = store.state.auth.avatar
<VListItemTitle>重启</VListItemTitle>
</VListItem>
<!-- Divider -->
<VDivider class="my-2" />
<!-- 👉 Logout -->
<VListItem @click="logout">
<VBtn color="error" block>

27
src/pages/appcenter.vue Normal file
View File

@@ -0,0 +1,27 @@
<script setup lang="ts">
import { NavMenu } from '@/@layouts/types'
import { SystemNavMenus } from '@/router/menu'
import store from '@/store'
// 从Vuex Store中获取superuser信息
const superUser = store.state.auth.superUser
// 根据分类获取菜单列表
const getMenuList = () => {
return SystemNavMenus.filter((item: NavMenu) => !item.admin || superUser)
}
</script>
<template>
<div class="ps ps--active-y mx-3" tabindex="0">
<VRow class="ma-0 mt-n1">
<VCol cols="6" md="4" lg="3" class="text-center cursor-pointer shortcut-icon" v-for="item in getMenuList()">
<VCard class="pa-4" :to="item.to" variant="flat">
<VAvatar size="64" variant="text">
<VIcon size="48" :icon="item.icon" color="primary" />
</VAvatar>
<h6 class="text-base font-weight-medium mt-2 mb-0">{{ item.full_title || item.title }}</h6>
</VCard>
</VCol>
</VRow>
</div>
</template>

View File

@@ -5,6 +5,13 @@ import { isNullOrEmptyObject } from '@/@core/utils'
import { DashboardItem } from '@/api/types'
import store from '@/store'
import DashboardElement from '@/components/misc/DashboardElement.vue'
import { useDisplay } from 'vuetify'
// APP
const display = useDisplay()
const appMode = computed(() => {
return localStorage.getItem('MP_APPMODE') == '1' && display.mdAndDown.value
})
// 从Vuex Store中获取superuser信息
const superUser = store.state.auth.superUser
@@ -314,7 +321,16 @@ onBeforeMount(async () => {
</draggable>
<!-- 底部操作按钮 -->
<VFab icon="mdi-view-dashboard-edit" location="bottom" size="x-large" fixed app appear @click="dialog = true" />
<VFab
icon="mdi-view-dashboard-edit"
location="bottom"
size="x-large"
fixed
app
appear
@click="dialog = true"
:class="{ 'mb-12': appMode }"
/>
<!-- 弹窗根据配置生成选项 -->
<VDialog v-model="dialog" max-width="35rem" scrollable>

View File

@@ -5,6 +5,13 @@ import type { Context } from '@/api/types'
import store from '@/store'
import TorrentCardListView from '@/views/discover/TorrentCardListView.vue'
import TorrentRowListView from '@/views/discover/TorrentRowListView.vue'
import { useDisplay } from 'vuetify'
// APP
const display = useDisplay()
const appMode = computed(() => {
return localStorage.getItem('MP_APPMODE') == '1' && display.mdAndDown.value
})
// 路由参数
const route = useRoute()
@@ -142,13 +149,25 @@ onUnmounted(() => {
<!-- 视图切换 -->
<VFab
v-if="viewType === 'list'"
class="mb-12"
icon="mdi-view-grid"
location="bottom"
size="x-large"
absolute
app
appear
@click="setViewType('card')"
:class="{ 'mb-12': appMode }"
/>
<VFab
v-else
icon="mdi-view-list"
location="bottom"
size="x-large"
fixed
app
appear
@click="setViewType('card')"
@click="setViewType('list')"
:class="{ 'mb-12': appMode }"
/>
<VFab v-else icon="mdi-view-list" location="bottom" size="x-large" fixed app appear @click="setViewType('list')" />
</template>

View File

@@ -137,6 +137,13 @@ const router = createRouter({
requiresAuth: true,
},
},
{
path: '/apps',
component: () => import('../pages/appcenter.vue'),
meta: {
requiresAuth: true,
},
},
],
},
{

View File

@@ -9,7 +9,7 @@ export const SystemNavMenus = [
},
{
title: '推荐',
icon: 'mdi-table-star',
icon: 'mdi-star-check-outline',
to: '/ranking',
header: '发现',
admin: false,
@@ -23,6 +23,7 @@ export const SystemNavMenus = [
},
{
title: '电影',
full_title: '电影订阅',
icon: 'mdi-movie-roll',
to: '/subscribe-movie?tab=mysub',
header: '订阅',
@@ -30,6 +31,7 @@ export const SystemNavMenus = [
},
{
title: '电视剧',
full_title: '电视剧订阅',
icon: 'mdi-television-classic',
to: '/subscribe-tv?tab=mysub',
header: '订阅',
@@ -37,6 +39,7 @@ export const SystemNavMenus = [
},
{
title: '日历',
full_title: '订阅日历',
icon: 'mdi-calendar',
to: '/calendar',
header: '订阅',

View File

@@ -16,6 +16,11 @@ const route = useRoute()
// 显示器宽度
const display = useDisplay()
// APP
const appMode = computed(() => {
return localStorage.getItem('MP_APPMODE') == '1' && display.mdAndDown.value
})
// 当前标签
const activeTab = ref(route.query.tab)
@@ -438,6 +443,7 @@ onBeforeMount(async () => {
app
appear
@click="SearchDialog = true"
:class="{ 'mb-12': appMode }"
/>
<VDialog
v-if="SearchDialog"

View File

@@ -7,6 +7,15 @@ import ReorganizeDialog from '@/components/dialog/ReorganizeDialog.vue'
import ProgressDialog from '@/components/dialog/ProgressDialog.vue'
import { useRoute } from 'vue-router'
import router from '@/router'
import { useDisplay } from 'vuetify'
// 显示器宽度
const display = useDisplay()
// APP
const appMode = computed(() => {
return localStorage.getItem('MP_APPMODE') == '1' && display.mdAndDown.value
})
// 提示框
const $toast = useToast()
@@ -473,10 +482,11 @@ onMounted(fetchData)
app
appear
@click="removeHistoryBatch"
:class="{ 'mb-12': appMode }"
/>
<VFab
v-if="selected.length > 0"
class="mb-16"
:class="appMode ? 'mb-28' : 'mb-16'"
icon="mdi-redo-variant"
location="bottom"
size="x-large"

View File

@@ -5,6 +5,15 @@ import type { Site } from '@/api/types'
import SiteCard from '@/components/cards/SiteCard.vue'
import NoDataFound from '@/components/NoDataFound.vue'
import SiteAddEditDialog from '@/components/dialog/SiteAddEditDialog.vue'
import { useDisplay } from 'vuetify'
// 显示器宽度
const display = useDisplay()
// APP
const appMode = computed(() => {
return localStorage.getItem('MP_APPMODE') == '1' && display.mdAndDown.value
})
// 数据列表
const dataList = ref<Site[]>([])
@@ -67,7 +76,16 @@ onBeforeMount(fetchData)
error-description="已添加并支持的站点将会在这里显示"
/>
<!-- 新增站点按钮 -->
<VFab icon="mdi-plus" location="bottom" size="x-large" fixed app appear @click="siteAddDialog = true" />
<VFab
icon="mdi-plus"
location="bottom"
size="x-large"
fixed
app
appear
@click="siteAddDialog = true"
:class="{ 'mb-12': appMode }"
/>
<!-- 新增站点弹窗 -->
<SiteAddEditDialog
v-if="siteAddDialog"

View File

@@ -7,6 +7,15 @@ import SubscribeCard from '@/components/cards/SubscribeCard.vue'
import SubscribeEditDialog from '@/components/dialog/SubscribeEditDialog.vue'
import SubscribeHistoryDialog from '@/components/dialog/SubscribeHistoryDialog.vue'
import store from '@/store'
import { useDisplay } from 'vuetify'
// 显示器宽度
const display = useDisplay()
// APP
const appMode = computed(() => {
return localStorage.getItem('MP_APPMODE') == '1' && display.mdAndDown.value
})
// 输入参数
const props = defineProps({
@@ -97,13 +106,14 @@ onMounted(async () => {
app
appear
@click="subscribeEditDialog = true"
:class="{ 'mb-12': appMode }"
/>
<VFab
v-if="store.state.auth.superUser"
icon="mdi-history"
color="info"
location="bottom"
class="mb-16"
:class="appMode ? 'mb-28' : 'mb-16'"
size="x-large"
fixed
app