xto-fronted 0.1.1 → 0.1.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (67) hide show
  1. package/.env.development +3 -4
  2. package/.env.production +3 -4
  3. package/bin/cli.js +104 -0
  4. package/dist/{403-MQkNUulz.js → 403-DM5wfQkM.js} +6 -6
  5. package/dist/{404-BOFYLq4X.js → 404-BurAu5LC.js} +7 -7
  6. package/dist/api/auth.d.ts +9 -8
  7. package/dist/api/menu.d.ts +3 -0
  8. package/dist/api/user.d.ts +2 -12
  9. package/dist/composables/index.d.ts +8 -0
  10. package/dist/composables/useApp.d.ts +64 -0
  11. package/dist/composables/useAuth.d.ts +19 -4
  12. package/dist/composables/useMenu.d.ts +34 -0
  13. package/dist/config/index.d.ts +11 -0
  14. package/dist/index-BNiEld34.js +15 -0
  15. package/dist/index-Be9RiEfo.js +98 -0
  16. package/dist/index-BqRv1bdN.js +1185 -0
  17. package/dist/index-CQLVXvNJ.js +15 -0
  18. package/dist/index-CyiE8n2V.js +15 -0
  19. package/dist/index-xauR1bOL.js +15 -0
  20. package/dist/index.d.ts +7 -4
  21. package/dist/index.es.js +50 -66
  22. package/dist/index.umd.js +1 -1
  23. package/dist/stores/auth.d.ts +60 -23
  24. package/dist/stores/menu.d.ts +40 -29
  25. package/dist/stores/user.d.ts +63 -84
  26. package/dist/style.css +1 -1
  27. package/dist/utils/auth.d.ts +15 -7
  28. package/dist/utils/permission.d.ts +1 -6
  29. package/dist/views/system/menu/index.vue.d.ts +1 -3
  30. package/dist/views/system/role/index.vue.d.ts +1 -3
  31. package/dist/views/system/user/index.vue.d.ts +1 -3
  32. package/package.json +27 -19
  33. package/src/api/auth.ts +34 -25
  34. package/src/api/menu.ts +13 -0
  35. package/src/api/user.ts +11 -45
  36. package/src/components/Layout/Header.vue +334 -389
  37. package/src/components/Layout/Sidebar.vue +212 -296
  38. package/src/components/Layout/Tabs.vue +19 -133
  39. package/src/composables/index.ts +9 -0
  40. package/src/composables/useApp.ts +170 -0
  41. package/src/composables/useAuth.ts +69 -44
  42. package/src/composables/useMenu.ts +141 -0
  43. package/src/config/index.ts +19 -0
  44. package/src/directives/permission.ts +40 -37
  45. package/src/index.ts +9 -4
  46. package/src/router/index.ts +70 -80
  47. package/src/stores/auth.ts +44 -31
  48. package/src/stores/menu.ts +157 -79
  49. package/src/stores/user.ts +40 -72
  50. package/src/types/api.d.ts +102 -83
  51. package/src/types/xto.d.ts +148 -148
  52. package/src/utils/auth.ts +85 -61
  53. package/src/utils/permission.ts +29 -41
  54. package/src/utils/request.ts +125 -125
  55. package/src/utils/storage.ts +10 -1
  56. package/src/views/dashboard/index.vue +31 -283
  57. package/src/views/login/index.vue +140 -247
  58. package/src/views/system/menu/index.vue +31 -380
  59. package/src/views/system/role/index.vue +31 -303
  60. package/src/views/system/user/index.vue +31 -326
  61. package/vite.config.ts +3 -3
  62. package/dist/index-BJxYdNPy.js +0 -475
  63. package/dist/index-BvnIIBR1.js +0 -142
  64. package/dist/index-CEvAq6KE.js +0 -372
  65. package/dist/index-DPkqej__.js +0 -345
  66. package/dist/index-pq9Z5K62.js +0 -184
  67. package/dist/index-vVfjShJR.js +0 -1183
