feat:完成所有功能

This commit is contained in:
zy7y 2022-09-19 18:02:37 +08:00
parent 7b1d87aa65
commit 5566e6693b
32 changed files with 343 additions and 124 deletions

View File

@ -1,5 +1,5 @@
from core.utils import list_to_tree from core.utils import list_to_tree
from dbhelper.menu import del_menu, get_tree_menu, insert_menu, put_menu from dbhelper.menu import del_menu, get_menu, get_tree_menu, insert_menu, put_menu
from schemas import MenuIn, MenuRead, Response from schemas import MenuIn, MenuRead, Response
@ -9,10 +9,16 @@ async def menu_add(data: MenuIn) -> Response[MenuRead]:
async def menu_arr() -> Response: async def menu_arr() -> Response:
menus = await get_tree_menu() menus = await get_tree_menu()
return Response(data=list_to_tree(menus)) try:
data = list_to_tree(menus)
except KeyError:
return Response(code=400, msg="菜单根节点丢失")
return Response(data=data)
async def menu_del(pk: int) -> Response: async def menu_del(pk: int) -> Response:
if await get_menu({"pid": pk}) is not None:
return Response(code=400, msg="请先删除子节点")
if await del_menu(pk) == 0: if await del_menu(pk) == 0:
return Response(code=400, msg="菜单不存在") return Response(code=400, msg="菜单不存在")
return Response() return Response()

View File

@ -26,8 +26,7 @@ async def role_has_menu(rid: int):
rid: 角色ID rid: 角色ID
""" """
menus = await get_role_menus(rid) menus = await get_role_menus(rid)
for obj in menus:
obj["meta"] = json.loads(obj["meta"]) if obj["meta"] is not None else None
try: try:
result = list_to_tree(menus) result = list_to_tree(menus)
except KeyError: except KeyError:

View File

@ -70,7 +70,7 @@ async def check_permissions(request: Request, user: UserModel = Depends(check_to
result = await get_user_info(user) result = await get_user_info(user)
active_rid = result["roles"][0]["id"] active_rid = result["roles"][0]["id"]
# 白名单 # 白名单 登录用户信息, 登录用户菜单信息
whitelist = [f"/user/{user.id}", f"/role/{active_rid}/menu"] whitelist = [f"/user/{user.id}", f"/role/{active_rid}/menu"]
flag = request.url.path in whitelist and request.method == "GET" flag = request.url.path in whitelist and request.method == "GET"
if flag: if flag:

View File

@ -45,7 +45,7 @@ async def get_menu(kwargs):
async def del_menu(mid: int): async def del_menu(mid: int):
"""删除用户""" """删除菜单"""
return await MenuModel.filter(id=mid).update(status=9) return await MenuModel.filter(id=mid).update(status=9)

View File

@ -9,11 +9,12 @@ async def get_role_menus(rid: int):
根据角色id 获取菜单 根据角色id 获取菜单
""" """
db = connections.get("default") db = connections.get("default")
# asc 降序
return await db.execute_query_dict( return await db.execute_query_dict(
""" """
select m.id, m.name, m.meta, m.path, m.type, m.component, m.pid, m.identifier, m.api, m.method select m.id, m.name, m.icon, m.path, m.type, m.component, m.pid, m.identifier, m.api, m.method
FROM sys_menu as m, sys_role_menu WHERE m.id = sys_role_menu.mid FROM sys_menu as m, sys_role_menu WHERE m.id = sys_role_menu.mid
AND sys_role_menu.rid = (?) AND sys_role_menu.`status` = 1""", AND sys_role_menu.rid = (?) AND sys_role_menu.`status` = 1 order by m.id asc""",
[rid], [rid],
) )

View File

@ -31,7 +31,6 @@ async def get_user_info(user: UserModel):
""", """,
[user.id], [user.id],
) )
return { return {
**jsonable_encoder(user), **jsonable_encoder(user),
"roles": sql_result, "roles": sql_result,
@ -81,9 +80,9 @@ async def put_user(uid: int, data: UserPut):
"""更新用户""" """更新用户"""
from core.security import get_password_hash from core.security import get_password_hash
roles = data.roles rids = data.roles
del data.roles del data.roles
for role in roles: for role in rids:
if await get_role({"id": role.rid, "status__not": 9}) is None: if await get_role({"id": role.rid, "status__not": 9}) is None:
return role.rid return role.rid
# 更新用户 # 更新用户
@ -104,13 +103,12 @@ async def put_user(uid: int, data: UserPut):
""", """,
[uid], [uid],
) )
print(has_roles)
# 2. 将先有的数据标记 删除 # 2. 将先有的数据标记 删除
[ [
await db.execute_query_dict( await db.execute_query_dict(
""" """
update sys_user_role set status = 9 where rid = (?) update sys_user_role set status = 9 where rid = (?)
""", """,
[role["id"]], [role["id"]],
) )
@ -119,11 +117,13 @@ async def put_user(uid: int, data: UserPut):
# 2. 新增次此更新的数据 # 2. 新增次此更新的数据
await UserRoleModel.bulk_create( await UserRoleModel.bulk_create(
[UserRoleModel(uid=uid, **role.dict()) for role in roles] [UserRoleModel(uid=uid, **role.dict()) for role in rids]
) )
async def select_role(uid: int, rid: int): async def select_role(uid: int, rid: int):
"""用户切换角色""" """用户切换角色"""
await UserRoleModel.filter(uid=uid, rid__not=rid).update(status=1) # 1.将用户id 未删除角色状态置为正常 1 除切换角色id
return await UserRoleModel.filter(uid=uid, rid=rid).update(status=5) await UserRoleModel.filter(uid=uid, rid__not=rid, status__not=9).update(status=1)
# 2.将用户id 角色id 和当前角色匹配的数据置为选中
return await UserRoleModel.filter(uid=uid, rid=rid, status__not=9).update(status=5)

BIN
backend/mini.db Normal file

Binary file not shown.

BIN
backend/mini.db-shm Normal file

Binary file not shown.

BIN
backend/mini.db-wal Normal file

Binary file not shown.

View File

@ -7,9 +7,9 @@ class MenuModel(Table):
""" """
name = fields.CharField(max_length=20, description="名称", null=True) name = fields.CharField(max_length=20, description="名称", null=True)
meta = fields.JSONField(description="元数据信息", null=True) icon = fields.CharField(max_length=100, description="菜单图标", null=True)
path = fields.CharField(max_length=128, description="菜单url", null=True) path = fields.CharField(max_length=128, description="菜单url", null=True)
type = fields.SmallIntField(description="菜单类型 0目录 1组件 2按钮") type = fields.SmallIntField(description="菜单类型 0目录 1组件 2按钮 3数据")
component = fields.CharField(max_length=128, description="组件地址", null=True) component = fields.CharField(max_length=128, description="组件地址", null=True)
pid = fields.IntField(description="父id", null=True) pid = fields.IntField(description="父id", null=True)
identifier = fields.CharField(max_length=30, description="权限标识 user:add", null=True) identifier = fields.CharField(max_length=30, description="权限标识 user:add", null=True)

View File

@ -121,7 +121,6 @@ class Route(routing.APIRoute):
has_perm = {"dependencies": [Depends(check_permissions)]} has_perm = {"dependencies": [Depends(check_permissions)]}
has_perm = {}
routes = [ routes = [
Route.post("/login", endpoint=login, tags=["公共"], summary="登录"), Route.post("/login", endpoint=login, tags=["公共"], summary="登录"),

View File

@ -6,8 +6,8 @@ from schemas.common import ReadBase
class MenuBasic(BaseModel): class MenuBasic(BaseModel):
name: str name: str = Field(..., description="菜单名称")
meta: dict = Field(default=None, description="元信息") icon: str = Field(default=None, description="菜单图标")
path: Optional[str] = Field(default=None, description="前端路由地址") path: Optional[str] = Field(default=None, description="前端路由地址")
type: int = Field(description="0 目录 1 组件 2 按钮 3数据") type: int = Field(description="0 目录 1 组件 2 按钮 3数据")
component: Optional[str] = Field(default=None, description="前端组件地址") component: Optional[str] = Field(default=None, description="前端组件地址")

View File

@ -13,29 +13,21 @@ dirs = [
( (
"/menu", "/menu",
MenuIn( # id 1 MenuIn( # id 1
name="系统管理", name="系统面板",
meta={"icon": "AppstoreOutlined"}, icon="DashboardOutlined",
path="/system", path="/dashboard",
type=0, type=0,
component=None,
pid=0, pid=0,
identifier=None,
api=None,
method=None,
).dict(), ).dict(),
), ),
( (
"/menu", "/menu",
MenuIn( # id 2 MenuIn( # id 2
name="系统设置", name="系统管理",
meta={"icon": "SettingOutlined"}, icon="AppstoreOutlined",
path="/system", path="/system",
type=0, type=0,
component=None,
pid=0, pid=0,
identifier=None,
api=None,
method=None,
).dict(), ).dict(),
), ),
] ]
@ -54,44 +46,44 @@ menus = [
"/menu", "/menu",
MenuIn( # id 3 MenuIn( # id 3
name="用户管理", name="用户管理",
meta={"icon": "TeamOutlined", "title": "用户管理"}, icon="TeamOutlined",
path="/system/user", path="/system/user",
type=1, type=1,
component="/system/user/user.vue", component="/system/user/user.vue",
pid=1, pid=2,
).dict(), ).dict(),
), ),
( (
"/menu", "/menu",
MenuIn( # id 4 MenuIn( # id 4
name="角色管理", name="角色管理",
meta={"icon": "UserOutlined", "title": "角色管理"}, icon="UserOutlined",
path="/system/role", path="/system/role",
type=1, type=1,
component="/system/role/role.vue", component="/system/role/role.vue",
pid=1, pid=2,
).dict(), ).dict(),
), ),
( (
"/menu", "/menu",
MenuIn( # id 5 MenuIn( # id 5
name="菜单管理", name="菜单管理",
meta={"icon": "MenuOutlined", "title": "菜单管理"}, icon="MenuOutlined",
path="/system/menu", path="/system/menu",
type=1, type=1,
component="/system/menu/menu.vue", component="/system/menu/menu.vue",
pid=1, pid=2,
).dict(), ).dict(),
), ),
( (
"/menu", "/menu",
MenuIn( # id 6 MenuIn( # id 6
name="关于", name="数据面板",
meta={"icon": "DashboardOutlined", "title": "关于"}, icon="AreaChartOutlined",
path="/setting/about", path="/dashboard/index",
type=1, type=1,
component="/setting/about/about.vue", component="/dashboard/index/index.vue",
pid=2, pid=1,
).dict(), ).dict(),
), ),
] ]
@ -174,7 +166,6 @@ role_manager_pre = [
), ),
MenuIn( MenuIn(
name="角色查询", name="角色查询",
meta={"icon": "Search"},
type=2, type=2,
identifier="role:query", identifier="role:query",
api="/role/query", api="/role/query",
@ -260,13 +251,14 @@ menus_len = (
+ len(dirs) + len(dirs)
+ len(role_manager_pre) + len(role_manager_pre)
+ len(menu_manager_pre) + len(menu_manager_pre)
+ 1
) )
datas = [ datas = [
( (
"/role", "/role",
RoleIn( RoleIn(
name="superStar", name="超管",
remark="全部权限", remark="全部权限",
menus=[num for num in range(1, menus_len)], menus=[num for num in range(1, menus_len)],
), ),
@ -276,7 +268,7 @@ datas = [
"/user", "/user",
UserAdd( UserAdd(
username="admin", username="admin",
nickname="666管理员", nickname="乐师高渐离",
password="123456", password="123456",
roles=[RoleActive(rid=1, status=5)], roles=[RoleActive(rid=1, status=5)],
), ),

View File

@ -1,6 +1,6 @@
<script setup> <script setup>
import { ref } from 'vue' import { ref } from 'vue'
import UserInfo from '@/components/layout/layout-info/layout-info.vue' import UserInfo from '@/components/layout/right/info.vue'
import HeaderCrumb from './header-crumb.vue' import HeaderCrumb from './header-crumb.vue'
// //

View File

@ -14,7 +14,7 @@ const roleChangeRef = ref()
const onClick = ({ key }) => { const onClick = ({ key }) => {
if (key === '1') { if (key === '1') {
// //
roleChangeRef.value?.showModal() roleChangeRef.value.visible = true
} else { } else {
store.$reset() store.$reset()
router.push('/login') router.push('/login')

View File

@ -1,10 +1,11 @@
<script setup> <script setup>
import { ref, computed } from 'vue' import { ref, computed } from 'vue'
import { userStore } from '@/stores/user' import { userStore } from '@/stores/user'
import { useRouter } from 'vue-router'
const store = userStore() const store = userStore()
const router = useRouter()
const loading = ref(false)
const visible = ref(false) const visible = ref(false)
const currentRoleId = ref(store.userInfo.roles[0].id) const currentRoleId = ref(store.userInfo.roles[0].id)
@ -17,35 +18,32 @@ const options = computed(() => {
})) }))
}) })
const showModal = () => {
visible.value = true
}
const handleOk = () => { const handleOk = () => {
loading.value = true visible.value = !visible.value
store.userSelectRole(currentRoleId.value) store.userSelectRole(currentRoleId.value)
// todo // todo
visible.value = false router.replace({
path: '/back'
})
visible.value = !visible.value
} }
const handleCancel = () => { const handleCancel = () => {
visible.value = false visible.value = !visible.value
} }
defineExpose({ defineExpose({
showModal visible
}) })
</script> </script>
<template> <template>
<div class="select-role"> <div class="select-role">
<a-modal v-model:visible="visible" title="切换角色" @ok="handleOk"> <a-modal v-model:visible="visible" title="切换角色">
<template #footer> <template #footer>
<a-button key="back" @click="handleCancel">取消</a-button> <a-button key="back" @click="handleCancel">取消</a-button>
<a-button <a-button
key="submit" key="submit"
type="primary" type="primary"
:loading="loading"
@click="handleOk" @click="handleOk"
:disabled="currentRoleId === store.userInfo.roles[0]['id']" :disabled="currentRoleId === store.userInfo.roles[0]['id']"
>确定</a-button >确定</a-button

View File

@ -20,14 +20,14 @@ const menuClick = (menu) => {
<template v-if="menu.type === 0"> <template v-if="menu.type === 0">
<a-sub-menu :key="menu.id"> <a-sub-menu :key="menu.id">
<template #icon> <template #icon>
<component :is="$loadIconCpn(menu.meta.icon)"></component> <component :is="$loadIconCpn(menu.icon)"></component>
</template> </template>
<template #title>{{ menu.name }}</template> <template #title>{{ menu.name }}</template>
<!-- 1 组件 子菜单项 --> <!-- 1 组件 子菜单项 -->
<template v-for="sub in menu.children" :key="sub.id"> <template v-for="sub in menu.children" :key="sub.id">
<a-menu-item @click="menuClick(sub)"> <a-menu-item @click="menuClick(sub)">
<template #icon> <template #icon>
<component :is="$loadIconCpn(sub.meta.icon)"></component> <component :is="$loadIconCpn(sub.icon)"></component>
</template> </template>
<span>{{ sub.name }}</span> <span>{{ sub.name }}</span>
</a-menu-item> </a-menu-item>

View File

@ -1,21 +1,5 @@
import { ref } from 'vue' import { ref } from 'vue'
// 菜单类型映射
export const menuType = {
0: '目录',
1: '菜单',
2: '按钮',
3: '数据'
}
// 请求方法颜色映射
export const methodColor = {
GET: '#61AFFE',
POST: '#49CC90',
DELETE: '#F93E3E',
PUT: '#FCA130'
}
export const tableTree = () => { export const tableTree = () => {
// 1.适配菜单表格 // 1.适配菜单表格
// 展开行 https://blog.csdn.net/weixin_52691965/article/details/120494451 // 展开行 https://blog.csdn.net/weixin_52691965/article/details/120494451

View File

@ -1,5 +1,6 @@
<script setup> <script setup>
import { menuType, methodColor, tableTree } from './conf' import { tableTree } from './conf'
import { menuType, methodColor } from '@/views/main/system/menu/conf'
/**接受父组件传递过来的值 */ /**接受父组件传递过来的值 */
defineProps({ defineProps({
@ -73,8 +74,8 @@ const expand = tableTree()
> >
<template #bodyCell="{ column, record }"> <template #bodyCell="{ column, record }">
<!-- 适配菜单表格 --> <!-- 适配菜单表格 -->
<template v-if="column.key === 'meta'"> <template v-if="column.key === 'icon'">
<component :is="$loadIconCpn(record.meta?.icon)"></component> <component :is="$loadIconCpn(record.icon)"></component>
</template> </template>
<template v-if="column.key === 'type'"> <template v-if="column.key === 'type'">
{{ menuType[record.type] }} {{ menuType[record.type] }}

View File

@ -1,5 +1,4 @@
import { createRouter, createWebHistory } from 'vue-router' import { createRouter, createWebHistory } from 'vue-router'
import { message } from 'ant-design-vue'
import { userStore } from '@/stores/user' import { userStore } from '@/stores/user'
const routes = [ const routes = [
@ -21,6 +20,10 @@ const routes = [
{ {
path: '/:pathMatch(.*)*', path: '/:pathMatch(.*)*',
component: () => import('@/views/error/404.vue') component: () => import('@/views/error/404.vue')
},
{
path: '/back',
component: () => import('@/views/error/back.vue')
} }
] ]
@ -31,18 +34,17 @@ const router = createRouter({
// 导航守卫 // 导航守卫
router.beforeEach((to) => { router.beforeEach((to) => {
// 修改页面标题
if (to.meta.title) {
document.title = to.meta.title
}
if (to.path !== '/login') { if (to.path !== '/login') {
if (userStore().token) { if (userStore().token) {
return return
} }
message.warning('请登录')
return '/login' return '/login'
} }
}) })
router.afterEach((next) => {
// 修改页面标题
document.title = next.name || 'Mini RBAC'
})
export default router export default router

View File

@ -1,3 +1,5 @@
import { message } from 'ant-design-vue'
import { formatTime } from './format' import { formatTime } from './format'
import { loadIconCpn } from './loadCpn' import { loadIconCpn } from './loadCpn'
@ -5,3 +7,10 @@ export const registerFilter = (app) => {
app.config.globalProperties.$formatTime = (value) => formatTime(value) app.config.globalProperties.$formatTime = (value) => formatTime(value)
app.config.globalProperties.$loadIconCpn = (value) => loadIconCpn(value) app.config.globalProperties.$loadIconCpn = (value) => loadIconCpn(value)
} }
// 响应msg
export const messageTip = (res) => {
if (res.code === 200) {
message.success(res.msg)
}
}

View File

@ -0,0 +1,14 @@
<script setup>
import { useRouter } from 'vue-router'
const router = useRouter()
router.replace({
path: '/main'
})
</script>
<template>
<div></div>
</template>
<style scoped></style>

View File

@ -2,7 +2,7 @@
import { ref } from 'vue' import { ref } from 'vue'
import SiderMenu from '@/components/layout/sider-menu.vue' import SiderMenu from '@/components/layout/sider-menu.vue'
import LayoutHeader from '@/components/layout/layout-header.vue' import Header from '@/components/layout/header.vue'
// a-ayout-sider // a-ayout-sider
const collapsed = ref(false) const collapsed = ref(false)
@ -23,7 +23,7 @@ const changeSiderFold = (subValue) => {
<a-layout> <a-layout>
<a-layout-header style="background: #fff; padding: 0"> <a-layout-header style="background: #fff; padding: 0">
<!-- 页头 --> <!-- 页头 -->
<LayoutHeader @changeFold="changeSiderFold" /> <Header @changeFold="changeSiderFold" />
</a-layout-header> </a-layout-header>
<a-layout-content <a-layout-content
class="content" class="content"

View File

@ -1,3 +1,4 @@
import * as icons from '@ant-design/icons-vue'
export const columns = [ export const columns = [
{ {
title: '名称', title: '名称',
@ -7,8 +8,8 @@ export const columns = [
}, },
{ {
title: '图标', title: '图标',
dataIndex: 'meta', dataIndex: 'icon',
key: 'meta', key: 'icon',
width: 60 width: 60
}, },
{ {
@ -65,3 +66,53 @@ export const columns = [
width: 120 width: 120
} }
] ]
// 菜单类型映射
export const menuType = {
0: '目录',
1: '菜单',
2: '按钮',
3: '数据'
}
// 请求方法颜色映射
export const methodColor = {
GET: '#61AFFE',
POST: '#49CC90',
DELETE: '#F93E3E',
PUT: '#FCA130'
}
const nullOption = {
label: null,
value: null
}
// 转换成select 需要的options
export const menuTypeMap = () => {
return Object.keys(menuType).map((k) => ({ label: menuType[k], value: parseInt(k) }))
}
export const methodMap = () => {
let arr = Object.keys(methodColor).map((k) => ({ label: k, value: k }))
arr.unshift(nullOption)
return arr
}
export const iconMap = () => {
let arr = Object.keys(icons)
.filter((k) => k.indexOf('Outlined') !== -1)
.map((k) => ({ label: k, value: k }))
arr.unshift(nullOption)
return arr
}
export const rules = {
name: [
{ required: true, message: '请输入名称', trigger: 'blur' },
{ min: 3, max: 12, message: '3-12', trigger: 'blur' }
],
path: [
{ required: true, message: '请输入路由', trigger: 'blur' },
{ min: 1, max: 20, message: '1~20', trigger: 'blur' }
]
}

View File

@ -1,5 +1,10 @@
<script setup> <script setup>
import { reactive, watch, toRefs } from 'vue'
import useModal from '@/hooks/useModal' import useModal from '@/hooks/useModal'
import { menuTypeMap, methodMap, iconMap, rules } from './conf'
import { getMenus, addMenu, putMenu } from '@/service/menu'
import { userStore } from '@/stores/user'
import { messageTip } from '@/utils'
const props = defineProps({ const props = defineProps({
modalTitle: { modalTitle: {
@ -15,19 +20,74 @@ const props = defineProps({
const { showModal, updateId, formRef } = useModal() const { showModal, updateId, formRef } = useModal()
//
const data = reactive({
//
menuForm: {
name: '',
icon: null,
path: null,
type: 0,
component: null,
pid: 0,
identifier: null,
api: null,
method: null
},
//
menusOptions: []
})
//
const filterTreeNode = (inputValue, treeNode) => {
return treeNode.name.toLowerCase().indexOf(inputValue.toLowerCase()) >= 0
}
//
const filterOption = (input, option) => {
if (option.value) {
return option.value.toLowerCase().indexOf(input.toLowerCase()) >= 0
}
}
watch(showModal, async (newValue) => {
if (newValue) {
const res = await getMenus()
data.menusOptions = res.data
data.menusOptions.unshift({
id: 0,
name: '顶层菜单'
})
}
})
// //
const openModal = (record) => { const openModal = (record) => {
showModal.value = true showModal.value = true
updateId.value = record.id updateId.value = record.id
data.menuForm = record
} }
const onOk = () => { const onOk = () => {
// //
console.log(props) formRef.value.validateFields().then(async () => {
let res
if (props.modalType === 'create') {
res = await addMenu(data.menuForm)
} else {
res = await putMenu(updateId.value, data.menuForm)
}
messageTip(res)
formRef.value.resetFields()
showModal.value = !showModal.value
userStore().isPush = true
})
} }
const onCancel = () => {} const onCancel = () => {
formRef.value.resetFields()
}
const { menuForm, menusOptions } = toRefs(data)
defineExpose({ openModal, showModal }) defineExpose({ openModal, showModal })
</script> </script>
@ -42,13 +102,87 @@ defineExpose({ openModal, showModal })
@ok="onOk" @ok="onOk"
@cancel="onCancel" @cancel="onCancel"
> >
<a-form ref="formRef"> <a-form ref="formRef" :model="menuForm" class="form" :rules="rules">
<a-form-item> <a-form-item name="pid" label="上级菜单" class="item">
<a-input></a-input> <a-tree-select
v-model:value="menuForm.pid"
show-search
style="width: 100%"
:dropdown-style="{ maxHeight: '400px', overflow: 'auto' }"
allow-clear
:tree-data="menusOptions"
:field-names="{
children: 'children',
label: 'name',
value: 'id'
}"
:filterTreeNode="filterTreeNode"
></a-tree-select>
</a-form-item>
<a-form-item name="name" label="名称">
<a-input v-model:value="menuForm.name" />
</a-form-item>
<a-form-item name="icon" label="图标">
<a-select
v-model:value="menuForm.icon"
style="width: 100%"
show-search
:filterOption="filterOption"
>
<template v-for="option in iconMap()" :key="option.value">
<a-select-option :value="option.value">
<component :is="$loadIconCpn(option.label)"></component>
{{ option.label }}
</a-select-option>
</template>
</a-select>
</a-form-item>
<a-form-item name="path" label="路由">
<a-input v-model:value="menuForm.path" />
</a-form-item>
<a-form-item name="type" label="类型">
<a-select
v-model:value="menuForm.type"
style="width: 100%"
:options="menuTypeMap()"
></a-select>
</a-form-item>
<a-form-item name="component" label="组件">
<a-input v-model:value="menuForm.component" placeholder="views/main" />
</a-form-item>
<a-form-item name="identifier" label="权限">
<a-input v-model:value="menuForm.identifier" />
</a-form-item>
<a-form-item name="api" label="接口">
<a-input v-model:value="menuForm.api" />
</a-form-item>
<a-form-item name="method" label="方法">
<a-select
v-model:value="menuForm.method"
style="width: 100%"
:options="methodMap()"
></a-select>
</a-form-item> </a-form-item>
</a-form> </a-form>
</a-modal> </a-modal>
</div> </div>
</template> </template>
<style scoped></style> <style scoped>
.ant-form {
display: flex;
justify-content: space-between; /* 横向中间自动空间 */
align-content: space-between; /* 竖向中间自动空间 */
flex-wrap: wrap; /* 换行 */
}
.ant-form-item:nth-child(0) {
width: 100%;
}
.ant-form-item {
width: 48%;
}
.item {
width: 100%;
}
</style>

View File

@ -1,16 +1,33 @@
<script setup> <script setup>
import { ref, reactive, toRefs } from 'vue' import { ref, reactive, toRefs, onMounted } from 'vue'
import { columns } from './conf' import { columns } from './conf'
import { getMenus } from '@/service/menu' import { delMenu, getMenus } from '@/service/menu'
import Table from '@/components/table/table.vue' import Table from '@/components/table/table.vue'
import MenuModal from './menu-modal.vue' import MenuModal from './menu-modal.vue'
import { userStore } from '@/stores/user'
import { messageTip } from '@/utils'
const store = userStore()
store.$subscribe((mutation, state) => {
if (state.isPush) {
getPageData()
state.isPush = false
}
})
// //
const dataSource = ref([]) const dataSource = ref([])
getMenus().then((res) => (dataSource.value = res.data)) function getPageData() {
getMenus().then((res) => (dataSource.value = res.data))
}
onMounted(() => {
getPageData()
})
const modalRef = ref() const modalRef = ref()
const modalConf = reactive({ const modalConf = reactive({
@ -24,16 +41,16 @@ const addClick = () => {
modalRef.value.showModal = true modalRef.value.showModal = true
} }
//
const putClick = (record) => { const putClick = (record) => {
console.log(record)
modalConf.title = '编辑菜单' modalConf.title = '编辑菜单'
modalConf.type = 'create' modalConf.type = 'update'
modalRef.value.openModal(record) modalRef.value.openModal(record)
} }
const delClick = (record) => { const delClick = async (record) => {
console.log('点击', record) const res = await delMenu(record.id)
messageTip(res)
getPageData()
} }
const { title, type } = toRefs(modalConf) const { title, type } = toRefs(modalConf)

View File

@ -1,7 +1,5 @@
<script setup> <script setup>
import { ref, reactive, watch } from 'vue' import { ref, reactive, watch } from 'vue'
import { message } from 'ant-design-vue'
import { rules, treeFieldNames } from './conf' import { rules, treeFieldNames } from './conf'
import { addRole, putRole } from '@/service/role' import { addRole, putRole } from '@/service/role'
import { getMenus as getRoleMenu } from '@/service/user' import { getMenus as getRoleMenu } from '@/service/user'
@ -9,6 +7,7 @@ import { getMenus } from '@/service/menu'
import { userStore } from '@/stores/user' import { userStore } from '@/stores/user'
import useModal from '@/hooks/useModal' import useModal from '@/hooks/useModal'
import { messageTip } from '@/utils'
const props = defineProps({ const props = defineProps({
modalTitle: { modalTitle: {
@ -98,9 +97,9 @@ const onOk = () => {
} else { } else {
res = await putRole(updateId.value, roleForm) res = await putRole(updateId.value, roleForm)
} }
message.success(res.msg) messageTip(res)
resetData() resetData()
showModal.value = false showModal.value = !showModal.value
userStore().isPush = true userStore().isPush = true
}) })
} }

View File

@ -8,6 +8,7 @@ import RoleSearch from './role-search.vue'
import RoleModal from './role-modal.vue' import RoleModal from './role-modal.vue'
import { userStore } from '@/stores/user' import { userStore } from '@/stores/user'
import { messageTip } from '@/utils'
const store = userStore() const store = userStore()
@ -75,8 +76,9 @@ const resetQueryForm = () => {
} }
// //
const delClick = (record) => { const delClick = async (record) => {
delRole(record.id) const res = await delRole(record.id)
messageTip(res)
getPageData() getPageData()
} }

View File

@ -1,12 +1,13 @@
<script setup> <script setup>
import { reactive, ref, watch } from 'vue' import { reactive, ref, watch } from 'vue'
import { message } from 'ant-design-vue'
import { addUserRules, putUserRules } from './conf' import { addUserRules, putUserRules } from './conf'
import { addUser, putUser, getUserInfo } from '@/service/user' import { addUser, putUser, getUserInfo } from '@/service/user'
import { userStore } from '@/stores/user' import { userStore } from '@/stores/user'
import { getRoles } from '@/service/role' import { getRoles } from '@/service/role'
import useModal from '@/hooks/useModal' import useModal from '@/hooks/useModal'
import { messageTip } from '@/utils'
import { message } from 'ant-design-vue'
const props = defineProps({ const props = defineProps({
modalTitle: { modalTitle: {
@ -67,6 +68,7 @@ const openModal = async (record) => {
const onOk = () => { const onOk = () => {
formRef.value.validateFields().then(async () => { formRef.value.validateFields().then(async () => {
let res let res
let flag = false
if (props.modalType === 'create') { if (props.modalType === 'create') {
newUserForm.roles = newUserForm.roles.map((e, i) => ({ rid: e, status: i === 0 ? 5 : 1 })) newUserForm.roles = newUserForm.roles.map((e, i) => ({ rid: e, status: i === 0 ? 5 : 1 }))
res = await addUser(newUserForm) res = await addUser(newUserForm)
@ -79,13 +81,14 @@ const onOk = () => {
roles: rids roles: rids
}) })
if (updateId.value === store.userInfo.id) { if (updateId.value === store.userInfo.id) {
// message.warning('修改登录用户信息,重新登录生效.')
if (rids[0]['rid'] !== store.userInfo.roles[0]['id']) { flag = true
store.getUserData(updateId.value)
}
} }
} }
message.success(res.msg) if (!flag) {
messageTip(res)
}
formRef.value.resetFields() formRef.value.resetFields()
showModal.value = !showModal.value showModal.value = !showModal.value
store.isPush = true store.isPush = true

View File

@ -9,7 +9,7 @@ import { columns } from './conf'
import UserSearch from './user-search.vue' import UserSearch from './user-search.vue'
import UserModal from './user-modal.vue' import UserModal from './user-modal.vue'
import { userStore } from '@/stores/user' import { userStore } from '@/stores/user'
import { message } from 'ant-design-vue' import { messageTip } from '@/utils'
const store = userStore() const store = userStore()
@ -85,7 +85,7 @@ const resetQueryForm = () => {
// //
const delClick = async (record) => { const delClick = async (record) => {
const res = await delUser(record.id) const res = await delUser(record.id)
message.success(res.msg) messageTip(res)
getPageData() getPageData()
} }
@ -120,7 +120,8 @@ const putClick = async (record) => {
@create-click="addClick" @create-click="addClick"
@update-click="putClick" @update-click="putClick"
@delete-click="delClick" @delete-click="delClick"
/> >
</Table>
<!-- 新增&编辑 --> <!-- 新增&编辑 -->
<UserModal ref="modalRef" :modal-title="modalConf.title" :modal-type="modalConf.type" /> <UserModal ref="modalRef" :modal-title="modalConf.title" :modal-type="modalConf.type" />

View File

@ -0,0 +1,7 @@
<script setup></script>
<template>
<div>debug123123</div>
</template>
<style scoped></style>