index.vue 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. <template>
  2. <z-paging ref="paging" v-model="list" bgColor="#f7f7f7" @query="query" hide-no-more-inside>
  3. <template #top>
  4. <ut-navbar title="码生成及管理" :fixed="false"></ut-navbar>
  5. </template>
  6. <template>
  7. <!-- 搜索栏 -->
  8. <view class="d-flex a-c pd-24 pb-0 bg-#f7f7f7">
  9. <view class="min-w-220 flex1">
  10. <ut-action-sheet v-model="form.downloadFlag"
  11. :tabs="[{ label: '全部', value: '' }, ...pt_code_downflag]" @change="onRefresh" title="选择类型">
  12. <view class="d-flex search-select-item a-c">
  13. <view class="flex1 ov-hd f-s-28 c-333 text-center f-w-5 w-s-no d-flex a-c j-c">
  14. <view class="w-s-no">{{ selectDictLabel(pt_code_downflag, form?.downloadFlag) || '全部' }}
  15. </view>
  16. </view>
  17. <up-icon size="24rpx" color="#333" name="arrow-down-fill" class="mr-5"></up-icon>
  18. </view>
  19. </ut-action-sheet>
  20. </view>
  21. <view class="h-86 pl-20 w-100%">
  22. <ut-search ref="searchRef" v-model="form.keyword" @search="changeSeach" @change="changeSeach"
  23. margin="0" :border="false" placeholder="搜生成批号、操作人" bgColor="#fff" height="86rpx"
  24. borderRadius="10rpx"></ut-search>
  25. </view>
  26. </view>
  27. <!-- 列表 -->
  28. <view class="pd-24">
  29. <view v-for="(item, index) in list" :key="item.id || index">
  30. <CodeItem :item="item" :showButtons="true" @refresh="handleRefresh" @download="handleDownload"
  31. @findAgent="handleFindAgent" @viewOrder="handleViewOrder" @reDownload="handleReDownload"
  32. @itemClick="handleItemClick" />
  33. </view>
  34. </view>
  35. </template>
  36. <template #empty>
  37. <view class="d-flex flex-cln a-c">
  38. <ut-empty class="mg-at" color="#ccc" size="28rpx"
  39. image="https://yujin-szyy.oss-cn-chengdu.aliyuncs.com/szyy/images-plt/plant/noEmpty.png">暂无追溯码数据</ut-empty>
  40. </view>
  41. </template>
  42. <template #bottom>
  43. <ut-tabar activeTab="more"></ut-tabar>
  44. </template>
  45. </z-paging>
  46. <!-- 悬浮按钮 -->
  47. <ut-suspension v-if="sus?.left" :imageWidth="60" :imageHeight="60" :x="sus?.left" :y="sus?.bottom" :inertia="false"
  48. :snap-threshold="40">
  49. <image src="https://yujin-szyy.oss-cn-chengdu.aliyuncs.com/szyy/images-plt/common/btn_add_logo.png" mode="widthFix"
  50. class="w-120 h-120" @click="showGenerateDialog = true"></image>
  51. </ut-suspension>
  52. <!-- 生成追溯码弹框 -->
  53. <GenerateDialog v-model:show="showGenerateDialog" @success="handleGenerateSuccess" />
  54. </template>
  55. <script setup lang="ts">
  56. import { useClientRequest } from '@/utils/request';
  57. import CodeItem from './models/code-item.vue';
  58. import GenerateDialog from './models/generate-dialog.vue';
  59. const { proxy } = getCurrentInstance() as ComponentInternalInstance;
  60. const { pt_code_downflag } = toRefs<any>(proxy?.useDict('pt_code_downflag'));
  61. const paging = ref();
  62. const list = ref<any[]>([]);
  63. // 搜索表单
  64. const form = ref({
  65. downloadFlag: '',
  66. keyword: '',
  67. });
  68. // 悬浮按钮位置
  69. const sus = ref({
  70. left: 0,
  71. bottom: 0,
  72. });
  73. // 弹框显示状态
  74. const showGenerateDialog = ref(false);
  75. // 初始化悬浮按钮位置
  76. const initSuspension = () => {
  77. const systemInfo = uni.getSystemInfoSync();
  78. const windowInfo = uni.getWindowInfo();
  79. sus.value.left = systemInfo.windowWidth - 10;
  80. sus.value.bottom = systemInfo.windowHeight - 200;
  81. };
  82. // 分页查询
  83. const query = async (pageNum: number, pageSize: number) => {
  84. const params = {
  85. pageNum,
  86. pageSize,
  87. ...form.value,
  88. };
  89. try {
  90. const res = await useClientRequest.get('/plt-api/app/traceCodeLog/list', params);
  91. if (res) {
  92. const { rows } = res;
  93. paging.value.complete(rows);
  94. }
  95. } catch (error) {
  96. console.error('查询列表失败:', error);
  97. paging.value.complete([]);
  98. }
  99. };
  100. // 搜索
  101. const changeSeach = () => {
  102. paging.value.reload();
  103. };
  104. // 刷新
  105. const onRefresh = () => {
  106. paging.value.reload();
  107. };
  108. // 处理刷新按钮点击(单条数据刷新)
  109. const handleRefresh = async (id: number) => {
  110. try {
  111. const res = await useClientRequest.get(`/plt-api/app/traceCodeLog/${id}`);
  112. if (res && res.code === 200) {
  113. const newData = res.data;
  114. // 找到对应项并更新
  115. const index = list.value.findIndex((item) => item.id === id);
  116. if (index !== -1) {
  117. list.value[index] = newData;
  118. }
  119. uni.showToast({
  120. title: '刷新成功',
  121. icon: 'success',
  122. });
  123. }
  124. } catch (error) {
  125. console.error('刷新失败:', error);
  126. uni.showToast({
  127. title: '刷新失败',
  128. icon: 'none',
  129. });
  130. }
  131. };
  132. // 处理下载按钮点击
  133. const handleDownload = async (item: any) => {
  134. if (!item.id) {
  135. uni.showToast({
  136. title: '缺少下载参数',
  137. icon: 'none',
  138. });
  139. return;
  140. }
  141. try {
  142. const downloadUrl = `/plt-api/app/traceCodeLog/download/${item.id}`;
  143. await useClientRequest.down(
  144. {
  145. url: downloadUrl,
  146. fileName: `traceCode_${item.id}.xlsx`,
  147. },
  148. '下载中',
  149. );
  150. } catch (error) {
  151. console.error('下载失败:', error);
  152. }
  153. };
  154. // 处理找人代制作按钮点击
  155. const handleFindAgent = (item: any) => {
  156. console.log('找人代制作:', item);
  157. uni.showToast({
  158. title: '代制作功能开发中',
  159. icon: 'none',
  160. });
  161. };
  162. // 处理查看代做订单按钮点击
  163. const handleViewOrder = (item: any) => {
  164. console.log('查看代做订单:', item);
  165. uni.showToast({
  166. title: '订单功能开发中',
  167. icon: 'none',
  168. });
  169. };
  170. // 处理重新下载按钮点击(与下载使用同一接口)
  171. const handleReDownload = (item: any) => {
  172. handleDownload(item);
  173. };
  174. // 处理 item 点击
  175. const handleItemClick = (item: any) => {
  176. console.log('item 点击:', item);
  177. // 可以跳转到详情页
  178. };
  179. // 处理生成成功
  180. const handleGenerateSuccess = () => {
  181. // 重新加载列表
  182. paging.value.reload();
  183. };
  184. onMounted(() => {
  185. initSuspension();
  186. });
  187. </script>
  188. <style scoped lang="scss">
  189. .search-select-item {
  190. height: 86rpx;
  191. background-color: #fff;
  192. border-radius: 10rpx;
  193. box-sizing: border-box;
  194. padding: 12rpx;
  195. }
  196. </style>