@@ -1,8 +1,7 @@
1
- import { AppRoute } from '../types/router';
2
- import { MenuItem } from '../types/api';
1
+ import { MenuItem, RemoteMenuItem } from '../types/api';
3
2
  export declare const useMenuStore: import('pinia').StoreDefinition<"menu", Pick<{
4
3
  menuList: import('vue').Ref<{
5
- id: number | string;
4
+ code: string;
6
5
  name: string;
7
6
  path: string;
8
7
  component?: string | undefined;
@@ -12,11 +11,12 @@ export declare const useMenuStore: import('pinia').StoreDefinition<"menu", Pick<
12
11
  hidden?: boolean | undefined;
13
12
  keepAlive?: boolean | undefined;
14
13
  affix?: boolean | undefined;
15
- order?: number | undefined;
16
- parentId?: number | string | null | undefined;
14
+ default?: boolean | undefined;
15
+ out?: boolean | undefined;
16
+ closable?: boolean | undefined;
17
17
  children?: /*elided*/ any[] | undefined;
18
18
  }[], MenuItem[] | {
19
- id: number | string;
19
+ code: string;
20
20
  name: string;
21
21
  path: string;
22
22
  component?: string | undefined;
@@ -26,18 +26,21 @@ export declare const useMenuStore: import('pinia').StoreDefinition<"menu", Pick<
26
26
  hidden?: boolean | undefined;
27
27
  keepAlive?: boolean | undefined;
28
28
  affix?: boolean | undefined;
29
- order?: number | undefined;
30
- parentId?: number | string | null | undefined;
29
+ default?: boolean | undefined;
30
+ out?: boolean | undefined;
31
+ closable?: boolean | undefined;
31
32
  children?: /*elided*/ any[] | undefined;
32
33
  }[]>;
34
+ menuBtnListMap: import('vue').Ref<Record<string, MenuItem[]>, Record<string, MenuItem[]>>;
33
35
  hasMenu: import('vue').ComputedRef<boolean>;
36
+ index: import('vue').ComputedRef<string>;
37
+ indexMenu: MenuItem;
34
38
  setMenuList: (menus: MenuItem[]) => void;
39
+ setMenuFromRemote: (remoteMenus: RemoteMenuItem[]) => void;
35
40
  clearMenu: () => void;
36
- generateRoutes: (menus: MenuItem[]) => AppRoute[];
37
- addRoutes: (menus: MenuItem[]) => void;
38
- }, "menuList">, Pick<{
41
+ }, "menuList" | "menuBtnListMap" | "indexMenu">, Pick<{
39
42
  menuList: import('vue').Ref<{
40
- id: number | string;
43
+ code: string;
41
44
  name: string;
42
45
  path: string;
43
46
  component?: string | undefined;
@@ -47,11 +50,12 @@ export declare const useMenuStore: import('pinia').StoreDefinition<"menu", Pick<
47
50
  hidden?: boolean | undefined;
48
51
  keepAlive?: boolean | undefined;
49
52
  affix?: boolean | undefined;
50
- order?: number | undefined;
51
- parentId?: number | string | null | undefined;
53
+ default?: boolean | undefined;
54
+ out?: boolean | undefined;
55
+ closable?: boolean | undefined;
52
56
  children?: /*elided*/ any[] | undefined;
53
57
  }[], MenuItem[] | {
54
- id: number | string;
58
+ code: string;
55
59
  name: string;
56
60
  path: string;
57
61
  component?: string | undefined;
@@ -61,18 +65,21 @@ export declare const useMenuStore: import('pinia').StoreDefinition<"menu", Pick<
61
65
  hidden?: boolean | undefined;
62
66
  keepAlive?: boolean | undefined;
63
67
  affix?: boolean | undefined;
64
- order?: number | undefined;
65
- parentId?: number | string | null | undefined;
68
+ default?: boolean | undefined;
69
+ out?: boolean | undefined;
70
+ closable?: boolean | undefined;
66
71
  children?: /*elided*/ any[] | undefined;
67
72
  }[]>;
73
+ menuBtnListMap: import('vue').Ref<Record<string, MenuItem[]>, Record<string, MenuItem[]>>;
68
74
  hasMenu: import('vue').ComputedRef<boolean>;
75
+ index: import('vue').ComputedRef<string>;
76
+ indexMenu: MenuItem;
69
77
  setMenuList: (menus: MenuItem[]) => void;
78
+ setMenuFromRemote: (remoteMenus: RemoteMenuItem[]) => void;
70
79
  clearMenu: () => void;
71
- generateRoutes: (menus: MenuItem[]) => AppRoute[];
72
- addRoutes: (menus: MenuItem[]) => void;
73
- }, "hasMenu">, Pick<{
80
+ }, "hasMenu" | "index">, Pick<{
74
81
  menuList: import('vue').Ref<{
75
- id: number | string;
82
+ code: string;
76
83
  name: string;
77
84
  path: string;
78
85
  component?: string | undefined;
@@ -82,11 +89,12 @@ export declare const useMenuStore: import('pinia').StoreDefinition<"menu", Pick<
82
89
  hidden?: boolean | undefined;
83
90
  keepAlive?: boolean | undefined;
84
91
  affix?: boolean | undefined;
85
- order?: number | undefined;
86
- parentId?: number | string | null | undefined;
92
+ default?: boolean | undefined;
93
+ out?: boolean | undefined;
94
+ closable?: boolean | undefined;
87
95
  children?: /*elided*/ any[] | undefined;
88
96
  }[], MenuItem[] | {
89
- id: number | string;
97
+ code: string;
90
98
  name: string;
91
99
  path: string;
92
100
  component?: string | undefined;
@@ -96,13 +104,16 @@ export declare const useMenuStore: import('pinia').StoreDefinition<"menu", Pick<
96
104
  hidden?: boolean | undefined;
97
105
  keepAlive?: boolean | undefined;
98
106
  affix?: boolean | undefined;
99
- order?: number | undefined;
100
- parentId?: number | string | null | undefined;
107
+ default?: boolean | undefined;
108
+ out?: boolean | undefined;
109
+ closable?: boolean | undefined;
101
110
  children?: /*elided*/ any[] | undefined;
102
111
  }[]>;
112
+ menuBtnListMap: import('vue').Ref<Record<string, MenuItem[]>, Record<string, MenuItem[]>>;
103
113
  hasMenu: import('vue').ComputedRef<boolean>;
114
+ index: import('vue').ComputedRef<string>;
115
+ indexMenu: MenuItem;
104
116
  setMenuList: (menus: MenuItem[]) => void;
117
+ setMenuFromRemote: (remoteMenus: RemoteMenuItem[]) => void;
105
118
  clearMenu: () => void;
106
- generateRoutes: (menus: MenuItem[]) => AppRoute[];
107
- addRoutes: (menus: MenuItem[]) => void;
108
- }, "setMenuList" | "clearMenu" | "generateRoutes" | "addRoutes">>;
119
+ }, "setMenuList" | "setMenuFromRemote" | "clearMenu">>;
@@ -1,113 +1,92 @@
1
1
  import { UserInfo } from '../types/api';
