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(() => { ...@@ -359,14 +359,9 @@ onMounted(() => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
flex-shrink: 0; flex-shrink: 0;
.add-btn {
margin-left: auto;
}
} }
// 表格区域 // 表格区域
.table-section { .table-section {
flex: 1; flex: 1;
......
...@@ -3,56 +3,50 @@ ...@@ -3,56 +3,50 @@
<div class="official-tag-page"> <div class="official-tag-page">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<div class="search-section"> <div class="search-section">
<div class="flex-1 flex gap-2"> <el-input
<el-input v-model="searchParams.title"
v-model="searchParams.title" placeholder="请输入栏目标题"
placeholder="请输入栏目标题" class="w-200px"
class="w-200px" ></el-input>
></el-input>
<!-- 审核状态 -->
<!-- 审核状态 --> <el-select
<el-select v-model="searchParams.isAudit"
v-model="searchParams.isAudit" placeholder="请选择审核状态"
placeholder="请选择审核状态" class="search-select mx-12px"
class="search-select" clearable
clearable >
> <el-option
<el-option v-for="item in auditTypeListOptions"
v-for="item in auditTypeListOptions" :key="item.value"
:key="item.value" :label="item.label"
:label="item.label" :value="item.value"
:value="item.value" />
/> </el-select>
</el-select>
<!-- 发布状态 -->
<!-- 发布状态 --> <el-select
<el-select v-model="searchParams.isUse"
v-model="searchParams.isUse" placeholder="请选择发布状态"
placeholder="请选择发布状态" class="search-select mr-12px"
class="search-select" clearable
clearable >
> <el-option
<el-option v-for="item in usageStatusListOptions"
v-for="item in usageStatusListOptions" :key="item.value"
:key="item.value" :label="item.label"
:label="item.label" :value="item.value"
:value="item.value" />
/> </el-select>
</el-select> <el-button type="primary" @click="refresh">
<div> <el-icon><IEpSearch /></el-icon>
<el-button type="primary" @click="refresh"> 搜索</el-button
<el-icon><IEpSearch /></el-icon> >
搜索</el-button <el-button @click="reset">重置</el-button>
>
<el-button @click="reset">重置</el-button> <ImportExcelBtn :api="importCaseExcel" @success="refresh" @error="handleError" />
</div> <el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
</div> <el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
<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>
</div> </div>
<!-- 表格区域 --> <!-- 表格区域 -->
...@@ -297,7 +291,7 @@ import { ...@@ -297,7 +291,7 @@ import {
AuditStatusEnum, AuditStatusEnum,
UsageStatusEnum, UsageStatusEnum,
} from '@/constants' } from '@/constants'
import UploadExcel from '@/components/common/UploadExcel/index.vue' import ImportExcelBtn from '@/components/common/ImportExcelBtn/index.vue'
import dayjs from 'dayjs' import dayjs from 'dayjs'
import ErrorListDialog from './components/errorListDialog.vue' import ErrorListDialog from './components/errorListDialog.vue'
import EditDialog from './components/editDialog.vue' import EditDialog from './components/editDialog.vue'
...@@ -397,8 +391,9 @@ const handleError = (list: BackendCaseListItemDto[]) => { ...@@ -397,8 +391,9 @@ const handleError = (list: BackendCaseListItemDto[]) => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
flex-shrink: 0; flex-shrink: 0;
gap: 12px 0;
.search-select { .search-select {
width: 200px; 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 @@ ...@@ -3,35 +3,31 @@
<div class="official-tag-page"> <div class="official-tag-page">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<div class="search-section"> <div class="search-section">
<div class="flex-1 flex gap-2"> <el-input
<el-input v-model="searchParams.title"
v-model="searchParams.title" placeholder="请输入栏目标题"
placeholder="请输入栏目标题" class="w-200px"
class="w-200px" clearable
clearable ></el-input>
></el-input>
<el-select
<el-select v-model="searchParams.status"
v-model="searchParams.status" placeholder="请选择发布状态"
placeholder="请选择发布状态" class="search-select mx-12px"
class="search-select" clearable
clearable >
> <el-option label="发布" :value="1" />
<el-option label="发布" :value="1" /> <el-option label="隐藏" :value="0" />
<el-option label="隐藏" :value="0" /> </el-select>
</el-select> <el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button> <el-button @click="reset">重置</el-button>
<el-button @click="reset">重置</el-button>
</div> <el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
<div class="flex justify-end"> 新增
<el-button type="primary" @click="handleAdd"> </el-button>
<el-icon><Plus /></el-icon> <el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
新增 <el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
</div> </div>
<!-- 表格区域 --> <!-- 表格区域 -->
...@@ -271,7 +267,8 @@ const handleBatchDelete = async () => { ...@@ -271,7 +267,8 @@ const handleBatchDelete = async () => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0; flex-shrink: 0;
.search-select { .search-select {
......
...@@ -3,35 +3,37 @@ ...@@ -3,35 +3,37 @@
<div class="official-tag-page"> <div class="official-tag-page">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<div class="search-section"> <div class="search-section">
<div class="flex-1 flex gap-2"> <el-input
<el-input v-model="searchParams.title"
v-model="searchParams.title" placeholder="请输入栏目标题"
placeholder="请输入栏目标题" class="w-200px"
class="w-200px" clearable
clearable ></el-input>
></el-input> <el-select
v-model="searchParams.status"
<el-select placeholder="请选择发布状态"
v-model="searchParams.status" class="search-select mx-12px"
placeholder="请选择发布状态" clearable
class="search-select" >
clearable <el-option label="发布" :value="1" />
> <el-option label="隐藏" :value="0" />
<el-option label="发布" :value="1" /> </el-select>
<el-option label="隐藏" :value="0" /> <el-button type="primary" @click="refresh">
</el-select> <el-icon><IEpSearch /></el-icon>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button> 搜索
<el-button @click="reset">重置</el-button> </el-button>
</div> <el-button type="primary" @click="refresh">
<el-icon><Search /></el-icon>
<div class="flex justify-end"> 搜索
<el-button type="primary" @click="handleAdd"> </el-button>
<el-icon><Plus /></el-icon> <el-button @click="reset">重置</el-button>
新增
</el-button> <el-button type="primary" @click="handleAdd">
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button> <el-icon><IEpPlus /></el-icon>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button> 新增
</div> </el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div> </div>
<!-- 表格区域 --> <!-- 表格区域 -->
...@@ -120,7 +122,7 @@ ...@@ -120,7 +122,7 @@
<template #footer> <template #footer>
<el-button @click="dialogVisible = false">取消</el-button> <el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit"> <el-button type="primary" @click="handleSubmit">
<el-icon class="btn-icon"><Upload /></el-icon> <el-icon class="btn-icon"><IEpUpload /></el-icon>
保存 保存
</el-button> </el-button>
</template> </template>
...@@ -129,7 +131,7 @@ ...@@ -129,7 +131,7 @@
</template> </template>
<script setup lang="ts"> <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 { usePageSearch, useResetData } from '@/hooks'
import { listOfCultureColumn, addOrUpdateColumn, deleteColumn, hideColumn } from '@/api/backend' import { listOfCultureColumn, addOrUpdateColumn, deleteColumn, hideColumn } from '@/api/backend'
import type { FormInstance, FormRules } from 'element-plus' import type { FormInstance, FormRules } from 'element-plus'
...@@ -271,7 +273,8 @@ const handleBatchDelete = async () => { ...@@ -271,7 +273,8 @@ const handleBatchDelete = async () => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0; flex-shrink: 0;
.search-select { .search-select {
......
...@@ -2,35 +2,31 @@ ...@@ -2,35 +2,31 @@
<div class="official-tag-page"> <div class="official-tag-page">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<div class="search-section"> <div class="search-section">
<div class="flex-1 flex gap-2"> <el-input
<el-input v-model="searchParams.title"
v-model="searchParams.title" placeholder="请输入栏目标题"
placeholder="请输入栏目标题" class="w-200px"
class="w-200px" clearable
clearable ></el-input>
></el-input>
<el-select
<el-select v-model="searchParams.status"
v-model="searchParams.status" placeholder="请选择发布状态"
placeholder="请选择发布状态" class="search-select mx-12px"
class="search-select" clearable
clearable >
> <el-option label="发布" :value="1" />
<el-option label="发布" :value="1" /> <el-option label="隐藏" :value="0" />
<el-option label="隐藏" :value="0" /> </el-select>
</el-select> <el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button> <el-button @click="reset">重置</el-button>
<el-button @click="reset">重置</el-button>
</div> <el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
<div class="flex justify-end"> 新增
<el-button type="primary" @click="handleAdd"> </el-button>
<el-icon><Plus /></el-icon> <el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
新增 <el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</el-button>
<el-button type="primary" @click="handleBatchPublish"> 批量发布/隐藏 </el-button>
<el-button type="danger" @click="handleBatchDelete"> 批量删除 </el-button>
</div>
</div> </div>
<!-- 表格区域 --> <!-- 表格区域 -->
...@@ -270,7 +266,8 @@ const handleBatchDelete = async () => { ...@@ -270,7 +266,8 @@ const handleBatchDelete = async () => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0; flex-shrink: 0;
.search-select { .search-select {
......
...@@ -112,13 +112,13 @@ const handleShowContent = (row: ArticleItemDto) => { ...@@ -112,13 +112,13 @@ const handleShowContent = (row: ArticleItemDto) => {
<div class="official-tag-page"> <div class="official-tag-page">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<div class="search-section"> <div class="search-section">
<div class="flex-1 flex gap-2"> <el-input
<el-input v-model="searchParams.title" placeholder="请输入标题" class="w-200px"></el-input> v-model="searchParams.title"
<div> placeholder="请输入标题"
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button> class="w-200px mr-12px"
<el-button @click="reset">重置</el-button> ></el-input>
</div> <el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
</div> <el-button @click="reset">重置</el-button>
</div> </div>
<!-- 表格区域 --> <!-- 表格区域 -->
...@@ -251,7 +251,8 @@ const handleShowContent = (row: ArticleItemDto) => { ...@@ -251,7 +251,8 @@ const handleShowContent = (row: ArticleItemDto) => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0; flex-shrink: 0;
.search-select { .search-select {
......
...@@ -112,13 +112,13 @@ const handleShowContent = (row: ArticleItemDto) => { ...@@ -112,13 +112,13 @@ const handleShowContent = (row: ArticleItemDto) => {
<div class="official-tag-page"> <div class="official-tag-page">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<div class="search-section"> <div class="search-section">
<div class="flex-1 flex gap-2"> <el-input
<el-input v-model="searchParams.title" placeholder="请输入标题" class="w-200px"></el-input> v-model="searchParams.title"
<div> placeholder="请输入标题"
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button> class="w-200px mr-12px"
<el-button @click="reset">重置</el-button> ></el-input>
</div> <el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
</div> <el-button @click="reset">重置</el-button>
</div> </div>
<!-- 表格区域 --> <!-- 表格区域 -->
...@@ -251,7 +251,8 @@ const handleShowContent = (row: ArticleItemDto) => { ...@@ -251,7 +251,8 @@ const handleShowContent = (row: ArticleItemDto) => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0; flex-shrink: 0;
.search-select { .search-select {
......
...@@ -112,13 +112,13 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => { ...@@ -112,13 +112,13 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
<div class="official-tag-page"> <div class="official-tag-page">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<div class="search-section"> <div class="search-section">
<div class="flex-1 flex gap-2"> <el-input
<el-input v-model="searchParams.title" placeholder="请输入标题" class="w-200px"></el-input> v-model="searchParams.title"
<div> placeholder="请输入标题"
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button> class="w-200px mr-12px"
<el-button @click="reset">重置</el-button> ></el-input>
</div> <el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
</div> <el-button @click="reset">重置</el-button>
</div> </div>
<!-- 表格区域 --> <!-- 表格区域 -->
...@@ -259,7 +259,8 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => { ...@@ -259,7 +259,8 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0; flex-shrink: 0;
.search-select { .search-select {
......
...@@ -102,13 +102,13 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => { ...@@ -102,13 +102,13 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
<div class="official-tag-page"> <div class="official-tag-page">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<div class="search-section"> <div class="search-section">
<div class="flex-1 flex gap-2"> <el-input
<el-input v-model="searchParams.title" placeholder="请输入标题" class="w-200px"></el-input> v-model="searchParams.title"
<div> placeholder="请输入标题"
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button> class="w-200px mr-12px"
<el-button @click="reset">重置</el-button> ></el-input>
</div> <el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
</div> <el-button @click="reset">重置</el-button>
</div> </div>
<!-- 表格区域 --> <!-- 表格区域 -->
...@@ -252,7 +252,8 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => { ...@@ -252,7 +252,8 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0; flex-shrink: 0;
.search-select { .search-select {
......
...@@ -2,36 +2,43 @@ ...@@ -2,36 +2,43 @@
<div class="official-tag-page"> <div class="official-tag-page">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<div class="search-section"> <div class="search-section">
<div class="flex-1 flex gap-2"> <el-input
<el-input v-model="searchParams.itemName"
v-model="searchParams.itemName" placeholder="请输入商品名称"
placeholder="请输入商品名称" class="w-200px"
class="w-200px" ></el-input>
></el-input>
<el-select
<el-select v-model="searchParams.status"
v-model="searchParams.status" placeholder="请选择发布状态"
placeholder="请选择发布状态" class="search-select mx-12px"
class="search-select" clearable
clearable >
> <el-option label="待发货" :value="0" />
<el-option label="待发货" :value="0" /> <el-option label="已发货" :value="1" />
<el-option label="已发货" :value="1" /> <el-option label="已取消" :value="2" />
<el-option label="已取消" :value="2" /> </el-select>
</el-select> <el-select
<el-select v-model="searchParams.source"
v-model="searchParams.source" placeholder="请选择来源"
placeholder="请选择来源" class="search-select"
class="search-select" clearable
clearable >
> <el-option label="商城" :value="1" />
<el-option label="商城" :value="1" /> <el-option label="大转盘" :value="2" />
<el-option label="大转盘" :value="2" /> <el-option label="每日抽奖" :value="3" />
<el-option label="每日抽奖" :value="3" /> </el-select>
</el-select> <el-button type="primary" @click="refresh" class="ml-12px">
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button> <el-icon><IEpSearch /></el-icon>
<el-button @click="reset">重置</el-button> 搜索
</div> </el-button>
<el-button @click="reset">重置</el-button>
<ExportExcelBtn
:api="exportShopItemList"
:searchParams="searchParams"
:columns="columns"
fileName="商品分发数据"
/>
</div> </div>
<!-- 表格区域 --> <!-- 表格区域 -->
...@@ -74,7 +81,9 @@ ...@@ -74,7 +81,9 @@
<el-table-column prop="issueTime" label="发放时间" min-width="200"> <el-table-column prop="issueTime" label="发放时间" min-width="200">
<template #default="{ row }"> <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> </template>
</el-table-column> </el-table-column>
<el-table-column prop="status" label="发放状态" min-width="200"> <el-table-column prop="status" label="发放状态" min-width="200">
...@@ -82,7 +91,7 @@ ...@@ -82,7 +91,7 @@
{{ row.status === 0 ? '待发货' : row.status === 1 ? '已发货' : '已取消' }} {{ row.status === 0 ? '待发货' : row.status === 1 ? '已发货' : '已取消' }}
</template> </template>
</el-table-column> </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"> <el-table-column label="操作" width="150" fixed="right">
<template #default="{ row }"> <template #default="{ row }">
<el-button v-if="row.status === 0" type="primary" link @click="handleIssue(row)"> <el-button v-if="row.status === 0" type="primary" link @click="handleIssue(row)">
...@@ -140,7 +149,7 @@ ...@@ -140,7 +149,7 @@
<template #footer> <template #footer>
<el-button @click="dialogVisible = false">取消</el-button> <el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit"> <el-button type="primary" @click="handleSubmit">
<el-icon class="btn-icon"><Upload /></el-icon> <el-icon class="btn-icon"><IEpUpload /></el-icon>
保存 保存
</el-button> </el-button>
</template> </template>
...@@ -149,21 +158,90 @@ ...@@ -149,21 +158,90 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { Search, Upload } from '@element-plus/icons-vue'
import { usePageSearch, useResetData } from '@/hooks' 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 { 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 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 } = const { loading, list, total, reset, goToPage, changePageSize, refresh, searchParams, search } =
usePageSearch(getBackendExchangeList, { usePageSearch(getBackendExchangeList, {
defaultParams: { defaultParams: {},
type: ArticleTypeEnum.VIDEO,
},
}) })
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 dialogVisible = ref(false)
const dialogTitle = computed(() => (form.value.id ? '编辑标签' : '新增标签')) const dialogTitle = computed(() => (form.value.id ? '编辑标签' : '新增标签'))
...@@ -261,7 +339,8 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => { ...@@ -261,7 +339,8 @@ const handleSelectionChange = (selection: BackendColumnListItemDto[]) => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0; flex-shrink: 0;
.search-select { .search-select {
......
...@@ -21,8 +21,8 @@ export default defineComponent(() => { ...@@ -21,8 +21,8 @@ export default defineComponent(() => {
// { path: '/backend/videoSettings', title: '视频——栏目管理' }, // { path: '/backend/videoSettings', title: '视频——栏目管理' },
// { path: '/backend/videoManage', title: '视频管理' }, // { path: '/backend/videoManage', title: '视频管理' },
{ path: '/backend/caseManage', title: 'YAYA案例库管理' }, { path: '/backend/caseManage', title: 'YAYA案例库管理' },
{ path: '/backend/goodsManage', title: '积分商城——商品配置' }, // { path: '/backend/goodsManage', title: '积分商城——商品配置' },
{ path: '/backend/goodsDistribution', title: '积分商城——商品分发' }, { path: '/backend/goodsDistribution', title: '商品分发' },
// 内容管理 // 内容管理
{ {
path: '/backend/contentsMenu', 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 @@ ...@@ -2,54 +2,49 @@
<div class="official-tag-page"> <div class="official-tag-page">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<div class="search-section"> <div class="search-section">
<div class="flex-1 flex gap-2"> <el-input v-model="searchParams.name" placeholder="请输入商品名称" class="w-200px"></el-input>
<el-input
v-model="searchParams.name" <el-select
placeholder="请输入商品名称" v-model="searchParams.itemType"
class="w-200px" placeholder="请选择商品类型"
></el-input> class="search-select mx-12px"
clearable
<el-select >
v-model="searchParams.itemType" <el-option label="实物" :value="ShopGoodsTypeEnum.REAL_GOODS" />
placeholder="请选择商品类型" <el-option label="虚拟" :value="ShopGoodsTypeEnum.VIRTUAL_GOODS" />
class="search-select" </el-select>
clearable <el-select
> v-model="searchParams.region"
<el-option label="实物" :value="ShopGoodsTypeEnum.REAL_GOODS" /> placeholder="请选择上架地区"
<el-option label="虚拟" :value="ShopGoodsTypeEnum.VIRTUAL_GOODS" /> class="search-select"
</el-select> clearable
<el-select >
v-model="searchParams.region" <el-option
placeholder="请选择上架地区" v-for="item in regionListOptions"
class="search-select" :key="item.value"
clearable :label="item.label"
> :value="item.value"
<el-option />
v-for="item in regionListOptions" </el-select>
:key="item.value" <el-select
:label="item.label" v-model="searchParams.enable"
:value="item.value" placeholder="是否展示在前台"
/> class="search-select mx-12px"
</el-select> clearable
<el-select >
v-model="searchParams.enable" <el-option label="是" :value="BooleanFlag.YES" />
placeholder="是否展示在前台" <el-option label="否" :value="BooleanFlag.NO" />
class="search-select" </el-select>
clearable <el-button type="primary" @click="refresh">
> <el-icon><IEpSearch /></el-icon>
<el-option label="是" :value="BooleanFlag.YES" /> 搜索
<el-option label="否" :value="BooleanFlag.NO" /> </el-button>
</el-select> <el-button @click="reset">重置</el-button>
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button>
<el-button @click="reset">重置</el-button> <el-button type="primary" @click="handleAdd">
</div> <el-icon><IEpPlus /></el-icon>
新增
<div class="flex justify-end"> </el-button>
<el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon>
新增
</el-button>
</div>
</div> </div>
<!-- 表格区域 --> <!-- 表格区域 -->
...@@ -182,7 +177,7 @@ ...@@ -182,7 +177,7 @@
<template #footer> <template #footer>
<el-button @click="dialogVisible = false">取消</el-button> <el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit"> <el-button type="primary" @click="handleSubmit">
<el-icon class="btn-icon"><Upload /></el-icon> <el-icon class="btn-icon"><IEpUpload /></el-icon>
保存 保存
</el-button> </el-button>
</template> </template>
...@@ -191,7 +186,6 @@ ...@@ -191,7 +186,6 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { Search, Plus, Upload } from '@element-plus/icons-vue'
import { usePageSearch, useResetData } from '@/hooks' import { usePageSearch, useResetData } from '@/hooks'
import { getShopItemList, addOrUpdateShopItem, deleteShopItem } from '@/api/backend' import { getShopItemList, addOrUpdateShopItem, deleteShopItem } from '@/api/backend'
import type { FormInstance, FormRules } from 'element-plus' import type { FormInstance, FormRules } from 'element-plus'
...@@ -322,7 +316,8 @@ const handleEnableChange = async (row: BackendColumnListItemDto) => { ...@@ -322,7 +316,8 @@ const handleEnableChange = async (row: BackendColumnListItemDto) => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
gap: 12px 0;
flex-shrink: 0; flex-shrink: 0;
.search-select { .search-select {
......
...@@ -13,12 +13,14 @@ ...@@ -13,12 +13,14 @@
<el-option label="年度主推关键词" value="year_recommend" /> <el-option label="年度主推关键词" value="year_recommend" />
<el-option label="关联场景" value="related_scenarios" /> <el-option label="关联场景" value="related_scenarios" />
</el-select> </el-select>
<el-button type="primary" @click="refresh" class="ml-12px">
<el-button type="primary" :icon="Search" @click="refresh">搜索</el-button> <el-icon><IEpSearch /></el-icon>
搜索
</el-button>
<el-button @click="reset">重置</el-button> <el-button @click="reset">重置</el-button>
<el-button type="primary" class="add-btn" @click="handleAdd"> <el-button type="primary" @click="handleAdd">
<el-icon><Plus /></el-icon> <el-icon><IEpPlus /></el-icon>
新增 新增
</el-button> </el-button>
</div> </div>
...@@ -108,7 +110,9 @@ ...@@ -108,7 +110,9 @@
<template #footer> <template #footer>
<el-button @click="dialogVisible = false">取消</el-button> <el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSubmit"> <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> </el-button>
</template> </template>
...@@ -117,7 +121,6 @@ ...@@ -117,7 +121,6 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { Search, Plus, Upload } from '@element-plus/icons-vue'
import { usePageSearch, useResetData } from '@/hooks' import { usePageSearch, useResetData } from '@/hooks'
import { getTagList, addOrUpdateTag, deleteTag } from '@/api/backend' import { getTagList, addOrUpdateTag, deleteTag } from '@/api/backend'
import type { FormInstance, FormRules } from 'element-plus' import type { FormInstance, FormRules } from 'element-plus'
...@@ -228,16 +231,12 @@ const handleSubmit = async () => { ...@@ -228,16 +231,12 @@ const handleSubmit = async () => {
border-radius: 8px; border-radius: 8px;
padding: 20px; padding: 20px;
display: flex; display: flex;
gap: 12px; flex-wrap: wrap;
flex-shrink: 0; flex-shrink: 0;
gap: 12px 0;
.search-select { .search-select {
width: 200px; 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 () => { ...@@ -365,7 +365,7 @@ const handleSwitchAccount = async () => {
cutEmail: selectedEmail.value, cutEmail: selectedEmail.value,
}) })
console.log(data) console.log(data)
sessionStorage.clear() localStorage.clear()
await userStore.getUserInfoByCode({ await userStore.getUserInfoByCode({
code: data, code: data,
isCodeLogin: 1, isCodeLogin: 1,
...@@ -386,7 +386,6 @@ const handleSwitchAccount = async () => { ...@@ -386,7 +386,6 @@ const handleSwitchAccount = async () => {
} }
const handleBackUser = () => { const handleBackUser = () => {
sessionStorage.clear()
wxLogin(route.fullPath) wxLogin(route.fullPath)
} }
...@@ -402,7 +401,7 @@ const handleClearCache = async () => { ...@@ -402,7 +401,7 @@ const handleClearCache = async () => {
userId: userInfo.value.userId, userId: userInfo.value.userId,
}) })
// 清理sessionStorage // 清理sessionStorage
sessionStorage.clear() localStorage.clear()
await userStore.getUserInfoByCode({ await userStore.getUserInfoByCode({
code: data, code: data,
isCodeLogin: 1, 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