adminforth 2.4.0-next.31 → 2.4.0-next.310

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 (176) 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 +69 -17
  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 +12 -2
  40. package/dist/index.d.ts.map +1 -1
  41. package/dist/index.js +45 -22
  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 +172 -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 +59 -174
  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 +127 -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 +315 -73
  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 +16 -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 +90 -23
  106. package/dist/spa/src/components/ResourceForm.vue +94 -51
  107. package/dist/spa/src/components/ResourceListTable.vue +115 -85
  108. package/dist/spa/src/components/ResourceListTableVirtual.vue +114 -80
  109. package/dist/spa/src/components/ShowTable.vue +21 -15
  110. package/dist/spa/src/components/Sidebar.vue +470 -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 +163 -23
  130. package/dist/spa/src/types/Common.ts +91 -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 -2
  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/index.ts +8 -0
  137. package/dist/spa/src/utils.ts +291 -11
  138. package/dist/spa/src/views/CreateView.vue +63 -21
  139. package/dist/spa/src/views/EditView.vue +55 -22
  140. package/dist/spa/src/views/ListView.vue +144 -87
  141. package/dist/spa/src/views/LoginView.vue +26 -35
  142. package/dist/spa/src/views/ResourceParent.vue +2 -2
  143. package/dist/spa/src/views/SettingsView.vue +121 -0
  144. package/dist/spa/src/views/ShowView.vue +83 -53
  145. package/dist/spa/src/websocket.ts +6 -1
  146. package/dist/spa/tsconfig.app.json +1 -1
  147. package/dist/spa/vite.config.ts +45 -2
  148. package/dist/types/Back.d.ts +146 -14
  149. package/dist/types/Back.d.ts.map +1 -1
  150. package/dist/types/Back.js +15 -0
  151. package/dist/types/Back.js.map +1 -1
  152. package/dist/types/Common.d.ts +106 -29
  153. package/dist/types/Common.d.ts.map +1 -1
  154. package/dist/types/Common.js.map +1 -1
  155. package/dist/types/FrontendAPI.d.ts +31 -3
  156. package/dist/types/FrontendAPI.d.ts.map +1 -1
  157. package/dist/types/FrontendAPI.js.map +1 -1
  158. package/dist/types/adapters/CaptchaAdapter.d.ts +30 -0
  159. package/dist/types/adapters/CaptchaAdapter.d.ts.map +1 -0
  160. package/dist/types/adapters/CaptchaAdapter.js +5 -0
  161. package/dist/types/adapters/CaptchaAdapter.js.map +1 -0
  162. package/dist/types/adapters/EmailAdapter.d.ts +1 -1
  163. package/dist/types/adapters/ImageVisionAdapter.d.ts +25 -0
  164. package/dist/types/adapters/ImageVisionAdapter.d.ts.map +1 -0
  165. package/dist/types/adapters/ImageVisionAdapter.js +2 -0
  166. package/dist/types/adapters/ImageVisionAdapter.js.map +1 -0
  167. package/dist/types/adapters/KeyValueAdapter.d.ts +10 -0
  168. package/dist/types/adapters/KeyValueAdapter.d.ts.map +1 -0
  169. package/dist/types/adapters/KeyValueAdapter.js +2 -0
  170. package/dist/types/adapters/KeyValueAdapter.js.map +1 -0
  171. package/dist/types/adapters/index.d.ts +9 -0
  172. package/dist/types/adapters/index.d.ts.map +1 -0
  173. package/dist/types/adapters/index.js +2 -0
  174. package/dist/types/adapters/index.js.map +1 -0
  175. package/package.json +4 -2
  176. package/dist/spa/src/types/adapters/index.js +0 -5
@@ -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
  }
@@ -1,10 +1,10 @@
1
1
  <template>
2
- <div>
2
+ <div class="">
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 coreStore?.config?.globalInjections?.userMenu || []" 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,22 @@
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
+ :class="{
86
+ 'sm:ml-18': isSidebarIconOnly,
87
+ 'sm:ml-[16.5rem]': !isSidebarIconOnly,
88
+ 'sm:max-w-[calc(100%-4.5rem)]': isSidebarIconOnly,
89
+ 'sm:max-w-[calc(100%-16rem)]': !isSidebarIconOnly
90
+ }"
179
91
  v-if="loggedIn && routerIsReady && loginRedirectCheckIsReady && defaultLayout">
180
92
  <div class="p-0 dark:border-gray-700 mt-14">
181
93
  <RouterView/>
@@ -193,7 +105,7 @@
193
105
  </div>
194
106
  </div>
195
107
  <AcceptModal />
196
- <div v-if="toastStore.toasts.length>0" class="fixed bottom-5 right-5 flex gap-1 flex-col-reverse z-50">
108
+ <div v-if="toastStore.toasts.length>0" class="fixed bottom-5 right-5 flex gap-1 flex-col-reverse z-[100]">
197
109
  <transition-group
198
110
  name="fade"
199
111
  tag="div"
@@ -235,10 +147,20 @@
235
147
  @apply opacity-100;
236
148
  }
237
149
 
150
+ @media (min-width: 640px) {
151
+ .sm\:ml-18 {
152
+ margin-left: 4.5rem;
153
+ }
154
+ .sm\:max-w-\[calc\(100\%-4\.5rem\)\] {
155
+ max-width: calc(100% - 4.5rem);
156
+ }
157
+ }
158
+
159
+
238
160
  </style>
239
161
 
240
162
  <script setup lang="ts">
241
- import { computed, onMounted, ref, watch, onBeforeMount, nextTick, type Ref } from 'vue';
163
+ import { computed, onMounted, ref, watch, onBeforeMount } from 'vue';
242
164
  import { RouterView } from 'vue-router';
243
165
  import { Dropdown } from 'flowbite'
244
166
  import './index.scss'
@@ -246,19 +168,15 @@ import { useCoreStore } from '@/stores/core';
246
168
  import { useUserStore } from '@/stores/user';
247
169
  import { IconMoonSolid, IconSunSolid } from '@iconify-prerendered/vue-flowbite';
248
170
  import AcceptModal from './components/AcceptModal.vue';
249
- import MenuLink from './components/MenuLink.vue';
171
+ import Sidebar from './components/Sidebar.vue';
250
172
  import { useRoute, useRouter } from 'vue-router';
251
- import { getIcon, verySimpleHash } from '@/utils';
252
173
  import { createHead } from 'unhead'
253
- import { loadFile } from '@/utils';
174
+ import { getCustomComponent } from '@/utils';
254
175
  import Toast from './components/Toast.vue';
255
176
  import {useToastStore} from '@/stores/toast';
256
- import { getCustomComponent } from '@/utils';
257
- import type { AdminForthConfigMenuItem, AnnouncementBadgeResponse } from './types/Common';
258
- import { Tooltip } from '@/afcl';
259
177
  import { initFrontedAPI } from '@/adminforth';
260
178
  import adminforth from '@/adminforth';
261
-
179
+ import UserMenuSettingsButton from './components/UserMenuSettingsButton.vue';
262
180
 
263
181
  const coreStore = useCoreStore();
264
182
  const toastStore = useToastStore();
@@ -271,16 +189,15 @@ const sideBarOpen = ref(false);
271
189
  const defaultLayout = ref(true);
272
190
  const route = useRoute();
273
191
  const router = useRouter();
274
- //create a ref to store the opened menu items with ts type;
275
- const opened = ref<string[]>([]);
276
192
  const publicConfigLoaded = ref(false);
277
193
  const dropdownUserButton = ref(null);
278
194
 
279
- const sidebarAside = ref(null);
280
195
 
281
196
  const routerIsReady = ref(false);
282
197
  const loginRedirectCheckIsReady = ref(false);
283
198
 
199
+ const isSidebarIconOnly = ref(localStorage.getItem('afIconOnlySidebar') === 'true');
200
+
284
201
  const loggedIn = computed(() => !!coreStore?.adminUser);
285
202
 
286
203
  const theme = ref('light');
@@ -289,19 +206,16 @@ function hideSidebar(): void {
289
206
  sideBarOpen.value = false;
290
207
  }
291
208
 
209
+ function handleSidebarStateChange(state: { isSidebarIconOnly: boolean }) {
210
+ isSidebarIconOnly.value = state.isSidebarIconOnly;
211
+ }
212
+
213
+
292
214
  function toggleTheme() {
293
215
  theme.value = theme.value === 'light' ? 'dark' : 'light';
294
216
  coreStore.toggleTheme();
295
217
  }
296
218
 
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
219
 
306
220
  async function logout() {
307
221
  userStore.unauthorize();
@@ -316,7 +230,7 @@ async function initRouter() {
316
230
 
317
231
  async function loadMenu() {
318
232
  await initRouter();
319
- if (!route.meta.customLayout) {
233
+ if (route.meta.sidebarAndHeader !== 'none') {
320
234
  // for custom layouts we don't need to fetch menu
321
235
  await coreStore.fetchMenuAndResource();
322
236
  }
@@ -324,8 +238,11 @@ async function loadMenu() {
324
238
  }
325
239
 
326
240
  function handleCustomLayout() {
327
- if (route.meta?.customLayout) {
241
+ if (route.meta?.sidebarAndHeader === 'none') {
328
242
  defaultLayout.value = false;
243
+ } else if (route.meta?.sidebarAndHeader === 'preferIconOnly') {
244
+ defaultLayout.value = true;
245
+ isSidebarIconOnly.value = true;
329
246
  } else {
330
247
  defaultLayout.value = true;
331
248
  }
@@ -361,13 +278,6 @@ watch([route, () => coreStore.resourceById, () => coreStore.config], async () =>
361
278
 
362
279
  });
363
280
 
364
- watch(()=>coreStore.menu, () => {
365
- coreStore.menu.forEach((item, i) => {
366
- if (item.open) {
367
- opened.value.push(i);
368
- };
369
- });
370
- })
371
281
 
372
282
  watch(dropdownUserButton, (dropdownUserButton) => {
373
283
  if (dropdownUserButton) {
@@ -386,11 +296,6 @@ async function loadPublicConfig() {
386
296
  publicConfigLoaded.value = true;
387
297
  }
388
298
 
389
- watch(sidebarAside, (sidebarAside) => {
390
- if (sidebarAside) {
391
- coreStore.fetchMenuBadges();
392
- }
393
- })
394
299
 
395
300
  // initialize components based on data attribute selectors
396
301
  onMounted(async () => {
@@ -410,32 +315,12 @@ onBeforeMount(()=>{
410
315
  document.documentElement.classList.toggle('dark', theme.value === 'dark');
411
316
  })
412
317
 
413
-
414
- const ctaBadge: Ref<(AnnouncementBadgeResponse & { hash: string; }) | null> = computed(() => {
415
- const badge = coreStore.config?.announcementBadge;
416
- if (!badge) {
417
- return null;
418
- }
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;
318
+ watch(() => coreStore.config?.singleTheme, (singleTheme) => {
319
+ if (singleTheme) {
320
+ theme.value = singleTheme;
321
+ window.localStorage.setItem('af__theme', singleTheme);
322
+ document.documentElement.classList.toggle('dark', theme.value === 'dark');
429
323
  }
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
-
324
+ }, { immediate: true })
440
325
 
441
326
  </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>