mirror of
https://github.com/jxxghp/MoviePilot-Frontend.git
synced 2026-05-11 18:10:49 +08:00
888 lines
25 KiB
Vue
888 lines
25 KiB
Vue
<script setup lang="ts">
|
||
import api from '@/api'
|
||
import type { Site, Plugin, Subscribe } from '@/api/types'
|
||
import { getNavMenus, getSettingTabs } from '@/router/i18n-menu'
|
||
import { NavMenu } from '@/@layouts/types'
|
||
import { useUserStore, useGlobalSettingsStore } from '@/stores'
|
||
import SearchSiteDialog from '@/components/dialog/SearchSiteDialog.vue'
|
||
import { useI18n } from 'vue-i18n'
|
||
import { useDisplay } from 'vuetify'
|
||
import { hasPermission, filterMenusByPermission } from '@/utils/permission'
|
||
|
||
// 显示器宽度
|
||
const display = useDisplay()
|
||
|
||
// 多语言支持
|
||
const { t } = useI18n()
|
||
|
||
// 定义props,接收modelValue
|
||
const props = defineProps<{
|
||
modelValue: boolean
|
||
}>()
|
||
|
||
// 路由
|
||
const router = useRouter()
|
||
|
||
// 用户 Store
|
||
const userStore = useUserStore()
|
||
|
||
// 全局设置 Store
|
||
const globalSettingsStore = useGlobalSettingsStore()
|
||
const globalSettings = globalSettingsStore.globalSettings
|
||
|
||
// 超级用户
|
||
const superUser = userStore.superUser
|
||
|
||
// 当前用户名
|
||
const userName = userStore.userName
|
||
|
||
// 权限检查
|
||
const hasSearchPermission = computed(() => {
|
||
return hasPermission(
|
||
{
|
||
is_superuser: userStore.superUser,
|
||
...userStore.permissions,
|
||
},
|
||
'search',
|
||
)
|
||
})
|
||
|
||
const hasSubscribePermission = computed(() => {
|
||
return hasPermission(
|
||
{
|
||
is_superuser: userStore.superUser,
|
||
...userStore.permissions,
|
||
},
|
||
'subscribe',
|
||
)
|
||
})
|
||
|
||
const hasManagePermission = computed(() => {
|
||
return hasPermission(
|
||
{
|
||
is_superuser: userStore.superUser,
|
||
...userStore.permissions,
|
||
},
|
||
'manage',
|
||
)
|
||
})
|
||
|
||
// 是否显示合集搜索项(当SEARCH_SOURCE包含themoviedb时显示)
|
||
const showCollectionSearch = computed(() => {
|
||
return globalSettings.SEARCH_SOURCE?.includes('themoviedb') || false
|
||
})
|
||
|
||
// 所有订阅数据
|
||
const SubscribeItems = ref<Subscribe[]>([])
|
||
|
||
// 站点选择对话框
|
||
const chooseSiteDialog = ref(false)
|
||
const selectedSites = ref<number[]>([])
|
||
const allSites = ref<Site[]>([])
|
||
|
||
// 定义事件
|
||
const emit = defineEmits(['close', 'update:modelValue'])
|
||
|
||
// 对话框状态的本地计算属性
|
||
const dialog = computed({
|
||
get: () => props.modelValue,
|
||
set: val => emit('update:modelValue', val),
|
||
})
|
||
|
||
// 搜索词
|
||
const searchWord = ref<string | null>(null)
|
||
|
||
// ref
|
||
const searchWordInput = ref<HTMLElement | null>(null)
|
||
|
||
// 近期搜索词条
|
||
const recentSearches = ref<string[]>([])
|
||
|
||
// 检测操作系统是否是Mac
|
||
function isMac() {
|
||
return navigator.platform.toUpperCase().indexOf('MAC') >= 0
|
||
}
|
||
|
||
// 计算属性:根据操作系统显示不同的按键提示
|
||
const metaKey = computed(() => (isMac() ? '⌘+K' : 'Ctrl+K'))
|
||
|
||
// 保存近期搜索到本地
|
||
function saveRecentSearches(keyword: string) {
|
||
if (!keyword) return
|
||
if (recentSearches.value.includes(keyword)) return
|
||
recentSearches.value.unshift(keyword)
|
||
localStorage.setItem('MP_RecentSearches', JSON.stringify(recentSearches.value))
|
||
}
|
||
|
||
// 从本地加载近期搜索
|
||
function loadRecentSearches() {
|
||
const recentSearchesStr = localStorage.getItem('MP_RecentSearches')
|
||
if (recentSearchesStr) {
|
||
recentSearches.value = JSON.parse(recentSearchesStr)
|
||
// 只保留最近的 5 条
|
||
if (recentSearches.value.length > 5) {
|
||
recentSearches.value = recentSearches.value.slice(0, 5)
|
||
}
|
||
}
|
||
}
|
||
|
||
// 所有菜单功能
|
||
function getMenus(): NavMenu[] {
|
||
let menus: NavMenu[] = []
|
||
// 导航菜单
|
||
getNavMenus(t).forEach(
|
||
item =>
|
||
item &&
|
||
menus.push({
|
||
title: item.full_title ?? item.title,
|
||
icon: item.icon,
|
||
to: item.to,
|
||
header: item.header,
|
||
admin: item.admin,
|
||
}),
|
||
)
|
||
// 设置标签页
|
||
getSettingTabs(t).forEach(
|
||
item =>
|
||
item &&
|
||
menus.push({
|
||
title: t('navItems.setting') + ' -> ' + item.title,
|
||
icon: item.icon,
|
||
to: `/setting?tab=${item.tab}`,
|
||
header: '',
|
||
admin: true,
|
||
description: item.description,
|
||
}),
|
||
)
|
||
|
||
return menus
|
||
}
|
||
|
||
// 获取用户权限信息
|
||
const userPermissions = computed(() => ({
|
||
is_superuser: userStore.superUser,
|
||
...userStore.permissions,
|
||
}))
|
||
|
||
// 匹配的菜单列表
|
||
const matchedMenuItems = computed(() => {
|
||
if (!searchWord.value) return []
|
||
const lowerWord = (searchWord.value as string).toLowerCase()
|
||
const menuItems = getMenus()
|
||
if (menuItems) {
|
||
// 先根据用户权限过滤菜单
|
||
const filteredMenus = filterMenusByPermission(menuItems, userPermissions.value)
|
||
// 再根据搜索词过滤
|
||
return filteredMenus.filter(
|
||
item =>
|
||
item.title.toLowerCase().includes(lowerWord) ||
|
||
(item.description && item.description.toLowerCase().includes(lowerWord)),
|
||
)
|
||
}
|
||
return []
|
||
})
|
||
|
||
// 所有插件(已安装)
|
||
const pluginItems = ref<Plugin[]>([])
|
||
|
||
// 获取插件列表数据
|
||
async function fetchInstalledPlugins() {
|
||
try {
|
||
pluginItems.value = await api.get('plugin/', {
|
||
params: {
|
||
state: 'installed',
|
||
},
|
||
})
|
||
} catch (error) {
|
||
console.error(error)
|
||
}
|
||
}
|
||
|
||
// 匹配的插件列表
|
||
const matchedPluginItems = computed(() => {
|
||
if (!searchWord.value) return []
|
||
if (!hasManagePermission.value) return []
|
||
const lowerWord = (searchWord.value as string).toLowerCase()
|
||
return pluginItems.value.filter((item: Plugin) => {
|
||
if (!item.plugin_name && !item.plugin_desc) return false
|
||
return item.plugin_name?.toLowerCase().includes(lowerWord) || item.plugin_desc?.toLowerCase().includes(lowerWord)
|
||
})
|
||
})
|
||
|
||
// 获取订阅列表数据
|
||
async function fetchSubscribes() {
|
||
try {
|
||
SubscribeItems.value = await api.get('subscribe/')
|
||
} catch (error) {
|
||
console.error(error)
|
||
}
|
||
}
|
||
|
||
// 从接口加载用户站点偏好设置
|
||
const loadUserSitePreferences = async () => {
|
||
try {
|
||
const result = await api.get('system/setting/IndexerSites')
|
||
if (result && result.data && result.data.value) {
|
||
selectedSites.value = result.data.value
|
||
return
|
||
}
|
||
} catch (err) {
|
||
console.error(err)
|
||
}
|
||
}
|
||
|
||
// 查询所有站点
|
||
async function queryAllSites() {
|
||
try {
|
||
const data: Site[] = await api.get('site/')
|
||
// 过滤站点,只有启用的站点才显示
|
||
allSites.value = data.filter(item => item.is_active)
|
||
// 如果没有选择任何站点并且有可用站点,才默认选择全部
|
||
if (selectedSites.value.length === 0 && allSites.value.length > 0) {
|
||
selectedSites.value = allSites.value.map((site: Site) => site.id)
|
||
}
|
||
} catch (error) {
|
||
console.log(error)
|
||
}
|
||
}
|
||
|
||
// 打开站点选择对话框
|
||
const openSiteDialog = () => {
|
||
chooseSiteDialog.value = true
|
||
}
|
||
|
||
// 匹配的订阅列表
|
||
const matchedSubscribeItems = computed(() => {
|
||
if (!searchWord.value) return []
|
||
if (!hasSubscribePermission.value) return []
|
||
const lowerWord = (searchWord.value as string).toLowerCase()
|
||
return SubscribeItems.value.filter((item: Subscribe) => {
|
||
return (item.name.toLowerCase().includes(lowerWord) && (superUser || userName === item.username)) || false
|
||
})
|
||
})
|
||
|
||
// 搜索多站点
|
||
function searchSites(sites: number[]) {
|
||
chooseSiteDialog.value = false
|
||
selectedSites.value = sites
|
||
searchTorrent()
|
||
}
|
||
|
||
// 搜索资源
|
||
function searchTorrent() {
|
||
if (!searchWord.value) return
|
||
// 记录搜索词
|
||
saveRecentSearches(searchWord.value)
|
||
// 跳转到搜索页面
|
||
router.push({
|
||
path: '/resource',
|
||
query: {
|
||
keyword: searchWord.value,
|
||
area: 'title',
|
||
sites: selectedSites.value.join(','),
|
||
},
|
||
})
|
||
// 关闭搜索对话框
|
||
dialog.value = false
|
||
emit('close')
|
||
}
|
||
|
||
// 跳转媒体搜索页面
|
||
function searchMedia(searchType: string) {
|
||
// 搜索类型 media/person
|
||
if (!searchWord.value) return
|
||
saveRecentSearches(searchWord.value)
|
||
router.push({
|
||
path: '/browse/media/search',
|
||
query: {
|
||
title: searchWord.value,
|
||
type: searchType,
|
||
},
|
||
})
|
||
emit('close')
|
||
}
|
||
|
||
// 跳转到历史记录页面
|
||
function searchHistory() {
|
||
if (!searchWord.value) return
|
||
saveRecentSearches(searchWord.value)
|
||
router.push({
|
||
path: '/history',
|
||
query: {
|
||
search: searchWord.value,
|
||
},
|
||
})
|
||
emit('close')
|
||
}
|
||
|
||
// 跳转到订阅分享页面
|
||
function searchSubscribeShares() {
|
||
if (!searchWord.value) return
|
||
saveRecentSearches(searchWord.value)
|
||
router.push({
|
||
path: '/subscribe-share',
|
||
query: {
|
||
keyword: searchWord.value,
|
||
},
|
||
})
|
||
emit('close')
|
||
}
|
||
|
||
// 跳转插件页面
|
||
function showPlugin(pluginId: string) {
|
||
router.push({
|
||
path: `/plugins/`,
|
||
query: {
|
||
tab: 'installed',
|
||
id: pluginId,
|
||
},
|
||
})
|
||
emit('close')
|
||
}
|
||
|
||
// 跳转菜单页面
|
||
function goPage(to: string) {
|
||
router.push(to)
|
||
emit('close')
|
||
}
|
||
|
||
// 跳转订阅页面
|
||
function goSubscribe(subscribe: Subscribe) {
|
||
if (subscribe.type === '电影') {
|
||
router.push({
|
||
path: '/subscribe/movie',
|
||
query: {
|
||
id: subscribe.id,
|
||
},
|
||
})
|
||
} else {
|
||
router.push({
|
||
path: '/subscribe/tv',
|
||
query: {
|
||
id: subscribe.id,
|
||
},
|
||
})
|
||
}
|
||
emit('close')
|
||
}
|
||
|
||
onMounted(() => {
|
||
setTimeout(() => {
|
||
searchWordInput.value?.focus()
|
||
}, 500)
|
||
// 根据权限加载不同的数据
|
||
if (hasManagePermission.value) {
|
||
fetchInstalledPlugins()
|
||
}
|
||
if (hasSubscribePermission.value) {
|
||
fetchSubscribes()
|
||
}
|
||
loadRecentSearches()
|
||
if (hasSearchPermission.value) {
|
||
loadUserSitePreferences()
|
||
if (hasManagePermission.value) {
|
||
queryAllSites()
|
||
}
|
||
}
|
||
})
|
||
</script>
|
||
<template>
|
||
<VDialog v-model="dialog" max-width="40rem" scrollable :fullscreen="!display.mdAndUp.value">
|
||
<VCard class="search-dialog">
|
||
<!-- 搜索输入框区域 -->
|
||
<div class="search-header">
|
||
<div class="search-input-wrapper">
|
||
<VIcon icon="mdi-magnify" size="22" class="search-input-icon" />
|
||
<input
|
||
ref="searchWordInput"
|
||
v-model="searchWord"
|
||
type="text"
|
||
class="search-native-input"
|
||
:placeholder="t('dialog.searchBar.searchPlaceholder')"
|
||
@keydown.enter="searchMedia('media')"
|
||
@keydown.escape="emit('close')"
|
||
/>
|
||
<VBtn icon size="small" variant="text" class="search-submit-btn" @click="searchMedia('media')">
|
||
<VIcon icon="mdi-magnify" size="20" />
|
||
</VBtn>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 主内容区域 -->
|
||
<div class="search-content">
|
||
<!-- 有搜索词时显示搜索入口和匹配结果 -->
|
||
<VList lines="two" v-if="searchWord" class="search-list pa-0 py-2">
|
||
<!-- 媒体搜索入口 -->
|
||
<VListSubheader class="font-weight-medium text-uppercase px-4">
|
||
{{ t('common.media') }}
|
||
</VListSubheader>
|
||
|
||
<VListItem density="comfortable" link @click="searchMedia('media')" class="search-result-item mx-2 my-1">
|
||
<template #prepend>
|
||
<div class="result-icon-wrapper">
|
||
<VIcon icon="mdi-movie-search" size="small" color="medium-emphasis" />
|
||
</div>
|
||
</template>
|
||
<VListItemTitle class="font-weight-medium text-body-2">
|
||
{{ t('recommend.categoryMovie') }}、{{ t('recommend.categoryTV') }}
|
||
</VListItemTitle>
|
||
<VListItemSubtitle class="text-caption text-medium-emphasis">
|
||
{{ t('common.search') }} <span class="primary-text font-weight-medium">{{ searchWord }}</span>
|
||
{{ t('resource.title') }}
|
||
</VListItemSubtitle>
|
||
</VListItem>
|
||
|
||
<VListItem
|
||
v-if="showCollectionSearch"
|
||
density="comfortable"
|
||
link
|
||
@click="searchMedia('collection')"
|
||
class="search-result-item mx-2 my-1"
|
||
>
|
||
<template #prepend>
|
||
<div class="result-icon-wrapper">
|
||
<VIcon icon="mdi-movie-filter" size="small" color="medium-emphasis" />
|
||
</div>
|
||
</template>
|
||
<VListItemTitle class="font-weight-medium text-body-2">{{
|
||
t('dialog.searchBar.collections')
|
||
}}</VListItemTitle>
|
||
<VListItemSubtitle class="text-caption text-medium-emphasis">
|
||
{{ t('common.search') }} <span class="primary-text font-weight-medium">{{ searchWord }}</span>
|
||
{{ t('dialog.searchBar.collectionSearch') }}
|
||
</VListItemSubtitle>
|
||
</VListItem>
|
||
|
||
<VListItem density="comfortable" link @click="searchMedia('person')" class="search-result-item mx-2 my-1">
|
||
<template #prepend>
|
||
<div class="result-icon-wrapper">
|
||
<VIcon icon="mdi-account-search" size="small" color="medium-emphasis" />
|
||
</div>
|
||
</template>
|
||
<VListItemTitle class="font-weight-medium text-body-2">{{ t('browse.actor') }}</VListItemTitle>
|
||
<VListItemSubtitle class="text-caption text-medium-emphasis">
|
||
{{ t('common.search') }} <span class="primary-text font-weight-medium">{{ searchWord }}</span>
|
||
{{ t('dialog.searchBar.actorSearch') }}
|
||
</VListItemSubtitle>
|
||
</VListItem>
|
||
|
||
<VListItem
|
||
v-if="hasSubscribePermission"
|
||
density="comfortable"
|
||
link
|
||
@click="searchSubscribeShares"
|
||
class="search-result-item mx-2 my-1"
|
||
>
|
||
<template #prepend>
|
||
<div class="result-icon-wrapper">
|
||
<VIcon icon="mdi-share-variant" size="small" color="medium-emphasis" />
|
||
</div>
|
||
</template>
|
||
<VListItemTitle class="font-weight-medium text-body-2">{{ t('subscribe.searchShares') }}</VListItemTitle>
|
||
<VListItemSubtitle class="text-caption text-medium-emphasis">
|
||
{{ t('common.search') }} <span class="primary-text font-weight-medium">{{ searchWord }}</span>
|
||
{{ t('dialog.searchBar.subscribeShareSearch') }}
|
||
</VListItemSubtitle>
|
||
</VListItem>
|
||
|
||
<VListItem
|
||
v-if="hasManagePermission"
|
||
density="comfortable"
|
||
link
|
||
@click="searchHistory"
|
||
class="search-result-item mx-2 my-1"
|
||
>
|
||
<template #prepend>
|
||
<div class="result-icon-wrapper">
|
||
<VIcon icon="mdi-history" size="small" color="medium-emphasis" />
|
||
</div>
|
||
</template>
|
||
<VListItemTitle class="font-weight-medium text-body-2">{{ t('navItems.history') }}</VListItemTitle>
|
||
<VListItemSubtitle class="text-caption text-medium-emphasis">
|
||
{{ t('common.search') }} <span class="primary-text font-weight-medium">{{ searchWord }}</span>
|
||
{{ t('dialog.searchBar.historySearch') }}
|
||
</VListItemSubtitle>
|
||
</VListItem>
|
||
|
||
<!-- 匹配的订阅 -->
|
||
<template v-if="matchedSubscribeItems.length > 0">
|
||
<VDivider class="mx-4 my-2 search-divider" />
|
||
<VListSubheader class="font-weight-medium text-uppercase px-4">
|
||
{{ t('dialog.searchBar.subscriptions') }}
|
||
</VListSubheader>
|
||
<VListItem
|
||
v-for="subscribe in matchedSubscribeItems"
|
||
:key="subscribe.id"
|
||
density="comfortable"
|
||
link
|
||
@click="goSubscribe(subscribe)"
|
||
class="search-result-item mx-2 my-1"
|
||
>
|
||
<template #prepend>
|
||
<div class="result-icon-wrapper">
|
||
<VIcon
|
||
:icon="subscribe.type === '电影' ? 'mdi-movie-roll' : 'mdi-television-classic'"
|
||
size="small"
|
||
color="medium-emphasis"
|
||
/>
|
||
</div>
|
||
</template>
|
||
<VListItemTitle class="font-weight-medium text-body-2">
|
||
{{ subscribe.name }}
|
||
<span v-if="subscribe.season" class="text-caption">
|
||
{{ t('resource.season') }} {{ subscribe.season }}</span
|
||
>
|
||
</VListItemTitle>
|
||
<VListItemSubtitle class="text-caption text-medium-emphasis">
|
||
{{ subscribe.type }}
|
||
</VListItemSubtitle>
|
||
</VListItem>
|
||
</template>
|
||
|
||
<!-- 匹配的菜单/功能 -->
|
||
<template v-if="matchedMenuItems.length > 0">
|
||
<VDivider class="mx-4 my-2 search-divider" />
|
||
<VListSubheader class="font-weight-medium text-uppercase px-4">
|
||
{{ t('dialog.searchBar.functions') }}
|
||
</VListSubheader>
|
||
<VListItem
|
||
v-for="menu in matchedMenuItems"
|
||
:key="menu.title"
|
||
density="comfortable"
|
||
link
|
||
@click="goPage(menu.to as string)"
|
||
class="search-result-item mx-2 my-1"
|
||
>
|
||
<template #prepend>
|
||
<div class="result-icon-wrapper">
|
||
<VIcon :icon="menu.icon as string" size="small" color="medium-emphasis" />
|
||
</div>
|
||
</template>
|
||
<VListItemTitle class="font-weight-medium text-body-2">
|
||
{{ menu.title }}
|
||
</VListItemTitle>
|
||
<VListItemSubtitle v-if="menu.description" class="text-caption text-medium-emphasis">
|
||
{{ menu.description }}
|
||
</VListItemSubtitle>
|
||
</VListItem>
|
||
</template>
|
||
|
||
<!-- 匹配的插件 -->
|
||
<template v-if="matchedPluginItems.length > 0">
|
||
<VDivider class="mx-4 my-2 search-divider" />
|
||
<VListSubheader class="font-weight-medium text-uppercase px-4">
|
||
{{ t('dialog.searchBar.plugins') }}
|
||
</VListSubheader>
|
||
<VListItem
|
||
v-for="plugin in matchedPluginItems"
|
||
:key="plugin.id"
|
||
density="comfortable"
|
||
link
|
||
@click="showPlugin(plugin.id ?? '')"
|
||
class="search-result-item mx-2 my-1"
|
||
>
|
||
<template #prepend>
|
||
<div class="result-icon-wrapper">
|
||
<VIcon icon="mdi-puzzle" size="small" color="medium-emphasis" />
|
||
</div>
|
||
</template>
|
||
<VListItemTitle class="font-weight-medium text-body-2">
|
||
{{ plugin.plugin_name }}
|
||
</VListItemTitle>
|
||
<VListItemSubtitle class="text-caption text-medium-emphasis">
|
||
{{ plugin.plugin_desc }}
|
||
</VListItemSubtitle>
|
||
</VListItem>
|
||
</template>
|
||
|
||
<!-- 站点资源搜索 -->
|
||
<template v-if="hasSearchPermission">
|
||
<VDivider class="mx-4 my-2 search-divider" />
|
||
<VListSubheader class="font-weight-medium text-uppercase px-4">
|
||
{{ t('dialog.searchBar.siteResources') }}
|
||
</VListSubheader>
|
||
|
||
<VListItem density="comfortable" link @click="searchTorrent" class="search-result-item mx-2 my-1">
|
||
<template #prepend>
|
||
<div class="result-icon-wrapper">
|
||
<VIcon icon="mdi-file-search" size="small" color="medium-emphasis" />
|
||
</div>
|
||
</template>
|
||
<VListItemTitle class="font-weight-medium text-body-2">{{
|
||
t('dialog.searchBar.searchInSites')
|
||
}}</VListItemTitle>
|
||
<VListItemSubtitle class="text-caption text-medium-emphasis">
|
||
{{ t('common.search') }} <span class="primary-text font-weight-medium">{{ searchWord }}</span>
|
||
{{ t('dialog.searchBar.relatedResources') }}
|
||
</VListItemSubtitle>
|
||
<template #append>
|
||
<VBtn
|
||
v-if="hasManagePermission"
|
||
size="x-small"
|
||
variant="tonal"
|
||
color="primary"
|
||
rounded="pill"
|
||
@click.stop="openSiteDialog"
|
||
>
|
||
{{ t('dialog.searchBar.selectSites') }}
|
||
</VBtn>
|
||
</template>
|
||
</VListItem>
|
||
</template>
|
||
</VList>
|
||
|
||
<!-- 无搜索词时显示空状态 -->
|
||
<div v-else class="search-empty-state">
|
||
<!-- 有最近搜索 -->
|
||
<div v-if="!searchWord && recentSearches.length > 0" class="recent-searches-section">
|
||
<div class="text-body-2 font-weight-medium text-medium-emphasis mb-3">
|
||
{{ t('dialog.searchBar.recentSearches') }}
|
||
</div>
|
||
<div class="d-flex flex-wrap gap-2">
|
||
<VChip
|
||
v-for="(word, index) in recentSearches"
|
||
:key="index"
|
||
variant="flat"
|
||
color="primary"
|
||
size="small"
|
||
@click="searchWord = word"
|
||
>
|
||
<VIcon start size="x-small">mdi-history</VIcon>
|
||
{{ word }}
|
||
</VChip>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 空状态提示 -->
|
||
<div v-else class="empty-hint">
|
||
<span class="text-body-1 text-medium-emphasis">{{ t('dialog.searchBar.emptySearchHint') }}</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 底部区域 -->
|
||
<!-- 桌面端:快捷键提示 -->
|
||
<div v-if="display.mdAndUp.value" class="search-footer">
|
||
<div class="shortcut-group">
|
||
<kbd>Esc</kbd>
|
||
<span class="shortcut-label">{{ t('dialog.searchBar.escClose') }}</span>
|
||
</div>
|
||
<div class="shortcut-group">
|
||
<kbd>{{ metaKey }}</kbd>
|
||
<span class="shortcut-label">{{ t('dialog.searchBar.openSearch') }}</span>
|
||
</div>
|
||
</div>
|
||
<!-- 移动端:关闭图标 -->
|
||
<div v-else class="search-footer-mobile">
|
||
<VBtn icon variant="tonal" @click="emit('close')">
|
||
<VIcon icon="mdi-close" size="20" />
|
||
</VBtn>
|
||
</div>
|
||
</VCard>
|
||
</VDialog>
|
||
|
||
<!-- 站点选择对话框 -->
|
||
<SearchSiteDialog
|
||
v-if="chooseSiteDialog"
|
||
v-model="chooseSiteDialog"
|
||
:sites="allSites"
|
||
:selected="selectedSites"
|
||
@search="searchSites"
|
||
@close="chooseSiteDialog = false"
|
||
@reload="queryAllSites"
|
||
/>
|
||
</template>
|
||
|
||
<style scoped>
|
||
.search-dialog {
|
||
display: flex;
|
||
overflow: hidden;
|
||
flex-direction: column;
|
||
border-radius: 16px !important;
|
||
box-shadow: 0 12px 40px rgba(0, 0, 0, 12%) !important;
|
||
}
|
||
|
||
/* 搜索头部区域 */
|
||
.search-header {
|
||
padding-block: 16px 12px;
|
||
padding-inline: 16px;
|
||
}
|
||
|
||
/* 搜索输入框容器 */
|
||
.search-input-wrapper {
|
||
display: flex;
|
||
align-items: center;
|
||
border: 1.5px solid rgba(var(--v-theme-on-surface), 0.15);
|
||
border-radius: 28px;
|
||
background-color: rgba(var(--v-theme-surface-variant), 0.04);
|
||
block-size: 48px;
|
||
padding-inline: 14px 6px;
|
||
transition:
|
||
border-color 0.2s ease,
|
||
box-shadow 0.2s ease;
|
||
}
|
||
|
||
.search-input-wrapper:focus-within {
|
||
border-color: rgba(var(--v-theme-on-surface), 0.3);
|
||
box-shadow: 0 0 0 3px rgba(var(--v-theme-on-surface), 0.04);
|
||
}
|
||
|
||
.search-input-icon {
|
||
flex-shrink: 0;
|
||
color: rgba(var(--v-theme-on-surface), 0.45);
|
||
margin-inline-end: 10px;
|
||
}
|
||
|
||
.search-native-input {
|
||
flex: 1;
|
||
border: none;
|
||
background: transparent;
|
||
color: rgba(var(--v-theme-on-surface), 0.87);
|
||
font-size: 15px;
|
||
line-height: 1.5;
|
||
min-inline-size: 0;
|
||
outline: none;
|
||
}
|
||
|
||
.search-native-input::placeholder {
|
||
color: rgba(var(--v-theme-on-surface), 0.38);
|
||
}
|
||
|
||
.search-submit-btn {
|
||
flex-shrink: 0;
|
||
border-radius: 50% !important;
|
||
background-color: rgba(var(--v-theme-on-surface), 0.06) !important;
|
||
block-size: 36px !important;
|
||
color: rgba(var(--v-theme-on-surface), 0.6) !important;
|
||
inline-size: 36px !important;
|
||
transition: background-color 0.2s ease;
|
||
}
|
||
|
||
.search-submit-btn:hover {
|
||
background-color: rgba(var(--v-theme-on-surface), 0.12) !important;
|
||
}
|
||
|
||
/* 主内容区域 */
|
||
.search-content {
|
||
max-block-size: 600px;
|
||
min-block-size: 150px;
|
||
overflow-y: auto;
|
||
}
|
||
|
||
.search-list {
|
||
background: transparent !important;
|
||
}
|
||
|
||
.search-result-item {
|
||
border-radius: 10px !important;
|
||
margin-block-end: 2px;
|
||
transition: background-color 0.15s ease;
|
||
}
|
||
|
||
.search-result-item:hover {
|
||
background-color: rgba(var(--v-theme-on-surface), 0.04);
|
||
}
|
||
|
||
.result-icon-wrapper {
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
border-radius: 8px;
|
||
background-color: rgba(var(--v-theme-surface-variant), 0.1);
|
||
block-size: 32px;
|
||
inline-size: 32px;
|
||
margin-inline-end: 12px;
|
||
}
|
||
|
||
.search-divider {
|
||
opacity: 0.08;
|
||
}
|
||
|
||
.primary-text {
|
||
color: rgb(var(--v-theme-primary));
|
||
}
|
||
|
||
/* 空状态 */
|
||
.search-empty-state {
|
||
display: flex;
|
||
align-items: start;
|
||
justify-content: center;
|
||
min-block-size: 150px;
|
||
padding-block: 0;
|
||
padding-inline: 1.5rem;
|
||
}
|
||
|
||
.recent-searches-section {
|
||
inline-size: 100%;
|
||
}
|
||
|
||
.empty-hint {
|
||
text-align: center;
|
||
}
|
||
|
||
/* 底部快捷键提示 */
|
||
.search-footer {
|
||
display: flex;
|
||
align-items: center;
|
||
border-block-start: 1px solid rgba(var(--v-theme-on-surface), 0.08);
|
||
gap: 16px;
|
||
padding-block: 10px;
|
||
padding-inline: 16px;
|
||
}
|
||
|
||
.shortcut-group {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 6px;
|
||
}
|
||
|
||
kbd {
|
||
border: 1px solid rgba(var(--v-theme-on-surface), 0.15);
|
||
border-radius: 5px;
|
||
background-color: rgba(var(--v-theme-on-surface), 0.04);
|
||
color: rgba(var(--v-theme-on-surface), 0.6);
|
||
font-family: inherit;
|
||
font-size: 11px;
|
||
font-weight: 500;
|
||
line-height: 1;
|
||
padding-block: 3px;
|
||
padding-inline: 6px;
|
||
}
|
||
|
||
.shortcut-label {
|
||
color: rgba(var(--v-theme-on-surface), 0.45);
|
||
font-size: 12px;
|
||
}
|
||
|
||
/* 移动端底部关闭图标 */
|
||
.search-footer-mobile {
|
||
display: flex;
|
||
justify-content: center;
|
||
margin-block-start: auto;
|
||
padding-block: 12px;
|
||
padding-block-end: calc(12px + env(safe-area-inset-bottom));
|
||
}
|
||
|
||
/* 响应式 */
|
||
@media (width <= 600px) {
|
||
.search-header {
|
||
padding-block: 12px 10px;
|
||
padding-inline: 12px;
|
||
}
|
||
|
||
.search-input-wrapper {
|
||
block-size: 44px;
|
||
padding-inline: 12px 4px;
|
||
}
|
||
|
||
.search-native-input {
|
||
font-size: 14px;
|
||
}
|
||
|
||
.search-footer {
|
||
padding-block: 8px;
|
||
padding-inline: 12px;
|
||
}
|
||
}
|
||
</style>
|