mirror of
https://github.com/jxxghp/MoviePilot-Frontend.git
synced 2026-05-30 21:00:43 +08:00
537 lines
12 KiB
Vue
537 lines
12 KiB
Vue
<script lang="ts" setup>
|
|
import { useToast } from 'vue-toast-notification'
|
|
import { useConfirm } from 'vuetify-use-dialog'
|
|
import api from '@/api'
|
|
import type { Plugin } from '@/api/types'
|
|
import FormRender from '@/components/render/FormRender.vue'
|
|
import PageRender from '@/components/render/PageRender.vue'
|
|
import { isNullOrEmptyObject } from '@core/utils'
|
|
import noImage from '@images/logos/plugin.png'
|
|
import { getDominantColor } from '@/@core/utils/image'
|
|
import store from '@/store'
|
|
|
|
// 输入参数
|
|
const props = defineProps({
|
|
plugin: Object as PropType<Plugin>,
|
|
width: String,
|
|
height: String,
|
|
})
|
|
|
|
// 定义触发的自定义事件
|
|
const emit = defineEmits(['remove', 'save'])
|
|
|
|
// 背景颜色
|
|
const backgroundColor = ref('#28A9E1')
|
|
|
|
// 图片对象
|
|
const imageRef = ref<any>()
|
|
|
|
// 提示框
|
|
const $toast = useToast()
|
|
|
|
// 确认框
|
|
const createConfirm = useConfirm()
|
|
|
|
// 本身是否可见
|
|
const isVisible = ref(true)
|
|
|
|
// 插件配置页面
|
|
const pluginConfigDialog = ref(false)
|
|
|
|
// 插件配置表单数据
|
|
const pluginConfigForm = ref({})
|
|
|
|
// 进度框
|
|
const progressDialog = ref(false)
|
|
|
|
// 插件表单配置项
|
|
let pluginFormItems = reactive([])
|
|
|
|
// 插件数据页面
|
|
const pluginInfoDialog = ref(false)
|
|
|
|
// 进度框文本
|
|
const progressText = ref('正在更新插件...')
|
|
|
|
// 插件数据页面配置项
|
|
let pluginPageItems = reactive([])
|
|
|
|
// 图片是否加载完成
|
|
const isImageLoaded = ref(false)
|
|
|
|
// 图片是否加载失败
|
|
const imageLoadError = ref(false)
|
|
|
|
// 图片加载完成
|
|
async function imageLoaded() {
|
|
isImageLoaded.value = true
|
|
const imageElement = imageRef.value?.$el.querySelector('img') as HTMLImageElement
|
|
// 从图片中提取背景色
|
|
backgroundColor.value = await getDominantColor(imageElement)
|
|
}
|
|
|
|
// 调用API卸载插件
|
|
async function uninstallPlugin() {
|
|
const isConfirmed = await createConfirm({
|
|
title: '确认',
|
|
content: `是否确认卸载插件 ${props.plugin?.plugin_name} ?`,
|
|
confirmationText: '确认',
|
|
cancellationText: '取消',
|
|
dialogProps: {
|
|
maxWidth: '50rem',
|
|
},
|
|
confirmationButtonProps: {
|
|
variant: 'tonal',
|
|
},
|
|
})
|
|
|
|
if (!isConfirmed)
|
|
return
|
|
|
|
try {
|
|
// 显示等待提示框
|
|
progressDialog.value = true
|
|
progressText.value = `正在卸载 ${props.plugin?.plugin_name} ...`
|
|
const result: { [key: string]: any } = await api.delete(`plugin/${props.plugin?.id}`)
|
|
// 隐藏等待提示框
|
|
progressDialog.value = false
|
|
if (result.success) {
|
|
$toast.success(`插件 ${props.plugin?.plugin_name} 已卸载`)
|
|
|
|
// 通知父组件刷新
|
|
emit('remove')
|
|
}
|
|
else {
|
|
$toast.error(`插件 ${props.plugin?.plugin_name} 卸载失败:${result.message}}`)
|
|
}
|
|
}
|
|
catch (error) {
|
|
console.error(error)
|
|
}
|
|
}
|
|
|
|
// 调用API读取表单页面
|
|
async function loadPluginForm() {
|
|
try {
|
|
const result: { [key: string]: any } = await api.get(`plugin/form/${props.plugin?.id}`)
|
|
if (result) {
|
|
pluginFormItems = result.conf
|
|
if (result.model)
|
|
pluginConfigForm.value = result.model
|
|
}
|
|
}
|
|
catch (error) {
|
|
console.error(error)
|
|
}
|
|
}
|
|
|
|
// 调用API读取数据页面
|
|
async function loadPluginPage() {
|
|
try {
|
|
const result: [] = await api.get(`plugin/page/${props.plugin?.id}`)
|
|
if (result)
|
|
pluginPageItems = result
|
|
}
|
|
catch (error) {
|
|
console.error(error)
|
|
}
|
|
}
|
|
|
|
// 调用API读取配置数据
|
|
async function loadPluginConf() {
|
|
try {
|
|
const result: { [key: string]: any } = await api.get(`plugin/${props.plugin?.id}`)
|
|
if (!isNullOrEmptyObject(result))
|
|
pluginConfigForm.value = result
|
|
}
|
|
catch (error) {
|
|
console.error(error)
|
|
}
|
|
}
|
|
|
|
// 调用API保存配置数据
|
|
async function savePluginConf() {
|
|
try {
|
|
const result: { [key: string]: any } = await api.put(`plugin/${props.plugin?.id}`, pluginConfigForm.value)
|
|
if (result.success) {
|
|
$toast.success(`插件 ${props.plugin?.plugin_name} 配置已保存`)
|
|
pluginConfigDialog.value = false
|
|
// 通知父组件刷新
|
|
emit('save')
|
|
}
|
|
else {
|
|
$toast.error(`插件 ${props.plugin?.plugin_name} 配置保存失败:${result.message}}`)
|
|
}
|
|
}
|
|
catch (error) {
|
|
console.error(error)
|
|
}
|
|
}
|
|
|
|
// 显示插件数据
|
|
async function showPluginInfo() {
|
|
// 加载数据
|
|
await loadPluginPage()
|
|
pluginConfigDialog.value = false
|
|
pluginInfoDialog.value = true
|
|
}
|
|
|
|
// 显示插件配置
|
|
async function showPluginConfig() {
|
|
// 加载表单
|
|
await loadPluginForm()
|
|
// 加载配置
|
|
await loadPluginConf()
|
|
// 显示对话框
|
|
pluginInfoDialog.value = false
|
|
pluginConfigDialog.value = true
|
|
}
|
|
|
|
// 计算图标路径
|
|
const iconPath: Ref<string> = computed(() => {
|
|
if (imageLoadError.value)
|
|
return noImage
|
|
// 如果是网络图片则使用代理后返回
|
|
if (props.plugin?.plugin_icon?.startsWith('http'))
|
|
return `${import.meta.env.VITE_API_BASE_URL}system/img/${encodeURIComponent(props.plugin?.plugin_icon)}/1`
|
|
|
|
return `./plugin_icon/${props.plugin?.plugin_icon}`
|
|
})
|
|
|
|
// 重置插件
|
|
async function resetPlugin() {
|
|
const isConfirmed = await createConfirm({
|
|
title: '确认',
|
|
content: `是否确认重置插件 ${props.plugin?.plugin_name} 的配置数据?`,
|
|
confirmationText: '确认',
|
|
cancellationText: '取消',
|
|
dialogProps: {
|
|
maxWidth: '50rem',
|
|
},
|
|
confirmationButtonProps: {
|
|
variant: 'tonal',
|
|
},
|
|
})
|
|
|
|
if (!isConfirmed)
|
|
return
|
|
|
|
try {
|
|
const result: { [key: string]: any } = await api.get(`plugin/reset/${props.plugin?.id}`)
|
|
if (result.success) {
|
|
$toast.success(`插件 ${props.plugin?.plugin_name} 数据已重置`)
|
|
// 通知父组件刷新
|
|
emit('save')
|
|
}
|
|
else {
|
|
$toast.error(`插件 ${props.plugin?.plugin_name} 重置失败:${result.message}}`)
|
|
}
|
|
}
|
|
catch (error) {
|
|
console.error(error)
|
|
}
|
|
}
|
|
|
|
// 更新插件
|
|
async function updatePlugin() {
|
|
try {
|
|
// 显示等待提示框
|
|
progressDialog.value = true
|
|
progressText.value = `正在更新 ${props.plugin?.plugin_name} ...`
|
|
|
|
const result: { [key: string]: any } = await api.get(
|
|
`plugin/install/${props.plugin?.id}`,
|
|
{
|
|
params: {
|
|
repo_url: props.plugin?.repo_url,
|
|
force: true,
|
|
},
|
|
},
|
|
)
|
|
|
|
// 隐藏等待提示框
|
|
progressDialog.value = false
|
|
|
|
if (result.success) {
|
|
$toast.success(`插件 ${props.plugin?.plugin_name} 更新成功!`)
|
|
|
|
// 通知父组件刷新
|
|
emit('save')
|
|
}
|
|
else {
|
|
$toast.error(`插件 ${props.plugin?.plugin_name} 更新失败:${result.message}`)
|
|
}
|
|
}
|
|
catch (error) {
|
|
console.error(error)
|
|
}
|
|
}
|
|
|
|
// 访问作者主页
|
|
function visitAuthorPage() {
|
|
window.open(props.plugin?.author_url, '_blank')
|
|
}
|
|
|
|
// 查看日志URL
|
|
function openLoggerWindow() {
|
|
const token = store.state.auth.token
|
|
const url = `${import.meta.env.VITE_API_BASE_URL}system/logging?token=${token}&length=-1&logfile=plugins/${props.plugin?.id?.toLowerCase()}.log`
|
|
window.open(url, '_blank')
|
|
}
|
|
|
|
// 弹出菜单
|
|
const dropdownItems = ref([
|
|
{
|
|
title: '查看数据',
|
|
value: 1,
|
|
show: props.plugin?.has_page,
|
|
props: {
|
|
prependIcon: 'mdi-information-outline',
|
|
click: showPluginInfo,
|
|
},
|
|
},
|
|
{
|
|
title: '设置',
|
|
value: 2,
|
|
show: true,
|
|
props: {
|
|
prependIcon: 'mdi-cog-outline',
|
|
click: showPluginConfig,
|
|
},
|
|
},
|
|
{
|
|
title: '更新',
|
|
value: 3,
|
|
show: props.plugin?.has_update,
|
|
props: {
|
|
prependIcon: 'mdi-cancel',
|
|
color: 'success',
|
|
click: updatePlugin,
|
|
},
|
|
},
|
|
{
|
|
title: '重置',
|
|
value: 4,
|
|
show: true,
|
|
props: {
|
|
prependIcon: 'mdi-cancel',
|
|
color: 'warning',
|
|
click: resetPlugin,
|
|
},
|
|
},
|
|
{
|
|
title: '卸载',
|
|
value: 5,
|
|
show: true,
|
|
props: {
|
|
prependIcon: 'mdi-trash-can-outline',
|
|
color: 'error',
|
|
click: uninstallPlugin,
|
|
},
|
|
},
|
|
{
|
|
title: '查看日志',
|
|
value: 6,
|
|
show: true,
|
|
props: {
|
|
prependIcon: 'mdi-file-document-outline',
|
|
click: () => {
|
|
openLoggerWindow()
|
|
},
|
|
},
|
|
},
|
|
{
|
|
title: '作者主页',
|
|
value: 7,
|
|
show: true,
|
|
props: {
|
|
prependIcon: 'mdi-home-circle-outline',
|
|
click: visitAuthorPage,
|
|
},
|
|
},
|
|
])
|
|
|
|
// 监听插件状态变化
|
|
watch(() => props.plugin?.has_update, (newHasUpdate, oldHasUpdate) => {
|
|
const updateItemIndex = dropdownItems.value.findIndex(item => item.value === 3)
|
|
if (updateItemIndex !== -1)
|
|
dropdownItems.value[updateItemIndex].show = newHasUpdate
|
|
})
|
|
</script>
|
|
|
|
<template>
|
|
<!-- 插件卡片 -->
|
|
<VCard
|
|
v-if="isVisible"
|
|
:width="props.width"
|
|
:height="props.height"
|
|
@click="() => {
|
|
if (props.plugin?.has_page)
|
|
showPluginInfo()
|
|
else
|
|
showPluginConfig()
|
|
}"
|
|
>
|
|
<div
|
|
class="relative pa-4 text-center card-cover-blurred"
|
|
:style="{ background: `${backgroundColor}` }"
|
|
>
|
|
<div
|
|
v-if="props.plugin?.has_update"
|
|
class="me-n3 absolute top-0 left-1"
|
|
>
|
|
<VIcon
|
|
icon="mdi-new-box"
|
|
class="text-white"
|
|
/>
|
|
</div>
|
|
<div class="me-n3 absolute top-0 right-3">
|
|
<IconBtn>
|
|
<VIcon icon="mdi-dots-vertical" class="text-white" />
|
|
<VMenu
|
|
activator="parent"
|
|
close-on-content-click
|
|
>
|
|
<VList>
|
|
<VListItem
|
|
v-for="(item, i) in dropdownItems"
|
|
v-show="item.show"
|
|
:key="i"
|
|
variant="plain"
|
|
:base-color="item.props.color"
|
|
@click="item.props.click"
|
|
>
|
|
<template #prepend>
|
|
<VIcon :icon="item.props.prependIcon" />
|
|
</template>
|
|
<VListItemTitle v-text="item.title" />
|
|
</VListItem>
|
|
</VList>
|
|
</VMenu>
|
|
</IconBtn>
|
|
</div>
|
|
<VAvatar
|
|
size="8rem"
|
|
>
|
|
<VImg
|
|
ref="imageRef"
|
|
:src="iconPath"
|
|
aspect-ratio="4/3"
|
|
cover
|
|
:class="{ shadow: isImageLoaded }"
|
|
@load="imageLoaded"
|
|
@error="imageLoadError = true"
|
|
/>
|
|
</VAvatar>
|
|
</div>
|
|
<VCardItem class="py-2">
|
|
<VCardTitle class="flex items-center flex-row">
|
|
<VBadge v-if="props.plugin?.state" dot inline color="success" class="me-1 mb-1" />
|
|
{{ props.plugin?.plugin_name }}<span class="text-sm ms-2 mt-1 text-gray-500">v{{ props.plugin?.plugin_version }}</span>
|
|
</VCardTitle>
|
|
</VCardItem>
|
|
<VCardText>
|
|
{{ props.plugin?.plugin_desc }}
|
|
</VCardText>
|
|
</VCard>
|
|
<!-- 插件配置页面 -->
|
|
<VDialog
|
|
v-model="pluginConfigDialog"
|
|
scrollable
|
|
max-width="60rem"
|
|
>
|
|
<VCard
|
|
:title="`${props.plugin?.plugin_name} - 配置`"
|
|
class="rounded-t"
|
|
>
|
|
<DialogCloseBtn @click="pluginConfigDialog = false" />
|
|
<VCardText>
|
|
<FormRender
|
|
v-for="(item, index) in pluginFormItems"
|
|
:key="index"
|
|
:config="item"
|
|
:form="pluginConfigForm"
|
|
/>
|
|
</VCardText>
|
|
<VCardActions>
|
|
<VBtn v-if="pluginPageItems.length > 0" @click="showPluginInfo">
|
|
查看数据
|
|
</VBtn>
|
|
<VSpacer />
|
|
<VBtn
|
|
variant="tonal"
|
|
@click="savePluginConf"
|
|
>
|
|
保存
|
|
</VBtn>
|
|
</VCardActions>
|
|
</VCard>
|
|
</VDialog>
|
|
|
|
<!-- 插件数据页面 -->
|
|
<VDialog
|
|
v-model="pluginInfoDialog"
|
|
scrollable
|
|
max-width="80rem"
|
|
>
|
|
<VCard
|
|
:title="`${props.plugin?.plugin_name}`"
|
|
class="rounded-t"
|
|
>
|
|
<DialogCloseBtn @click="pluginInfoDialog = false" />
|
|
<VCardText>
|
|
<PageRender
|
|
v-for="(item, index) in pluginPageItems"
|
|
:key="index"
|
|
:config="item"
|
|
/>
|
|
</VCardText>
|
|
<VCardActions>
|
|
<VBtn
|
|
@click="showPluginConfig"
|
|
>
|
|
配置
|
|
</VBtn>
|
|
<VSpacer />
|
|
<VBtn
|
|
variant="tonal"
|
|
@click="pluginInfoDialog = false"
|
|
>
|
|
关闭
|
|
</VBtn>
|
|
</VCardActions>
|
|
</VCard>
|
|
</VDialog>
|
|
<!-- 更新插件进度框 -->
|
|
<VDialog
|
|
v-model="progressDialog"
|
|
:scrim="false"
|
|
width="25rem"
|
|
>
|
|
<VCard
|
|
color="primary"
|
|
>
|
|
<VCardText class="text-center">
|
|
{{ progressText }}
|
|
<VProgressLinear
|
|
indeterminate
|
|
color="white"
|
|
class="mb-0 mt-1"
|
|
/>
|
|
</VCardText>
|
|
</VCard>
|
|
</VDialog>
|
|
</template>
|
|
|
|
<style lang="scss" scoped>
|
|
.card-cover-blurred::before {
|
|
position: absolute;
|
|
/* stylelint-disable-next-line property-no-vendor-prefix */
|
|
-webkit-backdrop-filter: blur(2px);
|
|
backdrop-filter: blur(2px);
|
|
background: rgba(29, 39, 59, 48%);
|
|
content: "";
|
|
inset: 0;
|
|
}
|
|
</style>
|