adminforth 2.4.0-next.13 → 2.4.0-next.130

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 (149) hide show
  1. package/commands/callTsProxy.js +14 -4
  2. package/commands/cli.js +12 -4
  3. package/commands/createApp/templates/custom/tsconfig.json.hbs +2 -3
  4. package/commands/createApp/templates/index.ts.hbs +10 -2
  5. package/commands/createApp/templates/package.json.hbs +1 -1
  6. package/commands/createApp/utils.js +27 -2
  7. package/commands/createCustomComponent/configLoader.js +3 -0
  8. package/dist/dataConnectors/baseConnector.d.ts.map +1 -1
  9. package/dist/dataConnectors/baseConnector.js +16 -3
  10. package/dist/dataConnectors/baseConnector.js.map +1 -1
  11. package/dist/dataConnectors/clickhouse.d.ts.map +1 -1
  12. package/dist/dataConnectors/clickhouse.js +4 -0
  13. package/dist/dataConnectors/clickhouse.js.map +1 -1
  14. package/dist/dataConnectors/mongo.d.ts.map +1 -1
  15. package/dist/dataConnectors/mongo.js +14 -14
  16. package/dist/dataConnectors/mongo.js.map +1 -1
  17. package/dist/index.d.ts +2 -1
  18. package/dist/index.d.ts.map +1 -1
  19. package/dist/index.js +20 -9
  20. package/dist/index.js.map +1 -1
  21. package/dist/modules/codeInjector.d.ts.map +1 -1
  22. package/dist/modules/codeInjector.js +22 -5
  23. package/dist/modules/codeInjector.js.map +1 -1
  24. package/dist/modules/configValidator.d.ts.map +1 -1
  25. package/dist/modules/configValidator.js +53 -2
  26. package/dist/modules/configValidator.js.map +1 -1
  27. package/dist/modules/restApi.d.ts.map +1 -1
  28. package/dist/modules/restApi.js +147 -25
  29. package/dist/modules/restApi.js.map +1 -1
  30. package/dist/modules/styles.d.ts +457 -13
  31. package/dist/modules/styles.d.ts.map +1 -1
  32. package/dist/modules/styles.js +513 -31
  33. package/dist/modules/styles.js.map +1 -1
  34. package/dist/modules/utils.d.ts +1 -0
  35. package/dist/modules/utils.d.ts.map +1 -1
  36. package/dist/modules/utils.js +9 -0
  37. package/dist/modules/utils.js.map +1 -1
  38. package/dist/spa/index.html +1 -1
  39. package/dist/spa/src/App.vue +33 -15
  40. package/dist/spa/src/adminforth.ts +15 -6
  41. package/dist/spa/src/afcl/BarChart.vue +2 -2
  42. package/dist/spa/src/afcl/Button.vue +6 -6
  43. package/dist/spa/src/afcl/Checkbox.vue +21 -13
  44. package/dist/spa/src/afcl/CountryFlag.vue +4 -1
  45. package/dist/spa/src/{components/CustomDatePicker.vue → afcl/DatePicker.vue} +95 -9
  46. package/dist/spa/src/afcl/Dialog.vue +44 -27
  47. package/dist/spa/src/afcl/Dropzone.vue +12 -12
  48. package/dist/spa/src/afcl/Input.vue +9 -7
  49. package/dist/spa/src/afcl/JsonViewer.vue +25 -0
  50. package/dist/spa/src/afcl/Link.vue +1 -1
  51. package/dist/spa/src/afcl/LinkButton.vue +2 -2
  52. package/dist/spa/src/afcl/PieChart.vue +5 -5
  53. package/dist/spa/src/afcl/ProgressBar.vue +7 -7
  54. package/dist/spa/src/afcl/Select.vue +68 -34
  55. package/dist/spa/src/afcl/Skeleton.vue +6 -6
  56. package/dist/spa/src/afcl/Table.vue +130 -39
  57. package/dist/spa/src/afcl/Textarea.vue +31 -0
  58. package/dist/spa/src/afcl/Toggle.vue +32 -0
  59. package/dist/spa/src/afcl/Tooltip.vue +2 -3
  60. package/dist/spa/src/afcl/VerticalTabs.vue +3 -3
  61. package/dist/spa/src/afcl/index.ts +4 -3
  62. package/dist/spa/src/components/AcceptModal.vue +7 -7
  63. package/dist/spa/src/components/Breadcrumbs.vue +5 -5
  64. package/dist/spa/src/components/ColumnValueInput.vue +37 -18
  65. package/dist/spa/src/components/ColumnValueInputWrapper.vue +4 -3
  66. package/dist/spa/src/components/CustomDateRangePicker.vue +9 -8
  67. package/dist/spa/src/components/CustomRangePicker.vue +37 -8
  68. package/dist/spa/src/components/Filters.vue +83 -37
  69. package/dist/spa/src/components/GroupsTable.vue +9 -8
  70. package/dist/spa/src/components/MenuLink.vue +3 -3
  71. package/dist/spa/src/components/ResourceForm.vue +94 -51
  72. package/dist/spa/src/components/ResourceListTable.vue +78 -80
  73. package/dist/spa/src/components/ResourceListTableVirtual.vue +70 -72
  74. package/dist/spa/src/components/ShowTable.vue +17 -12
  75. package/dist/spa/src/components/SingleSkeletLoader.vue +6 -6
  76. package/dist/spa/src/components/SkeleteLoader.vue +3 -3
  77. package/dist/spa/src/components/ThreeDotsMenu.vue +73 -14
  78. package/dist/spa/src/components/Toast.vue +2 -7
  79. package/dist/spa/src/components/ValueRenderer.vue +43 -16
  80. package/dist/spa/src/controls/BoolToggle.vue +34 -0
  81. package/dist/spa/src/i18n.ts +1 -1
  82. package/dist/spa/src/shims-vue.d.ts +5 -0
  83. package/dist/spa/src/spa_types/core.ts +8 -1
  84. package/dist/spa/src/stores/core.ts +1 -1
  85. package/dist/spa/src/stores/modal.ts +6 -1
  86. package/dist/spa/src/types/Back.ts +93 -21
  87. package/dist/spa/src/types/Common.ts +44 -31
  88. package/dist/spa/src/types/FrontendAPI.ts +8 -0
  89. package/dist/spa/src/types/adapters/CompletionAdapter.ts +25 -0
  90. package/dist/spa/src/types/adapters/EmailAdapter.ts +27 -0
  91. package/dist/spa/src/types/adapters/ImageGenerationAdapter.ts +50 -0
  92. package/dist/spa/src/types/adapters/ImageVisionAdapter.ts +30 -0
  93. package/dist/spa/src/types/adapters/OAuth2Adapter.ts +34 -0
  94. package/dist/spa/src/types/adapters/StorageAdapter.ts +73 -0
  95. package/dist/spa/src/types/adapters/index.ts +6 -0
  96. package/dist/spa/src/utils.ts +217 -7
  97. package/dist/spa/src/views/CreateView.vue +18 -19
  98. package/dist/spa/src/views/EditView.vue +25 -19
  99. package/dist/spa/src/views/ListView.vue +117 -79
  100. package/dist/spa/src/views/LoginView.vue +34 -35
  101. package/dist/spa/src/views/ResourceParent.vue +1 -1
  102. package/dist/spa/src/views/ShowView.vue +59 -39
  103. package/dist/spa/src/websocket.ts +6 -1
  104. package/dist/spa/tsconfig.app.json +1 -1
  105. package/dist/spa/vite.config.ts +45 -2
  106. package/dist/types/Back.d.ts +69 -14
  107. package/dist/types/Back.d.ts.map +1 -1
  108. package/dist/types/Back.js +3 -0
  109. package/dist/types/Back.js.map +1 -1
  110. package/dist/types/Common.d.ts +37 -28
  111. package/dist/types/Common.d.ts.map +1 -1
  112. package/dist/types/Common.js.map +1 -1
  113. package/dist/types/FrontendAPI.d.ts +7 -0
  114. package/dist/types/FrontendAPI.d.ts.map +1 -1
  115. package/dist/types/FrontendAPI.js.map +1 -1
  116. package/dist/types/adapters/CompletionAdapter.d.ts +20 -0
  117. package/dist/types/adapters/CompletionAdapter.d.ts.map +1 -0
  118. package/dist/types/adapters/CompletionAdapter.js +2 -0
  119. package/dist/types/adapters/CompletionAdapter.js.map +1 -0
  120. package/dist/types/adapters/EmailAdapter.d.ts +20 -0
  121. package/dist/types/adapters/EmailAdapter.d.ts.map +1 -0
  122. package/dist/types/adapters/EmailAdapter.js +2 -0
  123. package/dist/types/adapters/EmailAdapter.js.map +1 -0
  124. package/dist/types/adapters/ImageGenerationAdapter.d.ts +37 -0
  125. package/dist/types/adapters/ImageGenerationAdapter.d.ts.map +1 -0
  126. package/dist/types/adapters/ImageGenerationAdapter.js +2 -0
  127. package/dist/types/adapters/ImageGenerationAdapter.js.map +1 -0
  128. package/dist/types/adapters/ImageVisionAdapter.d.ts +25 -0
  129. package/dist/types/adapters/ImageVisionAdapter.d.ts.map +1 -0
  130. package/dist/types/adapters/ImageVisionAdapter.js +2 -0
  131. package/dist/types/adapters/ImageVisionAdapter.js.map +1 -0
  132. package/dist/types/adapters/OAuth2Adapter.d.ts +32 -0
  133. package/dist/types/adapters/OAuth2Adapter.d.ts.map +1 -0
  134. package/dist/types/adapters/OAuth2Adapter.js +2 -0
  135. package/dist/types/adapters/OAuth2Adapter.js.map +1 -0
  136. package/dist/types/adapters/StorageAdapter.d.ts +63 -0
  137. package/dist/types/adapters/StorageAdapter.d.ts.map +1 -0
  138. package/dist/types/adapters/StorageAdapter.js +2 -0
  139. package/dist/types/adapters/StorageAdapter.js.map +1 -0
  140. package/dist/types/adapters/index.d.ts +7 -0
  141. package/dist/types/adapters/index.d.ts.map +1 -0
  142. package/dist/types/adapters/index.js +2 -0
  143. package/dist/types/adapters/index.js.map +1 -0
  144. package/package.json +2 -2
  145. package/dist/spa/src/types/Adapters.ts +0 -213
  146. package/dist/types/Adapters.d.ts +0 -168
  147. package/dist/types/Adapters.d.ts.map +0 -1
  148. package/dist/types/Adapters.js +0 -2
  149. package/dist/types/Adapters.js.map +0 -1
