Browse Source

菜单管理

ZaiZai 1 year ago
parent
commit
9a0cf80199
5 changed files with 324 additions and 26 deletions
  1. 1 1
      package.json
  2. 37 0
      src/api/modules/system/client.js
  3. 257 4
      src/views/system/client.vue
  4. 25 17
      src/views/system/menu.vue
  5. 4 4
      yarn.lock

+ 1 - 1
package.json

@@ -21,7 +21,7 @@
         "dayjs": "^1.11.10",
         "echarts": "^5.4.3",
         "element-plus": "2.4.4",
-        "hc-vue3-ui": "^2.8.2",
+        "hc-vue3-ui": "^2.8.3",
         "js-base64": "^3.7.5",
         "js-fast-way": "^0.3.8",
         "js-md5": "^0.8.3",

+ 37 - 0
src/api/modules/system/client.js

@@ -0,0 +1,37 @@
+import { HcApi } from '../../request/index'
+
+export default {
+    async page(form) {
+        return HcApi({
+            url: '/api/blade-system/client/list',
+            method: 'get',
+            params: form,
+        }, false)
+    },
+    //菜单详情
+    async detail(id) {
+        return HcApi({
+            url: '/api/blade-system/client/detail',
+            method: 'get',
+            params: {
+                id,
+            },
+        }, false)
+    },
+    async submit(form) {
+        return HcApi({
+            url: '/api/blade-system/client/submit',
+            method: 'post',
+            data: form,
+        }, false)
+    },
+    async del(ids) {
+        return HcApi({
+            url: '/api/blade-system/client/remove',
+            method: 'post',
+            params: {
+                ids,
+            },
+        }, false)
+    },
+}

+ 257 - 4
src/views/system/client.vue

@@ -1,11 +1,264 @@
 <template>
