mirror of https://github.com/veypi/OneAuth.git
feat: 添加角色管理模块(API + UI)
parent
4101daeed3
commit
691f1df75b
@ -0,0 +1,43 @@
|
||||
package role
|
||||
|
||||
import (
|
||||
"github.com/veypi/vbase/cfg"
|
||||
"github.com/veypi/vbase/models"
|
||||
"github.com/veypi/vigo"
|
||||
)
|
||||
|
||||
type CreateReq struct {
|
||||
Code string `json:"code" src:"json" desc:"Role Code"`
|
||||
Name string `json:"name" src:"json" desc:"Role Name"`
|
||||
Description string `json:"description" src:"json" desc:"Role Description"`
|
||||
OrgID string `json:"org_id" src:"json" desc:"Organization ID (Optional)"`
|
||||
}
|
||||
|
||||
func create(x *vigo.X, req *CreateReq) (*models.Role, error) {
|
||||
// Check if role code already exists
|
||||
var count int64
|
||||
if err := cfg.DB().Model(&models.Role{}).Where("code = ?", req.Code).Count(&count).Error; err != nil {
|
||||
return nil, vigo.ErrInternalServer.WithError(err)
|
||||
}
|
||||
if count > 0 {
|
||||
return nil, vigo.ErrAlreadyExists.WithArgs("Role Code")
|
||||
}
|
||||
|
||||
role := &models.Role{
|
||||
Code: req.Code,
|
||||
Name: req.Name,
|
||||
Description: req.Description,
|
||||
IsSystem: false, // Default to false for user created roles
|
||||
Status: 1,
|
||||
}
|
||||
|
||||
if req.OrgID != "" {
|
||||
role.OrgID = &req.OrgID
|
||||
}
|
||||
|
||||
if err := cfg.DB().Create(role).Error; err != nil {
|
||||
return nil, vigo.ErrDatabase.WithError(err)
|
||||
}
|
||||
|
||||
return role, nil
|
||||
}
|
||||
@ -0,0 +1,28 @@
|
||||
package role
|
||||
|
||||
import (
|
||||
"github.com/veypi/vbase/cfg"
|
||||
"github.com/veypi/vbase/models"
|
||||
"github.com/veypi/vigo"
|
||||
)
|
||||
|
||||
type DelReq struct {
|
||||
ID string `src:"path@id" desc:"Role ID"`
|
||||
}
|
||||
|
||||
func del(x *vigo.X, req *DelReq) error {
|
||||
var role models.Role
|
||||
if err := cfg.DB().First(&role, "id = ?", req.ID).Error; err != nil {
|
||||
return vigo.ErrNotFound
|
||||
}
|
||||
|
||||
if role.IsSystem {
|
||||
return vigo.NewError("cannot delete system role").WithCode(40300)
|
||||
}
|
||||
|
||||
if err := cfg.DB().Delete(&role).Error; err != nil {
|
||||
return vigo.ErrDatabase.WithError(err)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
@ -0,0 +1,19 @@
|
||||
package role
|
||||
|
||||
import (
|
||||
"github.com/veypi/vbase/cfg"
|
||||
"github.com/veypi/vbase/models"
|
||||
"github.com/veypi/vigo"
|
||||
)
|
||||
|
||||
type GetReq struct {
|
||||
ID string `src:"path@id" desc:"Role ID"`
|
||||
}
|
||||
|
||||
func get(x *vigo.X, req *GetReq) (*models.Role, error) {
|
||||
var role models.Role
|
||||
if err := cfg.DB().First(&role, "id = ?", req.ID).Error; err != nil {
|
||||
return nil, vigo.ErrNotFound
|
||||
}
|
||||
return &role, nil
|
||||
}
|
||||
@ -0,0 +1,18 @@
|
||||
package role
|
||||
|
||||
import (
|
||||
"github.com/veypi/vbase/auth"
|
||||
"github.com/veypi/vigo"
|
||||
)
|
||||
|
||||
var Router = vigo.NewRouter()
|
||||
|
||||
func init() {
|
||||
Router.Get("/", "List Roles", auth.VBaseAuth.Perm("role:read"), list)
|
||||
Router.Get("/{id}", "Get Role Detail", auth.VBaseAuth.Perm("role:read"), get)
|
||||
Router.Post("/", "Create Role", auth.VBaseAuth.Perm("role:create"), create)
|
||||
Router.Patch("/{id}", "Update Role", auth.VBaseAuth.Perm("role:update"), patch)
|
||||
Router.Delete("/{id}", "Delete Role", auth.VBaseAuth.Perm("role:delete"), del)
|
||||
Router.Get("/{id}/permissions", "Get Role Permissions", auth.VBaseAuth.Perm("role:read"), getPermissions)
|
||||
Router.Put("/{id}/permissions", "Update Role Permissions", auth.VBaseAuth.Perm("role:update"), updatePermissions)
|
||||
}
|
||||
@ -0,0 +1,58 @@
|
||||
package role
|
||||
|
||||
import (
|
||||
"github.com/veypi/vbase/cfg"
|
||||
"github.com/veypi/vbase/models"
|
||||
"github.com/veypi/vigo"
|
||||
)
|
||||
|
||||
type ListReq struct {
|
||||
Page int `json:"page" src:"query" default:"1"`
|
||||
PageSize int `json:"page_size" src:"query" default:"20"`
|
||||
OrgID *string `json:"org_id" src:"query" desc:"Organization ID"`
|
||||
Keyword *string `json:"keyword" src:"query" desc:"Search Keyword"`
|
||||
}
|
||||
|
||||
type ListResp struct {
|
||||
Items []models.Role `json:"items"`
|
||||
Total int64 `json:"total"`
|
||||
Page int `json:"page"`
|
||||
PageSize int `json:"page_size"`
|
||||
TotalPages int `json:"total_pages"`
|
||||
}
|
||||
|
||||
func list(x *vigo.X, req *ListReq) (*ListResp, error) {
|
||||
db := cfg.DB().Model(&models.Role{})
|
||||
|
||||
if req.OrgID != nil {
|
||||
db = db.Where("org_id = ?", *req.OrgID)
|
||||
}
|
||||
|
||||
if req.Keyword != nil && *req.Keyword != "" {
|
||||
db = db.Where("name LIKE ? OR code LIKE ?", "%"+*req.Keyword+"%", "%"+*req.Keyword+"%")
|
||||
}
|
||||
|
||||
var total int64
|
||||
if err := db.Count(&total).Error; err != nil {
|
||||
return nil, vigo.ErrInternalServer.WithError(err)
|
||||
}
|
||||
|
||||
var roles []models.Role
|
||||
offset := (req.Page - 1) * req.PageSize
|
||||
if err := db.Order("created_at DESC").Offset(offset).Limit(req.PageSize).Find(&roles).Error; err != nil {
|
||||
return nil, vigo.ErrInternalServer.WithError(err)
|
||||
}
|
||||
|
||||
totalPages := int(total) / req.PageSize
|
||||
if int(total)%req.PageSize > 0 {
|
||||
totalPages++
|
||||
}
|
||||
|
||||
return &ListResp{
|
||||
Items: roles,
|
||||
Total: total,
|
||||
Page: req.Page,
|
||||
PageSize: req.PageSize,
|
||||
TotalPages: totalPages,
|
||||
}, nil
|
||||
}
|
||||
@ -0,0 +1,44 @@
|
||||
package role
|
||||
|
||||
import (
|
||||
"github.com/veypi/vbase/cfg"
|
||||
"github.com/veypi/vbase/models"
|
||||
"github.com/veypi/vigo"
|
||||
)
|
||||
|
||||
type PatchReq struct {
|
||||
ID string `src:"path@id" desc:"Role ID"`
|
||||
Name *string `json:"name" src:"json" desc:"Role Name"`
|
||||
Description *string `json:"description" src:"json" desc:"Role Description"`
|
||||
Status *int `json:"status" src:"json" desc:"Status"`
|
||||
}
|
||||
|
||||
func patch(x *vigo.X, req *PatchReq) (*models.Role, error) {
|
||||
var role models.Role
|
||||
if err := cfg.DB().First(&role, "id = ?", req.ID).Error; err != nil {
|
||||
return nil, vigo.ErrNotFound
|
||||
}
|
||||
|
||||
if role.IsSystem {
|
||||
return nil, vigo.NewError("cannot modify system role").WithCode(40300)
|
||||
}
|
||||
|
||||
updates := map[string]interface{}{}
|
||||
if req.Name != nil {
|
||||
updates["name"] = *req.Name
|
||||
}
|
||||
if req.Description != nil {
|
||||
updates["description"] = *req.Description
|
||||
}
|
||||
if req.Status != nil {
|
||||
updates["status"] = *req.Status
|
||||
}
|
||||
|
||||
if len(updates) > 0 {
|
||||
if err := cfg.DB().Model(&role).Updates(updates).Error; err != nil {
|
||||
return nil, vigo.ErrDatabase.WithError(err)
|
||||
}
|
||||
}
|
||||
|
||||
return &role, nil
|
||||
}
|
||||
@ -0,0 +1,64 @@
|
||||
package role
|
||||
|
||||
import (
|
||||
"github.com/veypi/vbase/cfg"
|
||||
"github.com/veypi/vbase/models"
|
||||
"github.com/veypi/vigo"
|
||||
"gorm.io/gorm"
|
||||
)
|
||||
|
||||
type GetPermissionsReq struct {
|
||||
RoleID string `src:"path@id" desc:"Role ID"`
|
||||
}
|
||||
|
||||
func getPermissions(x *vigo.X, req *GetPermissionsReq) ([]models.Permission, error) {
|
||||
var rolePermissions []models.RolePermission
|
||||
if err := cfg.DB().Preload("Permission").Where("role_id = ?", req.RoleID).Find(&rolePermissions).Error; err != nil {
|
||||
return nil, vigo.ErrDatabase.WithError(err)
|
||||
}
|
||||
|
||||
permissions := make([]models.Permission, 0, len(rolePermissions))
|
||||
for _, rp := range rolePermissions {
|
||||
permissions = append(permissions, rp.Permission)
|
||||
}
|
||||
return permissions, nil
|
||||
}
|
||||
|
||||
type UpdatePermissionsReq struct {
|
||||
RoleID string `src:"path@id" desc:"Role ID"`
|
||||
PermissionIDs []string `json:"permission_ids" src:"json" desc:"List of Permission IDs"`
|
||||
}
|
||||
|
||||
func updatePermissions(x *vigo.X, req *UpdatePermissionsReq) error {
|
||||
var role models.Role
|
||||
if err := cfg.DB().First(&role, "id = ?", req.RoleID).Error; err != nil {
|
||||
return vigo.ErrNotFound
|
||||
}
|
||||
|
||||
if role.IsSystem {
|
||||
return vigo.NewError("cannot modify permissions of system role").WithCode(40300)
|
||||
}
|
||||
|
||||
return cfg.DB().Transaction(func(tx *gorm.DB) error {
|
||||
// Delete existing permissions
|
||||
if err := tx.Where("role_id = ?", req.RoleID).Delete(&models.RolePermission{}).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// Add new permissions
|
||||
if len(req.PermissionIDs) > 0 {
|
||||
rolePermissions := make([]models.RolePermission, 0, len(req.PermissionIDs))
|
||||
for _, pid := range req.PermissionIDs {
|
||||
rolePermissions = append(rolePermissions, models.RolePermission{
|
||||
RoleID: req.RoleID,
|
||||
PermissionID: pid,
|
||||
Condition: "none", // Default condition
|
||||
})
|
||||
}
|
||||
if err := tx.Create(&rolePermissions).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
})
|
||||
}
|
||||
@ -0,0 +1,66 @@
|
||||
package user
|
||||
|
||||
import (
|
||||
"github.com/veypi/vbase/cfg"
|
||||
"github.com/veypi/vbase/models"
|
||||
"github.com/veypi/vigo"
|
||||
"gorm.io/gorm"
|
||||
)
|
||||
|
||||
// User specific permissions (data-level or direct assignment)
|
||||
|
||||
type GetPermissionsReq struct {
|
||||
UserID string `src:"path@user_id" desc:"User ID"`
|
||||
}
|
||||
|
||||
func getPermissions(x *vigo.X, req *GetPermissionsReq) ([]models.UserPermission, error) {
|
||||
var userPermissions []models.UserPermission
|
||||
if err := cfg.DB().Where("user_id = ?", req.UserID).Find(&userPermissions).Error; err != nil {
|
||||
return nil, vigo.ErrDatabase.WithError(err)
|
||||
}
|
||||
return userPermissions, nil
|
||||
}
|
||||
|
||||
type UpdatePermissionsReq struct {
|
||||
UserID string `src:"path@user_id" desc:"User ID"`
|
||||
Permissions []struct {
|
||||
PermissionID string `json:"permission_id"`
|
||||
ResourceID string `json:"resource_id"`
|
||||
} `json:"permissions" src:"json" desc:"List of User Permissions"`
|
||||
}
|
||||
|
||||
func updatePermissions(x *vigo.X, req *UpdatePermissionsReq) error {
|
||||
var user models.User
|
||||
if err := cfg.DB().First(&user, "id = ?", req.UserID).Error; err != nil {
|
||||
return vigo.ErrNotFound
|
||||
}
|
||||
|
||||
grantor := ""
|
||||
if uid := x.Get("user_id"); uid != nil {
|
||||
if s, ok := uid.(string); ok {
|
||||
grantor = s
|
||||
}
|
||||
}
|
||||
|
||||
return cfg.DB().Transaction(func(tx *gorm.DB) error {
|
||||
if err := tx.Where("user_id = ?", req.UserID).Delete(&models.UserPermission{}).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if len(req.Permissions) > 0 {
|
||||
userPermissions := make([]models.UserPermission, 0, len(req.Permissions))
|
||||
for _, p := range req.Permissions {
|
||||
userPermissions = append(userPermissions, models.UserPermission{
|
||||
UserID: req.UserID,
|
||||
PermissionID: p.PermissionID,
|
||||
ResourceID: p.ResourceID,
|
||||
GrantedBy: grantor,
|
||||
})
|
||||
}
|
||||
if err := tx.Create(&userPermissions).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
})
|
||||
}
|
||||
@ -0,0 +1,57 @@
|
||||
package user
|
||||
|
||||
import (
|
||||
"github.com/veypi/vbase/cfg"
|
||||
"github.com/veypi/vbase/models"
|
||||
"github.com/veypi/vigo"
|
||||
"gorm.io/gorm"
|
||||
)
|
||||
|
||||
type GetRolesReq struct {
|
||||
UserID string `src:"path@user_id" desc:"User ID"`
|
||||
}
|
||||
|
||||
func getRoles(x *vigo.X, req *GetRolesReq) ([]models.Role, error) {
|
||||
var userRoles []models.UserRole
|
||||
if err := cfg.DB().Preload("Role").Where("user_id = ?", req.UserID).Find(&userRoles).Error; err != nil {
|
||||
return nil, vigo.ErrDatabase.WithError(err)
|
||||
}
|
||||
|
||||
roles := make([]models.Role, 0, len(userRoles))
|
||||
for _, ur := range userRoles {
|
||||
roles = append(roles, ur.Role)
|
||||
}
|
||||
return roles, nil
|
||||
}
|
||||
|
||||
type UpdateRolesReq struct {
|
||||
UserID string `src:"path@user_id" desc:"User ID"`
|
||||
RoleIDs []string `json:"role_ids" src:"json" desc:"Role IDs"`
|
||||
}
|
||||
|
||||
func updateRoles(x *vigo.X, req *UpdateRolesReq) error {
|
||||
var user models.User
|
||||
if err := cfg.DB().First(&user, "id = ?", req.UserID).Error; err != nil {
|
||||
return vigo.ErrNotFound
|
||||
}
|
||||
|
||||
return cfg.DB().Transaction(func(tx *gorm.DB) error {
|
||||
if err := tx.Where("user_id = ?", req.UserID).Delete(&models.UserRole{}).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if len(req.RoleIDs) > 0 {
|
||||
userRoles := make([]models.UserRole, 0, len(req.RoleIDs))
|
||||
for _, rid := range req.RoleIDs {
|
||||
userRoles = append(userRoles, models.UserRole{
|
||||
UserID: req.UserID,
|
||||
RoleID: rid,
|
||||
})
|
||||
}
|
||||
if err := tx.Create(&userRoles).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
})
|
||||
}
|
||||
@ -0,0 +1,286 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
|
||||
<head>
|
||||
<meta name="description" content="Role Management">
|
||||
<title>{{ $t('nav.roles') }}</title>
|
||||
<style>
|
||||
body {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: var(--spacing-lg);
|
||||
padding: var(--spacing-lg);
|
||||
box-sizing: border-box;
|
||||
background-color: var(--bg-color);
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
flex-wrap: wrap;
|
||||
gap: var(--spacing-md);
|
||||
padding-bottom: var(--spacing-md);
|
||||
border-bottom: 1px solid var(--border-color);
|
||||
}
|
||||
|
||||
.page-title {
|
||||
font-size: var(--font-size-2xl);
|
||||
font-weight: var(--font-weight-bold);
|
||||
color: var(--text-color);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: var(--spacing-sm);
|
||||
}
|
||||
|
||||
.search-box {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: var(--spacing-sm);
|
||||
background: var(--bg-color-secondary);
|
||||
padding: 0 var(--spacing-md);
|
||||
border-radius: var(--radius-full);
|
||||
border: 1px solid var(--border-color);
|
||||
min-width: 320px;
|
||||
height: 40px;
|
||||
transition: all var(--transition-base);
|
||||
}
|
||||
|
||||
.search-box:focus-within {
|
||||
border-color: var(--color-primary);
|
||||
box-shadow: 0 0 0 3px color-mix(in srgb, var(--color-primary), transparent 85%);
|
||||
}
|
||||
|
||||
.search-box input {
|
||||
border: none;
|
||||
background: transparent;
|
||||
outline: none;
|
||||
font-size: var(--font-size-md);
|
||||
color: var(--text-color);
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
}
|
||||
|
||||
.table-container {
|
||||
flex: 1;
|
||||
overflow: auto;
|
||||
background: var(--bg-color-secondary);
|
||||
border-radius: var(--radius-lg);
|
||||
box-shadow: var(--shadow-sm);
|
||||
border: 1px solid var(--border-color);
|
||||
}
|
||||
|
||||
table {
|
||||
width: 100%;
|
||||
border-collapse: collapse;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
th,
|
||||
td {
|
||||
text-align: left;
|
||||
padding: var(--spacing-md);
|
||||
border-bottom: 1px solid var(--border-color);
|
||||
}
|
||||
|
||||
th {
|
||||
background-color: var(--bg-color-tertiary);
|
||||
font-weight: 600;
|
||||
position: sticky;
|
||||
top: 0;
|
||||
z-index: 10;
|
||||
}
|
||||
|
||||
tr:last-child td {
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
tr:hover td {
|
||||
background-color: var(--bg-color-tertiary);
|
||||
}
|
||||
|
||||
.status-badge {
|
||||
display: inline-block;
|
||||
padding: 2px 8px;
|
||||
border-radius: var(--radius-full);
|
||||
font-size: var(--font-size-xs);
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.status-active {
|
||||
background-color: color-mix(in srgb, var(--color-success), transparent 85%);
|
||||
color: var(--color-success);
|
||||
}
|
||||
|
||||
.status-inactive {
|
||||
background-color: color-mix(in srgb, var(--text-color-disabled), transparent 85%);
|
||||
color: var(--text-color-secondary);
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<div class="page-header">
|
||||
<div class="page-title">
|
||||
<i class="fas fa-user-tag" style="color: var(--color-primary);"></i>
|
||||
{{ $t('nav.roles') }}
|
||||
</div>
|
||||
<div style="display: flex; gap: var(--spacing-md); align-items: center;">
|
||||
<div class="search-box">
|
||||
<i class="fas fa-search" style="color: var(--text-color-tertiary);"></i>
|
||||
<input type="text" v:value="searchQuery" :placeholder="$t('role.search_placeholder')">
|
||||
</div>
|
||||
<v-btn color="primary" :click="openCreateModal">
|
||||
<i class="fas fa-plus"></i>
|
||||
{{ $t('common.create') }}
|
||||
</v-btn>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="table-container">
|
||||
<table>
|
||||
<thead>
|
||||
<tr>
|
||||
<th>ID</th>
|
||||
<th>{{ $t('role.code') }}</th>
|
||||
<th>{{ $t('role.name') }}</th>
|
||||
<th>{{ $t('role.description') }}</th>
|
||||
<th>System</th>
|
||||
<th>Status</th>
|
||||
<th>{{ $t('common.actions') }}</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<tr v-for="r in filteredRoles">
|
||||
<td>{{ r.id }}</td>
|
||||
<td>{{ r.code }}</td>
|
||||
<td>{{ r.name }}</td>
|
||||
<td>{{ r.description }}</td>
|
||||
<td>
|
||||
<span class="status-badge" :class="r.is_system ? 'status-active' : 'status-inactive'">
|
||||
{{ r.is_system ? 'Yes' : 'No' }}
|
||||
</span>
|
||||
</td>
|
||||
<td>
|
||||
<span class="status-badge" :class="r.status === 1 ? 'status-active' : 'status-inactive'">
|
||||
{{ r.status === 1 ? 'Active' : 'Inactive' }}
|
||||
</span>
|
||||
</td>
|
||||
<td>
|
||||
<div style="display: flex; gap: 8px;">
|
||||
<v-btn icon size="sm" variant="outline" :click="() => openEditModal(r)" :title="$t('common.edit')" :disabled="r.is_system">
|
||||
<i class="fas fa-edit"></i>
|
||||
</v-btn>
|
||||
<v-btn icon size="sm" color="danger" variant="outline" :click="() => deleteRole(r)" :title="$t('common.delete')" :disabled="r.is_system">
|
||||
<i class="fas fa-trash"></i>
|
||||
</v-btn>
|
||||
</div>
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
|
||||
<!-- Create/Edit Dialog -->
|
||||
<v-dialog v:visible="showModal"
|
||||
:title="isEdit ? $t('role.edit') : $t('role.create')">
|
||||
<form @submit.prevent="saveRole" style="display: grid; gap: 16px;">
|
||||
<v-input :label="$t('role.code')" required v:value="formData.code" :disabled="isEdit" placeholder="e.g. admin"></v-input>
|
||||
<v-input :label="$t('role.name')" required v:value="formData.name" placeholder="e.g. Administrator"></v-input>
|
||||
<v-input :label="$t('role.description')" v:value="formData.description" placeholder="Description..."></v-input>
|
||||
</form>
|
||||
<div vslot="footer">
|
||||
<v-btn variant="outline" :click="closeModal">{{ $t('common.cancel') }}</v-btn>
|
||||
<v-btn color="primary" :click="saveRole">
|
||||
{{ isEdit ? $t('common.save') : $t('common.create') }}
|
||||
</v-btn>
|
||||
</div>
|
||||
</v-dialog>
|
||||
</body>
|
||||
<script setup>
|
||||
roles = [];
|
||||
searchQuery = "";
|
||||
showModal = false;
|
||||
isEdit = false;
|
||||
formData = {
|
||||
id: null,
|
||||
code: "",
|
||||
name: "",
|
||||
description: ""
|
||||
};
|
||||
|
||||
loadRoles = async () => {
|
||||
try {
|
||||
const res = await $axios.get('/api/roles');
|
||||
roles = res.items || [];
|
||||
} catch (e) {
|
||||
$message.error(e.message);
|
||||
}
|
||||
};
|
||||
|
||||
filteredRoles = () => {
|
||||
if (!searchQuery) return roles;
|
||||
const query = searchQuery.toLowerCase();
|
||||
return roles.filter(r =>
|
||||
r.name.toLowerCase().includes(query) ||
|
||||
r.code.toLowerCase().includes(query) ||
|
||||
(r.description && r.description.toLowerCase().includes(query))
|
||||
);
|
||||
};
|
||||
|
||||
openCreateModal = () => {
|
||||
isEdit = false;
|
||||
formData = { id: null, code: "", name: "", description: "" };
|
||||
showModal = true;
|
||||
};
|
||||
|
||||
openEditModal = (r) => {
|
||||
isEdit = true;
|
||||
formData = { ...r };
|
||||
showModal = true;
|
||||
};
|
||||
|
||||
closeModal = () => {
|
||||
showModal = false;
|
||||
};
|
||||
|
||||
saveRole = async () => {
|
||||
if (!formData.code || !formData.name) {
|
||||
$message.error($t('org.required_fields')); // Reusing existing message or add new one
|
||||
return;
|
||||
}
|
||||
try {
|
||||
if (isEdit) {
|
||||
const payload = {
|
||||
name: formData.name,
|
||||
description: formData.description
|
||||
};
|
||||
await $axios.patch(`/api/roles/${formData.id}`, payload);
|
||||
$message.success($t('org.updated')); // Reusing
|
||||
} else {
|
||||
await $axios.post('/api/roles', formData);
|
||||
$message.success($t('org.created')); // Reusing
|
||||
}
|
||||
closeModal();
|
||||
loadRoles();
|
||||
} catch (e) {
|
||||
$message.error(e.message);
|
||||
}
|
||||
};
|
||||
|
||||
deleteRole = async (r) => {
|
||||
try {
|
||||
await $message.confirm($t('role.delete_confirm'));
|
||||
await $axios.delete(`/api/roles/${r.id}`);
|
||||
$message.success($t('org.deleted')); // Reusing
|
||||
loadRoles();
|
||||
} catch (e) {
|
||||
// Cancelled
|
||||
}
|
||||
};
|
||||
</script>
|
||||
<script>
|
||||
$data.loadRoles();
|
||||
</script>
|
||||
|
||||
</html>
|
||||
Loading…
Reference in New Issue