@finos/legend-application-studio
Version:
Legend Studio application core
228 lines • 18.3 kB
JavaScript
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
/**
* Copyright (c) 2020-present, Goldman Sachs
*
* 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 { useState, useCallback, useRef } from 'react';
import { observer } from 'mobx-react-lite';
import { UMLEditorState, UML_EDITOR_TAB, } from '../../../../stores/editor/editor-state/element-editor-state/UMLEditorState.js';
import { useDrag, useDrop } from 'react-dnd';
import { CORE_DND_TYPE, } from '../../../../stores/editor/utils/DnDUtils.js';
import { prettyCONSTName } from '@finos/legend-shared';
import { BlankPanelContent, clsx, getCollapsiblePanelGroupProps, InputWithInlineValidation, ResizablePanel, ResizablePanelGroup, ResizablePanelSplitter, ResizablePanelSplitterLine, PlusIcon, TimesIcon, LongArrowRightIcon, LockIcon, FireIcon, StickArrowCircleRightIcon, PanelEntryDragHandle, DragPreviewLayer, useDragPreviewLayer, PanelDropZone, Panel, PanelContent, PanelDnDEntry, PanelHeader, PanelHeaderActionItem, PanelHeaderActions, PanelContentLists, } from '@finos/legend-art';
import { LEGEND_STUDIO_TEST_ID } from '../../../../__lib__/LegendStudioTesting.js';
import { StereotypeDragPreviewLayer, StereotypeSelector, } from './StereotypeSelector.js';
import { TaggedValueDragPreviewLayer, TaggedValueEditor, } from './TaggedValueEditor.js';
import { useEditorStore } from '../../EditorStoreProvider.js';
import { Profile, StereotypeExplicitReference, stub_TaggedValue, stub_Tag, stub_Profile, stub_Stereotype, stub_Enum, } from '@finos/legend-graph';
import { enum_setName, annotatedElement_deleteStereotype, annotatedElement_addStereotype, annotatedElement_addTaggedValue, annotatedElement_deleteTaggedValue, enum_deleteValue, enum_addValue, enum_swapValues, } from '../../../../stores/graph-modifier/DomainGraphModifierHelper.js';
import { useApplicationNavigationContext } from '@finos/legend-application';
import { LEGEND_STUDIO_APPLICATION_NAVIGATION_CONTEXT_KEY } from '../../../../__lib__/LegendStudioApplicationNavigationContext.js';
const ENUM_VALUE_DND_TYPE = 'ENUMERATION';
const EnumBasicEditor = observer((props) => {
const ref = useRef(null);
const handleRef = useRef(null);
const { enumValue, selectValue, deleteValue, isReadOnly } = props;
const changeValue = (event) => {
enum_setName(enumValue, event.target.value);
};
const isEnumValueDuplicated = (val) => enumValue._OWNER.values.filter((value) => value.name === val.name)
.length >= 2;
// Drag and Drop
const handleHover = useCallback((item) => {
const draggingEnumeration = item.enumValue;
const hoveredEnumeration = enumValue;
enum_swapValues(enumValue._OWNER, draggingEnumeration, hoveredEnumeration);
}, [enumValue]);
const [{ isBeingDraggedEnumeration }, dropConnector] = useDrop(() => ({
accept: [ENUM_VALUE_DND_TYPE],
hover: (item) => handleHover(item),
collect: (monitor) => ({
isBeingDraggedEnumeration: monitor.getItem()?.enumValue,
}),
}), [handleHover]);
const isBeingDragged = enumValue === isBeingDraggedEnumeration;
const [, dragConnector, dragPreviewConnector] = useDrag(() => ({
type: ENUM_VALUE_DND_TYPE,
item: () => ({
enumValue: enumValue,
}),
}), [enumValue]);
dragConnector(handleRef);
dropConnector(ref);
useDragPreviewLayer(dragPreviewConnector);
return (_jsxs(PanelDnDEntry, { ref: ref, showPlaceholder: isBeingDragged, placeholder: _jsx("div", { className: "dnd__placeholder--light" }), className: "enum-basic-editor__container", children: [_jsx(PanelEntryDragHandle, { dragSourceConnector: handleRef, isDragging: isBeingDragged }), _jsxs("div", { className: "enum-basic-editor", children: [_jsx(InputWithInlineValidation, { className: "enum-basic-editor__name input-group__input", spellCheck: false, disabled: isReadOnly, value: enumValue.name, onChange: changeValue, placeholder: "Enum name", error: isEnumValueDuplicated(enumValue) ? 'Duplicated enum' : undefined }), _jsx("button", { className: "uml-element-editor__basic__detail-btn", onClick: selectValue, tabIndex: -1, title: "See detail", children: _jsx(LongArrowRightIcon, {}) }), !isReadOnly && (_jsx("button", { className: "uml-element-editor__remove-btn", disabled: isReadOnly, onClick: deleteValue, tabIndex: -1, title: "Remove", children: _jsx(TimesIcon, {}) }))] })] }));
});
const EnumEditor = observer((props) => {
const { _enum, deselectValue, isReadOnly } = props;
// Tab
const [selectedTab, setSelectedTab] = useState(UML_EDITOR_TAB.TAGGED_VALUES);
const tabs = [UML_EDITOR_TAB.TAGGED_VALUES, UML_EDITOR_TAB.STEREOTYPES];
const changeTab = (tab) => () => setSelectedTab(tab);
// Tagged value and Stereotype
let addButtonTitle = '';
switch (selectedTab) {
case UML_EDITOR_TAB.TAGGED_VALUES:
addButtonTitle = 'Add tagged value';
break;
case UML_EDITOR_TAB.STEREOTYPES:
addButtonTitle = 'Add stereotype';
break;
default:
break;
}
const addValue = () => {
if (!isReadOnly) {
if (selectedTab === UML_EDITOR_TAB.TAGGED_VALUES) {
annotatedElement_addTaggedValue(_enum, stub_TaggedValue(stub_Tag(stub_Profile())));
}
else if (selectedTab === UML_EDITOR_TAB.STEREOTYPES) {
annotatedElement_addStereotype(_enum, StereotypeExplicitReference.create(stub_Stereotype(stub_Profile())));
}
}
};
const _deleteStereotype = (val) => () => annotatedElement_deleteStereotype(_enum, val);
const _deleteTaggedValue = (val) => () => annotatedElement_deleteTaggedValue(_enum, val);
// Drag and Drop
const handleDropTaggedValue = useCallback((item) => {
if (!isReadOnly && item.data.packageableElement instanceof Profile) {
annotatedElement_addTaggedValue(_enum, stub_TaggedValue(stub_Tag(item.data.packageableElement)));
}
}, [_enum, isReadOnly]);
const [{ isTaggedValueDragOver }, taggedValueDropConnector] = useDrop(() => ({
accept: [CORE_DND_TYPE.PROJECT_EXPLORER_PROFILE],
drop: (item) => handleDropTaggedValue(item),
collect: (monitor) => ({
isTaggedValueDragOver: monitor.isOver({ shallow: true }),
}),
}), [handleDropTaggedValue]);
const handleDropStereotype = useCallback((item) => {
if (!isReadOnly && item.data.packageableElement instanceof Profile) {
annotatedElement_addStereotype(_enum, StereotypeExplicitReference.create(stub_Stereotype(item.data.packageableElement)));
}
}, [_enum, isReadOnly]);
const [{ isStereotypeDragOver }, stereotypeDropConnector] = useDrop(() => ({
accept: [CORE_DND_TYPE.PROJECT_EXPLORER_PROFILE],
drop: (item) => handleDropStereotype(item),
collect: (monitor) => ({
isStereotypeDragOver: monitor.isOver({ shallow: true }),
}),
}), [handleDropStereotype]);
return (_jsx("div", { className: "uml-element-editor enum-editor", children: _jsxs("div", { "data-testid": LEGEND_STUDIO_TEST_ID.PANEL, className: "panel", children: [_jsxs(PanelHeader, { children: [_jsxs("div", { className: "panel__header__title", children: [isReadOnly && (_jsx("div", { className: "uml-element-editor__header__lock", children: _jsx(LockIcon, {}) })), _jsx("div", { className: "panel__header__title__label", children: "enum" }), _jsx("div", { className: "panel__header__title__content", children: _enum.name })] }), _jsx(PanelHeaderActions, { children: _jsx(PanelHeaderActionItem, { onClick: deselectValue, title: "Close", children: _jsx(TimesIcon, {}) }) })] }), _jsxs("div", { "data-testid": LEGEND_STUDIO_TEST_ID.UML_ELEMENT_EDITOR__TABS_HEADER, className: "panel__header uml-element-editor__tabs__header", children: [_jsx("div", { className: "uml-element-editor__tabs", children: tabs.map((tab) => (_jsx("div", { onClick: changeTab(tab), className: clsx('uml-element-editor__tab', {
'uml-element-editor__tab--active': tab === selectedTab,
}), children: prettyCONSTName(tab) }, tab))) }), _jsx(PanelHeaderActions, { children: _jsx(PanelHeaderActionItem, { disabled: isReadOnly, onClick: addValue, title: addButtonTitle, children: _jsx(PlusIcon, {}) }) })] }), _jsxs(PanelContent, { children: [selectedTab === UML_EDITOR_TAB.TAGGED_VALUES && (_jsx(PanelDropZone, { isDragOver: isTaggedValueDragOver && !isReadOnly, dropTargetConnector: taggedValueDropConnector, children: _jsxs(PanelContentLists, { children: [_jsx(TaggedValueDragPreviewLayer, {}), _enum.taggedValues.map((taggedValue) => (_jsx(TaggedValueEditor, { annotatedElement: _enum, taggedValue: taggedValue, deleteValue: _deleteTaggedValue(taggedValue), isReadOnly: isReadOnly }, taggedValue._UUID)))] }) })), selectedTab === UML_EDITOR_TAB.STEREOTYPES && (_jsx(PanelDropZone, { isDragOver: isStereotypeDragOver && !isReadOnly, dropTargetConnector: stereotypeDropConnector, children: _jsxs(PanelContentLists, { children: [_jsx(StereotypeDragPreviewLayer, {}), _enum.stereotypes.map((stereotype) => (_jsx(StereotypeSelector, { annotatedElement: _enum, stereotype: stereotype, deleteStereotype: _deleteStereotype(stereotype), isReadOnly: isReadOnly }, stereotype.value._UUID)))] }) }))] })] }) }));
});
export const EnumerationEditor = observer((props) => {
const { enumeration } = props;
const editorStore = useEditorStore();
const editorState = editorStore.tabManagerState.getCurrentEditorState(UMLEditorState);
const isReadOnly = editorState.isReadOnly;
// Selected enum value
const [selectedEnum, setSelectedEnum] = useState();
const deselectValue = () => setSelectedEnum(undefined);
const selectValue = (val) => () => setSelectedEnum(val);
// Tab
const selectedTab = editorState.selectedTab;
const tabs = [
UML_EDITOR_TAB.ENUM_VALUES,
UML_EDITOR_TAB.TAGGED_VALUES,
UML_EDITOR_TAB.STEREOTYPES,
];
const changeTab = (tab) => () => {
editorState.setSelectedTab(tab);
setSelectedEnum(undefined);
};
// Tagged value and Stereotype
let addButtonTitle = '';
switch (selectedTab) {
case UML_EDITOR_TAB.ENUM_VALUES:
addButtonTitle = 'Add enum value';
break;
case UML_EDITOR_TAB.TAGGED_VALUES:
addButtonTitle = 'Add tagged value';
break;
case UML_EDITOR_TAB.STEREOTYPES:
addButtonTitle = 'Add stereotype';
break;
default:
break;
}
const add = () => {
if (!isReadOnly) {
if (selectedTab === UML_EDITOR_TAB.ENUM_VALUES) {
enum_addValue(enumeration, stub_Enum(enumeration));
}
else if (selectedTab === UML_EDITOR_TAB.TAGGED_VALUES) {
annotatedElement_addTaggedValue(enumeration, stub_TaggedValue(stub_Tag(stub_Profile())));
}
else if (selectedTab === UML_EDITOR_TAB.STEREOTYPES) {
annotatedElement_addStereotype(enumeration, StereotypeExplicitReference.create(stub_Stereotype(stub_Profile())));
}
}
};
const deleteValue = (val) => () => {
enum_deleteValue(enumeration, val);
if (val === selectedEnum) {
setSelectedEnum(undefined);
}
};
const _deleteStereotype = (val) => () => annotatedElement_deleteStereotype(enumeration, val);
const _deleteTaggedValue = (val) => () => annotatedElement_deleteTaggedValue(enumeration, val);
// Drag and Drop
const handleDropTaggedValue = useCallback((item) => {
if (!isReadOnly && item.data.packageableElement instanceof Profile) {
annotatedElement_addTaggedValue(enumeration, stub_TaggedValue(stub_Tag(item.data.packageableElement)));
}
}, [enumeration, isReadOnly]);
const [{ isTaggedValueDragOver }, taggedValueDropConnector] = useDrop(() => ({
accept: [CORE_DND_TYPE.PROJECT_EXPLORER_PROFILE],
drop: (item) => handleDropTaggedValue(item),
collect: (monitor) => ({
isTaggedValueDragOver: monitor.isOver({ shallow: true }),
}),
}), [handleDropTaggedValue]);
const handleDropStereotype = useCallback((item) => {
if (!isReadOnly && item.data.packageableElement instanceof Profile) {
annotatedElement_addStereotype(enumeration, StereotypeExplicitReference.create(stub_Stereotype(item.data.packageableElement)));
}
}, [enumeration, isReadOnly]);
const [{ isStereotypeDragOver }, stereotypeDropConnector] = useDrop(() => ({
accept: [CORE_DND_TYPE.PROJECT_EXPLORER_PROFILE],
drop: (item) => handleDropStereotype(item),
collect: (monitor) => ({
isStereotypeDragOver: monitor.isOver({ shallow: true }),
}),
}), [handleDropStereotype]);
// Generation
const generationParentElementPath = editorStore.graphState.graphGenerationState.findGenerationParentPath(enumeration.path);
const generationParentElement = generationParentElementPath
? editorStore.graphManagerState.graph.getNullableElement(generationParentElementPath)
: undefined;
const visitGenerationParentElement = () => {
if (generationParentElement) {
editorStore.graphEditorMode.openElement(generationParentElement);
}
};
useApplicationNavigationContext(LEGEND_STUDIO_APPLICATION_NAVIGATION_CONTEXT_KEY.ENUMERATION_EDITOR);
// layout
const enumEditorCollapsiblePanelGroupProps = getCollapsiblePanelGroupProps(!selectedEnum, {
size: 250,
});
return (_jsx("div", { "data-testid": LEGEND_STUDIO_TEST_ID.ENUMERATION_EDITOR, className: "uml-element-editor enumeration-editor", children: _jsxs(ResizablePanelGroup, { orientation: "horizontal", children: [_jsx(ResizablePanel, { ...enumEditorCollapsiblePanelGroupProps.remainingPanel, minSize: 56, children: _jsxs(Panel, { children: [_jsxs("div", { className: "panel__header", children: [_jsxs("div", { className: "panel__header__title", children: [_jsx("div", { className: "panel__header__title__label", children: "enumeration" }), _jsx("div", { className: "panel__header__title__content", children: enumeration.name })] }), _jsx("div", { className: "panel__header__actions", children: generationParentElement && (_jsxs("button", { className: "uml-element-editor__header__generation-origin", onClick: visitGenerationParentElement, tabIndex: -1, title: `Visit generation parent '${generationParentElement.path}'`, children: [_jsx("div", { className: "uml-element-editor__header__generation-origin__label", children: _jsx(FireIcon, {}) }), _jsx("div", { className: "uml-element-editor__header__generation-origin__parent-name", children: generationParentElement.name }), _jsx("div", { className: "uml-element-editor__header__generation-origin__visit-btn", children: _jsx(StickArrowCircleRightIcon, {}) })] })) })] }), _jsxs("div", { "data-testid": LEGEND_STUDIO_TEST_ID.UML_ELEMENT_EDITOR__TABS_HEADER, className: "panel__header uml-element-editor__tabs__header", children: [_jsx("div", { className: "uml-element-editor__tabs", children: tabs.map((tab) => (_jsx("div", { onClick: changeTab(tab), className: clsx('uml-element-editor__tab', {
'uml-element-editor__tab--active': tab === selectedTab,
}), children: prettyCONSTName(tab) }, tab))) }), _jsx("div", { className: "panel__header__actions", children: _jsx("button", { className: "panel__header__action", onClick: add, tabIndex: -1, title: addButtonTitle, children: _jsx(PlusIcon, {}) }) })] }), _jsxs(PanelContent, { children: [selectedTab === UML_EDITOR_TAB.ENUM_VALUES && (_jsxs(PanelContentLists, { children: [_jsx(DragPreviewLayer, { labelGetter: (item) => item.enumValue.name === ''
? '(unknown)'
: item.enumValue.name, types: [ENUM_VALUE_DND_TYPE] }), enumeration.values.map((enumValue) => (_jsx(EnumBasicEditor, { enumValue: enumValue, deleteValue: deleteValue(enumValue), selectValue: selectValue(enumValue), isReadOnly: isReadOnly }, enumValue._UUID)))] })), selectedTab === UML_EDITOR_TAB.TAGGED_VALUES && (_jsx(PanelDropZone, { isDragOver: isTaggedValueDragOver && !isReadOnly, dropTargetConnector: taggedValueDropConnector, children: _jsxs(PanelContentLists, { children: [_jsx(TaggedValueDragPreviewLayer, {}), enumeration.taggedValues.map((taggedValue) => (_jsx(TaggedValueEditor, { annotatedElement: enumeration, taggedValue: taggedValue, deleteValue: _deleteTaggedValue(taggedValue), isReadOnly: isReadOnly }, taggedValue._UUID)))] }) })), selectedTab === UML_EDITOR_TAB.STEREOTYPES && (_jsx(PanelDropZone, { isDragOver: isStereotypeDragOver && !isReadOnly, dropTargetConnector: stereotypeDropConnector, children: _jsxs(PanelContentLists, { children: [_jsx(StereotypeDragPreviewLayer, {}), enumeration.stereotypes.map((stereotype) => (_jsx(StereotypeSelector, { annotatedElement: enumeration, stereotype: stereotype, deleteStereotype: _deleteStereotype(stereotype), isReadOnly: isReadOnly }, stereotype.value._UUID)))] }) }))] })] }) }), _jsx(ResizablePanelSplitter, { children: _jsx(ResizablePanelSplitterLine, { color: "var(--color-light-grey-200)" }) }), _jsx(ResizablePanel, { ...enumEditorCollapsiblePanelGroupProps.collapsiblePanel, direction: -1, children: selectedEnum ? (_jsx(EnumEditor, { _enum: selectedEnum, deselectValue: deselectValue, isReadOnly: isReadOnly })) : (_jsx("div", { className: "uml-element-editor__sub-editor", children: _jsx(BlankPanelContent, { children: "No enum selected" }) })) })] }) }));
});
//# sourceMappingURL=EnumerationEditor.js.map