-    <div>11</div>
+    <hc-new-card>
+        <template #header>
+            <div class="w-60">
+                <hc-search-input v-model="searchForm.clientId" placeholder="请输入应用id关键词" @search="searchClick" />
+            </div>
+        </template>
+        <template #extra>
+            <el-button hc-btn type="primary" @click="addClick">新增</el-button>
+            <el-button hc-btn type="danger" @click="delClick">删除</el-button>
+        </template>
+        <hc-table
+            ref="tableRef" :column="tableColumn" :datas="tableData" :loading="tableLoading"
+            is-new is-check :check-style="{ width: 29 }" :index-style="{ width: 60 }"
+            @selection-change="tableCheckChange"
+        >
+            <template #action="{ row }">
+                <el-link type="warning" @click="editRowClick(row)">修改</el-link>
+                <el-link type="danger" @click="delRowClick(row)">删除</el-link>
+            </template>
+        </hc-table>
+        <template #action>
+            <hc-pages :pages="searchForm" @change="pageChange" />
+        </template>
+
+        <!-- 新增/修改 菜单 -->
+        <hc-new-dialog v-model="isDialogShow" widths="800px" is-footer-center :title="dialogTitle" @close="dialogClose">
+            <el-form ref="formRef" :model="formModel" :rules="formRules" label-position="top" label-width="auto">
+                <el-row :gutter="20">
+                    <el-col :span="8">
+                        <el-form-item label="应用名称:" prop="name">
+                            <el-input v-model="formModel.name" clearable placeholder="请输入应用名称" />
+                        </el-form-item>
+                    </el-col>
+                    <el-col :span="8">
+                        <el-form-item label="应用id:" prop="clientId">
+                            <el-input v-model="formModel.clientId" clearable placeholder="请输入应用id" />
+                        </el-form-item>
+                    </el-col>
+                    <el-col :span="8">
+                        <el-form-item label="应用密钥:" prop="clientSecret">
+                            <el-input v-model="formModel.clientSecret" clearable placeholder="请输入应用密钥" />
+                        </el-form-item>
+                    </el-col>
+                    <el-col :span="24">
+                        <el-form-item label="授权类型:" prop="authorizedGrantTypes">
+                            <div class="form-item-div">
+                                <el-checkbox-group v-model="formModel.authorizedGrantTypes" size="large">
+                                    <el-checkbox label="refresh_token" />
+                                    <el-checkbox label="password" />
+                                    <el-checkbox label="authorization_code" />
+                                    <el-checkbox label="captcha" />
+                                    <el-checkbox label="social" />
+                                </el-checkbox-group>
+                            </div>
+                        </el-form-item>
+                    </el-col>
+                    <el-col :span="8">
+                        <el-form-item label="授权范围:" prop="scope">
+                            <el-input v-model="formModel.scope" clearable placeholder="请输入授权范围" />
+                        </el-form-item>
+                    </el-col>
+                    <el-col :span="8">
+                        <el-form-item label="令牌秒数:" prop="accessTokenValidity">
+                            <el-input-number v-model="formModel.accessTokenValidity" :min="1" block controls-position="right" placeholder="请输入令牌秒数" />
+                        </el-form-item>
+                    </el-col>
+                    <el-col :span="8">
+                        <el-form-item label="刷新秒数:" prop="refreshTokenValidity">
+                            <el-input-number v-model="formModel.refreshTokenValidity" :min="1" block controls-position="right" placeholder="请输入令牌秒数" />
+                        </el-form-item>
+                    </el-col>
+                    <el-col :span="8">
+                        <el-form-item label="资源集合:">
+                            <el-input v-model="formModel.resourceIds" clearable placeholder="请输入资源集合" />
+                        </el-form-item>
+                    </el-col>
+                    <el-col :span="8">
+                        <el-form-item label="权限:">
+                            <el-input v-model="formModel.authorities" clearable placeholder="请输入权限" />
+                        </el-form-item>
+                    </el-col>
+                    <el-col :span="8">
+                        <el-form-item label="自动授权:">
+                            <el-input v-model="formModel.autoapprove" clearable placeholder="请输入自动授权" />
+                        </el-form-item>
+                    </el-col>
+                    <el-col :span="24">
+                        <el-form-item label="回调地址:" prop="webServerRedirectUri">
+                            <el-input v-model="formModel.webServerRedirectUri" clearable placeholder="请输入回调地址" />
+                        </el-form-item>
+                    </el-col>
+                </el-row>
+            </el-form>
+            <template #footer>
+                <el-button hc-btn @click="dialogClose">取消</el-button>
+                <el-button hc-btn type="primary" :loading="submitLoading" @click="dialogSubmit">提交</el-button>
+            </template>
+        </hc-new-dialog>
+    </hc-new-card>
 </template>
 
 <script setup>
+import { nextTick, onActivated, ref } from 'vue'
+import { arrToId, formValidate, getArrValue } from 'js-fast-way'
+import { delMessage, reloadPage } from '~uti/tools'
+import mainApi from '~api/system/client'
 
-</script>
+//激活
+onActivated(() => {
+    searchForm.value.current = 1
+    getTableData()
+})
+
+//搜索表单
+const searchForm = ref({ clientId: null, current: 1, size: 30, total: 0 })
+
+//搜索
+const searchClick = () => {
+    searchForm.value.current = 1
+    getTableData()
+}
+
+//分页
+const pageChange = ({ current, size }) => {
+    searchForm.value.current = current
+    searchForm.value.size = size
+    getTableData()
+}
+
+//表格数据
+const tableRef = ref(null)
+const tableColumn = ref([
+    { key: 'name', name: '应用名称' },
+    { key: 'clientId', name: '应用ID' },
+    { key: 'clientSecret', name: '应用密钥' },
+    { key: 'scope', name: '授权范围' },
+    { key: 'accessTokenValidity', name: '令牌秒数' },
+    { key: 'refreshTokenValidity', name: '刷新秒数' },
+    { key: 'action', name: '操作', width: 100, align: 'center' },
+])
+
+//获取表格数据
+const tableLoading = ref(false)
+const tableData = ref([{}])
+const getTableData = async () => {
+    tableData.value = []
+    tableLoading.value = true
+    const { error, code, data } = await mainApi.page(searchForm.value)
+    tableLoading.value = false
+    if (!error && code === 200) {
+        tableData.value = getArrValue(data['records'])
+        searchForm.value.total = data['total']
+    } else {
+        tableData.value = []
+        searchForm.value.total = 0
+    }
+}
+
+//表格被选择
+const tableCheckKeys = ref([])
+const tableCheckChange = (rows) => {
+    tableCheckKeys.value = rows
+}
 
