diff --git a/.sonarcloud.properties b/.sonarcloud.properties
new file mode 100644
index 0000000..61ad04c
--- /dev/null
+++ b/.sonarcloud.properties
@@ -0,0 +1,18 @@
+# Path to sources
+sonar.sources=src
+#sonar.exclusions=
+#sonar.inclusions=
+
+# Path to tests
+sonar.tests=test
+#sonar.test.exclusions=
+#sonar.test.inclusions=
+
+# Source encoding
+sonar.sourceEncoding=UTF-8
+
+# Exclusions for copy-paste detection
+sonar.cpd.exclusions=stories,test
+
+sonar.language=ts
+sonar.typescript.tsconfigPath=tsconfig.json
diff --git a/src/components/diagram/Graph.tsx b/src/components/diagram/Graph.tsx
index 9b20424..358ca77 100644
--- a/src/components/diagram/Graph.tsx
+++ b/src/components/diagram/Graph.tsx
@@ -68,6 +68,9 @@ export const Graph = ({
graph.resize(width, height);
};
resizeFn();
+ const resizeObserver = new ResizeObserver((entries) => {
+ resizeFn();
+ });
graph.selection.widget.collection.on('updated', (e) => {
const nodeIds = graph.selection.widget.collection.cells.map((c) => {
return { id: c.id, ...c.store.data.subject };
@@ -76,10 +79,12 @@ export const Graph = ({
});
graph.enableRubberband();
window.addEventListener('resize', resizeFn);
+ resizeObserver.observe(refWrap.current);
// dispose attached HTML objects
return () => {
graph.dispose();
window.removeEventListener('resize', resizeFn);
+ resizeObserver.disconnect();
};
}, []);
@@ -143,7 +148,6 @@ export const Graph = ({
Load More
*/}
-
((props) => {
props,
store,
);
+ console.log('GraphRenderer - start');
+ // ensure that we render Graph only after all data is not empty
+ let isAllNotEmpty = true;
+ const elements = findElementsRecursive(viewDescr, viewKindElement.elements || [], (params: any) => {
+ const collIriOverride = params[2]; ////[id, collIri, collIriOverride, inCollPath, viewKindElement, viewDescrElement]
+ if (!store.getColl(collIriOverride) || store.getColl(collIriOverride)?.data.length <= 0) {
+ isAllNotEmpty = false;
+ console.log('GraphRenderer - data - empty', collIriOverride);
+ return true;
+ }
+ return false;
+ });
- const options = viewKindElement.options || {};
- const regStencils = (stencils, arr) => {
- arr.forEach((e) => {
+ if (!isAllNotEmpty) {
+ console.log('GraphRenderer - data != OK');
+ return
;
+ } else {
+ console.log('GraphRenderer - data = OK');
+ const options = viewKindElement.options || {};
+ const regStencils = (stencils, arr) => {
+ arr.forEach((e) => {
+ if (e.elements) {
+ regStencils(stencils, e.elements);
+ }
+ if (e['@type'] === 'aldkg:DiagramNodeVKElement') {
+ stencils[e['@id']] = e;
+ }
+ });
+ };
+ const stencilPanel: any = {};
+ const viewKindStencils = ((viewKindElement as any)?.elements || []).reduce((acc, e) => {
if (e.elements) {
- regStencils(stencils, e.elements);
- }
- if (e['@type'] === 'aldkg:DiagramNodeVKElement') {
- stencils[e['@id']] = e;
+ regStencils(acc, e.elements);
}
- });
- };
- const stencilPanel: any = {};
- const viewKindStencils = ((viewKindElement as any)?.elements || []).reduce((acc, e) => {
- if (e.elements) {
- regStencils(acc, e.elements);
- }
- acc[e['@id']] = e;
- stencilPanel[e['@id']] = e;
- return acc;
- }, {});
+ acc[e['@id']] = e;
+ stencilPanel[e['@id']] = e;
+ return acc;
+ }, {});
- const dataSource = ((viewKindElement as any)?.elements || []).reduce((acc, e) => {
- if (e.resultsScope) {
- const dataUri = (viewDescr as any).collsConstrs.filter((el) => compareByIri(el['@parent'], e.resultsScope));
- if (dataUri.length > 0) {
- const graphData = store.getColl(dataUri[0]);
- acc[e['@id']] = graphData?.data ? getSnapshot(graphData?.data) : [];
- } else {
- console.log('No data for element', e);
+ const dataSource = ((viewKindElement as any)?.elements || []).reduce((acc, e) => {
+ if (e.resultsScope) {
+ const dataUri = (viewDescr as any).collsConstrs.filter((el) => compareByIri(el['@parent'], e.resultsScope));
+ if (dataUri.length > 0) {
+ const graphData = store.getColl(dataUri[0]);
+ acc[e['@id']] = graphData?.data ? getSnapshot(graphData?.data) : [];
+ } else {
+ console.log('No data for element', e);
+ }
}
- }
- return acc;
- }, {});
- const scope = viewKindElement.resultsScope;
- const withConnections = options.connections;
- const onChange = (data: any) => {
- if (data) {
- store.setSelectedData(scope, data);
- withConnections && store.editConn(withConnections, data[0]);
- }
- };
+ return acc;
+ }, {});
+ const scope = viewKindElement.resultsScope;
+ const withConnections = options.connections;
+ const onSelect = (data: any) => {
+ if (data) {
+ store.setSelectedData(scope, data);
+ withConnections && store.editConn(withConnections, data[0]);
+ }
+ };
- return (
-
- );
+ return (
+
+ );
+ }
});
export const graphRenderers = [
{ tester: graphRendererTester, renderer: GraphRenderer },
{ tester: testTester, renderer: Test },
];
+
+function findElementsRecursive(viewDescr, array: any[], condition: any): any[] {
+ const elements = array.map((a) => {
+ const params = processViewKindOverride({ viewKindElement: a, viewDescr }, undefined);
+ return [...(condition(params) ? [a] : []), ...findElementsRecursive(viewDescr, a.elements || [], condition).flat()];
+ });
+ return elements.flat();
+}
diff --git a/stories/GraphRenderer.stories.tsx b/stories/GraphRenderer.stories.tsx
index 345f798..cc77e99 100644
--- a/stories/GraphRenderer.stories.tsx
+++ b/stories/GraphRenderer.stories.tsx
@@ -20,13 +20,302 @@ import {
import { MstDiagramNodeVKElement, MstDiagramEdgeVKElement } from '../src/stores/MstDiagramEditorSchemas';
import { graphRenderers } from '../src/components/graphRenderer';
+import '@antv/x6/dist/x6.css';
import '../src/index.css';
import '../src/App.css';
/**
- * Mktp Cards ViewKinds
+ * Mktp Graph + SplitPane ViewKinds
*/
-const mktpViewKinds = [
+const mktpGraphViewKinds = [
+ {
+ '@id': 'mktp:_8g34sKh',
+ '@type': 'aldkg:ViewKind',
+ collsConstrs: [
+ // Categories
+ {
+ '@id': 'mktp:_kwe56Hgs',
+ '@type': 'aldkg:CollConstr',
+ entConstrs: [
+ {
+ '@id': 'mktp:_aS57dj',
+ '@type': 'aldkg:EntConstr',
+ schema: 'aldkg:UsedInDiagramAsRootNodeShape',
+ conditions: {
+ '@id': 'mktp:_Sdf72d',
+ '@type': 'aldkg:EntConstrCondition',
+ subject: '?eIri1',
+ },
+ },
+ {
+ '@id': 'mktp:_3Kjd6sF',
+ '@type': 'aldkg:EntConstr',
+ schema: 'hs:CategoryShape',
+ },
+ ],
+ },
+ // Products
+ {
+ '@id': 'mktp:_58DfdH',
+ '@type': 'aldkg:CollConstr',
+ entConstrs: [
+ {
+ '@id': 'mktp:_oG67s',
+ '@type': 'aldkg:EntConstr',
+ schema: 'aldkg:UsedInDiagramAsRootNodeShape',
+ conditions: {
+ '@id': 'mktp:_64G7Fd',
+ '@type': 'aldkg:EntConstrCondition',
+ subject: '?eIri1',
+ },
+ },
+ {
+ '@id': 'mktp:_8Fd5S',
+ '@type': 'aldkg:EntConstr',
+ schema: 'mktp:ProductShape',
+ },
+ ],
+ },
+ // ProductCards
+ {
+ '@id': 'mktp:_lf68D7',
+ '@type': 'aldkg:CollConstr',
+ entConstrs: [
+ {
+ '@id': 'mktp:_q8H6d',
+ '@type': 'aldkg:EntConstr',
+ schema: 'aldkg:UsedInDiagramAsRootNodeShape',
+ conditions: {
+ '@id': 'mktp:_90Hgs6',
+ '@type': 'aldkg:EntConstrCondition',
+ subject: '?eIri1',
+ },
+ },
+ {
+ '@id': 'mktp:_4hg5Df',
+ '@type': 'aldkg:EntConstr',
+ schema: 'hs:ProductCardShape',
+ },
+ ],
+ },
+ // SubcatInCatLink
+ {
+ '@id': 'mktp:_js5Jdf',
+ '@type': 'aldkg:CollConstr',
+ entConstrs: [
+ {
+ '@id': 'mktp:_Sdf73k',
+ '@type': 'aldkg:EntConstr',
+ schema: 'aldkg:UsedInDiagramAsArrowShape',
+ conditions: {
+ '@id': 'mktp:_9kJgd8',
+ '@type': 'aldkg:EntConstrCondition',
+ subject: '?eIri1',
+ },
+ },
+ {
+ '@id': 'mktp:_p9Dsk6',
+ '@type': 'aldkg:CollConstr',
+ schema: 'hs:SubcatInCatLinkShape',
+ },
+ ],
+ },
+ // CardInCatLink
+ {
+ '@id': 'mktp:_fS67d',
+ '@type': 'aldkg:CollConstr',
+ entConstrs: [
+ {
+ '@id': 'mktp:_kf578s',
+ '@type': 'aldkg:EntConstr',
+ schema: 'aldkg:UsedInDiagramAsArrowShape',
+ conditions: {
+ '@id': 'mktp:_9kJgd8',
+ '@type': 'aldkg:EntConstrCondition',
+ subject: '?eIri1',
+ },
+ },
+ {
+ '@id': 'mktp:_o6sD6f',
+ '@type': 'aldkg:CollConstr',
+ schema: 'hs:CardInCatLinkShape',
+ },
+ ],
+ },
+ // CardInProdLink
+ {
+ '@id': 'mktp:_ld98Sdg',
+ '@type': 'aldkg:CollConstr',
+ entConstrs: [
+ {
+ '@id': 'mktp:_5Sd7fG',
+ '@type': 'aldkg:EntConstr',
+ schema: 'aldkg:UsedInDiagramAsArrowShape',
+ conditions: {
+ '@id': 'mktp:_67Aqw6D',
+ '@type': 'aldkg:EntConstrCondition',
+ subject: '?eIri1',
+ },
+ },
+ {
+ '@id': 'mktp:_n90D6sf',
+ '@type': 'aldkg:CollConstr',
+ schema: 'mktp:CardInProdLinkShape',
+ },
+ ],
+ },
+ ],
+ elements: [
+ {
+ '@id': 'mktp:_45hfg93',
+ '@type': 'aldkg:DiagramEditorVKElement',
+ elements: [
+ /**
+ * Nodes
+ */
+ {
+ '@id': 'mktp:CategoryStencil', // stencil should be registered under this @id
+ '@type': 'aldkg:DiagramNodeVKElement',
+ protoStencil: 'aldkg:CardNode', //reference to the base stencil which should be customized additionally with 'style' and registered under the different id from @id property
+ resultsScope: 'mktp:_kwe56Hgs',
+ // img, title, description are the fields from Cart stencil
+ img: {
+ fallback:
+ '',
+ scope: 'subject/imageUrl',
+ },
+ title: {
+ default: 'Категория',
+ scope: 'subject/name',
+ },
+ description: {
+ scope: 'subject/description',
+ },
+ // style for the root DIV for the Cart stencil
+ style: {
+ borderRight: '4px solid #582dcf',
+ boxShadow: '0 0 4px 0 rgba(0, 0, 0, 0.4)',
+ // child embedding should be disabled here
+ },
+ paletteOrder: 0, // sorting order for stencils palette
+ },
+ {
+ '@id': 'mktp:ProductStencil',
+ '@type': 'aldkg:DiagramNodeVKElement',
+ protoStencil: 'aldkg:CardNode',
+ resultsScope: 'mktp:_58DfdH',
+ img: {
+ fallback:
+ '',
+ scope: 'subject/imageUrl',
+ },
+ title: {
+ default: 'Продукт',
+ scope: 'subject/title',
+ },
+ description: {
+ scope: 'subject/description',
+ },
+ style: {
+ borderRight: '4px solid #832dcf',
+ boxShadow: '0 0 4px 0 rgba(0, 0, 0, 0.4)',
+ // child embedding should be disabled here
+ },
+ paletteOrder: 1,
+ },
+ {
+ '@id': 'mktp:ProductCardStencil',
+ '@type': 'aldkg:DiagramNodeVKElement',
+ protoStencil: 'aldkg:CardNode',
+ resultsScope: 'mktp:_lf68D7',
+ img: {
+ fallback:
+ '',
+ scope: 'subject/imageUrl',
+ },
+ title: {
+ default: 'ProductCard',
+ scope: 'subject/name',
+ },
+ description: {
+ scope: 'subject/description',
+ },
+ style: {
+ borderRight: '4px solid #b42dcf',
+ boxShadow: '0 0 4px 0 rgba(0, 0, 0, 0.4)',
+ // child embedding should be disabled here
+ },
+ paletteOrder: 2,
+ },
+ /**
+ * Edges (arrows)
+ */
+ {
+ '@id': 'mktp:SubcategoryArrowStencil',
+ '@type': 'aldkg:DiagramEdgeVKElement',
+ protoStencil: 'aldkg:CardNode',
+ resultsScope: 'mktp:_js5Jdf',
+ title: 'подкатегория',
+ description: 'Подкатегория в категории',
+ paletteOrder: 4,
+ // styles for targetMarker
+ line: {
+ stroke: '#808080',
+ strokeWidth: 1,
+ targetMarker: {
+ name: 'block',
+ strokeWidth: 1,
+ fill: 'white',
+ },
+ },
+ },
+ {
+ '@id': 'mktp:CardToCategoryArrowStencil',
+ '@type': 'aldkg:DiagramEdgeVKElement',
+ protoStencil: 'aldkg:CardNode',
+ resultsScope: 'mktp:_fS67d',
+ title: 'в категории',
+ description: 'Карточка товара состоит в категории',
+ paletteOrder: 3,
+ line: {
+ stroke: '#808080',
+ strokeWidth: 1,
+ targetMarker: {
+ name: 'block',
+ strokeWidth: 1,
+ open: true,
+ },
+ },
+ },
+ {
+ '@id': 'mktp:CardToProductArrowStencil',
+ '@type': 'aldkg:DiagramEdgeVKElement',
+ protoStencil: 'aldkg:CardNode',
+ resultsScope: 'mktp:_ld98Sdg',
+ title: 'похожесть',
+ description: 'Карточка похожего товара объединена по сходству в один товар',
+ paletteOrder: 4,
+ // styles for targetMarker
+ line: {
+ stroke: '#808080',
+ strokeWidth: 1,
+ targetMarker: {
+ name: 'block',
+ strokeWidth: 1,
+ fill: '#808080',
+ },
+ },
+ },
+ ],
+ },
+ ],
+ },
+];
+
+/**
+ * Mktp Graph + SplitPane ViewKinds
+ */
+const mktpGraphSplitPaneViewKinds = [
{
'@id': 'mktp:_8g34sKh',
'@type': 'aldkg:ViewKind',
@@ -184,6 +473,10 @@ const mktpViewKinds = [
'@id': 'mktp:_934Jfg7',
'@type': 'aldkg:SplitPaneLayout',
options: {
+ style: {
+ width: '100%',
+ height: '100%',
+ },
defaultSize: {
'mktp:_45hfg93': '70%',
'mktp:_45hfgll': '30%',
@@ -533,11 +826,35 @@ const mktpViewDescrs = [
/**
* Collections Configs Data
*/
-const additionalColls: CollState[] = [
+const additionalGraphColls: CollState[] = [
// ViewKinds Collection
{
constr: viewKindCollConstr,
- data: mktpViewKinds,
+ data: mktpGraphViewKinds,
+ opt: {
+ updPeriod: undefined,
+ lastSynced: moment.now(),
+ //resolveCollConstrs: false, // disable data loading from the server for viewKinds.collConstrs
+ },
+ },
+ // ViewDescrs Collection
+ {
+ constr: viewDescrCollConstr,
+ data: mktpViewDescrs,
+ opt: {
+ updPeriod: undefined,
+ lastSynced: moment.now(),
+ //resolveCollConstrs: false, // 'true' here (by default) triggers data loading from the server
+ // for viewDescrs.collConstrs (it loads lazily -- after the first access)
+ },
+ },
+];
+
+const additionalGraphSplitPaneColls: CollState[] = [
+ // ViewKinds Collection
+ {
+ constr: viewKindCollConstr,
+ data: mktpGraphSplitPaneViewKinds,
opt: {
updPeriod: undefined,
lastSynced: moment.now(),
@@ -562,7 +879,7 @@ export default {
component: Form,
} as Meta;
-const Template: Story
= (args: any) => {
+const Template: Story = ({ viewDescrCollId, viewDescrId, additionalColls }) => {
const renderers = [...antdLayoutRenderers, ...graphRenderers, ...antdControlRenderers];
registerMstViewKindSchema(MstDiagramNodeVKElement);
registerMstViewKindSchema(MstDiagramEdgeVKElement);
@@ -575,18 +892,26 @@ const Template: Story = (args: any) => {
// eslint-disable-next-line @typescript-eslint/no-var-requires
connectReduxDevtools(require('remotedev'), rootStore);
return (
-
-
-
-
-
-
-
+
);
};
-export const RemoteData = Template.bind({});
-RemoteData.args = {
+export const RemoteDataGraph = Template.bind({});
+RemoteDataGraph.args = {
+ viewDescrCollId: viewDescrCollConstr['@id'],
+ viewDescrId: mktpViewDescrs[0]['@id'],
+ additionalColls: additionalGraphColls,
+};
+
+export const RemoteDataGraphSplitPane = Template.bind({});
+RemoteDataGraphSplitPane.args = {
viewDescrCollId: viewDescrCollConstr['@id'],
viewDescrId: mktpViewDescrs[0]['@id'],
+ additionalColls: additionalGraphSplitPaneColls,
};