From 9b46895c0dc433412858aadaa44006f766fecf49 Mon Sep 17 00:00:00 2001 From: Arturo Manzoli Date: Tue, 15 Oct 2024 11:34:53 -0300 Subject: [PATCH] Components: Very-generic-input-widget: Add component Signed-off-by: Arturo Manzoli --- src/assets/defaults.ts | 94 +++ src/assets/widgets/CustomWidgetBase.png | Bin 0 -> 1129 bytes .../CustomWidgetElementInstantiator.vue | 34 ++ src/components/EditMenu.vue | 115 +++- src/components/ElementConfigPanel.vue | 233 +++++++ src/components/ElementHugger.vue | 274 +++++++++ src/components/WidgetHugger.vue | 9 +- .../custom-widget-elements/Button.vue | 61 ++ .../custom-widget-elements/Checkbox.vue | 52 ++ .../custom-widget-elements/Dial.vue | 168 +++++ .../custom-widget-elements/Dropdown.vue | 112 ++++ .../custom-widget-elements/Label.vue | 63 ++ .../custom-widget-elements/Slider.vue | 98 +++ .../custom-widget-elements/Switch.vue | 50 ++ src/components/widgets/CustomWidgetBase.vue | 397 ++++++++++++ src/stores/widgetManager.ts | 159 ++++- src/types/widgets.ts | 574 +++++++++++++++++- src/views/WidgetsView.vue | 1 + 18 files changed, 2477 insertions(+), 17 deletions(-) create mode 100644 src/assets/widgets/CustomWidgetBase.png create mode 100644 src/components/CustomWidgetElementInstantiator.vue create mode 100644 src/components/ElementConfigPanel.vue create mode 100644 src/components/ElementHugger.vue create mode 100644 src/components/custom-widget-elements/Button.vue create mode 100644 src/components/custom-widget-elements/Checkbox.vue create mode 100644 src/components/custom-widget-elements/Dial.vue create mode 100644 src/components/custom-widget-elements/Dropdown.vue create mode 100644 src/components/custom-widget-elements/Label.vue create mode 100644 src/components/custom-widget-elements/Slider.vue create mode 100644 src/components/custom-widget-elements/Switch.vue create mode 100644 src/components/widgets/CustomWidgetBase.vue diff --git a/src/assets/defaults.ts b/src/assets/defaults.ts index a5d2e76dc..e3d568c7a 100644 --- a/src/assets/defaults.ts +++ b/src/assets/defaults.ts @@ -4,6 +4,8 @@ import { DistanceDisplayUnit } from '@/libs/units' import { type MiniWidgetProfile, type Profile, + CustomWidgetManagerVars, + CustomWidgetVarsType, MiniWidgetManagerVars, MiniWidgetType, WidgetManagerVars, @@ -29,6 +31,7 @@ export const defaultWidgetManagerVars: WidgetManagerVars = { lastNonMaximizedWidth: 0.2, lastNonMaximizedHeight: 0.36, highlighted: false, + disableResponsiveness: false, } export const defaultMiniWidgetManagerVars: MiniWidgetManagerVars = { @@ -37,6 +40,97 @@ export const defaultMiniWidgetManagerVars: MiniWidgetManagerVars = { highlighted: false, } +export const defaultCustomWidgetManagerVars: CustomWidgetManagerVars = { + everMounted: false, + configMenuOpen: false, + highlighted: false, + customWidgetVars: CustomWidgetVarsType.Button, + cockpitActions: [], +} + +export const defaultCustomWidgetContainers = [ + { + name: '0-left', + elements: [], + }, + { + name: '1-left', + elements: [], + }, + { + name: '2-left', + elements: [], + }, + { + name: '3-left', + elements: [], + }, + { + name: '4-left', + elements: [], + }, + { + name: '5-left', + elements: [], + }, + { + name: '6-left', + elements: [], + }, + { + name: '7-left', + elements: [], + }, + { + name: '8-left', + elements: [], + }, + { + name: '9-left', + elements: [], + }, + { + name: '0-right', + elements: [], + }, + { + name: '1-right', + elements: [], + }, + { + name: '2-right', + elements: [], + }, + { + name: '3-right', + elements: [], + }, + { + name: '4-right', + elements: [], + }, + { + name: '5-right', + elements: [], + }, + { + name: '6-right', + elements: [], + }, + { + name: '7-right', + elements: [], + }, + { + name: '8-right', + elements: [], + }, + { + name: '9-right', + elements: [], + }, +] + const hostname = window.location.hostname export const defaultBlueOsAddress = 'http://blueos-avahi.local' export const defaultGlobalAddress = hostname == '' || hostname == undefined ? defaultBlueOsAddress : hostname diff --git a/src/assets/widgets/CustomWidgetBase.png b/src/assets/widgets/CustomWidgetBase.png new file mode 100644 index 0000000000000000000000000000000000000000..032ac7f12db21a6ec7cf6edc83af1cd1ea8d64b1 GIT binary patch literal 1129 zcmeAS@N?(olHy`uVBq!ia0vp^FMzm#gAGVB@~Qd*$(BrK=KxP5mDwse2q#6!~Cgpz-XP>U~1pOZCEb&6w)jTQSRGS(epPZp;4A zbJ26QX3gFD{$#v_-V5#1G0RQ6M8r0-a4Lyh33L=Sd$i^Cl86%*DlZ+kfAmYWNb||6 zhKb3HrWq4r?SIJsdABxn&a5w!4tXy4ezef;Si`>91y>8^yiY!!IhWyJ`U9``KeBxj zzD@q(=H4yK*nIoI(Ng!|Zw;qq6MmeW;Cpjw)0UPb!3z7u`KOqjw>xBS-3T%8GOEq6uM1{7NUV9+(SX4w zxV_41F7w}8@f9zn-IJTk`AaYIpQ@Y`8(J$JmHqkR*04DS6Yt&Ic0b8}`?WCVNx%2~ z51vyXI`Q4FPh~Ie&y}=gSyuY*{qsDJO>dsconc+~NW_+XmqZWqy6Fy%3mU+fv9ZYk zf|;0D8=;tmi-{G2IYd~vAXq?yLj-~q0t7T5*kOS}00ct}0~tD44E#{fY^&y(otr9S zu=m=nP5*6pJnO!;Iy|&mbG*@RuKc!Z&(fGD&ztvIW&W>Ei7QssPCBl#Sw(u{p(TZR zzly>`kH3=MysvQI?|&YjB0|}do4eb0zM4PdbzOkmq@2~Swz}z`oph)F-Tni=Z>#={ z3tjIS^nXR=Md#Q)zmNw-eLp{^zYYHXq~^2qt*g1;{#=i~0!(oXp00i_>zopr0NvZg A-~a#s literal 0 HcmV?d00001 diff --git a/src/components/CustomWidgetElementInstantiator.vue b/src/components/CustomWidgetElementInstantiator.vue new file mode 100644 index 000000000..df2ea8f39 --- /dev/null +++ b/src/components/CustomWidgetElementInstantiator.vue @@ -0,0 +1,34 @@ + + + diff --git a/src/components/EditMenu.vue b/src/components/EditMenu.vue index 63aef0f46..5d2373e3d 100644 --- a/src/components/EditMenu.vue +++ b/src/components/EditMenu.vue @@ -453,7 +453,7 @@ theme="dark" variant="filled" density="compact" - :items="['Regular', 'Mini']" + :items="['Regular', 'Mini', 'Custom']" class="bg-[#27384255] 2xl:scale-100 scale-[80%]" hide-details @change="widgetMode = $event" @@ -463,11 +463,8 @@
To be placed on the main view area
-
- Click or drag to add +
+ (Click on card to add)
To be placed on the top and bottom bars @@ -475,6 +472,14 @@
(Drag card in place to add)
+
+ Create Custom Widget + +
@@ -491,7 +496,10 @@ @dragstart="onRegularWidgetDragStart" @dragend="onRegularWidgetDragEnd(widgetType)" > - +
+ +
+ + + diff --git a/src/components/ElementHugger.vue b/src/components/ElementHugger.vue new file mode 100644 index 000000000..7f305a5a8 --- /dev/null +++ b/src/components/ElementHugger.vue @@ -0,0 +1,274 @@ + + + + + diff --git a/src/components/WidgetHugger.vue b/src/components/WidgetHugger.vue index a45a7246f..6f427f046 100644 --- a/src/components/WidgetHugger.vue +++ b/src/components/WidgetHugger.vue @@ -59,6 +59,10 @@ export interface Props { * To hide or not content that overflows the widget area */ hideOverflow?: boolean + /** + * To change the widget sizes according to window size + */ + disableResponsiveness?: boolean } const props = withDefaults(defineProps(), { @@ -75,6 +79,7 @@ const outerWidgetRef = ref() const innerWidgetRef = ref() const widgetView = computed(() => outerWidgetRef.value?.parentElement) const widgetResizeHandles = computed(() => outerWidgetRef.value?.getElementsByClassName('resize-handle')) +const disableResponsiveness = toRefs(props).disableResponsiveness const devStore = useDevelopmentStore() @@ -272,8 +277,8 @@ const temporaryPosition = computed(() => { }) const sizeStyle = computed(() => ({ - width: `${100 * size.value.width}%`, - height: `${100 * size.value.height}%`, + width: disableResponsiveness.value ? `${1000 * size.value.width}px` : `${100 * size.value.width}%`, + height: disableResponsiveness.value ? `${1000 * size.value.height}px` : `${100 * size.value.height}%`, })) const positionStyle = computed(() => ({ diff --git a/src/components/custom-widget-elements/Button.vue b/src/components/custom-widget-elements/Button.vue new file mode 100644 index 000000000..458a57d16 --- /dev/null +++ b/src/components/custom-widget-elements/Button.vue @@ -0,0 +1,61 @@ + + + + + diff --git a/src/components/custom-widget-elements/Checkbox.vue b/src/components/custom-widget-elements/Checkbox.vue new file mode 100644 index 000000000..b23e13c56 --- /dev/null +++ b/src/components/custom-widget-elements/Checkbox.vue @@ -0,0 +1,52 @@ + + + + + diff --git a/src/components/custom-widget-elements/Dial.vue b/src/components/custom-widget-elements/Dial.vue new file mode 100644 index 000000000..27a8cbcc0 --- /dev/null +++ b/src/components/custom-widget-elements/Dial.vue @@ -0,0 +1,168 @@ + + + + + diff --git a/src/components/custom-widget-elements/Dropdown.vue b/src/components/custom-widget-elements/Dropdown.vue new file mode 100644 index 000000000..81b42345e --- /dev/null +++ b/src/components/custom-widget-elements/Dropdown.vue @@ -0,0 +1,112 @@ + + + + + diff --git a/src/components/custom-widget-elements/Label.vue b/src/components/custom-widget-elements/Label.vue new file mode 100644 index 000000000..5431accf0 --- /dev/null +++ b/src/components/custom-widget-elements/Label.vue @@ -0,0 +1,63 @@ + + + + + diff --git a/src/components/custom-widget-elements/Slider.vue b/src/components/custom-widget-elements/Slider.vue new file mode 100644 index 000000000..f6bdef828 --- /dev/null +++ b/src/components/custom-widget-elements/Slider.vue @@ -0,0 +1,98 @@ + + + + + diff --git a/src/components/custom-widget-elements/Switch.vue b/src/components/custom-widget-elements/Switch.vue new file mode 100644 index 000000000..f37b01725 --- /dev/null +++ b/src/components/custom-widget-elements/Switch.vue @@ -0,0 +1,50 @@ + + + + + diff --git a/src/components/widgets/CustomWidgetBase.vue b/src/components/widgets/CustomWidgetBase.vue new file mode 100644 index 000000000..c09b0cf5d --- /dev/null +++ b/src/components/widgets/CustomWidgetBase.vue @@ -0,0 +1,397 @@ + + + + + diff --git a/src/stores/widgetManager.ts b/src/stores/widgetManager.ts index 6f0220b01..1553f06e7 100644 --- a/src/stores/widgetManager.ts +++ b/src/stores/widgetManager.ts @@ -7,6 +7,7 @@ import { v4 as uuid4 } from 'uuid' import { computed, onBeforeMount, onBeforeUnmount, Ref, ref, watch } from 'vue' import { + defaultCustomWidgetContainers, defaultMiniWidgetManagerVars, defaultProfileVehicleCorrespondency, defaultWidgetManagerVars, @@ -16,6 +17,7 @@ import { miniWidgetsProfile } from '@/assets/defaults' import { useInteractionDialog } from '@/composables/interactionDialog' import { resetJustMadeKey, useBlueOsStorage } from '@/composables/settingsSyncer' import { openSnackbar } from '@/composables/snackbar' +import { useSnackbar } from '@/composables/snackbar' import { MavType } from '@/libs/connection/m2r/messages/mavlink2rest-enum' import * as Words from '@/libs/funny-name/words' import { @@ -32,6 +34,8 @@ import { type Profile, type View, type Widget, + CustomWidgetElement, + CustomWidgetElementContainer, MiniWidgetManagerVars, validateProfile, validateView, @@ -40,6 +44,7 @@ import { } from '@/types/widgets' const { showDialog } = useInteractionDialog() +const { showSnackbar } = useSnackbar() export const savedProfilesKey = 'cockpit-saved-profiles-v8' @@ -60,6 +65,131 @@ export const useWidgetManagerStore = defineStore('widget-manager', () => { ) const _widgetManagerVars: Ref> = ref({}) const _miniWidgetManagerVars: Ref> = ref({}) + const isElementsPropsDrawerVisible = ref(false) + const elementToShowOnDrawer = ref() + const widgetToEdit = ref() + + const editWidgetByHash = (hash: string): Widget | undefined => { + widgetToEdit.value = currentProfile.value.views + .flatMap((view) => view.widgets) + .find((widget) => widget.hash === hash) + return widgetToEdit.value + } + + const getElementByHash = (hash: string): CustomWidgetElement | undefined => { + const customWidgetElement = currentProfile.value.views + .flatMap((view) => view.widgets) + .filter((widget) => widget.component === WidgetType.CustomWidgetBase) + .flatMap((widget) => widget.options.elementContainers) + .flatMap((container) => container.elements) + .find((element) => element.hash === hash) + + return customWidgetElement + } + + const showElementPropsDrawer = (customWidgetElementHash: string): void => { + console.log('🚀 ~ customWidgetElementHash:', customWidgetElementHash) + const customWidgetElement = getElementByHash(customWidgetElementHash) + if (!customWidgetElement) { + showSnackbar({ variant: 'error', message: 'Could not find element with the given hash.', duration: 3000 }) + return + } + elementToShowOnDrawer.value = customWidgetElement + isElementsPropsDrawerVisible.value = true + } + + const removeElementFromCustomWidget = (elementHash: string): void => { + const customWidgetElement = getElementByHash(elementHash) + if (!customWidgetElement) { + showSnackbar({ variant: 'error', message: 'Could not find element with the given hash.', duration: 3000 }) + return + } + + const customWidget = currentProfile.value.views + .flatMap((view) => view.widgets) + .filter((widget) => widget.component === WidgetType.CustomWidgetBase) + .find((widget) => + widget.options.elementContainers.some((container: CustomWidgetElementContainer) => + container.elements.includes(customWidgetElement) + ) + ) + + if (!customWidget) { + showSnackbar({ + variant: 'error', + message: 'Could not find the custom widget containing the element.', + duration: 3000, + }) + return + } + + const customWidgetContainer = customWidget.options.elementContainers.find( + (container: CustomWidgetElementContainer) => container.elements.includes(customWidgetElement) + ) + if (!customWidgetContainer) { + showSnackbar({ + variant: 'error', + message: 'Could not find the container containing the element.', + duration: 3000, + }) + return + } + + customWidgetContainer.elements = customWidgetContainer.elements.filter( + (element: CustomWidgetElement) => element.hash !== elementHash + ) + } + + const downloadWidgetAsBrw = (widget: Widget): void => { + const blob = new Blob([JSON.stringify(widget)], { type: 'application/json;charset=utf-8' }) + const fileName = `${widget.name}.brw` + saveAs(blob, fileName) + } + + const loadWidgetFromFile = (widgetHash: string, loadedWidget: Widget): void => { + const currentViewWidgets = currentProfile.value.views[currentViewIndex.value].widgets + const widgetIndex = currentViewWidgets.findIndex((widget) => widget.hash === widgetHash) + + if (widgetIndex === -1) { + showSnackbar({ variant: 'error', message: 'Widget not found with the given hash.', duration: 3000 }) + return + } + + const currentPosition = currentViewWidgets[widgetIndex].position + const newWidgetHash = uuid4() + loadedWidget.hash = newWidgetHash + loadedWidget.position = currentPosition + + currentViewWidgets[widgetIndex] = loadedWidget + showSnackbar({ variant: 'success', message: 'Widget loaded successfully with new hash.', duration: 3000 }) + } + + /** + * Updates the options of a custom widget element by its hash. + * @param {string} elementHash - The unique identifier of the element. + * @param {Record} newOptions - The new options to merge with the existing ones. + */ + const updateElementOptions = (elementHash: string, newOptions: Record): void => { + const element = getElementByHash(elementHash) + if (element) { + element.options = { + ...element.options, + ...newOptions, + } + } else { + showSnackbar({ variant: 'error', message: 'Element not found.', duration: 3000 }) + } + } + + const allowMovingAndResizing = (widgetHash: string, forcedState: boolean): void => { + currentProfile.value.views.forEach((view) => { + view.widgets.forEach((widget) => { + if (widget.hash === widgetHash) { + widgetManagerVars(widgetHash).allowMoving = forcedState + } + }) + }) + } const widgetManagerVars = (widgetHash: string): WidgetManagerVars => { if (!_widgetManagerVars.value[widgetHash]) { @@ -412,6 +542,7 @@ export const useWidgetManagerStore = defineStore('widget-manager', () => { */ function addWidget(widgetType: WidgetType, view: View): void { const widgetHash = uuid4() + let disableWidgetResponsiveness = false const widget = { hash: widgetHash, @@ -422,8 +553,23 @@ export const useWidgetManagerStore = defineStore('widget-manager', () => { options: {}, } + if (widgetType === WidgetType.CustomWidgetBase) { + disableWidgetResponsiveness = true + widget.options = { + elementContainers: defaultCustomWidgetContainers, + columns: 1, + leftColumnWidth: 50, + backgroundOpacity: 0.2, + backgroundColor: '#FFFFFF', + backgroundBlur: 25, + } + } + view.widgets.unshift(widget) - Object.assign(widgetManagerVars(widget.hash), { ...defaultWidgetManagerVars, ...{ allowMoving: true } }) + Object.assign(widgetManagerVars(widget.hash), { + ...defaultWidgetManagerVars, + ...{ allowMoving: true, disableResponsiveness: disableWidgetResponsiveness }, + }) } /** @@ -434,6 +580,7 @@ export const useWidgetManagerStore = defineStore('widget-manager', () => { const view = viewFromWidget(widget) const index = view.widgets.indexOf(widget) view.widgets.splice(index, 1) + elementToShowOnDrawer.value = undefined } /** @@ -682,6 +829,7 @@ export const useWidgetManagerStore = defineStore('widget-manager', () => { currentMiniWidgetsProfile, savedProfiles, vehicleTypeProfileCorrespondency, + allowMovingAndResizing, loadProfile, saveProfile, resetSavedProfiles, @@ -713,5 +861,14 @@ export const useWidgetManagerStore = defineStore('widget-manager', () => { visibleAreaMinClearancePixels, currentTopBarHeightPixels, currentBottomBarHeightPixels, + showElementPropsDrawer, + isElementsPropsDrawerVisible, + elementToShowOnDrawer, + updateElementOptions, + removeElementFromCustomWidget, + downloadWidgetAsBrw, + loadWidgetFromFile, + widgetToEdit, + editWidgetByHash, } }) diff --git a/src/types/widgets.ts b/src/types/widgets.ts index 23e2f704e..df5094154 100644 --- a/src/types/widgets.ts +++ b/src/types/widgets.ts @@ -1,3 +1,5 @@ +import { CockpitAction } from '@/libs/joystick/protocols/cockpit-actions' + import type { Point2D, SizeRect2D } from './general' /** @@ -7,8 +9,9 @@ import type { Point2D, SizeRect2D } from './general' export enum WidgetType { Attitude = 'Attitude', Compass = 'Compass', - DepthHUD = 'DepthHUD', CompassHUD = 'CompassHUD', + CustomWidgetBase = 'CustomWidgetBase', + DepthHUD = 'DepthHUD', IFrame = 'IFrame', ImageView = 'ImageView', Map = 'Map', @@ -41,6 +44,467 @@ export enum MiniWidgetType { ViewSelector = 'ViewSelector', } +/** + * Available elements to be used in the Custom Widget creator. + * The enum value is equal to the component's filename, without the '.vue' extension + */ +export enum CustomWidgetElementType { + Button = 'Button', + Checkbox = 'Checkbox', + Dial = 'Dial', + Dropdown = 'Dropdown', + Label = 'Label', + Slider = 'Slider', + Switch = 'Switch', +} + +/** + * Available variables to be used in the Custom Widget creator. + */ +export enum CustomWidgetVarsType { + Button = 'boolean', + Checkbox = 'boolean', + Dial = 'number', + Dropdown = 'string', + Label = 'string', + Slider = 'number', + Switch = 'boolean', +} + +/** + * Available containers to be used in the Custom Widget creator. + */ +export enum CustomWidgetElementContainers { + Left0 = '0-left', + Left1 = '1-left', + Left2 = '2-left', + Left3 = '3-left', + Left4 = '4-left', + Left5 = '5-left', + Left6 = '6-left', + Left7 = '7-left', + Left8 = '8-left', + Left9 = '9-left', + Right0 = '0-right', + Right1 = '1-right', + Right2 = '2-right', + Right3 = '3-right', + Right4 = '4-right', + Right5 = '5-right', + Right6 = '6-right', + Right7 = '7-right', + Right8 = '8-right', + Right9 = '9-right', +} + +export type SelectorOption = { + /** + * The name of the option + */ + name: string + /** + * The value of the option + */ + value: string +} + +/** + * Options for the Cockpit Actions parameters + */ +export interface CockpitActionParameter { + /** + * Parameter ID, equals to initial name of the parameter + */ + id: string + /** + * Parameter name + */ + name: string + /** + * Parameter type + */ + type: 'string' | 'boolean' | 'number' + /** + * Parameter description + */ + description?: string +} + +/** + * Options for the Custom Widgets inner elements + */ +export type CustomWidgetElementOptions = { + /** + * Custom widget element - Label + */ + [CustomWidgetElementType.Label]: { + /** + * Element hash + */ + hash: string + /** + * Element name + */ + name: string + /** + * Element options + */ + options: { + /** + * Variable type + */ + variableType: string + /** + * Action parameter + */ + actionParameter: CockpitActionParameter + /** + * The label text + */ + text: string + /** + * Layout options + */ + layout: { + /** + * The size of the label's font (in pixels) + */ + textSize: number + /** + * Alignment of the element + */ + align: 'start' | 'center' | 'end' + /** + * The weight of the label's font + */ + weight: 'normal' | 'bold' | 'bolder' | 'lighter' + /** + * The decoration for the label's text + */ + decoration: 'none' | 'underline' | 'line-through' | 'overline' + /** + * The color of the label's text + */ + color: string + } + } + } + /** + * Custom widget element - Button + */ + [CustomWidgetElementType.Button]: { + /** + * Element hash + */ + hash: string + /** + * Element name + */ + name: string + /** + * Element options + */ + options: { + /** + * Variable type + */ + variableType: string + /** + * Action parameter + */ + cockpitAction: CockpitAction + /** + * Layout options + */ + layout: { + /** + * Alignment of the element + */ + align: 'start' | 'center' | 'end' + /** + * The label of the button + */ + label: string + /** + * The size of the button + */ + buttonSize: 'small' | 'default' | 'large' + /** + * The color of the button + */ + backgroundColor: string + /** + * The color of the button's text + */ + textColor: string + /** + * The variant of the button + */ + variant: 'text' | 'outlined' | 'flat' | 'elevated' | 'tonal' | 'plain' + } + } + } + /** + * Custom widget element - Checkbox + */ + [CustomWidgetElementType.Checkbox]: { + /** + * Element hash + */ + hash: string + /** + * Element name + */ + name: string + /** + * Element value + */ + checked: boolean + /** + * Element options + */ + options: { + /** + * Variable type + */ + variableType: string + /** + * Action parameter + */ + actionParameter: CockpitActionParameter + /** + * Layout props for the element + */ + layout: { + /** + * Alignment of the element + */ + align: 'start' | 'center' | 'end' + /** + * The size of the checkbox + */ + color: string + /** + * The label of the checkbox + */ + label: string + } + } + } + /** + * Custom widget element - Dial + */ + [CustomWidgetElementType.Dial]: { + /** + * Element hash + */ + hash: string + /** + * Element name + */ + name: string + /** + * Element options + */ + options: { + /** + * Variable type + */ + variableType: string + /** + * Action parameter + */ + actionParameter: CockpitActionParameter + /** + * Layout options + */ + layout: { + /** + * Alignment of the element + */ + align: 'start' | 'center' | 'end' + /** + * The size of the dial + */ + size: 'small' | 'medium' | 'large' + /** + * The color of the dial + */ + color: string + /** + * The minimum value of the dial + */ + minValue: number + /** + * The maximum value of the dial + */ + maxValue: number + /** + * The step value of the dial + */ + showValue: boolean + } + } + } + /** + * Custom widget element - Dropdown + */ + [CustomWidgetElementType.Dropdown]: { + /** + * Element hash + */ + hash: string + /** + * Element name + */ + name: string + /** + * Element options + */ + options: { + /** + * Variable type + */ + variableType: string + /** + * Action parameter + */ + actionParameter: CockpitActionParameter + /** + * Last selected value + */ + lastSelected: SelectorOption + /** + * Layout options + */ + layout: { + /** + * Alignment of the element + */ + selectorOptions: SelectorOption[] + /** + * Alignment of the element + */ + align: 'start' | 'center' | 'end' + /** + * The size of the dropdown + */ + width: number + } + } + } + /** + * Custom widget element - Slider + */ + [CustomWidgetElementType.Slider]: { + /** + * Element hash + */ + hash: string + /** + * Element name + */ + name: string + /** + * Element options + */ + options: { + /** + * Variable type + */ + variableType: string + /** + * Action parameter + */ + actionParameter: CockpitActionParameter + /** + * Layout options + */ + layout: { + /** + * Alignment of the element + */ + align: 'start' | 'center' | 'end' + /** + * The size of the slider + */ + size: 'small' | 'medium' | 'large' + /** + * The color of the slider + */ + color: string + /** + * The minimum value of the slider + */ + minValue: number + /** + * The maximum value of the slider + */ + maxValue: number + /** + * The step value of the slider + */ + showTooltip: boolean + /** + * The label of the slider + */ + label: string + /** + * The width of the label + */ + labelWidth: number + } + } + } + /** + * Custom widget element - Switch + */ + [CustomWidgetElementType.Switch]: { + /** + * Element hash + */ + hash: string + /** + * Element name + */ + name: string + /** + * Element options + */ + options: { + /** + * Variable type + */ + variableType: string + /** + * Action parameter + */ + actionParameter: CockpitActionParameter + /** + * Layout options + */ + layout: { + /** + * Alignment of the element + */ + align: 'start' | 'center' | 'end' + /** + * The size of the switch + */ + size: 'small' | 'medium' | 'large' + /** + * The color of the switch + */ + color: string + /** + * The label of the switch + */ + label: string + } + } + } +} + /** * External variables used by the widget manager */ @@ -77,6 +541,10 @@ export type WidgetManagerVars = { * Wether thewidget should be highlited or not */ highlighted: boolean + /** + * Wether the widget should be resized according to te window size + */ + disableResponsiveness: boolean } /** @@ -97,6 +565,32 @@ export type MiniWidgetManagerVars = { highlighted: boolean } +/** + * External variables used by the custom widget manager + */ +export type CustomWidgetManagerVars = { + /** + * Number of times the custom widget was mounted + */ + everMounted: boolean + /** + * If the configuration menu is open or not + */ + configMenuOpen: boolean + /** + * Wether the custom widget should be highlited or not + */ + highlighted: boolean + /** + * Variables that are used by the custom widget + */ + customWidgetVars: CustomWidgetVarsType + /** + * Cockpit actions that are used by the custom widget + */ + cockpitActions: CockpitAction[] +} + export type Widget = { /** * Unique identifier for the widget @@ -143,6 +637,73 @@ export type MiniWidget = { options: Record // eslint-disable-line @typescript-eslint/no-explicit-any } +export type CustomWidget = { + /** + * Unique identifier for the widget + */ + hash: string + /** + * Component type of the widget + */ + component: WidgetType + /** + * 2D position of the widget (top-left corner) + */ + position: Point2D + /** + * Size of the widget box + */ + size: SizeRect2D + /** + * Editable name for the widget + */ + name: string + /** + * Internal options of the widget + */ + elementContainers: Array<{ + /** + * Editable name for the container + */ + name: CustomWidgetElementContainers + /** + * Array of elements that are stored in the container + */ + elements: CustomWidgetElement[] + }> + /** + * Internal options of the widget + */ + options: Record // eslint-disable-line @typescript-eslint/no-explicit-any +} + +export type CustomWidgetElement = { + /** + * Unique identifier for the widget + */ + hash: string + /** + * Editable name for the widget + */ + name: string + /** + * Component type of the element + */ + component: CustomWidgetElementType + /** + * If the element is a boolean input or not + */ + isBoolean: boolean + /** + * Internal options of the widget + */ + options: Record // eslint-disable-line @typescript-eslint/no-explicit-any + /** + * Internal variables of the widget + */ + cockpitActions: CockpitAction[] +} + export type MiniWidgetContainer = { /** * Array of widgets that are stored in the container @@ -154,6 +715,17 @@ export type MiniWidgetContainer = { name: string } +export type CustomWidgetElementContainer = { + /** + * Array of widgets that are stored in the container + */ + elements: CustomWidgetElement[] + /** + * Editable name for the container + */ + name: string +} + export type MiniWidgetProfile = { /** * Array of views that are stored in the profile diff --git a/src/views/WidgetsView.vue b/src/views/WidgetsView.vue index 0764e77db..c015b48da 100644 --- a/src/views/WidgetsView.vue +++ b/src/views/WidgetsView.vue @@ -18,6 +18,7 @@ :widget="widget" :allow-moving="store.widgetManagerVars(widget.hash).allowMoving" :allow-resizing="store.editingMode" + :disable-responsiveness="store.widgetManagerVars(widget.hash).disableResponsiveness" >