@narrative.io/jsonforms-provider-protocols 3.0.0-beta.2 → 3.0.0-beta.21

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 (124) hide show
  1. package/dist/core/initFormData.d.ts +17 -0
  2. package/dist/core/initFormData.d.ts.map +1 -0
  3. package/dist/core/initFormData.js +99 -0
  4. package/dist/core/initFormData.js.map +1 -0
  5. package/dist/core/projection.d.ts +4 -0
  6. package/dist/core/projection.d.ts.map +1 -1
  7. package/dist/core/projection.js +17 -14
  8. package/dist/core/projection.js.map +1 -1
  9. package/dist/core/refs.d.ts +58 -0
  10. package/dist/core/refs.d.ts.map +1 -0
  11. package/dist/core/refs.js +70 -0
  12. package/dist/core/refs.js.map +1 -0
  13. package/dist/core/resolveScope.d.ts +6 -0
  14. package/dist/core/resolveScope.d.ts.map +1 -1
  15. package/dist/core/resolveScope.js +14 -8
  16. package/dist/core/resolveScope.js.map +1 -1
  17. package/dist/core/transforms.d.ts.map +1 -1
  18. package/dist/core/transforms.js +3 -1
  19. package/dist/core/transforms.js.map +1 -1
  20. package/dist/core/types.d.ts +1 -0
  21. package/dist/core/types.d.ts.map +1 -1
  22. package/dist/index.d.ts +3 -0
  23. package/dist/index.d.ts.map +1 -1
  24. package/dist/index.js +6 -1
  25. package/dist/index.js.map +1 -1
  26. package/dist/jsonforms-provider-protocols.css +2 -2
  27. package/dist/no-eval-ajv.d.ts +70 -0
  28. package/dist/no-eval-ajv.d.ts.map +1 -0
  29. package/dist/no-eval-ajv.js +247 -0
  30. package/dist/no-eval-ajv.js.map +1 -0
  31. package/dist/vue/components/ProviderAutocomplete.vue.d.ts.map +1 -1
  32. package/dist/vue/components/ProviderAutocomplete.vue.js +10 -6
  33. package/dist/vue/components/ProviderAutocomplete.vue.js.map +1 -1
  34. package/dist/vue/components/ProviderMultiSelect.vue.d.ts.map +1 -1
  35. package/dist/vue/components/ProviderMultiSelect.vue.js +1 -1
  36. package/dist/vue/components/ProviderMultiSelect.vue2.js +17 -9
  37. package/dist/vue/components/ProviderMultiSelect.vue2.js.map +1 -1
  38. package/dist/vue/components/ProviderSelect.vue.d.ts.map +1 -1
  39. package/dist/vue/components/ProviderSelect.vue.js +1 -1
  40. package/dist/vue/components/ProviderSelect.vue2.js +19 -9
  41. package/dist/vue/components/ProviderSelect.vue2.js.map +1 -1
  42. package/dist/vue/composables/useDataLayer.d.ts +1 -0
  43. package/dist/vue/composables/useDataLayer.d.ts.map +1 -1
  44. package/dist/vue/composables/useDataLayer.js +1 -0
  45. package/dist/vue/composables/useDataLayer.js.map +1 -1
  46. package/dist/vue/composables/useDerive.d.ts +1 -1
  47. package/dist/vue/composables/useDerive.d.ts.map +1 -1
  48. package/dist/vue/composables/useDerive.js +19 -2
  49. package/dist/vue/composables/useDerive.js.map +1 -1
  50. package/dist/vue/composables/useDeriveInitialValue.d.ts +36 -0
  51. package/dist/vue/composables/useDeriveInitialValue.d.ts.map +1 -0
  52. package/dist/vue/composables/useDeriveInitialValue.js +125 -0
  53. package/dist/vue/composables/useDeriveInitialValue.js.map +1 -0
  54. package/dist/vue/composables/useDirtyValidation.d.ts +3 -3
  55. package/dist/vue/composables/useDirtyValidation.d.ts.map +1 -1
  56. package/dist/vue/composables/useDirtyValidation.js +2 -2
  57. package/dist/vue/composables/useDirtyValidation.js.map +1 -1
  58. package/dist/vue/composables/useProjection.d.ts +7 -0
  59. package/dist/vue/composables/useProjection.d.ts.map +1 -1
  60. package/dist/vue/composables/useProjection.js +87 -4
  61. package/dist/vue/composables/useProjection.js.map +1 -1
  62. package/dist/vue/composables/useProvider.d.ts +2 -2
  63. package/dist/vue/composables/useProvider.d.ts.map +1 -1
  64. package/dist/vue/composables/useProvider.js +14 -10
  65. package/dist/vue/composables/useProvider.js.map +1 -1
  66. package/dist/vue/index.d.ts +2 -0
  67. package/dist/vue/index.d.ts.map +1 -1
  68. package/dist/vue/index.js +30 -10
  69. package/dist/vue/index.js.map +1 -1
  70. package/dist/vue/primevue/JfBoolean.vue.d.ts.map +1 -1
  71. package/dist/vue/primevue/JfBoolean.vue.js +17 -6
  72. package/dist/vue/primevue/JfBoolean.vue.js.map +1 -1
  73. package/dist/vue/primevue/JfEnum.vue.d.ts.map +1 -1
  74. package/dist/vue/primevue/JfEnum.vue.js +22 -10
  75. package/dist/vue/primevue/JfEnum.vue.js.map +1 -1
  76. package/dist/vue/primevue/JfEnumArray.vue.d.ts.map +1 -1
  77. package/dist/vue/primevue/JfEnumArray.vue.js +20 -10
  78. package/dist/vue/primevue/JfEnumArray.vue.js.map +1 -1
  79. package/dist/vue/primevue/JfNumber.vue.d.ts.map +1 -1
  80. package/dist/vue/primevue/JfNumber.vue.js +18 -10
  81. package/dist/vue/primevue/JfNumber.vue.js.map +1 -1
  82. package/dist/vue/primevue/JfText.vue.d.ts.map +1 -1
  83. package/dist/vue/primevue/JfText.vue.js +27 -12
  84. package/dist/vue/primevue/JfText.vue.js.map +1 -1
  85. package/dist/vue/primevue/JfTextArea.vue.d.ts.map +1 -1
  86. package/dist/vue/primevue/JfTextArea.vue.js +15 -9
  87. package/dist/vue/primevue/JfTextArea.vue.js.map +1 -1
  88. package/dist/vue/primevue/index.d.ts.map +1 -1
  89. package/dist/vue/primevue/index.js +93 -16
  90. package/dist/vue/primevue/index.js.map +1 -1
  91. package/dist/vue/utils/autoSelect.js.map +1 -1
  92. package/dist/vue/utils/placeholder.d.ts +17 -0
  93. package/dist/vue/utils/placeholder.d.ts.map +1 -0
  94. package/dist/vue/utils/placeholder.js +17 -0
  95. package/dist/vue/utils/placeholder.js.map +1 -0
  96. package/package.json +10 -2
  97. package/src/core/initFormData.ts +208 -0
  98. package/src/core/projection.ts +33 -22
  99. package/src/core/refs.ts +166 -0
  100. package/src/core/resolveScope.ts +23 -8
  101. package/src/core/transforms.ts +33 -6
  102. package/src/core/types.ts +1 -0
  103. package/src/index.ts +14 -2
  104. package/src/no-eval-ajv.ts +381 -0
  105. package/src/vue/components/ProviderAutocomplete.vue +9 -7
  106. package/src/vue/components/ProviderMultiSelect.vue +20 -15
  107. package/src/vue/components/ProviderSelect.vue +21 -14
  108. package/src/vue/composables/useDataLayer.ts +1 -1
  109. package/src/vue/composables/useDerive.ts +46 -3
  110. package/src/vue/composables/useDeriveInitialValue.ts +195 -0
  111. package/src/vue/composables/useDirtyValidation.ts +8 -3
  112. package/src/vue/composables/useProjection.ts +172 -1
  113. package/src/vue/composables/useProvider.ts +28 -11
  114. package/src/vue/index.ts +28 -9
  115. package/src/vue/primevue/JfBoolean.vue +10 -5
  116. package/src/vue/primevue/JfEnum.vue +23 -14
  117. package/src/vue/primevue/JfEnumArray.vue +22 -17
  118. package/src/vue/primevue/JfNumber.vue +20 -12
  119. package/src/vue/primevue/JfText.vue +31 -16
  120. package/src/vue/primevue/JfTextArea.vue +15 -13
  121. package/src/vue/primevue/index.ts +104 -23
  122. package/src/vue/styles.css +26 -1
  123. package/src/vue/utils/autoSelect.ts +2 -2
  124. package/src/vue/utils/placeholder.ts +42 -0
@@ -1,17 +1,19 @@
1
1
  import { defineComponent, computed, inject, watch, createElementBlock, openBlock, createCommentVNode, createVNode, unref, toDisplayString } from "vue";
2
2
  import { useJsonFormsControl } from "@jsonforms/vue";
3
3
  import { useProvider } from "../composables/useProvider.js";
4
+ import { useDeriveInitialValue } from "../composables/useDeriveInitialValue.js";
4
5
  import { useProjection } from "../composables/useProjection.js";
5
6
  import { shouldAutoSelect } from "../utils/autoSelect.js";
7
+ import { resolvePlaceholder } from "../utils/placeholder.js";
6
8
  import Dropdown from "primevue/dropdown";
7
- const _hoisted_1 = { class: "flex flex-column gap-2" };
9
+ const _hoisted_1 = { class: "jf-control" };
8
10
  const _hoisted_2 = {
9
11
  key: 0,
10
- class: "text-color text-left"
12
+ class: "jf-label"
11
13
  };
12
14
  const _hoisted_3 = {
13
15
  key: 1,
14
- class: "text-color-secondary text-left"
16
+ class: "jf-description"
15
17
  };
