From c2c63c6382174ff82f7f5b204d67002cb6796f38 Mon Sep 17 00:00:00 2001 From: Sharon Gratch Date: Mon, 6 Jan 2025 17:38:08 +0200 Subject: [PATCH] Migrate deprecated Wizard component to PF5 and fix Wizard's focus issue References: https://issues.redhat.com/browse/MTV-1051 https://issues.redhat.com/browse/MTV-1863 1. Migrate the deprecated Wizard component to PF5 - see https://v5-archive.patternfly.org/components/wizard/react-deprecated 2. When moving between steps in plan wizard, set the initial focus to the top of the step's page. Signed-off-by: Sharon Gratch --- .../views/create/PlanCreatePage.style.css | 10 +- .../Plans/views/create/PlanCreatePage.tsx | 125 ++++++++++-------- 2 files changed, 80 insertions(+), 55 deletions(-) diff --git a/packages/forklift-console-plugin/src/modules/Plans/views/create/PlanCreatePage.style.css b/packages/forklift-console-plugin/src/modules/Plans/views/create/PlanCreatePage.style.css index f5134fcd5..d1cab2104 100644 --- a/packages/forklift-console-plugin/src/modules/Plans/views/create/PlanCreatePage.style.css +++ b/packages/forklift-console-plugin/src/modules/Plans/views/create/PlanCreatePage.style.css @@ -80,6 +80,12 @@ cursor: pointer; } -.forklift--create-plan--wizard-appearance-order { - z-index: 1; +.forklift--create-plan--wizard-content { + flex-flow: initial; +} + +.forklift--create-plan--wizard-container { + height: 100%; + display: flex; + flex-direction: column; } \ No newline at end of file diff --git a/packages/forklift-console-plugin/src/modules/Plans/views/create/PlanCreatePage.tsx b/packages/forklift-console-plugin/src/modules/Plans/views/create/PlanCreatePage.tsx index dc0dbe9f6..b848459b6 100644 --- a/packages/forklift-console-plugin/src/modules/Plans/views/create/PlanCreatePage.tsx +++ b/packages/forklift-console-plugin/src/modules/Plans/views/create/PlanCreatePage.tsx @@ -1,15 +1,19 @@ import React, { useReducer } from 'react'; -import { useHistory } from 'react-router'; +import { getResourceUrl } from 'src/modules/Providers/utils/helpers'; import { useCreateVmMigrationData } from 'src/modules/Providers/views/migrate'; import ProvidersCreateVmMigrationPage from 'src/modules/Providers/views/migrate/ProvidersCreateVmMigrationPage'; import { startCreate } from 'src/modules/Providers/views/migrate/reducer/actions'; import { useFetchEffects } from 'src/modules/Providers/views/migrate/useFetchEffects'; import { useSaveEffect } from 'src/modules/Providers/views/migrate/useSaveEffect'; -import { ProviderModelGroupVersionKind, V1beta1Provider } from '@kubev2v/types'; +import { + PlanModelRef, + ProviderModelGroupVersionKind, + ProviderModelRef, + V1beta1Provider, +} from '@kubev2v/types'; import { useActiveNamespace, useK8sWatchResource } from '@openshift-console/dynamic-plugin-sdk'; -import { PageSection, Title } from '@patternfly/react-core'; -import { Wizard } from '@patternfly/react-core/deprecated'; +import { PageSection, Title, Wizard, WizardStep } from '@patternfly/react-core'; import { findProviderByID } from './components'; import { planCreatePageInitialState, planCreatePageReducer } from './states'; @@ -20,8 +24,8 @@ import './PlanCreatePage.style.css'; export const PlanCreatePage: React.FC<{ namespace: string }> = ({ namespace }) => { // Get optional initial state context const { data } = useCreateVmMigrationData(); - const history = useHistory(); - const startAtStep = data?.provider !== undefined ? 2 : 1; + const createPlanFromPlansList = data?.provider !== undefined ? false : true; + const startAtStep = createPlanFromPlansList ? 1 : 2; const [activeNamespace, setActiveNamespace] = useActiveNamespace(); const defaultNamespace = process?.env?.DEFAULT_NAMESPACE || 'default'; const projectName = @@ -29,6 +33,18 @@ export const PlanCreatePage: React.FC<{ namespace: string }> = ({ namespace }) = (activeNamespace === '#ALL_NS#' ? 'openshift-mtv' : activeNamespace) || defaultNamespace; + const plansListURL = getResourceUrl({ + reference: PlanModelRef, + namespace: namespace, + namespaced: namespace !== undefined, + }); + + const providerURL = getResourceUrl({ + reference: ProviderModelRef, + name: data?.provider?.metadata?.name, + namespace: data?.provider?.metadata?.namespace, + }); + // Init Select source provider form state const [filterState, filterDispatch] = useReducer(planCreatePageReducer, { ...planCreatePageInitialState, @@ -59,65 +75,68 @@ export const PlanCreatePage: React.FC<{ namespace: string }> = ({ namespace }) = }); useSaveEffect(state, dispatch); - const steps = [ - { - id: 'step-1', - name: 'Select source provider', - component: ( - - ), - enableNext: filterState?.selectedVMs?.length > 0, - }, - { - id: 'step-2', - name: 'Create migration plan', - component: ( - - ), - enableNext: - !emptyContext && - !( - !!state?.flow?.apiError || - Object.values(state?.validation || []).some((validation) => validation === 'error') || - state?.validation?.planName === 'default' - ), - canJumpTo: filterState?.selectedVMs?.length > 0, - nextButtonText: 'Create migration plan', - }, - ]; + const anyValidationError = Object.values(state?.validation || []).some( + (validation) => validation === 'error', + ); + + const planNameValidationDefault = state?.validation?.planName === 'default'; const title = 'Plans wizard'; + return ( <> {'Create migration plan'} - - + + window.location.replace(createPlanFromPlansList ? plansListURL : `${providerURL}/vms`) + } onSave={() => { setActiveNamespace(state.underConstruction.projectName); dispatch(startCreate()); }} - onClose={() => history.goBack()} - startAtStep={startAtStep} - /> + > + + + + + + + );