From ebebe6a02122f8603195f26732c196c219c014ec Mon Sep 17 00:00:00 2001 From: Shivam Gupta Date: Wed, 17 Jul 2024 20:03:30 +0530 Subject: [PATCH] feat(form): Introduce a 'User Modified' from section --- .../Form/dataFormat/DataFormatEditor.test.tsx | 7 +- .../Form/dataFormat/DataFormatEditor.tsx | 6 +- .../loadBalancer/LoadBalancerEditor.test.tsx | 7 +- .../Form/loadBalancer/LoadBalancerEditor.tsx | 6 +- .../StepExpressionEditor.test.tsx | 3 +- .../stepExpression/StepExpressionEditor.tsx | 6 +- .../Visualization/Canvas/CanvasForm.test.tsx | 352 +--------- .../Visualization/Canvas/CanvasForm.tsx | 69 +- ...tsx => CanvasFormTabs.exhaustive.test.tsx} | 2 +- .../Visualization/Canvas/CanvasFormTabs.scss | 3 + .../Canvas/CanvasFormTabs.test.tsx | 646 ++++++++++++++++++ .../Visualization/Canvas/CanvasFormTabs.tsx | 128 ++++ .../__snapshots__/CanvasForm.test.tsx.snap | 38 ++ ...get-user-updated-properties-schema.test.ts | 171 +++++ .../get-user-updated-properties-schema.ts | 41 ++ packages/ui/src/utils/index.ts | 1 + 16 files changed, 1057 insertions(+), 429 deletions(-) rename packages/ui/src/components/Visualization/Canvas/{CanvasForm.exhaustive.test.tsx => CanvasFormTabs.exhaustive.test.tsx} (99%) create mode 100644 packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.scss create mode 100644 packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.test.tsx create mode 100644 packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.tsx create mode 100644 packages/ui/src/utils/get-user-updated-properties-schema.test.ts create mode 100644 packages/ui/src/utils/get-user-updated-properties-schema.ts diff --git a/packages/ui/src/components/Form/dataFormat/DataFormatEditor.test.tsx b/packages/ui/src/components/Form/dataFormat/DataFormatEditor.test.tsx index 92382ade0..e626f49a1 100644 --- a/packages/ui/src/components/Form/dataFormat/DataFormatEditor.test.tsx +++ b/packages/ui/src/components/Form/dataFormat/DataFormatEditor.test.tsx @@ -8,6 +8,7 @@ import { getFirstCatalogMap } from '../../../stubs/test-load-catalog'; import { MetadataEditor } from '../../MetadataEditor'; import { CanvasNode } from '../../Visualization/Canvas/canvas.models'; import { DataFormatEditor } from './DataFormatEditor'; +import { FormTabsModes } from '../../Visualization/Canvas/CanvasFormTabs'; describe('DataFormatEditor', () => { let mockNode: CanvasNode; @@ -46,7 +47,7 @@ describe('DataFormatEditor', () => { }); it('should render', async () => { - render(); + render(); const buttons = screen.getAllByRole('button', { name: 'Menu toggle' }); await act(async () => { fireEvent.click(buttons[0]); @@ -58,7 +59,7 @@ describe('DataFormatEditor', () => { }); it('should filter candidates with a text input', async () => { - render(); + render(); const buttons = screen.getAllByRole('button', { name: 'Menu toggle' }); await act(async () => { fireEvent.click(buttons[0]); @@ -74,7 +75,7 @@ describe('DataFormatEditor', () => { }); it('should clear filter and close the dropdown with close button', async () => { - render(); + render(); const buttons = screen.getAllByRole('button', { name: 'Menu toggle' }); await act(async () => { fireEvent.click(buttons[0]); diff --git a/packages/ui/src/components/Form/dataFormat/DataFormatEditor.tsx b/packages/ui/src/components/Form/dataFormat/DataFormatEditor.tsx index afa26a3d1..6526fb4b8 100644 --- a/packages/ui/src/components/Form/dataFormat/DataFormatEditor.tsx +++ b/packages/ui/src/components/Form/dataFormat/DataFormatEditor.tsx @@ -12,10 +12,12 @@ import { CanvasNode } from '../../Visualization/Canvas/canvas.models'; import './DataFormatEditor.scss'; import { DataFormatService } from './dataformat.service'; import { TypeaheadEditor } from '../customField/TypeaheadEditor'; -import { getSerializedModel } from '../../../utils'; +import { getSerializedModel, isDefined } from '../../../utils'; +import { FormTabsModes } from '../../Visualization/Canvas/CanvasFormTabs'; interface DataFormatEditorProps { selectedNode: CanvasNode; + formMode: FormTabsModes; } export const DataFormatEditor: FunctionComponent = (props) => { @@ -78,6 +80,8 @@ export const DataFormatEditor: FunctionComponent = (props [entitiesContext, dataFormatCatalogMap, props.selectedNode.data?.vizNode], ); + if (props.formMode === FormTabsModes.USER_MODIFIED && !isDefined(selectedDataFormatOption)) return null; + return (
diff --git a/packages/ui/src/components/Form/loadBalancer/LoadBalancerEditor.test.tsx b/packages/ui/src/components/Form/loadBalancer/LoadBalancerEditor.test.tsx index 179d8a4d6..34d9426c4 100644 --- a/packages/ui/src/components/Form/loadBalancer/LoadBalancerEditor.test.tsx +++ b/packages/ui/src/components/Form/loadBalancer/LoadBalancerEditor.test.tsx @@ -8,6 +8,7 @@ import { getFirstCatalogMap } from '../../../stubs/test-load-catalog'; import { MetadataEditor } from '../../MetadataEditor'; import { CanvasNode } from '../../Visualization/Canvas/canvas.models'; import { LoadBalancerEditor } from './LoadBalancerEditor'; +import { FormTabsModes } from '../../Visualization/Canvas/CanvasFormTabs'; describe('LoadBalancerEditor', () => { let mockNode: CanvasNode; @@ -47,7 +48,7 @@ describe('LoadBalancerEditor', () => { }); it('should render', async () => { - render(); + render(); const buttons = screen.getAllByRole('button', { name: 'Menu toggle' }); await act(async () => { fireEvent.click(buttons[0]); @@ -61,7 +62,7 @@ describe('LoadBalancerEditor', () => { }); it('should filter candidates with a text input', async () => { - render(); + render(); const buttons = screen.getAllByRole('button', { name: 'Menu toggle' }); await act(async () => { fireEvent.click(buttons[0]); @@ -77,7 +78,7 @@ describe('LoadBalancerEditor', () => { }); it('should clear filter and close the dropdown with close button', async () => { - render(); + render(); const buttons = screen.getAllByRole('button', { name: 'Menu toggle' }); await act(async () => { fireEvent.click(buttons[0]); diff --git a/packages/ui/src/components/Form/loadBalancer/LoadBalancerEditor.tsx b/packages/ui/src/components/Form/loadBalancer/LoadBalancerEditor.tsx index 4ac586903..e6709e0ba 100644 --- a/packages/ui/src/components/Form/loadBalancer/LoadBalancerEditor.tsx +++ b/packages/ui/src/components/Form/loadBalancer/LoadBalancerEditor.tsx @@ -12,10 +12,12 @@ import { CanvasNode } from '../../Visualization/Canvas/canvas.models'; import { LoadBalancerService } from './loadbalancer.service'; import './LoadBalancerEditor.scss'; import { TypeaheadEditor } from '../customField/TypeaheadEditor'; -import { getSerializedModel } from '../../../utils'; +import { getSerializedModel, isDefined } from '../../../utils'; +import { FormTabsModes } from '../../Visualization/Canvas/CanvasFormTabs'; interface LoadBalancerEditorProps { selectedNode: CanvasNode; + formMode: FormTabsModes; } export const LoadBalancerEditor: FunctionComponent = (props) => { @@ -78,6 +80,8 @@ export const LoadBalancerEditor: FunctionComponent = (p [entitiesContext, loadBalancerCatalogMap, props.selectedNode.data?.vizNode], ); + if (props.formMode === FormTabsModes.USER_MODIFIED && !isDefined(selectedLoadBalancerOption)) return null; + return (
diff --git a/packages/ui/src/components/Form/stepExpression/StepExpressionEditor.test.tsx b/packages/ui/src/components/Form/stepExpression/StepExpressionEditor.test.tsx index dd83a15ae..cbcd3beea 100644 --- a/packages/ui/src/components/Form/stepExpression/StepExpressionEditor.test.tsx +++ b/packages/ui/src/components/Form/stepExpression/StepExpressionEditor.test.tsx @@ -9,6 +9,7 @@ import { MetadataEditor } from '../../MetadataEditor'; import { CanvasNode } from '../../Visualization/Canvas/canvas.models'; import { SchemaService } from '../schema.service'; import { StepExpressionEditor } from './StepExpressionEditor'; +import { FormTabsModes } from '../../Visualization/Canvas/CanvasFormTabs'; describe('StepExpressionEditor', () => { let mockNode: CanvasNode; @@ -46,7 +47,7 @@ describe('StepExpressionEditor', () => { }); it('should render', async () => { - render(); + render(); const launcherButton = screen.getAllByRole('button', { name: 'Configure Expression' }); await act(async () => { fireEvent.click(launcherButton[0]); diff --git a/packages/ui/src/components/Form/stepExpression/StepExpressionEditor.tsx b/packages/ui/src/components/Form/stepExpression/StepExpressionEditor.tsx index 0586933d4..dfe74bdfe 100644 --- a/packages/ui/src/components/Form/stepExpression/StepExpressionEditor.tsx +++ b/packages/ui/src/components/Form/stepExpression/StepExpressionEditor.tsx @@ -5,10 +5,12 @@ import { EntitiesContext } from '../../../providers'; import { CanvasNode } from '../../Visualization/Canvas/canvas.models'; import { ExpressionService } from '..//expression/expression.service'; import { ExpressionModalLauncher } from '../expression/ExpressionModalLauncher'; -import { getSerializedModel } from '../../../utils'; +import { getSerializedModel, isDefined } from '../../../utils'; +import { FormTabsModes } from '../../Visualization/Canvas/CanvasFormTabs'; interface StepExpressionEditorProps { selectedNode: CanvasNode; + formMode: FormTabsModes; } export const StepExpressionEditor: FunctionComponent = (props) => { @@ -65,6 +67,8 @@ export const StepExpressionEditor: FunctionComponent const title = props.selectedNode.label; const description = title ? `Configure expression for "${title}" parameter` : 'Configure expression'; + if (props.formMode === FormTabsModes.USER_MODIFIED && !isDefined(preparedLanguage)) return null; + return ( languageCatalogMap && (
diff --git a/packages/ui/src/components/Visualization/Canvas/CanvasForm.test.tsx b/packages/ui/src/components/Visualization/Canvas/CanvasForm.test.tsx index ea30cc14c..9c7bf7d28 100644 --- a/packages/ui/src/components/Visualization/Canvas/CanvasForm.test.tsx +++ b/packages/ui/src/components/Visualization/Canvas/CanvasForm.test.tsx @@ -1,5 +1,5 @@ import catalogLibrary from '@kaoto/camel-catalog/index.json'; -import { CatalogLibrary, RouteDefinition } from '@kaoto/camel-catalog/types'; +import { CatalogLibrary } from '@kaoto/camel-catalog/types'; import { act, fireEvent, render, screen } from '@testing-library/react'; import { CamelCatalogService, @@ -17,7 +17,6 @@ import { VisibleFlowsContext, VisibleFlowsProvider } from '../../../providers'; import { EntitiesContext, EntitiesProvider } from '../../../providers/entities.provider'; import { camelRouteJson, kameletJson } from '../../../stubs'; import { getFirstCatalogMap } from '../../../stubs/test-load-catalog'; -import { SchemaService } from '../../Form'; import { CanvasForm } from './CanvasForm'; import { CanvasNode } from './canvas.models'; import { CanvasService } from './canvas.service'; @@ -266,353 +265,4 @@ describe('CanvasForm', () => { expect(kameletVisualEntity.id).toEqual(newName); expect(dispatchSpy).toHaveBeenCalledWith({ type: 'renameFlow', flowId, newName }); }); - - describe('should persists changes from both expression editor and main form', () => { - beforeEach(() => { - jest.spyOn(console, 'error').mockImplementation(() => {}); - }); - - it('expression => main form', async () => { - const camelRoute = { - from: { - uri: 'timer:tutorial', - steps: [ - { - setHeader: { - name: 'foo', - }, - }, - ], - }, - } as RouteDefinition; - const entity = new CamelRouteVisualEntity(camelRoute); - const rootNode: IVisualizationNode = entity.toVizNode(); - const setHeaderNode = rootNode.getChildren()![0].getChildren()![0]; - const selectedNode = { - id: '1', - type: 'node', - data: { - vizNode: setHeaderNode, - }, - }; - - render( - - - - - , - ); - const launchExpression = screen.getByTestId('launch-expression-modal-btn'); - act(() => { - fireEvent.click(launchExpression); - }); - const button = screen - .getAllByTestId('typeahead-select-input') - .filter((input) => input.innerHTML.includes(SchemaService.DROPDOWN_PLACEHOLDER)); - act(() => { - fireEvent.click(button[0]); - }); - const simple = screen.getByTestId('expression-dropdownitem-simple'); - act(() => { - fireEvent.click(simple.getElementsByTagName('button')[0]); - }); - const expressionInput = screen - .getAllByRole('textbox') - .filter((textbox) => textbox.getAttribute('name') === 'expression'); - const applyBtn = screen.getAllByRole('button').filter((button) => button.textContent === 'Apply'); - act(() => { - fireEvent.input(expressionInput[0], { target: { value: '${header.foo}' } }); - fireEvent.click(applyBtn[0]); - }); - /* eslint-disable @typescript-eslint/no-explicit-any */ - expect((camelRoute.from.steps[0].setHeader!.expression as any).simple.expression).toEqual('${header.foo}'); - expect(camelRoute.from.steps[0].setHeader!.name).toEqual('foo'); - - const filtered = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Name'); - act(() => { - fireEvent.input(filtered[0], { target: { value: 'bar' } }); - }); - /* eslint-disable @typescript-eslint/no-explicit-any */ - expect((camelRoute.from.steps[0].setHeader!.expression as any).simple.expression).toEqual('${header.foo}'); - expect(camelRoute.from.steps[0].setHeader!.name).toEqual('bar'); - }); - - it('main form => expression', async () => { - const camelRoute = { - from: { - uri: 'timer:tutorial', - steps: [ - { - setHeader: { - name: 'foo', - }, - }, - ], - }, - } as RouteDefinition; - const entity = new CamelRouteVisualEntity(camelRoute); - const rootNode: IVisualizationNode = entity.toVizNode(); - const setHeaderNode = rootNode.getChildren()![0].getChildren()![0]; - const selectedNode = { - id: '1', - type: 'node', - data: { - vizNode: setHeaderNode, - }, - }; - - render( - - - - - , - ); - const filtered = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Name'); - act(() => { - fireEvent.input(filtered[0], { target: { value: 'bar' } }); - }); - expect(camelRoute.from.steps[0].setHeader!.expression).toBeUndefined(); - expect(camelRoute.from.steps[0].setHeader!.name).toEqual('bar'); - - const launchExpression = screen.getByTestId('launch-expression-modal-btn'); - act(() => { - fireEvent.click(launchExpression); - }); - const button = screen - .getAllByTestId('typeahead-select-input') - .filter((input) => input.innerHTML.includes(SchemaService.DROPDOWN_PLACEHOLDER)); - act(() => { - fireEvent.click(button[0]); - }); - const simple = screen.getByTestId('expression-dropdownitem-simple'); - act(() => { - fireEvent.click(simple.getElementsByTagName('button')[0]); - }); - const expressionInput = screen - .getAllByRole('textbox') - .filter((textbox) => textbox.getAttribute('name') === 'expression'); - const applyBtn = screen.getAllByRole('button').filter((button) => button.textContent === 'Apply'); - act(() => { - fireEvent.input(expressionInput[0], { target: { value: '${header.foo}' } }); - fireEvent.click(applyBtn[0]); - }); - /* eslint-disable @typescript-eslint/no-explicit-any */ - expect((camelRoute.from.steps[0].setHeader!.expression as any).simple.expression).toEqual('${header.foo}'); - expect(camelRoute.from.steps[0].setHeader!.name).toEqual('bar'); - }); - }); - - describe('should persists changes from both dataformat editor and main form', () => { - beforeEach(() => { - jest.spyOn(console, 'error').mockImplementation(() => {}); - }); - - it('dataformat => main form', async () => { - const camelRoute = { - from: { - uri: 'timer:tutorial', - steps: [ - { - marshal: { - id: 'ms', - }, - }, - ], - }, - } as RouteDefinition; - const entity = new CamelRouteVisualEntity(camelRoute); - const rootNode: IVisualizationNode = entity.toVizNode(); - const marshalNode = rootNode.getChildren()![0].getChildren()![0]; - const selectedNode = { - id: '1', - type: 'node', - data: { - vizNode: marshalNode, - }, - }; - - render( - - - - - , - ); - const button = screen.getAllByRole('button', { name: 'Menu toggle' }); - await act(async () => { - fireEvent.click(button[0]); - }); - const avro = screen.getByTestId('dataformat-dropdownitem-avro'); - await act(async () => { - fireEvent.click(avro.getElementsByTagName('button')[0]); - }); - expect(camelRoute.from.steps[0].marshal!.avro).toBeDefined(); - expect(camelRoute.from.steps[0].marshal!.id).toEqual('ms'); - - const idInput = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Id'); - await act(async () => { - fireEvent.input(idInput[1], { target: { value: 'modified' } }); - }); - expect(camelRoute.from.steps[0].marshal!.avro).toBeDefined(); - expect(camelRoute.from.steps[0].marshal!.id).toEqual('modified'); - }); - - it('main form => dataformat', async () => { - const camelRoute = { - from: { - uri: 'timer:tutorial', - steps: [ - { - marshal: { - id: 'ms', - }, - }, - ], - }, - } as RouteDefinition; - const entity = new CamelRouteVisualEntity(camelRoute); - const rootNode: IVisualizationNode = entity.toVizNode(); - const marshalNode = rootNode.getChildren()![0].getChildren()![0]; - const selectedNode = { - id: '1', - type: 'node', - data: { - vizNode: marshalNode, - }, - }; - - render( - - - - - , - ); - const idInput = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Id'); - await act(async () => { - fireEvent.input(idInput[0], { target: { value: 'modified' } }); - }); - expect(camelRoute.from.steps[0].marshal!.avro).toBeUndefined(); - expect(camelRoute.from.steps[0].marshal!.id).toEqual('modified'); - - const button = screen.getAllByRole('button', { name: 'Menu toggle' }); - await act(async () => { - fireEvent.click(button[0]); - }); - const avro = screen.getByTestId('dataformat-dropdownitem-avro'); - await act(async () => { - fireEvent.click(avro.getElementsByTagName('button')[0]); - }); - expect(camelRoute.from.steps[0].marshal!.avro).toBeDefined(); - expect(camelRoute.from.steps[0].marshal!.id).toEqual('modified'); - }); - }); - - describe('should persists changes from both loadbalancer editor and main form', () => { - beforeEach(() => { - jest.spyOn(console, 'error').mockImplementation(() => {}); - }); - - it('loadbalancer => main form', async () => { - const camelRoute = { - from: { - uri: 'timer:tutorial', - steps: [ - { - loadBalance: { - id: 'lb', - }, - }, - ], - }, - } as RouteDefinition; - const entity = new CamelRouteVisualEntity(camelRoute); - const rootNode: IVisualizationNode = entity.toVizNode(); - const loadBalanceNode = rootNode.getChildren()![0].getChildren()![0]; - const selectedNode = { - id: '1', - type: 'node', - data: { - vizNode: loadBalanceNode, - }, - }; - - render( - - - - - , - ); - const button = screen.getAllByRole('button', { name: 'Menu toggle' }); - await act(async () => { - fireEvent.click(button[0]); - }); - const avro = screen.getByTestId('loadbalancer-dropdownitem-weightedLoadBalancer'); - await act(async () => { - fireEvent.click(avro.getElementsByTagName('button')[0]); - }); - expect(camelRoute.from.steps[0].loadBalance!.weightedLoadBalancer).toBeDefined(); - expect(camelRoute.from.steps[0].loadBalance!.id).toEqual('lb'); - - const idInput = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Id'); - await act(async () => { - fireEvent.input(idInput[1], { target: { value: 'modified' } }); - }); - expect(camelRoute.from.steps[0].loadBalance!.weightedLoadBalancer).toBeDefined(); - expect(camelRoute.from.steps[0].loadBalance!.id).toEqual('modified'); - }); - - it('main form => loadbalancer', async () => { - const camelRoute = { - from: { - uri: 'timer:tutorial', - steps: [ - { - loadBalance: { - id: 'lb', - }, - }, - ], - }, - } as RouteDefinition; - const entity = new CamelRouteVisualEntity(camelRoute); - const rootNode: IVisualizationNode = entity.toVizNode(); - const loadBalanceNode = rootNode.getChildren()![0].getChildren()![0]; - const selectedNode = { - id: '1', - type: 'node', - data: { - vizNode: loadBalanceNode, - }, - }; - - render( - - - - - , - ); - const idInput = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Id'); - await act(async () => { - fireEvent.input(idInput[0], { target: { value: 'modified' } }); - }); - expect(camelRoute.from.steps[0].loadBalance!.weighted).toBeUndefined(); - expect(camelRoute.from.steps[0].loadBalance!.id).toEqual('modified'); - - const button = screen.getAllByRole('button', { name: 'Menu toggle' }); - await act(async () => { - fireEvent.click(button[0]); - }); - const avro = screen.getByTestId('loadbalancer-dropdownitem-weightedLoadBalancer'); - await act(async () => { - fireEvent.click(avro.getElementsByTagName('button')[0]); - }); - expect(camelRoute.from.steps[0].loadBalance!.weightedLoadBalancer).toBeDefined(); - expect(camelRoute.from.steps[0].loadBalance!.id).toEqual('modified'); - }); - }); }); diff --git a/packages/ui/src/components/Visualization/Canvas/CanvasForm.tsx b/packages/ui/src/components/Visualization/Canvas/CanvasForm.tsx index 64a17cf27..b674007ea 100644 --- a/packages/ui/src/components/Visualization/Canvas/CanvasForm.tsx +++ b/packages/ui/src/components/Visualization/Canvas/CanvasForm.tsx @@ -1,18 +1,11 @@ import { Card, CardBody, CardHeader, SearchInput } from '@patternfly/react-core'; import { FunctionComponent, useCallback, useContext, useEffect, useMemo, useRef } from 'react'; import { VisibleFlowsContext, FilteredFieldContext } from '../../../providers'; -import { EntitiesContext } from '../../../providers/entities.provider'; -import { SchemaBridgeProvider } from '../../../providers/schema-bridge.provider'; -import { isDefined, setValue } from '../../../utils'; import { ErrorBoundary } from '../../ErrorBoundary'; -import { CustomAutoForm, CustomAutoFormRef } from '../../Form/CustomAutoForm'; -import { DataFormatEditor } from '../../Form/dataFormat/DataFormatEditor'; -import { LoadBalancerEditor } from '../../Form/loadBalancer/LoadBalancerEditor'; -import { StepExpressionEditor } from '../../Form/stepExpression/StepExpressionEditor'; -import { UnknownNode } from '../Custom/UnknownNode'; import './CanvasForm.scss'; import { CanvasFormHeader } from './Form/CanvasFormHeader'; import { CanvasNode } from './canvas.models'; +import { CanvasFormTabs } from './CanvasFormTabs'; interface CanvasFormProps { selectedNode: CanvasNode; @@ -21,12 +14,8 @@ interface CanvasFormProps { export const CanvasForm: FunctionComponent = (props) => { const { visualFlowsApi } = useContext(VisibleFlowsContext)!; - const entitiesContext = useContext(EntitiesContext); const { filteredFieldText, onFilterChange } = useContext(FilteredFieldContext); - const formRef = useRef(null); - const divRef = useRef(null); const flowIdRef = useRef(undefined); - const omitFields = useRef(props.selectedNode.data?.vizNode?.getBaseEntity()?.getOmitFormFields() || []); const visualComponentSchema = useMemo(() => { const answer = props.selectedNode.data?.vizNode?.getComponentSchema(); @@ -36,7 +25,6 @@ export const CanvasForm: FunctionComponent = (props) => { } return answer; }, [props.selectedNode.data?.vizNode]); - const model = visualComponentSchema?.definition; const title = visualComponentSchema?.title; /** Store the flow's initial Id */ @@ -44,41 +32,6 @@ export const CanvasForm: FunctionComponent = (props) => { flowIdRef.current = props.selectedNode.data?.vizNode?.getBaseEntity()?.getId(); }, []); - useEffect(() => { - formRef.current?.form.reset(); - }, [props.selectedNode.data?.vizNode]); - - const handleOnChangeIndividualProp = useCallback( - (path: string, value: unknown) => { - if (!props.selectedNode.data?.vizNode) { - return; - } - - let updatedValue = value; - if (typeof value === 'string' && value.trim() === '') { - updatedValue = undefined; - } - - const newModel = props.selectedNode.data.vizNode.getComponentSchema()?.definition || {}; - setValue(newModel, path, updatedValue); - props.selectedNode.data.vizNode.updateModel(newModel); - entitiesContext?.updateSourceCodeFromEntities(); - }, - [entitiesContext, props.selectedNode.data?.vizNode], - ); - - const stepFeatures = useMemo(() => { - const comment = visualComponentSchema?.schema?.['$comment'] ?? ''; - const isExpressionAwareStep = comment.includes('expression'); - const isDataFormatAwareStep = comment.includes('dataformat'); - const isLoadBalanceAwareStep = comment.includes('loadbalance'); - const isUnknownComponent = - !isDefined(visualComponentSchema) || - !isDefined(visualComponentSchema.schema) || - Object.keys(visualComponentSchema.schema).length === 0; - return { isExpressionAwareStep, isDataFormatAwareStep, isLoadBalanceAwareStep, isUnknownComponent }; - }, [visualComponentSchema]); - const onClose = useCallback(() => { props.onClose?.(); const newId = props.selectedNode.data?.vizNode?.getBaseEntity()?.getId(); @@ -108,25 +61,7 @@ export const CanvasForm: FunctionComponent = (props) => { - {stepFeatures.isUnknownComponent ? ( - - ) : ( - - {stepFeatures.isExpressionAwareStep && } - {stepFeatures.isDataFormatAwareStep && } - {stepFeatures.isLoadBalanceAwareStep && } - -
- - )} + diff --git a/packages/ui/src/components/Visualization/Canvas/CanvasForm.exhaustive.test.tsx b/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.exhaustive.test.tsx similarity index 99% rename from packages/ui/src/components/Visualization/Canvas/CanvasForm.exhaustive.test.tsx rename to packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.exhaustive.test.tsx index f4bf6c76e..ce31c0351 100644 --- a/packages/ui/src/components/Visualization/Canvas/CanvasForm.exhaustive.test.tsx +++ b/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.exhaustive.test.tsx @@ -14,7 +14,7 @@ import { getFirstCatalogMap } from '../../../stubs/test-load-catalog'; import { SchemaService } from '../../Form'; import { CustomAutoFieldDetector } from '../../Form/CustomAutoField'; -describe('CanvasForm', () => { +describe('CanvasFormTabs', () => { let componentCatalogMap: Record; let patternCatalogMap: Record; let kameletCatalogMap: Record; diff --git a/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.scss b/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.scss new file mode 100644 index 000000000..82d3dfe6f --- /dev/null +++ b/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.scss @@ -0,0 +1,3 @@ +.form-tabs { + margin-bottom: 15px; +} diff --git a/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.test.tsx b/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.test.tsx new file mode 100644 index 000000000..95f0b2b2b --- /dev/null +++ b/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.test.tsx @@ -0,0 +1,646 @@ +import catalogLibrary from '@kaoto/camel-catalog/index.json'; +import { CatalogLibrary, RouteDefinition } from '@kaoto/camel-catalog/types'; +import { act, fireEvent, render, screen } from '@testing-library/react'; +import { + CamelCatalogService, + CamelRouteVisualEntity, + CatalogKind, + ICamelComponentDefinition, + ICamelProcessorDefinition, + IKameletDefinition, +} from '../../../models'; +import { IVisualizationNode } from '../../../models/visualization/base-visual-entity'; +import { VisibleFlowsProvider } from '../../../providers'; +import { EntitiesContext, EntitiesProvider } from '../../../providers/entities.provider'; +import { camelRouteJson } from '../../../stubs'; +import { getFirstCatalogMap } from '../../../stubs/test-load-catalog'; +import { SchemaService } from '../../Form'; +import { CanvasNode } from './canvas.models'; +import { CanvasService } from './canvas.service'; +import { CanvasFormTabs } from './CanvasFormTabs'; + +describe('CanvasFormTabs', () => { + let camelRouteVisualEntity: CamelRouteVisualEntity; + let selectedNode: CanvasNode; + let componentCatalogMap: Record; + let patternCatalogMap: Record; + let kameletCatalogMap: Record; + + beforeAll(async () => { + const catalogsMap = await getFirstCatalogMap(catalogLibrary as CatalogLibrary); + componentCatalogMap = catalogsMap.componentCatalogMap; + patternCatalogMap = catalogsMap.patternCatalogMap; + kameletCatalogMap = catalogsMap.kameletsCatalogMap; + + CamelCatalogService.setCatalogKey(CatalogKind.Component, componentCatalogMap); + CamelCatalogService.setCatalogKey(CatalogKind.Pattern, patternCatalogMap); + CamelCatalogService.setCatalogKey(CatalogKind.Kamelet, kameletCatalogMap); + CamelCatalogService.setCatalogKey(CatalogKind.Processor, catalogsMap.modelCatalogMap); + CamelCatalogService.setCatalogKey(CatalogKind.Language, catalogsMap.languageCatalog); + CamelCatalogService.setCatalogKey(CatalogKind.Dataformat, catalogsMap.dataformatCatalog); + CamelCatalogService.setCatalogKey(CatalogKind.Loadbalancer, catalogsMap.loadbalancerCatalog); + CamelCatalogService.setCatalogKey(CatalogKind.Entity, catalogsMap.entitiesCatalog); + }); + + beforeEach(() => { + camelRouteVisualEntity = new CamelRouteVisualEntity(camelRouteJson); + const { nodes } = CanvasService.getFlowDiagram(camelRouteVisualEntity.toVizNode()); + selectedNode = nodes[0]; // timer + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + describe('should show the User-updated field under the modified tab', () => { + it('normal text field', async () => { + render( + + + + + , + ); + + const defaultTab = screen.getByRole('button', { name: 'All Fields' }); + const modifiedTab = screen.getByRole('button', { name: 'User Modified' }); + + expect(defaultTab).toBeInTheDocument(); + expect(modifiedTab).toBeInTheDocument(); + + act(() => { + fireEvent.click(modifiedTab); + }); + + const inputVariableReceiveModifiedTabElement = screen + .queryAllByRole('textbox') + .filter((textbox) => textbox.getAttribute('label') === 'Variable Receive'); + expect(inputVariableReceiveModifiedTabElement).toHaveLength(0); + + act(() => { + fireEvent.click(defaultTab); + }); + + await act(async () => { + const inputVariableReceiveDefaultTabElement = screen + .getAllByRole('textbox') + .filter((textbox) => textbox.getAttribute('label') === 'Variable Receive'); + fireEvent.change(inputVariableReceiveDefaultTabElement[0], { target: { value: 'test' } }); + fireEvent.blur(inputVariableReceiveDefaultTabElement[0]); + }); + + act(() => { + fireEvent.click(modifiedTab); + }); + + const inputVariableReceiveModifiedTabElementNew = screen + .getAllByRole('textbox') + .filter((textbox) => textbox.getAttribute('label') === 'Variable Receive'); + expect(inputVariableReceiveModifiedTabElementNew).toHaveLength(1); + }); + + it('expression field', async () => { + const camelRoute = { + from: { + uri: 'timer:tutorial', + steps: [ + { + setHeader: { + name: 'foo', + }, + }, + ], + }, + } as RouteDefinition; + const entity = new CamelRouteVisualEntity(camelRoute); + const rootNode: IVisualizationNode = entity.toVizNode(); + const setHeaderNode = rootNode.getChildren()![0].getChildren()![0]; + const selectedNode = { + id: '1', + type: 'node', + data: { + vizNode: setHeaderNode, + }, + }; + + render( + + + + + , + ); + + const defaultTab = screen.getByRole('button', { name: 'All Fields' }); + const modifiedTab = screen.getByRole('button', { name: 'User Modified' }); + + act(() => { + fireEvent.click(modifiedTab); + }); + + expect(screen.queryByTestId('launch-expression-modal-btn')).toBeNull(); + + act(() => { + fireEvent.click(defaultTab); + }); + + const launchExpressionDefaultTab = screen.getByTestId('launch-expression-modal-btn'); + + act(() => { + fireEvent.click(launchExpressionDefaultTab); + }); + const button = screen + .getAllByTestId('typeahead-select-input') + .filter((input) => input.innerHTML.includes(SchemaService.DROPDOWN_PLACEHOLDER)); + act(() => { + fireEvent.click(button[0]); + }); + const simple = screen.getByTestId('expression-dropdownitem-simple'); + act(() => { + fireEvent.click(simple.getElementsByTagName('button')[0]); + }); + const expressionInput = screen + .getAllByRole('textbox') + .filter((textbox) => textbox.getAttribute('name') === 'expression'); + const applyBtn = screen.getAllByRole('button').filter((button) => button.textContent === 'Apply'); + act(() => { + fireEvent.input(expressionInput[0], { target: { value: '${header.foo}' } }); + fireEvent.click(applyBtn[0]); + }); + + act(() => { + fireEvent.click(modifiedTab); + }); + + expect(screen.getByTestId('launch-expression-modal-btn')).toBeInTheDocument(); + }); + + it('dataformat field', async () => { + const camelRoute = { + from: { + uri: 'timer:tutorial', + steps: [ + { + marshal: { + id: 'ms', + }, + }, + ], + }, + } as RouteDefinition; + const entity = new CamelRouteVisualEntity(camelRoute); + const rootNode: IVisualizationNode = entity.toVizNode(); + const marshalNode = rootNode.getChildren()![0].getChildren()![0]; + const selectedNode = { + id: '1', + type: 'node', + data: { + vizNode: marshalNode, + }, + }; + + render( + + + + + , + ); + + const defaultTab = screen.getByRole('button', { name: 'All Fields' }); + const modifiedTab = screen.getByRole('button', { name: 'User Modified' }); + act(() => { + fireEvent.click(modifiedTab); + }); + + expect(screen.queryByRole('button', { name: 'Menu toggle' })).toBeNull(); + + act(() => { + fireEvent.click(defaultTab); + }); + + const dataformatDefaultTabButton = screen.getAllByRole('button', { name: 'Menu toggle' }); + await act(async () => { + fireEvent.click(dataformatDefaultTabButton[0]); + }); + const asn1 = screen.getByTestId('dataformat-dropdownitem-asn1'); + await act(async () => { + fireEvent.click(asn1.getElementsByTagName('button')[0]); + }); + + act(() => { + fireEvent.click(modifiedTab); + }); + + expect(screen.queryByRole('button', { name: 'Menu toggle' })).toBeInTheDocument(); + }); + + it('loadbalancer field', async () => { + const camelRoute = { + from: { + uri: 'timer:tutorial', + steps: [ + { + loadBalance: { + id: 'lb', + }, + }, + ], + }, + } as RouteDefinition; + const entity = new CamelRouteVisualEntity(camelRoute); + const rootNode: IVisualizationNode = entity.toVizNode(); + const loadBalanceNode = rootNode.getChildren()![0].getChildren()![0]; + const selectedNode = { + id: '1', + type: 'node', + data: { + vizNode: loadBalanceNode, + }, + }; + + render( + + + + + , + ); + const defaultTab = screen.getByRole('button', { name: 'All Fields' }); + const modifiedTab = screen.getByRole('button', { name: 'User Modified' }); + act(() => { + fireEvent.click(modifiedTab); + }); + + expect(screen.queryByRole('button', { name: 'Menu toggle' })).toBeNull(); + + act(() => { + fireEvent.click(defaultTab); + }); + + const button = screen.getAllByRole('button', { name: 'Menu toggle' }); + await act(async () => { + fireEvent.click(button[0]); + }); + const weighted = screen.getByTestId('loadbalancer-dropdownitem-weighted'); + await act(async () => { + fireEvent.click(weighted.getElementsByTagName('button')[0]); + }); + + act(() => { + fireEvent.click(modifiedTab); + }); + + expect(screen.queryByRole('button', { name: 'Menu toggle' })).toBeInTheDocument(); + }); + }); + + describe('should persists changes from both expression editor and main form', () => { + beforeEach(() => { + jest.spyOn(console, 'error').mockImplementation(() => {}); + }); + + it('expression => main form', async () => { + const camelRoute = { + from: { + uri: 'timer:tutorial', + steps: [ + { + setHeader: { + name: 'foo', + }, + }, + ], + }, + } as RouteDefinition; + const entity = new CamelRouteVisualEntity(camelRoute); + const rootNode: IVisualizationNode = entity.toVizNode(); + const setHeaderNode = rootNode.getChildren()![0].getChildren()![0]; + const selectedNode = { + id: '1', + type: 'node', + data: { + vizNode: setHeaderNode, + }, + }; + + render( + + + + + , + ); + const launchExpression = screen.getByTestId('launch-expression-modal-btn'); + act(() => { + fireEvent.click(launchExpression); + }); + const button = screen + .getAllByTestId('typeahead-select-input') + .filter((input) => input.innerHTML.includes(SchemaService.DROPDOWN_PLACEHOLDER)); + act(() => { + fireEvent.click(button[0]); + }); + const simple = screen.getByTestId('expression-dropdownitem-simple'); + act(() => { + fireEvent.click(simple.getElementsByTagName('button')[0]); + }); + const expressionInput = screen + .getAllByRole('textbox') + .filter((textbox) => textbox.getAttribute('name') === 'expression'); + const applyBtn = screen.getAllByRole('button').filter((button) => button.textContent === 'Apply'); + act(() => { + fireEvent.input(expressionInput[0], { target: { value: '${header.foo}' } }); + fireEvent.click(applyBtn[0]); + }); + /* eslint-disable @typescript-eslint/no-explicit-any */ + expect((camelRoute.from.steps[0].setHeader!.expression as any).simple.expression).toEqual('${header.foo}'); + expect(camelRoute.from.steps[0].setHeader!.name).toEqual('foo'); + + const filtered = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Name'); + act(() => { + fireEvent.input(filtered[0], { target: { value: 'bar' } }); + }); + /* eslint-disable @typescript-eslint/no-explicit-any */ + expect((camelRoute.from.steps[0].setHeader!.expression as any).simple.expression).toEqual('${header.foo}'); + expect(camelRoute.from.steps[0].setHeader!.name).toEqual('bar'); + }); + + it('main form => expression', async () => { + const camelRoute = { + from: { + uri: 'timer:tutorial', + steps: [ + { + setHeader: { + name: 'foo', + }, + }, + ], + }, + } as RouteDefinition; + const entity = new CamelRouteVisualEntity(camelRoute); + const rootNode: IVisualizationNode = entity.toVizNode(); + const setHeaderNode = rootNode.getChildren()![0].getChildren()![0]; + const selectedNode = { + id: '1', + type: 'node', + data: { + vizNode: setHeaderNode, + }, + }; + + render( + + + + + , + ); + const filtered = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Name'); + act(() => { + fireEvent.input(filtered[0], { target: { value: 'bar' } }); + }); + expect(camelRoute.from.steps[0].setHeader!.expression).toBeUndefined(); + expect(camelRoute.from.steps[0].setHeader!.name).toEqual('bar'); + + const launchExpression = screen.getByTestId('launch-expression-modal-btn'); + act(() => { + fireEvent.click(launchExpression); + }); + const button = screen + .getAllByTestId('typeahead-select-input') + .filter((input) => input.innerHTML.includes(SchemaService.DROPDOWN_PLACEHOLDER)); + act(() => { + fireEvent.click(button[0]); + }); + const simple = screen.getByTestId('expression-dropdownitem-simple'); + act(() => { + fireEvent.click(simple.getElementsByTagName('button')[0]); + }); + const expressionInput = screen + .getAllByRole('textbox') + .filter((textbox) => textbox.getAttribute('name') === 'expression'); + const applyBtn = screen.getAllByRole('button').filter((button) => button.textContent === 'Apply'); + act(() => { + fireEvent.input(expressionInput[0], { target: { value: '${header.foo}' } }); + fireEvent.click(applyBtn[0]); + }); + /* eslint-disable @typescript-eslint/no-explicit-any */ + expect((camelRoute.from.steps[0].setHeader!.expression as any).simple.expression).toEqual('${header.foo}'); + expect(camelRoute.from.steps[0].setHeader!.name).toEqual('bar'); + }); + }); + + describe('should persists changes from both dataformat editor and main form', () => { + beforeEach(() => { + jest.spyOn(console, 'error').mockImplementation(() => {}); + }); + + it('dataformat => main form', async () => { + const camelRoute = { + from: { + uri: 'timer:tutorial', + steps: [ + { + marshal: { + id: 'ms', + }, + }, + ], + }, + } as RouteDefinition; + const entity = new CamelRouteVisualEntity(camelRoute); + const rootNode: IVisualizationNode = entity.toVizNode(); + const marshalNode = rootNode.getChildren()![0].getChildren()![0]; + const selectedNode = { + id: '1', + type: 'node', + data: { + vizNode: marshalNode, + }, + }; + + render( + + + + + , + ); + const button = screen.getAllByRole('button', { name: 'Menu toggle' }); + await act(async () => { + fireEvent.click(button[0]); + }); + const avro = screen.getByTestId('dataformat-dropdownitem-avro'); + await act(async () => { + fireEvent.click(avro.getElementsByTagName('button')[0]); + }); + expect(camelRoute.from.steps[0].marshal!.avro).toBeDefined(); + expect(camelRoute.from.steps[0].marshal!.id).toEqual('ms'); + + const idInput = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Id'); + await act(async () => { + fireEvent.input(idInput[1], { target: { value: 'modified' } }); + }); + expect(camelRoute.from.steps[0].marshal!.avro).toBeDefined(); + expect(camelRoute.from.steps[0].marshal!.id).toEqual('modified'); + }); + + it('main form => dataformat', async () => { + const camelRoute = { + from: { + uri: 'timer:tutorial', + steps: [ + { + marshal: { + id: 'ms', + }, + }, + ], + }, + } as RouteDefinition; + const entity = new CamelRouteVisualEntity(camelRoute); + const rootNode: IVisualizationNode = entity.toVizNode(); + const marshalNode = rootNode.getChildren()![0].getChildren()![0]; + const selectedNode = { + id: '1', + type: 'node', + data: { + vizNode: marshalNode, + }, + }; + + render( + + + + + , + ); + const idInput = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Id'); + await act(async () => { + fireEvent.input(idInput[0], { target: { value: 'modified' } }); + }); + expect(camelRoute.from.steps[0].marshal!.avro).toBeUndefined(); + expect(camelRoute.from.steps[0].marshal!.id).toEqual('modified'); + + const button = screen.getAllByRole('button', { name: 'Menu toggle' }); + await act(async () => { + fireEvent.click(button[0]); + }); + const avro = screen.getByTestId('dataformat-dropdownitem-avro'); + await act(async () => { + fireEvent.click(avro.getElementsByTagName('button')[0]); + }); + expect(camelRoute.from.steps[0].marshal!.avro).toBeDefined(); + expect(camelRoute.from.steps[0].marshal!.id).toEqual('modified'); + }); + }); + + describe('should persists changes from both loadbalancer editor and main form', () => { + beforeEach(() => { + jest.spyOn(console, 'error').mockImplementation(() => {}); + }); + + it('loadbalancer => main form', async () => { + const camelRoute = { + from: { + uri: 'timer:tutorial', + steps: [ + { + loadBalance: { + id: 'lb', + }, + }, + ], + }, + } as RouteDefinition; + const entity = new CamelRouteVisualEntity(camelRoute); + const rootNode: IVisualizationNode = entity.toVizNode(); + const loadBalanceNode = rootNode.getChildren()![0].getChildren()![0]; + const selectedNode = { + id: '1', + type: 'node', + data: { + vizNode: loadBalanceNode, + }, + }; + + render( + + + + + , + ); + const button = screen.getAllByRole('button', { name: 'Menu toggle' }); + await act(async () => { + fireEvent.click(button[0]); + }); + const weighted = screen.getByTestId('loadbalancer-dropdownitem-weighted'); + await act(async () => { + fireEvent.click(weighted.getElementsByTagName('button')[0]); + }); + expect(camelRoute.from.steps[0].loadBalance!.weighted).toBeDefined(); + expect(camelRoute.from.steps[0].loadBalance!.id).toEqual('lb'); + + const idInput = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Id'); + await act(async () => { + fireEvent.input(idInput[1], { target: { value: 'modified' } }); + }); + expect(camelRoute.from.steps[0].loadBalance!.weighted).toBeDefined(); + expect(camelRoute.from.steps[0].loadBalance!.id).toEqual('modified'); + }); + + it('main form => loadbalancer', async () => { + const camelRoute = { + from: { + uri: 'timer:tutorial', + steps: [ + { + loadBalance: { + id: 'lb', + }, + }, + ], + }, + } as RouteDefinition; + const entity = new CamelRouteVisualEntity(camelRoute); + const rootNode: IVisualizationNode = entity.toVizNode(); + const loadBalanceNode = rootNode.getChildren()![0].getChildren()![0]; + const selectedNode = { + id: '1', + type: 'node', + data: { + vizNode: loadBalanceNode, + }, + }; + + render( + + + + + , + ); + const idInput = screen.getAllByRole('textbox').filter((textbox) => textbox.getAttribute('label') === 'Id'); + await act(async () => { + fireEvent.input(idInput[0], { target: { value: 'modified' } }); + }); + expect(camelRoute.from.steps[0].loadBalance!.weighted).toBeUndefined(); + expect(camelRoute.from.steps[0].loadBalance!.id).toEqual('modified'); + + const button = screen.getAllByRole('button', { name: 'Menu toggle' }); + await act(async () => { + fireEvent.click(button[0]); + }); + const weighted = screen.getByTestId('loadbalancer-dropdownitem-weighted'); + await act(async () => { + fireEvent.click(weighted.getElementsByTagName('button')[0]); + }); + expect(camelRoute.from.steps[0].loadBalance!.weighted).toBeDefined(); + expect(camelRoute.from.steps[0].loadBalance!.id).toEqual('modified'); + }); + }); +}); diff --git a/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.tsx b/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.tsx new file mode 100644 index 000000000..774188b6c --- /dev/null +++ b/packages/ui/src/components/Visualization/Canvas/CanvasFormTabs.tsx @@ -0,0 +1,128 @@ +import { ToggleGroup, ToggleGroupItem } from '@patternfly/react-core'; +import { FunctionComponent, useCallback, useContext, useEffect, useMemo, useRef, useState } from 'react'; +import { EntitiesContext } from '../../../providers/entities.provider'; +import { SchemaBridgeProvider } from '../../../providers/schema-bridge.provider'; +import { getUserUpdatedPropertiesSchema, isDefined, setValue } from '../../../utils'; +import { CustomAutoForm, CustomAutoFormRef } from '../../Form/CustomAutoForm'; +import { DataFormatEditor } from '../../Form/dataFormat/DataFormatEditor'; +import { LoadBalancerEditor } from '../../Form/loadBalancer/LoadBalancerEditor'; +import { StepExpressionEditor } from '../../Form/stepExpression/StepExpressionEditor'; +import { UnknownNode } from '../Custom/UnknownNode'; +import './CanvasFormTabs.scss'; +import { CanvasNode } from './canvas.models'; + +interface CanvasFormTabsProps { + selectedNode: CanvasNode; +} + +export enum FormTabsModes { + ALL_FIELDS = 'All Fields', + USER_MODIFIED = 'User Modified', +} + +export const CanvasFormTabs: FunctionComponent = (props) => { + const entitiesContext = useContext(EntitiesContext); + const divRef = useRef(null); + const formRef = useRef(null); + const omitFields = useRef(props.selectedNode.data?.vizNode?.getBaseEntity()?.getOmitFormFields() || []); + const [selectedTab, setSelectedTab] = useState(FormTabsModes.ALL_FIELDS); + + const visualComponentSchema = useMemo(() => { + const answer = props.selectedNode.data?.vizNode?.getComponentSchema(); + // Overriding parameters with an empty object When the parameters property is mistakenly set to null + if (answer?.definition?.parameters === null) { + answer!.definition.parameters = {}; + } + return answer; + }, [props.selectedNode.data?.vizNode, selectedTab]); + const model = visualComponentSchema?.definition; + let processedSchema = visualComponentSchema?.schema; + if (selectedTab === FormTabsModes.USER_MODIFIED) { + processedSchema = { + ...visualComponentSchema?.schema, + properties: getUserUpdatedPropertiesSchema(visualComponentSchema?.schema.properties ?? {}, model), + }; + } + + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const handleItemClick = (event: any, _isSelected: boolean) => { + const id = event.currentTarget.id; + setSelectedTab(id); + }; + + useEffect(() => { + formRef.current?.form.reset(); + }, [props.selectedNode.data?.vizNode, selectedTab]); + + const stepFeatures = useMemo(() => { + const comment = visualComponentSchema?.schema?.['$comment'] ?? ''; + const isExpressionAwareStep = comment.includes('expression'); + const isDataFormatAwareStep = comment.includes('dataformat'); + const isLoadBalanceAwareStep = comment.includes('loadbalance'); + const isUnknownComponent = + !isDefined(visualComponentSchema) || + !isDefined(visualComponentSchema.schema) || + Object.keys(visualComponentSchema.schema).length === 0; + return { isExpressionAwareStep, isDataFormatAwareStep, isLoadBalanceAwareStep, isUnknownComponent }; + }, [visualComponentSchema]); + + const handleOnChangeIndividualProp = useCallback( + (path: string, value: unknown) => { + if (!props.selectedNode.data?.vizNode) { + return; + } + + let updatedValue = value; + if (typeof value === 'string' && value.trim() === '') { + updatedValue = undefined; + } + + const newModel = props.selectedNode.data.vizNode.getComponentSchema()?.definition || {}; + setValue(newModel, path, updatedValue); + props.selectedNode.data.vizNode.updateModel(newModel); + entitiesContext?.updateSourceCodeFromEntities(); + }, + [entitiesContext, props.selectedNode.data?.vizNode], + ); + + return ( + <> + {stepFeatures.isUnknownComponent ? ( + + ) : ( + + + {Object.values(FormTabsModes).map((mode) => ( + + ))} + + {stepFeatures.isExpressionAwareStep && ( + + )} + {stepFeatures.isDataFormatAwareStep && ( + + )} + {stepFeatures.isLoadBalanceAwareStep && ( + + )} + +
+ + )} + + ); +}; diff --git a/packages/ui/src/components/Visualization/Canvas/__snapshots__/CanvasForm.test.tsx.snap b/packages/ui/src/components/Visualization/Canvas/__snapshots__/CanvasForm.test.tsx.snap index 0cc30f65b..5c93e506a 100644 --- a/packages/ui/src/components/Visualization/Canvas/__snapshots__/CanvasForm.test.tsx.snap +++ b/packages/ui/src/components/Visualization/Canvas/__snapshots__/CanvasForm.test.tsx.snap @@ -105,6 +105,44 @@ exports[`CanvasForm should render 1`] = `
+
+
+ +
+
+ +
+
{ + const schema = { + type: 'object', + properties: { + id: { + title: 'Id', + type: 'string', + }, + description: { + title: 'Description', + type: 'string', + }, + uri: { + title: 'Uri', + type: 'string', + }, + variableReceive: { + title: 'Variable Receive', + type: 'string', + }, + parameters: { + type: 'object', + title: 'Endpoint Properties', + description: 'Endpoint properties description', + properties: { + timerName: { + title: 'Timer Name', + type: 'string', + }, + delay: { + title: 'Delay', + type: 'string', + default: '1000', + }, + fixedRate: { + title: 'Fixed Rate', + type: 'boolean', + default: false, + }, + includeMetadata: { + title: 'Include Metadata', + type: 'boolean', + default: false, + }, + period: { + title: 'Period', + type: 'string', + default: '1000', + }, + repeatCount: { + title: 'Repeat Count', + type: 'integer', + }, + exceptionHandler: { + title: 'Exception Handler', + type: 'string', + $comment: 'class:org.apache.camel.spi.ExceptionHandler', + }, + exchangePattern: { + title: 'Exchange Pattern', + type: 'string', + enum: ['InOnly', 'InOut'], + }, + synchronous: { + title: 'Synchronous', + type: 'boolean', + default: false, + }, + timer: { + title: 'Timer', + type: 'string', + $comment: 'class:java.util.Timer', + }, + runLoggingLevel: { + title: 'Run Logging Level', + type: 'string', + default: 'TRACE', + enum: ['TRACE', 'DEBUG', 'INFO', 'WARN', 'ERROR', 'OFF'], + }, + }, + required: ['timerName'], + }, + }, + } as unknown as KaotoSchemaDefinition['schema']; + + const inputModel: Record = { + id: 'from-2860', + uri: 'timer', + variableReceive: 'test', + parameters: { + period: '1000', + timerName: 'template', + exceptionHandler: '#test', + exchangePattern: 'InOnly', + fixedRate: true, + runLoggingLevel: 'OFF', + repeatCount: '2', + time: undefined, + timer: '#testbean', + }, + steps: [ + { + log: { + id: 'log-2942', + message: 'template message', + }, + }, + ], + }; + + const expectedSchema = { + id: { + title: 'Id', + type: 'string', + }, + uri: { + title: 'Uri', + type: 'string', + }, + parameters: { + type: 'object', + title: 'Endpoint Properties', + description: 'Endpoint properties description', + properties: { + timerName: { + title: 'Timer Name', + type: 'string', + }, + fixedRate: { + title: 'Fixed Rate', + type: 'boolean', + default: false, + }, + repeatCount: { + title: 'Repeat Count', + type: 'integer', + }, + exceptionHandler: { + title: 'Exception Handler', + type: 'string', + $comment: 'class:org.apache.camel.spi.ExceptionHandler', + }, + exchangePattern: { + title: 'Exchange Pattern', + type: 'string', + enum: ['InOnly', 'InOut'], + }, + timer: { + title: 'Timer', + type: 'string', + $comment: 'class:java.util.Timer', + }, + runLoggingLevel: { + title: 'Run Logging Level', + type: 'string', + default: 'TRACE', + enum: ['TRACE', 'DEBUG', 'INFO', 'WARN', 'ERROR', 'OFF'], + }, + }, + required: ['timerName'], + }, + }; + + it('should return only the properties which are user Modified', () => { + const procesedSchema = getUserUpdatedPropertiesSchema(schema.properties!, inputModel); + expect(procesedSchema).toMatchObject(expectedSchema); + }); +}); diff --git a/packages/ui/src/utils/get-user-updated-properties-schema.ts b/packages/ui/src/utils/get-user-updated-properties-schema.ts new file mode 100644 index 000000000..e43e6a29c --- /dev/null +++ b/packages/ui/src/utils/get-user-updated-properties-schema.ts @@ -0,0 +1,41 @@ +import { isDefined } from './is-defined'; + +export function getUserUpdatedPropertiesSchema( + // eslint-disable-next-line @typescript-eslint/no-explicit-any + obj1: Record, + // eslint-disable-next-line @typescript-eslint/no-explicit-any + obj2: Record, + // eslint-disable-next-line @typescript-eslint/no-explicit-any +): Record { + if (!isDefined(obj1) || !isDefined(obj2)) return {}; + + const nonDefaultFormSchema = Object.entries(obj1).reduce( + (acc, item) => { + if (item[0] in obj2 && isDefined(obj2[item[0]])) { + if (item[1]['type'] === 'string' || item[1]['type'] === 'boolean' || item[1]['type'] === 'integer') { + if ('default' in item[1]) { + if (!(item[1]['default'] == obj2[item[0]])) { + acc[item[0]] = item[1]; + } + } else { + acc[item[0]] = item[1]; + } + } else if (item[1]['type'] === 'object' && Object.keys(obj2[item[0]]).length > 0) { + if ('properties' in item[1]) { + const subSchema = getUserUpdatedPropertiesSchema(item[1]['properties'], obj2[item[0]]); + acc[item[0]] = { ...item[1], properties: subSchema }; + } else { + acc[item[0]] = item[1]; + } + } else if (item[1]['type'] === 'array' && obj2[item[0]].length > 0) { + acc[item[0]] = item[1]; + } + } + + return acc; + }, + {} as Record, + ); + + return nonDefaultFormSchema; +} diff --git a/packages/ui/src/utils/index.ts b/packages/ui/src/utils/index.ts index 3937fc85e..44fb2af7f 100644 --- a/packages/ui/src/utils/index.ts +++ b/packages/ui/src/utils/index.ts @@ -14,4 +14,5 @@ export * from './update-kamelet-from-custom-schema'; export * from './pipe-custom-schema'; export * from './get-field-groups'; export * from './get-serialized-model'; +export * from './get-user-updated-properties-schema'; export * from './weight-schema-properties';