adminforth 2.4.0-next.32 → 2.4.0-next.321

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 (177) hide show
  1. package/commands/callTsProxy.js +14 -4
  2. package/commands/createApp/templates/api.ts.hbs +10 -0
  3. package/commands/createApp/templates/custom/tsconfig.json.hbs +2 -3
  4. package/commands/createApp/templates/index.ts.hbs +12 -1
  5. package/commands/createApp/templates/package.json.hbs +1 -1
  6. package/commands/createApp/templates/prisma.config.ts.hbs +8 -0
  7. package/commands/createApp/templates/schema.prisma.hbs +0 -1
  8. package/commands/createApp/utils.js +10 -0
  9. package/commands/createCustomComponent/configLoader.js +17 -4
  10. package/commands/createCustomComponent/main.js +13 -7
  11. package/commands/createCustomComponent/templates/customCrud/beforeActionButtons.vue.hbs +38 -0
  12. package/commands/createCustomComponent/templates/customCrud/saveButton.vue.hbs +28 -0
  13. package/commands/createPlugin/templates/custom/tsconfig.json.hbs +2 -5
  14. package/commands/createPlugin/templates/package.json.hbs +1 -1
  15. package/commands/generateModels.js +30 -22
  16. package/dist/auth.d.ts +9 -1
  17. package/dist/auth.d.ts.map +1 -1
  18. package/dist/auth.js +21 -2
  19. package/dist/auth.js.map +1 -1
  20. package/dist/dataConnectors/baseConnector.d.ts +1 -1
  21. package/dist/dataConnectors/baseConnector.d.ts.map +1 -1
  22. package/dist/dataConnectors/baseConnector.js +70 -18
  23. package/dist/dataConnectors/baseConnector.js.map +1 -1
  24. package/dist/dataConnectors/clickhouse.d.ts.map +1 -1
  25. package/dist/dataConnectors/clickhouse.js +15 -0
  26. package/dist/dataConnectors/clickhouse.js.map +1 -1
  27. package/dist/dataConnectors/mongo.d.ts.map +1 -1
  28. package/dist/dataConnectors/mongo.js +50 -15
  29. package/dist/dataConnectors/mongo.js.map +1 -1
  30. package/dist/dataConnectors/mysql.d.ts.map +1 -1
  31. package/dist/dataConnectors/mysql.js +11 -0
  32. package/dist/dataConnectors/mysql.js.map +1 -1
  33. package/dist/dataConnectors/postgres.d.ts.map +1 -1
  34. package/dist/dataConnectors/postgres.js +43 -14
  35. package/dist/dataConnectors/postgres.js.map +1 -1
  36. package/dist/dataConnectors/sqlite.d.ts.map +1 -1
  37. package/dist/dataConnectors/sqlite.js +11 -0
  38. package/dist/dataConnectors/sqlite.js.map +1 -1
  39. package/dist/index.d.ts +11 -1
  40. package/dist/index.d.ts.map +1 -1
  41. package/dist/index.js +44 -21
  42. package/dist/index.js.map +1 -1
  43. package/dist/modules/codeInjector.d.ts +2 -0
  44. package/dist/modules/codeInjector.d.ts.map +1 -1
  45. package/dist/modules/codeInjector.js +62 -6
  46. package/dist/modules/codeInjector.js.map +1 -1
  47. package/dist/modules/configValidator.d.ts +6 -0
  48. package/dist/modules/configValidator.d.ts.map +1 -1
  49. package/dist/modules/configValidator.js +202 -25
  50. package/dist/modules/configValidator.js.map +1 -1
  51. package/dist/modules/restApi.d.ts +1 -1
  52. package/dist/modules/restApi.d.ts.map +1 -1
  53. package/dist/modules/restApi.js +184 -31
  54. package/dist/modules/restApi.js.map +1 -1
  55. package/dist/modules/styles.d.ts +499 -13
  56. package/dist/modules/styles.d.ts.map +1 -1
  57. package/dist/modules/styles.js +555 -31
  58. package/dist/modules/styles.js.map +1 -1
  59. package/dist/modules/utils.d.ts +7 -15
  60. package/dist/modules/utils.d.ts.map +1 -1
  61. package/dist/modules/utils.js +45 -68
  62. package/dist/modules/utils.js.map +1 -1
  63. package/dist/servers/express.d.ts +5 -0
  64. package/dist/servers/express.d.ts.map +1 -1
  65. package/dist/servers/express.js +40 -1
  66. package/dist/servers/express.js.map +1 -1
  67. package/dist/spa/index.html +1 -1
  68. package/dist/spa/package-lock.json +1208 -708
  69. package/dist/spa/package.json +34 -34
  70. package/dist/spa/src/App.vue +61 -173
  71. package/dist/spa/src/adminforth.ts +42 -18
  72. package/dist/spa/src/afcl/AreaChart.vue +0 -1
  73. package/dist/spa/src/afcl/BarChart.vue +2 -2
  74. package/dist/spa/src/afcl/Button.vue +6 -6
  75. package/dist/spa/src/afcl/ButtonGroup.vue +91 -0
  76. package/dist/spa/src/afcl/Card.vue +25 -0
  77. package/dist/spa/src/afcl/Checkbox.vue +21 -13
  78. package/dist/spa/src/afcl/CountryFlag.vue +4 -1
  79. package/dist/spa/src/{components/CustomDatePicker.vue → afcl/DatePicker.vue} +95 -9
  80. package/dist/spa/src/afcl/Dialog.vue +47 -27
  81. package/dist/spa/src/afcl/Dropzone.vue +145 -48
  82. package/dist/spa/src/afcl/Input.vue +14 -6
  83. package/dist/spa/src/afcl/JsonViewer.vue +25 -0
  84. package/dist/spa/src/afcl/LinkButton.vue +3 -3
  85. package/dist/spa/src/afcl/PieChart.vue +5 -5
  86. package/dist/spa/src/afcl/ProgressBar.vue +7 -7
  87. package/dist/spa/src/afcl/Select.vue +82 -34
  88. package/dist/spa/src/afcl/Skeleton.vue +6 -6
  89. package/dist/spa/src/afcl/Table.vue +313 -75
  90. package/dist/spa/src/afcl/Textarea.vue +31 -0
  91. package/dist/spa/src/afcl/Toggle.vue +32 -0
  92. package/dist/spa/src/afcl/Tooltip.vue +28 -18
  93. package/dist/spa/src/afcl/VerticalTabs.vue +21 -7
  94. package/dist/spa/src/afcl/index.ts +6 -3
  95. package/dist/spa/src/components/AcceptModal.vue +48 -14
  96. package/dist/spa/src/components/Breadcrumbs.vue +5 -5
  97. package/dist/spa/src/components/CallActionWrapper.vue +15 -0
  98. package/dist/spa/src/components/ColumnValueInput.vue +38 -18
  99. package/dist/spa/src/components/ColumnValueInputWrapper.vue +4 -3
  100. package/dist/spa/src/components/CustomDateRangePicker.vue +9 -8
  101. package/dist/spa/src/components/CustomRangePicker.vue +37 -21
  102. package/dist/spa/src/components/ErrorMessage.vue +21 -0
  103. package/dist/spa/src/components/Filters.vue +195 -132
  104. package/dist/spa/src/components/GroupsTable.vue +9 -8
  105. package/dist/spa/src/components/MenuLink.vue +95 -23
  106. package/dist/spa/src/components/ResourceForm.vue +94 -51
  107. package/dist/spa/src/components/ResourceListTable.vue +119 -94
  108. package/dist/spa/src/components/ResourceListTableVirtual.vue +118 -87
  109. package/dist/spa/src/components/ShowTable.vue +21 -15
  110. package/dist/spa/src/components/Sidebar.vue +472 -0
  111. package/dist/spa/src/components/SingleSkeletLoader.vue +6 -6
  112. package/dist/spa/src/components/SkeleteLoader.vue +3 -3
  113. package/dist/spa/src/components/ThreeDotsMenu.vue +84 -15
  114. package/dist/spa/src/components/Toast.vue +40 -29
  115. package/dist/spa/src/components/UserMenuSettingsButton.vue +69 -0
  116. package/dist/spa/src/components/ValueRenderer.vue +44 -17
  117. package/dist/spa/src/controls/BoolToggle.vue +34 -0
  118. package/dist/spa/src/i18n.ts +5 -3
  119. package/dist/spa/src/main.ts +1 -1
  120. package/dist/spa/src/renderers/CompactField.vue +1 -1
  121. package/dist/spa/src/renderers/CompactUUID.vue +1 -1
  122. package/dist/spa/src/router/index.ts +8 -0
  123. package/dist/spa/src/shims-vue.d.ts +5 -0
  124. package/dist/spa/src/spa_types/core.ts +13 -1
  125. package/dist/spa/src/stores/core.ts +13 -1
  126. package/dist/spa/src/stores/filters.ts +33 -2
  127. package/dist/spa/src/stores/modal.ts +6 -1
  128. package/dist/spa/src/stores/toast.ts +22 -3
  129. package/dist/spa/src/types/Back.ts +168 -23
  130. package/dist/spa/src/types/Common.ts +93 -32
  131. package/dist/spa/src/types/FrontendAPI.ts +31 -5
  132. package/dist/spa/src/types/adapters/CaptchaAdapter.ts +34 -0
  133. package/dist/spa/src/types/adapters/EmailAdapter.ts +2 -4
  134. package/dist/spa/src/types/adapters/ImageVisionAdapter.ts +30 -0
  135. package/dist/spa/src/types/adapters/KeyValueAdapter.ts +16 -0
  136. package/dist/spa/src/types/adapters/StorageAdapter.ts +4 -2
  137. package/dist/spa/src/types/adapters/index.ts +3 -0
  138. package/dist/spa/src/utils.ts +291 -11
  139. package/dist/spa/src/views/CreateView.vue +63 -21
  140. package/dist/spa/src/views/EditView.vue +55 -22
  141. package/dist/spa/src/views/ListView.vue +144 -87
  142. package/dist/spa/src/views/LoginView.vue +26 -35
  143. package/dist/spa/src/views/ResourceParent.vue +2 -2
  144. package/dist/spa/src/views/SettingsView.vue +121 -0
  145. package/dist/spa/src/views/ShowView.vue +83 -53
  146. package/dist/spa/src/websocket.ts +6 -1
  147. package/dist/spa/tsconfig.app.json +1 -1
  148. package/dist/spa/vite.config.ts +45 -2
  149. package/dist/types/Back.d.ts +151 -14
  150. package/dist/types/Back.d.ts.map +1 -1
  151. package/dist/types/Back.js +15 -0
  152. package/dist/types/Back.js.map +1 -1
  153. package/dist/types/Common.d.ts +108 -29
  154. package/dist/types/Common.d.ts.map +1 -1
  155. package/dist/types/Common.js.map +1 -1
  156. package/dist/types/FrontendAPI.d.ts +31 -3
  157. package/dist/types/FrontendAPI.d.ts.map +1 -1
  158. package/dist/types/FrontendAPI.js.map +1 -1
  159. package/dist/types/adapters/CaptchaAdapter.d.ts +30 -0
  160. package/dist/types/adapters/CaptchaAdapter.d.ts.map +1 -0
  161. package/dist/types/adapters/CaptchaAdapter.js +5 -0
  162. package/dist/types/adapters/CaptchaAdapter.js.map +1 -0
  163. package/dist/types/adapters/EmailAdapter.d.ts +2 -3
  164. package/dist/types/adapters/EmailAdapter.d.ts.map +1 -1
  165. package/dist/types/adapters/ImageVisionAdapter.d.ts +25 -0
  166. package/dist/types/adapters/ImageVisionAdapter.d.ts.map +1 -0
  167. package/dist/types/adapters/ImageVisionAdapter.js +2 -0
  168. package/dist/types/adapters/ImageVisionAdapter.js.map +1 -0
  169. package/dist/types/adapters/KeyValueAdapter.d.ts +10 -0
  170. package/dist/types/adapters/KeyValueAdapter.d.ts.map +1 -0
  171. package/dist/types/adapters/KeyValueAdapter.js +2 -0
  172. package/dist/types/adapters/KeyValueAdapter.js.map +1 -0
  173. package/dist/types/adapters/StorageAdapter.d.ts +2 -0
  174. package/dist/types/adapters/StorageAdapter.d.ts.map +1 -1
  175. package/dist/types/adapters/index.d.ts +3 -0
  176. package/dist/types/adapters/index.d.ts.map +1 -1
  177. package/package.json +4 -2
@@ -10,50 +10,50 @@
10
10
  "build-only": "vite build",
11
11
  "type-check": "vue-tsc --build --force",
12
12
  "lint": "eslint . --ext .vue,.js,.jsx,.cjs,.mjs,.ts,.tsx,.cts,.mts --fix --ignore-path .gitignore",
13
- "i18n:extract": "echo {} > i18n-empty.json && vue-i18n-extract report --vueFiles \"./src/**/*.{js,vue}\" --output ./i18n-messages.json --languageFiles \"i18n-empty.json\" --add"
13
+ "i18n:extract": "echo {} > i18n-empty.json && vue-i18n-extract report --vueFiles \"./src/**/*.{js,vue,ts}\" --output ./i18n-messages.json --languageFiles \"i18n-empty.json\" --add"
14
14
  },
15
15
  "dependencies": {
16
- "@iconify-prerendered/vue-flag": "^0.28.1748584105",
17
- "@iconify-prerendered/vue-flowbite": "^0.23.1714023977",
18
- "@unhead/vue": "^1.9.12",
19
- "@vueuse/core": "^10.10.0",
16
+ "@iconify-prerendered/vue-flag": "^0.28.1754899047",
17
+ "@iconify-prerendered/vue-flowbite": "^0.28.1754899090",
18
+ "@unhead/vue": "^1.11.20",
19
+ "@vueuse/core": "^10.11.1",
20
20
  "apexcharts": "^4.7.0",
21
- "dayjs": "^1.11.11",
22
- "debounce": "^2.1.0",
23
- "flowbite-datepicker": "^1.2.6",
24
- "javascript-time-ago": "^2.5.11",
25
- "pinia": "^2.1.7",
26
- "sanitize-html": "^2.13.0",
27
- "unhead": "^1.9.12",
21
+ "dayjs": "^1.11.19",
22
+ "debounce": "^2.2.0",
23
+ "flowbite-datepicker": "^1.3.2",
24
+ "javascript-time-ago": "^2.5.12",
25
+ "pinia": "^2.3.1",
26
+ "sanitize-html": "^2.17.0",
27
+ "unhead": "^1.11.20",
28
28
  "uuid": "^10.0.0",
29
- "vue": "^3.5.12",
29
+ "vue": "^3.5.22",
30
30
  "vue-diff": "^1.2.4",
31
- "vue-i18n": "^10.0.5",
32
- "vue-router": "^4.3.0",
31
+ "vue-i18n": "^10.0.8",
32
+ "vue-router": "^4.6.3",
33
33
  "vue-slider-component": "^4.1.0-beta.7"
34
34
  },
35
35
  "devDependencies": {
36
- "@rushstack/eslint-patch": "^1.8.0",
37
- "@tsconfig/node20": "^20.1.4",
38
- "@types/node": "^20.12.5",
39
- "@vitejs/plugin-vue": "^5.0.4",
36
+ "@rushstack/eslint-patch": "^1.14.1",
37
+ "@tsconfig/node20": "^20.1.6",
38
+ "@types/node": "^20.19.24",
39
+ "@vitejs/plugin-vue": "^5.2.4",
40
40
  "@vue/eslint-config-typescript": "^13.0.0",
41
- "@vue/tsconfig": "^0.5.1",
42
- "autoprefixer": "^10.4.19",
43
- "eslint": "^8.57.0",
44
- "eslint-plugin-vue": "^9.23.0",
45
- "flag-icons": "^7.2.3",
41
+ "@vue/tsconfig": "^0.8.1",
42
+ "autoprefixer": "^10.4.21",
43
+ "eslint": "^8.57.1",
44
+ "eslint-plugin-vue": "^9.33.0",
45
+ "flag-icons": "^7.5.0",
46
46
  "flowbite": "^3.1.2",
47
- "i18n-iso-countries": "^7.12.0",
48
- "npm-run-all2": "^6.1.2",
49
- "portfinder": "^1.0.32",
50
- "postcss": "^8.4.38",
51
- "sass": "^1.77.2",
52
- "tailwindcss": "^3.4.17",
53
- "typescript": "~5.4.0",
54
- "vite": "^5.2.13",
47
+ "i18n-iso-countries": "^7.14.0",
48
+ "npm-run-all2": "^6.2.6",
49
+ "portfinder": "^1.0.38",
50
+ "postcss": "^8.5.6",
51
+ "sass": "^1.93.3",
52
+ "tailwindcss": "^3.4.18",
53
+ "typescript": "~5.9.3",
54
+ "vite": "^5.4.21",
55
55
  "vue-i18n-extract": "^2.0.7",
56
- "vue-tsc": "^2.0.11",
57
- "vue3-json-viewer": "^2.2.2"
56
+ "vue-tsc": "^2.2.12",
57
+ "vue3-json-viewer": "^2.4.1"
58
58
  }
59
59
  }
@@ -2,9 +2,9 @@
2
2
  <div>
3
3
  <nav
4
4
  v-if="loggedIn && routerIsReady && loginRedirectCheckIsReady && defaultLayout"
5
- class="fixed h-14 top-0 z-20 w-full border-b shadow-sm bg-lightNavbar shadow-headerShadow dark:bg-darkNavbar dark:border-darkSidebarDevider"
5
+ class="fixed h-14 top-0 z-30 w-full border-b shadow-sm bg-lightNavbar shadow-headerShadow dark:bg-darkNavbar dark:border-darkSidebarDevider"
6
6
  >
7
- <div class="px-3 lg:px-5 lg:pl-3 flex items-center justify-between h-full w-full" >
7
+ <div class="af-header px-3 lg:px-5 lg:pl-3 flex items-center justify-between h-full w-full" >
8
8
  <div class="flex items-center justify-start rtl:justify-end">
9
9
  <button @click="sideBarOpen = !sideBarOpen"
10
10
  type="button" class="inline-flex items-center p-2 text-sm rounded-lg sm:hidden hover:bg-lightSidebarItemHover focus:outline-none focus:ring-2 focus:ring-lightSidebarDevider dark:text-darkSidebarIcons dark:hover:bg-darkSidebarHover dark:focus:ring-lightSidebarDevider">
@@ -24,8 +24,9 @@
24
24
  />
25
25
 
26
26
  <div class="flex items-center ms-3 ">
27
- <span
28
- @click="toggleTheme" class="cursor-pointer flex items-center gap-1 block px-4 py-2 text-sm text-black hover:bg-lightHtml dark:text-darkSidebarTextHover dark:hover:bg-darkHtml dark:hover:text-darkSidebarTextActive" role="menuitem">
27
+ <span
28
+ v-if="!coreStore.config?.singleTheme"
29
+ @click="toggleTheme" class="cursor-pointer flex items-center gap-1 block px-4 py-2 text-sm text-black dark:text-darkSidebarTextHover dark:hover:text-darkSidebarTextActive" role="menuitem">
29
30
  <IconMoonSolid class="w-5 h-5 text-blue-300" v-if="coreStore.theme !== 'dark'" />
30
31
  <IconSunSolid class="w-5 h-5 text-yellow-300" v-else />
31
32
  </span>
@@ -40,7 +41,7 @@
40
41
  </button>
41
42
  </div>
42
43
 
43
- <div class="z-50 hidden my-4 text-base list-none bg-white divide-y divide-gray-100 rounded shadow dark:shadow-black dark:bg-darkSidebar dark:divide-darkSidebarDevider dark:shadow-black" id="dropdown-user">
44
+ <div class="z-50 hidden my-4 text-base list-none bg-lightUserMenuBackground divide-y divide-lightUserMenuBorder text-lightUserMenuText rounded shadow dark:shadow-black dark:bg-darkUserMenuBackground dark:divide-darkUserMenuBorder text-darkUserMenuText dark:shadow-black" id="dropdown-user">
44
45
  <div class="px-4 py-3" role="none">
45
46
  <p class="text-sm text-gray-900 dark:text-darkNavbarText" role="none" v-if="coreStore.userFullname">
46
47
  {{ coreStore.userFullname }}
@@ -51,15 +52,18 @@
51
52
  </div>
52
53
 
53
54
  <ul class="py-1" role="none">
54
- <li v-for="c in coreStore?.config?.globalInjections?.userMenu || []" >
55
+ <li v-for="c in userMenuComponents" class="bg-lightUserMenuItemBackground hover:bg-lightUserMenuItemBackgroundHover text-lightUserMenuItemText hover:text-lightUserMenuItemText dark:bg-darkUserMenuItemBackground dark:hover:bg-darkUserMenuItemBackgroundHover dark:text-darkUserMenuItemText dark:hover:darkUserMenuItemTextHover" >
55
56
  <component
56
57
  :is="getCustomComponent(c)"
57
58
  :meta="c.meta"
58
59
  :adminUser="coreStore.adminUser"
59
60
  />
60
61
  </li>
62
+ <li v-if="coreStore?.config?.settingPages && coreStore.config.settingPages.length > 0">
63
+ <UserMenuSettingsButton />
64
+ </li>
61
65
  <li>
62
- <button @click="logout" class="cursor-pointer flex items-center gap-1 block px-4 py-2 text-sm text-black hover:bg-html dark:text-darkSidebarTextHover dark:hover:bg-darkSidebarItemHover dark:hover:text-darkSidebarTextActive w-full" role="menuitem">{{ $t('Sign out') }}</button>
66
+ <button @click="logout" class="cursor-pointer flex items-center gap-1 block px-4 py-2 text-sm bg-lightUserMenuItemBackground hover:bg-lightUserMenuItemBackgroundHover text-lightUserMenuItemText hover:text-lightUserMenuItemText dark:bg-darkUserMenuItemBackground dark:hover:bg-darkUserMenuItemBackgroundHover dark:text-darkUserMenuItemText dark:hover:darkUserMenuItemTextHover w-full" role="menuitem">{{ $t('Sign out') }}</button>
63
67
  </li>
64
68
  </ul>
65
69
  </div>
@@ -68,114 +72,20 @@
68
72
  </div>
69
73
  </nav>
70
74
 
71
- <aside
72
- ref="sidebarAside"
75
+ <Sidebar
73
76
  v-if="loggedIn && routerIsReady && loginRedirectCheckIsReady && defaultLayout"
74
- id="logo-lightSidebar" class="fixed border-none top-0 left-0 z-30 w-64 h-screen transition-transform bg-lightSidebar dark:bg-darkSidebar border-r border-lightSidebarBorder sm:translate-x-0 dark:bg-darkSidebar dark:border-darkSidebarBorder"
75
- :class="{ '-translate-x-full': !sideBarOpen, 'transform-none': sideBarOpen }"
76
- aria-label="Sidebar"
77
- >
78
- <div class="h-full px-3 pb-4 overflow-y-auto bg-lightSidebar dark:bg-darkSidebar border-r border-lightSidebarBorder dark:border-darkSidebarBorder">
79
- <div class="flex ms-2 m-4">
80
- <img :src="loadFile(coreStore.config?.brandLogo || '@/assets/logo.svg')" :alt="`${ coreStore.config?.brandName } Logo`" class="h-8 me-3" />
81
- <span
82
- v-if="coreStore.config?.showBrandNameInSidebar"
83
- class="self-center text-lightNavbarText-size font-semibold sm:text-lightNavbarText-size whitespace-nowrap dark:text-darkSidebarText text-lightSidebarText"
84
- >
85
- {{ coreStore.config?.brandName }}
86
- </span>
87
- </div>
88
-
89
- <ul class="space-y-2 font-medium">
90
- <template v-for="(item, i) in coreStore.menu" :key="`menu-${i}`">
91
- <div v-if="item.type === 'divider'" class="border-t border-lightSidebarDevider dark:border-darkSidebarDevider"></div>
92
- <div v-else-if="item.type === 'gap'" class="flex items-center justify-center h-8"></div>
93
- <div v-else-if="item.type === 'heading'" class="flex items-center justify-left pl-2 h-8 text-lightSidebarHeading dark:text-darkSidebarHeading
94
- ">{{ item.label }}</div>
95
- <li v-else-if="item.children" class=" ">
96
- <button @click="clickOnMenuItem(i)" type="button" class="flex items-center w-full p-2 text-base text-lightSidebarText rounded-default transition duration-75 group hover:bg-lightSidebarItemHover hover:text-lightSidebarTextHover dark:text-darkSidebarText dark:hover:bg-darkSidebarHover dark:hover:text-darkSidebarTextHover"
97
- :aria-controls="`dropdown-example${i}`"
98
- :data-collapse-toggle="`dropdown-example${i}`"
99
- >
100
-
101
- <component v-if="item.icon" :is="getIcon(item.icon)" class="w-5 h-5 text-lightSidebarIcons group-hover:text-lightSidebarIconsHover transition duration-75 dark:group-hover:text-darkSidebarIconsHover dark:text-darkSidebarIcons" ></component>
102
-
103
- <span class="text-ellipsis overflow-hidden flex-1 ms-3 text-left rtl:text-right whitespace-nowrap">{{ item.label }}
104
-
105
- <span v-if="item.badge" class="inline-flex items-center justify-center w-3 h-3 p-3 ms-3 text-sm font-medium rounded-full bg-lightAnnouncementBG dark:bg-darkAnnouncementBG
106
- fill-lightAnnouncementText dark:fill-darkAccent text-lightAnnouncementText dark:text-darkAccent">
107
- <Tooltip v-if="item.badgeTooltip">
108
- {{ item.badge }}
109
- <template #tooltip>
110
- {{ item.badgeTooltip }}
111
- </template>
112
- </Tooltip>
113
- <template v-else>
114
- {{ item.badge }}
115
- </template>
116
- </span>
117
- </span>
118
-
119
- <svg :class="{'rotate-180': opened.includes(i) }" class="w-3 h-3" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 10 6">
120
- <path stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="m1 1 4 4 4-4"/>
121
- </svg>
122
- </button>
123
-
124
- <ul :id="`dropdown-example${i}`" role="none" class="af-sidebar-dropdown pt-1 space-y-1" :class="{ 'hidden': !opened.includes(i) }">
125
- <template v-for="(child, j) in item.children" :key="`menu-${i}-${j}`">
126
- <li>
127
- <MenuLink :item="child" isChild="true" @click="hideSidebar"/>
128
- </li>
129
- </template>
130
- </ul>
131
- </li>
132
- <li v-else>
133
- <MenuLink :item="item" @click="hideSidebar"/>
134
- </li>
135
- </template>
136
- </ul>
137
-
138
-
139
- <div id="dropdown-cta" class="p-4 mt-6 rounded-lg bg-lightAnnouncementBG dark:bg-darkAnnouncementBG
140
- fill-lightAnnouncementText dark:fill-darkAccent text-lightAnnouncementText dark:text-darkAccent text-sm" role="alert"
141
- v-if="ctaBadge"
142
- >
143
- <div class="flex items-center mb-3" :class="!ctaBadge.title ? 'float-right' : ''">
144
- <!-- <span class="bg-lightPrimaryOpacity dark:bg-darkPrimaryOpacity text-sm font-semibold me-2 px-2.5 py-0.5 rounded "
145
- v-if="ctaBadge.title"
146
- > -->
147
- <span>
148
- {{ctaBadge.title}}
149
- </span>
150
- <button type="button"
151
- class="ms-auto -mx-1.5 -my-1.5 bg-lightPrimaryOpacity dark:bg-darkPrimaryOpacity inline-flex justify-center items-center w-6 h-6 rounded-lg p-1 hover:brightness-110"
152
-
153
- data-dismiss-target="#dropdown-cta" aria-label="Close"
154
- v-if="ctaBadge?.closable" @click="closeCTA"
155
- >
156
- <span class="sr-only">Close</span>
157
- <svg class="w-2.5 h-2.5" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 14 14">
158
- <path stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="m1 1 6 6m0 0 6 6M7 7l6-6M7 7l-6 6"/>
159
- </svg>
160
- </button>
161
- </div>
162
- <p class="mb-3 text-sm " v-if="ctaBadge.html" v-html="ctaBadge.html"></p>
163
- <p class="mb-3 text-sm fill-lightNavbarText dark:fill-darkPrimary text-lightNavbarText dark:text-darkNavbarPrimary" v-else>
164
- {{ ctaBadge.text }}
165
- </p>
166
- <!-- <a class="text-sm text-lightPrimary underline font-medium hover:text-blue-900 dark:text-blue-400 dark:hover:text-blue-300" href="#">Turn new navigation off</a> -->
167
- </div>
168
-
169
- <component
170
- v-for="c in coreStore?.config?.globalInjections?.sidebar || []"
171
- :is="getCustomComponent(c)"
172
- :meta="c.meta"
173
- :adminUser="coreStore.adminUser"
174
- />
175
- </div>
176
- </aside>
77
+ :sideBarOpen="sideBarOpen"
78
+ :forceIconOnly="route.meta?.sidebarAndHeader === 'preferIconOnly'"
79
+ @hideSidebar="hideSidebar"
80
+ @loadMenu="loadMenu"
81
+ @sidebarStateChange="handleSidebarStateChange"
82
+ />
177
83
 
178
- <div class="sm:ml-64 max-w-[100vw] sm:max-w-[calc(100%-16rem)]"
84
+ <div class="af-content-wrapper transition-all duration-300 ease-in-out max-w-[100vw]"
85
+ :style="{
86
+ marginLeft: isSidebarIconOnly ? '4.5rem' : expandedWidth,
87
+ maxWidth: isSidebarIconOnly ? 'calc(100% - 4.5rem)' : `calc(100% - ${expandedWidth})`
88
+ }"
179
89
  v-if="loggedIn && routerIsReady && loginRedirectCheckIsReady && defaultLayout">
180
90
  <div class="p-0 dark:border-gray-700 mt-14">
181
91
  <RouterView/>
@@ -193,7 +103,7 @@
193
103
  </div>
194
104
  </div>
195
105
  <AcceptModal />
196
- <div v-if="toastStore.toasts.length>0" class="fixed bottom-5 right-5 flex gap-1 flex-col-reverse z-50">
106
+ <div v-if="toastStore.toasts.length>0" class="fixed bottom-5 right-5 flex gap-1 flex-col-reverse z-[100]">
197
107
  <transition-group
198
108
  name="fade"
199
109
  tag="div"
@@ -235,10 +145,18 @@
235
145
  @apply opacity-100;
236
146
  }
237
147
 
148
+ @media (max-width: 640px) {
149
+ .af-content-wrapper {
150
+ margin-left: 0 !important;
151
+ max-width: 100% !important;
152
+ }
153
+ }
154
+
155
+
238
156
  </style>
239
157
 
240
158
  <script setup lang="ts">
241
- import { computed, onMounted, ref, watch, onBeforeMount, nextTick, type Ref } from 'vue';
159
+ import { computed, onMounted, ref, watch, onBeforeMount } from 'vue';
242
160
  import { RouterView } from 'vue-router';
243
161
  import { Dropdown } from 'flowbite'
244
162
  import './index.scss'
@@ -246,19 +164,15 @@ import { useCoreStore } from '@/stores/core';
246
164
  import { useUserStore } from '@/stores/user';
247
165
  import { IconMoonSolid, IconSunSolid } from '@iconify-prerendered/vue-flowbite';
248
166
  import AcceptModal from './components/AcceptModal.vue';
249
- import MenuLink from './components/MenuLink.vue';
167
+ import Sidebar from './components/Sidebar.vue';
250
168
  import { useRoute, useRouter } from 'vue-router';
251
- import { getIcon, verySimpleHash } from '@/utils';
252
169
  import { createHead } from 'unhead'
253
- import { loadFile } from '@/utils';
170
+ import { getCustomComponent } from '@/utils';
254
171
  import Toast from './components/Toast.vue';
255
172
  import {useToastStore} from '@/stores/toast';
256
- import { getCustomComponent } from '@/utils';
257
- import type { AdminForthConfigMenuItem, AnnouncementBadgeResponse } from './types/Common';
258
- import { Tooltip } from '@/afcl';
259
173
  import { initFrontedAPI } from '@/adminforth';
260
174
  import adminforth from '@/adminforth';
261
-
175
+ import UserMenuSettingsButton from './components/UserMenuSettingsButton.vue';
262
176
 
263
177
  const coreStore = useCoreStore();
264
178
  const toastStore = useToastStore();
@@ -271,37 +185,40 @@ const sideBarOpen = ref(false);
271
185
  const defaultLayout = ref(true);
272
186
  const route = useRoute();
273
187
  const router = useRouter();
274
- //create a ref to store the opened menu items with ts type;
275
- const opened = ref<string[]>([]);
276
188
  const publicConfigLoaded = ref(false);
277
189
  const dropdownUserButton = ref(null);
278
190
 
279
- const sidebarAside = ref(null);
280
191
 
281
192
  const routerIsReady = ref(false);
282
193
  const loginRedirectCheckIsReady = ref(false);
283
194
 
195
+ const isSidebarIconOnly = ref(localStorage.getItem('afIconOnlySidebar') === 'true');
196
+
284
197
  const loggedIn = computed(() => !!coreStore?.adminUser);
285
198
 
199
+ const expandedWidth = computed(() => coreStore.config?.iconOnlySidebar?.expandedSidebarWidth || '16.5rem');
200
+
286
201
  const theme = ref('light');
287
202
 
203
+ const userMenuComponents = computed(() => {
204
+ console.log('🪲🆕 userMenuComponents recomputed', coreStore?.config?.globalInjections?.userMenu);
205
+ return coreStore?.config?.globalInjections?.userMenu || [];
206
+ })
207
+
288
208
  function hideSidebar(): void {
289
209
  sideBarOpen.value = false;
290
210
  }
291
211
 
212
+ function handleSidebarStateChange(state: { isSidebarIconOnly: boolean }) {
213
+ isSidebarIconOnly.value = state.isSidebarIconOnly;
214
+ }
215
+
216
+
292
217
  function toggleTheme() {
293
218
  theme.value = theme.value === 'light' ? 'dark' : 'light';
294
219
  coreStore.toggleTheme();
295
220
  }
296
221
 
297
- function clickOnMenuItem(label: string) {
298
- if (opened.value.includes(label)) {
299
- opened.value = opened.value.filter((item) => item !== label);
300
- } else {
301
- opened.value.push(label);
302
- }
303
-
304
- }
305
222
 
