2 次代碼提交 89bff031ed ... b2aebb2ef4

作者 SHA1 備註 提交日期
  guicheng b2aebb2ef4 组件表格封装完成 1 月之前
  guicheng 26f4fd8b7e 表格组件封装 1 月之前

+ 1 - 1
.vscode/settings.json

@@ -2,7 +2,7 @@
   "editor.formatOnType": true,
   "editor.formatOnSave": true,
   "[vue]": {
-    "editor.defaultFormatter": "Vue.volar"
+    "editor.defaultFormatter": "esbenp.prettier-vscode"
   },
   "editor.tabSize": 2,
   "editor.formatOnPaste": true,

+ 6 - 6
mock/login.ts

@@ -1,10 +1,10 @@
 /*
- * @Author: 
+ * @Author:
  * @Date: 2023-09-27 14:33:29
  * @LastEditors: Please set LastEditors
  * @LastEditTime: 2024-01-03 14:16:53
  * @Description: kxs files
- * @filePath: 
+ * @filePath:
  */
 // 根据角色动态生成路由
 import { MockMethod } from "vite-plugin-mock";
@@ -15,7 +15,7 @@ export default [
     method: "post",
     response: () => {
       return {
-        status: '1',
+        status: "1",
         data: {
           SysAdminId: "1",
           realName: "admin",
@@ -24,7 +24,7 @@ export default [
           accessToken: "eyJhbGciOiJIUzUxMiJ9.admin",
           refreshToken: "eyJhbGciOiJIUzUxMiJ9.adm inRefresh",
           expires: "2023/10/30 00:00:00",
-          rightList: [],
+          rightList: []
         }
       };
     }
@@ -34,7 +34,7 @@ export default [
     method: "post",
     response: () => {
       return {
-        status: '1',
+        status: "1",
         data: {
           SysAdminId: "1",
           realName: "admin",
@@ -43,7 +43,7 @@ export default [
           accessToken: "eyJhbGciOiJIUzUxMiJ9.admin",
           refreshToken: "eyJhbGciOiJIUzUxMiJ9.adm inRefresh",
           expires: "2023/10/30 00:00:00",
-          rightList: [],
+          rightList: []
         }
       };
     }

+ 10 - 78
mock/welcome.ts

@@ -1,96 +1,28 @@
 /*
- * @Author: 
+ * @Author:
  * @Date: 2023-09-27 14:33:29
- * @LastEditors: Please set LastEditors
- * @LastEditTime: 2024-01-03 17:28:27
+ * @LastEditors: guicheng 1625811865@qq.com
+ * @LastEditTime: 2025-02-11 14:56:07
  * @Description: kxs files
- * @filePath: 
+ * @filePath:
  */
 // 根据角色动态生成路由
 import { MockMethod } from "vite-plugin-mock";
 
 export default [
-  {
-    url: "/pie",
-    method: "get",
-    response: () => {
-      return {
-        status: '1',
-        message: '',
-        data: [
-          {
-            name: '数据1',
-            value: 100
-          },
-          {
-            name: '数据2',
-            value: 125
-          },
-          {
-            name: '数据3',
-            value: 635
-          },
-        ]
-      };
-    }
-  },
-  {
-    url: "/line",
-    method: "get",
-    response: () => {
-      return {
-        status: '1',
-        message: '',
-        data: [
-          {
-            name: '数据1',
-            data: [
-              509, 917, 2455, 2610, 2719, 3033, 3044, 3085, 2708, 2809, 2117, 2000, 1455, 1210, 719, 733, 944, 2285, 2208, 3372, 3936, 3693, 2962, 2810, 3519, 2455, 2610, 2719, 2484, 2078]
-          },
-          {
-            name: '数据2',
-            data: [
-              2136, 3693, 2962, 3810, 3519, 3484, 3915, 3823, 3455, 4310, 4019, 3433, 3544, 3885, 4208, 3372, 3484, 3915, 3748, 3675, 4009, 4433, 3544, 3285, 4208, 3372, 3484, 3915, 3823, 4265, 4298
-            ]
-          },
-          {
-            name: '数据3',
-            data: [
-              1361, 2693, 5962, 3810, 3519, 3484, 3915, 3823, 3455, 4310, 4019, 4332, 3544, 3885, 4208, 3372, 3484, 3915, 3748, 3675, 4009, 4433, 3544, 3283, 4208, 3372, 3484, 3915, 3823, 4265, 4298
-            ]
-          }
-        ]
-      };
-    }
-  },
-  {
-    url: "/bar",
-    method: "get",
-    response: () => {
-      return {
-        status: '1',
-        message: '',
-        data: [
-          { name: '数据1', data: [100, 500, 1000], date: ["2021", "2022", "2023"] },
-          { name: '数据2', data: [10, 50, 100], date: ["2021", "2022", "2023"] },
-          { name: '数据3', data: [80, 130, 600], date: ["2021", "2022", "2023"] },
-        ]
-      };
-    }
-  },
   {
     url: "/list",
     method: "get",
     response: () => {
       return {
-        status: '1',
-        message: '',
+        status: "1",
+        message: "",
         data: [
-          { content: '这是第一条', date: '2024/01/01 00:00:15' },
-          { content: '这是第二条', date: '2024/01/02 10:30:57' },
-          { content: '这是第三条', date: '2024/01/03 11:08:29' },
+          { content: "一一一一一一一一", date: "2024/01/01 00:00:15" },
+          { content: "二二二二二二二二", date: "2024/01/02 10:30:57" },
+          { content: "三三三三三三三三", date: "2024/01/03 11:08:29" }
         ]
       };
     }
-  },
+  }
 ] as MockMethod[];

+ 3 - 3
src/api/utils.ts

@@ -2,9 +2,9 @@
  * @Author: Gui
  * @Date: 2023-03-08 11:24:34
  * @LastEditors: guicheng 1625811865@qq.com
- * @LastEditTime: 2025-02-07 16:18:12
+ * @LastEditTime: 2025-02-17 11:39:44
  * @Description: kxs files
  * @filePath:
  */
-// export const environment = "development";
-export const environment = "production";
+export const environment = "development";
+// export const environment = "production";

+ 1 - 14
src/api/welcome.ts

@@ -8,20 +8,7 @@
  */
 import { http } from "@/utils/http";
 import { dataResult } from "@/api/apiResult";
-
-/** 饼图数据 */
-export const welcomePie = (data?: object): dataResult => {
-  return http.request("get", "/pie", { data });
-};
-/** 折线图数据 */
-export const line = (data?: object) => {
-  return http.request("get", "/line ", { data });
-};
-/** 柱状图数据 */
-export const bar = (data?: object) => {
-  return http.request("get", "/bar ", { data });
-};
 /** 列表数据 */
-export const noticelist = (data?: object) => {
+export const noticelist = (data?: object): dataResult => {
   return http.request("get", "/list ", { data });
 };

+ 0 - 23
src/assets/brandList/brandlist.js

@@ -1,23 +0,0 @@
-/*
- * @Author: Gui
- * @Date: 2023-05-11 16:33:30
- * @LastEditors:
- * @LastEditTime: 2023-05-16 11:49:09
- * @Description: kxs files
- * @filePath:
- */
-const brandlist = [
-  "金控电签",
-  "开店宝电签",
-  "金控大POS",
-  "乐刷电签",
-  "乐刷大POS",
-  "立刷云电签",
-  "盛付通电签",
-  "海科电签",
-  "海科大POS",
-  "联动电签",
-  "联动大POS",
-  "盒易付电签"
-];
-export default brandlist;

+ 519 - 0
src/components/OmageTable/index.vue

@@ -0,0 +1,519 @@
+<!--
+ * @Author: Gui
+ * @Date: 2023-03-01 19:20:44
+ * @LastEditors: error: error: git config user.name & please set dead value or install git && error: git config user.email & please set dead value or install git & please set dead value or install git
+ * @LastEditTime: 2025-02-14 15:48:40
+ * @Description: kxs files
+ * @filePath: 
+-->
+<script setup lang="ts">
+import { ref } from "vue";
+interface ColumnsType {
+  hide: Function;
+  slot: string;
+  cellRenderer: Function;
+  formatter: Function;
+  type: string;
+  index: number | Function;
+  label: string;
+  columnKey: string;
+  prop: string;
+  width: string | number;
+  minWidth: string | number;
+  fixed: string | boolean;
+  headerRenderer: Function;
+  sortable: boolean | string;
+  sortMethod: Function;
+  sortBy: Function | string | object;
+  sortOrders: object;
+  resizable: boolean;
+  formatterTable: Function;
+  showOverflowTooltip: boolean | object;
+  align: string;
+  headerAlign: string;
+  className: string;
+  labelClassName: string;
+  selectable: Function;
+  reserveSelection: boolean;
+  filters: Array<{ text: string; value: string }>;
+  filterPlacement: string;
+  filterClassName: string;
+  filterMultiple: boolean;
+  filterMethod: Function;
+  filteredValue: object;
+  tooltipFormatter: Function;
+}
+const props = defineProps({
+  data: {
+    type: Array,
+    default: () => []
+  },
+  height: {
+    type: String || Number,
+    default: null
+  },
+  maxHeight: {
+    type: String || Number,
+    default: null
+  },
+  stripe: {
+    type: Boolean,
+    default: true
+  },
+  border: {
+    type: Boolean,
+    default: true
+  },
+  size: {
+    type: String,
+    default: "default"
+  },
+  fit: {
+    type: Boolean,
+    default: true
+  },
+  showHeader: {
+    type: Boolean,
+    default: true
+  },
+  highlightCurrentRow: {
+    type: Boolean,
+    default: false
+  },
+  currentRowKey: {
+    type: String || Number,
+    default: null
+  },
+  rowClassName: {
+    type: Function || String,
+    default: null
+  },
+  rowStyle: {
+    type: Function || Object,
+    default: null
+  },
+  cellClassName: {
+    type: Function || String,
+    default: null
+  },
+  cellStyle: {
+    type: Function || Object,
+    default: null
+  },
+  headerRowClassName: {
+    type: Function || String,
+    default: null
+  },
+  headerRowStyle: {
+    type: Function || Object,
+    default: null
+  },
+  headerCellClassName: {
+    type: Function || String,
+    default: null
+  },
+  headerCellStyle: {
+    type: Object,
+    default: () => {
+      return {
+        background: "var(--el-table-row-hover-bg-color)",
+        color: "var(--el-text-color-primary)"
+      };
+    }
+  },
+  rowKey: {
+    type: String,
+    default: ""
+  },
+  emptyText: {
+    type: String,
+    default: ""
+  },
+  defaultExpandAll: {
+    type: Boolean,
+    default: false
+  },
+  expandRowKeys: {
+    type: Boolean,
+    default: null
+  },
+  defaultSort: {
+    type: Object,
+    default: () => {
+      return { prop: "date", order: "descending" };
+    }
+  },
+  tooltipEffect: {
+    type: String,
+    default: "dark"
+  },
+  tooltipOptions: {
+    type: Object,
+    default: () => {
+      return {
+        enterable: true,
+        placement: "top",
+        showArrow: true,
+        hideAfter: 200,
+        popperOptions: { strategy: "fixed" }
+      };
+    }
+  },
+  appendFilterPanelTo: {
+    type: String,
+    default: null
+  },
+  showSummary: {
+    type: Boolean,
+    default: false
+  },
+  sumText: {
+    type: String,
+    default: "Sum"
+  },
+  summaryMethod: {
+    type: Function,
+    default: null
+  },
+  spanMethod: {
+    type: Function,
+    default: null
+  },
+  selectOnIndeterminate: {
+    type: Boolean,
+    default: true
+  },
+  indent: {
+    type: Number,
+    default: 16
+  },
+  lazy: {
+    type: Boolean,
+    default: false
+  },
+  load: {
+    type: Function,
+    default: null
+  },
+  treeProps: {
+    type: Object,
+    default: () => {
+      return {
+        hasChildren: "hasChildren",
+        children: "children",
+        checkStrictly: false
+      };
+    }
+  },
+  tableLayout: {
+    type: String,
+    default: "fixed"
+  },
+  scrollbarAlwaysOn: {
+    type: Boolean,
+    default: false
+  },
+  showOverflowTooltip: {
+    type: Object,
+    default: null
+  },
+  flexible: {
+    type: Boolean,
+    default: false
+  },
+  scrollbarTabindex: {
+    type: String || Number,
+    default: null
+  },
+  allowDragLastColumn: {
+    type: Boolean,
+    default: false
+  },
+  tooltipFormatter: {
+    type: Function,
+    default: null
+  },
+  alignWhole: {
+    type: String,
+    default: "center"
+  },
+  loading: {
+    type: Boolean,
+    default: false
+  },
+  columns: {
+    type: Array<ColumnsType>,
+    default: () => []
+  },
+  checkList: {
+    type: Array,
+    default: () => []
+  },
+  disabled: {
+    type: Boolean,
+    default: false
+  },
+  pagination: {
+    type: Object,
+    default: () => {
+      return {
+        // 总数
+        total: 0,
+        // 条数
+        pageSize: 30,
+        // 页码
+        currentPage: 1,
+        background: true,
+        pageSizes: [10, 30, 50, 100]
+      };
+    }
+  },
+  paginationSmall: {
+    type: Boolean,
+    default: true
+  }
+});
+const emit = defineEmits([
+  "select",
+  "select-all",
+  "selection-change",
+  "cell-mouse-enter",
+  "cell-mouse-leave",
+  "cell-click",
+  "cell-dblclick",
+  "cell-contextmenu",
+  "row-click",
+  "row-contextmenu",
+  "row-dblclick",
+  "header-click",
+  "header-contextmenu",
+  "sort-change",
+  "filter-change",
+  "table-current-change",
+  "header-dragend",
+  "expand-change",
+  "scroll",
+  "size-change",
+  "current-change"
+]);
+const handleSizeChange = val => {
+  emit("size-change", val);
+};
+const handleCurrentChange = val => {
+  emit("current-change", val);
+};
+const handleSelectAll = val => {
+  emit("select-all", val);
+};
+const handleSelect = val => {
+  emit("select", val);
+};
+const handleSelectionChange = val => {
+  emit("selection-change", val);
+};
+const handleCellMouseEnter = val => {
+  emit("cell-mouse-enter", val);
+};
+const handleCellMouseLeave = val => {
+  emit("cell-mouse-leave", val);
+};
+const handleCellClick = val => {
+  emit("cell-click", val);
+};
+const handleCellDBLClick = val => {
+  emit("cell-dblclick", val);
+};
+const handleCellContextmenu = val => {
+  emit("cell-contextmenu", val);
+};
+const handleRowClick = val => {
+  emit("row-click", val);
+};
+const handleRowContextmenu = val => {
+  emit("row-contextmenu", val);
+};
+const handleRowDBLclick = val => {
+  emit("row-dblclick", val);
+};
+const handleHeaderClick = val => {
+  emit("header-click", val);
+};
+const handleHeaderContextmenu = val => {
+  emit("header-contextmenu", val);
+};
+const handleSortChange = val => {
+  emit("sort-change", val);
+};
+const handleFilterChange = val => {
+  emit("filter-change", val);
+};
+const handleTableCurrentChange = val => {
+  emit("table-current-change", val);
+};
+const handleHeaderDragend = val => {
+  emit("header-dragend", val);
+};
+const handleExpandChange = val => {
+  emit("expand-change", val);
+};
+const handleScroll = val => {
+  emit("scroll", val);
+};
+props.columns.forEach(element => {
+  element.formatterTable = element.cellRenderer
+    ? (row, column, index) => {
+        return element.cellRenderer({
+          row,
+          column,
+          index,
+          props: { size: props.size }
+        });
+      }
+    : element.formatter
+    ? row => {
+        return element.formatter(row);
+      }
+    : null;
+});
+const tableRef = ref();
+
+defineExpose(
+  new Proxy(
+    {},
+    {
+      get(_target, prop) {
+        return tableRef.value?.[prop];
+      },
+      has(_target, prop) {
+        return prop in tableRef.value;
+      }
+    }
+  )
+);
+</script>
+<script></script>
+<template>
+  <el-table
+    :data="data"
+    :height="height"
+    :max-height="maxHeight"
+    :stripe="stripe"
+    :border="border"
+    :size="size"
+    :fit="fit"
+    :show-header="showHeader"
+    :highlight-current-row="highlightCurrentRow"
+    :current-row-key="currentRowKey"
+    :row-class-name="rowClassName"
+    :row-style="rowStyle"
+    :cell-class-name="cellClassName"
+    :cell-style="cellStyle"
+    :header-row-class-name="headerRowClassName"
+    :header-row-style="headerRowStyle"
+    :header-cell-class-name="headerCellClassName"
+    :header-cell-style="headerCellStyle"
+    :row-key="rowKey"
+    :empty-text="emptyText"
+    :default-expand-all="defaultExpandAll"
+    :expand-row-keys="expandRowKeys"
+    :default-sort="defaultSort"
+    :tooltip-effect="tooltipEffect"
+    :tooltip-options="tooltipOptions"
+    :append-filter-panel-to="appendFilterPanelTo"
+    :show-summary="showSummary"
+    :sum-text="sumText"
+    :summary-method="summaryMethod"
+    :span-method="spanMethod"
+    :select-on-indeterminate="selectOnIndeterminate"
+    :indent="indent"
+    :lazy="lazy"
+    :load="load"
+    :tree-props="treeProps"
+    :table-layout="tableLayout"
+    :scrollbar-always-on="scrollbarAlwaysOn"
+    :show-overflow-tooltip="showOverflowTooltip"
+    :flexible="flexible"
+    :scrollbar-tabindex="scrollbarTabindex"
+    :allow-drag-last-column="allowDragLastColumn"
+    :tooltip-formatter="tooltipFormatter"
+    v-loading="loading"
+    style="width: 100%"
+    @select="handleSelect"
+    @select-all="handleSelectAll"
+    @selection-change="handleSelectionChange"
+    @cell-mouse-enter="handleCellMouseEnter"
+    @cell-mouse-leave="handleCellMouseLeave"
+    @cell-click="handleCellClick"
+    @cell-dblclick="handleCellDBLClick"
+    @cell-contextmenu="handleCellContextmenu"
+    @row-click="handleRowClick"
+    @row-contextmenu="handleRowContextmenu"
+    @row-dblclick="handleRowDBLclick"
+    @header-click="handleHeaderClick"
+    @header-contextmenu="handleHeaderContextmenu"
+    @sort-change="handleSortChange"
+    @filter-change="handleFilterChange"
+    @current-change="handleTableCurrentChange"
+    @header-dragend="handleHeaderDragend"
+    @expand-change="handleExpandChange"
+    @scroll="handleScroll"
+    ref="tableRef"
+  >
+    <template #append> <slot name="append" /> </template>
+    <template #empty> <slot name="empty" /> </template>
+
+    <template v-for="(item, index) in columns" :key="index">
+      <el-table-column
+        v-if="!item.hide || !item.hide({ checkList })"
+        :type="item.type"
+        :index="item.index"
+        :label="item.label"
+        :column-key="item.columnKey"
+        :prop="item.prop"
+        :width="item.width"
+        :min-width="item.minWidth"
+        :fixed="item.fixed"
+        :render-header="item.headerRenderer"
+        :sortable="item.sortable"
+        :sort-method="item.sortMethod"
+        :sort-by="item.sortBy"
+        :sort-orders="item.sortOrders"
+        :sort-resizable="item.resizable"
+        :formatter="item.formatterTable"
+        :show-overflow-tooltip="item.showOverflowTooltip"
+        :align="item.align || alignWhole"
+        :header-align="item.headerAlign"
+        :class-name="item.className"
+        :label-class-name="item.labelClassName"
+        :selectable="item.selectable"
+        :reserve-selection="item.reserveSelection"
+        :filters="item.filters"
+        :filter-placement="item.filterPlacement"
+        :filter-class-name="item.filterClassName"
+        :filter-multiple="item.filterMultiple"
+        :filter-method="item.filterMethod"
+        :filtered-value="item.filteredValue"
+        :tooltip-formatter="item.tooltipFormatter"
+      >
+        <template v-slot="{ row }">
+          <slot name="operation" :row="row" v-if="item.slot == 'operation'" />
+        </template>
+      </el-table-column>
+    </template>
+  </el-table>
+  <div class="flex justify-end mb-4 mt-4">
+    <el-pagination
+      :current-page="pagination.currentPage"
+      :page-sizes="pagination.pageSizes"
+      :page-size="pagination.pageSize"
+      :size="size"
+      :disabled="disabled"
+      layout="total, sizes, prev, pager, next, jumper"
+      :total="pagination.total"
+      background
+      @size-change="handleSizeChange"
+      @current-change="handleCurrentChange"
+    />
+  </div>
+</template>
+<style></style>

+ 6 - 4
src/main.ts

@@ -1,8 +1,8 @@
 /*
  * @Author: Gui
  * @Date: 2023-03-01 19:20:44
- * @LastEditors:
- * @LastEditTime: 2023-09-04 14:42:43
+ * @LastEditors: guicheng 1625811865@qq.com
+ * @LastEditTime: 2025-02-12 16:45:30
  * @Description: kxs files
  * @filePath:
  */
@@ -17,7 +17,8 @@ import { MotionPlugin } from "@vueuse/motion";
 import { useEcharts } from "@/plugins/echarts";
 import { injectResponsiveStorage } from "@/utils/responsive";
 
-import Table from "@pureadmin/table";
+// import Table from "@pureadmin/table";
+import Table from "@/components/OmageTable/index.vue";
 import PureDescriptions from "@pureadmin/descriptions";
 import * as ElementPlusIconsVue from "@element-plus/icons-vue";
 
@@ -54,6 +55,7 @@ import {
 app.component("IconifyIconOffline", IconifyIconOffline);
 app.component("IconifyIconOnline", IconifyIconOnline);
 app.component("FontIcon", FontIcon);
+app.component("PureTable", Table);
 
 // 全局注册按钮级别权限组件
 import { Auth } from "@/components/ReAuth";
@@ -68,7 +70,7 @@ getServerConfig(app).then(async config => {
     .use(MotionPlugin)
     .use(useI18n)
     .use(ElementPlus)
-    .use(Table)
+    // .use(Table)
     .use(PureDescriptions)
     .use(useEcharts);
   app.mount("#app");

+ 0 - 38
src/utils/BrandList.ts

@@ -1,38 +0,0 @@
-/*
- * @Author: Gui
- * @Date: 2023-04-07 16:09:33
- * @LastEditors: Please set LastEditors
- * @LastEditTime: 2023-06-01 17:57:43
- * @Description: kxs files
- * @filePath:
- */
-import { BrandList } from "@/api/brandlist";
-import { onMounted, ref } from "vue";
-type Data = {
-  status?: string;
-  info?: string;
-  other?: any;
-  data?: Array<objectData>;
-};
-type objectData = {
-  Name?: string;
-  Id?: Number;
-};
-
-export function BrandListHook() {
-  const Brand: any = ref([]);
-  function getBrandList() {
-    // 开始加载
-    BrandList({}).then((data?: Data) => {
-      Brand.value = data.data.map(data => {
-        return { Name: data.Name, Id: data.Id };
-      });
-    });
-  }
-  onMounted(() => {
-    getBrandList();
-  });
-  return {
-    Brand
-  };
-}

+ 1 - 1
src/views/admin/menu/index.vue

@@ -126,7 +126,7 @@ const rules = reactive({
         @selection-change="handleSelectionChange",
         @size-change="handleSizeChange",
         @current-change="handleCurrentChange"
-      )          
+      )
         template(#operation="{ row }")
           el-button.reset-margin(
             link

+ 209 - 0
src/views/app/appTracking/hook.tsx

@@ -0,0 +1,209 @@
+import {
+  reactive,
+  onMounted,
+  ref,
+  ElMessage,
+  ElMessageBox,
+  http,
+  getGroupUrl,
+  RegularVerification,
+  verification,
+  PaginationProps
+} from "@/utils/importUsed";
+// 表单实例
+const ruleFormRef = ref();
+export function useAppBottomNavs() {
+  // 接口列表实例
+  let UrlList = reactive(null);
+  // 获取当前板块接口列表
+  onMounted(async () => {
+    UrlList = await getGroupUrl(["kxsConfigServer"]);
+    onSearch(ruleFormRef.value);
+  });
+  const form = reactive({
+    // kind: "creater", // 分类
+    appVersion: "", // 上报版本号
+    userTag: "", // 用户标识
+    pageTitle: "", // 页面标题
+    pageUrl: "", // 页面地址
+    deviceSize: "", // 设备尺寸
+    reportType: "" // 上报类型
+  });
+  const dataList = ref([]);
+  const loading = ref(false);
+  const dialogAddVisible = ref(false);
+  const pagination = reactive<PaginationProps>({
+    total: 0,
+    pageSize: 30,
+    currentPage: 1,
+    background: true,
+    pageSizes: [10, 30, 50, 100]
+  });
+  const columns: TableColumnList = [
+    {
+      type: "selection",
+      width: 55,
+      align: "left",
+      hide: ({ checkList }) => !checkList.includes("勾选列")
+    },
+    {
+      label: "序号",
+      type: "index",
+      width: 70,
+      hide: ({ checkList }) => !checkList.includes("序号列")
+    },
+    {
+      label: "页面标题",
+      prop: "pageTitle",
+      minWidth: 200
+    },
+    {
+      label: "页面地址",
+      prop: "pageUrl",
+      minWidth: 200
+    },
+    {
+      label: "用户系统信息",
+      prop: "userSysInfo",
+      minWidth: 200
+    },
+    {
+      label: "设备尺寸",
+      prop: "deviceSize",
+      minWidth: 200
+    },
+    {
+      label: "上报类型",
+      prop: "reportType",
+      minWidth: 200
+    },
+    {
+      label: "上报数据",
+      prop: "reportData",
+      minWidth: 200
+    },
+    {
+      label: "用户标识",
+      prop: "userTag",
+      minWidth: 200
+    },
+    {
+      label: "创建时间",
+      prop: "createTime",
+      minWidth: 200
+    },
+    {
+      label: "ID",
+      prop: "id",
+      minWidth: 200
+    }
+
+    // {
+    //   label: "操作",
+    //   fixed: "right",
+    //   width: 200,
+    //   slot: "operation"
+    // }
+  ];
+  // 当前页数量切换
+  function handleSizeChange(val: number) {
+    if (typeof val === "number") {
+      pagination.pageSize = val;
+      onSearch(ruleFormRef.value);
+    }
+  }
+  // 当前页码切换
+  function handleCurrentChange(val: number) {
+    console.log(`current page: ${val}`);
+    if (typeof val === "number") {
+      pagination.currentPage = val;
+      onSearch(ruleFormRef.value);
+    }
+  }
+  // 选择表格项
+  function handleSelectionChange(val) {
+    console.log(`SelectionChange: ${val}`);
+    onSearch(ruleFormRef.value);
+  }
+  // 搜索列表
+  async function onSearch(formEl) {
+    // 表单校验拦截
+    if (!formEl) return;
+    await formEl.validate(async (valid, fields) => {
+      if (valid) {
+        //表单校验成功回调
+        console.log("submit!");
+        // 状态调整为加载中
+        loading.value = true;
+        // 调用接口(需动态生成接口)
+        const { status, msg, data }: any = await http.Request({
+          method: "get",
+          url: "https://test-app-manager.kexiaoshuang.com/v1/api/AppReportRecord/getAppReportRecordList",
+          // method: UrlList.kxsConfigServer.appBottomNavslist.method,
+          // url: UrlList.kxsConfigServer.appBottomNavslist.url,
+          params: {
+            ...form,
+            pageSize: pagination.pageSize,
+            pageNum: pagination.currentPage
+          }
+        });
+        dataList.value = data.records;
+        pagination.total = data.total;
+        setTimeout(() => {
+          loading.value = false;
+        }, 500);
+      } else {
+        //表单校验失败回调
+        ElMessage({
+          message: "请输入完整信息",
+          type: "error"
+        });
+      }
+    });
+  }
+
+  // 删除
+  function handleDelete(row) {
+    ElMessageBox.confirm(`是否删除该app底部导航? `, "提示", {
+      confirmButtonText: "删除",
+      cancelButtonText: "取消",
+      type: "warning"
+    }).then(async () => {
+      const { status, msg }: any = await http.Request({
+        method: UrlList.kxsConfigServer.appBottomNavsdelete.method,
+        url: UrlList.kxsConfigServer.appBottomNavsdelete.url,
+        params: String(row.id)
+      });
+      if (status === 1) {
+        ElMessage({
+          message: "删除成功",
+          type: "success"
+        });
+        onSearch(ruleFormRef.value);
+      } else {
+        ElMessageBox.alert(msg, "提示", {
+          confirmButtonText: "关闭",
+          type: "warning"
+        });
+      }
+    });
+  }
+
+  // 分类选项数据
+  const kindOptionList = [{ label: "创客版", id: "creater" }];
+
+  return {
+    form,
+    loading,
+    columns,
+    dataList,
+    pagination,
+    onSearch,
+    handleSizeChange,
+    handleCurrentChange,
+    handleSelectionChange,
+    kindOptionList,
+    ruleFormRef,
+    handleDelete
+  };
+}

+ 155 - 0
src/views/app/appTracking/index.vue

@@ -0,0 +1,155 @@
+<script lang="ts">
+// 声明额外的选项
+export default {
+  name: "AppBottomNavs"
+};
+</script>
+<script setup lang="ts">
+import { provide } from "vue";
+import { useAppBottomNavs } from "./hook";
+import { http } from "@/utils/http";
+import { useRenderIcon, hasAuth, PureTableBar } from "@/utils/importUsed";
+import Search from "@iconify-icons/ep/search";
+import Addicon from "@iconify-icons/ep/document-add";
+import Delete from "@iconify-icons/ep/delete";
+import EditPen from "@iconify-icons/ep/edit-pen";
+import More from "@iconify-icons/ep/more-filled";
+
+const {
+  form,
+  loading,
+  columns,
+  dataList,
+  pagination,
+  onSearch,
+  handleSizeChange,
+  handleCurrentChange,
+  handleSelectionChange,
+  ruleFormRef,
+  handleAdd,
+  handleUpdate,
+  kindOptionList,
+  handleDelete
+} = useAppBottomNavs();
+</script>
+
+<template lang="pug">
+.main
+  div
+    //- 搜索表格组件条件
+    el-form.bg-bg_color.pl-8.pt-4.pr-8(
+      label-position="left"
+      label-width="100px"
+      :inline="true",
+      :model="form",
+      class="w-[99/100]"
+      ref="ruleFormRef"
+      :rules="rules"
+    )
+      el-form-item(label="分类", prop="kind")
+        el-select(
+          v-model="form.kind",
+          placeholder="请选择分类",
+          clearable,
+        )
+          el-option(:label="item.label", :value="item.id" v-for="(item,index) in kindOptionList")
+      el-form-item(label='版本号' prop="appVersion")
+        el-input(v-model='form.appVersion' autocomplete='off'
+          placeholder="请输入版本号")
+      el-form-item(label='用户标识' prop="userTag")
+        el-input(v-model='form.userTag' autocomplete='off'
+          placeholder="请输入用户标识")
+      el-form-item(label='页面标题' prop="pageTitle")
+        el-input(v-model='form.pageTitle' autocomplete='off'
+          placeholder="请输入页面标题")
+      el-form-item(label='页面地址' prop="pageUrl")
+        el-input(v-model='form.pageUrl' autocomplete='off'
+          placeholder="请输入页面地址")
+      el-form-item(label='设备尺寸' prop="deviceSize")
+        el-input(v-model='form.deviceSize' autocomplete='off'
+          placeholder="请输入设备尺寸")
+      el-form-item(label='上报类型' prop="reportType")
+        el-input(v-model='form.reportType' autocomplete='off'
+          placeholder="请输入上报类型")
+
+      el-form-item
+        el-button(
+          type="primary",
+          :icon="useRenderIcon(Search)",
+          :loading="loading",
+          @click="onSearch(ruleFormRef)"
+        ) 查询
+    //- 表格组件
+    PureTableBar(title="app底部导航", @refresh="onSearch(ruleFormRef)" )
+      //- template(#buttons)
+        el-button(type="primary" :icon="useRenderIcon(Addicon)" @click="handleAdd()" ) 同步配置至新版本
+
+      template(v-slot="{ size, checkList }")
+        pure-table( 
+          row-key="id"
+          ref="tableRef"
+          adaptive
+          align-whole="center"
+          table-layout="auto"
+          :loading="loading"
+          :size="size",
+          :data="dataList",
+          :columns="columns",
+          :checkList="checkList",
+          :pagination="pagination",
+          :paginationSmall="size === 'default' ? true : false",
+          :header-cell-style="{ background: 'var(--el-table-row-hover-bg-color)', color: 'var(--el-text-color-primary)' }",
+          @selection-change="handleSelectionChange",
+          @size-change="handleSizeChange",
+          @current-change="handleCurrentChange"
+        )
+          template(#operation="{ row }")
+            el-button.reset-margin(
+              link
+              type="primary"
+              size="small"
+              @click="handleUpdate(row)"
+              :icon="useRenderIcon(Delete)"
+              v-if="hasAuth(['update'])"
+            ) 编辑
+            el-button.reset-margin(
+              link
+              type="primary"
+              size="small"
+              @click="handleDelete(row)"
+              :icon="useRenderIcon(Delete)"
+              v-if="hasAuth(['delete'])"
+            ) 删除
+
+                    
+</template>
+
+<style scoped lang="scss">
+:deep(.el-dropdown-menu__item i) {
+  margin: 0;
+}
+
+:deep(.el-form-item__label) {
+  font-weight: 700;
+}
+
+:deep(.el-pagination) {
+  flex-flow: wrap;
+}
+
+:deep(.is-draggable) {
+  max-height: 80vh;
+  overflow: auto;
+}
+
+:deep(.el-dialog__header) {
+  position: sticky;
+  top: 0;
+  z-index: 2;
+  background: #fff;
+}
+
+:deep(.el-descriptions__header) {
+  margin: 16px 0 !important;
+}
+</style>

+ 54 - 48
src/views/app/fileUpdateInfo/hook.tsx

@@ -1,28 +1,38 @@
-import { reactive, onMounted, ref, ElMessage, ElMessageBox, http, getGroupUrl, RegularVerification, verification, PaginationProps } from "@/utils/importUsed"
+import {
+  reactive,
+  onMounted,
+  ref,
+  ElMessage,
+  ElMessageBox,
+  http,
+  getGroupUrl,
+  RegularVerification,
+  verification,
+  PaginationProps
+} from "@/utils/importUsed";
 // 表单实例
-const ruleFormRef = ref()
+const ruleFormRef = ref();
 export function useFileUpdateInfo() {
   // 接口列表实例
-  let UrlList = reactive(null)
+  let UrlList = reactive(null);
   // 获取当前板块接口列表
   onMounted(async () => {
     UrlList = await getGroupUrl(["kxsConfigServer"]);
     onSearch(ruleFormRef.value);
-
   });
-  let form = reactive({
+  const form = reactive({
     kind: "creater", //分类
-    appVersion: "",
-
+    appVersion: ""
   });
   const dataList = ref([]);
   const loading = ref(false);
   const dialogAddVisible = ref(false);
   const pagination = reactive<PaginationProps>({
     total: 0,
-    pageSize: 10,
+    pageSize: 30,
     currentPage: 1,
-    background: true
+    background: true,
+    pageSizes: [10, 30, 50, 100]
   });
   const columns: TableColumnList = [
     {
@@ -82,17 +92,17 @@ export function useFileUpdateInfo() {
   }
   // 选择表格项
   function handleSelectionChange(val) {
-    console.log(`SelectionChange: ${val}`);
-    onSearch(ruleFormRef.value);
+    console.log(`SelectionChange:`, val);
+    // onSearch(ruleFormRef.value);
   }
   // 搜索列表
   async function onSearch(formEl) {
     // 表单校验拦截
-    if (!formEl) return
+    if (!formEl) return;
     await formEl.validate(async (valid, fields) => {
       if (valid) {
         //表单校验成功回调
-        console.log('submit!')
+        console.log("submit!");
         // 状态调整为加载中
         loading.value = true;
         // 调用接口(需动态生成接口)
@@ -117,23 +127,21 @@ export function useFileUpdateInfo() {
           type: "error"
         });
       }
-    })
+    });
   }
 
-
-
   // 删除
   function handleDelete(row) {
-    ElMessageBox.confirm(
-      `是否删除该资源文件更新信息? `,
-      "提示",
-      {
-        confirmButtonText: "删除",
-        cancelButtonText: "取消",
-        type: "warning"
-      }
-    ).then(async () => {
-      const { status, msg }: any = await http.Request({ method: UrlList.kxsConfigServer.fileUpdateInfodelete.method, url: UrlList.kxsConfigServer.fileUpdateInfodelete.url, params: String(row.id) });
+    ElMessageBox.confirm(`是否删除该资源文件更新信息? `, "提示", {
+      confirmButtonText: "删除",
+      cancelButtonText: "取消",
+      type: "warning"
+    }).then(async () => {
+      const { status, msg }: any = await http.Request({
+        method: UrlList.kxsConfigServer.fileUpdateInfodelete.method,
+        url: UrlList.kxsConfigServer.fileUpdateInfodelete.url,
+        params: String(row.id)
+      });
       if (status === 1) {
         ElMessage({
           message: "删除成功",
@@ -145,8 +153,8 @@ export function useFileUpdateInfo() {
           confirmButtonText: "关闭",
           type: "warning"
         });
-      };
-    })
+      }
+    });
   }
 
   function handleUpdateFile() {
@@ -157,16 +165,16 @@ export function useFileUpdateInfo() {
       });
       return;
     }
-    ElMessageBox.confirm(
-      `是否同步资源文件? `,
-      "提示",
-      {
-        confirmButtonText: "同步",
-        cancelButtonText: "取消",
-        type: "warning"
-      }
-    ).then(async () => {
-      const { status, msg }: any = await http.Request({ method: UrlList.kxsConfigServer.fileUpdateInfoupdateFile.method, url: UrlList.kxsConfigServer.fileUpdateInfoupdateFile.url, params: { kind: form.kind, appVersion: form.appVersion } });
+    ElMessageBox.confirm(`是否同步资源文件? `, "提示", {
+      confirmButtonText: "同步",
+      cancelButtonText: "取消",
+      type: "warning"
+    }).then(async () => {
+      const { status, msg }: any = await http.Request({
+        method: UrlList.kxsConfigServer.fileUpdateInfoupdateFile.method,
+        url: UrlList.kxsConfigServer.fileUpdateInfoupdateFile.url,
+        params: { kind: form.kind, appVersion: form.appVersion }
+      });
       if (status === 1) {
         ElMessage({
           message: "同步成功",
@@ -178,24 +186,25 @@ export function useFileUpdateInfo() {
           confirmButtonText: "关闭",
           type: "warning"
         });
-      };
-    })
+      }
+    });
   }
 
   // 新增
   const addVisible = ref(false);
   function handleAdd() {
     addVisible.value = true;
-  };
+  }
 
   // 修改
   const editUpdateVisible = ref(false);
   const editUpdateFormData = ref({});
   function handleUpdate(row) {
+    console.log("[ row ] >", row);
     editUpdateVisible.value = true;
     // 表格数据赋值
     editUpdateFormData.value = row;
-  };
+  }
   // 更新版本号
   const editUpVersionVisible = ref(false);
   const editUpVersionFormData = ref({});
@@ -222,12 +231,10 @@ export function useFileUpdateInfo() {
         type: "warning"
       });
     }
-  };
+  }
 
   // 分类选项数据
-  const kindOptionList = [
-    { label: '创客版', id: 'creater' }
-  ]
+  const kindOptionList = [{ label: "创客版", id: "creater" }];
 
   return {
     form,
@@ -251,7 +258,6 @@ export function useFileUpdateInfo() {
     handleUpVersion,
     handleUpdateFile,
     editUpVersionVisible,
-    editUpVersionFormData,
-
+    editUpVersionFormData
   };
 }

+ 0 - 118
src/views/welcome/components/Bar.vue

@@ -1,118 +0,0 @@
-<script setup lang="ts">
-import { ref, computed, watch, type Ref } from "vue";
-import { useAppStoreHook } from "@/store/modules/app";
-import {
-  delay,
-  useDark,
-  useECharts,
-  type EchartOptions,
-  randomColor
-} from "@pureadmin/utils";
-import * as echarts from "echarts/core";
-import { bar } from "@/api/welcome";
-
-const { isDark } = useDark();
-
-const theme: EchartOptions["theme"] = computed(() => {
-  return isDark.value ? "dark" : "light";
-});
-
-const barChartRef = ref<HTMLDivElement | null>(null);
-const { setOptions, resize } = useECharts(barChartRef as Ref<HTMLDivElement>, {
-  theme
-});
-
-let barxList = [];
-let nameList = [];
-function getList() {
-  bar()
-    .then((data) => {
-      if (data.status == '1') {
-        data.data.forEach((item) => {
-          const colort = randomColor({ type: "hex" }) as string;
-          const colorb = randomColor({ type: "hex" }) as string;
-          nameList.push(item.name)
-          barxList = item.date
-          item.type = "bar"
-          item.barWidth = "15%";
-          item.itemStyle = {
-            color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
-              {
-                offset: 0,
-                color: colort
-              },
-              {
-                offset: 1,
-                color: colorb
-              }
-            ])
-          }
-        });
-        setOptions(
-          {
-            tooltip: {
-              trigger: "axis",
-              axisPointer: {
-                type: "shadow"
-              }
-            },
-            grid: {
-              bottom: "20px",
-              right: "10px"
-            },
-            legend: {
-              //@ts-expect-error
-              right: true,
-              data: nameList
-            },
-            xAxis: [
-              {
-                type: "category",
-                axisTick: {
-                  alignWithLabel: true
-                },
-                axisLabel: {
-                  interval: 0
-                  // width: "70",
-                  // overflow: "truncate"
-                },
-                data: barxList,
-                triggerEvent: true
-              }
-            ],
-            yAxis: [
-              {
-                type: "value",
-                triggerEvent: true
-              }
-            ],
-            series: data.data,
-            addTooltip: true
-          },
-          {
-            name: "click",
-            callback: params => {
-              console.log("click", params);
-            }
-          }
-        );
-      } else {
-        console.log(data.message);
-      };
-    }).catch(err => {
-      console.log(err);
-    })
-}
-
-getList();
-watch(
-  () => useAppStoreHook().getSidebarStatus,
-  () => {
-    delay(600).then(() => resize());
-  }
-);
-</script>
-
-<template>
-  <div ref="barChartRef" style="width: 100%; height: 35vh" />
-</template>

+ 0 - 177
src/views/welcome/components/Line.vue

@@ -1,177 +0,0 @@
-<!--
- * @Author: 
- * @Date: 2023-03-01 19:20:44
- * @LastEditors: Please set LastEditors
- * @LastEditTime: 2024-01-03 15:42:58
- * @Description: kxs files
- * @filePath: 
--->
-<script setup lang="ts">
-import { useIntervalFn } from "@vueuse/core";
-import { ref, computed, watch, type Ref } from "vue";
-import { useAppStoreHook } from "@/store/modules/app";
-import { line } from "@/api/welcome"
-import {
-  delay,
-  useDark,
-  useECharts,
-  type EchartOptions,
-  randomColor
-} from "@pureadmin/utils";
-
-const { isDark } = useDark();
-
-const theme: EchartOptions["theme"] = computed(() => {
-  return isDark.value ? "dark" : "default";
-});
-
-const lineChartRef = ref<HTMLDivElement | null>(null);
-const { setOptions, getInstance, resize } = useECharts(
-  lineChartRef as Ref<HTMLDivElement>,
-  { theme }
-);
-
-const xData = (() => {
-  const data: any[] = [];
-  for (let i = 1; i < 31; i++) {
-    data.push(`${i}日`);
-  }
-  return data;
-})();
-let pieColorList = [];
-function getList() {
-  line()
-    .then((data) => {
-      if (data.status == '1') {
-        data.data.forEach(item => {
-          let color = randomColor({ type: "hex" });
-          let BGcolor = randomColor({ type: "hex" });
-          pieColorList.push(item.name)
-          item.type = "line",
-            item.symbolSize = 10,
-            item.symbol = "circle",
-            item.color = BGcolor,
-            item.markPoint = {
-              label: {
-                color: color
-              },
-              data: [
-                {
-                  type: "max",
-                  name: "最大值"
-                },
-                {
-                  type: "min",
-                  name: "最小值"
-                }
-              ]
-            }
-        });
-        setOptions(
-          {
-            tooltip: {
-              trigger: "axis",
-              axisPointer: {
-                type: "shadow"
-              }
-            },
-            grid: {
-              bottom: "20px",
-              right: "10px"
-            },
-            legend: {
-              //@ts-expect-error
-              right: true,
-              data: pieColorList
-            },
-            calculable: true,
-            xAxis: [
-              {
-                triggerEvent: true,
-                type: "category",
-                splitLine: {
-                  show: false
-                },
-                axisTick: {
-                  show: false
-                },
-                data: xData
-              }
-            ],
-            yAxis: [
-              {
-                triggerEvent: true,
-                type: "value",
-                splitLine: {
-                  show: false
-                },
-                axisLine: {
-                  show: true
-                }
-              }
-            ],
-            dataZoom: [
-              {
-                type: "slider",
-                show: false,
-                realtime: true,
-                startValue: 0,
-                endValue: 24
-              }
-            ],
-            series: data.data,
-            addTooltip: true
-          },
-          {
-            name: "click",
-            callback: params => {
-              console.log("click", params);
-            }
-          },
-          {
-            name: "contextmenu",
-            callback: params => {
-              console.log("contextmenu", params);
-            }
-          },
-          // 点击空白处
-          {
-            type: "zrender",
-            name: "click",
-            callback: params => {
-              console.log("点击空白处", params);
-            }
-          }
-        );
-      } else {
-        console.log(data.message);
-      };
-    }).catch(err => {
-      console.log(err);
-    })
-}
-let a = 1;
-useIntervalFn(() => {
-  if (a == xData.length - 24) {
-    a = 0;
-  }
-  getInstance()!.dispatchAction({
-    type: "dataZoom",
-    startValue: a,
-    endValue: a + 24
-  });
-  a++;
-}, 2000);
-
-watch(
-  () => useAppStoreHook().getSidebarStatus,
-  () => {
-    delay(600).then(() => resize());
-  }
-);
-getList();
-</script>
-
-<template>
-  <div ref="lineChartRef" style="width: 100%; height: 35vh" />
-</template>

+ 0 - 102
src/views/welcome/components/Pie.vue

@@ -1,102 +0,0 @@
-<!--
- * @Author: 
- * @Date: 2023-03-01 19:20:44
- * @LastEditors: Please set LastEditors
- * @LastEditTime: 2024-01-03 14:47:29
- * @Description: kxs files
- * @filePath: 
--->
-<script setup lang="ts">
-import { ref, computed, watch, type Ref } from "vue";
-import { useAppStoreHook } from "@/store/modules/app";
-import { welcomePie } from "@/api/welcome"
-import {
-  delay,
-  useDark,
-  useECharts,
-  type EchartOptions,
-  randomColor
-} from "@pureadmin/utils";
-
-const { isDark } = useDark();
-
-const theme: EchartOptions["theme"] = computed(() => {
-  return isDark.value ? "dark" : "light";
-});
-const pieChartRef = ref<HTMLDivElement | null>(null);
-const { setOptions, resize } = useECharts(pieChartRef as Ref<HTMLDivElement>, {
-  theme
-})
-const pieColorList = [];
-function getList() {
-  welcomePie()
-    .then((data) => {
-      if (data.status == '1') {
-        data.data.forEach(() => {
-          pieColorList.push(randomColor({ type: "hex" }))
-        });
-        setOptions(
-          {
-            tooltip: {
-              trigger: "item"
-            },
-            legend: {
-              icon: "circle",
-              //@ts-expect-error
-              right: true
-            },
-            series: [
-              {
-                name: "分类占比",
-                type: "pie",
-                top: "20%",
-                radius: "80%",
-                center: ["40%", "50%"],
-                color: pieColorList,
-                data: data.data
-                // emphasis: {
-                //   itemStyle: {
-                //     shadowBlur: 10,
-                //     shadowOffsetX: 0,
-                //     shadowColor: "rgba(0, 0, 0, 0.5)"
-                //   }
-                // }
-              }
-            ]
-          },
-          {
-            name: "click",
-            callback: params => {
-              console.log("click", params);
-            }
-          },
-          // 点击空白处
-          {
-            type: "zrender",
-            name: "click",
-            callback: params => {
-              console.log("点击空白处", params);
-            }
-          }
-        );
-
-      } else {
-        console.log(data.message);
-      };
-    }).catch((err) => {
-      console.log(err);
-    })
-};
-getList();
-
-watch(
-  () => useAppStoreHook().getSidebarStatus,
-  () => {
-    delay(600).then(() => resize());
-  }
-);
-</script>
-
-<template>
-  <div ref="pieChartRef" style="width: 100%; height: 35vh" />
-</template>

+ 0 - 3
src/views/welcome/index.vue

@@ -2,9 +2,6 @@
 import dayjs from "dayjs";
 import axios from "axios";
 import MdEditor from "md-editor-v3";
-import Bar from "./components/Bar.vue";
-import Pie from "./components/Pie.vue";
-import Line from "./components/Line.vue";
 import TypeIt from "@/components/ReTypeit";
 import { ref, computed, markRaw, onMounted } from "vue";
 import Table from "./components/Table.vue";