@@ -1,10 +1,10 @@
1
1
  <template>
2
- <div class="relative flex items-center justify-center min-h-screen bg-gray-100 dark:bg-gray-800 relative w-screen h-screen"
2
+ <div class="relative flex items-center justify-center min-h-screen bg-lightHtml dark:bg-darkHtml w-screen h-screen"
3
3
  :style="coreStore.config?.loginBackgroundImage && backgroundPosition === 'over' ? {
4
4
  'background-image': 'url(' + loadFile(coreStore.config?.loginBackgroundImage) + ')',
5
5
  'background-size': 'cover',
6
6
  'background-position': 'center',
7
- 'background-blend-mode': 'darken'
7
+ 'background-blend-mode': coreStore.config?.removeBackgroundBlendMode ? 'normal' : 'darken'
8
8
  }: {}"
9
9
  >
10
10
 
@@ -27,19 +27,19 @@
27
27
  overflow-x-hidden z-50 min-w-[350px] justify-center items-center md:inset-0 h-[calc(100%-1rem)] max-h-full">
28
28
  <div class="relative p-4 w-full max-h-full max-w-[400px]">
29
29
  <!-- Modal content -->
30
- <div class="af-login-modal-content relative bg-white rounded-lg shadow dark:bg-gray-700 dark:shadow-black" >
30
+ <div class="af-login-modal-content relative bg-lightLoginViewBackground rounded-lg shadow dark:bg-darkLoginViewBackground dark:shadow-black" >
31
31
  <!-- Modal header -->