-<style scoped lang="scss">
+//新增/修改 弹窗
+const isDialogShow = ref(false)
+const dialogTitle = ref('')
 
-</style>
+//菜单表单
+const formRef = ref(null)
+const formModel = ref({})
+const formRules = {
+    name: {
+        required: true,
+        trigger: 'blur',
+        message: '请输入菜单名称',
+    },
+}
+
+//新增
+const addClick = () => {
+    dialogTitle.value = '新增菜单'
+    formModel.value = {
+        category: 1, isOpen: 1, isLayout: 1, isShowButton: 1,
+        sort: 1,
+    }
+    //显示表单弹窗
+    nextTick(() => {
+        isDialogShow.value = true
+    })
+}
+
+//修改菜单
+const editRowClick = (row) => {
+    formModel.value = {}
+    dialogTitle.value = '修改菜单'
+    formModel.value = { ...row }
+    //显示表单弹窗
+    nextTick(() => {
+        isDialogShow.value = true
+    })
+}
+
+//删除菜单
+const delRowClick = (row) => {
+    delMessage(async () => {
+        const { code, msg } = await mainApi.del(row.id)
+        if (code === 200) {
+            window.$message.success('删除成功')
+            reloadPage()
+        } else {
+            window.$message.error(msg ?? '删除失败')
+        }
+    })
+}
+
+//批量删除菜单
+const delClick = () => {
+    const rows = tableCheckKeys.value
+    if (rows.length <= 0) {
+        window.$message.warning('请选择要删除的菜单')
+        return false
+    }
+    //确认删除菜单
+    delMessage(async () => {
+        const ids = arrToId(rows)
+        const { code, msg } = await mainApi.del(ids)
+        if (code === 200) {
+            window.$message.success('删除成功')
+            reloadPage()
+        } else {
+            window.$message.error(msg ?? '删除失败')
+        }
+    })
+}
+
+//提交表单
+const submitLoading = ref(false)
+const dialogSubmit = async () => {
+    const formRes = await formValidate(formRef.value)
+    if (!formRes) return false
+    submitLoading.value = true
+    //处理数据
+    const form = formModel.value
+    form.parentId = form.parentId ?? 0
+    form.alias = form.alias ?? form.code
+    //发起请求
+    const { error, code, msg } = await mainApi.submit(form)
+    if (!error && code === 200) {
+        window?.$message?.error('操作成功')
+        reloadPage()
+    } else {
+        window?.$message?.error(msg ?? '操作失败')
+    }
+}
+
+//关闭弹窗
+const dialogClose = () => {
+    isDialogShow.value = false
+    submitLoading.value = false
+    formModel.value = {}
+}
+</script>

+ 25 - 17
src/views/system/menu.vue

@@ -78,34 +78,42 @@
                     </el-col>
                     <el-col :span="8">
                         <el-form-item label="菜单类型:" prop="category">
-                            <el-radio-group v-model="formModel.category">
-                                <el-radio :label="1">菜单</el-radio>
-                                <el-radio :label="2">按钮</el-radio>
-                            </el-radio-group>
+                            <div class="form-item-div">
+                                <el-radio-group v-model="formModel.category">
+                                    <el-radio :label="1">菜单</el-radio>
+                                    <el-radio :label="2">按钮</el-radio>
+                                </el-radio-group>
+                            </div>
                         </el-form-item>
                     </el-col>
                     <el-col v-if="formModel.category === 2" :span="8">
                         <el-form-item label="按钮是否显示:" prop="isShowButton">
-                            <el-radio-group v-model="formModel.isShowButton">
-                                <el-radio :label="2">否</el-radio>
-                                <el-radio :label="1">是</el-radio>
-                            </el-radio-group>
+                            <div class="form-item-div">
+                                <el-radio-group v-model="formModel.isShowButton">
+                                    <el-radio :label="2">否</el-radio>
+                                    <el-radio :label="1">是</el-radio>
+                                </el-radio-group>
+                            </div>
                         </el-form-item>
                     </el-col>
                     <el-col v-if="formModel.category === 1" :span="8">
                         <el-form-item label="是否外层:" prop="isLayout">
-                            <el-radio-group v-model="formModel.isLayout">
-                                <el-radio :label="1">否</el-radio>
-                                <el-radio :label="2">是</el-radio>
-                            </el-radio-group>
+                            <div class="form-item-div">
+                                <el-radio-group v-model="formModel.isLayout">
+                                    <el-radio :label="1">否</el-radio>
+                                    <el-radio :label="2">是</el-radio>
+                                </el-radio-group>
+                            </div>
                         </el-form-item>
                     </el-col>
                     <el-col :span="8">
                         <el-form-item label="是否新窗口:" prop="isOpen">
-                            <el-radio-group v-model="formModel.isOpen">
-                                <el-radio :label="1">否</el-radio>
-                                <el-radio :label="2">是</el-radio>
-                            </el-radio-group>
+                            <div class="form-item-div">
+                                <el-radio-group v-model="formModel.isOpen">
+                                    <el-radio :label="1">否</el-radio>
+                                    <el-radio :label="2">是</el-radio>
+                                </el-radio-group>
+                            </div>
                         </el-form-item>
                     </el-col>
                     <el-col :span="8">
@@ -273,6 +281,7 @@ const getLevelMenuData = async () => {
 
 //新增/修改 弹窗
 const isDialogShow = ref(false)
+const iconDialogTitle = ref('')
 
 //菜单表单
 const formRef = ref(null)
@@ -327,7 +336,6 @@ const formRules = {
 
 //图标选择
 const isIconShow = ref(false)
-const iconDialogTitle = ref('')
 const menuIconFinish = (icon) => {
     formModel.value.source = icon
     isIconShow.value = false

+ 4 - 4
yarn.lock

@@ -1915,10 +1915,10 @@ has-flag@^4.0.0:
   resolved "http://47.110.251.215:9000/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b"
   integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==
 
-hc-vue3-ui@^2.8.2:
-  version "2.8.2"
-  resolved "http://47.110.251.215:9000/hc-vue3-ui/-/hc-vue3-ui-2.8.2.tgz#923453b02615c8b71df2632860917fb9867bb23c"
-  integrity sha512-L5ASUHBHe4xXZSLdU4XaPTtMsPHv/RSNCSF3lwyciNQBP13TbFOpD77pBjB7F+lPRQ54fChQ7VMBYMWenVkkTQ==
+hc-vue3-ui@^2.8.3:
+  version "2.8.3"
+  resolved "http://47.110.251.215:9000/hc-vue3-ui/-/hc-vue3-ui-2.8.3.tgz#1ef51ebc3f06ea4a2b1f870aa7ae83f1cf5b5832"
+  integrity sha512-lSgR/uXm5LXLyct89WewXMonw81PQCCrjnhQP/4Sj0O2rdWwkdU6YOrNj5djEv2l48khUuF/798UoZnuUEEj6A==
 
 human-signals@^2.1.0:
   version "2.1.0"