index.vue 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. <template>
  2. <div class="icon-page">
  3. <div class="icon-container">
  4. <!-- 标题栏(独立卡片) -->
  5. <div class="header-card">
  6. <div class="table-title">福礼图标广告信息列表</div>
  7. <div class="action-btns">
  8. <el-button type="primary" plain icon="Plus" @click="handleAdd">添加图标广告</el-button>
  9. <el-button icon="Refresh" @click="getList">刷新</el-button>
  10. </div>
  11. </div>
  12. <!-- 表格区域(独立卡片) -->
  13. <div class="table-card">
  14. <!-- 表格 -->
  15. <el-table v-loading="loading" :data="iconList" border>
  16. <el-table-column label="分类标题" align="center" prop="title" min-width="160" />
  17. <el-table-column label="封面图片" align="center" width="140">
  18. <template #default="scope">
  19. <el-image
  20. :src="scope.row.imageUrl"
  21. :preview-src-list="[scope.row.imageUrl]"
  22. fit="cover"
  23. style="width: 64px; height: 64px"
  24. lazy
  25. >
  26. <template #error>
  27. <div class="image-slot">
  28. <el-icon><Picture /></el-icon>
  29. </div>
  30. </template>
  31. </el-image>
  32. </template>
  33. </el-table-column>
  34. <el-table-column label="链接地址" align="center" prop="link" :show-overflow-tooltip="true" min-width="220">
  35. <template #default="scope">
  36. <span class="link-text">{{ scope.row.link || '#' }}</span>
  37. </template>
  38. </el-table-column>
  39. <el-table-column label="状态" align="center" width="100">
  40. <template #default="scope">
  41. <el-tag :type="scope.row.status === 1 ? 'success' : 'info'">
  42. {{ scope.row.status === 1 ? '显示' : '隐藏' }}
  43. </el-tag>
  44. </template>
  45. </el-table-column>
  46. <el-table-column label="排序" align="center" prop="sort" width="80" />
  47. <el-table-column label="操作" align="center" width="150">
  48. <template #default="scope">
  49. <span class="action-link primary" @click="handleUpdate(scope.row)">编辑</span>
  50. <span class="action-link danger" @click="handleDelete(scope.row)">删除</span>
  51. </template>
  52. </el-table-column>
  53. </el-table>
  54. <!-- 分页 -->
  55. <pagination
  56. v-show="total > 0"
  57. v-model:page="queryParams.pageNum"
  58. v-model:limit="queryParams.pageSize"
  59. :total="total"
  60. @pagination="getList"
  61. />
  62. </div>
  63. </div>
  64. <!-- 新增/编辑对话框 -->
  65. <el-dialog v-model="dialog.visible" :title="dialog.title" width="620px" append-to-body @close="cancel">
  66. <el-form ref="iconFormRef" :model="form" :rules="rules" label-width="90px">
  67. <el-row :gutter="20">
  68. <el-col :span="12">
  69. <el-form-item label="公告标题" prop="title">
  70. <el-input v-model="form.title" placeholder="请输入公告标题" />
  71. </el-form-item>
  72. </el-col>
  73. <el-col :span="12">
  74. <el-form-item label="公告链接" prop="link">
  75. <el-input v-model="form.link" placeholder="请输入公告链接" />
  76. </el-form-item>
  77. </el-col>
  78. </el-row>
  79. <el-row :gutter="20">
  80. <el-col :span="12">
  81. <el-form-item label="排序" prop="sort">
  82. <el-input-number v-model="form.sort" :min="0" controls-position="right" style="width: 100%" />
  83. </el-form-item>
  84. </el-col>
  85. <el-col :span="12">
  86. <el-form-item label="状态" prop="status">
  87. <el-switch
  88. v-model="form.status"
  89. :active-value="1"
  90. :inactive-value="0"
  91. active-text="显示"
  92. inactive-text="隐藏"
  93. />
  94. </el-form-item>
  95. </el-col>
  96. </el-row>
  97. <el-form-item label="封面图片" prop="imageUrl">
  98. <image-upload v-model="form.imageUrl" :limit="1" />
  99. </el-form-item>
  100. </el-form>
  101. <template #footer>
  102. <div class="dialog-footer">
  103. <el-button type="primary" @click="submitForm">确认</el-button>
  104. <el-button @click="cancel">取消</el-button>
  105. </div>
  106. </template>
  107. </el-dialog>
  108. </div>
  109. </template>
  110. <script setup name="GiftIcon" lang="ts">
  111. import { getCurrentInstance, onMounted, reactive, ref } from 'vue';
  112. import type { ComponentInternalInstance } from 'vue';
  113. import type { FormInstance } from 'element-plus';
  114. import dayjs from 'dayjs';
  115. import { listAdContent, addAdContent, updateAdContent, delAdContent } from '@/api/ad/content';
  116. const { proxy } = getCurrentInstance() as ComponentInternalInstance;
  117. const loading = ref(false);
  118. const total = ref(0);
  119. const iconList = ref<any[]>([]);
  120. const queryParams = ref({
  121. pageNum: 1,
  122. pageSize: 10,
  123. adType: 'gift_icon'
  124. });
  125. const dialog = reactive<DialogOption>({
  126. visible: false,
  127. title: ''
  128. });
  129. const iconFormRef = ref<FormInstance>();
  130. const initFormData = {
  131. id: undefined as number | undefined,
  132. adType: 'gift_icon',
  133. title: '',
  134. imageUrl: '',
  135. link: '',
  136. sort: 0,
  137. status: 1,
  138. color: '#ffffff',
  139. remark: '',
  140. extJson: '{}',
  141. startTime: '',
  142. endTime: ''
  143. };
  144. const form = ref({ ...initFormData });
  145. const rules = reactive({
  146. title: [{ required: true, message: '公告标题不能为空', trigger: 'blur' }]
  147. });
  148. const getList = async () => {
  149. loading.value = true;
  150. const res = await listAdContent({
  151. pageNum: queryParams.value.pageNum,
  152. pageSize: queryParams.value.pageSize,
  153. adType: queryParams.value.adType
  154. });
  155. iconList.value = res.rows || [];
  156. total.value = res.total || 0;
  157. loading.value = false;
  158. };
  159. const reset = () => {
  160. form.value = { ...initFormData };
  161. iconFormRef.value?.resetFields();
  162. };
  163. const handleAdd = () => {
  164. reset();
  165. const today = dayjs().format('YYYY-MM-DD');
  166. const future = dayjs().add(365, 'day').format('YYYY-MM-DD');
  167. form.value.startTime = today;
  168. form.value.endTime = future;
  169. dialog.visible = true;
  170. dialog.title = '新增图标广告';
  171. };
  172. const handleUpdate = (row: any) => {
  173. reset();
  174. form.value = {
  175. ...row,
  176. adType: 'gift_icon',
  177. // 编辑时用ossId,这样image-upload组件才能正确回显
  178. imageUrl: row.imageOssId || row.imageUrl || '',
  179. status: row.status ?? 1,
  180. color: row.color || '#ffffff',
  181. remark: row.remark || '',
  182. extJson: row.extJson || '{}',
  183. startTime: row.startTime ? dayjs(row.startTime).format('YYYY-MM-DD') : '',
  184. endTime: row.endTime ? dayjs(row.endTime).format('YYYY-MM-DD') : ''
  185. };
  186. dialog.visible = true;
  187. dialog.title = '编辑图标广告';
  188. };
  189. const submitForm = () => {
  190. iconFormRef.value?.validate((valid: boolean) => {
  191. if (valid) {
  192. const api = form.value.id ? updateAdContent : addAdContent;
  193. api(form.value).then(() => {
  194. proxy?.$modal.msgSuccess(form.value.id ? '修改成功' : '新增成功');
  195. dialog.visible = false;
  196. getList();
  197. });
  198. }
  199. });
  200. };
  201. const handleDelete = (row: any) => {
  202. proxy?.$modal.confirm(`是否确认删除分类标题为\"${row.title}\"的数据项?`).then(() => {
  203. delAdContent(row.id).then(() => {
  204. proxy?.$modal.msgSuccess('删除成功');
  205. getList();
  206. });
  207. });
  208. };
  209. const cancel = () => {
  210. reset();
  211. dialog.visible = false;
  212. };
  213. onMounted(() => {
  214. getList();
  215. });
  216. </script>
  217. <style scoped lang="scss">
  218. .icon-page {
  219. min-height: 100vh;
  220. background: #f5f5f5;
  221. padding: 20px;
  222. }
  223. .icon-container {
  224. max-width: 1200px;
  225. margin: 0 auto;
  226. }
  227. .header-card {
  228. background: #fff;
  229. border-radius: 4px;
  230. padding: 15px 20px;
  231. margin-bottom: 12px;
  232. display: flex;
  233. justify-content: space-between;
  234. align-items: center;
  235. .table-title {
  236. font-size: 16px;
  237. font-weight: 600;
  238. color: #303133;
  239. }
  240. .action-btns {
  241. display: flex;
  242. gap: 10px;
  243. }
  244. }
  245. .table-card {
  246. background: #fff;
  247. border-radius: 4px;
  248. padding: 20px;
  249. }
  250. .image-slot {
  251. width: 64px;
  252. height: 64px;
  253. display: flex;
  254. align-items: center;
  255. justify-content: center;
  256. background: #f5f7fa;
  257. color: #c0c4cc;
  258. border-radius: 4px;
  259. }
  260. .link-text {
  261. color: #409eff;
  262. }
  263. .action-link {
  264. cursor: pointer;
  265. margin: 0 8px;
  266. &.primary {
  267. color: #409eff;
  268. &:hover { color: #66b1ff; }
  269. }
  270. &.danger {
  271. color: #f56c6c;
  272. &:hover { color: #f78989; }
  273. }
  274. }
  275. </style>