@effect-app/vue-components 4.0.0-beta.22 → 4.0.0-beta.220
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.
- package/README.md +13 -9
- package/dist/reset.css +39 -38
- package/dist/types/components/CommandButton.vue.d.ts +6 -4
- package/dist/types/components/OmegaForm/OmegaArray.vue.d.ts +1 -1
- package/dist/types/components/OmegaForm/OmegaAutoGen.vue.d.ts +1 -1
- package/dist/types/components/OmegaForm/OmegaErrorsInternal.vue.d.ts +1 -1
- package/dist/types/components/OmegaForm/OmegaFormInput.vue.d.ts +1 -1
- package/dist/types/components/OmegaForm/OmegaInput.vue.d.ts +1 -1
- package/dist/types/components/OmegaForm/OmegaInternalInput.vue.d.ts +2 -1
- package/dist/types/components/OmegaForm/OmegaWrapper.vue.d.ts +1 -1
- package/dist/types/components/OmegaForm/createUseFormWithCustomInput.d.ts +2 -2
- package/dist/types/components/OmegaForm/errors.d.ts +33 -0
- package/dist/types/components/OmegaForm/getOmegaStore.d.ts +1 -1
- package/dist/types/components/OmegaForm/hocs.d.ts +3 -0
- package/dist/types/components/OmegaForm/index.d.ts +13 -3
- package/dist/types/components/OmegaForm/inputs.d.ts +4 -0
- package/dist/types/components/OmegaForm/meta/checks.d.ts +4 -0
- package/dist/types/components/OmegaForm/meta/createMeta.d.ts +32 -0
- package/dist/types/components/OmegaForm/meta/defaults.d.ts +2 -0
- package/dist/types/components/OmegaForm/meta/redacted.d.ts +2 -0
- package/dist/types/components/OmegaForm/meta/types.d.ts +56 -0
- package/dist/types/components/OmegaForm/meta/walker.d.ts +18 -0
- package/dist/types/components/OmegaForm/persistency.d.ts +58 -0
- package/dist/types/components/OmegaForm/submit.d.ts +60 -0
- package/dist/types/components/OmegaForm/types.d.ts +281 -0
- package/dist/types/components/OmegaForm/useOmegaForm.d.ts +7 -213
- package/dist/types/components/OmegaForm/validation/localized.d.ts +10 -0
- package/dist/types/index.d.ts +0 -1
- package/dist/types/utils/index.d.ts +6 -6
- package/dist/vue-components.es.js +29 -44
- package/dist/vue-components10.es.js +5 -0
- package/dist/vue-components11.es.js +20 -0
- package/dist/vue-components12.es.js +49 -0
- package/dist/vue-components13.es.js +128 -0
- package/dist/vue-components14.es.js +65 -0
- package/dist/vue-components15.es.js +60 -0
- package/dist/vue-components16.es.js +22 -0
- package/dist/vue-components17.es.js +5 -0
- package/dist/vue-components18.es.js +80 -0
- package/dist/vue-components19.es.js +92 -0
- package/dist/vue-components2.es.js +11 -0
- package/dist/vue-components20.es.js +73 -0
- package/dist/vue-components21.es.js +12 -0
- package/dist/vue-components22.es.js +56 -0
- package/dist/vue-components23.es.js +5 -0
- package/dist/vue-components24.es.js +44 -0
- package/dist/vue-components25.es.js +5 -0
- package/dist/vue-components26.es.js +84 -0
- package/dist/vue-components28.es.js +8 -0
- package/dist/vue-components29.es.js +9 -0
- package/dist/vue-components3.es.js +86 -0
- package/dist/vue-components30.es.js +269 -0
- package/dist/vue-components32.es.js +8 -0
- package/dist/vue-components33.es.js +73 -0
- package/dist/vue-components34.es.js +5 -0
- package/dist/vue-components35.es.js +52 -0
- package/dist/vue-components36.es.js +5 -0
- package/dist/vue-components37.es.js +24 -0
- package/dist/vue-components38.es.js +5 -0
- package/dist/vue-components39.es.js +59 -0
- package/dist/vue-components4.es.js +5 -0
- package/dist/vue-components40.es.js +5 -0
- package/dist/vue-components41.es.js +12 -0
- package/dist/vue-components42.es.js +22 -0
- package/dist/vue-components44.es.js +9 -0
- package/dist/vue-components45.es.js +4 -0
- package/dist/vue-components46.es.js +38 -0
- package/dist/vue-components47.es.js +27 -0
- package/dist/vue-components48.es.js +28 -0
- package/dist/vue-components49.es.js +7 -0
- package/dist/vue-components5.es.js +24 -0
- package/dist/vue-components50.es.js +18 -0
- package/dist/vue-components51.es.js +36 -0
- package/dist/vue-components52.es.js +18 -0
- package/dist/vue-components53.es.js +21 -0
- package/dist/vue-components54.es.js +30 -0
- package/dist/vue-components55.es.js +7 -0
- package/dist/vue-components56.es.js +9 -0
- package/dist/vue-components57.es.js +38 -0
- package/dist/vue-components58.es.js +25 -0
- package/dist/vue-components59.es.js +128 -0
- package/dist/vue-components6.es.js +13 -0
- package/dist/vue-components60.es.js +24 -0
- package/dist/vue-components61.es.js +21 -0
- package/dist/vue-components62.es.js +9 -0
- package/dist/vue-components63.es.js +19 -0
- package/dist/vue-components64.es.js +5 -0
- package/dist/vue-components65.es.js +29 -0
- package/dist/vue-components66.es.js +5 -0
- package/dist/vue-components67.es.js +29 -0
- package/dist/vue-components68.es.js +6 -0
- package/dist/vue-components69.es.js +18 -0
- package/dist/vue-components7.es.js +13 -0
- package/dist/vue-components70.es.js +40 -0
- package/dist/vue-components71.es.js +81 -0
- package/dist/vue-components72.es.js +33 -0
- package/dist/vue-components73.es.js +19 -0
- package/dist/vue-components74.es.js +48 -0
- package/dist/vue-components8.es.js +35 -0
- package/dist/vue-components9.es.js +47 -0
- package/package.json +29 -29
- package/src/components/CommandButton.vue +55 -7
- package/src/components/OmegaForm/OmegaArray.vue +2 -4
- package/src/components/OmegaForm/OmegaAutoGen.vue +2 -1
- package/src/components/OmegaForm/OmegaErrorsInternal.vue +1 -1
- package/src/components/OmegaForm/OmegaFormInput.vue +1 -1
- package/src/components/OmegaForm/OmegaInput.vue +7 -36
- package/src/components/OmegaForm/OmegaInputVuetify.vue +5 -2
- package/src/components/OmegaForm/OmegaInternalInput.vue +12 -6
- package/src/components/OmegaForm/OmegaTaggedUnion.vue +2 -1
- package/src/components/OmegaForm/OmegaTaggedUnionInternal.vue +1 -1
- package/src/components/OmegaForm/OmegaWrapper.vue +1 -1
- package/src/components/OmegaForm/blockDialog.ts +18 -6
- package/src/components/OmegaForm/createUseFormWithCustomInput.ts +2 -1
- package/src/components/OmegaForm/errors.ts +136 -0
- package/src/components/OmegaForm/getOmegaStore.ts +1 -1
- package/src/components/OmegaForm/hocs.ts +19 -0
- package/src/components/OmegaForm/index.ts +16 -4
- package/src/components/OmegaForm/inputs.ts +22 -0
- package/src/components/OmegaForm/meta/checks.ts +81 -0
- package/src/components/OmegaForm/meta/createMeta.ts +138 -0
- package/src/components/OmegaForm/meta/defaults.ts +132 -0
- package/src/components/OmegaForm/meta/redacted.ts +66 -0
- package/src/components/OmegaForm/meta/types.ts +78 -0
- package/src/components/OmegaForm/meta/walker.ts +248 -0
- package/src/components/OmegaForm/persistency.ts +247 -0
- package/src/components/OmegaForm/submit.ts +128 -0
- package/src/components/OmegaForm/types.ts +751 -0
- package/src/components/OmegaForm/useOmegaForm.ts +58 -893
- package/src/components/OmegaForm/validation/localized.ts +202 -0
- package/src/index.ts +0 -1
- package/src/reset.css +39 -38
- package/src/utils/index.ts +10 -7
- package/dist/types/components/OmegaForm/OmegaFormStuff.d.ts +0 -159
- package/dist/types/constants/index.d.ts +0 -1
- package/dist/vue-components.es10.js +0 -239
- package/dist/vue-components.es11.js +0 -32
- package/dist/vue-components.es12.js +0 -503
- package/dist/vue-components.es13.js +0 -49
- package/dist/vue-components.es14.js +0 -4
- package/dist/vue-components.es15.js +0 -4
- package/dist/vue-components.es16.js +0 -6
- package/dist/vue-components.es17.js +0 -13
- package/dist/vue-components.es18.js +0 -57
- package/dist/vue-components.es19.js +0 -56
- package/dist/vue-components.es2.js +0 -30
- package/dist/vue-components.es20.js +0 -8
- package/dist/vue-components.es21.js +0 -8
- package/dist/vue-components.es22.js +0 -5
- package/dist/vue-components.es23.js +0 -5
- package/dist/vue-components.es24.js +0 -4
- package/dist/vue-components.es25.js +0 -4
- package/dist/vue-components.es26.js +0 -4
- package/dist/vue-components.es27.js +0 -4
- package/dist/vue-components.es28.js +0 -19
- package/dist/vue-components.es29.js +0 -13
- package/dist/vue-components.es3.js +0 -17
- package/dist/vue-components.es30.js +0 -194
- package/dist/vue-components.es32.js +0 -31
- package/dist/vue-components.es33.js +0 -6
- package/dist/vue-components.es34.js +0 -4
- package/dist/vue-components.es35.js +0 -4
- package/dist/vue-components.es36.js +0 -113
- package/dist/vue-components.es38.js +0 -9
- package/dist/vue-components.es39.js +0 -34
- package/dist/vue-components.es4.js +0 -52
- package/dist/vue-components.es41.js +0 -6
- package/dist/vue-components.es42.js +0 -25
- package/dist/vue-components.es43.js +0 -7
- package/dist/vue-components.es44.js +0 -23
- package/dist/vue-components.es45.js +0 -32
- package/dist/vue-components.es46.js +0 -24
- package/dist/vue-components.es47.js +0 -14
- package/dist/vue-components.es48.js +0 -7
- package/dist/vue-components.es49.js +0 -21
- package/dist/vue-components.es5.js +0 -52
- package/dist/vue-components.es50.js +0 -11
- package/dist/vue-components.es51.js +0 -33
- package/dist/vue-components.es52.js +0 -50
- package/dist/vue-components.es53.js +0 -28
- package/dist/vue-components.es54.js +0 -13
- package/dist/vue-components.es55.js +0 -67
- package/dist/vue-components.es56.js +0 -58
- package/dist/vue-components.es57.js +0 -19
- package/dist/vue-components.es58.js +0 -35
- package/dist/vue-components.es59.js +0 -31
- package/dist/vue-components.es6.js +0 -69
- package/dist/vue-components.es60.js +0 -44
- package/dist/vue-components.es61.js +0 -4
- package/dist/vue-components.es62.js +0 -46
- package/dist/vue-components.es63.js +0 -4
- package/dist/vue-components.es7.js +0 -83
- package/dist/vue-components.es8.js +0 -63
- package/dist/vue-components.es9.js +0 -21
- package/src/components/OmegaForm/OmegaFormStuff.ts +0 -1276
- package/src/constants/index.ts +0 -1
package/package.json
CHANGED
|
@@ -1,39 +1,39 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@effect-app/vue-components",
|
|
3
|
-
"version": "4.0.0-beta.
|
|
3
|
+
"version": "4.0.0-beta.220",
|
|
4
4
|
"peerDependencies": {
|
|
5
5
|
"@mdi/js": "^7.4.47",
|
|
6
|
-
"effect": "^4.0.0-beta.
|
|
7
|
-
"intl-messageformat": "^11.
|
|
6
|
+
"effect": "^4.0.0-beta.62",
|
|
7
|
+
"intl-messageformat": "^11.2.4",
|
|
8
8
|
"mdi-js": "^1.0.1",
|
|
9
9
|
"primeflex": "^4.0.0",
|
|
10
10
|
"primeicons": "^7.0.0",
|
|
11
|
-
"primevue": "^4.5.
|
|
12
|
-
"vue": "^3.5.
|
|
13
|
-
"vuetify": "^4.0.
|
|
11
|
+
"primevue": "^4.5.5",
|
|
12
|
+
"vue": "^3.5.34",
|
|
13
|
+
"vuetify": "^4.0.6"
|
|
14
14
|
},
|
|
15
15
|
"devDependencies": {
|
|
16
|
-
"@storybook/vue3": "^10.
|
|
17
|
-
"@storybook/vue3-vite": "^10.
|
|
18
|
-
"@types/node": "^25.
|
|
19
|
-
"@vitejs/plugin-vue": "^6.0.
|
|
20
|
-
"@vue/test-utils": "^2.4.
|
|
21
|
-
"@vueuse/core": "^14.
|
|
22
|
-
"
|
|
23
|
-
"jsdom": "^28.1.0",
|
|
16
|
+
"@storybook/vue3": "^10.3.6",
|
|
17
|
+
"@storybook/vue3-vite": "^10.3.6",
|
|
18
|
+
"@types/node": "^25.6.0",
|
|
19
|
+
"@vitejs/plugin-vue": "^6.0.6",
|
|
20
|
+
"@vue/test-utils": "^2.4.10",
|
|
21
|
+
"@vueuse/core": "^14.3.0",
|
|
22
|
+
"jsdom": "^29.1.1",
|
|
24
23
|
"rimraf": "^6.1.3",
|
|
25
|
-
"sass": "^1.
|
|
26
|
-
"storybook": "^10.
|
|
27
|
-
"
|
|
28
|
-
"
|
|
29
|
-
"vite
|
|
24
|
+
"sass": "^1.99.0",
|
|
25
|
+
"storybook": "^10.3.6",
|
|
26
|
+
"storybook-vue3-router": "^7.0.0",
|
|
27
|
+
"typescript": "~6.0.3",
|
|
28
|
+
"vite": "^8.0.10",
|
|
29
|
+
"vite-plugin-css-injected-by-js": "^5.0.1",
|
|
30
30
|
"vitepress": "^1.6.4",
|
|
31
|
-
"vitest": "^4.
|
|
32
|
-
"vue-router": "^5.0.
|
|
31
|
+
"vitest": "^4.1.5",
|
|
32
|
+
"vue-router": "^5.0.6",
|
|
33
33
|
"vue-toastification": "^2.0.0-rc.5",
|
|
34
|
-
"vue-tsc": "^3.2.
|
|
35
|
-
"vuetify": "^4.0.
|
|
36
|
-
"@effect-app/eslint-shared-config": "0.
|
|
34
|
+
"vue-tsc": "^3.2.8",
|
|
35
|
+
"vuetify": "^4.0.6",
|
|
36
|
+
"@effect-app/eslint-shared-config": "0.6.0-beta.25"
|
|
37
37
|
},
|
|
38
38
|
"files": [
|
|
39
39
|
"src",
|
|
@@ -52,13 +52,13 @@
|
|
|
52
52
|
}
|
|
53
53
|
},
|
|
54
54
|
"dependencies": {
|
|
55
|
-
"@opentelemetry/api": "^1.9.
|
|
55
|
+
"@opentelemetry/api": "^1.9.1",
|
|
56
56
|
"@tanstack/vue-form": "^1.23.5",
|
|
57
57
|
"highlight.js": "^11.11.1",
|
|
58
58
|
"mitt": "^3.0.1",
|
|
59
59
|
"vue3-highlightjs": "^1.0.5",
|
|
60
|
-
"@effect-app/vue": "4.0.0-beta.
|
|
61
|
-
"effect-app": "4.0.0-beta.
|
|
60
|
+
"@effect-app/vue": "4.0.0-beta.220",
|
|
61
|
+
"effect-app": "4.0.0-beta.220"
|
|
62
62
|
},
|
|
63
63
|
"scripts": {
|
|
64
64
|
"check": "vue-tsc",
|
|
@@ -67,10 +67,10 @@
|
|
|
67
67
|
"docs:dev": "vitepress dev docs",
|
|
68
68
|
"docs:build": "vitepress build docs",
|
|
69
69
|
"docs:serve": "vitepress serve docs",
|
|
70
|
-
"lint": "NODE_OPTIONS=--max-old-space-size=8192 eslint src stories .storybook",
|
|
70
|
+
"lint": "oxlint --quiet --type-aware src stories && NODE_OPTIONS=--max-old-space-size=8192 ESLINT_TS=1 eslint --quiet src stories .storybook && pnpm exec dprint check --config ../../dprint.jsonc .",
|
|
71
71
|
"ncu": "ncu",
|
|
72
72
|
"clean": "rm -rf dist",
|
|
73
|
-
"lint-fix": "pnpm
|
|
73
|
+
"lint-fix": "oxlint --quiet --type-aware --fix src stories && NODE_OPTIONS=--max-old-space-size=8192 ESLINT_TS=1 eslint --quiet --fix src stories .storybook && pnpm exec dprint fmt --config ../../dprint.jsonc .",
|
|
74
74
|
"storybook": "storybook dev -p 6006",
|
|
75
75
|
"build-storybook": "storybook build",
|
|
76
76
|
"test": "vitest",
|
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
<script
|
|
2
2
|
setup
|
|
3
3
|
lang="ts"
|
|
4
|
-
generic="I = never"
|
|
4
|
+
generic="I = never, RA = unknown, RE = unknown"
|
|
5
5
|
>
|
|
6
|
-
import type { CommandBase } from "@effect-app/vue"
|
|
6
|
+
import type { CommandBase, Progress } from "@effect-app/vue"
|
|
7
|
+
import type * as AsyncResult from "effect/unstable/reactivity/AsyncResult"
|
|
7
8
|
import { computed } from "vue"
|
|
8
9
|
import type { VBtn } from "vuetify/components"
|
|
9
10
|
|
|
@@ -14,11 +15,11 @@ const props = defineProps<
|
|
|
14
15
|
& (
|
|
15
16
|
| {
|
|
16
17
|
input: NoInfer<I>
|
|
17
|
-
command: CommandBase<I>
|
|
18
|
+
command: CommandBase<I, any, RA, RE>
|
|
18
19
|
empty?: boolean
|
|
19
20
|
}
|
|
20
21
|
| {
|
|
21
|
-
command: CommandBase
|
|
22
|
+
command: CommandBase<any, any, RA, RE>
|
|
22
23
|
input?: undefined
|
|
23
24
|
empty?: boolean
|
|
24
25
|
}
|
|
@@ -26,12 +27,32 @@ const props = defineProps<
|
|
|
26
27
|
& {
|
|
27
28
|
disabled?: ButtonProps["disabled"]
|
|
28
29
|
title?: string // why isn't it part of VBtnProps??
|
|
30
|
+
mapProgress?: (result: AsyncResult.AsyncResult<RA, RE>) => Progress | undefined
|
|
29
31
|
}
|
|
30
32
|
& ButtonProps
|
|
31
33
|
>()
|
|
32
34
|
|
|
33
35
|
const isDisabled = computed(() => props.command.blocked || props.disabled)
|
|
34
36
|
|
|
37
|
+
const resolvedProgress = computed(() => {
|
|
38
|
+
if (props.mapProgress) {
|
|
39
|
+
const result = props.command.result
|
|
40
|
+
return result !== undefined ? props.mapProgress(result) : undefined
|
|
41
|
+
}
|
|
42
|
+
return props.command.progress
|
|
43
|
+
})
|
|
44
|
+
|
|
45
|
+
const progressText = computed(() => {
|
|
46
|
+
const p = resolvedProgress.value
|
|
47
|
+
if (p === undefined) return undefined
|
|
48
|
+
return typeof p === "string" ? p : p.text
|
|
49
|
+
})
|
|
50
|
+
|
|
51
|
+
const progressPercentage = computed(() => {
|
|
52
|
+
const p = resolvedProgress.value
|
|
53
|
+
return typeof p === "object" && p !== null ? p.percentage : undefined
|
|
54
|
+
})
|
|
55
|
+
|
|
35
56
|
const handleClick = () => {
|
|
36
57
|
// Block execution if button is disabled
|
|
37
58
|
if (isDisabled.value) {
|
|
@@ -41,7 +62,9 @@ const handleClick = () => {
|
|
|
41
62
|
const input = ("input" in props && props.input
|
|
42
63
|
? props.input
|
|
43
64
|
: undefined) as unknown as I
|
|
44
|
-
|
|
65
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- command.handle has a generic signature mismatched by erased input type
|
|
66
|
+
const handle = props.command.handle as any
|
|
67
|
+
handle(input)
|
|
45
68
|
}
|
|
46
69
|
</script>
|
|
47
70
|
<script lang="ts">
|
|
@@ -60,13 +83,25 @@ export default {
|
|
|
60
83
|
:class="{ 'v-btn--disabled': isDisabled }"
|
|
61
84
|
@click="handleClick"
|
|
62
85
|
>
|
|
86
|
+
<template
|
|
87
|
+
v-if="progressText !== undefined"
|
|
88
|
+
#loader
|
|
89
|
+
>
|
|
90
|
+
<v-progress-circular
|
|
91
|
+
:indeterminate="progressPercentage === undefined"
|
|
92
|
+
:model-value="progressPercentage"
|
|
93
|
+
size="20"
|
|
94
|
+
width="2"
|
|
95
|
+
/>
|
|
96
|
+
<span class="ml-2">{{ progressText }}</span>
|
|
97
|
+
</template>
|
|
63
98
|
<slot
|
|
64
99
|
:loading="command.waiting"
|
|
65
100
|
:disabled="isDisabled"
|
|
66
101
|
:label="command.label"
|
|
67
102
|
:title="title ?? command.action"
|
|
68
103
|
>
|
|
69
|
-
|
|
104
|
+
{{ command.label }}
|
|
70
105
|
</slot>
|
|
71
106
|
</v-btn>
|
|
72
107
|
<v-btn
|
|
@@ -77,5 +112,18 @@ export default {
|
|
|
77
112
|
:title="title ?? command.action"
|
|
78
113
|
:class="{ 'v-btn--disabled': isDisabled }"
|
|
79
114
|
@click="handleClick"
|
|
80
|
-
|
|
115
|
+
>
|
|
116
|
+
<template
|
|
117
|
+
v-if="progressText !== undefined"
|
|
118
|
+
#loader
|
|
119
|
+
>
|
|
120
|
+
<v-progress-circular
|
|
121
|
+
:indeterminate="progressPercentage === undefined"
|
|
122
|
+
:model-value="progressPercentage"
|
|
123
|
+
size="20"
|
|
124
|
+
width="2"
|
|
125
|
+
/>
|
|
126
|
+
<span class="ml-2">{{ progressText }}</span>
|
|
127
|
+
</template>
|
|
128
|
+
</v-btn>
|
|
81
129
|
</template>
|
|
@@ -12,9 +12,7 @@
|
|
|
12
12
|
:is="form.Field"
|
|
13
13
|
v-for="(_, i) of items"
|
|
14
14
|
:key="`${name}[${Number(i)}]`"
|
|
15
|
-
:name="
|
|
16
|
-
`${name}[${Number(i)}]` as DeepKeys<From>
|
|
17
|
-
"
|
|
15
|
+
:name="`${name}[${Number(i)}]` as DeepKeys<From>"
|
|
18
16
|
>
|
|
19
17
|
<template #default="{ field: subField, state: subState }">
|
|
20
18
|
<slot
|
|
@@ -46,7 +44,7 @@
|
|
|
46
44
|
>
|
|
47
45
|
import { type DeepKeys } from "@tanstack/vue-form"
|
|
48
46
|
import { computed, onMounted, provide } from "vue"
|
|
49
|
-
import { type OmegaArrayProps } from "./
|
|
47
|
+
import { type OmegaArrayProps } from "./types"
|
|
50
48
|
|
|
51
49
|
const props = defineProps<OmegaArrayProps<From, To, Name>>()
|
|
52
50
|
|
|
@@ -22,7 +22,8 @@
|
|
|
22
22
|
import { type DeepKeys } from "@tanstack/vue-form"
|
|
23
23
|
import { Order } from "effect-app"
|
|
24
24
|
import { computed } from "vue"
|
|
25
|
-
import { type FieldMeta
|
|
25
|
+
import { type FieldMeta } from "./meta/types"
|
|
26
|
+
import { type FieldPath, type OmegaAutoGenMeta, type OmegaInputProps } from "./types"
|
|
26
27
|
|
|
27
28
|
type NewMeta = OmegaAutoGenMeta<From, To, Name>
|
|
28
29
|
|
|
@@ -102,7 +102,7 @@
|
|
|
102
102
|
import type { StandardSchemaV1Issue } from "@tanstack/vue-form"
|
|
103
103
|
import { computed, getCurrentInstance } from "vue"
|
|
104
104
|
import { useIntl } from "../../utils"
|
|
105
|
-
import { type OmegaError } from "./
|
|
105
|
+
import { type OmegaError } from "./types"
|
|
106
106
|
|
|
107
107
|
const instance = getCurrentInstance()
|
|
108
108
|
const vuetified = instance?.appContext.components["VAlert"]
|
|
@@ -17,8 +17,8 @@
|
|
|
17
17
|
import { type DeepKeys } from "@tanstack/vue-form"
|
|
18
18
|
import { inject } from "vue"
|
|
19
19
|
import type { MergedInputProps } from "./InputProps"
|
|
20
|
-
import type { BaseProps, DefaultTypeProps, OmegaInputProps } from "./OmegaFormStuff"
|
|
21
20
|
import OmegaInput from "./OmegaInput.vue"
|
|
21
|
+
import type { BaseProps, DefaultTypeProps, OmegaInputProps } from "./types"
|
|
22
22
|
import { OmegaFormKey } from "./useOmegaForm"
|
|
23
23
|
|
|
24
24
|
const form = inject(OmegaFormKey) as unknown as OmegaInputProps<
|
|
@@ -1,18 +1,14 @@
|
|
|
1
1
|
<template>
|
|
2
2
|
<component
|
|
3
3
|
:is="form.Field"
|
|
4
|
-
:key="fieldKey"
|
|
5
4
|
:name="name"
|
|
6
|
-
:validators="
|
|
7
|
-
onChange: schema,
|
|
8
|
-
...validators
|
|
9
|
-
}"
|
|
5
|
+
:validators="validators"
|
|
10
6
|
>
|
|
11
7
|
<template #default="{ field, state }">
|
|
12
8
|
<OmegaInternalInput
|
|
13
9
|
v-if="meta"
|
|
14
10
|
v-bind="{ ...$attrs, ...$props, inputClass: computedClass }"
|
|
15
|
-
:field="field"
|
|
11
|
+
:field="field as any"
|
|
16
12
|
:state="state"
|
|
17
13
|
:register="form.registerField"
|
|
18
14
|
:label="label ?? errori18n(propsName)"
|
|
@@ -40,12 +36,13 @@
|
|
|
40
36
|
lang="ts"
|
|
41
37
|
generic="From extends Record<PropertyKey, any>, To extends Record<PropertyKey, any>, Name extends DeepKeys<From>"
|
|
42
38
|
>
|
|
39
|
+
/* eslint-disable @typescript-eslint/no-explicit-any -- TanStack Form Field generic interop and slot prop typing */
|
|
43
40
|
import { type DeepKeys } from "@tanstack/vue-form"
|
|
44
41
|
import { computed, inject, type Ref, useAttrs } from "vue"
|
|
45
|
-
import {
|
|
46
|
-
import { type FieldMeta
|
|
42
|
+
import { useErrorLabel } from "./errors"
|
|
43
|
+
import { type FieldMeta } from "./meta/types"
|
|
47
44
|
import OmegaInternalInput from "./OmegaInternalInput.vue"
|
|
48
|
-
import {
|
|
45
|
+
import { type OmegaInputPropsBase } from "./types"
|
|
49
46
|
|
|
50
47
|
const props = defineProps<OmegaInputPropsBase<From, To, Name>>()
|
|
51
48
|
|
|
@@ -57,13 +54,10 @@ defineSlots<{
|
|
|
57
54
|
default?: (props: any) => any
|
|
58
55
|
}>()
|
|
59
56
|
|
|
60
|
-
defineOptions({
|
|
61
|
-
inheritAttrs: false
|
|
62
|
-
})
|
|
57
|
+
defineOptions({ inheritAttrs: false })
|
|
63
58
|
|
|
64
59
|
const attrs = useAttrs()
|
|
65
60
|
|
|
66
|
-
// Compute the class to use based on inputClass prop
|
|
67
61
|
const computedClass = computed(() => {
|
|
68
62
|
if (props.inputClass === null) return undefined
|
|
69
63
|
if (props.inputClass !== undefined) return props.inputClass
|
|
@@ -82,28 +76,5 @@ const meta = computed(() => {
|
|
|
82
76
|
return props.form.meta[propsName.value]
|
|
83
77
|
})
|
|
84
78
|
|
|
85
|
-
// Key to force Field re-mount when meta type changes (for TaggedUnion support)
|
|
86
|
-
const fieldKey = computed(() => {
|
|
87
|
-
const m = meta.value
|
|
88
|
-
if (!m) return propsName.value
|
|
89
|
-
// Include type and key constraints in the key so Field re-mounts when validation rules change
|
|
90
|
-
// Cast to any since not all FieldMeta variants have these properties
|
|
91
|
-
const fm = m as any
|
|
92
|
-
return `${propsName.value}-${fm.type}-${fm.minLength ?? ""}-${fm.maxLength ?? ""}-${fm.minimum ?? ""}-${
|
|
93
|
-
fm.maximum ?? ""
|
|
94
|
-
}`
|
|
95
|
-
})
|
|
96
|
-
|
|
97
|
-
// Call useIntl during setup to avoid issues when computed re-evaluates
|
|
98
|
-
const { trans } = useIntl()
|
|
99
|
-
|
|
100
|
-
const schema = computed(() => {
|
|
101
|
-
if (!meta.value) {
|
|
102
|
-
console.log(props.name, Object.keys(props.form.meta), props.form.meta)
|
|
103
|
-
throw new Error("Meta is undefined")
|
|
104
|
-
}
|
|
105
|
-
return generateInputStandardSchemaFromFieldMeta(meta.value, trans)
|
|
106
|
-
})
|
|
107
|
-
|
|
108
79
|
const errori18n = useErrorLabel(props.form)
|
|
109
80
|
</script>
|
|
@@ -1,7 +1,10 @@
|
|
|
1
1
|
<template>
|
|
2
2
|
<div
|
|
3
3
|
class="omega-input"
|
|
4
|
-
@focusout="
|
|
4
|
+
@focusout="(e) => {
|
|
5
|
+
$emit('blur', e)
|
|
6
|
+
field.handleBlur()
|
|
7
|
+
}"
|
|
5
8
|
@focusin="$emit('focus', $event)"
|
|
6
9
|
>
|
|
7
10
|
<component
|
|
@@ -204,8 +207,8 @@
|
|
|
204
207
|
generic="From extends Record<PropertyKey, any>, Name extends DeepKeys<From>"
|
|
205
208
|
>
|
|
206
209
|
import { type DeepKeys } from "@tanstack/vue-form"
|
|
207
|
-
import { getInputType } from "../OmegaForm/OmegaFormStuff"
|
|
208
210
|
import type { VuetifyInputProps } from "./InputProps"
|
|
211
|
+
import { getInputType } from "./inputs"
|
|
209
212
|
|
|
210
213
|
defineProps<VuetifyInputProps<From, Name>>()
|
|
211
214
|
|
|
@@ -24,11 +24,13 @@
|
|
|
24
24
|
lang="ts"
|
|
25
25
|
generic="From extends Record<PropertyKey, any>, Name extends DeepKeys<From>"
|
|
26
26
|
>
|
|
27
|
+
/* eslint-disable @typescript-eslint/no-explicit-any -- TanStack Form / Vue attrs interop */
|
|
27
28
|
import { type DeepKeys, useStore } from "@tanstack/vue-form"
|
|
28
29
|
import { computed, type ComputedRef, getCurrentInstance, useAttrs, useId, useSlots } from "vue"
|
|
29
30
|
import type { InputProps, OmegaFieldInternalApi } from "./InputProps"
|
|
30
|
-
import type {
|
|
31
|
+
import type { MetaRecord, NestedKeyOf } from "./meta/types"
|
|
31
32
|
import OmegaInputVuetify from "./OmegaInputVuetify.vue"
|
|
33
|
+
import type { FieldValidators, TypeOverride } from "./types"
|
|
32
34
|
|
|
33
35
|
defineOptions({
|
|
34
36
|
inheritAttrs: false
|
|
@@ -84,7 +86,8 @@ const id = useId()
|
|
|
84
86
|
|
|
85
87
|
const fieldApi = props.field
|
|
86
88
|
|
|
87
|
-
|
|
89
|
+
// Subscribed for side-effect: keeps component reactive to fieldApi.store changes
|
|
90
|
+
const _fieldState = useStore(fieldApi.store, (state) => state)
|
|
88
91
|
|
|
89
92
|
// Get errors from form-level fieldMeta (persists across Field re-mounts)
|
|
90
93
|
const formFieldMeta = useStore(fieldApi.form.store, (state) => state.fieldMeta)
|
|
@@ -128,6 +131,13 @@ const handleChange: OmegaFieldInternalApi<From, Name>["handleChange"] = (value)
|
|
|
128
131
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
129
132
|
: null as any
|
|
130
133
|
)
|
|
134
|
+
} else if (props.meta?.isOptionalKey) {
|
|
135
|
+
// `S.optionalKey` expects the key to be ABSENT from the submitted
|
|
136
|
+
// object, not present-with-undefined. Remove it from form state
|
|
137
|
+
// rather than setting it to `undefined`. Note: this is distinct
|
|
138
|
+
// from `required: false`, which may also just mean "empty string
|
|
139
|
+
// is valid" for unconstrained `S.String` fields.
|
|
140
|
+
props.field.form.deleteField(props.field.name)
|
|
131
141
|
} else {
|
|
132
142
|
// Keep the actual value (e.g., empty string for S.String fields)
|
|
133
143
|
props.field.handleChange(value)
|
|
@@ -135,10 +145,6 @@ const handleChange: OmegaFieldInternalApi<From, Name>["handleChange"] = (value)
|
|
|
135
145
|
} else {
|
|
136
146
|
props.field.handleChange(value)
|
|
137
147
|
}
|
|
138
|
-
|
|
139
|
-
// whenever we change the field, regardless if we set it to null, we should reset onSubmit.
|
|
140
|
-
// not sure why this is not the case in tanstack form.
|
|
141
|
-
props.field.setMeta((m) => ({ ...m, errorMap: { ...m.errorMap, onSubmit: undefined } }))
|
|
142
148
|
}
|
|
143
149
|
|
|
144
150
|
// Note: Default value normalization (converting empty strings to null/undefined for nullable fields)
|
|
@@ -8,8 +8,8 @@
|
|
|
8
8
|
import { type DeepKeys } from "@tanstack/vue-form"
|
|
9
9
|
import { computed, provide, ref, watch } from "vue"
|
|
10
10
|
import { type TaggedUnionOption } from "./InputProps"
|
|
11
|
-
import { type FieldPath } from "./OmegaFormStuff"
|
|
12
11
|
import OmegaTaggedUnionInternal from "./OmegaTaggedUnionInternal.vue"
|
|
12
|
+
import { type FieldPath } from "./types"
|
|
13
13
|
import { type useOmegaForm } from "./useOmegaForm"
|
|
14
14
|
|
|
15
15
|
const props = defineProps<{
|
|
@@ -30,6 +30,7 @@ watch(
|
|
|
30
30
|
() => {
|
|
31
31
|
const path = tagPath.value
|
|
32
32
|
// Navigate to the nested value
|
|
33
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- traversing arbitrary nested values
|
|
33
34
|
return path.split(".").reduce((acc: any, key) => acc?.[key], formValues.value) as string | null
|
|
34
35
|
},
|
|
35
36
|
(newTag) => {
|
|
@@ -30,7 +30,7 @@
|
|
|
30
30
|
import { useStore } from "@tanstack/vue-form"
|
|
31
31
|
import { usePreventClose } from "./blockDialog"
|
|
32
32
|
import { getOmegaStore } from "./getOmegaStore"
|
|
33
|
-
import { type DefaultTypeProps, type OmegaFormApi, type OmegaFormState } from "./
|
|
33
|
+
import { type DefaultTypeProps, type OmegaFormApi, type OmegaFormState } from "./types"
|
|
34
34
|
import { type OmegaFormReturn } from "./useOmegaForm"
|
|
35
35
|
|
|
36
36
|
type OmegaWrapperProps = {
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import mitt from "mitt"
|
|
2
2
|
import { inject, type InjectionKey, provide, type Ref } from "vue"
|
|
3
|
+
import { useIntl } from "../../utils"
|
|
3
4
|
import { onMountedWithCleanup } from "./onMountedWithCleanup"
|
|
4
5
|
|
|
5
6
|
export type DialogClosing = { prevent?: boolean | Promise<boolean> }
|
|
@@ -19,11 +20,19 @@ export const usePreventClose = (mkIsDirty: () => Ref<boolean>) => {
|
|
|
19
20
|
if (!bus) {
|
|
20
21
|
return
|
|
21
22
|
}
|
|
23
|
+
const { formatMessage, trans } = useIntl()
|
|
22
24
|
const isDirty = mkIsDirty()
|
|
25
|
+
const defaultMessage = "There are unsaved changes. Are you sure you want to close?"
|
|
23
26
|
onMountedWithCleanup(() => {
|
|
24
27
|
const onDialogClosing = (evt: DialogClosing) => {
|
|
25
28
|
if (isDirty.value) {
|
|
26
|
-
|
|
29
|
+
// Mirror the guard pattern in errors.ts: a custom `useIntl` mock may
|
|
30
|
+
// only provide `trans`, so fall back through trans → defaultMessage.
|
|
31
|
+
const message = formatMessage
|
|
32
|
+
? formatMessage({ id: "form.unsaved_changes_confirm", defaultMessage })
|
|
33
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- key may not be registered in the locale catalog
|
|
34
|
+
: trans?.("form.unsaved_changes_confirm" as any) ?? defaultMessage
|
|
35
|
+
if (!confirm(message)) {
|
|
27
36
|
evt.prevent = true
|
|
28
37
|
}
|
|
29
38
|
}
|
|
@@ -46,11 +55,14 @@ export const useOnClose = (close: () => void) => {
|
|
|
46
55
|
bus.emit("dialog-closing", evt)
|
|
47
56
|
if (evt.prevent) {
|
|
48
57
|
if (typeof evt.prevent === "object" && "then" in evt.prevent) {
|
|
49
|
-
evt
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
58
|
+
evt
|
|
59
|
+
.prevent
|
|
60
|
+
.then((r) => {
|
|
61
|
+
if (r) {
|
|
62
|
+
close()
|
|
63
|
+
}
|
|
64
|
+
})
|
|
65
|
+
.catch(console.error)
|
|
54
66
|
}
|
|
55
67
|
} else {
|
|
56
68
|
close()
|
|
@@ -1,8 +1,9 @@
|
|
|
1
|
+
/* eslint-disable @typescript-eslint/no-explicit-any -- TanStack Form / Vue render-fn slot interop */
|
|
1
2
|
import type { DeepKeys } from "@tanstack/vue-form"
|
|
2
3
|
import { type Component, h } from "vue"
|
|
3
4
|
import type { MergedInputProps } from "./InputProps"
|
|
4
|
-
import { type DefaultTypeProps } from "./OmegaFormStuff"
|
|
5
5
|
import OmegaInput from "./OmegaInput.vue"
|
|
6
|
+
import { type DefaultTypeProps } from "./types"
|
|
6
7
|
import { useOmegaForm } from "./useOmegaForm"
|
|
7
8
|
|
|
8
9
|
export const createUseFormWithCustomInput = <
|
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
2
|
+
|
|
3
|
+
import { type Component, computed, type ComputedRef, type ConcreteComponent, h, onUnmounted, type Ref, ref, watch } from "vue"
|
|
4
|
+
import { useIntl } from "../../utils"
|
|
5
|
+
import type { OmegaError } from "./types"
|
|
6
|
+
import type { OF } from "./useOmegaForm"
|
|
7
|
+
|
|
8
|
+
export const useErrorLabel = (form: OF<any, any>) => {
|
|
9
|
+
const { formatMessage } = useIntl()
|
|
10
|
+
const humanize = (str: string) => {
|
|
11
|
+
return str
|
|
12
|
+
.replace(/([A-Z])/g, " $1") // Add space before capital letters
|
|
13
|
+
.replace(/^./, (char) => char.toUpperCase()) // Capitalize the first letter
|
|
14
|
+
.trim() // Remove leading/trailing spaces
|
|
15
|
+
}
|
|
16
|
+
const fallback = (propsName: string) =>
|
|
17
|
+
formatMessage
|
|
18
|
+
? formatMessage({ id: `general.fields.${propsName}`, defaultMessage: humanize(propsName) })
|
|
19
|
+
: humanize(propsName)
|
|
20
|
+
const i18n = (propsName: string) =>
|
|
21
|
+
form.i18nNamespace
|
|
22
|
+
? formatMessage({ id: `${form.i18nNamespace}.fields.${propsName}`, defaultMessage: fallback(propsName) })
|
|
23
|
+
: fallback(propsName)
|
|
24
|
+
|
|
25
|
+
return i18n
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export const eHoc = (errorProps: {
|
|
29
|
+
form: OF<any, any>
|
|
30
|
+
fieldMap: Ref<Map<string, { id: string; label: string }>>
|
|
31
|
+
}) => {
|
|
32
|
+
return function FormHoc<P>(
|
|
33
|
+
WrappedComponent: Component<P>
|
|
34
|
+
): ConcreteComponent<P> {
|
|
35
|
+
return {
|
|
36
|
+
setup() {
|
|
37
|
+
const { fieldMap, form } = errorProps
|
|
38
|
+
const generalErrors = form.useStore((state) => state.errors)
|
|
39
|
+
const fieldMeta = form.useStore((state) => state.fieldMeta)
|
|
40
|
+
const errorMap = form.useStore((state) => state.errorMap)
|
|
41
|
+
|
|
42
|
+
const errorLabel = useErrorLabel(form)
|
|
43
|
+
|
|
44
|
+
const errors = computed(() => {
|
|
45
|
+
// Collect errors from fieldMeta (field-level errors for registered fields)
|
|
46
|
+
const fieldErrors = Object.entries(fieldMeta.value).reduce<OmegaError[]>((acc, [key, m]) => {
|
|
47
|
+
const fieldErrors = (m as { errors?: Array<{ message?: string }> } | undefined)?.errors ?? []
|
|
48
|
+
if (!fieldErrors.length) {
|
|
49
|
+
return acc
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
const fieldInfo = fieldMap.value.get(key)
|
|
53
|
+
if (!fieldInfo) {
|
|
54
|
+
return acc
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
acc.push({
|
|
58
|
+
label: fieldInfo.label,
|
|
59
|
+
inputId: fieldInfo.id,
|
|
60
|
+
errors: [fieldErrors[0]?.message].filter(Boolean) as string[]
|
|
61
|
+
})
|
|
62
|
+
|
|
63
|
+
return acc
|
|
64
|
+
}, [])
|
|
65
|
+
|
|
66
|
+
// Collect errors from errorMap.onDynamic / errorMap.onSubmit ONLY for fields that are NOT registered
|
|
67
|
+
// (registered fields already have their errors in fieldMeta).
|
|
68
|
+
// Our localized standard schema writes to onDynamic via validationLogic: revalidateLogic();
|
|
69
|
+
// caller-provided validators.onSubmit (via tanstackFormOptions spread) writes to onSubmit.
|
|
70
|
+
const submitErrors: OmegaError[] = []
|
|
71
|
+
const submitIssueMaps = [errorMap.value.onDynamic, errorMap.value.onSubmit].filter(
|
|
72
|
+
Boolean
|
|
73
|
+
) as unknown as Array<
|
|
74
|
+
Record<string, unknown>
|
|
75
|
+
>
|
|
76
|
+
const seenPaths = new Set<string>()
|
|
77
|
+
for (const issuesByPath of submitIssueMaps) {
|
|
78
|
+
for (const [_, issues] of Object.entries(issuesByPath)) {
|
|
79
|
+
if (Array.isArray(issues) && issues.length) {
|
|
80
|
+
for (const issue of issues) {
|
|
81
|
+
const issAny: any = issue
|
|
82
|
+
if (issAny?.path && Array.isArray(issAny.path) && issAny.path.length) {
|
|
83
|
+
const fieldPath = issAny.path.join(".")
|
|
84
|
+
if (!fieldMap.value.has(fieldPath) && !seenPaths.has(fieldPath)) {
|
|
85
|
+
seenPaths.add(fieldPath)
|
|
86
|
+
submitErrors.push({
|
|
87
|
+
label: errorLabel(fieldPath),
|
|
88
|
+
inputId: fieldPath,
|
|
89
|
+
errors: [issAny.message].filter(Boolean)
|
|
90
|
+
})
|
|
91
|
+
break
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
// Combine both error sources (no need to check for duplicates since they're mutually exclusive)
|
|
100
|
+
return [...fieldErrors, ...submitErrors]
|
|
101
|
+
})
|
|
102
|
+
|
|
103
|
+
return {
|
|
104
|
+
generalErrors,
|
|
105
|
+
errors
|
|
106
|
+
}
|
|
107
|
+
},
|
|
108
|
+
render({ errors, generalErrors }: any) {
|
|
109
|
+
return h(WrappedComponent, {
|
|
110
|
+
errors,
|
|
111
|
+
generalErrors,
|
|
112
|
+
...this.$attrs
|
|
113
|
+
}, this.$slots)
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
export const makeFieldMap = () => {
|
|
120
|
+
const fieldMap = ref(new Map<string, { label: string; id: string }>())
|
|
121
|
+
const registerField = (field: ComputedRef<{ name: string; label: string; id: string }>) => {
|
|
122
|
+
watch(field, (f) => {
|
|
123
|
+
fieldMap.value.set(f.name, { label: f.label, id: f.id })
|
|
124
|
+
}, { immediate: true })
|
|
125
|
+
onUnmounted(() => {
|
|
126
|
+
// Only delete if we still own this entry (id matches)
|
|
127
|
+
// This prevents old components from deleting entries registered by new components
|
|
128
|
+
// during re-mount transitions (e.g., when :key changes)
|
|
129
|
+
const currentEntry = fieldMap.value.get(field.value.name)
|
|
130
|
+
if (currentEntry?.id === field.value.id) {
|
|
131
|
+
fieldMap.value.delete(field.value.name)
|
|
132
|
+
}
|
|
133
|
+
})
|
|
134
|
+
}
|
|
135
|
+
return { fieldMap, registerField }
|
|
136
|
+
}
|