32
- <div class="af-login-modal-header flex items-center justify-between p-4 md:p-5 border-b rounded-t dark:border-gray-600">
32
+ <div class="af-login-modal-header flex items-center justify-between flex-col p-4 md:p-5 border-b rounded-t dark:border-gray-600">
33
33
 
34
- <div v-if="coreStore?.config?.loginPageInjections?.panelHeader.length > 0">
34
+ <template v-if="coreStore?.config?.loginPageInjections?.panelHeader.length > 0">
35
35
  <component
36
36
  v-for="(c, index) in coreStore?.config?.loginPageInjections?.panelHeader || []"
37
37
  :key="index"
38
38
  :is="getCustomComponent(c)"
39
39
  :meta="c.meta"
40
40
  />
41
- </div>
42
- <h3 v-else class="text-xl font-semibold text-gray-900 dark:text-white">
41
+ </template>
42
+ <h3 v-else class="text-xl font-semibold text-lightLoginViewText dark:text-darkLoginViewTextColor">
43
43
  {{ $t('Sign in to') }} {{ coreStore.config?.brandName }}
44
44
  </h3>
45
45
  </div>
@@ -47,28 +47,28 @@
47
47
  <div class="af-login-modal-body p-4 md:p-5">
48
48
  <form class="space-y-4" @submit.prevent>
49
49
  <div>
50
- <label for="username" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">{{ $t('Your') }} {{ coreStore.config?.usernameFieldName?.toLowerCase() }}</label>
50
+ <label for="username" class="block mb-2 text-sm font-medium text-lightLoginViewText dark:text-darkLoginViewTextColor">{{ $t('Your') }} {{ coreStore.config?.usernameFieldName?.toLowerCase() }}</label>
51
51
  <Input
52
+ v-model="username"
52
53
  autocomplete="username"
53
54
  type="username"
54
55
  name="username"
55
56
  id="username"
56
57
  ref="usernameInput"
57
- oninput="setCustomValidity('')"
58
58
  @keydown.enter="passwordInput.focus()"
59
59
  class="w-full"
