Commit ea5de822 by lijiabin

【需求 20331】 feat: 完善后台管理页面优化菜单等、完成前后台限时竞拍相关功能

parent 2a7fbb51
/**
* 确认兑换商品的弹窗内容
*/
import { ShopGoodsTypeEnum, regionListOptions } from '@/constants'
import type { BackendShopItemDto, ExchangeGoodsParams } from '@/api'
import type { SetupContext } from 'vue'
type ExchangeContentProps = {
item: BackendShopItemDto
modelValue: ExchangeGoodsParams
}
type ExchangeContentEvents = {
'update:modelValue'(data: ExchangeGoodsParams): void
}
export default function ExchangeContent(
{ item, modelValue }: ExchangeContentProps,
context: SetupContext<ExchangeContentEvents>,
) {
return (
<div class="exchange-content py-6 px-4">
{/* 商品图片区域 */}
<div class="flex justify-center mb-8">
<div class="relative">
<div class="w-32 h-32 bg-gradient-to-br from-orange-100 to-pink-100 rounded-3xl flex items-center justify-center shadow-lg">
<div class="w-20 h-20 bg-white rounded-lg flex items-center justify-center shadow-sm">
<img src={item.imageUrl} alt={item.name} class="w-16 h-16 object-contain" />
</div>
</div>
<div class="absolute -top-2 -right-2 w-7 h-7 bg-blue-500 rounded-full flex items-center justify-center shadow-md">
<span class="text-white text-sm font-medium">{item.stock}</span>
</div>
</div>
</div>
{/* 商品信息 */}
<div class="space-y-3 mb-8">
<div class="flex items-center gap-3 px-4">
<div class="w-1.5 h-1.5 bg-gray-400 rounded-full flex-shrink-0"></div>
<span class="text-gray-600 text-sm min-w-12">名称:</span>
<span class="font-medium text-gray-900 flex-1">{item.name}</span>
</div>
<div class="flex items-center gap-3 px-4">
<div class="w-1.5 h-1.5 bg-orange-400 rounded-full flex-shrink-0"></div>
<span class="text-gray-600 text-sm min-w-12">积分:</span>
<span class="font-semibold text-orange-500 text-lg">{item.price}YA币</span>
</div>
</div>
{/* 办公点选择和数量 */}
{item.itemType === ShopGoodsTypeEnum.REAL_GOODS && (
<div class=" rounded-lg px-5 mx-2 space-y-4">
{/* 办公点选择 */}
<div>
<label class="text-gray-700 text-sm font-medium mb-2 block">办公点</label>
<el-select
modelValue={modelValue.deliveryInfo}
onUpdate:modelValue={(value: string) =>
context.emit('update:modelValue', { ...modelValue, deliveryInfo: value })
}
placeholder="请选择办公点"
class="w-full"
>
{regionListOptions.map((office) => (
<el-option key={office.value} label={office.label} value={office.value} />
))}
</el-select>
</div>
{/* 数量选择 */}
<div>
<label class="text-gray-700 text-sm font-medium mb-2 block">选择数量</label>
<el-input-number
min={1}
max={item.stock}
modelValue={modelValue.num}
onUpdate:modelValue={(value: number) =>
context.emit('update:modelValue', { ...modelValue, num: value })
}
class="w-full"
controls-position="right"
/>
</div>
</div>
)}
</div>
)
}
ExchangeContent.props = {
item: {
type: Object as PropType<BackendShopItemDto>,
required: true,
},
modelValue: {
type: Object as PropType<ExchangeGoodsParams>,
required: true,
},
}
ExchangeContent.emits = {
'update:modelValue': (value: ExchangeGoodsParams) => value,
}
<template>
<el-dialog
v-model="visible"
title="商品领取记录"
width="1100px"
class="exchange-record-dialog"
align-center
destroy-on-close
>
<div class="px-6 py-4">
<!-- 搜索筛选区域 - 优化布局 -->
<div class="mb-4 flex items-center gap-3 justify-end">
<div class="w-750px flex items-center gap-3">
<el-select v-model="searchParams.itemType" placeholder="选择商品类型" clearable>
<el-option label="亚声实物" :value="ShopGoodsTypeEnum.REAL_GOODS" />
<el-option label="虚拟装饰" :value="ShopGoodsTypeEnum.VIRTUAL_GOODS" />
</el-select>
<el-date-picker
v-model="timeRange"
type="daterange"
range-separator="至"
start-placeholder="开始日期"
end-placeholder="结束日期"
format="YYYY-MM-DD"
value-format="x"
clearable
style="width: 700px"
/>
<el-button type="primary" @click="refresh" style="width: 80px"> 搜索 </el-button>
</div>
</div>
<!-- 表格 - 调整列宽 -->
<el-table :data="list" stripe height="420" class="w-full">
<el-table-column prop="itemName" label="名称" min-width="140" show-overflow-tooltip />
<el-table-column prop="itemType" label="商品类型" width="100" align="center">
<template #default="scope">
{{ scope.row.itemType === ShopGoodsTypeEnum.REAL_GOODS ? '亚声实物' : '虚拟装饰' }}
</template>
</el-table-column>
<!-- <el-table-column prop="price" label="商品单价" width="100" align="center" /> -->
<el-table-column prop="num" label="兑换数量" width="100" align="center" />
<el-table-column label="扣除YA币" width="110" align="center">
<template #default="scope">
<span class="text-red-500 font-semibold"> -{{ scope.row.price }} </span>
</template>
</el-table-column>
<el-table-column
prop="deliveryInfo"
label="收货地址"
min-width="160"
show-overflow-tooltip
/>
<el-table-column prop="createTime" label="领取时间" width="200">
<template #default="scope">
{{ dayjs(scope.row.createTime * 1000).format('YYYY-MM-DD HH:mm:ss') }}
</template>
</el-table-column>
</el-table>
<!-- 分页 - 优化对齐 -->
<div class="flex justify-end mt-4">
<el-pagination
v-model:current-page="searchParams.current"
:page-size="searchParams.size"
@current-change="goToPage"
:total="total"
layout="total, prev, pager, next"
small
/>
</div>
</div>
</el-dialog>
</template>
<script setup lang="ts">
import { usePageSearch } from '@/hooks'
import { getExchangeGoodsRecordList } from '@/api'
import dayjs from 'dayjs'
import { ShopGoodsTypeEnum } from '@/constants'
const { total, searchParams, refresh, goToPage, list } = usePageSearch(getExchangeGoodsRecordList, {
immediate: false,
})
const timeRange = computed({
get() {
if (!searchParams.value.receiveTimeStart || !searchParams.value.receiveTimeEnd) {
return []
} else {
return [searchParams.value.receiveTimeStart * 1000, searchParams.value.receiveTimeEnd * 1000]
}
},
set(value) {
console.log(value)
if (!value) {
searchParams.value.receiveTimeStart = 0
searchParams.value.receiveTimeEnd = 0
} else {
searchParams.value.receiveTimeStart = value[0]! / 1000
searchParams.value.receiveTimeEnd = value[1]! / 1000
}
},
})
const visible = shallowRef(false)
const open = () => {
visible.value = true
refresh()
}
defineExpose({
open,
})
</script>
<style scoped></style>
<template>
<el-dialog
v-model="visible"
title="YA币收支记录"
width="1100px"
class="exchange-record-dialog"
align-center
destroy-on-close
>
<div class="px-6 py-4">
<!-- 搜索筛选区域 - 优化布局 -->
<div class="mb-4 flex items-center gap-3 justify-end">
<div class="w-750px flex items-center gap-3">
<!-- <el-select v-model="searchParams.type" placeholder="选择商品类型" clearable>
<el-option label="亚声实物" :value="ShopGoodsTypeEnum.REAL_GOODS" />
<el-option label="虚拟装饰" :value="ShopGoodsTypeEnum.VIRTUAL_GOODS" />
</el-select> -->
<el-date-picker
v-model="timeRange"
type="daterange"
range-separator="至"
start-placeholder="开始日期"
end-placeholder="结束日期"
format="YYYY-MM-DD"
value-format="x"
clearable
style="width: 700px"
/>
<el-button type="primary" @click="onSearch" style="width: 80px"> 搜索 </el-button>
</div>
</div>
<!-- 表格 - 调整列宽 -->
<el-table :data="list" stripe height="420" class="w-full">
<el-table-column prop="remark" label="名称" min-width="140" show-overflow-tooltip />
<el-table-column prop="itemType" label="YA币" width="100" align="center">
<template #default="scope">
<span
class="font-semibold"
:class="{ 'text-red-500': !scope.row.isIncr, 'text-green-500': scope.row.isIncr }"
>
<!-- {{ scope.row.isIncr ? '+' : '-' }} -->
{{ scope.row.scoreAyabi }}
</span>
</template>
</el-table-column>
<el-table-column prop="createdAt" label="时间" width="300">
<template #default="scope">
{{ dayjs(scope.row.createdAt * 1000).format('YYYY-MM-DD HH:mm:ss') }}
</template>
</el-table-column>
</el-table>
<!-- 分页 - 优化对齐 -->
<div class="flex justify-end mt-4">
<el-pagination
v-model:current-page="searchParams.current"
:page-size="searchParams.size"
@current-change="goToPage"
:total="total"
layout="total, prev, pager, next"
small
/>
</div>
</div>
</el-dialog>
</template>
<script setup lang="ts">
import { usePageSearch } from '@/hooks'
import { getExchangeYabiRecordList } from '@/api'
import dayjs from 'dayjs'
const { total, searchParams, refresh, goToPage, list } = usePageSearch(getExchangeYabiRecordList, {
immediate: false,
})
const timeRange = computed({
get() {
if (!searchParams.value.startTime || !searchParams.value.endTime) {
return []
} else {
return [searchParams.value.startTime * 1000, searchParams.value.endTime * 1000]
}
},
set(value) {
if (!value) {
searchParams.value.startTime = undefined
searchParams.value.endTime = undefined
} else {
searchParams.value.startTime = value[0]! / 1000
searchParams.value.endTime = value[1]! / 1000
}
},
})
const visible = shallowRef(false)
const open = () => {
visible.value = true
refresh()
}
const onSearch = () => {
refresh()
}
defineExpose({
open,
})
</script>
<style scoped></style>
......@@ -359,14 +359,9 @@ onMounted(() => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
flex-shrink: 0;
.add-btn {
margin-left: auto;
}
}
// 表格区域
.table-section {
flex: 1;
......
......@@ -3,56 +3,50 @@
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input
v-model="searchParams.title"
placeholder="请输入栏目标题"
class="w-200px"
></el-input>
<!-- 审核状态 -->
<el-select
v-model="searchParams.isAudit"
placeholder="请选择审核状态"
class="search-select"
clearable
>
<el-option
v-for="item in auditTypeListOptions"
:key="item.value"
:label="item.label"
:value="item.value"
/>
</el-select>
<!-- 发布状态 -->
<el-select
v-model="searchParams.isUse"
placeholder="请选择发布状态"
class="search-select"
clearable
>
<el-option
v-for="item in usageStatusListOptions"
:key="item.value"
:label="item.label"
:value="item.value"
/>
</el-select>
<div>
<el-button type="primary" @click="refresh">
<el-icon><IEpSearch /></el-icon>
搜索</el-button
>
<el-button @click="reset">重置</el-button>
</div>
</div>
<div class="flex justify-end">
<UploadExcel :api="importCaseExcel" @success="refresh" @error="handleError" />
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
<el-input
v-model="searchParams.title"
placeholder="请输入栏目标题"
class="w-200px"
></el-input>
<!-- 审核状态 -->
<el-select
v-model="searchParams.isAudit"
placeholder="请选择审核状态"
class="search-select mx-12px"
clearable
>
<el-option
v-for="item in auditTypeListOptions"
:key="item.value"
:label="item.label"
:value="item.value"
/>
</el-select>
<!-- 发布状态 -->
<el-select
v-model="searchParams.isUse"
placeholder="请选择发布状态"
class="search-select mr-12px"
clearable
>
<el-option
v-for="item in usageStatusListOptions"
:key="item.value"
:label="item.label"
:value="item.value"
/>
</el-select>
<el-button type="primary" @click="refresh">
<el-icon><IEpSearch /></el-icon>
搜索</el-button
>
<el-button @click="reset">重置</el-button>
<ImportExcelBtn :api="importCaseExcel" @success="refresh" @error="handleError" />
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
<!-- 表格区域 -->
......@@ -297,7 +291,7 @@ import {
AuditStatusEnum,
UsageStatusEnum,
} from '@/constants'
import UploadExcel from '@/components/common/UploadExcel/index.vue'
import ImportExcelBtn from '@/components/common/ImportExcelBtn/index.vue'
import dayjs from 'dayjs'
import ErrorListDialog from './components/errorListDialog.vue'
import EditDialog from './components/editDialog.vue'
......@@ -397,8 +391,9 @@ const handleError = (list: BackendCaseListItemDto[]) => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
flex-shrink: 0;
gap: 12px 0;
.search-select {
width: 200px;
......
<!-- views/backend/official/index.vue -->
<template>
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input
v-model="searchParams.title"
placeholder="请输入栏目标题"
class="w-200px"
clearable
></el-input>
<el-select
v-model="searchParams.status"
placeholder="请选择发布状态"
class="search-select"
clearable
>
<el-option label="发布" :value="1" />
<el-option label="隐藏" :value="0" />
</el-select>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<div class="flex justify-end">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
新增
</el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
</div>
<!-- 表格区域 -->
<div class="table-section">
<!-- 表格 -->
<div class="table-wrapper">
<el-table
v-loading="loading"
:data="list"
height="100%"
@selection-change="handleSelectionChange"
>
<el-table-column type="selection" width="55"> </el-table-column>
<el-table-column prop="sort" label="栏目顺序" width="180"> </el-table-column>
<el-table-column prop="title" label="栏目名称" min-width="200" />
<el-table-column prop="postCount" label="栏目帖子数量" min-width="200" />
<el-table-column prop="color" label="颜色" width="300">
<template #default="{ row }">
<div class="color-cell">
<div class="color-block" :style="{ backgroundColor: row.color }" />
<span class="color-text">{{ row.color }}</span>
</div>
</template>
</el-table-column>
<el-table-column prop="createUserId" label="创建人" min-width="200" />
<el-table-column prop="createTime" label="创建时间" min-width="200">
<template #default="{ row }">
{{ dayjs(row.createTime * 1000).format('YYYY-MM-DD HH:mm:ss') }}
</template>
</el-table-column>
<el-table-column prop="status" label="状态" min-width="200">
<template #default="{ row }">
<el-switch
:model-value="row.status"
:active-value="1"
:inactive-value="0"
@change="handleStatusChange(row)"
/>
</template>
</el-table-column>
<el-table-column label="操作" width="150" fixed="right">
<template #default="{ row }">
<el-button type="primary" link @click="handleEdit(row)">编辑</el-button>
<el-button type="danger" link @click="handleDelete(row)">删除</el-button>
</template>
</el-table-column>
</el-table>
</div>
<!-- 分页 -->
<div class="pagination-wrapper">
<el-pagination
v-model:current-page="searchParams.current"
v-model:page-size="searchParams.size"
:total="total"
:page-sizes="[10, 20, 30]"
layout="total, sizes, prev, pager, next, jumper"
@size-change="changePageSize"
@current-change="goToPage"
/>
</div>
</div>
<!-- 新增/编辑对话框 -->
<el-dialog
v-model="dialogVisible"
:title="dialogTitle"
width="500px"
:close-on-click-modal="false"
>
<el-form ref="formRef" :model="form" :rules="formRules" label-width="80px">
<el-form-item label="标题" prop="title">
<el-input v-model="form.title" placeholder="请输入标题" />
</el-form-item>
<el-form-item label="颜色" prop="color">
<el-color-picker v-model="form.color" />
<span class="color-value">{{ form.color }}</span>
</el-form-item>
<el-form-item label="排序" prop="sort">
<el-input-number v-model="form.sort" :min="0" :max="100" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit">
<el-icon class="btn-icon"><Upload /></el-icon>
保存
</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup lang="ts">
import { Search, Plus, Upload } from '@element-plus/icons-vue'
import { usePageSearch, useResetData } from '@/hooks'
import { listOfCultureColumn, addOrUpdateColumn, deleteColumn, hideColumn } from '@/api/backend'
import type { FormInstance, FormRules } from 'element-plus'
import type { BackendColumnListItemDto, AddOrUpdateColumnDto } from '@/api/backend'
import dayjs from 'dayjs'
const { loading, list, total, reset, goToPage, changePageSize, refresh, searchParams, search } =
usePageSearch(listOfCultureColumn, {
defaultParams: {
type: 'column',
},
})
// 对话框
const dialogVisible = ref(false)
const dialogTitle = computed(() => (form.value.id ? '编辑标签' : '新增标签'))
const formRef = ref<FormInstance>()
// 表单数据
const [form, resetForm] = useResetData<AddOrUpdateColumnDto>({
title: '',
color: '#000000',
id: undefined,
sort: 0,
type: 'column',
})
// 表单验证规则
const formRules: FormRules = {
title: [{ required: true, message: '请输入标题', trigger: 'blur' }],
color: [{ required: true, message: '请选择颜色', trigger: 'change' }],
sort: [{ required: true, message: '请输入排序', trigger: 'blur' }],
}
// 新增
const handleAdd = () => {
resetForm()
dialogVisible.value = true
}
// 编辑
const handleEdit = (row: BackendColumnListItemDto) => {
resetForm()
form.value = {
title: row.title,
color: row.color,
id: row.id,
sort: row.sort,
type: 'column',
}
dialogVisible.value = true
}
// 状态改变
const handleStatusChange = async (row: BackendColumnListItemDto) => {
await hideColumn([row.id])
refresh()
}
// 删除
const handleDelete = async (row: BackendColumnListItemDto) => {
try {
await ElMessageBox.confirm(`确定要删除标签"${row.title}"吗?`, '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning',
})
await deleteColumn([row.id])
ElMessage.success('删除成功')
refresh()
} catch (error) {
if (error !== 'cancel') {
ElMessage.error('删除失败')
}
}
}
// 提交表单
const handleSubmit = async () => {
if (!formRef.value) return
try {
await formRef.value.validate()
if (form.value.id) {
await addOrUpdateColumn(form.value)
} else {
await addOrUpdateColumn(form.value)
}
ElMessage.success(form.value.id ? '编辑成功' : '新增成功')
dialogVisible.value = false
if (form.value.id) {
search()
} else {
refresh()
}
} catch (error) {
console.error('表单验证失败:', error)
}
}
const selectedRows = ref<BackendColumnListItemDto[]>([])
// 选择
const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
selectedRows.value = selection
}
// 批量发布/隐藏
const handleBatchPublish = async () => {
await hideColumn(selectedRows.value.map((item) => item.id))
refresh()
selectedRows.value = []
ElMessage.success('发布/隐藏成功')
}
// 批量删除
const handleBatchDelete = async () => {
await deleteColumn(selectedRows.value.map((item) => item.id))
refresh()
selectedRows.value = []
ElMessage.success('删除成功')
}
</script>
<style scoped lang="scss">
.official-tag-page {
height: 100%;
display: flex;
flex-direction: column;
gap: 16px;
}
// 搜索区域
.search-section {
background: #fff;
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-shrink: 0;
.search-select {
width: 200px;
}
}
// 表格区域
.table-section {
flex: 1;
background: #fff;
border-radius: 8px;
padding: 20px;
display: flex;
flex-direction: column;
min-height: 0;
}
.table-wrapper {
flex: 1;
min-height: 0;
.color-cell {
display: flex;
align-items: center;
gap: 12px;
.color-block {
width: 100%;
height: 36px;
border-radius: 4px;
border: 1px solid #e5e7eb;
flex: 1;
}
.color-text {
color: #fff;
font-size: 14px;
font-weight: 500;
position: absolute;
left: 50%;
transform: translateX(-50%);
text-shadow: 0 0 3px rgba(0, 0, 0, 0.5);
}
}
}
.pagination-wrapper {
display: flex;
justify-content: flex-end;
padding-top: 16px;
flex-shrink: 0;
}
// 对话框内的颜色显示
.color-value {
margin-left: 12px;
color: #606266;
font-family: monospace;
}
.btn-icon {
margin-right: 4px;
}
</style>
......@@ -3,35 +3,31 @@
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input
v-model="searchParams.title"
placeholder="请输入栏目标题"
class="w-200px"
clearable
></el-input>
<el-select
v-model="searchParams.status"
placeholder="请选择发布状态"
class="search-select"
clearable
>
<el-option label="发布" :value="1" />
<el-option label="隐藏" :value="0" />
</el-select>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<div class="flex justify-end">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
新增
</el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
<el-input
v-model="searchParams.title"
placeholder="请输入栏目标题"
class="w-200px"
clearable
></el-input>
<el-select
v-model="searchParams.status"
placeholder="请选择发布状态"
class="search-select mx-12px"
clearable
>
<el-option label="发布" :value="1" />
<el-option label="隐藏" :value="0" />
</el-select>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
新增
</el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
<!-- 表格区域 -->
......@@ -271,7 +267,8 @@ const handleBatchDelete = async () => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0;
.search-select {
......
......@@ -3,35 +3,37 @@
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input
v-model="searchParams.title"
placeholder="请输入栏目标题"
class="w-200px"
clearable
></el-input>
<el-select
v-model="searchParams.status"
placeholder="请选择发布状态"
class="search-select"
clearable
>
<el-option label="发布" :value="1" />
<el-option label="隐藏" :value="0" />
</el-select>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<div class="flex justify-end">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
新增
</el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
<el-input
v-model="searchParams.title"
placeholder="请输入栏目标题"
class="w-200px"
clearable
></el-input>
<el-select
v-model="searchParams.status"
placeholder="请选择发布状态"
class="search-select mx-12px"
clearable
>
<el-option label="发布" :value="1" />
<el-option label="隐藏" :value="0" />
</el-select>
<el-button type="primary" @click="refresh">
<el-icon><IEpSearch /></el-icon>
搜索
</el-button>
<el-button type="primary" @click="refresh">
<el-icon><Search /></el-icon>
搜索
</el-button>
<el-button @click="reset">重置</el-button>
<el-button type="primary" @click="handleAdd">
<el-icon><IEpPlus /></el-icon>
新增
</el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
<!-- 表格区域 -->
......@@ -120,7 +122,7 @@
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit">
<el-icon class="btn-icon"><Upload /></el-icon>
<el-icon class="btn-icon"><IEpUpload /></el-icon>
保存
</el-button>
</template>
......@@ -129,7 +131,7 @@
</template>
<script setup lang="ts">
import { Search, Plus, Upload } from '@element-plus/icons-vue'
import { Search } from '@element-plus/icons-vue'
import { usePageSearch, useResetData } from '@/hooks'
import { listOfCultureColumn, addOrUpdateColumn, deleteColumn, hideColumn } from '@/api/backend'
import type { FormInstance, FormRules } from 'element-plus'
......@@ -271,7 +273,8 @@ const handleBatchDelete = async () => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0;
.search-select {
......
......@@ -2,35 +2,31 @@
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input
v-model="searchParams.title"
placeholder="请输入栏目标题"
class="w-200px"
clearable
></el-input>
<el-select
v-model="searchParams.status"
placeholder="请选择发布状态"
class="search-select"
clearable
>
<el-option label="发布" :value="1" />
<el-option label="隐藏" :value="0" />
</el-select>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<div class="flex justify-end">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
新增
</el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
<el-input
v-model="searchParams.title"
placeholder="请输入栏目标题"
class="w-200px"
clearable
></el-input>
<el-select
v-model="searchParams.status"
placeholder="请选择发布状态"
class="search-select mx-12px"
clearable
>
<el-option label="发布" :value="1" />
<el-option label="隐藏" :value="0" />
</el-select>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
新增
</el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
<!-- 表格区域 -->
......@@ -270,7 +266,8 @@ const handleBatchDelete = async () => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0;
.search-select {
......
......@@ -112,13 +112,13 @@ const handleShowContent = (row: ArticleItemDto) => {
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input v-model="searchParams.title" placeholder="请输入标题" class="w-200px"></el-input>
<div>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
</div>
<el-input
v-model="searchParams.title"
placeholder="请输入标题"
class="w-200px mr-12px"
></el-input>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<!-- 表格区域 -->
......@@ -251,7 +251,8 @@ const handleShowContent = (row: ArticleItemDto) => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0;
.search-select {
......
......@@ -112,13 +112,13 @@ const handleShowContent = (row: ArticleItemDto) => {
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input v-model="searchParams.title" placeholder="请输入标题" class="w-200px"></el-input>
<div>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
</div>
<el-input
v-model="searchParams.title"
placeholder="请输入标题"
class="w-200px mr-12px"
></el-input>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<!-- 表格区域 -->
......@@ -251,7 +251,8 @@ const handleShowContent = (row: ArticleItemDto) => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0;
.search-select {
......
......@@ -112,13 +112,13 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input v-model="searchParams.title" placeholder="请输入标题" class="w-200px"></el-input>
<div>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
</div>
<el-input
v-model="searchParams.title"
placeholder="请输入标题"
class="w-200px mr-12px"
></el-input>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<!-- 表格区域 -->
......@@ -259,7 +259,8 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0;
.search-select {
......
......@@ -102,13 +102,13 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input v-model="searchParams.title" placeholder="请输入标题" class="w-200px"></el-input>
<div>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
</div>
<el-input
v-model="searchParams.title"
placeholder="请输入标题"
class="w-200px mr-12px"
></el-input>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<!-- 表格区域 -->
......@@ -252,7 +252,8 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0;
.search-select {
......
......@@ -2,36 +2,43 @@
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input
v-model="searchParams.itemName"
placeholder="请输入商品名称"
class="w-200px"
></el-input>
<el-select
v-model="searchParams.status"
placeholder="请选择发布状态"
class="search-select"
clearable
>
<el-option label="待发货" :value="0" />
<el-option label="已发货" :value="1" />
<el-option label="已取消" :value="2" />
</el-select>
<el-select
v-model="searchParams.source"
placeholder="请选择来源"
class="search-select"
clearable
>
<el-option label="商城" :value="1" />
<el-option label="大转盘" :value="2" />
<el-option label="每日抽奖" :value="3" />
</el-select>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<el-input
v-model="searchParams.itemName"
placeholder="请输入商品名称"
class="w-200px"
></el-input>
<el-select
v-model="searchParams.status"
placeholder="请选择发布状态"
class="search-select mx-12px"
clearable
>
<el-option label="待发货" :value="0" />
<el-option label="已发货" :value="1" />
<el-option label="已取消" :value="2" />
</el-select>
<el-select
v-model="searchParams.source"
placeholder="请选择来源"
class="search-select"
clearable
>
<el-option label="商城" :value="1" />
<el-option label="大转盘" :value="2" />
<el-option label="每日抽奖" :value="3" />
</el-select>
<el-button type="primary" @click="refresh" class="ml-12px">
<el-icon><IEpSearch /></el-icon>
搜索
</el-button>
<el-button @click="reset">重置</el-button>
<ExportExcelBtn
:api="exportShopItemList"
:searchParams="searchParams"
:columns="columns"
fileName="商品分发数据"
/>
</div>
<!-- 表格区域 -->
......@@ -74,7 +81,9 @@
<el-table-column prop="issueTime" label="发放时间" min-width="200">
<template #default="{ row }">
{{ dayjs(row.createTime * 1000).format('YYYY-MM-DD HH:mm:ss') }}
{{
row.issueTime ? dayjs(row.issueTime * 1000).format('YYYY-MM-DD HH:mm:ss') : '未发货'
}}
</template>
</el-table-column>
<el-table-column prop="status" label="发放状态" min-width="200">
......@@ -82,7 +91,7 @@
{{ row.status === 0 ? '待发货' : row.status === 1 ? '已发货' : '已取消' }}
</template>
</el-table-column>
<el-table-column prop="showName" label="发放人备注" min-width="200" />
<el-table-column prop="memo" label="发放人备注" min-width="200" />
<el-table-column label="操作" width="150" fixed="right">
<template #default="{ row }">
<el-button v-if="row.status === 0" type="primary" link @click="handleIssue(row)">
......@@ -140,7 +149,7 @@
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit">
<el-icon class="btn-icon"><Upload /></el-icon>
<el-icon class="btn-icon"><IEpUpload /></el-icon>
保存
</el-button>
</template>
......@@ -149,21 +158,90 @@
</template>
<script setup lang="ts">
import { Search, Upload } from '@element-plus/icons-vue'
import { usePageSearch, useResetData } from '@/hooks'
import { addOrUpdateColumn, getBackendExchangeList, issueProduct } from '@/api/backend'
import {
addOrUpdateColumn,
getBackendExchangeList,
issueProduct,
exportShopItemList,
} from '@/api/backend'
import type { FormInstance, FormRules } from 'element-plus'
import type { BackendColumnListItemDto, AddOrUpdateColumnDto } from '@/api/backend'
import type {
BackendColumnListItemDto,
AddOrUpdateColumnDto,
BackendShopItemDto,
} from '@/api/backend'
import dayjs from 'dayjs'
import { ArticleTypeEnum } from '@/constants'
import ExportExcelBtn from '@/components/common/ExportExcelBtn/index.vue'
import type { ExportColumn } from '@/utils'
const { loading, list, total, reset, goToPage, changePageSize, refresh, searchParams, search } =
usePageSearch(getBackendExchangeList, {
defaultParams: {
type: ArticleTypeEnum.VIDEO,
},
defaultParams: {},
})
const columns: ExportColumn<BackendShopItemDto>[] = [
{
title: '商品名称',
key: 'itemName',
},
{
title: '图片',
key: 'imageUrl',
},
{
title: '来源',
key: 'source',
formatter: (value) => {
return value === 1 ? '商城' : value === 2 ? '大转盘' : '每日抽奖'
},
},
{
title: '兑换数量',
key: 'num',
},
{
title: '兑换时间',
key: 'createTime',
formatter: (value) => {
return dayjs((value as number) * 1000).format('YYYY-MM-DD HH:mm:ss')
},
},
{
title: '兑换人',
key: 'userName',
},
{
title: '发放人',
key: 'issuerName',
},
{
title: '发放备注',
key: 'memo',
},
{
title: '发放时间',
key: 'issueTime',
formatter: (value) => {
return value ? dayjs((value as number) * 1000).format('YYYY-MM-DD HH:mm:ss') : '未发货'
},
},
{
title: '发放状态',
key: 'status',
formatter: (value) => {
return value === 0 ? '待发货' : value === 1 ? '已发货' : '已取消'
},
},
{
title: '发放人备注',
key: 'memo',
},
]
// 对话框
const dialogVisible = ref(false)
const dialogTitle = computed(() => (form.value.id ? '编辑标签' : '新增标签'))
......@@ -261,7 +339,8 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0;
.search-select {
......
......@@ -21,8 +21,8 @@ export default defineComponent(() => {
// { path: '/backend/videoSettings', title: '视频——栏目管理' },
// { path: '/backend/videoManage', title: '视频管理' },
{ path: '/backend/caseManage', title: 'YAYA案例库管理' },
{ path: '/backend/goodsManage', title: '积分商城——商品配置' },
{ path: '/backend/goodsDistribution', title: '积分商城——商品分发' },
// { path: '/backend/goodsManage', title: '积分商城——商品配置' },
{ path: '/backend/goodsDistribution', title: '商品分发' },
// 内容管理
{
path: '/backend/contentsMenu',
......
<template>
<el-dialog v-model="visible" title="竞拍记录" width="50%">
<el-table :data="list" style="width: 100%">
<el-table-column prop="userName" label="竞拍人" />
<el-table-column prop="bidTime" label="竞拍时间">
<template #default="{ row }">
{{ dayjs.unix(row.bidTime).format('YYYY-MM-DD HH:mm:ss') }}
</template>
</el-table-column>
<el-table-column prop="bidPrice" label="竞拍价格">
<template #default="{ row }"> {{ row.bidPrice }} YA币 </template>
</el-table-column>
</el-table>
<div class="flex justify-end mt-3">
<el-pagination
v-model:current-page="searchParams.current"
v-model:page-size="searchParams.size"
:total="total"
@size-change="changePageSize"
@current-change="goToPage"
layout="total, sizes, prev, pager, next, jumper"
/>
</div>
</el-dialog>
</template>
<script setup lang="ts">
import { usePageSearch } from '@/hooks'
import { getAuctionRecord } from '@/api/backend'
import dayjs from 'dayjs'
const visible = ref(false)
const { list, total, goToPage, changePageSize, searchParams, search } = usePageSearch(
getAuctionRecord,
{
immediate: false,
},
)
const open = (id: number) => {
searchParams.value.id = id
search()
visible.value = true
}
defineExpose({
open,
})
</script>
......@@ -2,54 +2,49 @@
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input
v-model="searchParams.name"
placeholder="请输入商品名称"
class="w-200px"
></el-input>
<el-select
v-model="searchParams.itemType"
placeholder="请选择商品类型"
class="search-select"
clearable
>
<el-option label="实物" :value="ShopGoodsTypeEnum.REAL_GOODS" />
<el-option label="虚拟" :value="ShopGoodsTypeEnum.VIRTUAL_GOODS" />
</el-select>
<el-select
v-model="searchParams.region"
placeholder="请选择上架地区"
class="search-select"
clearable
>
<el-option
v-for="item in regionListOptions"
:key="item.value"
:label="item.label"
:value="item.value"
/>
</el-select>
<el-select
v-model="searchParams.enable"
placeholder="是否展示在前台"
class="search-select"
clearable
>
<el-option label="是" :value="BooleanFlag.YES" />
<el-option label="否" :value="BooleanFlag.NO" />
</el-select>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<div class="flex justify-end">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
新增
</el-button>
</div>
<el-input v-model="searchParams.name" placeholder="请输入商品名称" class="w-200px"></el-input>
<el-select
v-model="searchParams.itemType"
placeholder="请选择商品类型"
class="search-select mx-12px"
clearable
>
<el-option label="实物" :value="ShopGoodsTypeEnum.REAL_GOODS" />
<el-option label="虚拟" :value="ShopGoodsTypeEnum.VIRTUAL_GOODS" />
</el-select>
<el-select
v-model="searchParams.region"
placeholder="请选择上架地区"
class="search-select"
clearable
>
<el-option
v-for="item in regionListOptions"
:key="item.value"
:label="item.label"
:value="item.value"
/>
</el-select>
<el-select
v-model="searchParams.enable"
placeholder="是否展示在前台"
class="search-select mx-12px"
clearable
>
<el-option label="是" :value="BooleanFlag.YES" />
<el-option label="否" :value="BooleanFlag.NO" />
</el-select>
<el-button type="primary" @click="refresh">
<el-icon><IEpSearch /></el-icon>
搜索
</el-button>
<el-button @click="reset">重置</el-button>
<el-button type="primary" @click="handleAdd">
<el-icon><IEpPlus /></el-icon>
新增
</el-button>
</div>
<!-- 表格区域 -->
......@@ -182,7 +177,7 @@
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit">
<el-icon class="btn-icon"><Upload /></el-icon>
<el-icon class="btn-icon"><IEpUpload /></el-icon>
保存
</el-button>
</template>
......@@ -191,7 +186,6 @@
</template>
<script setup lang="ts">
import { Search, Plus, Upload } from '@element-plus/icons-vue'
import { usePageSearch, useResetData } from '@/hooks'
import { getShopItemList, addOrUpdateShopItem, deleteShopItem } from '@/api/backend'
import type { FormInstance, FormRules } from 'element-plus'
......@@ -322,7 +316,8 @@ const handleEnableChange = async (row: BackendColumnListItemDto) => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0;
.search-select {
......
......@@ -13,12 +13,14 @@
<el-option label="年度主推关键词" value="year_recommend" />
<el-option label="关联场景" value="related_scenarios" />
</el-select>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button type="primary" @click="refresh" class="ml-12px">
<el-icon><IEpSearch /></el-icon>
搜索
</el-button>
<el-button @click="reset">重置</el-button>
<el-button type="primary" class="add-btn" @click="handleAdd">
<el-icon><Plus /></el-icon>
<el-button type="primary" @click="handleAdd">
<el-icon><IEpPlus /></el-icon>
新增
</el-button>
</div>
......@@ -108,7 +110,9 @@
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit">
<el-icon class="btn-icon"><Upload /></el-icon>
<el-icon class="btn-icon"
><el-icon><IEpUpload /></el-icon
></el-icon>
保存
</el-button>
</template>
......@@ -117,7 +121,6 @@
</template>
<script setup lang="ts">
import { Search, Plus, Upload } from '@element-plus/icons-vue'
import { usePageSearch, useResetData } from '@/hooks'
import { getTagList, addOrUpdateTag, deleteTag } from '@/api/backend'
import type { FormInstance, FormRules } from 'element-plus'
......@@ -228,16 +231,12 @@ const handleSubmit = async () => {
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-wrap: wrap;
flex-shrink: 0;
gap: 12px 0;
.search-select {
width: 200px;
}
.add-btn {
margin-left: auto;
}
}
// 表格区域
......
<template>
<div class="official-tag-page">
<!-- 搜索栏 -->
<div class="search-section">
<div class="flex-1 flex gap-2">
<el-input
v-model="searchParams.title"
placeholder="请输入栏目标题"
class="w-200px"
clearable
></el-input>
<el-select
v-model="searchParams.status"
placeholder="请选择发布状态"
class="search-select"
clearable
>
<el-option label="发布" :value="1" />
<el-option label="隐藏" :value="0" />
</el-select>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button>
</div>
<div class="flex justify-end">
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
新增
</el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
</div>
<!-- 表格区域 -->
<div class="table-section">
<!-- 表格 -->
<div class="table-wrapper">
<el-table
v-loading="loading"
:data="list"
height="100%"
@selection-change="handleSelectionChange"
>
<el-table-column type="selection" width="55"> </el-table-column>
<el-table-column prop="sort" label="栏目顺序" width="180"> </el-table-column>
<el-table-column prop="title" label="栏目名称" min-width="200" />
<el-table-column prop="postCount" label="栏目帖子数量" min-width="200" />
<el-table-column prop="color" label="颜色" width="300">
<template #default="{ row }">
<div class="color-cell">
<div class="color-block" :style="{ backgroundColor: row.color }" />
<span class="color-text">{{ row.color }}</span>
</div>
</template>
</el-table-column>
<el-table-column prop="createUserId" label="创建人" min-width="200" />
<el-table-column prop="createTime" label="创建时间" min-width="200">
<template #default="{ row }">
{{ dayjs(row.createTime * 1000).format('YYYY-MM-DD HH:mm:ss') }}
</template>
</el-table-column>
<el-table-column prop="status" label="状态" min-width="200">
<template #default="{ row }">
<el-switch
:model-value="row.status"
:active-value="1"
:inactive-value="0"
@change="handleStatusChange(row)"
/>
</template>
</el-table-column>
<el-table-column label="操作" width="150" fixed="right">
<template #default="{ row }">
<el-button type="primary" link @click="handleEdit(row)">编辑</el-button>
<el-button type="danger" link @click="handleDelete(row)">删除</el-button>
</template>
</el-table-column>
</el-table>
</div>
<!-- 分页 -->
<div class="pagination-wrapper">
<el-pagination
v-model:current-page="searchParams.current"
v-model:page-size="searchParams.size"
:total="total"
:page-sizes="[10, 20, 30]"
layout="total, sizes, prev, pager, next, jumper"
@size-change="changePageSize"
@current-change="goToPage"
/>
</div>
</div>
<!-- 新增/编辑对话框 -->
<el-dialog
v-model="dialogVisible"
:title="dialogTitle"
width="500px"
:close-on-click-modal="false"
>
<el-form ref="formRef" :model="form" :rules="formRules" label-width="80px">
<el-form-item label="标题" prop="title">
<el-input v-model="form.title" placeholder="请输入标题" />
</el-form-item>
<el-form-item label="颜色" prop="color">
<el-color-picker v-model="form.color" />
<span class="color-value">{{ form.color }}</span>
</el-form-item>
<el-form-item label="排序" prop="sort">
<el-input-number v-model="form.sort" :min="0" :max="100" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit">
<el-icon class="btn-icon"><Upload /></el-icon>
保存
</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup lang="ts">
import { Search, Plus, Upload } from '@element-plus/icons-vue'
import { usePageSearch, useResetData } from '@/hooks'
import { listOfCultureColumn, addOrUpdateColumn, deleteColumn, hideColumn } from '@/api/backend'
import type { FormInstance, FormRules } from 'element-plus'
import type { BackendColumnListItemDto, AddOrUpdateColumnDto } from '@/api/backend'
import dayjs from 'dayjs'
const { loading, list, total, reset, goToPage, changePageSize, refresh, searchParams, search } =
usePageSearch(listOfCultureColumn, {
defaultParams: {
type: 'video',
},
})
// 对话框
const dialogVisible = ref(false)
const dialogTitle = computed(() => (form.value.id ? '编辑标签' : '新增标签'))
const formRef = ref<FormInstance>()
// 表单数据
const [form, resetForm] = useResetData<AddOrUpdateColumnDto>({
title: '',
color: '#000000',
id: undefined,
sort: 0,
type: 'video',
})
// 表单验证规则
const formRules: FormRules = {
title: [{ required: true, message: '请输入标题', trigger: 'blur' }],
color: [{ required: true, message: '请选择颜色', trigger: 'change' }],
sort: [{ required: true, message: '请输入排序', trigger: 'blur' }],
}
// 新增
const handleAdd = () => {
resetForm()
dialogVisible.value = true
}
// 编辑
const handleEdit = (row: BackendColumnListItemDto) => {
resetForm()
form.value = {
title: row.title,
color: row.color,
id: row.id,
sort: row.sort,
type: 'video',
}
dialogVisible.value = true
}
// 状态改变
const handleStatusChange = async (row: BackendColumnListItemDto) => {
await hideColumn([row.id])
refresh()
}
// 删除
const handleDelete = async (row: BackendColumnListItemDto) => {
try {
await ElMessageBox.confirm(`确定要删除标签"${row.title}"吗?`, '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning',
})
await deleteColumn([row.id])
ElMessage.success('删除成功')
refresh()
} catch (error) {
if (error !== 'cancel') {
ElMessage.error('删除失败')
}
}
}
// 提交表单
const handleSubmit = async () => {
if (!formRef.value) return
try {
await formRef.value.validate()
if (form.value.id) {
await addOrUpdateColumn(form.value)
} else {
await addOrUpdateColumn(form.value)
}
ElMessage.success(form.value.id ? '编辑成功' : '新增成功')
dialogVisible.value = false
if (form.value.id) {
search()
} else {
refresh()
}
} catch (error) {
console.error('表单验证失败:', error)
}
}
const selectedRows = ref<BackendColumnListItemDto[]>([])
// 选择
const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
selectedRows.value = selection
}
// 批量发布/隐藏
const handleBatchPublish = async () => {
await hideColumn(selectedRows.value.map((item) => item.id))
refresh()
selectedRows.value = []
ElMessage.success('发布/隐藏成功')
}
// 批量删除
const handleBatchDelete = async () => {
await deleteColumn(selectedRows.value.map((item) => item.id))
refresh()
selectedRows.value = []
ElMessage.success('删除成功')
}
</script>
<style scoped lang="scss">
.official-tag-page {
height: 100%;
display: flex;
flex-direction: column;
gap: 16px;
}
// 搜索区域
.search-section {
background: #fff;
border-radius: 8px;
padding: 20px;
display: flex;
gap: 12px;
flex-shrink: 0;
.search-select {
width: 200px;
}
}
// 表格区域
.table-section {
flex: 1;
background: #fff;
border-radius: 8px;
padding: 20px;
display: flex;
flex-direction: column;
min-height: 0;
}
.table-wrapper {
flex: 1;
min-height: 0;
.color-cell {
display: flex;
align-items: center;
gap: 12px;
.color-block {
width: 100%;
height: 36px;
border-radius: 4px;
border: 1px solid #e5e7eb;
flex: 1;
}
.color-text {
color: #fff;
font-size: 14px;
font-weight: 500;
position: absolute;
left: 50%;
transform: translateX(-50%);
text-shadow: 0 0 3px rgba(0, 0, 0, 0.5);
}
}
}
.pagination-wrapper {
display: flex;
justify-content: flex-end;
padding-top: 16px;
flex-shrink: 0;
}
// 对话框内的颜色显示
.color-value {
margin-left: 12px;
color: #606266;
font-family: monospace;
}
.btn-icon {
margin-right: 4px;
}
</style>
<template>
<div class="flex-1 flex flex-col" v-loading="loading">
<div class="flex-1 p-4 pt-1">
<div class="relative">
<el-tabs v-model="searchParams.type" @tab-change="toggleTab">
<el-tab-pane
v-for="tab in activityTypeListOptions"
:key="tab.value"
:label="tab.label"
:name="tab.value"
/>
</el-tabs>
<div class="absolute right-0 top-2.5 z-1000">
<el-icon
size="15"
class="cursor-pointer hover:rotate-180 transition-all duration-300"
@click="refresh"
><IEpRefresh
/></el-icon>
</div>
</div>
<!-- 加一行提示 -->
<div class="flex justify-end">
<p class="text-gray-500 text-sm mb-1 flex items-center gap-1">
<el-icon><IEpInfoFilled /></el-icon>
页面仅展示竞拍成功的记录
</p>
</div>
<div v-if="!list.length" class="flex flex-col items-center justify-center h-64">
<div class="w-16 h-16 bg-gray-100 rounded-full flex items-center justify-center mb-4">
<el-icon class="text-2xl text-gray-300"><IEpDocument /></el-icon>
</div>
<div class="text-gray-500 text-lg mb-2">暂无内容</div>
</div>
<div v-else class="space-y-4">
<el-table height="500" :data="list" stripe border>
<el-table-column prop="name" label="名称" />
<el-table-column prop="startingPrice" label="起拍价" />
<el-table-column prop="bidPrice" label="支出YA币" />
</el-table>
</div>
</div>
<div
v-if="list.length"
class="flex items-center justify-end px-6 py-4 border-t border-gray-200"
>
<div class="pagination-wrapper bg-white rounded-lg shadow-sm border border-gray-100 p-2">
<el-pagination
v-model:current-page="searchParams.current"
v-model:page-size="searchParams.size"
@size-change="changePageSize"
@current-change="goToPage"
:page-sizes="[10, 20, 30, 40]"
layout="prev, pager, next, jumper, total"
:total="total"
class="custom-pagination"
/>
</div>
</div>
</div>
</template>
<script lang="ts" setup>
import { getSelfAuctionRecord } from '@/api'
import { usePageSearch } from '@/hooks'
import { activityTypeListOptions } from '@/constants/options'
import { ActivityTypeEnum } from '@/constants/enums'
import type { TabPaneName } from 'element-plus'
const toggleTab = (key: TabPaneName) => {
searchParams.value.type = key as ActivityTypeEnum
refresh()
}
const { list, loading, searchParams, total, refresh, goToPage, changePageSize } = usePageSearch(
getSelfAuctionRecord,
{
defaultParams: {
type: activityTypeListOptions[0]!.value,
},
immediate: false,
},
)
onActivated(() => {
searchParams.value.type = activityTypeListOptions[0]!.value
refresh()
})
defineExpose({
refresh: () => {
searchParams.value.type = activityTypeListOptions[0]!.value
refresh()
},
})
</script>
......@@ -365,7 +365,7 @@ const handleSwitchAccount = async () => {
cutEmail: selectedEmail.value,
})
console.log(data)
sessionStorage.clear()
localStorage.clear()
await userStore.getUserInfoByCode({
code: data,
isCodeLogin: 1,
......@@ -386,7 +386,6 @@ const handleSwitchAccount = async () => {
}
const handleBackUser = () => {
sessionStorage.clear()
wxLogin(route.fullPath)
}
......@@ -402,7 +401,7 @@ const handleClearCache = async () => {
userId: userInfo.value.userId,
})
// 清理sessionStorage
sessionStorage.clear()
localStorage.clear()
await userStore.getUserInfoByCode({
code: data,
isCodeLogin: 1,
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment