index.vue 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631
  1. <template>
  2. <div class="p-2">
  3. <transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave">
  4. <div v-show="showSearch" class="mb-[10px]">
  5. <el-card shadow="hover" :style="{ height: '70px' }">
  6. <el-row :gutter="20" align="middle">
  7. <!-- 节目总数 -->
  8. <el-col :span="8" style="display: flex; justify-content: center; align-items: center">
  9. <el-statistic :value="totalNum">
  10. <template #title>
  11. <div style="display: inline-flex; align-items: center">节目总数</div>
  12. </template>
  13. </el-statistic>
  14. </el-col>
  15. <!-- 轮播总数 -->
  16. <el-col :span="8" style="display: flex; justify-content: center; align-items: center">
  17. <el-statistic :value="lbNum">
  18. <template #title>
  19. <div style="display: inline-flex; align-items: center">轮播总数</div>
  20. </template>
  21. </el-statistic>
  22. </el-col>
  23. <!-- 节目总数 -->
  24. <el-col :span="8" style="display: flex; justify-content: center; align-items: center">
  25. <el-statistic :value="jmNum">
  26. <template #title>
  27. <div style="display: inline-flex; align-items: center">节目总数</div>
  28. </template>
  29. </el-statistic>
  30. </el-col>
  31. </el-row>
  32. </el-card>
  33. <el-card shadow="hover" :style="{ marginTop: '10px', height: '60px' }">
  34. <el-form ref="queryFormRef" :model="queryParams" :inline="true" label-width="40px">
  35. <el-form-item label="名称" prop="itemName">
  36. <el-input v-model="queryParams.itemName" style="width: 150px" placeholder="请输入名称" clearable @keyup.enter="handleQuery" />
  37. </el-form-item>
  38. <el-form-item label="类型" prop="itemType">
  39. <el-select v-model="queryParams.itemType" style="width: 150px" placeholder="请选择类型" clearable>
  40. <el-option v-for="dict in smsb_item_type" :key="dict.value" :label="dict.label" :value="dict.value" />
  41. </el-select>
  42. </el-form-item>
  43. <el-form-item label="时间" style="width: 250px">
  44. <el-date-picker
  45. v-model="dateRangeCreateTime"
  46. value-format="YYYY-MM-DD HH:mm:ss"
  47. type="daterange"
  48. range-separator="-"
  49. start-placeholder="开始日期"
  50. end-placeholder="结束日期"
  51. :default-time="[new Date(2000, 1, 1, 0, 0, 0), new Date(2000, 1, 1, 23, 59, 59)]"
  52. ></el-date-picker>
  53. </el-form-item>
  54. <el-form-item>
  55. <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
  56. <el-button icon="Refresh" @click="resetQuery">重置</el-button>
  57. <el-button type="primary" plain icon="Plus" @click="handleAddL" v-hasPermi="['source:item:add']"> 新增轮播组 </el-button>
  58. <el-button type="primary" plain icon="Plus" @click="handleAddJ" v-hasPermi="['source:item:add']"> 新增分屏组 </el-button>
  59. <el-button type="success" plain icon="Edit" :disabled="single" @click="handleUpdate()" v-hasPermi="['source:item:edit']"
  60. >修改
  61. </el-button>
  62. <el-button type="danger" plain icon="Delete" :disabled="multiple" @click="handleDelete()" v-hasPermi="['source:item:remove']"
  63. >删除
  64. </el-button>
  65. <el-button type="warning" plain icon="Download" @click="handleExport" v-hasPermi="['source:item:export']"> 导出 </el-button>
  66. </el-form-item>
  67. </el-form>
  68. </el-card>
  69. </div>
  70. </transition>
  71. <el-card shadow="never">
  72. <el-table v-loading="loading" :data="itemList" @selection-change="handleSelectionChange">
  73. <el-table-column type="selection" width="55" align="center" />
  74. <el-table-column label="ID" align="left" prop="id" v-if="true" width="180" :show-overflow-tooltip="true" />
  75. <el-table-column label="名称" align="left" prop="itemName" />
  76. <el-table-column label="类型" align="center" prop="itemType" width="100">
  77. <template #default="scope">
  78. <dict-tag :options="smsb_item_type" :value="scope.row.itemType" />
  79. </template>
  80. </el-table-column>
  81. <el-table-column label="分屏" align="center" prop="splitScreen" width="100">
  82. <template #default="scope">
  83. <dict-tag :options="smsb_split_screen" :value="scope.row.splitScreen" />
  84. </template>
  85. </el-table-column>
  86. <el-table-column label="资源数量" align="center" prop="sourceNum" width="100" />
  87. <el-table-column label="创建人" align="left" prop="createUser" width="120" :show-overflow-tooltip="true" />
  88. <el-table-column label="创建时间" align="left" prop="createTime" width="160" />
  89. <el-table-column label="更新时间" align="left" prop="updateTime" width="160" />
  90. <el-table-column label="操作" align="center" class-name="small-padding fixed-width" width="120">
  91. <template #default="scope">
  92. <el-tooltip content="修改" placement="top">
  93. <el-button link type="primary" icon="Edit" @click="handleUpdate(scope.row)" v-hasPermi="['source:item:edit']"></el-button>
  94. </el-tooltip>
  95. <el-tooltip content="删除" placement="top">
  96. <el-button link type="primary" icon="Delete" @click="handleDelete(scope.row)" v-hasPermi="['source:item:remove']"></el-button>
  97. </el-tooltip>
  98. </template>
  99. </el-table-column>
  100. </el-table>
  101. <pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" />
  102. </el-card>
  103. <!-- 添加或修改轮播组对话框 -->
  104. <el-dialog :title="dialog.title" v-model="dialog.visible" width="1200px" append-to-body>
  105. <div class="dialog-container">
  106. <!-- 左侧文件列表 -->
  107. <div class="table-container">
  108. <!-- 界面名称输入框 -->
  109. <el-input v-model="itemName" placeholder="请输入轮播组名称" class="interface-input"></el-input>
  110. <el-table :data="minioDataList" @selection-change="handleSelectionFile">
  111. <el-table-column type="selection" width="55" align="center" />
  112. <el-table-column label="类型" align="center" prop="type" width="80">
  113. <template #default="scope">
  114. <dict-tag :options="smsb_source_type" :value="scope.row.type" />
  115. </template>
  116. </el-table-column>
  117. <el-table-column label="原名" align="left" prop="originalName" width="150" :show-overflow-tooltip="true" />
  118. <el-table-column label="大小" align="center" prop="size" />
  119. <el-table-column label="时长" align="center" prop="duration" />
  120. <el-table-column label="截图" align="center" prop="screenshot">
  121. <template #default="scope">
  122. <image-preview :src="scope.row.screenshot" style="width: 40px; height: 40px; cursor: pointer" />
  123. </template>
  124. </el-table-column>
  125. </el-table>
  126. <pagination
  127. v-show="fileTotal > 0"
  128. :total="fileTotal"
  129. v-model:page="dialogQueryParams.pageNum"
  130. v-model:limit="dialogQueryParams.pageSize"
  131. @pagination="getFileList"
  132. />
  133. </div>
  134. <!-- 右侧选中文件列表 -->
  135. <div class="selected-container">
  136. <el-table ref="selectedTable" :data="selectedFiles" border>
  137. <!-- 显示排序数字 -->
  138. <el-table-column label="排序" width="80">
  139. <template #default="{ row }">
  140. <span class="order-number">{{ row.order }}</span>
  141. </template>
  142. </el-table-column>
  143. <el-table-column label="文件名">
  144. <template #default="{ row }">
  145. <span>{{ row.name }}</span>
  146. </template>
  147. </el-table-column>
  148. <el-table-column label="播放时长">
  149. <template #default="{ row }">
  150. <el-input-number :disabled="row.type !== 1" v-model="row.duration" :min="1" :max="300"></el-input-number>
  151. </template>
  152. </el-table-column>
  153. </el-table>
  154. </div>
  155. </div>
  156. <template #footer>
  157. <div class="dialog-footer">
  158. <el-button :loading="buttonLoading" type="primary" @click="submitForm">确 定</el-button>
  159. <el-button @click="cancel">取 消</el-button>
  160. </div>
  161. </template>
  162. </el-dialog>
  163. <!-- 添加或修改分屏组对话框 -->
  164. <el-dialog :title="splitDialog.title" v-model="splitDialog.visible" width="1200px" append-to-body>
  165. <div class="dialog-container">
  166. <!-- 左侧示意图 -->
  167. <div class="table-container">示意图</div>
  168. <!-- 右侧基础信息 -->
  169. <div class="selected-container">
  170. <el-form ref="itemFormRef" :model="form" :rules="rules" label-width="60px">
  171. <el-form-item label="名称" prop="itemName">
  172. <el-input v-model="form.itemName" placeholder="请输入分屏组名称" />
  173. </el-form-item>
  174. <el-row>
  175. <el-col :span="10">
  176. <el-form-item label="分辨率" prop="width">
  177. <el-input-number
  178. v-model="form.width"
  179. controls-position="right"
  180. placeholder="请输入宽度"
  181. :min="0"
  182. style="width: 180px"
  183. maxlength="6"
  184. />
  185. </el-form-item>
  186. </el-col>
  187. <el-col :span="10">
  188. <el-form-item label="" prop="height" width="10px">
  189. <el-input-number
  190. v-model="form.height"
  191. controls-position="right"
  192. placeholder="请输入高度"
  193. :min="0"
  194. style="width: 180px"
  195. maxlength="6"
  196. />
  197. </el-form-item>
  198. </el-col>
  199. </el-row>
  200. <el-form-item label="分屏" prop="splitScreen">
  201. <el-radio-group v-model="form.splitScreen">
  202. <el-radio v-for="dict in smsb_split_screen" :key="dict.value" :value="parseInt(dict.value)">
  203. {{ dict.label }}
  204. </el-radio>
  205. </el-radio-group>
  206. </el-form-item>
  207. <el-form-item label="坐标" prop="position">
  208. <el-row>
  209. <el-col :span="12">
  210. <el-input-number
  211. v-model="position1"
  212. controls-position="right"
  213. placeholder="请输入坐标1"
  214. :min="0"
  215. style="width: 180px"
  216. maxlength="6"
  217. />
  218. </el-col>
  219. <el-col :span="12">
  220. <el-input-number
  221. v-model="position2"
  222. controls-position="right"
  223. placeholder="请输入坐标2"
  224. :min="0"
  225. style="width: 180px; margin-left: 15px"
  226. maxlength="6"
  227. v-if="form.splitScreen !== 2"
  228. />
  229. </el-col>
  230. </el-row>
  231. </el-form-item>
  232. <el-form-item label="宽高" prop="position" v-if="form.splitScreen === 4">
  233. <el-row>
  234. <el-col :span="12">
  235. <el-input-number
  236. v-model="positionW"
  237. controls-position="right"
  238. placeholder="请输入坐标1"
  239. :min="0"
  240. style="width: 180px"
  241. maxlength="6"
  242. />
  243. </el-col>
  244. <el-col :span="12">
  245. <el-input-number
  246. v-model="positionH"
  247. controls-position="right"
  248. placeholder="请输入坐标2"
  249. :min="0"
  250. style="width: 180px; margin-left: 15px"
  251. maxlength="6"
  252. />
  253. </el-col>
  254. </el-row>
  255. </el-form-item>
  256. <el-row>
  257. <el-col :span="12">
  258. <el-form-item label="跑马灯" prop="hasPmd">
  259. <el-radio-group v-model="form.hasPmd">
  260. <el-radio v-for="dict in smsb_yes_no" :key="dict.value" :value="parseInt(dict.value)">{{ dict.label }} </el-radio>
  261. </el-radio-group>
  262. </el-form-item>
  263. </el-col>
  264. <el-col :span="12">
  265. <el-form-item label="位置" prop="positionPmd">
  266. <el-radio-group v-model="form.positionPmd">
  267. <el-radio-button label="上方" value="1" />
  268. <el-radio-button label="下方" value="2" />
  269. </el-radio-group>
  270. </el-form-item>
  271. </el-col>
  272. </el-row>
  273. <el-form-item label="内容" prop="contentPmd">
  274. <el-input v-model="form.contentPmd" type="textarea" placeholder="请输入跑马灯内容" />
  275. </el-form-item>
  276. <el-row>
  277. <el-col :span="12">
  278. <el-form-item label="天气" prop="hasWeather">
  279. <el-radio-group v-model="form.hasWeather">
  280. <el-radio v-for="dict in smsb_yes_no" :key="dict.value" :value="parseInt(dict.value)">{{ dict.label }} </el-radio>
  281. </el-radio-group>
  282. </el-form-item>
  283. </el-col>
  284. <el-col :span="12">
  285. <el-form-item label="时间" prop="hasTime">
  286. <el-radio-group v-model="form.hasTime">
  287. <el-radio v-for="dict in smsb_yes_no" :key="dict.value" :value="parseInt(dict.value)">{{ dict.label }} </el-radio>
  288. </el-radio-group>
  289. </el-form-item>
  290. </el-col>
  291. </el-row>
  292. </el-form>
  293. </div>
  294. </div>
  295. <template #footer>
  296. <div class="dialog-footer">
  297. <el-button :loading="buttonLoading" type="primary" @click="submitSplit">确 定</el-button>
  298. <el-button @click="cancel">取 消</el-button>
  299. </div>
  300. </template>
  301. </el-dialog>
  302. </div>
  303. </template>
  304. <script setup name="Item" lang="ts">
  305. import { listItem, getItem, delItem, addItem, updateItem, itemStatistics } from '@/api/smsb/source/item';
  306. import { ItemVO, ItemQuery, ItemForm } from '@/api/smsb/source/item_type';
  307. import { MinioDataQuery, MinioDataVO } from '@/api/smsb/source/minioData_type';
  308. import { listMinioData } from '@/api/smsb/source/minioData';
  309. import Sortable from 'sortablejs';
  310. import { nextTick } from 'vue';
  311. const { proxy } = getCurrentInstance() as ComponentInternalInstance;
  312. const { smsb_item_type, smsb_split_screen, smsb_source_type, smsb_yes_no } = toRefs<any>(
  313. proxy?.useDict('smsb_item_type', 'smsb_split_screen', 'smsb_source_type', 'smsb_yes_no')
  314. );
  315. const itemList = ref<ItemVO[]>([]);
  316. const buttonLoading = ref(false);
  317. const loading = ref(true);
  318. const showSearch = ref(true);
  319. const ids = ref<Array<string | number>>([]);
  320. const single = ref(true);
  321. const multiple = ref(true);
  322. const total = ref(0);
  323. const fileTotal = ref(0);
  324. const totalNum = ref(0);
  325. const lbNum = ref(0);
  326. const jmNum = ref(0);
  327. const position1 = ref(0);
  328. const position2 = ref(0);
  329. const positionW = ref(0);
  330. const positionH = ref(0);
  331. const itemName = ref<string>('');
  332. const queryFormRef = ref<ElFormInstance>();
  333. const itemFormRef = ref<ElFormInstance>();
  334. const minioDataList = ref<MinioDataVO[]>([]);
  335. const dateRangeCreateTime = ref<[DateModelType, DateModelType]>(['', '']);
  336. // 选中的文件
  337. const selectedFiles = ref<{ id: number; name: string; duration: number; order: number; type: number }[]>([]);
  338. const dialog = reactive<DialogOption>({
  339. visible: false,
  340. title: ''
  341. });
  342. const splitDialog = reactive<DialogOption>({
  343. visible: false,
  344. title: ''
  345. });
  346. const initFormData: ItemForm = {
  347. itemName: undefined,
  348. itemType: undefined,
  349. splitScreen: 2,
  350. selectedFiles: undefined,
  351. width: undefined,
  352. height: undefined,
  353. background: undefined,
  354. position: undefined,
  355. hasPmd: 0,
  356. positionPmd: '1',
  357. contentPmd: undefined,
  358. hasWeather: 0,
  359. hasTime: 0
  360. };
  361. const data = reactive<PageData<ItemForm, ItemQuery>>({
  362. form: { ...initFormData },
  363. queryParams: {
  364. pageNum: 1,
  365. pageSize: 10,
  366. itemName: undefined,
  367. itemType: undefined,
  368. splitScreen: undefined,
  369. createUser: undefined,
  370. params: {}
  371. },
  372. rules: {}
  373. });
  374. const dialogData = reactive<DialogPageData<MinioDataQuery>>({
  375. dialogQueryParams: {
  376. pageNum: 1,
  377. pageSize: 10,
  378. params: {}
  379. }
  380. });
  381. const { queryParams, form, rules } = toRefs(data);
  382. const { dialogQueryParams } = toRefs(dialogData);
  383. /** 查询节目管理列表 */
  384. const getList = async () => {
  385. loading.value = true;
  386. // const res = await listItem(queryParams.value);
  387. const res = await listItem(proxy?.addDateRange(queryParams.value, dateRangeCreateTime.value, 'CreateTime'));
  388. itemList.value = res.rows;
  389. total.value = res.total;
  390. loading.value = false;
  391. };
  392. /** 查询文件资源列表 */
  393. const getFileList = async () => {
  394. const res = await listMinioData(dialogQueryParams.value);
  395. minioDataList.value = res.rows;
  396. minioDataList.value.forEach((data) => {
  397. data.size = parseFloat(data.size / 1024).toFixed(3) + 'MB';
  398. });
  399. fileTotal.value = res.total;
  400. };
  401. /** 取消按钮 */
  402. const cancel = () => {
  403. reset();
  404. dialog.visible = false;
  405. splitDialog.visible = false;
  406. };
  407. /** 表单重置 */
  408. const reset = () => {
  409. form.value = { ...initFormData };
  410. itemName.value = '';
  411. itemFormRef.value?.resetFields();
  412. };
  413. /** 搜索按钮操作 */
  414. const handleQuery = () => {
  415. queryParams.value.pageNum = 1;
  416. getList();
  417. };
  418. /** 重置按钮操作 */
  419. const resetQuery = () => {
  420. queryFormRef.value?.resetFields();
  421. handleQuery();
  422. };
  423. /** 多选框选中数据 */
  424. const handleSelectionChange = (selection: ItemVO[]) => {
  425. ids.value = selection.map((item) => item.id);
  426. single.value = selection.length != 1;
  427. multiple.value = !selection.length;
  428. };
  429. /** 多选框选中文件数据 */
  430. const handleSelectionFile = (selection: MinioDataVO[]) => {
  431. selectedFiles.value = selection.map((item: any, index: number) => ({
  432. id: item.id,
  433. name: item.originalName,
  434. type: item.type,
  435. duration: item.type === 1 ? 10 : item.duration, // 默认播放时长 10s
  436. order: index + 1 // 默认排序号
  437. }));
  438. };
  439. // 重新编号排序
  440. const updateOrder = () => {
  441. selectedFiles.value.forEach((item, index) => {
  442. item.order = index + 1;
  443. });
  444. };
  445. // 删除选中文件
  446. const removeFile = (file: any) => {
  447. selectedFiles.value = selectedFiles.value.filter((f) => f.id !== file.id);
  448. };
  449. /** 新增轮播按钮操作 */
  450. const handleAddL = () => {
  451. reset();
  452. dialog.visible = true;
  453. dialog.title = '新增轮播组';
  454. form.value.itemType = 1;
  455. getFileList();
  456. };
  457. /** 新增节目按钮操作 */
  458. const handleAddJ = () => {
  459. reset();
  460. splitDialog.visible = true;
  461. splitDialog.title = '新增分屏组';
  462. form.value.itemType = 2;
  463. };
  464. /** 修改按钮操作 */
  465. const handleUpdate = async (row?: ItemVO) => {
  466. reset();
  467. const _id = row?.id || ids.value[0];
  468. const res = await getItem(_id);
  469. if (1 === res.data.itemType) {
  470. Object.assign(form.value, res.data);
  471. dialog.visible = true;
  472. dialog.title = '修改节目管理';
  473. } else {
  474. // 跳转页面进行数据关联
  475. proxy.$router.push('/source/split/edit/' + res.data.id);
  476. }
  477. };
  478. /** 提交按钮 */
  479. const submitForm = async () => {
  480. buttonLoading.value = true;
  481. if (form.value.id) {
  482. await updateItem(form.value).finally(() => (buttonLoading.value = false));
  483. } else {
  484. if (form.value.itemType === 1) {
  485. form.value.selectedFiles = selectedFiles.value;
  486. form.value.itemName = itemName.value;
  487. }
  488. await addItem(form.value).finally(() => (buttonLoading.value = false));
  489. }
  490. proxy?.$modal.msgSuccess('操作成功');
  491. dialog.visible = false;
  492. await getList();
  493. };
  494. const submitSplit = async () => {
  495. // 根据分辨率判断横屏还是竖屏
  496. const isWidth = form.value.width > form.value.height;
  497. // 2分屏的坐标
  498. if (form.value.splitScreen === 2) {
  499. if (isWidth) {
  500. form.value.position = '(0,' + position1.value + '),(' + form.value.height + ',' + position1.value + ')';
  501. } else {
  502. form.value.position = '(0,' + position1.value + '),(' + form.value.width + ',' + position1.value + ')';
  503. }
  504. }
  505. // 3分屏的坐标
  506. if (form.value.splitScreen === 3) {
  507. if (isWidth) {
  508. const xy1 = '(0,' + position1.value + '),(' + form.value.height + ',' + position1.value + ')';
  509. const xy2 = '(0,' + position2.value + '),(' + form.value.height + ',' + position2.value + ')';
  510. form.value.position = xy1 + ',' + xy2;
  511. } else {
  512. const xy1 = '(0,' + position1.value + '),(' + form.value.width + ',' + position1.value + ')';
  513. const xy2 = '(0,' + position2.value + '),(' + form.value.width + ',' + position2.value + ')';
  514. form.value.position = xy1 + ',' + xy2;
  515. }
  516. }
  517. // 播放框
  518. if (form.value.splitScreen === 4) {
  519. const xy1 = '(' + (position1.value - positionW.value / 2) + ',' + (position2.value + positionH.value / 2) + ')';
  520. const xy2 = '(' + (position1.value + positionW.value / 2) + ',' + (position2.value + positionH.value / 2) + ')';
  521. const xy3 = '(' + (position1.value - positionW.value / 2) + ',' + (position2.value - positionH.value / 2) + ')';
  522. const xy4 = '(' + (position1.value + positionW.value / 2) + ',' + (position2.value - positionH.value / 2) + ')';
  523. form.value.position = xy1 + ',' + xy2 + ',' + xy3 + ',' + xy4;
  524. }
  525. const res = await addItem(form.value).finally(() => (buttonLoading.value = false));
  526. proxy?.$modal.msgSuccess('操作成功');
  527. splitDialog.visible = false;
  528. const itemId = res.data;
  529. // 跳转页面进行数据关联
  530. proxy.$router.push('/source/split/edit/' + itemId);
  531. };
  532. /** 删除按钮操作 */
  533. const handleDelete = async (row?: ItemVO) => {
  534. const _ids = row?.id || ids.value;
  535. await proxy?.$modal.confirm('是否确认删除节目管理编号为"' + _ids + '"的数据项?').finally(() => (loading.value = false));
  536. await delItem(_ids);
  537. proxy?.$modal.msgSuccess('删除成功');
  538. await getList();
  539. };
  540. /** 导出按钮操作 */
  541. const handleExport = () => {
  542. proxy?.download(
  543. 'source/item/export',
  544. {
  545. ...queryParams.value
  546. },
  547. `item_${new Date().getTime()}.xlsx`
  548. );
  549. };
  550. const getItemStatistics = async () => {
  551. const res = await itemStatistics();
  552. console.log(res.data);
  553. totalNum.value = res.data.totalNum;
  554. lbNum.value = res.data.lbNum;
  555. jmNum.value = res.data.jmNum;
  556. };
  557. onMounted(() => {
  558. getList();
  559. getItemStatistics();
  560. nextTick(() => {
  561. const tableEl = document.querySelector('.selected-container .el-table__body-wrapper tbody') as HTMLElement;
  562. Sortable.create(tableEl, {
  563. animation: 150,
  564. handle: '.order-number', // 允许拖拽
  565. onEnd(event) {
  566. const { oldIndex, newIndex } = event;
  567. const movedItem = selectedFiles.value.splice(oldIndex!, 1)[0];
  568. selectedFiles.value.splice(newIndex!, 0, movedItem);
  569. updateOrder(); // 拖拽后重新排序
  570. }
  571. });
  572. });
  573. });
  574. </script>
  575. <style scoped>
  576. .dialog-container {
  577. display: flex;
  578. gap: 20px;
  579. }
  580. .table-container {
  581. flex: 1;
  582. }
  583. .selected-container {
  584. flex: 1;
  585. }
  586. .interface-input {
  587. margin-bottom: 10px;
  588. }
  589. .order-number {
  590. cursor: grab;
  591. display: inline-block;
  592. width: 30px;
  593. text-align: center;
  594. font-weight: bold;
  595. }
  596. </style>