60
60
  placeholder="name@company.com" required />
61
61
  </div>
62
62
  <div class="">
63
- <label for="password" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">{{ $t('Your password') }}</label>
63
+ <label for="password" class="block mb-2 text-sm font-medium text-lightLoginViewText dark:text-darkLoginViewTextColor">{{ $t('Your password') }}</label>
64
64
  <Input
65
+ v-model="password"
65
66
  ref="passwordInput"
66
67
  autocomplete="current-password"
67
- oninput="setCustomValidity('')"
68
68
  @keydown.enter="login"
69
69
  :type="!showPw ? 'password': 'text'" name="password" id="password" placeholder="••••••••" class="w-full" required>
70
70
  <template #rightIcon>
71
- <button type="button" @click="showPw = !showPw" class="text-gray-400 dark:text-gray-300">
71
+ <button type="button" @click="showPw = !showPw" class="text-lightLoginViewSubTextColor dark:text-darkLoginViewSubTextColor">
72
72
  <IconEyeSolid class="w-5 h-5" v-if="!showPw" />
73
73
  <IconEyeSlashSolid class="w-5 h-5" v-else />
74
74
  </button>
@@ -101,15 +101,15 @@
101
101
  {{ error }}
102
102
  </div>
103
103
  </div>
104
-
105
- <div v-if="coreStore.config?.loginPromptHTML"
106
- class="flex items-center p-4 mb-4 text-sm text-gray-800 rounded-lg bg-gray-50 dark:bg-gray-800 dark:text-gray-400" role="alert"
104
+
105
+ <div v-if="loginPromptHTML"
106
+ class="flex items-center p-4 mb-4 text-sm text-lightLoginViewPromptText rounded-lg bg-lightLoginViewPromptBackground dark:bg-darkLoginViewPromptBackground dark:text-darkLoginViewPromptText" role="alert"
107
107
  >
108
108
  <svg class="flex-shrink-0 inline w-4 h-4 me-3" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 20 20">
109
109
  <path d="M10 .5a9.5 9.5 0 1 0 9.5 9.5A9.51 9.51 0 0 0 10 .5ZM9.5 4a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM12 15H8a1 1 0 0 1 0-2h1v-3H8a1 1 0 0 1 0-2h2a1 1 0 0 1 1 1v4h1a1 1 0 0 1 0 2Z"/>
110
110
  </svg>
111
111
  <span class="sr-only">{{ $t('Info') }}</span>
112
- <div v-html="coreStore.config?.loginPromptHTML"></div>
112
+ <div v-html="loginPromptHTML"></div>
113
113
  </div>
114
114
  <Button @click="login" :loader="inProgress" :disabled="inProgress" class="w-full">
115
115
  {{ $t('Login to your account') }}
@@ -142,11 +142,13 @@ const { t } = useI18n();
142
142
  const passwordInput = ref(null);
143
143
  const usernameInput = ref(null);
144
144
  const rememberMeValue= ref(false);
145
+ const username = ref('');
146
+ const password = ref('');
145
147
 
146
148
  const route = useRoute();
147
149
  const router = useRouter();
148
150
  const inProgress = ref(false);
149
-
151
+ const loginPromptHTML = ref()
150
152
  const coreStore = useCoreStore();
151
153
  const user = useUserStore();
152
154
 
@@ -158,6 +160,15 @@ const backgroundPosition = computed(() => {
158
160
  return coreStore.config?.loginBackgroundPosition || '1/2';
159
161
  });
160
162
 