2
2
  export declare const useUserStore: import('pinia').StoreDefinition<"user", Pick<{
3
3
  userInfo: import('vue').Ref<{
4
- id: number | string;
5
- username: string;
6
- nickname: string;
7
- avatar?: string | undefined;
4
+ uId: string;
5
+ appId: string;
6
+ userId: string;
7
+ userName: string;
8
+ departmentName?: string | undefined;
8
9
  email?: string | undefined;
9
- phone?: string | undefined;
10
- status: number;
11
- roles: string[];
12
- permissions: string[];
13
- createTime?: string | undefined;
14
- updateTime?: string | undefined;
15
- } | null, UserInfo | {
16
- id: number | string;
17
- username: string;
18
- nickname: string;
10
+ mobilePhone?: string | undefined;
11
+ positionName?: string | undefined;
19
12
  avatar?: string | undefined;
13
+ workNo?: string | undefined;
14
+ } | null, UserInfo | {
15
+ uId: string;
16
+ appId: string;
17
+ userId: string;
18
+ userName: string;
19
+ departmentName?: string | undefined;
20
20
  email?: string | undefined;
21
- phone?: string | undefined;
22
- status: number;
23
- roles: string[];
24
- permissions: string[];
25
- createTime?: string | undefined;
26
- updateTime?: string | undefined;
21
+ mobilePhone?: string | undefined;
22
+ positionName?: string | undefined;
23
+ avatar?: string | undefined;
24
+ workNo?: string | undefined;
27
25
  } | null>;
28
- roles: import('vue').Ref<string[], string[]>;
29
- permissions: import('vue').Ref<string[], string[]>;
30
26
  isLoggedIn: import('vue').ComputedRef<boolean>;
31
- username: import('vue').ComputedRef<string>;
32
- nickname: import('vue').ComputedRef<string>;
27
+ userName: import('vue').ComputedRef<string>;
33
28
  avatar: import('vue').ComputedRef<string>;
34
- userId: import('vue').ComputedRef<string | number | undefined>;
29
+ userId: import('vue').ComputedRef<string | undefined>;
35
30
  setUserInfo: (info: UserInfo) => void;
36
31
  clearUserInfo: () => void;
37
- hasPermission: (permission: string | string[]) => boolean;
38
- hasRole: (role: string | string[]) => boolean;
39
- }, "userInfo" | "roles" | "permissions">, Pick<{
32
+ }, "userInfo">, Pick<{
40
33
  userInfo: import('vue').Ref<{
41
- id: number | string;
42
- username: string;
43
- nickname: string;
44
- avatar?: string | undefined;
34
+ uId: string;
35
+ appId: string;
36
+ userId: string;
37
+ userName: string;
38
+ departmentName?: string | undefined;
45
39
  email?: string | undefined;
46
- phone?: string | undefined;
47
- status: number;
48
- roles: string[];
49
- permissions: string[];
50
- createTime?: string | undefined;
51
- updateTime?: string | undefined;
52
- } | null, UserInfo | {
53
- id: number | string;
54
- username: string;
55
- nickname: string;
40
+ mobilePhone?: string | undefined;
41
+ positionName?: string | undefined;
56
42
  avatar?: string | undefined;
43
+ workNo?: string | undefined;
44
+ } | null, UserInfo | {
45
+ uId: string;
46
+ appId: string;
47
+ userId: string;
48
+ userName: string;
49
+ departmentName?: string | undefined;
57
50
  email?: string | undefined;
58
- phone?: string | undefined;
59
- status: number;
60
- roles: string[];
61
- permissions: string[];
62
- createTime?: string | undefined;
63
- updateTime?: string | undefined;
51
+ mobilePhone?: string | undefined;
52
+ positionName?: string | undefined;
53
+ avatar?: string | undefined;
54
+ workNo?: string | undefined;
64
55
  } | null>;
65
- roles: import('vue').Ref<string[], string[]>;
66
- permissions: import('vue').Ref<string[], string[]>;
67
56
  isLoggedIn: import('vue').ComputedRef<boolean>;
68
- username: import('vue').ComputedRef<string>;
69
- nickname: import('vue').ComputedRef<string>;
57
+ userName: import('vue').ComputedRef<string>;
70
58
  avatar: import('vue').ComputedRef<string>;
71
- userId: import('vue').ComputedRef<string | number | undefined>;
59
+ userId: import('vue').ComputedRef<string | undefined>;
72
60
  setUserInfo: (info: UserInfo) => void;
73
61
  clearUserInfo: () => void;
74
- hasPermission: (permission: string | string[]) => boolean;
75
- hasRole: (role: string | string[]) => boolean;
76
- }, "isLoggedIn" | "username" | "nickname" | "avatar" | "userId">, Pick<{
62
+ }, "isLoggedIn" | "userId" | "userName" | "avatar">, Pick<{
77
63
  userInfo: import('vue').Ref<{
78
- id: number | string;
79
- username: string;
80
- nickname: string;
81
- avatar?: string | undefined;
64
+ uId: string;
65
+ appId: string;
66
+ userId: string;
67
+ userName: string;
68
+ departmentName?: string | undefined;
82
69
  email?: string | undefined;
83
- phone?: string | undefined;
84
- status: number;
85
- roles: string[];
86
- permissions: string[];
87
- createTime?: string | undefined;
88
- updateTime?: string | undefined;
89
- } | null, UserInfo | {
90
- id: number | string;
91
- username: string;
92
- nickname: string;
70
+ mobilePhone?: string | undefined;
71
+ positionName?: string | undefined;
93
72
  avatar?: string | undefined;
73
+ workNo?: string | undefined;
74
+ } | null, UserInfo | {
75
+ uId: string;
76
+ appId: string;
77
+ userId: string;
78
+ userName: string;
79
+ departmentName?: string | undefined;
94
80
  email?: string | undefined;
95
- phone?: string | undefined;
96
- status: number;
97
- roles: string[];
98
- permissions: string[];
99
- createTime?: string | undefined;
100
- updateTime?: string | undefined;
81
+ mobilePhone?: string | undefined;
82
+ positionName?: string | undefined;
83
+ avatar?: string | undefined;
84
+ workNo?: string | undefined;
101
85
  } | null>;
102
- roles: import('vue').Ref<string[], string[]>;
103
- permissions: import('vue').Ref<string[], string[]>;
104
86
  isLoggedIn: import('vue').ComputedRef<boolean>;
105
- username: import('vue').ComputedRef<string>;
106
- nickname: import('vue').ComputedRef<string>;
87
+ userName: import('vue').ComputedRef<string>;
107
88
  avatar: import('vue').ComputedRef<string>;
108
- userId: import('vue').ComputedRef<string | number | undefined>;
89
+ userId: import('vue').ComputedRef<string | undefined>;
109
90
  setUserInfo: (info: UserInfo) => void;
110
91
  clearUserInfo: () => void;
111
- hasPermission: (permission: string | string[]) => boolean;
112
- hasRole: (role: string | string[]) => boolean;
113
- }, "setUserInfo" | "clearUserInfo" | "hasPermission" | "hasRole">>;
92
+ }, "setUserInfo" | "clearUserInfo">>;
package/dist/style.css CHANGED
@@ -1 +1 @@
1
- .sidebar[data-v-69e6558d]{height:100%;display:flex;flex-direction:column;background-color:var(--bg-color)}.sidebar__logo[data-v-69e6558d]{height:50px;display:flex;align-items:center;justify-content:center;gap:10px;border-bottom:1px solid var(--color-border-lighter)}.sidebar__logo-img[data-v-69e6558d]{width:32px;height:32px}.sidebar__logo-text[data-v-69e6558d]{font-size:16px;font-weight:600;color:var(--color-primary)}.sidebar__search[data-v-69e6558d]{padding:10px;border-bottom:1px solid var(--color-border-lighter);position:relative}.sidebar__search-results[data-v-69e6558d]{position:absolute;top:100%;left:0;right:0;background-color:var(--bg-color);border:1px solid var(--color-border-lighter);border-radius:var(--border-radius-base);box-shadow:var(--box-shadow);max-height:300px;overflow-y:auto;z-index:100}.sidebar__search-item[data-v-69e6558d]{display:flex;align-items:center;gap:10px;padding:10px 12px;cursor:pointer;transition:background-color .2s}.sidebar__search-item[data-v-69e6558d]:hover{background-color:var(--color-fill)}.sidebar__search-item-info[data-v-69e6558d]{display:flex;flex-direction:column;gap:2px}.sidebar__search-item-title[data-v-69e6558d]{font-size:14px;color:var(--color-text-primary)}.sidebar__search-item-parent[data-v-69e6558d]{font-size:12px;color:var(--color-text-secondary)}.sidebar__menu[data-v-69e6558d]{flex:1;border-right:none;overflow-y:auto}.sidebar__user[data-v-69e6558d]{padding:10px;border-top:1px solid var(--color-border-lighter);display:flex;align-items:center;justify-content:space-between}.sidebar__user-info[data-v-69e6558d]{display:flex;flex-direction:column;gap:2px}.sidebar__user-name[data-v-69e6558d]{font-size:14px;font-weight:500}.sidebar__user-role[data-v-69e6558d]{font-size:12px;color:var(--color-text-secondary)}.menu-icon[data-v-69e6558d]{margin-right:8px}.header[data-v-a8667482]{display:flex;align-items:center;justify-content:space-between;padding:0 20px;height:100%}.header__left[data-v-a8667482]{display:flex;align-items:center;gap:15px}.header__collapse[data-v-a8667482]{width:24px;height:24px;display:flex;align-items:center;justify-content:center;cursor:pointer;font-size:18px;color:var(--color-text-regular)}.header__collapse[data-v-a8667482]:hover{color:var(--color-primary)}.header__breadcrumb[data-v-a8667482]{font-size:14px;color:var(--color-text-secondary)}.header__breadcrumb .is-current[data-v-a8667482]{color:var(--color-text-primary);font-weight:500}.header__right[data-v-a8667482]{display:flex;align-items:center;gap:15px}.header__action[data-v-a8667482]{width:32px;height:32px;display:flex;align-items:center;justify-content:center;cursor:pointer;border-radius:var(--border-radius-base);font-size:16px}.header__action[data-v-a8667482]:hover{background-color:var(--color-fill)}.header__user[data-v-a8667482]{position:relative}.header__user-trigger[data-v-a8667482]{display:flex;align-items:center;gap:8px;cursor:pointer;padding:4px 8px;border-radius:var(--border-radius-base);transition:background-color .2s}.header__user-trigger[data-v-a8667482]:hover{background-color:var(--color-fill)}.header__user-name[data-v-a8667482]{font-size:14px;color:var(--color-text-primary)}.header__user-arrow[data-v-a8667482]{font-size:10px;color:var(--color-text-secondary);transition:transform .2s}.header__user-arrow.is-active[data-v-a8667482]{transform:rotate(180deg)}.header__avatar[data-v-a8667482]{width:32px;height:32px;border-radius:50%;background:linear-gradient(135deg,var(--color-primary),var(--color-primary-light-3));display:flex;align-items:center;justify-content:center;color:#fff;font-size:14px;font-weight:500}.header__dropdown[data-v-a8667482]{position:absolute;top:calc(100% + 8px);right:0;min-width:200px;background-color:var(--bg-color);border-radius:var(--border-radius-base);box-shadow:var(--box-shadow);overflow:hidden;z-index:100}.header__dropdown-header[data-v-a8667482]{display:flex;align-items:center;gap:12px;padding:16px}.header__dropdown-avatar[data-v-a8667482]{width:40px;height:40px;border-radius:50%;background:linear-gradient(135deg,var(--color-primary),var(--color-primary-light-3));display:flex;align-items:center;justify-content:center;color:#fff;font-size:16px;font-weight:500}.header__dropdown-info[data-v-a8667482]{flex:1}.header__dropdown-name[data-v-a8667482]{font-size:14px;font-weight:500;color:var(--color-text-primary)}.header__dropdown-role[data-v-a8667482]{font-size:12px;color:var(--color-text-secondary);margin-top:2px}.header__dropdown-divider[data-v-a8667482]{height:1px;background-color:var(--color-border-lighter)}.header__dropdown-menu[data-v-a8667482]{padding:8px 0}.header__dropdown-item[data-v-a8667482]{display:flex;align-items:center;gap:10px;padding:10px 16px;cursor:pointer;font-size:14px;color:var(--color-text-regular);transition:all .2s}.header__dropdown-item[data-v-a8667482]:hover{background-color:var(--color-fill);color:var(--color-text-primary)}.header__dropdown-item--danger[data-v-a8667482]{color:var(--color-danger)}.header__dropdown-item--danger[data-v-a8667482]:hover{background-color:var(--color-danger-light);color:var(--color-danger)}.header__dropdown-icon[data-v-a8667482]{font-size:16px}.dropdown-enter-active[data-v-a8667482],.dropdown-leave-active[data-v-a8667482]{transition:all .2s ease}.dropdown-enter-from[data-v-a8667482],.dropdown-leave-to[data-v-a8667482]{opacity:0;transform:translateY(-10px)}.layout[data-v-1a24e6d0]{display:flex;width:100%;height:100%}.layout__aside[data-v-1a24e6d0]{transition:width .3s;overflow:hidden;flex-shrink:0;height:100%}.layout__main[data-v-1a24e6d0]{flex:1;display:flex;flex-direction:column;overflow:hidden;height:100%}.layout__header[data-v-1a24e6d0]{height:50px;background-color:var(--bg-color);border-bottom:1px solid var(--color-border-lighter);flex-shrink:0}.layout__content[data-v-1a24e6d0]{flex:1;overflow:auto;background-color:var(--bg-color-page)}.tabs-wrapper[data-v-bd3be691]{width:100%;height:100%;padding:0 10px}.tabs-wrapper[data-v-bd3be691] .t-tabs{height:100%}.tabs-wrapper[data-v-bd3be691] .t-tabs__header{margin:0;border-bottom:none}.tabs-wrapper[data-v-bd3be691] .t-tabs__nav{border:none}.tabs-wrapper[data-v-bd3be691] .t-tabs__item{height:32px;line-height:32px;border:1px solid var(--color-border-lighter);margin-right:5px;border-radius:var(--border-radius-base);padding:0 15px}.tabs-wrapper[data-v-bd3be691] .t-tabs__item.is-active{background-color:var(--color-primary-light-9);border-color:var(--color-primary)}.tab-label[data-v-bd3be691]{display:flex;align-items:center;gap:5px}.tab-close[data-v-bd3be691]{width:14px;height:14px;display:flex;align-items:center;justify-content:center;font-size:10px;border-radius:50%}.tab-close[data-v-bd3be691]:hover{background-color:var(--color-danger-light);color:var(--color-danger)}.footer[data-v-317ae311]{width:100%;text-align:center;font-size:12px;color:var(--color-text-secondary)}.login[data-v-eb51a732]{width:100%;min-height:100vh;display:flex;align-items:center;justify-content:flex-end;padding-right:15%;background:linear-gradient(135deg,var(--color-primary-light-9) 0%,var(--color-primary-light-7) 100%)}.login__container[data-v-eb51a732]{width:400px;padding:40px;background-color:var(--bg-color);border-radius:var(--border-radius-large);box-shadow:var(--box-shadow-dark)}.login__header[data-v-eb51a732]{text-align:center;margin-bottom:30px}.login__logo[data-v-eb51a732]{width:60px;height:60px}.login__title[data-v-eb51a732]{font-size:28px;font-weight:600;color:var(--color-primary);margin:15px 0 5px}.login__subtitle[data-v-eb51a732]{font-size:14px;color:var(--color-text-secondary)}.login__form[data-v-eb51a732] .t-form-item{margin-bottom:20px}.login__form[data-v-eb51a732] .x-input__prefix{margin-right:8px}.login__submit[data-v-eb51a732]{width:100%}.login__footer[data-v-eb51a732]{text-align:center;margin-top:20px;font-size:12px;color:var(--color-text-placeholder)}.error-page[data-v-f1a1acb0]{width:100%;height:100vh;display:flex;align-items:center;justify-content:center;background-color:var(--bg-color-page)}.error-page__content[data-v-f1a1acb0]{text-align:center}.error-page__code[data-v-f1a1acb0]{font-size:120px;font-weight:600;color:var(--color-primary);line-height:1;margin-bottom:20px}.error-page__title[data-v-f1a1acb0]{font-size:24px;font-weight:500;color:var(--color-text-primary);margin-bottom:10px}.error-page__desc[data-v-f1a1acb0]{font-size:14px;color:var(--color-text-secondary);margin-bottom:30px}.error-page[data-v-f6105e3d]{width:100%;height:100vh;display:flex;align-items:center;justify-content:center;background-color:var(--bg-color-page)}.error-page__content[data-v-f6105e3d]{text-align:center}.error-page__code[data-v-f6105e3d]{font-size:120px;font-weight:600;color:var(--color-warning);line-height:1;margin-bottom:20px}.error-page__title[data-v-f6105e3d]{font-size:24px;font-weight:500;color:var(--color-text-primary);margin-bottom:10px}.error-page__desc[data-v-f6105e3d]{font-size:14px;color:var(--color-text-secondary);margin-bottom:30px}.dashboard[data-v-4bb1b10a]{padding:20px}.dashboard__stats[data-v-4bb1b10a]{display:grid;grid-template-columns:repeat(4,1fr);gap:20px;margin-bottom:20px}@media (max-width: 1200px){.dashboard__stats[data-v-4bb1b10a]{grid-template-columns:repeat(2,1fr)}}@media (max-width: 768px){.dashboard__stats[data-v-4bb1b10a]{grid-template-columns:1fr}}.dashboard__main[data-v-4bb1b10a]{display:grid;grid-template-columns:repeat(2,1fr);gap:20px;margin-bottom:20px}@media (max-width: 992px){.dashboard__main[data-v-4bb1b10a]{grid-template-columns:1fr}}.dashboard__quick[data-v-4bb1b10a],.dashboard__activity[data-v-4bb1b10a]{min-height:300px}.card-title[data-v-4bb1b10a]{font-size:16px;font-weight:500}.stat-card__content[data-v-4bb1b10a]{display:flex;align-items:center;gap:15px}.stat-card__icon[data-v-4bb1b10a]{width:50px;height:50px;display:flex;align-items:center;justify-content:center;font-size:24px;border-radius:var(--border-radius-base)}.stat-card__info[data-v-4bb1b10a]{flex:1}.stat-card__title[data-v-4bb1b10a]{font-size:14px;color:var(--color-text-secondary);margin-bottom:5px}.stat-card__value[data-v-4bb1b10a]{font-size:24px;font-weight:600}.quick-links[data-v-4bb1b10a]{display:grid;grid-template-columns:repeat(4,1fr);gap:15px}@media (max-width: 768px){.quick-links[data-v-4bb1b10a]{grid-template-columns:repeat(2,1fr)}}.quick-link[data-v-4bb1b10a]{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:20px;border-radius:var(--border-radius-base);background-color:var(--color-fill);text-decoration:none;transition:all var(--transition-duration-fast)}.quick-link[data-v-4bb1b10a]:hover{background-color:var(--color-primary-light-9);transform:translateY(-2px)}.quick-link__icon[data-v-4bb1b10a]{font-size:28px;margin-bottom:10px}.quick-link__title[data-v-4bb1b10a]{font-size:14px;color:var(--color-text-primary)}.activity-list[data-v-4bb1b10a]{display:flex;flex-direction:column;gap:15px}.activity-item[data-v-4bb1b10a]{display:flex;align-items:center;gap:10px;padding-bottom:15px;border-bottom:1px solid var(--color-border-lighter)}.activity-item[data-v-4bb1b10a]:last-child{border-bottom:none;padding-bottom:0}.activity-item__action[data-v-4bb1b10a]{flex:1;font-size:14px;color:var(--color-text-regular)}.activity-item__time[data-v-4bb1b10a]{font-size:12px;color:var(--color-text-placeholder)}.system-info[data-v-4bb1b10a]{display:grid;grid-template-columns:repeat(2,1fr);gap:20px}@media (max-width: 768px){.system-info[data-v-4bb1b10a]{grid-template-columns:1fr}}.system-info__item[data-v-4bb1b10a]{display:flex;align-items:center;gap:10px}.system-info__label[data-v-4bb1b10a]{font-size:14px;color:var(--color-text-secondary);min-width:80px}.system-info__value[data-v-4bb1b10a]{font-size:14px;color:var(--color-text-primary)}.user-page[data-v-4ebcfe66]{padding:20px}.user-page .search-card[data-v-4ebcfe66]{margin-bottom:20px}.user-page .toolbar[data-v-4ebcfe66]{margin-bottom:15px}.data-table[data-v-4ebcfe66]{width:100%;border-collapse:collapse}.data-table th[data-v-4ebcfe66],.data-table td[data-v-4ebcfe66]{padding:12px;text-align:left;border-bottom:1px solid var(--color-border-lighter)}.data-table th[data-v-4ebcfe66]{font-weight:500;color:var(--color-text-regular);background-color:var(--color-fill-light)}.data-table .loading-cell[data-v-4ebcfe66],.data-table .empty-cell[data-v-4ebcfe66]{text-align:center;color:var(--color-text-secondary);padding:40px}.pagination-wrapper[data-v-4ebcfe66]{display:flex;justify-content:flex-end;margin-top:20px}.role-page[data-v-b714caf1]{padding:20px}.role-page .search-card[data-v-b714caf1]{margin-bottom:20px}.role-page .toolbar[data-v-b714caf1]{margin-bottom:15px}.data-table[data-v-b714caf1]{width:100%;border-collapse:collapse}.data-table th[data-v-b714caf1],.data-table td[data-v-b714caf1]{padding:12px;text-align:left;border-bottom:1px solid var(--color-border-lighter)}.data-table th[data-v-b714caf1]{font-weight:500;color:var(--color-text-regular);background-color:var(--color-fill-light)}.data-table .loading-cell[data-v-b714caf1],.data-table .empty-cell[data-v-b714caf1]{text-align:center;color:var(--color-text-secondary);padding:40px}.pagination-wrapper[data-v-b714caf1]{display:flex;justify-content:flex-end;margin-top:20px}.menu-page[data-v-14a9dcbc]{padding:20px}.menu-page .toolbar[data-v-14a9dcbc]{margin-bottom:15px}.tree-table[data-v-14a9dcbc]{width:100%;border-collapse:collapse}.tree-table th[data-v-14a9dcbc],.tree-table td[data-v-14a9dcbc]{padding:12px;text-align:left;border-bottom:1px solid var(--color-border-lighter)}.tree-table th[data-v-14a9dcbc]{font-weight:500;color:var(--color-text-regular);background-color:var(--color-fill-light)}.tree-table .tree-row--level-1 td[data-v-14a9dcbc]:first-child{padding-left:30px}.tree-indent[data-v-14a9dcbc]{display:inline-block;width:20px}.menu-name[data-v-14a9dcbc]{cursor:pointer;color:var(--color-primary)}.menu-name[data-v-14a9dcbc]:hover{text-decoration:underline}
1
+ .sidebar[data-v-bd43a7b8]{height:100%;display:flex;flex-direction:column;background-color:var(--bg-color)}.sidebar__logo[data-v-bd43a7b8]{height:50px;display:flex;align-items:center;justify-content:center;gap:10px;border-bottom:1px solid var(--color-border-lighter)}.sidebar__logo-img[data-v-bd43a7b8]{width:32px;height:32px}.sidebar__logo-text[data-v-bd43a7b8]{font-size:16px;font-weight:600;color:var(--color-primary)}.sidebar__search[data-v-bd43a7b8]{padding:10px;border-bottom:1px solid var(--color-border-lighter);position:relative}.sidebar__search-results[data-v-bd43a7b8]{position:absolute;top:100%;left:0;right:0;background-color:var(--bg-color);border:1px solid var(--color-border-lighter);border-radius:var(--border-radius-base);box-shadow:var(--box-shadow);max-height:300px;overflow-y:auto;z-index:100}.sidebar__search-item[data-v-bd43a7b8]{display:flex;align-items:center;gap:10px;padding:10px 12px;cursor:pointer;transition:background-color .2s}.sidebar__search-item[data-v-bd43a7b8]:hover{background-color:var(--color-fill)}.sidebar__search-item-info[data-v-bd43a7b8]{display:flex;flex-direction:column;gap:2px}.sidebar__search-item-title[data-v-bd43a7b8]{font-size:14px;color:var(--color-text-primary)}.sidebar__search-item-parent[data-v-bd43a7b8]{font-size:12px;color:var(--color-text-secondary)}.sidebar__menu[data-v-bd43a7b8]{flex:1;border-right:none;overflow-y:auto}.sidebar__user[data-v-bd43a7b8]{padding:10px;border-top:1px solid var(--color-border-lighter);display:flex;align-items:center;justify-content:space-between}.sidebar__user-info[data-v-bd43a7b8]{display:flex;flex-direction:column;gap:2px}.sidebar__user-name[data-v-bd43a7b8]{font-size:14px;font-weight:500}.sidebar__user-role[data-v-bd43a7b8]{font-size:12px;color:var(--color-text-secondary)}.menu-icon[data-v-bd43a7b8]{margin-right:8px}.header[data-v-2bf8f3d2]{display:flex;align-items:center;justify-content:space-between;padding:0 20px;height:100%}.header__left[data-v-2bf8f3d2]{display:flex;align-items:center;gap:15px}.header__collapse[data-v-2bf8f3d2]{width:24px;height:24px;display:flex;align-items:center;justify-content:center;cursor:pointer;font-size:18px;color:var(--color-text-regular)}.header__collapse[data-v-2bf8f3d2]:hover{color:var(--color-primary)}.header__right[data-v-2bf8f3d2]{display:flex;align-items:center;gap:15px}.header__action[data-v-2bf8f3d2]{width:32px;height:32px;display:flex;align-items:center;justify-content:center;cursor:pointer;border-radius:var(--border-radius-base);font-size:16px}.header__action[data-v-2bf8f3d2]:hover{background-color:var(--color-fill)}.header__user[data-v-2bf8f3d2]{position:relative}.header__user-trigger[data-v-2bf8f3d2]{display:flex;align-items:center;gap:8px;cursor:pointer;padding:4px 8px;border-radius:var(--border-radius-base);transition:background-color .2s}.header__user-trigger[data-v-2bf8f3d2]:hover{background-color:var(--color-fill)}.header__user-name[data-v-2bf8f3d2]{font-size:14px;color:var(--color-text-primary)}.header__user-arrow[data-v-2bf8f3d2]{font-size:10px;color:var(--color-text-secondary);transition:transform .2s}.header__user-arrow.is-active[data-v-2bf8f3d2]{transform:rotate(180deg)}.header__avatar[data-v-2bf8f3d2]{width:32px;height:32px;border-radius:50%;background:linear-gradient(135deg,var(--color-primary),var(--color-primary-light-3));display:flex;align-items:center;justify-content:center;color:#fff;font-size:14px;font-weight:500}.header__dropdown[data-v-2bf8f3d2]{position:absolute;top:calc(100% + 8px);right:0;min-width:200px;background-color:var(--bg-color);border-radius:var(--border-radius-base);box-shadow:var(--box-shadow);overflow:hidden;z-index:100}.header__dropdown-header[data-v-2bf8f3d2]{display:flex;align-items:center;gap:12px;padding:16px}.header__dropdown-avatar[data-v-2bf8f3d2]{width:40px;height:40px;border-radius:50%;background:linear-gradient(135deg,var(--color-primary),var(--color-primary-light-3));display:flex;align-items:center;justify-content:center;color:#fff;font-size:16px;font-weight:500}.header__dropdown-info[data-v-2bf8f3d2]{flex:1}.header__dropdown-name[data-v-2bf8f3d2]{font-size:14px;font-weight:500;color:var(--color-text-primary)}.header__dropdown-role[data-v-2bf8f3d2]{font-size:12px;color:var(--color-text-secondary);margin-top:2px}.header__dropdown-divider[data-v-2bf8f3d2]{height:1px;background-color:var(--color-border-lighter)}.header__dropdown-menu[data-v-2bf8f3d2]{padding:8px 0}.header__dropdown-item[data-v-2bf8f3d2]{display:flex;align-items:center;gap:10px;padding:10px 16px;cursor:pointer;font-size:14px;color:var(--color-text-regular);transition:all .2s}.header__dropdown-item[data-v-2bf8f3d2]:hover{background-color:var(--color-fill);color:var(--color-text-primary)}.header__dropdown-item--danger[data-v-2bf8f3d2]{color:var(--color-danger)}.header__dropdown-item--danger[data-v-2bf8f3d2]:hover{background-color:var(--color-danger-light);color:var(--color-danger)}.header__dropdown-icon[data-v-2bf8f3d2]{font-size:16px}.dropdown-enter-active[data-v-2bf8f3d2],.dropdown-leave-active[data-v-2bf8f3d2]{transition:all .2s ease}.dropdown-enter-from[data-v-2bf8f3d2],.dropdown-leave-to[data-v-2bf8f3d2]{opacity:0;transform:translateY(-10px)}.layout[data-v-1a24e6d0]{display:flex;width:100%;height:100%}.layout__aside[data-v-1a24e6d0]{transition:width .3s;overflow:hidden;flex-shrink:0;height:100%}.layout__main[data-v-1a24e6d0]{flex:1;display:flex;flex-direction:column;overflow:hidden;height:100%}.layout__header[data-v-1a24e6d0]{height:50px;background-color:var(--bg-color);border-bottom:1px solid var(--color-border-lighter);flex-shrink:0}.layout__content[data-v-1a24e6d0]{flex:1;overflow:auto;background-color:var(--bg-color-page)}.tabs-wrapper[data-v-3b1e382a]{width:100%;height:100%;display:flex;align-items:center;justify-content:center;color:var(--color-text-secondary);font-size:12px}.footer[data-v-317ae311]{width:100%;text-align:center;font-size:12px;color:var(--color-text-secondary)}.login[data-v-728dba9d]{width:100%;min-height:100vh;display:flex;align-items:center;justify-content:flex-end;padding-right:15%;background:linear-gradient(135deg,var(--color-primary-light-9) 0%,var(--color-primary-light-7) 100%)}.login__container[data-v-728dba9d]{width:400px;padding:40px;background-color:var(--bg-color);border-radius:var(--border-radius-large);box-shadow:var(--box-shadow-dark)}.login__header[data-v-728dba9d]{text-align:center;margin-bottom:30px}.login__logo[data-v-728dba9d]{width:60px;height:60px}.login__title[data-v-728dba9d]{font-size:28px;font-weight:600;color:var(--color-primary);margin:15px 0 5px}.login__subtitle[data-v-728dba9d]{font-size:14px;color:var(--color-text-secondary)}.login__form[data-v-728dba9d] .t-form-item{margin-bottom:20px}.login__submit[data-v-728dba9d]{width:100%}.error-page[data-v-f1a1acb0]{width:100%;height:100vh;display:flex;align-items:center;justify-content:center;background-color:var(--bg-color-page)}.error-page__content[data-v-f1a1acb0]{text-align:center}.error-page__code[data-v-f1a1acb0]{font-size:120px;font-weight:600;color:var(--color-primary);line-height:1;margin-bottom:20px}.error-page__title[data-v-f1a1acb0]{font-size:24px;font-weight:500;color:var(--color-text-primary);margin-bottom:10px}.error-page__desc[data-v-f1a1acb0]{font-size:14px;color:var(--color-text-secondary);margin-bottom:30px}.error-page[data-v-f6105e3d]{width:100%;height:100vh;display:flex;align-items:center;justify-content:center;background-color:var(--bg-color-page)}.error-page__content[data-v-f6105e3d]{text-align:center}.error-page__code[data-v-f6105e3d]{font-size:120px;font-weight:600;color:var(--color-warning);line-height:1;margin-bottom:20px}.error-page__title[data-v-f6105e3d]{font-size:24px;font-weight:500;color:var(--color-text-primary);margin-bottom:10px}.error-page__desc[data-v-f6105e3d]{font-size:14px;color:var(--color-text-secondary);margin-bottom:30px}.dashboard[data-v-a57037a8]{padding:20px;height:100%}.dashboard__placeholder[data-v-a57037a8]{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%;color:var(--color-text-secondary)}.dashboard__placeholder h2[data-v-a57037a8]{margin-bottom:10px;color:var(--color-text-primary)}.page[data-v-715bfcf7]{padding:20px;height:100%}.page__placeholder[data-v-715bfcf7]{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%;color:var(--color-text-secondary)}.page__placeholder h2[data-v-715bfcf7]{margin-bottom:10px;color:var(--color-text-primary)}.page[data-v-0802c3db]{padding:20px;height:100%}.page__placeholder[data-v-0802c3db]{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%;color:var(--color-text-secondary)}.page__placeholder h2[data-v-0802c3db]{margin-bottom:10px;color:var(--color-text-primary)}.page[data-v-7bc14131]{padding:20px;height:100%}.page__placeholder[data-v-7bc14131]{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%;color:var(--color-text-secondary)}.page__placeholder h2[data-v-7bc14131]{margin-bottom:10px;color:var(--color-text-primary)}
@@ -1,18 +1,26 @@
1
1
  /**
2
2
  * Token 管理
3
3
  */
4
+ export interface LoginInfo {
5
+ accessToken: string;
6
+ refreshToken: string;
7
+ expiresTime: string;
8
+ refreshTime: string;
9
+ tokenType: string;
10
+ code: string;
11
+ }
4
12
  export interface TokenInfo {
5
13
  token: string;
6
14
  refreshToken: string;
7
15
  expireTime: number;
8
16
  }
17
+ export declare const getLoginInfo: () => LoginInfo | null;
18
+ export declare const setLoginInfo: (data: Record<string, unknown>) => void;
9
19
  export declare const getToken: () => string | null;
10
- export declare const setToken: (token: string) => void;
11
- export declare const getRefreshToken: () => string | null;
12
- export declare const setRefreshToken: (refreshToken: string) => void;
13
- export declare const getTokenExpire: () => number | null;
14
- export declare const setTokenExpire: (expireTime: number) => void;
15
- export declare const setTokenInfo: (info: TokenInfo) => void;
20
+ export declare const getTokenType: () => string;
21
+ export declare const getCode: () => string | null;
16
22
  export declare const clearToken: () => void;
17
- export declare const isTokenExpired: () => boolean;
18
23
  export declare const hasToken: () => boolean;
24
+ export declare const setUserInfo: (userInfo: Record<string, unknown>) => void;
25
+ export declare const getUserInfo: () => Record<string, unknown> | null;
26
+ export declare const clearUserInfo: () => void;
@@ -2,15 +2,10 @@
2
2
  * 权限工具函数
3
3
  */
4
4
  /**
5
- * 检查是否有权限
5
+ * 检查是否有权限(基于菜单按钮权限)
6
6
  * @param permission 权限标识
7
7
  */
8
8
  export declare function hasPermission(permission: string | string[]): boolean;
9
- /**
10
- * 检查是否有角色
11
- * @param role 角色标识
12
- */
13
- export declare function hasRole(role: string | string[]): boolean;
14
9
  /**
15
10
  * 检查是否是管理员
16
11
  */
@@ -1,4 +1,2 @@
1
- declare const _default: import('vue').DefineComponent<{}, {}, {}, {}, {}, import('vue').ComponentOptionsMixin, import('vue').ComponentOptionsMixin, {}, string, import('vue').PublicProps, Readonly<{}> & Readonly<{}>, {}, {}, {}, {}, string, import('vue').ComponentProvideOptions, true, {
2
- formRef: any;
3
- }, HTMLDivElement>;
1
+ declare const _default: import('vue').DefineComponent<{}, {}, {}, {}, {}, import('vue').ComponentOptionsMixin, import('vue').ComponentOptionsMixin, {}, string, import('vue').PublicProps, Readonly<{}> & Readonly<{}>, {}, {}, {}, {}, string, import('vue').ComponentProvideOptions, true, {}, HTMLDivElement>;
4
2
  export default _default;
@@ -1,4 +1,2 @@
1
- declare const _default: import('vue').DefineComponent<{}, {}, {}, {}, {}, import('vue').ComponentOptionsMixin, import('vue').ComponentOptionsMixin, {}, string, import('vue').PublicProps, Readonly<{}> & Readonly<{}>, {}, {}, {}, {}, string, import('vue').ComponentProvideOptions, true, {
2
- formRef: any;
3
- }, HTMLDivElement>;
1
+ declare const _default: import('vue').DefineComponent<{}, {}, {}, {}, {}, import('vue').ComponentOptionsMixin, import('vue').ComponentOptionsMixin, {}, string, import('vue').PublicProps, Readonly<{}> & Readonly<{}>, {}, {}, {}, {}, string, import('vue').ComponentProvideOptions, true, {}, HTMLDivElement>;
4
2
  export default _default;
@@ -1,4 +1,2 @@
1
- declare const _default: import('vue').DefineComponent<{}, {}, {}, {}, {}, import('vue').ComponentOptionsMixin, import('vue').ComponentOptionsMixin, {}, string, import('vue').PublicProps, Readonly<{}> & Readonly<{}>, {}, {}, {}, {}, string, import('vue').ComponentProvideOptions, true, {
2
- formRef: any;
3
- }, HTMLDivElement>;
1
+ declare const _default: import('vue').DefineComponent<{}, {}, {}, {}, {}, import('vue').ComponentOptionsMixin, import('vue').ComponentOptionsMixin, {}, string, import('vue').PublicProps, Readonly<{}> & Readonly<{}>, {}, {}, {}, {}, string, import('vue').ComponentProvideOptions, true, {}, HTMLDivElement>;
4
2
  export default _default;
package/package.json CHANGED
@@ -1,9 +1,9 @@
1
1
  {
2
2
  "name": "xto-fronted",
3
- "version": "0.1.1",
3
+ "version": "0.1.3",
4
4
  "private": false,
5
5
  "type": "module",
6
- "description": "XTO 前端应用框架",
6
+ "description": "XTO 前端应用脚手架,封装登录、退出、菜单渲染等统一逻辑",
7
7
  "main": "dist/index.umd.js",
8
8
  "module": "dist/index.es.js",
9
9
  "types": "dist/index.d.ts",
@@ -15,50 +15,54 @@
15
15
  },
16
16
  "./style.css": "./dist/style.css"
17
17
  },
18
+ "bin": {
19
+ "xto-fronted": "./bin/cli.js"
20
+ },
18
21
  "files": [
19
22
  "dist",
20
23
  "src",
21
24
  "public",
25
+ "bin",
22
26
  "index.html",
23
27
  "vite.config.ts",
24
28
  "tsconfig.json",
25
29
  "tsconfig.node.json",
26
- ".env.development",
27
- ".env.production"
30
+ ".env.production",
31
+ ".env.development"
28
32
  ],
29
33
  "scripts": {
30
34
  "dev": "vite",
31
35
  "build": "vue-tsc && vite build",
32
- "build:lib": "vue-tsc && vite build --mode lib",
36
+ "build:lib": "vite build --mode lib",
33
37
  "preview": "vite preview",
34
- "lint": "eslint . --ext .vue,.js,.jsx,.cjs,.mjs,.ts,.tsx,.cts,.mts --fix"
38
+ "lint": "eslint . --ext .vue,.js,.jsx,.cjs,.mjs,.ts,.tsx,.cts,.mts --fix",
39
+ "prepublishOnly": "npm run build:lib"
35
40
  },
36
41
  "peerDependencies": {
42
+ "@xto/base": "^0.1.0",
43
+ "@xto/feedback": "^0.1.0",
44
+ "@xto/form": "^0.1.0",
45
+ "@xto/navigation": "^0.1.0",
37
46
  "axios": "^1.6.0",
38
47
  "pinia": "^2.1.0",
39
48
  "vue": "^3.4.0",
40
49
  "vue-router": "^4.3.0"
41
50
  },
42
- "dependencies": {
43
- "@xto/base": "^0.1.0",
44
- "@xto/business": "^0.1.0",
45
- "@xto/core": "^0.1.0",
46
- "@xto/data": "^0.1.0",
47
- "@xto/feedback": "^0.1.0",
48
- "@xto/form": "^0.1.0",
49
- "@xto/layout": "^0.1.0",
50
- "@xto/navigation": "^0.1.0"
51
- },
51
+ "dependencies": {},
52
52
  "devDependencies": {
53
53
  "@types/node": "^20.11.30",
54
54
  "@vitejs/plugin-vue": "^5.0.4",
55
+ "@xto/base": "^0.1.0",
56
+ "@xto/feedback": "^0.1.0",
57
+ "@xto/form": "^0.1.0",
58
+ "@xto/navigation": "^0.1.0",
55
59
  "axios": "^1.6.8",
56
60
  "pinia": "^2.1.7",
57
61
  "sass": "^1.72.0",
58
62
  "sass-embedded": "^1.72.0",
59
63
  "typescript": "^5.4.2",
60
64
  "vite": "^5.2.0",
61
- "vite-plugin-dts": "^4.5.4",
65
+ "vite-plugin-dts": "^3.9.1",
62
66
  "vue": "^3.4.21",
63
67
  "vue-router": "^4.3.0",
64
68
  "vue-tsc": "^2.0.6"
@@ -68,7 +72,11 @@
68
72
  "vue3",
69
73
  "admin",
70
74
  "template",
71
- "xto"
75
+ "xto",
76
+ "scaffold",
77
+ "login",
78
+ "menu",
79
+ "cli"
72
80
  ],
73
81
  "author": "",
74
82
  "license": "MIT",
@@ -76,4 +84,4 @@
76
84
  "type": "git",
77
85
  "url": ""
78
86
  }