16
18
  const _hoisted_4 = {
17
19
  key: 2,
@@ -28,7 +30,11 @@ const _sfc_main = /* @__PURE__ */ defineComponent({
28
30
  setup(__props) {
29
31
  const props = __props;
30
32
  const { control, handleChange: rawHandleChange } = useJsonFormsControl(props);
31
- const { projectedData, handleProjectedChange: handleChange } = useProjection(control, rawHandleChange);
33
+ const {
34
+ projectedData,
35
+ projectedLabel,
36
+ handleProjectedChange: handleChange
37
+ } = useProjection(control, rawHandleChange);
32
38
  const binding = computed(() => {
33
39
  const provider = control.value.uischema?.options?.provider;
34
40
  if (provider && typeof provider === "object" && !provider.load) {
@@ -46,8 +52,9 @@ const _sfc_main = /* @__PURE__ */ defineComponent({
46
52
  data: rootData,
47
53
  // Pass the reactive reference
48
54
  path: control.value.path,
49
- dependsOnValues: depValues.value
55
+ dependsOnValues: depValues
50
56
  });
57
+ useDeriveInitialValue({ control, handleChange });
51
58
  watch(
52
59
  [items, loading],
53
60
  ([newItems, isLoading]) => {
@@ -69,17 +76,20 @@ const _sfc_main = /* @__PURE__ */ defineComponent({
69
76
  });
70
77
  const placeholder = computed(() => {
71
78
  if (loading.value) return "Loading…";
72
- const uischemaPlaceholder = control.value.uischema?.options?.placeholder;
73
- return uischemaPlaceholder || "Select…";
79
+ return resolvePlaceholder(
80
+ control.value.uischema,
81
+ projectedLabel.value,
82
+ "select"
83
+ );
74
84
  });
75
85
  return (_ctx, _cache) => {
76
86
  return openBlock(), createElementBlock("div", _hoisted_1, [
77
- unref(control).schema.title ? (openBlock(), createElementBlock("label", _hoisted_2, toDisplayString(unref(control).schema.title), 1)) : createCommentVNode("", true),
87
+ unref(projectedLabel) ? (openBlock(), createElementBlock("label", _hoisted_2, toDisplayString(unref(projectedLabel)), 1)) : createCommentVNode("", true),
78
88
  unref(control).description ? (openBlock(), createElementBlock("div", _hoisted_3, toDisplayString(unref(control).description), 1)) : createCommentVNode("", true),
79
89
  createVNode(unref(Dropdown), {
80
90
  modelValue: value.value,
81
91
  "onUpdate:modelValue": _cache[0] || (_cache[0] = ($event) => value.value = $event),
82
- class: "w-full",
92
+ class: "w-full!",
83
93
  options: unref(items),
84
94
  "option-label": "label",
85
95
  "option-value": "value",
@@ -1 +1 @@
1
- {"version":3,"file":"ProviderSelect.vue2.js","sources":["../../../src/vue/components/ProviderSelect.vue"],"sourcesContent":["<script setup lang=\"ts\">\nimport type { ControlElement, JsonSchema } from \"@jsonforms/core\";\nimport { useJsonFormsControl } from \"@jsonforms/vue\";\nimport { computed, inject, watch } from \"vue\";\nimport { useProvider } from \"../composables/useProvider\";\nimport { useProjection } from \"../composables/useProjection\";\nimport { shouldAutoSelect } from \"../utils/autoSelect\";\nimport Dropdown from \"primevue/dropdown\";\n\nconst props = defineProps<{\n uischema: ControlElement;\n schema: JsonSchema;\n path: string;\n}>();\nconst { control, handleChange: rawHandleChange } = useJsonFormsControl(props);\nconst { projectedData, handleProjectedChange: handleChange } = useProjection(control, rawHandleChange);\n\nconst binding = computed(() => {\n const provider = control.value.uischema?.options?.provider;\n // Ensure load property is set to 'mount' by default\n if (provider && typeof provider === \"object\" && !provider.load) {\n return { ...provider, load: \"mount\" };\n }\n return provider;\n});\n\nconst deps = computed(\n () =>\n ((\n (control.value.schema as Record<string, unknown>)?.[\n \"x-provider\"\n ] as Record<string, unknown>\n )?.dependsOn as string[]) ?? [],\n);\nconst depValues = computed(() => deps.value.map(() => null)); // you can resolve actual values via control.value.data & pointers\n\n// Get the root form data from JSONForms context for template URL resolution\nconst injectedFormData = inject<{ value: unknown }>(\"formData\", { value: {} });\nconst rootData = computed(() => injectedFormData.value || {});\n\nconst { items, loading, error } = useProvider(binding, {\n data: rootData, // Pass the reactive reference\n path: control.value.path,\n dependsOnValues: depValues.value,\n});\n\n// Provider will automatically reload when rootData changes due to reactive cache key\n\n// Auto-select when provider returns only one item (enabled by default)\nwatch(\n [items, loading],\n ([newItems, isLoading]) => {\n const valueToSelect = shouldAutoSelect({\n autoSelectSingle:\n control.value.uischema?.options?.autoSelectSingle !== false,\n isLoading,\n items: newItems,\n currentValue: projectedData.value,\n });\n\n if (valueToSelect !== null) {\n handleChange(control.value.path, valueToSelect);\n }\n },\n { immediate: true }\n);\n\nconst value = computed({\n get: () => projectedData.value,\n set: (v) => handleChange(control.value.path, v),\n});\n\nconst placeholder = computed(() => {\n if (loading.value) return \"Loading…\";\n // Check for placeholder in uischema options\n const uischemaPlaceholder = (\n control.value.uischema as { options?: { placeholder?: string } }\n )?.options?.placeholder;\n return uischemaPlaceholder || \"Select…\";\n});\n</script>\n\n<template>\n <div class=\"flex flex-column gap-2\">\n <label v-if=\"control.schema.title\" class=\"text-color text-left\">{{\n control.schema.title\n }}</label>\n <div v-if=\"control.description\" class=\"text-color-secondary text-left\">\n {{ control.description }}\n </div>\n <Dropdown\n v-model=\"value\"\n class=\"w-full\"\n :options=\"items\"\n option-label=\"label\"\n option-value=\"value\"\n :placeholder=\"placeholder\"\n :disabled=\"!control.enabled || loading\"\n :show-clear=\"true\"\n />\n <small v-if=\"error\" class=\"p-error\" role=\"alert\"\n >Failed to load: {{ error }}</small\n >\n </div>\n</template>\n\n<style scoped>\n:deep(.p-dropdown-label) {\n text-align: left;\n}\n</style>\n"],"names":["_openBlock","_createElementBlock","_unref","_toDisplayString","_createVNode"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;AASA,UAAM,QAAQ;AAKd,UAAM,EAAE,SAAS,cAAc,gBAAA,IAAoB,oBAAoB,KAAK;AAC5E,UAAM,EAAE,eAAe,uBAAuB,iBAAiB,cAAc,SAAS,eAAe;AAErG,UAAM,UAAU,SAAS,MAAM;AAC7B,YAAM,WAAW,QAAQ,MAAM,UAAU,SAAS;AAElD,UAAI,YAAY,OAAO,aAAa,YAAY,CAAC,SAAS,MAAM;AAC9D,eAAO,EAAE,GAAG,UAAU,MAAM,QAAA;AAAA,MAC9B;AACA,aAAO;AAAA,IACT,CAAC;AAED,UAAM,OAAO;AAAA,MACX,MAEK,QAAQ,MAAM,SACb,YACF,GACC,aAA0B,CAAA;AAAA,IAAC;AAElC,UAAM,YAAY,SAAS,MAAM,KAAK,MAAM,IAAI,MAAM,IAAI,CAAC;AAG3D,UAAM,mBAAmB,OAA2B,YAAY,EAAE,OAAO,CAAA,GAAI;AAC7E,UAAM,WAAW,SAAS,MAAM,iBAAiB,SAAS,CAAA,CAAE;AAE5D,UAAM,EAAE,OAAO,SAAS,MAAA,IAAU,YAAY,SAAS;AAAA,MACrD,MAAM;AAAA;AAAA,MACN,MAAM,QAAQ,MAAM;AAAA,MACpB,iBAAiB,UAAU;AAAA,IAAA,CAC5B;AAKD;AAAA,MACE,CAAC,OAAO,OAAO;AAAA,MACf,CAAC,CAAC,UAAU,SAAS,MAAM;AACzB,cAAM,gBAAgB,iBAAiB;AAAA,UACrC,kBACE,QAAQ,MAAM,UAAU,SAAS,qBAAqB;AAAA,UACxD;AAAA,UACA,OAAO;AAAA,UACP,cAAc,cAAc;AAAA,QAAA,CAC7B;AAED,YAAI,kBAAkB,MAAM;AAC1B,uBAAa,QAAQ,MAAM,MAAM,aAAa;AAAA,QAChD;AAAA,MACF;AAAA,MACA,EAAE,WAAW,KAAA;AAAA,IAAK;AAGpB,UAAM,QAAQ,SAAS;AAAA,MACrB,KAAK,MAAM,cAAc;AAAA,MACzB,KAAK,CAAC,MAAM,aAAa,QAAQ,MAAM,MAAM,CAAC;AAAA,IAAA,CAC/C;AAED,UAAM,cAAc,SAAS,MAAM;AACjC,UAAI,QAAQ,MAAO,QAAO;AAE1B,YAAM,sBACJ,QAAQ,MAAM,UACb,SAAS;AACZ,aAAO,uBAAuB;AAAA,IAChC,CAAC;;AAIC,aAAAA,UAAA,GAAAC,mBAoBM,OApBN,YAoBM;AAAA,QAnBSC,MAAA,OAAA,EAAQ,OAAO,SAA5BF,aAAAC,mBAEU,SAFV,YAEUE,gBADRD,eAAQ,OAAO,KAAK,GAAA,CAAA;QAEXA,MAAA,OAAA,EAAQ,eAAnBF,UAAA,GAAAC,mBAEM,OAFN,YAEME,gBADDD,MAAA,OAAA,EAAQ,WAAW,GAAA,CAAA;QAExBE,YASEF,MAAA,QAAA,GAAA;AAAA,sBARS,MAAA;AAAA,uEAAA,MAAK,QAAA;AAAA,UACd,OAAM;AAAA,UACL,SAASA,MAAA,KAAA;AAAA,UACV,gBAAa;AAAA,UACb,gBAAa;AAAA,UACZ,aAAa,YAAA;AAAA,UACb,UAAQ,CAAGA,MAAA,OAAA,EAAQ,WAAWA,MAAA,OAAA;AAAA,UAC9B,cAAY;AAAA,QAAA;QAEFA,MAAA,KAAA,KAAbF,aAAAC,mBAEC,SAFD,YACG,qCAAmBC,MAAA,KAAA,CAAK,GAAA,CAAA;;;;;"}
1
+ {"version":3,"file":"ProviderSelect.vue2.js","sources":["../../../src/vue/components/ProviderSelect.vue"],"sourcesContent":["<script setup lang=\"ts\">\nimport type { ControlElement, JsonSchema } from \"@jsonforms/core\";\nimport { useJsonFormsControl } from \"@jsonforms/vue\";\nimport { computed, inject, watch } from \"vue\";\nimport { useProvider } from \"../composables/useProvider\";\nimport { useDeriveInitialValue } from \"../composables/useDeriveInitialValue\";\nimport { useProjection } from \"../composables/useProjection\";\nimport { shouldAutoSelect } from \"../utils/autoSelect\";\nimport { resolvePlaceholder } from \"../utils/placeholder\";\nimport Dropdown from \"primevue/dropdown\";\n\nconst props = defineProps<{\n uischema: ControlElement;\n schema: JsonSchema;\n path: string;\n}>();\nconst { control, handleChange: rawHandleChange } = useJsonFormsControl(props);\nconst {\n projectedData,\n projectedLabel,\n handleProjectedChange: handleChange,\n} = useProjection(control, rawHandleChange);\n\nconst binding = computed(() => {\n const provider = control.value.uischema?.options?.provider;\n // Ensure load property is set to 'mount' by default\n if (provider && typeof provider === \"object\" && !provider.load) {\n return { ...provider, load: \"mount\" };\n }\n return provider;\n});\n\nconst deps = computed(\n () =>\n ((\n (control.value.schema as Record<string, unknown>)?.[\n \"x-provider\"\n ] as Record<string, unknown>\n )?.dependsOn as string[]) ?? [],\n);\nconst depValues = computed(() => deps.value.map(() => null)); // you can resolve actual values via control.value.data & pointers\n\n// Get the root form data from JSONForms context for template URL resolution\nconst injectedFormData = inject<{ value: unknown }>(\"formData\", { value: {} });\nconst rootData = computed(() => injectedFormData.value || {});\n\nconst { items, loading, error } = useProvider(binding, {\n data: rootData, // Pass the reactive reference\n path: control.value.path,\n dependsOnValues: depValues,\n});\n\n// Provider will automatically reload when rootData changes due to reactive cache key\n\n// deriveInitialValue — async API-based initial value seeding\nuseDeriveInitialValue({ control, handleChange });\n\n// Auto-select when provider returns only one item (enabled by default)\nwatch(\n [items, loading],\n ([newItems, isLoading]) => {\n const valueToSelect = shouldAutoSelect({\n autoSelectSingle:\n control.value.uischema?.options?.autoSelectSingle !== false,\n isLoading,\n items: newItems,\n currentValue: projectedData.value,\n });\n\n if (valueToSelect !== null) {\n handleChange(control.value.path, valueToSelect);\n }\n },\n { immediate: true },\n);\n\nconst value = computed({\n get: () => projectedData.value,\n set: (v) => handleChange(control.value.path, v),\n});\n\nconst placeholder = computed(() => {\n if (loading.value) return \"Loading…\";\n return resolvePlaceholder(\n control.value.uischema,\n projectedLabel.value,\n \"select\",\n );\n});\n</script>\n\n<template>\n <div class=\"jf-control\">\n <label v-if=\"projectedLabel\" class=\"jf-label\">{{ projectedLabel }}</label>\n <div v-if=\"control.description\" class=\"jf-description\">\n {{ control.description }}\n </div>\n <Dropdown\n v-model=\"value\"\n class=\"w-full!\"\n :options=\"items\"\n option-label=\"label\"\n option-value=\"value\"\n :placeholder=\"placeholder\"\n :disabled=\"!control.enabled || loading\"\n :show-clear=\"true\"\n />\n <small v-if=\"error\" class=\"p-error\" role=\"alert\"\n >Failed to load: {{ error }}</small\n >\n </div>\n</template>\n\n<style scoped>\n:deep(.p-dropdown-label) {\n text-align: left;\n}\n</style>\n"],"names":["_openBlock","_createElementBlock","_unref","_toDisplayString","_createVNode"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAWA,UAAM,QAAQ;AAKd,UAAM,EAAE,SAAS,cAAc,gBAAA,IAAoB,oBAAoB,KAAK;AAC5E,UAAM;AAAA,MACJ;AAAA,MACA;AAAA,MACA,uBAAuB;AAAA,IAAA,IACrB,cAAc,SAAS,eAAe;AAE1C,UAAM,UAAU,SAAS,MAAM;AAC7B,YAAM,WAAW,QAAQ,MAAM,UAAU,SAAS;AAElD,UAAI,YAAY,OAAO,aAAa,YAAY,CAAC,SAAS,MAAM;AAC9D,eAAO,EAAE,GAAG,UAAU,MAAM,QAAA;AAAA,MAC9B;AACA,aAAO;AAAA,IACT,CAAC;AAED,UAAM,OAAO;AAAA,MACX,MAEK,QAAQ,MAAM,SACb,YACF,GACC,aAA0B,CAAA;AAAA,IAAC;AAElC,UAAM,YAAY,SAAS,MAAM,KAAK,MAAM,IAAI,MAAM,IAAI,CAAC;AAG3D,UAAM,mBAAmB,OAA2B,YAAY,EAAE,OAAO,CAAA,GAAI;AAC7E,UAAM,WAAW,SAAS,MAAM,iBAAiB,SAAS,CAAA,CAAE;AAE5D,UAAM,EAAE,OAAO,SAAS,MAAA,IAAU,YAAY,SAAS;AAAA,MACrD,MAAM;AAAA;AAAA,MACN,MAAM,QAAQ,MAAM;AAAA,MACpB,iBAAiB;AAAA,IAAA,CAClB;AAKD,0BAAsB,EAAE,SAAS,cAAc;AAG/C;AAAA,MACE,CAAC,OAAO,OAAO;AAAA,MACf,CAAC,CAAC,UAAU,SAAS,MAAM;AACzB,cAAM,gBAAgB,iBAAiB;AAAA,UACrC,kBACE,QAAQ,MAAM,UAAU,SAAS,qBAAqB;AAAA,UACxD;AAAA,UACA,OAAO;AAAA,UACP,cAAc,cAAc;AAAA,QAAA,CAC7B;AAED,YAAI,kBAAkB,MAAM;AAC1B,uBAAa,QAAQ,MAAM,MAAM,aAAa;AAAA,QAChD;AAAA,MACF;AAAA,MACA,EAAE,WAAW,KAAA;AAAA,IAAK;AAGpB,UAAM,QAAQ,SAAS;AAAA,MACrB,KAAK,MAAM,cAAc;AAAA,MACzB,KAAK,CAAC,MAAM,aAAa,QAAQ,MAAM,MAAM,CAAC;AAAA,IAAA,CAC/C;AAED,UAAM,cAAc,SAAS,MAAM;AACjC,UAAI,QAAQ,MAAO,QAAO;AAC1B,aAAO;AAAA,QACL,QAAQ,MAAM;AAAA,QACd,eAAe;AAAA,QACf;AAAA,MAAA;AAAA,IAEJ,CAAC;;AAIC,aAAAA,UAAA,GAAAC,mBAkBM,OAlBN,YAkBM;AAAA,QAjBSC,MAAA,cAAA,kBAAbD,mBAA0E,SAA1E,YAA0EE,gBAAzBD,MAAA,cAAA,CAAc,GAAA,CAAA;QACpDA,MAAA,OAAA,EAAQ,eAAnBF,UAAA,GAAAC,mBAEM,OAFN,YAEME,gBADDD,MAAA,OAAA,EAAQ,WAAW,GAAA,CAAA;QAExBE,YASEF,MAAA,QAAA,GAAA;AAAA,sBARS,MAAA;AAAA,uEAAA,MAAK,QAAA;AAAA,UACd,OAAM;AAAA,UACL,SAASA,MAAA,KAAA;AAAA,UACV,gBAAa;AAAA,UACb,gBAAa;AAAA,UACZ,aAAa,YAAA;AAAA,UACb,UAAQ,CAAGA,MAAA,OAAA,EAAQ,WAAWA,MAAA,OAAA;AAAA,UAC9B,cAAY;AAAA,QAAA;QAEFA,MAAA,KAAA,KAAbF,aAAAC,mBAEC,SAFD,YACG,qCAAmBC,MAAA,KAAA,CAAK,GAAA,CAAA;;;;;"}
@@ -1,5 +1,6 @@
1
1
  import { type DeepReadonly, type Ref } from "vue";
2
2
  import type { ConnectorDataLayer } from "../../core/types";
3
+ export declare const DATA_LAYER_KEY: unique symbol;
3
4
  export interface DataLayer {
4
5
  push(data: Partial<ConnectorDataLayer>): void;
5
6
  state: DeepReadonly<Ref<ConnectorDataLayer>>;
@@ -1 +1 @@
1
- {"version":3,"file":"useDataLayer.d.ts","sourceRoot":"","sources":["../../../src/vue/composables/useDataLayer.ts"],"names":[],"mappings":"AAAA,OAAO,EAKL,KAAK,YAAY,EACjB,KAAK,GAAG,EACT,MAAM,KAAK,CAAC;AACb,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,kBAAkB,CAAC;AAI3D,MAAM,WAAW,SAAS;IACxB,IAAI,CAAC,IAAI,EAAE,OAAO,CAAC,kBAAkB,CAAC,GAAG,IAAI,CAAC;IAC9C,KAAK,EAAE,YAAY,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC,CAAC;CAC9C;AAED,wBAAgB,eAAe,IAAI,SAAS,CAa3C;AAED,wBAAgB,YAAY,IAAI,YAAY,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC,CAUpE"}
1
+ {"version":3,"file":"useDataLayer.d.ts","sourceRoot":"","sources":["../../../src/vue/composables/useDataLayer.ts"],"names":[],"mappings":"AAAA,OAAO,EAKL,KAAK,YAAY,EACjB,KAAK,GAAG,EACT,MAAM,KAAK,CAAC;AACb,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,kBAAkB,CAAC;AAE3D,eAAO,MAAM,cAAc,eAAsB,CAAC;AAElD,MAAM,WAAW,SAAS;IACxB,IAAI,CAAC,IAAI,EAAE,OAAO,CAAC,kBAAkB,CAAC,GAAG,IAAI,CAAC;IAC9C,KAAK,EAAE,YAAY,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC,CAAC;CAC9C;AAED,wBAAgB,eAAe,IAAI,SAAS,CAa3C;AAED,wBAAgB,YAAY,IAAI,YAAY,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC,CAUpE"}
@@ -19,6 +19,7 @@ function useDataLayer() {
19
19
  return readonly(ref({}));
20
20
  }
21
21
  export {
22
+ DATA_LAYER_KEY,
22
23
  createDataLayer,
23
24
  useDataLayer
24
25
  };
@@ -1 +1 @@
1
- {"version":3,"file":"useDataLayer.js","sources":["../../../src/vue/composables/useDataLayer.ts"],"sourcesContent":["import {\n ref,\n provide,\n inject,\n readonly,\n type DeepReadonly,\n type Ref,\n} from \"vue\";\nimport type { ConnectorDataLayer } from \"../../core/types\";\n\nconst DATA_LAYER_KEY = Symbol(\"dataLayer\");\n\nexport interface DataLayer {\n push(data: Partial<ConnectorDataLayer>): void;\n state: DeepReadonly<Ref<ConnectorDataLayer>>;\n}\n\nexport function createDataLayer(): DataLayer {\n const state = ref<ConnectorDataLayer>({});\n\n const dataLayer: DataLayer = {\n push(data: Partial<ConnectorDataLayer>) {\n state.value = { ...state.value, ...data };\n },\n state: readonly(state) as DeepReadonly<Ref<ConnectorDataLayer>>,\n };\n\n provide(DATA_LAYER_KEY, dataLayer);\n\n return dataLayer;\n}\n\nexport function useDataLayer(): DeepReadonly<Ref<ConnectorDataLayer>> {\n const dataLayer = inject<DataLayer | null>(DATA_LAYER_KEY, null);\n if (dataLayer) {\n return dataLayer.state;\n }\n\n // No dataLayer provided — return empty reactive ref\n return readonly(ref<ConnectorDataLayer>({})) as DeepReadonly<\n Ref<ConnectorDataLayer>\n >;\n}\n"],"names":[],"mappings":";AAUA,MAAM,iBAAiB,OAAO,WAAW;AAOlC,SAAS,kBAA6B;AAC3C,QAAM,QAAQ,IAAwB,EAAE;AAExC,QAAM,YAAuB;AAAA,IAC3B,KAAK,MAAmC;AACtC,YAAM,QAAQ,EAAE,GAAG,MAAM,OAAO,GAAG,KAAA;AAAA,IACrC;AAAA,IACA,OAAO,SAAS,KAAK;AAAA,EAAA;AAGvB,UAAQ,gBAAgB,SAAS;AAEjC,SAAO;AACT;AAEO,SAAS,eAAsD;AACpE,QAAM,YAAY,OAAyB,gBAAgB,IAAI;AAC/D,MAAI,WAAW;AACb,WAAO,UAAU;AAAA,EACnB;AAGA,SAAO,SAAS,IAAwB,CAAA,CAAE,CAAC;AAG7C;"}
1
+ {"version":3,"file":"useDataLayer.js","sources":["../../../src/vue/composables/useDataLayer.ts"],"sourcesContent":["import {\n ref,\n provide,\n inject,\n readonly,\n type DeepReadonly,\n type Ref,\n} from \"vue\";\nimport type { ConnectorDataLayer } from \"../../core/types\";\n\nexport const DATA_LAYER_KEY = Symbol(\"dataLayer\");\n\nexport interface DataLayer {\n push(data: Partial<ConnectorDataLayer>): void;\n state: DeepReadonly<Ref<ConnectorDataLayer>>;\n}\n\nexport function createDataLayer(): DataLayer {\n const state = ref<ConnectorDataLayer>({});\n\n const dataLayer: DataLayer = {\n push(data: Partial<ConnectorDataLayer>) {\n state.value = { ...state.value, ...data };\n },\n state: readonly(state) as DeepReadonly<Ref<ConnectorDataLayer>>,\n };\n\n provide(DATA_LAYER_KEY, dataLayer);\n\n return dataLayer;\n}\n\nexport function useDataLayer(): DeepReadonly<Ref<ConnectorDataLayer>> {\n const dataLayer = inject<DataLayer | null>(DATA_LAYER_KEY, null);\n if (dataLayer) {\n return dataLayer.state;\n }\n\n // No dataLayer provided — return empty reactive ref\n return readonly(ref<ConnectorDataLayer>({})) as DeepReadonly<\n Ref<ConnectorDataLayer>\n >;\n}\n"],"names":[],"mappings":";AAUO,MAAM,iBAAiB,OAAO,WAAW;AAOzC,SAAS,kBAA6B;AAC3C,QAAM,QAAQ,IAAwB,EAAE;AAExC,QAAM,YAAuB;AAAA,IAC3B,KAAK,MAAmC;AACtC,YAAM,QAAQ,EAAE,GAAG,MAAM,OAAO,GAAG,KAAA;AAAA,IACrC;AAAA,IACA,OAAO,SAAS,KAAK;AAAA,EAAA;AAGvB,UAAQ,gBAAgB,SAAS;AAEjC,SAAO;AACT;AAEO,SAAS,eAAsD;AACpE,QAAM,YAAY,OAAyB,gBAAgB,IAAI;AAC/D,MAAI,WAAW;AACb,WAAO,UAAU;AAAA,EACnB;AAGA,SAAO,SAAS,IAAwB,CAAA,CAAE,CAAC;AAG7C;"}
@@ -11,6 +11,6 @@ interface DeriveOptions {
11
11
  * matches the projected (unwrapped) value rather than raw scope data. */
12
12
  data?: Ref<unknown> | ComputedRef<unknown>;
13
13
  }
14
- export declare function useDerive({ control, handleChange, data: dataOverride }: DeriveOptions): void;
14
+ export declare function useDerive({ control, handleChange, data: dataOverride, }: DeriveOptions): void;
15
15
  export {};
16
16
  //# sourceMappingURL=useDerive.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"useDerive.d.ts","sourceRoot":"","sources":["../../../src/vue/composables/useDerive.ts"],"names":[],"mappings":"AAAA,OAAO,EAAkC,KAAK,GAAG,EAAE,KAAK,WAAW,EAAE,MAAM,KAAK,CAAC;AACjF,OAAO,EAAE,KAAK,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAGtD,UAAU,aAAa;IACrB,OAAO,EAAE,GAAG,CAAC;QACX,QAAQ,EAAE,cAAc,CAAC;QACzB,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,OAAO,CAAC;KACf,CAAC,CAAC;IACH,YAAY,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,KAAK,IAAI,CAAC;IACrD;8EAC0E;IAC1E,IAAI,CAAC,EAAE,GAAG,CAAC,OAAO,CAAC,GAAG,WAAW,CAAC,OAAO,CAAC,CAAC;CAC5C;AAED,wBAAgB,SAAS,CAAC,EAAE,OAAO,EAAE,YAAY,EAAE,IAAI,EAAE,YAAY,EAAE,EAAE,aAAa,QAiDrF"}
1
+ {"version":3,"file":"useDerive.d.ts","sourceRoot":"","sources":["../../../src/vue/composables/useDerive.ts"],"names":[],"mappings":"AAAA,OAAO,EAML,KAAK,GAAG,EACR,KAAK,WAAW,EACjB,MAAM,KAAK,CAAC;AACb,OAAO,EAAE,KAAK,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAGtD,UAAU,aAAa;IACrB,OAAO,EAAE,GAAG,CAAC;QACX,QAAQ,EAAE,cAAc,CAAC;QACzB,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,OAAO,CAAC;KACf,CAAC,CAAC;IACH,YAAY,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,KAAK,IAAI,CAAC;IACrD;8EAC0E;IAC1E,IAAI,CAAC,EAAE,GAAG,CAAC,OAAO,CAAC,GAAG,WAAW,CAAC,OAAO,CAAC,CAAC;CAC5C;AAED,wBAAgB,SAAS,CAAC,EACxB,OAAO,EACP,YAAY,EACZ,IAAI,EAAE,YAAY,GACnB,EAAE,aAAa,QAgFf"}
@@ -1,7 +1,11 @@
1
- import { inject, computed, watch, unref } from "vue";
1
+ import { inject, computed, ref, watch, unref } from "vue";
2
2
  import "@jsonforms/core";
3
3
  import { useDataLayer } from "./useDataLayer.js";
4
- function useDerive({ control, handleChange, data: dataOverride }) {
4
+ function useDerive({
5
+ control,
6
+ handleChange,
7
+ data: dataOverride
8
+ }) {
5
9
  const injectedFormData = inject("formData", {
6
10
  value: {}
7
11
  });
@@ -16,6 +20,8 @@ function useDerive({ control, handleChange, data: dataOverride }) {
16
20
  // 'follow' = auto-update, 'manual' = user controlled
17
21
  };
18
22
  });
23
+ const lastDerivedValue = ref(void 0);
24
+ let isFirstRun = true;
19
25
  watch(
20
26
  [rootData, dataLayerData, deriveConfig],
21
27
  ([data, extData, config]) => {
@@ -29,6 +35,17 @@ function useDerive({ control, handleChange, data: dataOverride }) {
29
35
  extData
30
36
  );
31
37
  const compareData = dataOverride ? unref(dataOverride) : control.value.data;
38
+ if (isFirstRun) {
39
+ isFirstRun = false;
40
+ lastDerivedValue.value = derivedValue;
41
+ const isFieldEmpty = compareData === void 0 || compareData === null || compareData === "";
42
+ if (isFieldEmpty && derivedValue !== compareData) {
43
+ handleChange(control.value.path, derivedValue);
44
+ }
45
+ return;
46
+ }
47
+ if (derivedValue === lastDerivedValue.value) return;
48
+ lastDerivedValue.value = derivedValue;
32
49
  if (derivedValue !== compareData) {
33
50
  handleChange(control.value.path, derivedValue);
34
51
  }
@@ -1 +1 @@
1
- {"version":3,"file":"useDerive.js","sources":["../../../src/vue/composables/useDerive.ts"],"sourcesContent":["import { computed, watch, unref, inject, type Ref, type ComputedRef } from \"vue\";\nimport { type ControlElement } from \"@jsonforms/core\";\nimport { useDataLayer } from \"./useDataLayer\";\n\ninterface DeriveOptions {\n control: Ref<{\n uischema: ControlElement;\n path: string;\n data: unknown;\n }>;\n handleChange: (path: string, value: unknown) => void;\n /** When projection is active, pass projectedData so the comparison\n * matches the projected (unwrapped) value rather than raw scope data. */\n data?: Ref<unknown> | ComputedRef<unknown>;\n}\n\nexport function useDerive({ control, handleChange, data: dataOverride }: DeriveOptions) {\n // Get the root form data from JSONForms context\n const injectedFormData = inject<{ value: unknown }>(\"formData\", {\n value: {},\n });\n const rootData = computed(() => injectedFormData.value || {});\n\n // Get data from the dataLayer\n const dataLayerState = useDataLayer();\n const dataLayerData = computed(() => dataLayerState.value || {});\n\n // Extract derive configuration from uischema options\n const deriveConfig = computed(() => {\n const options = control.value.uischema?.options as\n | { derive?: string; mode?: string }\n | undefined;\n return {\n expression: options?.derive,\n mode: options?.mode || \"follow\", // 'follow' = auto-update, 'manual' = user controlled\n };\n });\n\n // Watch for changes in form data and dataLayer and update derived field\n watch(\n [rootData, dataLayerData, deriveConfig],\n ([data, extData, config]) => {\n if (!config.expression || config.mode !== \"follow\") {\n return;\n }\n\n try {\n const derivedValue = resolveDeriveExpression(\n config.expression,\n data,\n extData,\n );\n const compareData = dataOverride ? unref(dataOverride) : control.value.data;\n if (derivedValue !== compareData) {\n handleChange(control.value.path, derivedValue);\n }\n } catch (error) {\n console.warn(\n `Failed to derive value for ${control.value.path}:`,\n error,\n );\n }\n },\n { deep: true, immediate: true },\n );\n}\n\nfunction resolveDeriveExpression(\n expression: string,\n data: unknown,\n dataLayerData?: unknown,\n): unknown {\n // Handle dataLayer() syntax\n if (expression.startsWith(\"dataLayer(\") && expression.endsWith(\")\")) {\n const propertyPath = expression.slice(10, -1); // Remove \"dataLayer(\" and \")\"\n return resolvePropertyPath(propertyPath, dataLayerData);\n }\n\n // Handle simple property paths like \"country.name\"\n if (\n !expression.includes(\"(\") &&\n !expression.includes(\"+\") &&\n !expression.includes(\"?\")\n ) {\n return resolvePropertyPath(expression, data);\n }\n\n // For now, we'll only support simple property paths and dataLayer() calls\n // Complex expressions would require a safe expression evaluator\n return resolvePropertyPath(expression, data);\n}\n\nfunction resolvePropertyPath(path: string, data: unknown): unknown {\n if (!path || !data) return \"\";\n\n const keys = path.split(\".\");\n let value: unknown = data;\n\n for (const key of keys) {\n if (value && typeof value === \"object\" && key in value) {\n value = (value as Record<string, unknown>)[key];\n } else {\n return null;\n }\n }\n\n return value;\n}\n"],"names":[],"mappings":";;;AAgBO,SAAS,UAAU,EAAE,SAAS,cAAc,MAAM,gBAA+B;AAEtF,QAAM,mBAAmB,OAA2B,YAAY;AAAA,IAC9D,OAAO,CAAA;AAAA,EAAC,CACT;AACD,QAAM,WAAW,SAAS,MAAM,iBAAiB,SAAS,CAAA,CAAE;AAG5D,QAAM,iBAAiB,aAAA;AACvB,QAAM,gBAAgB,SAAS,MAAM,eAAe,SAAS,CAAA,CAAE;AAG/D,QAAM,eAAe,SAAS,MAAM;AAClC,UAAM,UAAU,QAAQ,MAAM,UAAU;AAGxC,WAAO;AAAA,MACL,YAAY,SAAS;AAAA,MACrB,MAAM,SAAS,QAAQ;AAAA;AAAA,IAAA;AAAA,EAE3B,CAAC;AAGD;AAAA,IACE,CAAC,UAAU,eAAe,YAAY;AAAA,IACtC,CAAC,CAAC,MAAM,SAAS,MAAM,MAAM;AAC3B,UAAI,CAAC,OAAO,cAAc,OAAO,SAAS,UAAU;AAClD;AAAA,MACF;AAEA,UAAI;AACF,cAAM,eAAe;AAAA,UACnB,OAAO;AAAA,UACP;AAAA,UACA;AAAA,QAAA;AAEF,cAAM,cAAc,eAAe,MAAM,YAAY,IAAI,QAAQ,MAAM;AACvE,YAAI,iBAAiB,aAAa;AAChC,uBAAa,QAAQ,MAAM,MAAM,YAAY;AAAA,QAC/C;AAAA,MACF,SAAS,OAAO;AACd,gBAAQ;AAAA,UACN,8BAA8B,QAAQ,MAAM,IAAI;AAAA,UAChD;AAAA,QAAA;AAAA,MAEJ;AAAA,IACF;AAAA,IACA,EAAE,MAAM,MAAM,WAAW,KAAA;AAAA,EAAK;AAElC;AAEA,SAAS,wBACP,YACA,MACA,eACS;AAET,MAAI,WAAW,WAAW,YAAY,KAAK,WAAW,SAAS,GAAG,GAAG;AACnE,UAAM,eAAe,WAAW,MAAM,IAAI,EAAE;AAC5C,WAAO,oBAAoB,cAAc,aAAa;AAAA,EACxD;AAGA,MACE,CAAC,WAAW,SAAS,GAAG,KACxB,CAAC,WAAW,SAAS,GAAG,KACxB,CAAC,WAAW,SAAS,GAAG,GACxB;AACA,WAAO,oBAAoB,YAAY,IAAI;AAAA,EAC7C;AAIA,SAAO,oBAAoB,YAAY,IAAI;AAC7C;AAEA,SAAS,oBAAoB,MAAc,MAAwB;AACjE,MAAI,CAAC,QAAQ,CAAC,KAAM,QAAO;AAE3B,QAAM,OAAO,KAAK,MAAM,GAAG;AAC3B,MAAI,QAAiB;AAErB,aAAW,OAAO,MAAM;AACtB,QAAI,SAAS,OAAO,UAAU,YAAY,OAAO,OAAO;AACtD,cAAS,MAAkC,GAAG;AAAA,IAChD,OAAO;AACL,aAAO;AAAA,IACT;AAAA,EACF;AAEA,SAAO;AACT;"}
1
+ {"version":3,"file":"useDerive.js","sources":["../../../src/vue/composables/useDerive.ts"],"sourcesContent":["import {\n computed,\n ref,\n watch,\n unref,\n inject,\n type Ref,\n type ComputedRef,\n} from \"vue\";\nimport { type ControlElement } from \"@jsonforms/core\";\nimport { useDataLayer } from \"./useDataLayer\";\n\ninterface DeriveOptions {\n control: Ref<{\n uischema: ControlElement;\n path: string;\n data: unknown;\n }>;\n handleChange: (path: string, value: unknown) => void;\n /** When projection is active, pass projectedData so the comparison\n * matches the projected (unwrapped) value rather than raw scope data. */\n data?: Ref<unknown> | ComputedRef<unknown>;\n}\n\nexport function useDerive({\n control,\n handleChange,\n data: dataOverride,\n}: DeriveOptions) {\n // Get the root form data from JSONForms context\n const injectedFormData = inject<{ value: unknown }>(\"formData\", {\n value: {},\n });\n const rootData = computed(() => injectedFormData.value || {});\n\n // Get data from the dataLayer\n const dataLayerState = useDataLayer();\n const dataLayerData = computed(() => dataLayerState.value || {});\n\n // Extract derive configuration from uischema options\n const deriveConfig = computed(() => {\n const options = control.value.uischema?.options as\n | { derive?: string; mode?: string }\n | undefined;\n return {\n expression: options?.derive,\n mode: options?.mode || \"follow\", // 'follow' = auto-update, 'manual' = user controlled\n };\n });\n\n // Track the last resolved source value so we only update the field\n // when the source itself changes, not on every form data mutation.\n const lastDerivedValue = ref<unknown>(undefined);\n // First watch fire is special: a pre-populated field (edit flow seeded\n // from a saved connection) must not be clobbered by whatever the source\n // resolves to on mount. We record the source value but skip handleChange.\n let isFirstRun = true;\n\n // Watch for changes in form data and dataLayer and update derived field\n watch(\n [rootData, dataLayerData, deriveConfig],\n ([data, extData, config]) => {\n if (!config.expression || config.mode !== \"follow\") {\n return;\n }\n\n try {\n const derivedValue = resolveDeriveExpression(\n config.expression,\n data,\n extData,\n );\n const compareData = dataOverride\n ? unref(dataOverride)\n : control.value.data;\n\n if (isFirstRun) {\n isFirstRun = false;\n lastDerivedValue.value = derivedValue;\n // On mount, only populate the field if it's empty.\n // A non-empty seed represents user intent (edit flow) and must\n // be preserved. Subsequent source changes will still propagate.\n const isFieldEmpty =\n compareData === undefined ||\n compareData === null ||\n compareData === \"\";\n if (isFieldEmpty && derivedValue !== compareData) {\n handleChange(control.value.path, derivedValue);\n }\n return;\n }\n\n // Only update if the SOURCE value changed, not just the field value\n if (derivedValue === lastDerivedValue.value) return;\n lastDerivedValue.value = derivedValue;\n\n if (derivedValue !== compareData) {\n handleChange(control.value.path, derivedValue);\n }\n } catch (error) {\n console.warn(\n `Failed to derive value for ${control.value.path}:`,\n error,\n );\n }\n },\n { deep: true, immediate: true },\n );\n}\n\nfunction resolveDeriveExpression(\n expression: string,\n data: unknown,\n dataLayerData?: unknown,\n): unknown {\n // Handle dataLayer() syntax\n if (expression.startsWith(\"dataLayer(\") && expression.endsWith(\")\")) {\n const propertyPath = expression.slice(10, -1); // Remove \"dataLayer(\" and \")\"\n return resolvePropertyPath(propertyPath, dataLayerData);\n }\n\n // Handle simple property paths like \"country.name\"\n if (\n !expression.includes(\"(\") &&\n !expression.includes(\"+\") &&\n !expression.includes(\"?\")\n ) {\n return resolvePropertyPath(expression, data);\n }\n\n // For now, we'll only support simple property paths and dataLayer() calls\n // Complex expressions would require a safe expression evaluator\n return resolvePropertyPath(expression, data);\n}\n\nfunction resolvePropertyPath(path: string, data: unknown): unknown {\n if (!path || !data) return \"\";\n\n const keys = path.split(\".\");\n let value: unknown = data;\n\n for (const key of keys) {\n if (value && typeof value === \"object\" && key in value) {\n value = (value as Record<string, unknown>)[key];\n } else {\n return null;\n }\n }\n\n return value;\n}\n"],"names":[],"mappings":";;;AAwBO,SAAS,UAAU;AAAA,EACxB;AAAA,EACA;AAAA,EACA,MAAM;AACR,GAAkB;AAEhB,QAAM,mBAAmB,OAA2B,YAAY;AAAA,IAC9D,OAAO,CAAA;AAAA,EAAC,CACT;AACD,QAAM,WAAW,SAAS,MAAM,iBAAiB,SAAS,CAAA,CAAE;AAG5D,QAAM,iBAAiB,aAAA;AACvB,QAAM,gBAAgB,SAAS,MAAM,eAAe,SAAS,CAAA,CAAE;AAG/D,QAAM,eAAe,SAAS,MAAM;AAClC,UAAM,UAAU,QAAQ,MAAM,UAAU;AAGxC,WAAO;AAAA,MACL,YAAY,SAAS;AAAA,MACrB,MAAM,SAAS,QAAQ;AAAA;AAAA,IAAA;AAAA,EAE3B,CAAC;AAID,QAAM,mBAAmB,IAAa,MAAS;AAI/C,MAAI,aAAa;AAGjB;AAAA,IACE,CAAC,UAAU,eAAe,YAAY;AAAA,IACtC,CAAC,CAAC,MAAM,SAAS,MAAM,MAAM;AAC3B,UAAI,CAAC,OAAO,cAAc,OAAO,SAAS,UAAU;AAClD;AAAA,MACF;AAEA,UAAI;AACF,cAAM,eAAe;AAAA,UACnB,OAAO;AAAA,UACP;AAAA,UACA;AAAA,QAAA;AAEF,cAAM,cAAc,eAChB,MAAM,YAAY,IAClB,QAAQ,MAAM;AAElB,YAAI,YAAY;AACd,uBAAa;AACb,2BAAiB,QAAQ;AAIzB,gBAAM,eACJ,gBAAgB,UAChB,gBAAgB,QAChB,gBAAgB;AAClB,cAAI,gBAAgB,iBAAiB,aAAa;AAChD,yBAAa,QAAQ,MAAM,MAAM,YAAY;AAAA,UAC/C;AACA;AAAA,QACF;AAGA,YAAI,iBAAiB,iBAAiB,MAAO;AAC7C,yBAAiB,QAAQ;AAEzB,YAAI,iBAAiB,aAAa;AAChC,uBAAa,QAAQ,MAAM,MAAM,YAAY;AAAA,QAC/C;AAAA,MACF,SAAS,OAAO;AACd,gBAAQ;AAAA,UACN,8BAA8B,QAAQ,MAAM,IAAI;AAAA,UAChD;AAAA,QAAA;AAAA,MAEJ;AAAA,IACF;AAAA,IACA,EAAE,MAAM,MAAM,WAAW,KAAA;AAAA,EAAK;AAElC;AAEA,SAAS,wBACP,YACA,MACA,eACS;AAET,MAAI,WAAW,WAAW,YAAY,KAAK,WAAW,SAAS,GAAG,GAAG;AACnE,UAAM,eAAe,WAAW,MAAM,IAAI,EAAE;AAC5C,WAAO,oBAAoB,cAAc,aAAa;AAAA,EACxD;AAGA,MACE,CAAC,WAAW,SAAS,GAAG,KACxB,CAAC,WAAW,SAAS,GAAG,KACxB,CAAC,WAAW,SAAS,GAAG,GACxB;AACA,WAAO,oBAAoB,YAAY,IAAI;AAAA,EAC7C;AAIA,SAAO,oBAAoB,YAAY,IAAI;AAC7C;AAEA,SAAS,oBAAoB,MAAc,MAAwB;AACjE,MAAI,CAAC,QAAQ,CAAC,KAAM,QAAO;AAE3B,QAAM,OAAO,KAAK,MAAM,GAAG;AAC3B,MAAI,QAAiB;AAErB,aAAW,OAAO,MAAM;AACtB,QAAI,SAAS,OAAO,UAAU,YAAY,OAAO,OAAO;AACtD,cAAS,MAAkC,GAAG;AAAA,IAChD,OAAO;AACL,aAAO;AAAA,IACT;AAAA,EACF;AAEA,SAAO;AACT;"}
@@ -0,0 +1,36 @@
1
+ import { type Ref, type ComputedRef } from "vue";
2
+ import { type ControlElement } from "@jsonforms/core";
3
+ import { type TransformPipeline } from "../../core/transforms";
4
+ import type { AuthConfig } from "../../core/types";
5
+ export interface DeriveInitialValueCfg {
6
+ protocol: string;
7
+ config: {
8
+ url: string;
9
+ method?: "GET" | "POST";
10
+ headers?: Record<string, string>;
11
+ query?: Record<string, unknown>;
12
+ body?: unknown;
13
+ auth?: AuthConfig;
14
+ items: string;
15
+ map: {
16
+ value: string;
17
+ };
18
+ transforms?: TransformPipeline;
19
+ showError?: boolean;
20
+ };
21
+ }
22
+ interface DeriveInitialValueOptions {
23
+ control: Ref<{
24
+ uischema: ControlElement;
25
+ path: string;
26
+ data: unknown;
27
+ }>;
28
+ handleChange: (path: string, value: unknown) => void;
29
+ data?: Ref<unknown> | ComputedRef<unknown>;
30
+ }
31
+ export declare function useDeriveInitialValue({ control, handleChange, }: DeriveInitialValueOptions): {
32
+ loading: Ref<boolean, boolean>;
33
+ error: Ref<string | undefined, string | undefined>;
34
+ };
35
+ export {};
36
+ //# sourceMappingURL=useDeriveInitialValue.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useDeriveInitialValue.d.ts","sourceRoot":"","sources":["../../../src/vue/composables/useDeriveInitialValue.ts"],"names":[],"mappings":"AAAA,OAAO,EAAgC,KAAK,GAAG,EAAE,KAAK,WAAW,EAAE,MAAM,KAAK,CAAC;AAC/E,OAAO,EAAE,KAAK,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAGtD,OAAO,EAEL,KAAK,iBAAiB,EACvB,MAAM,uBAAuB,CAAC;AAC/B,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAEnD,MAAM,WAAW,qBAAqB;IACpC,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE;QACN,GAAG,EAAE,MAAM,CAAC;QACZ,MAAM,CAAC,EAAE,KAAK,GAAG,MAAM,CAAC;QACxB,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;QACjC,KAAK,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QAChC,IAAI,CAAC,EAAE,OAAO,CAAC;QACf,IAAI,CAAC,EAAE,UAAU,CAAC;QAClB,KAAK,EAAE,MAAM,CAAC;QACd,GAAG,EAAE;YAAE,KAAK,EAAE,MAAM,CAAA;SAAE,CAAC;QACvB,UAAU,CAAC,EAAE,iBAAiB,CAAC;QAC/B,SAAS,CAAC,EAAE,OAAO,CAAC;KACrB,CAAC;CACH;AAED,UAAU,yBAAyB;IACjC,OAAO,EAAE,GAAG,CAAC;QACX,QAAQ,EAAE,cAAc,CAAC;QACzB,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,OAAO,CAAC;KACf,CAAC,CAAC;IACH,YAAY,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,KAAK,IAAI,CAAC;IACrD,IAAI,CAAC,EAAE,GAAG,CAAC,OAAO,CAAC,GAAG,WAAW,CAAC,OAAO,CAAC,CAAC;CAC5C;AAsDD,wBAAgB,qBAAqB,CAAC,EACpC,OAAO,EACP,YAAY,GACb,EAAE,yBAAyB;;;EAuG3B"}
@@ -0,0 +1,125 @@
1
+ import { inject, computed, ref, watch } from "vue";
2
+ import "@jsonforms/core";
3
+ import { renderTpl, renderObj } from "../../core/templating.js";
4
+ import { jp } from "../../core/jsonpath.js";
5
+ import { applyTransformPipeline } from "../../core/transforms.js";
6
+ function buildAuthHeaders(auth, globalAuth) {
7
+ const headers = {};
8
+ if (!auth) return headers;
9
+ if (auth.use && globalAuth?.[auth.use]) {
10
+ const globalValue = globalAuth[auth.use];
11
+ const value = typeof globalValue === "function" ? globalValue() : globalValue;
12
+ if (auth.use === "apiKey") headers["X-API-Key"] = String(value);
13
+ else if (auth.use === "bearer")
14
+ headers["Authorization"] = `Bearer ${value}`;
15
+ else if (auth.use === "token") headers["Authorization"] = `Token ${value}`;
16
+ return headers;
17
+ }
18
+ if (auth.bearer) {
19
+ const v = typeof auth.bearer === "function" ? auth.bearer() : auth.bearer;
20
+ headers["Authorization"] = `Bearer ${v}`;
21
+ }
22
+ if (auth.apiKey) {
23
+ const v = typeof auth.apiKey === "function" ? auth.apiKey() : auth.apiKey;
24
+ headers["X-API-Key"] = String(v);
25
+ }
26
+ if (auth.token) {
27
+ const v = typeof auth.token === "function" ? auth.token() : auth.token;
28
+ headers["Authorization"] = `Token ${v}`;
29
+ }
30
+ return headers;
31
+ }
32
+ function hasUnresolvedTemplates(templateUrl, renderedUrl) {
33
+ if (!templateUrl.includes("{{")) return false;
34
+ const pathPart = renderedUrl.replace(/^https?:\/\/[^/]+/, "");
35
+ if (pathPart.includes("//")) return true;
36
+ if (renderedUrl.endsWith("/") && !templateUrl.endsWith("/")) return true;
37
+ return false;
38
+ }
39
+ function useDeriveInitialValue({
40
+ control,
41
+ handleChange
42
+ }) {
43
+ const injectedFormData = inject("formData", {
44
+ value: {}
45
+ });
46
+ const rootData = computed(() => injectedFormData.value || {});
47
+ const auth = inject("providerAuth", {});
48
+ const cfg = computed(() => {
49
+ return control.value.uischema?.options?.deriveInitialValue;
50
+ });
51
+ const resolvedUrl = computed(() => {
52
+ const c = cfg.value;
53
+ if (!c?.config?.url) return null;
54
+ const rendered = renderTpl(c.config.url, { data: rootData.value });
55
+ if (hasUnresolvedTemplates(c.config.url, rendered)) return null;
56
+ return rendered;
57
+ });
58
+ const lastFetchedUrl = ref(null);
59
+ const loading = ref(false);
60
+ const error = ref(void 0);
61
+ watch(
62
+ resolvedUrl,
63
+ async (url) => {
64
+ if (!url || !cfg.value) return;
65
+ if (url === lastFetchedUrl.value) return;
66
+ lastFetchedUrl.value = url;
67
+ loading.value = true;
68
+ error.value = void 0;
69
+ try {
70
+ const c = cfg.value.config;
71
+ const fullUrl = new URL(url);
72
+ const q = renderObj(c.query ?? {}, {
73
+ data: rootData.value
74
+ });
75
+ for (const [k, v] of Object.entries(q)) {
76
+ if (v !== void 0 && v !== "")
77
+ fullUrl.searchParams.set(k, String(v));
78
+ }
79
+ const baseHeaders = renderObj(c.headers ?? {}, {
80
+ data: rootData.value
81
+ });
82
+ const authHeaders = buildAuthHeaders(c.auth, auth);
83
+ const headers = { ...baseHeaders, ...authHeaders };
84
+ const method = c.method ?? "GET";
85
+ const requestInit = { method, headers };
86
+ if (method !== "GET" && c.body) {
87
+ requestInit.body = JSON.stringify(
88
+ renderObj(c.body, { data: rootData.value })
89
+ );
90
+ }
91
+ const res = await fetch(fullUrl.toString(), requestInit);
92
+ if (!res.ok) {
93
+ if (c.showError !== false) {
94
+ throw new Error(`REST ${res.status}`);
95
+ }
96
+ return;
97
+ }
98
+ const json = await res.json();
99
+ let items = jp(json, c.items);
100
+ if (c.transforms && c.transforms.length > 0) {
101
+ items = applyTransformPipeline(items, c.transforms);
102
+ }
103
+ if (items.length === 0) return;
104
+ const derivedValue = jp(items[0], c.map.value)[0];
105
+ if (derivedValue !== void 0) {
106
+ handleChange(control.value.path, derivedValue);
107
+ }
108
+ } catch (e) {
109
+ error.value = e?.message ?? String(e);
110
+ console.warn(
111
+ `deriveInitialValue fetch failed for ${control.value.path}:`,
112
+ e
113
+ );
114
+ } finally {
115
+ loading.value = false;
116
+ }
117
+ },
118
+ { immediate: true }
119
+ );
120
+ return { loading, error };
121
+ }
122
+ export {
123
+ useDeriveInitialValue
124
+ };
125
+ //# sourceMappingURL=useDeriveInitialValue.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useDeriveInitialValue.js","sources":["../../../src/vue/composables/useDeriveInitialValue.ts"],"sourcesContent":["import { computed, inject, ref, watch, type Ref, type ComputedRef } from \"vue\";\nimport { type ControlElement } from \"@jsonforms/core\";\nimport { renderTpl, renderObj } from \"../../core/templating\";\nimport { jp } from \"../../core/jsonpath\";\nimport {\n applyTransformPipeline,\n type TransformPipeline,\n} from \"../../core/transforms\";\nimport type { AuthConfig } from \"../../core/types\";\n\nexport interface DeriveInitialValueCfg {\n protocol: string;\n config: {\n url: string;\n method?: \"GET\" | \"POST\";\n headers?: Record<string, string>;\n query?: Record<string, unknown>;\n body?: unknown;\n auth?: AuthConfig;\n items: string;\n map: { value: string };\n transforms?: TransformPipeline;\n showError?: boolean;\n };\n}\n\ninterface DeriveInitialValueOptions {\n control: Ref<{\n uischema: ControlElement;\n path: string;\n data: unknown;\n }>;\n handleChange: (path: string, value: unknown) => void;\n data?: Ref<unknown> | ComputedRef<unknown>;\n}\n\nfunction buildAuthHeaders(\n auth?: AuthConfig,\n globalAuth?: Record<string, unknown>,\n): Record<string, string> {\n const headers: Record<string, string> = {};\n if (!auth) return headers;\n\n if (auth.use && globalAuth?.[auth.use]) {\n const globalValue = globalAuth[auth.use];\n const value =\n typeof globalValue === \"function\" ? globalValue() : globalValue;\n if (auth.use === \"apiKey\") headers[\"X-API-Key\"] = String(value);\n else if (auth.use === \"bearer\")\n headers[\"Authorization\"] = `Bearer ${value}`;\n else if (auth.use === \"token\") headers[\"Authorization\"] = `Token ${value}`;\n return headers;\n }\n\n if (auth.bearer) {\n const v = typeof auth.bearer === \"function\" ? auth.bearer() : auth.bearer;\n headers[\"Authorization\"] = `Bearer ${v}`;\n }\n if (auth.apiKey) {\n const v = typeof auth.apiKey === \"function\" ? auth.apiKey() : auth.apiKey;\n headers[\"X-API-Key\"] = String(v);\n }\n if (auth.token) {\n const v = typeof auth.token === \"function\" ? auth.token() : auth.token;\n headers[\"Authorization\"] = `Token ${v}`;\n }\n\n return headers;\n}\n\n/**\n * Returns true when the template URL contains `{{…}}` placeholders but\n * one or more of those placeholders resolved to an empty string, which\n * means a required data dependency hasn't been set yet.\n */\nfunction hasUnresolvedTemplates(\n templateUrl: string,\n renderedUrl: string,\n): boolean {\n if (!templateUrl.includes(\"{{\")) return false;\n // After protocol, empty segments indicate unresolved vars\n const pathPart = renderedUrl.replace(/^https?:\\/\\/[^/]+/, \"\");\n if (pathPart.includes(\"//\")) return true;\n // Trailing slash when the template didn't have one\n if (renderedUrl.endsWith(\"/\") && !templateUrl.endsWith(\"/\")) return true;\n return false;\n}\n\nexport function useDeriveInitialValue({\n control,\n handleChange,\n}: DeriveInitialValueOptions) {\n const injectedFormData = inject<{ value: unknown }>(\"formData\", {\n value: {},\n });\n const rootData = computed(() => injectedFormData.value || {});\n const auth = inject(\"providerAuth\", {}) as Record<string, unknown>;\n\n const cfg = computed<DeriveInitialValueCfg | undefined>(() => {\n return control.value.uischema?.options?.deriveInitialValue as\n | DeriveInitialValueCfg\n | undefined;\n });\n\n // Compute the resolved URL reactively\n const resolvedUrl = computed<string | null>(() => {\n const c = cfg.value;\n if (!c?.config?.url) return null;\n const rendered = renderTpl(c.config.url, { data: rootData.value });\n if (hasUnresolvedTemplates(c.config.url, rendered)) return null;\n return rendered;\n });\n\n const lastFetchedUrl = ref<string | null>(null);\n const loading = ref(false);\n const error = ref<string | undefined>(undefined);\n\n watch(\n resolvedUrl,\n async (url) => {\n if (!url || !cfg.value) return;\n // Only fetch when the URL changes (new context).\n // Same URL = same context; don't override user selection.\n if (url === lastFetchedUrl.value) return;\n lastFetchedUrl.value = url;\n\n loading.value = true;\n error.value = undefined;\n\n try {\n const c = cfg.value.config;\n const fullUrl = new URL(url);\n\n // Query params\n const q = renderObj(c.query ?? {}, {\n data: rootData.value,\n }) as Record<string, unknown>;\n for (const [k, v] of Object.entries(q)) {\n if (v !== undefined && v !== \"\")\n fullUrl.searchParams.set(k, String(v));\n }\n\n // Headers\n const baseHeaders = renderObj(c.headers ?? {}, {\n data: rootData.value,\n }) as Record<string, string>;\n const authHeaders = buildAuthHeaders(c.auth, auth);\n const headers = { ...baseHeaders, ...authHeaders };\n\n const method = c.method ?? \"GET\";\n const requestInit: RequestInit = { method, headers };\n if (method !== \"GET\" && c.body) {\n requestInit.body = JSON.stringify(\n renderObj(c.body, { data: rootData.value }),\n );\n }\n\n const res = await fetch(fullUrl.toString(), requestInit);\n if (!res.ok) {\n if (c.showError !== false) {\n throw new Error(`REST ${res.status}`);\n }\n return;\n }\n\n const json = await res.json();\n let items = jp(json, c.items);\n\n // Apply transforms if provided\n if (c.transforms && c.transforms.length > 0) {\n items = applyTransformPipeline(items, c.transforms);\n }\n\n if (items.length === 0) return; // No items → leave field empty\n\n // Extract value from first item\n const derivedValue = jp(items[0], c.map.value)[0];\n if (derivedValue !== undefined) {\n handleChange(control.value.path, derivedValue);\n }\n } catch (e) {\n error.value = (e as Error)?.message ?? String(e);\n console.warn(\n `deriveInitialValue fetch failed for ${control.value.path}:`,\n e,\n );\n } finally {\n loading.value = false;\n }\n },\n { immediate: true },\n );\n\n return { loading, error };\n}\n"],"names":[],"mappings":";;;;;AAoCA,SAAS,iBACP,MACA,YACwB;AACxB,QAAM,UAAkC,CAAA;AACxC,MAAI,CAAC,KAAM,QAAO;AAElB,MAAI,KAAK,OAAO,aAAa,KAAK,GAAG,GAAG;AACtC,UAAM,cAAc,WAAW,KAAK,GAAG;AACvC,UAAM,QACJ,OAAO,gBAAgB,aAAa,gBAAgB;AACtD,QAAI,KAAK,QAAQ,kBAAkB,WAAW,IAAI,OAAO,KAAK;AAAA,aACrD,KAAK,QAAQ;AACpB,cAAQ,eAAe,IAAI,UAAU,KAAK;AAAA,aACnC,KAAK,QAAQ,iBAAiB,eAAe,IAAI,SAAS,KAAK;AACxE,WAAO;AAAA,EACT;AAEA,MAAI,KAAK,QAAQ;AACf,UAAM,IAAI,OAAO,KAAK,WAAW,aAAa,KAAK,WAAW,KAAK;AACnE,YAAQ,eAAe,IAAI,UAAU,CAAC;AAAA,EACxC;AACA,MAAI,KAAK,QAAQ;AACf,UAAM,IAAI,OAAO,KAAK,WAAW,aAAa,KAAK,WAAW,KAAK;AACnE,YAAQ,WAAW,IAAI,OAAO,CAAC;AAAA,EACjC;AACA,MAAI,KAAK,OAAO;AACd,UAAM,IAAI,OAAO,KAAK,UAAU,aAAa,KAAK,UAAU,KAAK;AACjE,YAAQ,eAAe,IAAI,SAAS,CAAC;AAAA,EACvC;AAEA,SAAO;AACT;AAOA,SAAS,uBACP,aACA,aACS;AACT,MAAI,CAAC,YAAY,SAAS,IAAI,EAAG,QAAO;AAExC,QAAM,WAAW,YAAY,QAAQ,qBAAqB,EAAE;AAC5D,MAAI,SAAS,SAAS,IAAI,EAAG,QAAO;AAEpC,MAAI,YAAY,SAAS,GAAG,KAAK,CAAC,YAAY,SAAS,GAAG,EAAG,QAAO;AACpE,SAAO;AACT;AAEO,SAAS,sBAAsB;AAAA,EACpC;AAAA,EACA;AACF,GAA8B;AAC5B,QAAM,mBAAmB,OAA2B,YAAY;AAAA,IAC9D,OAAO,CAAA;AAAA,EAAC,CACT;AACD,QAAM,WAAW,SAAS,MAAM,iBAAiB,SAAS,CAAA,CAAE;AAC5D,QAAM,OAAO,OAAO,gBAAgB,EAAE;AAEtC,QAAM,MAAM,SAA4C,MAAM;AAC5D,WAAO,QAAQ,MAAM,UAAU,SAAS;AAAA,EAG1C,CAAC;AAGD,QAAM,cAAc,SAAwB,MAAM;AAChD,UAAM,IAAI,IAAI;AACd,QAAI,CAAC,GAAG,QAAQ,IAAK,QAAO;AAC5B,UAAM,WAAW,UAAU,EAAE,OAAO,KAAK,EAAE,MAAM,SAAS,OAAO;AACjE,QAAI,uBAAuB,EAAE,OAAO,KAAK,QAAQ,EAAG,QAAO;AAC3D,WAAO;AAAA,EACT,CAAC;AAED,QAAM,iBAAiB,IAAmB,IAAI;AAC9C,QAAM,UAAU,IAAI,KAAK;AACzB,QAAM,QAAQ,IAAwB,MAAS;AAE/C;AAAA,IACE;AAAA,IACA,OAAO,QAAQ;AACb,UAAI,CAAC,OAAO,CAAC,IAAI,MAAO;AAGxB,UAAI,QAAQ,eAAe,MAAO;AAClC,qBAAe,QAAQ;AAEvB,cAAQ,QAAQ;AAChB,YAAM,QAAQ;AAEd,UAAI;AACF,cAAM,IAAI,IAAI,MAAM;AACpB,cAAM,UAAU,IAAI,IAAI,GAAG;AAG3B,cAAM,IAAI,UAAU,EAAE,SAAS,CAAA,GAAI;AAAA,UACjC,MAAM,SAAS;AAAA,QAAA,CAChB;AACD,mBAAW,CAAC,GAAG,CAAC,KAAK,OAAO,QAAQ,CAAC,GAAG;AACtC,cAAI,MAAM,UAAa,MAAM;AAC3B,oBAAQ,aAAa,IAAI,GAAG,OAAO,CAAC,CAAC;AAAA,QACzC;AAGA,cAAM,cAAc,UAAU,EAAE,WAAW,CAAA,GAAI;AAAA,UAC7C,MAAM,SAAS;AAAA,QAAA,CAChB;AACD,cAAM,cAAc,iBAAiB,EAAE,MAAM,IAAI;AACjD,cAAM,UAAU,EAAE,GAAG,aAAa,GAAG,YAAA;AAErC,cAAM,SAAS,EAAE,UAAU;AAC3B,cAAM,cAA2B,EAAE,QAAQ,QAAA;AAC3C,YAAI,WAAW,SAAS,EAAE,MAAM;AAC9B,sBAAY,OAAO,KAAK;AAAA,YACtB,UAAU,EAAE,MAAM,EAAE,MAAM,SAAS,OAAO;AAAA,UAAA;AAAA,QAE9C;AAEA,cAAM,MAAM,MAAM,MAAM,QAAQ,SAAA,GAAY,WAAW;AACvD,YAAI,CAAC,IAAI,IAAI;AACX,cAAI,EAAE,cAAc,OAAO;AACzB,kBAAM,IAAI,MAAM,QAAQ,IAAI,MAAM,EAAE;AAAA,UACtC;AACA;AAAA,QACF;AAEA,cAAM,OAAO,MAAM,IAAI,KAAA;AACvB,YAAI,QAAQ,GAAG,MAAM,EAAE,KAAK;AAG5B,YAAI,EAAE,cAAc,EAAE,WAAW,SAAS,GAAG;AAC3C,kBAAQ,uBAAuB,OAAO,EAAE,UAAU;AAAA,QACpD;AAEA,YAAI,MAAM,WAAW,EAAG;AAGxB,cAAM,eAAe,GAAG,MAAM,CAAC,GAAG,EAAE,IAAI,KAAK,EAAE,CAAC;AAChD,YAAI,iBAAiB,QAAW;AAC9B,uBAAa,QAAQ,MAAM,MAAM,YAAY;AAAA,QAC/C;AAAA,MACF,SAAS,GAAG;AACV,cAAM,QAAS,GAAa,WAAW,OAAO,CAAC;AAC/C,gBAAQ;AAAA,UACN,uCAAuC,QAAQ,MAAM,IAAI;AAAA,UACzD;AAAA,QAAA;AAAA,MAEJ,UAAA;AACE,gBAAQ,QAAQ;AAAA,MAClB;AAAA,IACF;AAAA,IACA,EAAE,WAAW,KAAA;AAAA,EAAK;AAGpB,SAAO,EAAE,SAAS,MAAA;AACpB;"}
@@ -1,9 +1,9 @@
1
- import { type Ref } from "vue";
1
+ import { type Ref, type ComputedRef } from "vue";
2
2
  export declare function useDirtyValidation(control: Ref<{
3
3
  errors: string;
4
- }>): {
4
+ }>, errorsOverride?: Ref<string> | ComputedRef<string>): {
5
5
  hasInteracted: Ref<boolean, boolean>;
6
- showErrors: import("vue").ComputedRef<boolean>;
6
+ showErrors: ComputedRef<boolean>;
7
7
  markDirty: () => void;
8
8
  };
9
9
  //# sourceMappingURL=useDirtyValidation.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"useDirtyValidation.d.ts","sourceRoot":"","sources":["../../../src/vue/composables/useDirtyValidation.ts"],"names":[],"mappings":"AAAA,OAAO,EAAiB,KAAK,GAAG,EAAE,MAAM,KAAK,CAAC;AAE9C,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,GAAG,CAAC;IAAE,MAAM,EAAE,MAAM,CAAA;CAAE,CAAC;;;;EAYlE"}
1
+ {"version":3,"file":"useDirtyValidation.d.ts","sourceRoot":"","sources":["../../../src/vue/composables/useDirtyValidation.ts"],"names":[],"mappings":"AAAA,OAAO,EAAiB,KAAK,GAAG,EAAE,KAAK,WAAW,EAAE,MAAM,KAAK,CAAC;AAEhE,wBAAgB,kBAAkB,CAChC,OAAO,EAAE,GAAG,CAAC;IAAE,MAAM,EAAE,MAAM,CAAA;CAAE,CAAC,EAChC,cAAc,CAAC,EAAE,GAAG,CAAC,MAAM,CAAC,GAAG,WAAW,CAAC,MAAM,CAAC;;;;EAenD"}
@@ -1,8 +1,8 @@
1
1
  import { ref, computed } from "vue";
2
- function useDirtyValidation(control) {
2
+ function useDirtyValidation(control, errorsOverride) {
3
3
  const hasInteracted = ref(false);
4
4
  const showErrors = computed(
5
- () => hasInteracted.value && !!control.value.errors
5
+ () => hasInteracted.value && !!(errorsOverride ? errorsOverride.value : control.value.errors)
6
6
  );
7
7
  const markDirty = () => {
8
8
  hasInteracted.value = true;
@@ -1 +1 @@
1
- {"version":3,"file":"useDirtyValidation.js","sources":["../../../src/vue/composables/useDirtyValidation.ts"],"sourcesContent":["import { ref, computed, type Ref } from \"vue\";\n\nexport function useDirtyValidation(control: Ref<{ errors: string }>) {\n const hasInteracted = ref(false);\n\n const showErrors = computed(\n () => hasInteracted.value && !!control.value.errors,\n );\n\n const markDirty = () => {\n hasInteracted.value = true;\n };\n\n return { hasInteracted, showErrors, markDirty };\n}\n"],"names":[],"mappings":";AAEO,SAAS,mBAAmB,SAAkC;AACnE,QAAM,gBAAgB,IAAI,KAAK;AAE/B,QAAM,aAAa;AAAA,IACjB,MAAM,cAAc,SAAS,CAAC,CAAC,QAAQ,MAAM;AAAA,EAAA;AAG/C,QAAM,YAAY,MAAM;AACtB,kBAAc,QAAQ;AAAA,EACxB;AAEA,SAAO,EAAE,eAAe,YAAY,UAAA;AACtC;"}
1
+ {"version":3,"file":"useDirtyValidation.js","sources":["../../../src/vue/composables/useDirtyValidation.ts"],"sourcesContent":["import { ref, computed, type Ref, type ComputedRef } from \"vue\";\n\nexport function useDirtyValidation(\n control: Ref<{ errors: string }>,\n errorsOverride?: Ref<string> | ComputedRef<string>,\n) {\n const hasInteracted = ref(false);\n\n const showErrors = computed(\n () =>\n hasInteracted.value &&\n !!(errorsOverride ? errorsOverride.value : control.value.errors),\n );\n\n const markDirty = () => {\n hasInteracted.value = true;\n };\n\n return { hasInteracted, showErrors, markDirty };\n}\n"],"names":[],"mappings":";AAEO,SAAS,mBACd,SACA,gBACA;AACA,QAAM,gBAAgB,IAAI,KAAK;AAE/B,QAAM,aAAa;AAAA,IACjB,MACE,cAAc,SACd,CAAC,EAAE,iBAAiB,eAAe,QAAQ,QAAQ,MAAM;AAAA,EAAA;AAG7D,QAAM,YAAY,MAAM;AACtB,kBAAc,QAAQ;AAAA,EACxB;AAEA,SAAO,EAAE,eAAe,YAAY,UAAA;AACtC;"}
@@ -2,6 +2,9 @@ import { type ComputedRef, type Ref } from "vue";
2
2
  interface ProjectionControl {
3
3
  data: unknown;
4
4
  path: string;
5
+ errors: string;
6
+ label?: string;
7
+ required?: boolean;
5
8
  schema: Record<string, any>;
6
9
  uischema: {
7
10
  options?: {
@@ -19,6 +22,10 @@ export interface ProjectionResult {
19
22
  handleProjectedChange: (path: string, value: unknown) => void;
20
23
  /** Whether projection is active */
21
24
  hasProjection: boolean;
25
+ /** Resolved display label (options.label → projected schema title → control.label) */
26
+ projectedLabel: ComputedRef<string>;
27
+ /** Error string combining base-path and projected sub-path errors */
28
+ projectedErrors: ComputedRef<string>;
22
29
  }
23
30
  /**
24
31
  * Composable that wraps a JSON Forms control with projection support.
@@ -1 +1 @@
1
- {"version":3,"file":"useProjection.d.ts","sourceRoot":"","sources":["../../../src/vue/composables/useProjection.ts"],"names":[],"mappings":"AAAA,OAAO,EAAY,KAAK,WAAW,EAAE,KAAK,GAAG,EAAE,MAAM,KAAK,CAAC;AAO3D,UAAU,iBAAiB;IACzB,IAAI,EAAE,OAAO,CAAC;IACd,IAAI,EAAE,MAAM,CAAC;IAEb,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAC5B,QAAQ,EAAE;QAAE,OAAO,CAAC,EAAE;YAAE,UAAU,CAAC,EAAE,MAAM,CAAC;YAAC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAA;SAAE,CAAA;KAAE,CAAC;CACzE;AAED,MAAM,WAAW,gBAAgB;IAC/B,sDAAsD;IACtD,aAAa,EAAE,WAAW,CAAC,OAAO,CAAC,CAAC;IACpC,gEAAgE;IAEhE,eAAe,EAAE,WAAW,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC;IAClD,8DAA8D;IAC9D,qBAAqB,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,KAAK,IAAI,CAAC;IAC9D,mCAAmC;IACnC,aAAa,EAAE,OAAO,CAAC;CACxB;AAED;;;;;;;;;GASG;AACH,wBAAgB,aAAa,CAC3B,OAAO,EAAE,GAAG,CAAC,iBAAiB,CAAC,EAC/B,YAAY,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,KAAK,IAAI,GACnD,gBAAgB,CAiClB"}
1
+ {"version":3,"file":"useProjection.d.ts","sourceRoot":"","sources":["../../../src/vue/composables/useProjection.ts"],"names":[],"mappings":"AAAA,OAAO,EAAoB,KAAK,WAAW,EAAE,KAAK,GAAG,EAAE,MAAM,KAAK,CAAC;AASnE,UAAU,iBAAiB;IACzB,IAAI,EAAE,OAAO,CAAC;IACd,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,QAAQ,CAAC,EAAE,OAAO,CAAC;IAEnB,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAC5B,QAAQ,EAAE;QAAE,OAAO,CAAC,EAAE;YAAE,UAAU,CAAC,EAAE,MAAM,CAAC;YAAC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAA;SAAE,CAAA;KAAE,CAAC;CACzE;AAqGD,MAAM,WAAW,gBAAgB;IAC/B,sDAAsD;IACtD,aAAa,EAAE,WAAW,CAAC,OAAO,CAAC,CAAC;IACpC,gEAAgE;IAEhE,eAAe,EAAE,WAAW,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC;IAClD,8DAA8D;IAC9D,qBAAqB,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,KAAK,IAAI,CAAC;IAC9D,mCAAmC;IACnC,aAAa,EAAE,OAAO,CAAC;IACvB,sFAAsF;IACtF,cAAc,EAAE,WAAW,CAAC,MAAM,CAAC,CAAC;IACpC,qEAAqE;IACrE,eAAe,EAAE,WAAW,CAAC,MAAM,CAAC,CAAC;CACtC;AAED;;;;;;;;;GASG;AACH,wBAAgB,aAAa,CAC3B,OAAO,EAAE,GAAG,CAAC,iBAAiB,CAAC,EAC/B,YAAY,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,KAAK,IAAI,GACnD,gBAAgB,CAgGlB"}
@@ -1,21 +1,102 @@
1
- import { computed } from "vue";
2
- import { getProjectedValue, getProjectedSchema, setProjectedValue } from "../../core/projection.js";
1
+ import { computed, inject } from "vue";
2
+ import { getProjectedValue, getProjectedSchema, parseProjectionPath, setProjectedValue } from "../../core/projection.js";
3
+ import { deref } from "../../core/refs.js";
4
+ function resolveLabel(ctrl, schemaTitle, required) {
5
+ const base = ctrl.uischema?.options?.label ?? schemaTitle ?? ctrl.label ?? "";
6
+ if (!base) return base;
7
+ return required ? `${base} *` : base;
8
+ }
9
+ function isProjectedFieldRequired(schema, path, root) {
10
+ const segments = parseProjectionPath(path);
11
+ if (segments.length === 0) return false;
12
+ const rootSchema = root ?? schema;
13
+ let current = deref(schema, rootSchema);
14
+ for (let i = 0; i < segments.length - 1; i++) {
15
+ const seg = segments[i];
16
+ if (typeof seg === "number") {
17
+ current = deref(current?.items, rootSchema);
18
+ } else {
19
+ current = deref(current?.properties?.[seg], rootSchema);
20
+ }
21
+ if (!current) return false;
22
+ }
23
+ const last = segments[segments.length - 1];
24
+ if (typeof last !== "string") return false;
25
+ current = deref(current, rootSchema);
26
+ return Array.isArray(current?.required) && current.required.includes(last);
27
+ }
28
+ function normalizeErrors(errors) {
29
+ if (!errors) return errors;
30
+ return errors.replace(/is a required property/g, "is required").replace(/must have required property '[^']*'/g, "is required");
31
+ }
32
+ function prefixErrors(label, errors) {
33
+ if (!label || !errors) return errors;
34
+ return errors.split("\n").map((line) => `${label} ${line}`).join("\n");
35
+ }
36
+ function getErrorPath(error) {
37
+ let p = (error.instancePath || "").replace(/\//g, ".").replace(/^\./, "");
38
+ if (error.keyword === "required" && error.params?.missingProperty) {
39
+ p = p ? p + "." + error.params.missingProperty : error.params.missingProperty;
40
+ }
41
+ return p;
42
+ }
3
43
  function useProjection(control, handleChange) {
4
44
  const projection = control.value.uischema?.options?.projection;
5
45
  if (!projection) {
46
+ const label2 = computed(
47
+ () => resolveLabel(control.value, void 0, control.value.required)
48
+ );
6
49
  return {
7
50
  projectedData: computed(() => control.value.data),
8
51
  projectedSchema: computed(() => control.value.schema),
9
52
  handleProjectedChange: handleChange,
10
- hasProjection: false
53
+ hasProjection: false,
54
+ projectedLabel: label2,
55
+ projectedErrors: computed(
56
+ () => prefixErrors(
57
+ label2.value.replace(/\*$/, "").trim(),
58
+ normalizeErrors(control.value.errors)
59
+ )
60
+ )
11
61
  };
12
62
  }
63
+ const jsonforms = inject("jsonforms", null);
64
+ const fullProjectedPath = control.value.path + "." + projection;
13
65
  const projectedData = computed(
14
66
  () => getProjectedValue(control.value.data, projection)
15
67
  );
16
68
  const projectedSchema = computed(
17
69
  () => getProjectedSchema(control.value.schema, projection)
18
70
  );
71
+ const projectedRequired = computed(
72
+ () => isProjectedFieldRequired(
73
+ control.value.schema,
74
+ projection,
75
+ jsonforms?.core?.schema ?? control.value.schema
76
+ )
77
+ );
78
+ const label = computed(
79
+ () => resolveLabel(
80
+ control.value,
81
+ projectedSchema.value?.title,
82
+ projectedRequired.value
83
+ )
84
+ );
85
+ const projectedErrors = computed(() => {
86
+ const baseErrors = normalizeErrors(control.value.errors || "");
87
+ const rawErrors = jsonforms?.core?.errors ?? [];
88
+ const matching = rawErrors.filter(
89
+ (err) => getErrorPath(err) === fullProjectedPath
90
+ );
91
+ let errStr;
92
+ if (matching.length === 0) {
93
+ errStr = baseErrors;
94
+ } else {
95
+ const projMsg = matching.map((e) => e.keyword === "required" ? "is required" : e.message).filter(Boolean).join("\n");
96
+ errStr = [baseErrors, projMsg].filter(Boolean).join("\n");
97
+ }
98
+ return prefixErrors(label.value.replace(/\*$/, "").trim(), errStr);
99
+ });
19
100
  const handleProjectedChange = (path, value) => {
20
101
  const fullValue = setProjectedValue(control.value.data, projection, value);
21
102
  handleChange(path, fullValue);
@@ -24,7 +105,9 @@ function useProjection(control, handleChange) {
24
105
  projectedData,
25
106
  projectedSchema,
26
107
  handleProjectedChange,
27
- hasProjection: true
108
+ hasProjection: true,
109
+ projectedLabel: label,
110
+ projectedErrors
28
111
  };
29
112
  }
30
113
  export {