From 1770041757717cb47c0a4999f7f765370f93ae4b Mon Sep 17 00:00:00 2001 From: David Date: Wed, 4 Oct 2023 10:28:28 +0200 Subject: [PATCH] F OpenNebula/one#6286: Sunstone video section (#2769) --- .../inputOutput/index.js | 2 + .../inputOutput/schema.js | 3 + .../ExtraConfiguration/inputOutput/index.js | 9 +- .../ExtraConfiguration/inputOutput/schema.js | 3 + .../inputOutput/videoSchema.js | 284 ++++++++++++++++++ .../inputOutput/videoSection.js | 79 +++++ .../src/client/constants/translates.js | 20 ++ .../src/client/constants/vmTemplate.js | 15 + .../form-panels/create/wizard-tabs/io.js | 169 ++++++++++- .../create/wizard-tabs/io/html.hbs | 71 +++++ 10 files changed, 653 insertions(+), 2 deletions(-) create mode 100644 src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/videoSchema.js create mode 100644 src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/videoSection.js diff --git a/src/fireedge/src/client/components/Forms/Vm/UpdateConfigurationForm/inputOutput/index.js b/src/fireedge/src/client/components/Forms/Vm/UpdateConfigurationForm/inputOutput/index.js index 47db877455..e27faeb5b9 100644 --- a/src/fireedge/src/client/components/Forms/Vm/UpdateConfigurationForm/inputOutput/index.js +++ b/src/fireedge/src/client/components/Forms/Vm/UpdateConfigurationForm/inputOutput/index.js @@ -20,6 +20,7 @@ import { Stack } from '@mui/material' import { FormWithSchema } from 'client/components/Forms' import InputsSection from 'client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/inputsSection' +import VideoSection from 'client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/videoSection' import { GRAPHICS_FIELDS } from 'client/components/Forms/Vm/UpdateConfigurationForm/inputOutput/schema' import { T, HYPERVISORS } from 'client/constants' @@ -40,6 +41,7 @@ const InputOutput = ({ hypervisor }) => ( legend={T.Graphics} /> + ) diff --git a/src/fireedge/src/client/components/Forms/Vm/UpdateConfigurationForm/inputOutput/schema.js b/src/fireedge/src/client/components/Forms/Vm/UpdateConfigurationForm/inputOutput/schema.js index 46aa3508c4..703a031ed4 100644 --- a/src/fireedge/src/client/components/Forms/Vm/UpdateConfigurationForm/inputOutput/schema.js +++ b/src/fireedge/src/client/components/Forms/Vm/UpdateConfigurationForm/inputOutput/schema.js @@ -17,6 +17,8 @@ import { object, ObjectSchema } from 'yup' import * as ioSchema from 'client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/schema' +import { VIDEO_SCHEMA } from 'client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/videoSchema' + import { Field, filterFieldsByHypervisor, @@ -55,3 +57,4 @@ export const SCHEMA = ({ hypervisor }) => object() .concat(ioSchema.INPUTS_SCHEMA) .concat(GRAPHICS_SCHEMA({ hypervisor })) + .concat(VIDEO_SCHEMA(hypervisor)) diff --git a/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/index.js b/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/index.js index 0445779527..97e2617633 100644 --- a/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/index.js +++ b/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/index.js @@ -25,10 +25,11 @@ import { } from 'client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration' import InputsSection, { SECTION_ID as INPUT_ID } from './inputsSection' import PciDevicesSection, { SECTION_ID as PCI_ID } from './pciDevicesSection' +import VideoSection, { SECTION_ID as VIDEO_ID } from './videoSection' import { GRAPHICS_FIELDS } from './schema' import { T } from 'client/constants' -export const TAB_ID = ['GRAPHICS', INPUT_ID, PCI_ID] +export const TAB_ID = ['GRAPHICS', INPUT_ID, PCI_ID, VIDEO_ID] const InputOutput = ({ hypervisor, oneConfig, adminGroup }) => ( ( oneConfig={oneConfig} adminGroup={adminGroup} /> + ) diff --git a/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/schema.js b/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/schema.js index da0686aa05..07af20a5d2 100644 --- a/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/schema.js +++ b/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/schema.js @@ -18,6 +18,7 @@ import { object, ObjectSchema } from 'yup' import { GRAPHICS_SCHEMA } from './graphicsSchema' import { INPUTS_SCHEMA } from './inputsSchema' import { PCI_DEVICES_SCHEMA } from './pciDevicesSchema' +import { VIDEO_SCHEMA } from './videoSchema' /** * @param {string} [hypervisor] - VM hypervisor @@ -28,7 +29,9 @@ export const SCHEMA = (hypervisor) => .concat(INPUTS_SCHEMA) .concat(PCI_DEVICES_SCHEMA) .concat(GRAPHICS_SCHEMA(hypervisor)) + .concat(VIDEO_SCHEMA(hypervisor)) export * from './graphicsSchema' export * from './inputsSchema' export * from './pciDevicesSchema' +export * from './videoSchema' diff --git a/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/videoSchema.js b/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/videoSchema.js new file mode 100644 index 0000000000..29b7dc056e --- /dev/null +++ b/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/videoSchema.js @@ -0,0 +1,284 @@ +/* ------------------------------------------------------------------------- * + * Copyright 2002-2023, OpenNebula Project, OpenNebula Systems * + * * + * Licensed under the Apache License, Version 2.0 (the "License"); you may * + * not use this file except in compliance with the License. You may obtain * + * a copy of the License at * + * * + * http://www.apache.org/licenses/LICENSE-2.0 * + * * + * Unless required by applicable law or agreed to in writing, software * + * distributed under the License is distributed on an "AS IS" BASIS, * + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * + * See the License for the specific language governing permissions and * + * limitations under the License. * + * ------------------------------------------------------------------------- */ +import { string, ObjectSchema, boolean, number } from 'yup' + +import { + Field, + arrayToOptions, + filterFieldsByHypervisor, + getObjectSchemaFromFields, + disableFields, +} from 'client/utils' +import { + T, + INPUT_TYPES, + HYPERVISORS, + VIDEO_TYPES, + COMMON_RESOLUTIONS, +} from 'client/constants' + +const { kvm, dummy } = HYPERVISORS + +/** + * Schema for VIDEO section of the template. Considerations about fields behavior according to the core: + * - Only if the hypervisor it's kvm. + * - If the user select VIDEO_TYPE auto, the rest of the fields will be hidden and the request has not any VIDEO section. + * - If the user select VIDEO_TYPE none, the rest of the fields will be hidden and the request has VIDEO section but only with type=none. + * - If the user select VIDEO_TYPE cirrus, ATS, IOMMU and resolution will be hidden. + * - If the user select VIDEO_TYPE vga, ATS and IOMMU will be hidden. + * - If the user select VIDEO_TYPE virtio, all attributes will be show to the user. + * - Resolution will be one of the values of the COMMON_RESOLUTIONS or a custom resolution. + */ + +/** @type {Field} Type field */ +export const VIDEO_TYPE = { + name: 'VIDEO.TYPE', + type: INPUT_TYPES.SELECT, + label: T.VideoType, + tooltip: T.VideoTypeConcept, + onlyOnHypervisors: [kvm, dummy], + values: arrayToOptions(Object.values(VIDEO_TYPES), { addEmpty: false }), + validation: string() + .trim() + .default(() => VIDEO_TYPES.auto) + .afterSubmit((value, { context }) => { + // A valid hypervisor will be if the hypervisor it's kvm (when creating or updating templates) or when it's undefined (when updating the config of a vm) + const validHypervisor = + context?.general?.HYPERVISOR === kvm || + context?.general?.HYPERVISOR === dummy || + !context?.general?.HYPERVISOR + + // Not send to the request the value if it's not a valid hypervisor + return validHypervisor && value !== VIDEO_TYPES.auto ? value : undefined + }), + grid: { sm: 3, md: 3 }, +} + +/** @type {Field} IOMMU field */ +export const IOMMU = { + name: 'VIDEO.IOMMU', + label: T.IOMMU, + tooltip: T.IOMMUConcept, + onlyOnHypervisors: [kvm, dummy], + type: INPUT_TYPES.SWITCH, + dependOf: VIDEO_TYPE.name, + htmlType: (type) => + (!type || type !== VIDEO_TYPES.virtio) && INPUT_TYPES.HIDDEN, + validation: boolean() + .yesOrNo(false) + .afterSubmit((value, { context }) => { + // A valid hypervisor will be if the hypervisor it's kvm (when creating or updating templates) or when it's undefined (when updating the config of a vm) + const validHypervisor = + context?.general?.HYPERVISOR === kvm || + context?.general?.HYPERVISOR === dummy || + !context?.general?.HYPERVISOR + + // Get video type from the context on extra step (templates) or no step (vm) + const videoType = context?.extra?.VIDEO?.TYPE || context?.VIDEO?.TYPE + + // Not send to the request the value if it's not a valid hypervisor + return validHypervisor && videoType === VIDEO_TYPES.virtio + ? value + ? 'YES' + : 'NO' + : undefined + }), + grid: { sm: 12, md: 12 }, +} + +/** @type {Field} ATS field */ +export const ATS = { + name: 'VIDEO.ATS', + label: T.ATS, + tooltip: T.ATSConcept, + onlyOnHypervisors: [kvm, dummy], + type: INPUT_TYPES.SWITCH, + dependOf: VIDEO_TYPE.name, + htmlType: (type) => + (!type || type !== VIDEO_TYPES.virtio) && INPUT_TYPES.HIDDEN, + validation: boolean() + .yesOrNo(false) + .afterSubmit((value, { context }) => { + // A valid hypervisor will be if the hypervisor it's kvm (when creating or updating templates) or when it's undefined (when updating the config of a vm) + const validHypervisor = + context?.general?.HYPERVISOR === kvm || + context?.general?.HYPERVISOR === dummy || + !context?.general?.HYPERVISOR + + // Get video type from the context on extra step (templates) or no step (vm) + const videoType = context?.extra?.VIDEO?.TYPE || context?.VIDEO?.TYPE + + // Not send to the request the value if it's not a valid hypervisor + return validHypervisor && videoType === VIDEO_TYPES.virtio + ? value + ? 'YES' + : 'NO' + : undefined + }), + grid: { sm: 12, md: 12 }, +} + +/** @type {Field} VRAM field */ +export const VRAM = { + name: 'VIDEO.VRAM', + label: T.VRAM, + tooltip: T.VRAMConcept, + onlyOnHypervisors: [kvm, dummy], + type: INPUT_TYPES.TEXT, + dependOf: VIDEO_TYPE.name, + htmlType: (type) => + (!type || type === VIDEO_TYPES.auto || type === VIDEO_TYPES.none) && + INPUT_TYPES.HIDDEN, + validation: number() + .min(1024) + .afterSubmit((value, { context }) => { + // A valid hypervisor will be if the hypervisor it's kvm (when creating or updating templates) or when it's undefined (when updating the config of a vm) + const validHypervisor = + context?.general?.HYPERVISOR === kvm || + context?.general?.HYPERVISOR === dummy || + !context?.general?.HYPERVISOR + + // Get video type from the context on extra step (templates) or no step (vm) + const videoType = context?.extra?.VIDEO?.TYPE || context?.VIDEO?.TYPE + + // Not send to the request the value if it's not a valid hypervisor + return validHypervisor && + videoType !== VIDEO_TYPES.auto && + videoType !== VIDEO_TYPES.none + ? value + : undefined + }), + grid: { sm: 3, md: 3 }, +} + +/** @type {Field} Resolution field */ +export const RESOLUTION = { + name: 'VIDEO.RESOLUTION', + type: INPUT_TYPES.SELECT, + label: T.Resolution, + tooltip: T.ResolutionConcept, + onlyOnHypervisors: [kvm, dummy], + dependOf: VIDEO_TYPE.name, + htmlType: (type) => + (!type || + type === VIDEO_TYPES.auto || + type === VIDEO_TYPES.none || + type === VIDEO_TYPES.cirrus) && + INPUT_TYPES.HIDDEN, + values: arrayToOptions(Object.values(COMMON_RESOLUTIONS), { addEmpty: true }), + validation: string() + .trim() + .afterSubmit((value, { context }) => { + // Video type could be on extra (when creating or updating templates) or VIDEO (when updating the config of a vm) attributes + const videoType = context?.extra?.VIDEO?.TYPE || context?.VIDEO?.TYPE + + // A valid hypervisor will be if the hypervisor it's kvm (when creating or updating templates) or when it's undefined (when updating the config of a vm) + const validHypervisor = + context?.general?.HYPERVISOR === kvm || + context?.general?.HYPERVISOR === dummy || + !context?.general?.HYPERVISOR + + // Resolution (width and height, that only is set by the user when resolution it's custom) could be on extra (when creating or updating templates) or VIDEO (when updating the config of a vm) attributes + const resolutionWidth = + context?.extra?.VIDEO?.RESOLUTION_WIDTH || + context?.VIDEO?.RESOLUTION_WIDTH + const resolutionHeight = + context?.extra?.VIDEO?.RESOLUTION_HEIGHT || + context?.VIDEO?.RESOLUTION_HEIGHT + + // Return resolution only if the video type is not auto/none/cirrus and hypervisor is kvm (templates) or undefined (vms) + return !validHypervisor || + videoType === VIDEO_TYPES.auto || + videoType === VIDEO_TYPES.none || + videoType === VIDEO_TYPES.cirrus + ? undefined + : value === 'custom' + ? resolutionWidth + 'x' + resolutionHeight + : value + }), + grid: { sm: 6, md: 6 }, +} + +/** @type {Field} RESOLUTION_WIDTH field */ +export const RESOLUTION_WIDTH = { + name: 'VIDEO.RESOLUTION_WIDTH', + label: T.ResolutionWidth, + tooltip: T.ResolutionWidthConcept, + onlyOnHypervisors: [kvm, dummy], + type: INPUT_TYPES.TEXT, + dependOf: [RESOLUTION.name, VIDEO_TYPE.name], + htmlType: (type) => + (!type || + type[0] !== COMMON_RESOLUTIONS.custom || + type[1] === VIDEO_TYPES.auto || + type[1] === VIDEO_TYPES.none || + type[1] === VIDEO_TYPES.cirrus) && + INPUT_TYPES.HIDDEN, + validation: number() + .positive() + .afterSubmit(() => undefined), + grid: { sm: 3, md: 3 }, +} + +/** @type {Field} RESOLUTION_HEIGHT field */ +export const RESOLUTION_HEIGHT = { + name: 'VIDEO.RESOLUTION_HEIGHT', + label: T.ResolutionHeight, + tooltip: T.ResolutionHeightConcept, + onlyOnHypervisors: [kvm, dummy], + type: INPUT_TYPES.TEXT, + dependOf: [RESOLUTION.name, VIDEO_TYPE.name], + htmlType: (type) => + (!type || + type[0] !== COMMON_RESOLUTIONS.custom || + type[1] === VIDEO_TYPES.auto || + type[1] === VIDEO_TYPES.none || + type[1] === VIDEO_TYPES.cirrus) && + INPUT_TYPES.HIDDEN, + validation: number() + .positive() + .afterSubmit(() => undefined), + grid: { sm: 3, md: 3 }, +} + +/** + * @param {string} [hypervisor] - VM hypervisor + * @param {object} oneConfig - Config of oned.conf + * @param {boolean} adminGroup - User is admin or not + * @returns {Field[]} List of Video fields + */ +export const VIDEO_FIELDS = (hypervisor, oneConfig, adminGroup) => + disableFields( + filterFieldsByHypervisor( + [ + VIDEO_TYPE, + VRAM, + RESOLUTION, + RESOLUTION_WIDTH, + RESOLUTION_HEIGHT, + IOMMU, + ATS, + ], + hypervisor + ), + 'VIDEO', + oneConfig, + adminGroup + ) + +/** @type {ObjectSchema} Video schema */ +export const VIDEO_SCHEMA = (hypervisor) => + getObjectSchemaFromFields(VIDEO_FIELDS(hypervisor)) diff --git a/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/videoSection.js b/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/videoSection.js new file mode 100644 index 0000000000..cf7a6652ba --- /dev/null +++ b/src/fireedge/src/client/components/Forms/VmTemplate/CreateForm/Steps/ExtraConfiguration/inputOutput/videoSection.js @@ -0,0 +1,79 @@ +/* ------------------------------------------------------------------------- * + * Copyright 2002-2023, OpenNebula Project, OpenNebula Systems * + * * + * Licensed under the Apache License, Version 2.0 (the "License"); you may * + * not use this file except in compliance with the License. You may obtain * + * a copy of the License at * + * * + * http://www.apache.org/licenses/LICENSE-2.0 * + * * + * Unless required by applicable law or agreed to in writing, software * + * distributed under the License is distributed on an "AS IS" BASIS, * + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * + * See the License for the specific language governing permissions and * + * limitations under the License. * + * ------------------------------------------------------------------------- */ +import { useMemo } from 'react' +import PropTypes from 'prop-types' +import { VIDEO_FIELDS } from './schema' +import { T, COMMON_RESOLUTIONS } from 'client/constants' +import { FormWithSchema } from 'client/components/Forms' +import { useFormContext } from 'react-hook-form' +export const SECTION_ID = 'VIDEO' + +/** + * Video section to set values about VIDEO attribute. + * + * @param {object} param - Properties to use in the form + * @param {string} param.stepId - Name of the step + * @param {string} param.hypervisor - Type of hypervisor + * @param {object} param.oneConfig - ONE config + * @param {boolean} param.adminGroup - If the user is admin + * @returns {object} - The component rendered + */ +const VideoSection = ({ stepId, hypervisor, oneConfig, adminGroup }) => { + // Check resolution value to get if it's a custom value or one of the common resolutions + const { getValues, setValue } = useFormContext() + + // If the context has extra field, we are in template create/update that use extra.VIDEO fiel. If not, we are on update config on a vm, that uses VIDEO field + const videoField = getValues('extra') ? 'extra.VIDEO' : 'VIDEO' + + const resolution = getValues(`${videoField}.RESOLUTION`) + const commonsResolutions = Object.values(COMMON_RESOLUTIONS) + + // If resolution it's a custom value, set custom as resolution and set widht and height resolution + if (resolution && !commonsResolutions.includes(resolution)) { + setValue(`${videoField}.RESOLUTION`, 'custom') + const resolutionValues = resolution.split('x') + setValue(`${videoField}.RESOLUTION_WIDTH`, resolutionValues[0]) + setValue(`${videoField}.RESOLUTION_HEIGHT`, resolutionValues[1]) + } + + // Get video fields + const fields = useMemo( + () => VIDEO_FIELDS(hypervisor, oneConfig, adminGroup), + [hypervisor] + ) + + // Generate a form from the schema + return ( + + ) +} + +VideoSection.propTypes = { + hypervisor: PropTypes.string, + stepId: PropTypes.string, + oneConfig: PropTypes.object, + adminGroup: PropTypes.bool, +} + +VideoSection.displayName = 'VideoSection' + +export default VideoSection diff --git a/src/fireedge/src/client/constants/translates.js b/src/fireedge/src/client/constants/translates.js index d122d15016..df5282b84e 100644 --- a/src/fireedge/src/client/constants/translates.js +++ b/src/fireedge/src/client/constants/translates.js @@ -1005,6 +1005,26 @@ module.exports = { Device: 'Device', Vendor: 'Vendor', Class: 'Class', + Video: 'Video', + VideoType: 'Video device type', + VideoTypeConcept: + 'Select the driver for the virtual video device, auto will be generated by OpenNebula.', + IOMMU: 'Enable IOMMU', + IOMMUConcept: + 'Enable the use of emulated Input/Output Memory Management Unit by the device', + ATS: 'Enable ATS', + ATSConcept: 'Enable Address Translation Service support', + VRAM: 'VRAM (KB)', + VRAMConcept: + 'Define the amount of VRAM to assign to the video device in KB. Must be greater than or equal to 1024', + Resolution: 'Resolution', + ResolutionConcept: 'Set the default resolution for the video device', + ResolutionWidth: 'Resolution width', + ResolutionWidthConcept: + 'Set the default width resolution for the video device', + ResolutionHeight: 'Resolution height', + ResolutionHeightConcept: + 'Set the default heigth resolution for the video device', /* VM Template schema - Input/Output - graphics */ Graphics: 'Graphics', ListenOnIp: 'Listen on IP', diff --git a/src/fireedge/src/client/constants/vmTemplate.js b/src/fireedge/src/client/constants/vmTemplate.js index e7675a5006..09df38a0ae 100644 --- a/src/fireedge/src/client/constants/vmTemplate.js +++ b/src/fireedge/src/client/constants/vmTemplate.js @@ -81,6 +81,21 @@ export const SD_DISK_BUSES = ['scsi', 'sata'] export const DEVICE_TYPES = { mouse: 'mouse', tablet: 'tablet' } export const DEVICE_BUS_TYPES = { usb: 'usb', ps2: 'ps2' } +export const VIDEO_TYPES = { + auto: 'auto', + none: 'none', + vga: 'vga', + cirrus: 'cirrus', + virtio: 'virtio', +} +export const COMMON_RESOLUTIONS = { + '1920x1080': '1920x1080', + '1366x768': '1366x768', + '1536x864': '1536x864', + '1440x900': '1440x900', + '1280x720': '1280x720', + custom: 'custom', +} export const FIRMWARE_TYPES = ['BIOS'] diff --git a/src/sunstone/public/app/tabs/templates-tab/form-panels/create/wizard-tabs/io.js b/src/sunstone/public/app/tabs/templates-tab/form-panels/create/wizard-tabs/io.js index 2221e46f9a..18afe680b5 100644 --- a/src/sunstone/public/app/tabs/templates-tab/form-panels/create/wizard-tabs/io.js +++ b/src/sunstone/public/app/tabs/templates-tab/form-panels/create/wizard-tabs/io.js @@ -74,6 +74,7 @@ define(function(require) { function _setup(context) { $("input[name='graphics_type']", context).change(function() { + if ($(this).attr("value") !== '') { if($('input[wizard_field="LISTEN"]', context).val() == ""){ $('input[wizard_field="LISTEN"]', context).val("0.0.0.0"); @@ -125,11 +126,96 @@ define(function(require) { }); CreateUtils.setupPCIRows($(".pci_devices", context)); + + // Hide video section if the hypervisor it's kvm (Done here because there is a bug if we used the css classes) + $("input[name='hypervisor']").change(function() { + if (this.value === "kvm") { + $(".video").show() + } + else { + $(".video").hide() + } + }) + + // Event on change video type attribute + $("input[name='video_type']", context).change(function() { + + /** + * - Only if the hypervisor it's kvm. + * - If the user select VIDEO_TYPE auto, the rest of the fields will be hidden and the request has not any VIDEO section. + * - If the user select VIDEO_TYPE none, the rest of the fields will be hidden and the request has VIDEO section but only with type=none. + * - If the user select VIDEO_TYPE cirrus, ATS, IOMMU and resolution will be hidden. + * - If the user select VIDEO_TYPE vga, ATS and IOMMU will be hidden. + * - If the user select VIDEO_TYPE virtio, all attributes will be show to the user. + * - Resolution will be one of the values of the COMMON_RESOLUTIONS or a custom resolution. + */ + if ($(this).attr("value") === 'auto' || $(this).attr("value") === 'none') { + + $('.video-settings', context).css('display', 'none'); + } + else if ($(this).attr("value") === 'cirrus') { + + $('.video-settings', context).css('display', ''); + $('.video-settings-iommu', context).css('display', 'none'); + $('.video-settings-iommu-label', context).css('display', 'none'); + $('.video-settings-ats', context).css('display', 'none'); + $('.video-settings-ats-label', context).css('display', 'none'); + $('.video-settings-vram', context).css('display', ''); + $('.video-settings-resolution', context).css('display', 'none'); + $('.video-settings-resolution-label', context).css('display', 'none'); + $('.video-settings-resolution-width', context).css('display', 'none'); + $('.video-settings-resolution-height', context).css('display', 'none'); + } + else if ($(this).attr("value") === 'vga') { + + $('.video-settings', context).css('display', ''); + $('.video-settings-iommu', context).css('display', 'none'); + $('.video-settings-iommu-label', context).css('display', 'none'); + $('.video-settings-ats', context).css('display', 'none'); + $('.video-settings-ats-label', context).css('display', 'none'); + $('.video-settings-vram', context).css('display', ''); + $('.video-settings-resolution', context).css('display', ''); + $('.video-settings-resolution-label', context).css('display', ''); + $('.video-settings-resolution-width', context).css('display', ''); + $('.video-settings-resolution-height', context).css('display', ''); + $("select[name='resolution']", context).change() + } + else if ($(this).attr("value") === 'virtio') { + + $('.video-settings', context).css('display', ''); + $('.video-settings-iommu', context).css('display', ''); + $('.video-settings-iommu-label', context).css('display', ''); + $('.video-settings-ats', context).css('display', ''); + $('.video-settings-ats-label', context).css('display', ''); + $('.video-settings-vram', context).css('display', ''); + $('.video-settings-resolution', context).css('display', ''); + $('.video-settings-resolution-label', context).css('display', ''); + $('.video-settings-resolution-width', context).css('display', ''); + $('.video-settings-resolution-height', context).css('display', ''); + $("select[name='resolution']", context).change() + } + }); + + // Manage custom resolution + $("select[name='resolution']", context).change(function() { + + if ($(this).val() === "custom") { + + $('.video-settings-resolution-width', context).css('display', ''); + $('.video-settings-resolution-height', context).css('display', ''); + } + else { + $('.video-settings-resolution-width', context).css('display', 'none'); + $('.video-settings-resolution-height', context).css('display', 'none'); + } + + }) + } function _retrieve(context) { var templateJSON = {}; - var graphicsJSON = WizardFields.retrieve(context); + var graphicsJSON = WizardFields.retrieve(context.find("div.graphics")); if (!$.isEmptyObject(graphicsJSON) && $(".RANDOM_PASSWD:checked", context).length > 0) { graphicsJSON["RANDOM_PASSWD"] = "YES"; @@ -161,6 +247,43 @@ define(function(require) { } }); + // Add video section to the request + // - If video type is auto, don't send any video section + // - If video type is none, send only video type + // - Checkbox attributes will be YES/NO + var videoJSON = WizardFields.retrieve(context.find("div.video")); + + videoJSON.TYPE = videoJSON.VIDEO_TYPE + delete videoJSON.VIDEO_TYPE + + if (videoJSON.TYPE === "auto") { + videoJSON = {} + } + + if (videoJSON.TYPE === "none") { + videoJSON = { + "TYPE": "none" + } + } + + if (!$.isEmptyObject(videoJSON) && $(".video-settings-iommu:checked", context).length > 0) { + videoJSON["IOMMU"] = "YES"; + } + + if (!$.isEmptyObject(videoJSON) && $(".video-settings-ats:checked", context).length > 0) { + videoJSON["ATS"] = "YES"; + } + + if (!$.isEmptyObject(videoJSON) && videoJSON.RESOLUTION == "custom") { + videoJSON.RESOLUTION = videoJSON.RESOLUTION_WIDTH + "x" + videoJSON.RESOLUTION_HEIGHT + } + + delete videoJSON.RESOLUTION_WIDTH + delete videoJSON.RESOLUTION_HEIGHT + + + if (!$.isEmptyObject(videoJSON)) { templateJSON['VIDEO'] = videoJSON; }; + return templateJSON; } @@ -236,5 +359,49 @@ define(function(require) { }); delete templateJSON.PCI; + + // Video section + var videoJSON = templateJSON['VIDEO']; + + if (videoJSON) { + + var type = videoJSON.TYPE; + if (type) { + $(".video input[wizard_field='VIDEO_TYPE'][value='" + type + "']", context).click(); + } else { + $(".video input[wizard_field='VIDEO_TYPE'][value='']", context).click(); + } + + videoJSON.VIDEO_TYPE = type + delete videoJSON.TYPE + + if (videoJSON["IOMMU"] == "YES") { + $(".video-settings-iommu", context).attr("checked", "checked"); + delete videoJSON["IOMMU"] + } + + if (videoJSON["ATS"] == "YES") { + $(".video-settings-ats", context).attr("checked", "checked"); + delete videoJSON["ATS"] + } + + // Manage resolution if it's a custom value + let resolutions = $("select[name='resolution'] option").map(function() { return this.value; }).get(); + let resolution = videoJSON.RESOLUTION + if (resolutions && resolution && (resolution!== "") && !resolutions.includes(resolution)) { + let storeResolutions = resolution.split("x") + videoJSON.RESOLUTION_WIDTH = storeResolutions[0] + videoJSON.RESOLUTION_HEIGHT = storeResolutions[1] + videoJSON.RESOLUTION = "custom" + } + + WizardFields.fill(context, videoJSON); + + delete templateJSON['VIDEO'] + + } else { + $(".video input[wizard_field='TYPE'][value='']", context).click(); + } + } }); diff --git a/src/sunstone/public/app/tabs/templates-tab/form-panels/create/wizard-tabs/io/html.hbs b/src/sunstone/public/app/tabs/templates-tab/form-panels/create/wizard-tabs/io/html.hbs index ea0992746b..a356116415 100644 --- a/src/sunstone/public/app/tabs/templates-tab/form-panels/create/wizard-tabs/io/html.hbs +++ b/src/sunstone/public/app/tabs/templates-tab/form-panels/create/wizard-tabs/io/html.hbs @@ -160,4 +160,75 @@ +
+
+ + {{tr "Video"}} + {{{tip (tr "Select the driver for the virtual video device, auto will be generated by OpenNebula.")}}} + +
+
+ + + + + +
+
+ +
+