79
- }
87
+ }
package/src/api/auth.ts CHANGED
@@ -1,26 +1,35 @@
1
- /**
2
- * 认证 API
3
- */
4
-
5
- import { http } from '@/utils/request'
6
- import type { LoginParams, LoginResult, UserInfo } from '@/types/api'
7
-
8
- // 登录
9
- export function login(data: LoginParams) {
10
- return http.post<LoginResult>('/auth/login', data)
11
- }
12
-
13
- // 登出
14
- export function logout() {
15
- return http.post('/auth/logout')
16
- }
17
-
18
- // 获取用户信息
19
- export function getUserInfo() {
20
- return http.get<UserInfo>('/user/info')
21
- }
22
-
23
- // 刷新 Token
24
- export function refreshToken(refreshToken: string) {
25
- return http.post<{ token: string; expireTime: number }>('/auth/refresh', { refreshToken })
1
+ /**
2
+ * 认证 API
3
+ */
4
+
5
+ import { http } from '@/utils/request'
6
+ import type { ApiResponse } from '@/utils/request'
7
+ import type { LoginResult } from '@/types/api'
8
+ import config from '@/config'
9
+
10
+ // 登录参数
11
+ export interface LoginParams {
12
+ uid: string
13
+ password: string
14
+ }
15
+
16
+ // 登录
17
+ export function login(data: LoginParams): Promise<ApiResponse<LoginResult>> {
18
+ return http.post<LoginResult>('/user/v1.0/login/by-domain', {
19
+ appId: config.appId,
20
+ clientId: config.clientId,
21
+ uid: data.uid,
22
+ password: data.password,
23
+ code: true
24
+ })
25
+ }
26
+
27
+ // 退出登录
28
+ export function logout(): Promise<ApiResponse<null>> {
29
+ return http.put<null>('/user/v1.0/user/logout')
30
+ }
31
+
32
+ // 根据 code 登录
33
+ export function loginByCode(appId: string, code: string): Promise<ApiResponse<LoginResult>> {
34
+ return http.get<LoginResult>(`/user/v1.0/login/by-code?appId=${appId}&code=${code}`)
26
35
  }
@@ -0,0 +1,13 @@
1
+ /**
2
+ * 菜单 API
3
+ */
4
+
5
+ import { http } from '@/utils/request'
6
+ import type { ApiResponse } from '@/utils/request'
7
+ import type { RemoteMenuItem } from '@/types/api'
8
+ import config from '@/config'
9
+
10
+ // 获取用户菜单
11
+ export function getUserMenu(): Promise<ApiResponse<RemoteMenuItem[]>> {
12
+ return http.get<RemoteMenuItem[]>(`/user/v1.0/menu/get-menu?appId=${config.appId}`)
13
+ }