diff --git a/docs/framework/public/components/progressbar.png b/docs/framework/public/components/progressbar.png new file mode 100644 index 000000000..872528352 Binary files /dev/null and b/docs/framework/public/components/progressbar.png differ diff --git a/src/ui/src/components/core/base/BaseInputWrapper.vue b/src/ui/src/components/core/base/BaseInputWrapper.vue index 1cb3b391b..ed3b64273 100644 --- a/src/ui/src/components/core/base/BaseInputWrapper.vue +++ b/src/ui/src/components/core/base/BaseInputWrapper.vue @@ -4,16 +4,16 @@ class="BaseInputWrapper" :class="{ horizontal: isHorizontal }" > - + diff --git a/src/ui/src/composables/useFieldValue.ts b/src/ui/src/composables/useFieldValue.ts new file mode 100644 index 000000000..ed77bb76f --- /dev/null +++ b/src/ui/src/composables/useFieldValue.ts @@ -0,0 +1,33 @@ +import { ComputedRef, computed } from "vue"; + +type Fields = Record>; + +export function useFieldValueAsString( + fields: Fields, + key: string, + fallback?: string, +): ComputedRef { + return computed(() => + fields[key]?.value === undefined + ? fallback + : String(fields[key]?.value), + ); +} + +export function useFieldValueAsNumber( + fields: Fields, + key: string, + fallback?: number, +): ComputedRef { + return computed(() => { + if (fields[key]?.value === undefined) return fallback; + return Number(fields[key]?.value); + }); +} + +export function useFieldValueAsYesNo( + fields: Fields, + key: string, +): ComputedRef { + return computed(() => fields[key].value === "yes"); +} diff --git a/src/ui/src/composables/useFormater.ts b/src/ui/src/composables/useFormater.ts new file mode 100644 index 000000000..339229769 --- /dev/null +++ b/src/ui/src/composables/useFormater.ts @@ -0,0 +1,18 @@ +import { computed, ComputedRef } from "vue"; + +export function usePercentageFormater( + number: ComputedRef, + options: Pick< + Intl.NumberFormatOptions, + "minimumFractionDigits" | "maximumFractionDigits" + > = { + minimumFractionDigits: 0, + maximumFractionDigits: 1, + }, +) { + const formatter = new Intl.NumberFormat(undefined, { + style: "percent", + ...options, + }); + return computed(() => formatter.format(number.value)); +} diff --git a/src/ui/src/core/templateMap.ts b/src/ui/src/core/templateMap.ts index c3031ccbd..ec4b344f2 100644 --- a/src/ui/src/core/templateMap.ts +++ b/src/ui/src/core/templateMap.ts @@ -17,6 +17,7 @@ import CoreTags from "../components/core/content/CoreTags.vue"; import CoreAvatar from "../components/core/content/CoreAvatar.vue"; import CoreAnnotatedText from "../components/core/content/CoreAnnotatedText.vue"; import CoreJsonViewer from "../components/core/content/CoreJsonViewer.vue"; +import CoreProgressBar from "../components/core/content/CoreProgressBar.vue"; // input import CoreCheckboxInput from "../components/core/input/CoreCheckboxInput.vue"; @@ -136,6 +137,7 @@ const templateMap: TemplateMap = { jsonviewer: CoreJsonViewer, workflows_root: WorkflowsRoot, workflows_workflow: WorkflowsWorkflow, + progressbar: CoreProgressBar, }; const abstractTemplateMap: Record = {}; diff --git a/src/ui/src/renderer/syntheticEvents.ts b/src/ui/src/renderer/syntheticEvents.ts index df7bab08c..5fcf3b229 100644 --- a/src/ui/src/renderer/syntheticEvents.ts +++ b/src/ui/src/renderer/syntheticEvents.ts @@ -1,4 +1,7 @@ -export function getClick(ev: MouseEvent): CustomEvent { +/** + * @param ev event from a mouse click, or a keyboard click (using tab navigation, then click with `Enter`) + */ +export function getClick(ev: MouseEvent | KeyboardEvent): CustomEvent { const payload = { ctrlKey: ev.ctrlKey, shiftKey: ev.shiftKey,