|
@@ -0,0 +1,186 @@
|
|
|
+<script setup lang="ts">
|
|
|
+import { ref } from "vue";
|
|
|
+import { ElMessageBox, ElTree } from "element-plus";
|
|
|
+import { httpActionList, httpList } from "/@/api/mobile/stock1";
|
|
|
+import { useAsync, useResponseHandle } from "/@/hooks/useAsync";
|
|
|
+import { createActionButtons } from "./config/buttons";
|
|
|
+import { methods } from "./config/requeset";
|
|
|
+import ActionModal from "./cpns/action-modal.vue";
|
|
|
+
|
|
|
+const expandedKeys = ref<string[]>([]);
|
|
|
+const treeRef = ref<InstanceType<typeof ElTree>>(null);
|
|
|
+const actionModalRef = ref<InstanceType<typeof ActionModal>>(null);
|
|
|
+
|
|
|
+const defaultProps = {
|
|
|
+ children: "child",
|
|
|
+ isLeaf: "isLeaf",
|
|
|
+ label: data => {
|
|
|
+ if (data.action_name) return data.action_name;
|
|
|
+ return data.menu_name;
|
|
|
+ }
|
|
|
+};
|
|
|
+
|
|
|
+const responseHandle = useResponseHandle();
|
|
|
+
|
|
|
+const { data, run, loading } = useAsync<Menu[]>({
|
|
|
+ initalData: [],
|
|
|
+ success: () => data.value.forEach(({ id }) => expandedKeys.value.push(id))
|
|
|
+});
|
|
|
+
|
|
|
+async function handleLoad({ data }: { data: Menu }, resolve: any) {
|
|
|
+ if (data.child) return resolve(data.child);
|
|
|
+ //加载action
|
|
|
+ const { id } = data;
|
|
|
+ if (!id) return;
|
|
|
+
|
|
|
+ const { data: _data, code, message } = await httpActionList({ id });
|
|
|
+
|
|
|
+ responseHandle({
|
|
|
+ code,
|
|
|
+ message,
|
|
|
+ handler: () => {
|
|
|
+ const nodes = _data.map(item => ({ ...item, isLeaf: true }));
|
|
|
+ resolve(nodes);
|
|
|
+ }
|
|
|
+ });
|
|
|
+}
|
|
|
+
|
|
|
+function handleDelOrStatus(action: string, data, isMenu: boolean) {
|
|
|
+ const apis = isMenu ? methods.action : methods.action;
|
|
|
+ const isDelete = action === "delete";
|
|
|
+ const api = isDelete ? apis.del : apis.status;
|
|
|
+ const status = data.status === "0" ? "1" : "0";
|
|
|
+ const statusText = status === "0" ? "禁用" : "启用";
|
|
|
+
|
|
|
+ ElMessageBox.confirm(`确定要${isDelete ? "删除" : statusText}?`).then(
|
|
|
+ async () => {
|
|
|
+ const { code, message } = await api({
|
|
|
+ ...{ id: data.value },
|
|
|
+ ...(isDelete ? {} : { status })
|
|
|
+ });
|
|
|
+
|
|
|
+ responseHandle({
|
|
|
+ code,
|
|
|
+ message,
|
|
|
+ handler: () =>
|
|
|
+ handleUpdateTreeNode(data.menuid ? data.menuid : data.value)
|
|
|
+ });
|
|
|
+ }
|
|
|
+ );
|
|
|
+}
|
|
|
+
|
|
|
+function handleAction(event: MouseEvent, action: string, data: any, node: any) {
|
|
|
+ event.stopPropagation();
|
|
|
+ const isMenu = !data.action_code;
|
|
|
+ const level = Number(node.level);
|
|
|
+
|
|
|
+ //删除禁用操作不打开模态框
|
|
|
+ if (["status", "delete"].includes(action)) {
|
|
|
+ return handleDelOrStatus(action, data, isMenu);
|
|
|
+ }
|
|
|
+
|
|
|
+ actionModalRef.value.onDisplay(
|
|
|
+ action,
|
|
|
+ level === 1 || (level === 2 && action === "update"),
|
|
|
+ data
|
|
|
+ );
|
|
|
+}
|
|
|
+
|
|
|
+async function handleUpdateTreeNode(id: string) {
|
|
|
+ const { data: _data, message, code } = await httpActionList({ id });
|
|
|
+
|
|
|
+ responseHandle({
|
|
|
+ code,
|
|
|
+ message,
|
|
|
+ handler: () => {
|
|
|
+ const nodes = _data.map(item => ({ ...item, isLeaf: true }));
|
|
|
+ treeRef.value.updateKeyChildren(id, nodes);
|
|
|
+ }
|
|
|
+ });
|
|
|
+}
|
|
|
+
|
|
|
+const requesetMenulist = () => run(httpList());
|
|
|
+
|
|
|
+requesetMenulist();
|
|
|
+</script>
|
|
|
+
|
|
|
+<template>
|
|
|
+ <div class="p-2">
|
|
|
+ <div
|
|
|
+ class="w-full flex justify-between bg-white px-[15px] pb-5 pt-3 align-bottom"
|
|
|
+ >
|
|
|
+ <h1 class="font-bold">菜单设置</h1>
|
|
|
+ <div>
|
|
|
+ <ElButton type="success">添加</ElButton>
|
|
|
+ <ElButton type="primary" @click="() => requesetMenulist()"
|
|
|
+ >刷新</ElButton
|
|
|
+ >
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+
|
|
|
+ <ElTree
|
|
|
+ lazy
|
|
|
+ :load="handleLoad"
|
|
|
+ node-key="id"
|
|
|
+ ref="treeRef"
|
|
|
+ :data="data"
|
|
|
+ v-loading="loading"
|
|
|
+ :default-expanded-keys="expandedKeys"
|
|
|
+ :props="defaultProps"
|
|
|
+ >
|
|
|
+ <template #="{ node, data }">
|
|
|
+ <!-- 操作按钮 -->
|
|
|
+ <div class="mobile-tree-node" v-if="data.action_name">
|
|
|
+ <p>{{ data.action_name }}</p>
|
|
|
+ <div>
|
|
|
+ <ElButton
|
|
|
+ v-for="btn in createActionButtons(data.status).slice(1)"
|
|
|
+ size="small"
|
|
|
+ plain
|
|
|
+ :type="btn.type"
|
|
|
+ :key="btn.action"
|
|
|
+ @click="event => handleAction(event, btn.action, data, node)"
|
|
|
+ >{{ btn.label }}</ElButton
|
|
|
+ >
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+
|
|
|
+ <!-- 菜单 -->
|
|
|
+ <div class="mobile-tree-node" v-else>
|
|
|
+ <p>{{ data.menu_name }}</p>
|
|
|
+ <div>
|
|
|
+ <ElButton
|
|
|
+ v-for="btn in createActionButtons(data.status)"
|
|
|
+ size="small"
|
|
|
+ plain
|
|
|
+ :type="btn.type"
|
|
|
+ :key="btn.action"
|
|
|
+ @click="event => handleAction(event, btn.action, data, node)"
|
|
|
+ >{{ btn.label }}</ElButton
|
|
|
+ >
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </template>
|
|
|
+ </ElTree>
|
|
|
+
|
|
|
+ <ActionModal ref="actionModalRef" @node-update="handleUpdateTreeNode" />
|
|
|
+ </div>
|
|
|
+</template>
|
|
|
+
|
|
|
+<style lang="scss" scoped>
|
|
|
+:deep(.el-tree-node__label) {
|
|
|
+ flex: 1;
|
|
|
+}
|
|
|
+
|
|
|
+:deep(.el-tree-node__content) {
|
|
|
+ height: 35px;
|
|
|
+}
|
|
|
+
|
|
|
+.mobile-tree-node {
|
|
|
+ width: 100%;
|
|
|
+ display: flex;
|
|
|
+ justify-content: space-between;
|
|
|
+ padding-right: 15px;
|
|
|
+ align-items: center;
|
|
|
+}
|
|
|
+</style>
|