163
+
164
+ async function getLoginFormConfig() {
165
+ const response = await callAdminForthApi({
166
+ path: '/get_login_form_config',
167
+ method: 'GET',
168
+ });
169
+ loginPromptHTML.value = response.loginPromptHTML;
170
+ }
171
+
161
172
  onBeforeMount(() => {
162
173
  if (localStorage.getItem('isAuthorized') === 'true') {
163
174
  // if route has next param, redirect
@@ -171,29 +182,17 @@ onBeforeMount(() => {
171
182
  })
172
183
 
173
184
  onMounted(async () => {
185
+ getLoginFormConfig();
174
186
  if (coreStore.config?.demoCredentials) {
175
- const [username, password] = coreStore.config.demoCredentials.split(':');
176
- usernameInput.value.value = username;
177
- passwordInput.value.value = password;
187
+ const [demoUsername, demoPassword] = coreStore.config.demoCredentials.split(':');
188
+ username.value = demoUsername;
189
+ password.value = demoPassword;
178
190
  }
179
191
  usernameInput.value.focus();
180
192
  });
181
193
 
182
194
 
183
195
  async function login() {
184
-
185
- const username = usernameInput.value.value;
186
- const password = passwordInput.value.value;
187
-
188
- if (!username) {
189
- usernameInput.value.setCustomValidity(t('Please fill out this field.'));
190
- return;
191
- }
192
- if (!password) {
193
- passwordInput.value.setCustomValidity(t('Please fill out this field.'));
194
- return;
195
- }
196
-
197
196
  if (inProgress.value) {
198
197
  return;
199
198
  }
@@ -202,8 +201,8 @@ async function login() {
202
201
  path: '/login',
203
202
  method: 'POST',
204
203
  body: {
205
- username,
206
- password,
204
+ username: username.value,
205
+ password: password.value,
207
206
  rememberMe: rememberMeValue.value,
208
207
  }
209
208
  });
@@ -1,5 +1,5 @@
1
1
  <template>
2
- <div :key="`${$route?.params.resourceId}---${$route?.params.primaryKey}`" class="p-4 flex"
2
+ <div :key="`${$route?.params.resourceId}---${$route?.params.primaryKey}`" class="af-resource-parent p-4 flex"
3
3
  :class="limitHeightToPage ? 'h-[calc(100dvh-3.5rem)]': undefined"
4
4
  >
5
5
  <RouterView/>
@@ -4,7 +4,7 @@
4
4
  v-if="!loading"
5
5
  v-for="c in coreStore?.resourceOptions?.pageInjections?.show?.beforeBreadcrumbs || []"
6
6
  :is="getCustomComponent(c)"
7
- :meta="c.meta"
7
+ :meta="(c as AdminForthComponentDeclarationFull).meta"
8
8
  :record="coreStore.record"
9
9
  :resource="coreStore.resource"
10
10
  :adminUser="coreStore.adminUser"
@@ -15,7 +15,7 @@
15
15
  v-for="action in coreStore.resource.options.actions.filter(a => a.showIn?.showButton)"
16
16
  :key="action.id"
17
17
  @click="startCustomAction(action.id)"
18
- :disabled="actionLoadingStates[action.id]"
18
+ :disabled="actionLoadingStates[action.id!]"
19
19
  class="flex items-center py-1 px-3 text-sm font-medium text-gray-900 focus:outline-none bg-white rounded-default border border-gray-300 hover:bg-gray-100 hover:text-lightPrimary focus:z-10 focus:ring-4 focus:ring-gray-100 dark:focus:ring-gray-700 dark:bg-gray-800 dark:text-gray-400 dark:border-gray-600 dark:hover:text-white dark:hover:bg-gray-700"
20
20
  >
21
21
  <component
@@ -28,28 +28,28 @@
28
28
  </template>
29
29
  <RouterLink v-if="coreStore.resource?.options?.allowedActions?.create"
30
30
  :to="{ name: 'resource-create', params: { resourceId: $route.params.resourceId } }"
31
- class="flex items-center py-1 px-3 text-sm font-medium text-gray-900 focus:outline-none bg-white rounded border border-gray-300 hover:bg-gray-100 hover:text-lightPrimary focus:z-10 focus:ring-4 focus:ring-gray-100 dark:focus:ring-gray-700 dark:bg-gray-800 dark:text-gray-400 dark:border-gray-600 dark:hover:text-white dark:hover:bg-gray-700 rounded-default"
31
+ class="af-add-new-button flex items-center py-1 px-3 text-sm font-medium text-lightShowViewButtonText focus:outline-none bg-lightShowViewButtonBackground rounded border border-lightShowViewButtonBorder hover:bg-lightShowViewButtonBackgroundHover hover:text-lightShowViewButtonTextHover focus:z-10 focus:ring-4 focus:ring-lightShowViewButtonFocusRing dark:focus:ring-darkShowViewButtonFocusRing dark:bg-darkShowViewButtonBackground dark:text-darkShowViewButtonText dark:border-darkShowViewButtonBorder dark:hover:text-darkShowViewButtonTextHover dark:hover:bg-darkShowViewButtonBackgroundHover rounded-default gap-1"
32
32
  >
33
- <IconPlusOutline class="w-4 h-4 me-2"/>
33
+ <IconPlusOutline class="w-4 h-4"/>
34
34
  {{ $t('Add new') }}
35
35
  </RouterLink>
36
36
 
37
37
  <RouterLink v-if="coreStore?.resourceOptions?.allowedActions?.edit" :to="{ name: 'resource-edit', params: { resourceId: $route.params.resourceId, primaryKey: $route.params.primaryKey } }"
38
- class="flex items-center py-1 px-3 text-sm font-medium text-gray-900 focus:outline-none bg-white rounded-default border border-gray-300 hover:bg-gray-100 hover:text-lightPrimary focus:z-10 focus:ring-4 focus:ring-gray-100 dark:focus:ring-gray-700 dark:bg-gray-800 dark:text-gray-400 dark:border-gray-600 dark:hover:text-white dark:hover:bg-gray-700"
38
+ class="flex items-center af-edit-button py-1 px-3 text-sm font-medium text-lightShowViewButtonText focus:outline-none bg-lightShowViewButtonBackground rounded-default border border-lightShowViewButtonBorder hover:bg-lightShowViewButtonBackgroundHover hover:text-lightShowViewButtonTextHover focus:z-10 focus:ring-4 focus:ring-lightShowViewButtonFocusRing dark:focus:ring-darkShowViewButtonFocusRing dark:bg-darkShowViewButtonBackground dark:text-darkShowViewButtonText dark:border-darkShowViewButtonBorder dark:hover:text-darkShowViewButtonTextHover dark:hover:bg-darkShowViewButtonBackgroundHover gap-1"
39
39
  >
40
40
  <IconPenSolid class="w-4 h-4" />
41
41
  {{ $t('Edit') }}
42
42
  </RouterLink>
43
43
 
44
44
  <button v-if="coreStore?.resourceOptions?.allowedActions?.delete" @click="deleteRecord"
45
- class="flex items-center py-1 px-3 text-sm font-medium rounded-default text-red-600 focus:outline-none bg-white border border-gray-300 hover:bg-gray-100 hover:text-red-700 focus:z-10 focus:ring-4 focus:ring-gray-100 dark:focus:ring-gray-700 dark:bg-gray-800 dark:text-red-500 dark:border-gray-600 dark:hover:text-white dark:hover:bg-gray-700"
45
+ class="flex items-center af-delete-button py-1 px-3 text-sm font-medium rounded-default text-red-600 focus:outline-none bg-lightShowViewButtonBackground border border-lightShowViewButtonBorder hover:bg-lightShowViewButtonBackgroundHover hover:text-red-700 focus:z-10 focus:ring-4 focus:ring-lightShowViewButtonFocusRing dark:focus:ring-darkShowViewButtonFocusRing dark:bg-darkShowViewButtonBackground dark:text-red-500 dark:border-darkShowViewButtonBorder dark:hover:text-darkShowViewButtonTextHover dark:hover:bg-darkShowViewButtonBackgroundHover gap-1"
46
46
  >
47
47
  <IconTrashBinSolid class="w-4 h-4" />
48
48
  {{ $t('Delete') }}
49
49
  </button>
50
50
 
51
51
  <ThreeDotsMenu
52
- :threeDotsDropdownItems="coreStore.resourceOptions?.pageInjections?.show?.threeDotsDropdownItems"
52
+ :threeDotsDropdownItems="(coreStore.resourceOptions?.pageInjections?.show?.threeDotsDropdownItems as [])"
53
53
  :customActions="customActions"
54
54
  ></ThreeDotsMenu>
55
55
  </BreadcrumbsWithButtons>
@@ -57,7 +57,7 @@
57
57
  <component
58
58
  v-for="c in coreStore?.resourceOptions?.pageInjections?.show?.afterBreadcrumbs || []"
59
59
  :is="getCustomComponent(c)"
60
- :meta="c.meta"
60
+ :meta="(c as AdminForthComponentDeclarationFull).meta"
61
61
  :record="coreStore.record"
62
62
  :resource="coreStore.resource"
63
63
  :adminUser="coreStore.adminUser"
@@ -76,11 +76,11 @@
76
76
  v-else-if="coreStore.record"
77
77
  class="relative w-full flex flex-col gap-4"
78
78
  >
79
- <div v-if="!groups.length && allColumns.length">
79
+ <div v-if="!groups.length && allColumns?.length">
80
80
  <ShowTable
81
- :columns="allColumns"
82
81
  :resource="coreStore.resource"
83
82
  :record="coreStore.record"
83
+ :columns="allColumns as Array<{ name: string; label?: string; components?: any }>"
84
84
  />
85
85
  </div>
86
86
  <template v-else>
@@ -93,12 +93,12 @@
93
93
  :record="coreStore.record"
94
94
  />
95
95
  </template>
96
- <template v-if="otherColumns.length > 0">
96
+ <template v-if="otherColumns && otherColumns.length > 0">
97
97
  <ShowTable
98
- :columns="otherColumns"
99
98
  groupName="Other Fields"
100
99
  :resource="coreStore.resource"
101
100
  :record="coreStore.record"
101
+ :columns="otherColumns as Array<{ name: string; label?: string; components?: any }>"
102
102
  />
103
103
  </template>
104
104
  </template>
@@ -112,8 +112,7 @@
112
112
  v-if="!loading"
113
113
  v-for="c in coreStore?.resourceOptions?.pageInjections?.show?.bottom || []"
114
114
  :is="getCustomComponent(c)"
115
- :meta="c.meta"
116
- :column="column"
115
+ :meta="(c as AdminForthComponentDeclarationFull).meta"
117
116
  :record="coreStore.record"
118
117
  :resource="coreStore.resource"
119
118
  :adminUser="coreStore.adminUser"
@@ -140,6 +139,7 @@ import ShowTable from '@/components/ShowTable.vue';
140
139
  import adminforth from "@/adminforth";
141
140
  import { useI18n } from 'vue-i18n';
142
141
  import { getIcon } from '@/utils';
142
+ import { type AdminForthComponentDeclarationFull, type AdminForthResourceColumnCommon, type FieldGroup } from '@/types/Common.js';
143
143
 
144
144
  const route = useRoute();
145
145
  const router = useRouter();
@@ -147,62 +147,65 @@ const loading = ref(true);
147
147
  const { t } = useI18n();
148
148
  const coreStore = useCoreStore();
149
149
 
150
- const actionLoadingStates = ref({});
150
+ const actionLoadingStates = ref<Record<string, boolean>>({});
151
151
 
152
152
  const customActions = computed(() => {
153
- return coreStore.resource?.options?.actions?.filter(a => a.showIn?.showThreeDotsMenu) || [];
153
+ return coreStore.resource?.options?.actions?.filter((a: any) => a.showIn?.showThreeDotsMenu) || [];
154
154
  });
155
155
 
156
156
  onMounted(async () => {
157
157
  loading.value = true;
158
158
  await coreStore.fetchResourceFull({
159
- resourceId: route.params.resourceId
159
+ resourceId: route.params.resourceId as string,
160
160
  });
161
161
  initThreeDotsDropdown();
162
162
  await coreStore.fetchRecord({
163
- resourceId: route.params.resourceId,
164
- primaryKey: route.params.primaryKey,
163
+ resourceId: route.params.resourceId as string,
164
+ primaryKey: route.params.primaryKey as string,
165
165
  source: 'show',
166
166
  });
167
- checkAcessByAllowedActions(coreStore.resourceOptions.allowedActions,'show');
167
+ if(coreStore.resourceOptions){
168
+ checkAcessByAllowedActions(coreStore.resourceOptions.allowedActions,'show');
169
+ }
168
170
  loading.value = false;
169
171
  });
170
172
 
171
173
  const groups = computed(() => {
172
174
  let fieldGroupType;
173
- if (coreStore.resource.options?.showFieldGroups) {
174
- fieldGroupType = coreStore.resource.options.showFieldGroups;
175
- } else if (coreStore.resource.options?.showFieldGroups === null) {
176
- fieldGroupType = [];
177
- } else {
178
- fieldGroupType = coreStore.resource.options?.fieldGroups;
175
+ if (coreStore.resource) {
176
+ if (coreStore.resource.options?.showFieldGroups) {
177
+ fieldGroupType = coreStore.resource.options.showFieldGroups;
178
+ } else if (coreStore.resource.options?.showFieldGroups === null) {
179
+ fieldGroupType = [];
180
+ } else {
181
+ fieldGroupType = coreStore.resource.options?.fieldGroups;
182
+ }
179
183
  }
184
+ const activeGroups: typeof fieldGroupType | [] = fieldGroupType ?? [];
180
185
 
181
- const activeGroups = fieldGroupType ?? [];
182
-
183
- return activeGroups.map(group => ({
186
+ return activeGroups.map((group: FieldGroup) => ({
184
187
  ...group,
185
- columns: coreStore.resource.columns.filter(
186
- col => group.columns.includes(col.name) && col.showIn.show
188
+ columns: coreStore.resource?.columns.filter(
189
+ col => group.columns.includes(col.name) && col.showIn?.show
187
190
  ),
188
191
  }));
189
192
  });
190
193
 
191
194
  const allColumns = computed(() => {
192
- return coreStore.resource.columns.filter(col => col.showIn.show);
195
+ return coreStore.resource?.columns.filter(col => col.showIn?.show);
193
196
  });
194
197
 
195
198
  const otherColumns = computed(() => {
196
199
  const groupedColumnNames = new Set(
197
- groups.value.flatMap(group => group.columns.map(col => col.name))
200
+ groups.value.flatMap(group => group.columns.map((col: AdminForthResourceColumnCommon) => col.name))
198
201
  );
199
202
 
200
- return coreStore.resource.columns.filter(
201
- col => !groupedColumnNames.has(col.name) && col.showIn.show
203
+ return coreStore.resource?.columns.filter(
204
+ col => !groupedColumnNames.has(col.name) && col.showIn?.show
202
205
  );
203
206
  });
204
207
 
205
- async function deleteRecord(row) {
208
+ async function deleteRecord() {
206
209
  const data = await adminforth.confirm({
207
210
  message: t('Are you sure you want to delete this item?'),
208
211
  yes: t('Delete'),
@@ -231,7 +234,7 @@ async function deleteRecord(row) {
231
234
 
232
235
  }
233
236
 
234
- async function startCustomAction(actionId) {
237
+ async function startCustomAction(actionId: string) {
235
238
  actionLoadingStates.value[actionId] = true;
236
239
 
237
240
  const data = await callAdminForthApi({
@@ -263,8 +266,8 @@ async function startCustomAction(actionId) {
263
266
 
264
267
  if (data?.ok) {
265
268
  await coreStore.fetchRecord({
266
- resourceId: route.params.resourceId,
267
- primaryKey: route.params.primaryKey,
269
+ resourceId: route.params.resourceId as string,
270
+ primaryKey: route.params.primaryKey as string,
268
271
  source: 'show',
269
272
  });
270
273
 
@@ -281,4 +284,21 @@ async function startCustomAction(actionId) {
281
284
  }
282
285
  }
283
286
 
287
+ adminforth.show.refresh = () => {
288
+ (async () => {
289
+ try {
290
+ loading.value = true;
291
+ await coreStore.fetchRecord({
292
+ resourceId: String(route.params.resourceId),
293
+ primaryKey: String(route.params.primaryKey),
294
+ source: 'show',
295
+ });
296
+ } catch (e) {
297
+ showErrorTost((e as Error).message);
298
+ } finally {
299
+ loading.value = false;
300
+ }
301
+ })();
302
+ }
303
+
284
304
  </script>
@@ -1,8 +1,13 @@
1
1
 
2
2
  const subscriptions: { [topic: string]: ((data: any) => void)[] } = {};
3
+
4
+ interface ExtendedWebSocket extends WebSocket {
5
+ connected?: boolean;
6
+ }
7
+
3
8
  const state: {
4
9
  status: 'connecting' | 'connected' | 'disconnected';
5
- ws: WebSocket | null;
10
+ ws: ExtendedWebSocket | null;
6
11
  } = {
7
12
  status: 'connecting',
8
13
  ws: null
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "extends": "@vue/tsconfig/tsconfig.dom.json",
3
- "include": ["env.d.ts", "src/**/*", "src/**/*.vue"],
3
+ "include": ["env.d.ts", "src/**/*", "src/**/*.vue", "src/**/*.d.ts"],
4
4
  "exclude": ["src/**/__tests__/*"],
5
5
  "compilerOptions": {
6
6
  "composite": true,
@@ -1,18 +1,60 @@
1
1
  import { fileURLToPath, URL } from 'node:url'
2
-
3
2
  import { defineConfig } from 'vite'
4
3
  import vue from '@vitejs/plugin-vue'
5
4
  import portfinder from 'portfinder';
5
+ import { Plugin } from 'vite';
6
+ import tailwindcss from 'tailwindcss';
7
+
6
8
 
7
9
  /**
8
10
  * Find the next available port after a specified port.
9
11
  * @param {number} startPort - The starting port to check.
10
12
  * @returns {Promise<number>} - A promise that resolves with the next available port.
11
13
  */
12
- async function getNextAvailablePort(startPort) {
14
+ async function getNextAvailablePort(startPort: number | undefined) {
13
15
  return await portfinder.getPortPromise({ port: startPort });
14
16
  };
15
17
 
18
+ function ignoreTailwindErrors(): Plugin {
19
+ return {
20
+ name: 'ignore-tailwind-errors',
21
+ configureServer(server) {
22
+ server.middlewares.use((req, res, next) => {
23
+ const originalWrite = res.write;
24
+ res.write = function(chunk) {
25
+ if (typeof chunk === 'string' && chunk.includes('tailwind')) {
26
+ return true;
27
+ }
28
+ return originalWrite.call(this, chunk);
29
+ };
30
+ next();
31
+ });
32
+ },
33
+ config(config, { command }) {
34
+ if (command === 'build') {
35
+ // Override PostCSS config for build
36
+ config.css = config.css || {};
37
+ config.css.postcss = {
38
+ plugins: [
39
+ {
40
+ postcssPlugin: 'ignore-tailwind-errors',
41
+ Once(root, helpers) {
42
+ try {
43
+ return tailwindcss()(root, helpers);
44
+ } catch (error: any) {
45
+ console.warn('TailwindCSS warning ignored:', error.message);
46
+ return;
47
+ }
48
+ }
49
+ }
50
+ ]
51
+ };
52
+ }
53
+ }
54
+ };
55
+ }
56
+
57
+
16
58
  const appPort = await getNextAvailablePort(5173);
17
59
  const hmrPort = await getNextAvailablePort(5273);
18
60
  console.log(`SPA port: ${appPort}. HMR port: ${hmrPort}`);
@@ -27,6 +69,7 @@ export default defineConfig({
27
69
  },
28
70
  },
29
71
  plugins: [
72
+ //ignoreTailwindErrors(),
30
73
  vue(),
31
74
  ],
32
75
  resolve: {