新增了 PageHeader 通用组件,用于统一页面标题和操作按钮区域的布局。 该组件包含 title 属性以及 buttons 和 alerts 两个具名插槽,便于复用和维护。 已在以下页面中集成使用: - 图片管理页(ImageConfig.vue) - 音乐管理页(MusicConfig.vue) - 人员管理页(PersonAll/index.vue) - 中奖者管理页(PersonAlready.vue) - 奖品管理页(PrizeConfig.vue)
125 lines
3.6 KiB
Vue
125 lines
3.6 KiB
Vue
<script setup lang='ts'>
|
||
import type { IImage } from '@/types/storeType'
|
||
import localforage from 'localforage'
|
||
import { storeToRefs } from 'pinia'
|
||
import { onMounted, ref, watch } from 'vue'
|
||
import { useI18n } from 'vue-i18n'
|
||
import ImageSync from '@/components/ImageSync/index.vue'
|
||
import PageHeader from '@/components/PageHeader/index.vue'
|
||
import useStore from '@/store'
|
||
import { readFileData } from '@/utils/file'
|
||
|
||
const { t } = useI18n()
|
||
const globalConfig = useStore().globalConfig
|
||
const { getImageList: localImageList } = storeToRefs(globalConfig)
|
||
const limitType = ref('image/*')
|
||
const imgUploadToast = ref(0) // 0是不显示,1是成功,2是失败,3是不是图片
|
||
const imageDbStore = localforage.createInstance({
|
||
name: 'imgStore',
|
||
})
|
||
async function handleFileChange(e: Event) {
|
||
const isImage = /image*/.test(((e.target as HTMLInputElement).files as FileList)[0].type)
|
||
if (!isImage) {
|
||
imgUploadToast.value = 3
|
||
|
||
return
|
||
}
|
||
const { dataUrl, fileName } = await readFileData(((e.target as HTMLInputElement).files as FileList)[0])
|
||
imageDbStore.setItem(`${new Date().getTime().toString()}+${fileName}`, dataUrl)
|
||
.then(() => {
|
||
imgUploadToast.value = 1
|
||
getImageDbStore()
|
||
})
|
||
.catch(() => {
|
||
imgUploadToast.value = 2
|
||
})
|
||
}
|
||
|
||
async function getImageDbStore() {
|
||
const keys = await imageDbStore.keys()
|
||
if (keys.length > 0) {
|
||
imageDbStore.iterate((value, key) => {
|
||
globalConfig.addImage({
|
||
id: key,
|
||
name: key,
|
||
url: 'Storage',
|
||
})
|
||
})
|
||
}
|
||
}
|
||
|
||
function removeImage(item: IImage) {
|
||
if (item.url === 'Storage') {
|
||
imageDbStore.removeItem(item.id).then(() => {
|
||
globalConfig.removeImage(item.id)
|
||
})
|
||
}
|
||
globalConfig.removeImage(item.id)
|
||
}
|
||
onMounted(() => {
|
||
// getImageDbStore()
|
||
})
|
||
watch(() => imgUploadToast.value, (val) => {
|
||
if (val !== 0) {
|
||
setTimeout(() => {
|
||
imgUploadToast.value = 0
|
||
}, 2000)
|
||
}
|
||
})
|
||
</script>
|
||
|
||
<template>
|
||
<div class="toast toast-top toast-end">
|
||
<div v-if="imgUploadToast === 2" class="alert alert-error">
|
||
<span>{{ t('error.uploadFail') }}</span>
|
||
</div>
|
||
<div v-if="imgUploadToast === 1" class="alert alert-success">
|
||
<span>{{ t('error.uploadSuccess') }}</span>
|
||
</div>
|
||
<div v-if="imgUploadToast === 3" class="alert alert-error">
|
||
<span>{{ t('error.notImage') }}</span>
|
||
</div>
|
||
</div>
|
||
|
||
<div>
|
||
<PageHeader title="图片管理">
|
||
<template #buttons>
|
||
<div class="">
|
||
<label for="explore">
|
||
<input
|
||
id="explore" type="file" class="" style="display: none" :accept="limitType"
|
||
@change="handleFileChange"
|
||
>
|
||
<span class="btn btn-primary btn-sm">{{ t('button.upload') }}</span>
|
||
</label>
|
||
</div>
|
||
</template>
|
||
</PageHeader>
|
||
|
||
<ul class="p-0">
|
||
<li v-for="item in localImageList" :key="item.id" class="mb-3">
|
||
<div class="flex items-center gap-8">
|
||
<div class="avatar h-14">
|
||
<div class="w-12 h-12 mask mask-squircle hover:w-14 hover:h-14">
|
||
<!-- <img v-if="item.url!=='Storage'" :src="item.url" alt="Avatar Tailwind CSS Component" /> -->
|
||
<ImageSync :img-item="item" />
|
||
</div>
|
||
</div>
|
||
<div class="w-64">
|
||
<div class="overflow-hidden font-bold whitespace-nowrap text-ellipsis">
|
||
{{ item.name }}
|
||
</div>
|
||
</div>
|
||
<div>
|
||
<button class="btn btn-error btn-xs" @click="removeImage(item)">
|
||
{{ t('button.delete') }}
|
||
</button>
|
||
</div>
|
||
</div>
|
||
</li>
|
||
</ul>
|
||
</div>
|
||
</template>
|
||
|
||
<style lang='scss' scoped></style>
|