306
223
  async function logout() {
307
224
  userStore.unauthorize();
@@ -316,7 +233,7 @@ async function initRouter() {
316
233
 
317
234
  async function loadMenu() {
318
235
  await initRouter();
319
- if (!route.meta.customLayout) {
236
+ if (route.meta.sidebarAndHeader !== 'none') {
320
237
  // for custom layouts we don't need to fetch menu
321
238
  await coreStore.fetchMenuAndResource();
322
239
  }
@@ -324,8 +241,11 @@ async function loadMenu() {
324
241
  }
325
242
 
326
243
  function handleCustomLayout() {
327
- if (route.meta?.customLayout) {
244
+ if (route.meta?.sidebarAndHeader === 'none') {
328
245
  defaultLayout.value = false;
246
+ } else if (route.meta?.sidebarAndHeader === 'preferIconOnly') {
247
+ defaultLayout.value = true;
248
+ isSidebarIconOnly.value = true;
329
249
  } else {
330
250
  defaultLayout.value = true;
331
251
  }
@@ -361,13 +281,6 @@ watch([route, () => coreStore.resourceById, () => coreStore.config], async () =>
361
281
 
362
282
  });
363
283
 
364
- watch(()=>coreStore.menu, () => {
365
- coreStore.menu.forEach((item, i) => {
366
- if (item.open) {
367
- opened.value.push(i);
368
- };
369
- });
370
- })
371
284
 
372
285
  watch(dropdownUserButton, (dropdownUserButton) => {
373
286
  if (dropdownUserButton) {
@@ -386,11 +299,6 @@ async function loadPublicConfig() {
386
299
  publicConfigLoaded.value = true;
387
300
  }
388
301
 
389
- watch(sidebarAside, (sidebarAside) => {
390
- if (sidebarAside) {
391
- coreStore.fetchMenuBadges();
392
- }
393
- })
394
302
 
395
303
  // initialize components based on data attribute selectors
396
304
  onMounted(async () => {
@@ -410,32 +318,12 @@ onBeforeMount(()=>{
410
318
  document.documentElement.classList.toggle('dark', theme.value === 'dark');
411
319
  })
412
320
 
413
-
414
- const ctaBadge: Ref<(AnnouncementBadgeResponse & { hash: string; }) | null> = computed(() => {
415
- const badge = coreStore.config?.announcementBadge;
416
- if (!badge) {
417
- return null;
321
+ watch(() => coreStore.config?.singleTheme, (singleTheme) => {
322
+ if (singleTheme) {
323
+ theme.value = singleTheme;
324
+ window.localStorage.setItem('af__theme', singleTheme);
325
+ document.documentElement.classList.toggle('dark', theme.value === 'dark');
418
326
  }
419
- const hash = badge.closable ? verySimpleHash(JSON.stringify(badge)) : '';
420
- if (badge.closable && window.localStorage.getItem(`ctaBadge-${hash}`)) {
421
- return null;
422
- }
423
- return {...badge, hash};
424
- });
425
-
426
- function closeCTA() {
427
- if (!ctaBadge.value) {
428
- return;
429
- }
430
- const hash = ctaBadge.value.hash;
431
- window.localStorage.setItem(`ctaBadge-${hash}`, '1');
432
- nextTick( async() => {
433
- loadMenu();
434
- await coreStore.fetchMenuBadges();
435
- adminforth.menu.refreshMenuBadges();
436
- })
437
-
438
- }
439
-
327
+ }, { immediate: true })
440
328
 
441
329
  </script>
@@ -1,6 +1,5 @@
1
- import type { FilterParams, FrontendAPIInterface } from "./types/FrontendAPI";
2
- import type { FrontendAPIInterface, ConfirmParams, AlertParams, } from '@/types/FrontendAPI';
3
- import type { AdminForthFilterOperators, AdminForthResourceColumn } from '@/types/Common';
1
+ import type { FilterParams, FrontendAPIInterface, ConfirmParams, AlertParams, } from '@/types/FrontendAPI';
2
+ import type { AdminForthFilterOperators, AdminForthResourceColumnCommon } from '@/types/Common';
4
3
  import { useToastStore } from '@/stores/toast';
5
4
  import { useModalStore } from '@/stores/modal';
6
5
  import { useCoreStore } from '@/stores/core';
@@ -36,6 +35,10 @@ class FrontendAPI implements FrontendAPIInterface {
36
35
  refreshMenuBadges: () => void;
37
36
  }
38
37
 
38
+ public show: {
39
+ refresh(): void;
40
+ }
41
+
39
42
  closeUserMenuDropdown(): void {
40
43
  console.log('closeUserMenuDropdown')
41
44
  }
@@ -73,9 +76,15 @@ class FrontendAPI implements FrontendAPIInterface {
73
76
  updateFilter: this.updateListFilter.bind(this),
74
77
  clearFilters: this.clearListFilters.bind(this),
75
78
  }
79
+
80
+ this.show = {
81
+ refresh: () => {
82
+ console.log('show.refresh')
83
+ }
84
+ }
76
85
  }
77
86
 
78
- confirm(params: ConfirmParams): Promise<void> {
87
+ confirm(params: ConfirmParams): Promise<boolean> {
79
88
  return new Promise((resolve, reject) => {
80
89
  this.modalStore.setModalContent({
81
90
  content: params.message,
@@ -88,35 +97,50 @@ class FrontendAPI implements FrontendAPIInterface {
88
97
  })
89
98
  }
90
99
 
91
- alert(params: AlertParams): void {
92
- this.toastStore.addToast({
100
+ alert(params: AlertParams): void | Promise<string> | string {
101
+ const toats = {
93
102
  message: params.message,
94
103
  messageHtml: params.messageHtml,
95
104
  variant: params.variant,
96
- timeout: params.timeout
97
- })
105
+ timeout: params.timeout,
106
+ buttons: params.buttons,
107
+ }
108
+ if (params.buttons && params.buttons.length > 0) {
109
+ return new Promise<string>((resolve) => {
110
+ this.toastStore.addToast({
111
+ ...toats,
112
+ onResolve: (value?: any) => resolve(String(value ?? '')),
113
+ })
114
+ })
115
+ } else {
116
+ this.toastStore.addToast({...toats})
117
+ }
98
118
  }
99
119
 
100
120
  listFilterValidation(filter: FilterParams): boolean {
101
121
  if(router.currentRoute.value.meta.type !== 'list'){
102
122
  throw new Error(`Cannot use ${this.setListFilter.name} filter on a list page`)
103
- } else {
104
- console.log(this.coreStore.resourceColumnsWithFilters,'core store')
105
- const filterField = this.coreStore.resourceColumnsWithFilters.find((col: AdminForthResourceColumn) => col.name === filter.field)
106
- if(!filterField){
107
- throw new Error(`Field ${filter.field} is not available for filtering`)
108
- }
109
-
110
123
  }
111
124
  return true
112
125
  }
113
126
 
114
127
  setListFilter(filter: FilterParams): void {
115
128
  if(this.listFilterValidation(filter)){
116
- if(this.filtersStore.filters.some((f) => {return f.field === filter.field && f.operator === filter.operator})){
117
- throw new Error(`Filter ${filter.field} with operator ${filter.operator} already exists`)
129
+ const existingFilterIndex = this.filtersStore.filters.findIndex((f: any) => {
130
+ return f.field === filter.field && f.operator === filter.operator
131
+ });
132
+
133
+ if(existingFilterIndex !== -1){
134
+ // Update existing filter instead of throwing error
135
+ const filters = [...this.filtersStore.filters];
136
+ if (filter.value === undefined) {
137
+ filters.splice(existingFilterIndex, 1);
138
+ } else {
139
+ filters[existingFilterIndex] = filter;
140
+ }
141
+ this.filtersStore.setFilters(filters);
118
142
  } else {
119
- this.filtersStore.setFilter(filter)
143
+ this.filtersStore.setFilter(filter);
120
144
  }
121
145
  }
122
146
  }
@@ -142,7 +142,6 @@ watch(() => [options.value, chart.value], (value) => {
142
142
  if (!value || !chart.value) {
143
143
  return;
144
144
  }
145
- console.log('options changed', options.value);
146
145
  if (apexChart) {
147
146
  apexChart.updateOptions(options.value);
148
147
  } else {
@@ -60,7 +60,7 @@ const optionsBase = {
60
60
  tooltip: {
61
61
  shared: true,
62
62
  intersect: false,
63
- formatter: function (value) {
63
+ formatter: function (value: any) {
64
64
  return value
65
65
  },
66
66
  },
@@ -71,7 +71,7 @@ const optionsBase = {
71
71
  fontFamily: "Inter, sans-serif",
72
72
  cssClass: 'text-xs font-normal fill-gray-500 dark:fill-gray-400'
73
73
  },
74
- formatter: function (value) {
74
+ formatter: function (value: any) {
75
75
  return value
76
76
  }
77
77
  },
@@ -2,16 +2,15 @@
2
2
  <button
3
3
  v-bind="$attrs"
4
4
  type="submit"
5
- class="afcl-button flex items-center justify-center gap-1 text-lightPrimaryContrast bg-lightPrimary dark:bg-darkPrimary hover:brightness-110
6
- focus:ring-4 focus:outline-none focus:ring-lightPrimary focus:ring-opacity-50 font-medium rounded-lg text-sm px-5 py-2.5 text-center dark:focus:ring-darkPrimary dark:focus:ring-opacity-50"
5
+ class="afcl-button flex items-center justify-center gap-1 text-lightButtonsText bg-lightButtonsBackground border border-lightButtonsBorder dark:bg-darkButtonsBackground hover:bg-lightButtonsHover hover:border-lightButtonsBorderHover
6
+ focus:ring-4 focus:outline-none focus:ring-lightButtonFocusRing focus:ring-opacity-50 font-medium rounded-lg text-sm px-5 py-2.5 text-center dark:focus:ring-darkButtonFocusRing dark:text-darkButtonsText dark:border-darkButtonsBorder dark:hover:bg-darkButtonsHover dark:hover:border-darkButtonsBorderHover"
7
7
  :class="{
8
- 'cursor-default': props.disabled,
9
- 'opacity-50': props.disabled,
10
- 'pointer-events-none': props.disabled,
8
+ 'cursor-default opacity-50 pointer-events-none': props.disabled,
9
+ 'active brightness-200 hover:brightness-150' : props.active
11
10
  }"
12
11
  >
13
12
  <svg v-if="props.loader"
14
- aria-hidden="true" class="w-4 h-4 text-gray-200 animate-spin dark:text-gray-600 fill-lightPrimary dark:fill-darkPrimary"
13
+ aria-hidden="true" class="w-4 h-4 text-lightButtonsText animate-spin dark:text-darkButtonsText fill-lightButtonsBackground dark:fill-darkPrimary"
15
14
  viewBox="0 0 100 101" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M100 50.5908C100 78.2051 77.6142 100.591 50 100.591C22.3858 100.591 0 78.2051 0 50.5908C0 22.9766 22.3858 0.59082 50 0.59082C77.6142 0.59082 100 22.9766 100 50.5908ZM9.08144 50.5908C9.08144 73.1895 27.4013 91.5094 50 91.5094C72.5987 91.5094 90.9186 73.1895 90.9186 50.5908C90.9186 27.9921 72.5987 9.67226 50 9.67226C27.4013 9.67226 9.08144 27.9921 9.08144 50.5908Z" fill="currentColor"/><path d="M93.9676 39.0409C96.393 38.4038 97.8624 35.9116 97.0079 33.5539C95.2932 28.8227 92.871 24.3692 89.8167 20.348C85.8452 15.1192 80.8826 10.7238 75.2124 7.41289C69.5422 4.10194 63.2754 1.94025 56.7698 1.05124C51.7666 0.367541 46.6976 0.446843 41.7345 1.27873C39.2613 1.69328 37.813 4.19778 38.4501 6.62326C39.0873 9.04874 41.5694 10.4717 44.0505 10.1071C47.8511 9.54855 51.7191 9.52689 55.5402 10.0491C60.8642 10.7766 65.9928 12.5457 70.6331 15.2552C75.2735 17.9648 79.3347 21.5619 82.5849 25.841C84.9175 28.9121 86.7997 32.2913 88.1811 35.8758C89.083 38.2158 91.5421 39.6781 93.9676 39.0409Z" fill="currentFill"/></svg>
16
15
  <slot></slot>
17
16
  </button>
@@ -22,6 +21,7 @@
22
21
  const props = defineProps({
23
22
  loader: Boolean,
24
23
  disabled: Boolean,
24
+ active: Boolean,
25
25
  });
26
26
 
27
27
  </script>