@likec4/language-server 1.43.0 → 1.45.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/browser/package.json +1 -1
- package/browser-worker/package.json +1 -1
- package/dist/LikeC4LanguageServices.d.ts +13 -34
- package/dist/LikeC4LanguageServices.js +152 -0
- package/dist/Rpc.js +257 -0
- package/dist/ast.d.ts +5 -1
- package/dist/ast.js +257 -0
- package/dist/browser-worker.js +4 -0
- package/dist/browser.js +35 -0
- package/dist/bundled.js +44 -0
- package/dist/bundled.mjs +3568 -3900
- package/dist/documentation/documentation-provider.js +51 -0
- package/dist/documentation/index.js +1 -0
- package/dist/empty.js +2 -0
- package/dist/filesystem/ChokidarWatcher.d.ts +2 -0
- package/dist/filesystem/ChokidarWatcher.js +108 -0
- package/dist/filesystem/FileSystemWatcher.js +14 -0
- package/dist/filesystem/LikeC4FileSystem.d.ts +1 -2
- package/dist/filesystem/LikeC4FileSystem.js +126 -0
- package/dist/filesystem/index.d.ts +26 -0
- package/dist/filesystem/index.js +29 -0
- package/dist/formatting/LikeC4Formatter.js +637 -0
- package/dist/formatting/utils.js +18 -0
- package/dist/generated/ast.js +2155 -0
- package/dist/generated/{grammar.mjs → grammar.js} +6 -2
- package/dist/generated/module.d.ts +6 -1
- package/dist/generated/module.js +27 -0
- package/dist/generated-lib/{icons.mjs → icons.js} +11 -7
- package/dist/index.d.ts +10 -1
- package/dist/index.js +55 -0
- package/dist/{likec4lib.mjs → likec4lib.js} +3 -3
- package/dist/logger.js +81 -0
- package/dist/lsp/CodeActionProvider.d.ts +14 -0
- package/dist/lsp/CodeActionProvider.js +33 -0
- package/dist/lsp/CodeLensProvider.js +44 -0
- package/dist/lsp/CompletionProvider.d.ts +3 -1
- package/dist/lsp/CompletionProvider.js +200 -0
- package/dist/lsp/DocumentHighlightProvider.js +10 -0
- package/dist/lsp/DocumentLinkProvider.js +58 -0
- package/dist/lsp/DocumentSymbolProvider.js +306 -0
- package/dist/lsp/HoverProvider.js +106 -0
- package/dist/lsp/RenameProvider.js +6 -0
- package/dist/lsp/SemanticTokenProvider.js +257 -0
- package/dist/lsp/index.d.ts +1 -0
- package/dist/lsp/index.js +9 -0
- package/dist/mcp/MCPServerFactory.js +73 -0
- package/dist/mcp/NoopLikeC4MCPServer.js +17 -0
- package/dist/mcp/interfaces.js +5 -0
- package/dist/mcp/server/StdioLikeC4MCPServer.js +51 -0
- package/dist/mcp/server/StreamableLikeC4MCPServer.js +145 -0
- package/dist/mcp/server/WithMCPServer.js +56 -0
- package/dist/mcp/tools/_common.d.ts +8 -7
- package/dist/mcp/tools/_common.js +49 -0
- package/dist/mcp/tools/find-relationships.d.ts +7 -8
- package/dist/mcp/tools/find-relationships.js +150 -0
- package/dist/mcp/tools/list-projects.d.ts +3 -3
- package/dist/mcp/tools/list-projects.js +62 -0
- package/dist/mcp/tools/open-view.d.ts +6 -7
- package/dist/mcp/tools/open-view.js +52 -0
- package/dist/mcp/tools/read-deployment.d.ts +6 -7
- package/dist/mcp/tools/read-deployment.js +132 -0
- package/dist/mcp/tools/read-element.d.ts +6 -7
- package/dist/mcp/tools/read-element.js +194 -0
- package/dist/mcp/tools/read-project-summary.d.ts +5 -6
- package/dist/mcp/tools/read-project-summary.js +176 -0
- package/dist/mcp/tools/read-view.d.ts +6 -7
- package/dist/mcp/tools/read-view.js +203 -0
- package/dist/mcp/tools/search-element.d.ts +3 -3
- package/dist/mcp/tools/search-element.js +177 -0
- package/dist/mcp/utils.d.ts +2 -2
- package/dist/mcp/utils.js +48 -0
- package/dist/model/builder/MergedExtends.js +74 -0
- package/dist/model/builder/MergedSpecification.js +175 -0
- package/dist/model/builder/buildModel.js +176 -0
- package/dist/model/deployments-index.js +102 -0
- package/dist/model/fqn-index.d.ts +1 -2
- package/dist/model/fqn-index.js +247 -0
- package/dist/model/index.js +6 -0
- package/dist/model/model-builder.d.ts +13 -11
- package/dist/model/model-builder.js +232 -0
- package/dist/model/model-locator.d.ts +6 -5
- package/dist/model/model-locator.js +240 -0
- package/dist/model/model-parser-where.js +81 -0
- package/dist/model/model-parser.d.ts +309 -304
- package/dist/model/model-parser.js +126 -0
- package/dist/model/parser/Base.d.ts +2 -2
- package/dist/model/parser/Base.js +367 -0
- package/dist/model/parser/DeploymentModelParser.d.ts +2 -2
- package/dist/model/parser/DeploymentModelParser.js +176 -0
- package/dist/model/parser/DeploymentViewParser.d.ts +3 -3
- package/dist/model/parser/DeploymentViewParser.js +86 -0
- package/dist/model/parser/FqnRefParser.d.ts +2 -2
- package/dist/model/parser/FqnRefParser.js +382 -0
- package/dist/model/parser/GlobalsParser.d.ts +6 -6
- package/dist/model/parser/GlobalsParser.js +84 -0
- package/dist/model/parser/ImportsParser.d.ts +11 -12
- package/dist/model/parser/ImportsParser.js +24 -0
- package/dist/model/parser/ModelParser.d.ts +2 -2
- package/dist/model/parser/ModelParser.js +165 -0
- package/dist/model/parser/PredicatesParser.d.ts +2 -2
- package/dist/model/parser/PredicatesParser.js +45 -0
- package/dist/model/parser/SpecificationParser.d.ts +2 -2
- package/dist/model/parser/SpecificationParser.js +113 -0
- package/dist/model/parser/ValueConverter.js +12 -0
- package/dist/model/parser/ViewsParser.d.ts +3 -3
- package/dist/model/parser/ViewsParser.js +479 -0
- package/dist/model-change/ModelChanges.d.ts +8 -5
- package/dist/model-change/ModelChanges.js +129 -0
- package/dist/model-change/changeElementStyle.js +134 -0
- package/dist/model-change/changeViewLayout.d.ts +2 -2
- package/dist/model-change/changeViewLayout.js +28 -0
- package/dist/model-change/removeManualLayoutV1.d.ts +7 -0
- package/dist/model-change/removeManualLayoutV1.js +27 -0
- package/dist/module.d.ts +10 -5
- package/dist/module.js +143 -0
- package/dist/protocol.d.ts +34 -27
- package/dist/protocol.js +123 -0
- package/dist/references/index.js +3 -0
- package/dist/references/name-provider.js +37 -0
- package/dist/references/scope-computation.js +288 -0
- package/dist/references/scope-provider.d.ts +3 -3
- package/dist/references/scope-provider.js +242 -0
- package/dist/shared/NodeKindProvider.js +57 -0
- package/dist/shared/{WorkspaceSymbolProvider.mjs → WorkspaceSymbolProvider.js} +1 -1
- package/dist/shared/index.js +2 -0
- package/dist/test/index.js +1 -0
- package/dist/test/testServices.d.ts +16 -16
- package/dist/test/testServices.js +210 -0
- package/dist/utils/disposable.js +26 -0
- package/dist/utils/elementRef.d.ts +1 -1
- package/dist/utils/elementRef.js +27 -0
- package/dist/utils/fqnRef.js +63 -0
- package/dist/utils/index.js +35 -0
- package/dist/utils/printDocs.js +1 -0
- package/dist/utils/projectId.js +16 -0
- package/dist/utils/stringHash.js +5 -0
- package/dist/validation/DocumentValidator.js +17 -0
- package/dist/validation/_shared.js +26 -0
- package/dist/validation/deployment-checks.js +140 -0
- package/dist/validation/dynamic-view.js +67 -0
- package/dist/validation/element-ref.js +12 -0
- package/dist/validation/element.js +49 -0
- package/dist/validation/imports.js +46 -0
- package/dist/validation/index.d.ts +1 -1
- package/dist/validation/index.js +157 -0
- package/dist/validation/property-checks.js +108 -0
- package/dist/validation/relation.js +55 -0
- package/dist/validation/specification.js +190 -0
- package/dist/validation/view-predicates/fqn-expr-with.js +43 -0
- package/dist/validation/view-predicates/fqn-ref-expr.js +51 -0
- package/dist/validation/view-predicates/incoming.js +16 -0
- package/dist/validation/view-predicates/index.js +6 -0
- package/dist/validation/view-predicates/outgoing.js +20 -0
- package/dist/validation/view-predicates/relation-expr.js +46 -0
- package/dist/validation/view-predicates/relation-with.js +16 -0
- package/dist/validation/view.d.ts +1 -1
- package/dist/validation/view.js +42 -0
- package/dist/view-utils/assignNavigateTo.js +27 -0
- package/dist/view-utils/index.d.ts +1 -0
- package/dist/view-utils/index.js +2 -0
- package/dist/view-utils/manual-layout.d.ts +6 -0
- package/dist/view-utils/manual-layout.js +149 -0
- package/dist/views/ConfigurableLayouter.js +51 -0
- package/dist/views/LikeC4ManualLayouts.d.ts +42 -0
- package/dist/views/LikeC4ManualLayouts.js +209 -0
- package/dist/views/{likec4-views.d.ts → LikeC4Views.d.ts} +32 -10
- package/dist/views/LikeC4Views.js +216 -0
- package/dist/views/index.d.ts +4 -1
- package/dist/views/index.js +11 -0
- package/dist/workspace/AstNodeDescriptionProvider.js +18 -0
- package/dist/workspace/IndexManager.js +21 -0
- package/dist/workspace/LangiumDocuments.d.ts +4 -3
- package/dist/workspace/LangiumDocuments.js +72 -0
- package/dist/workspace/ProjectsManager.d.ts +25 -16
- package/dist/workspace/ProjectsManager.js +469 -0
- package/dist/workspace/WorkspaceManager.d.ts +3 -2
- package/dist/workspace/WorkspaceManager.js +98 -0
- package/dist/workspace/index.js +5 -0
- package/likec4lib/package.json +1 -1
- package/package.json +30 -28
- package/protocol/package.json +1 -1
- package/dist/LikeC4LanguageServices.mjs +0 -197
- package/dist/Rpc.mjs +0 -296
- package/dist/ast.mjs +0 -221
- package/dist/browser-worker.mjs +0 -2
- package/dist/browser.mjs +0 -32
- package/dist/documentation/documentation-provider.mjs +0 -48
- package/dist/documentation/index.mjs +0 -1
- package/dist/empty.mjs +0 -1
- package/dist/filesystem/ChokidarWatcher.mjs +0 -68
- package/dist/filesystem/FileSystemWatcher.mjs +0 -11
- package/dist/filesystem/LikeC4FileSystem.mjs +0 -64
- package/dist/filesystem/index.mjs +0 -19
- package/dist/formatting/LikeC4Formatter.mjs +0 -511
- package/dist/formatting/utils.mjs +0 -15
- package/dist/generated/ast.mjs +0 -2150
- package/dist/generated/module.mjs +0 -23
- package/dist/index.mjs +0 -50
- package/dist/logger.mjs +0 -82
- package/dist/lsp/CodeLensProvider.mjs +0 -42
- package/dist/lsp/CompletionProvider.mjs +0 -208
- package/dist/lsp/DocumentHighlightProvider.mjs +0 -10
- package/dist/lsp/DocumentLinkProvider.mjs +0 -53
- package/dist/lsp/DocumentSymbolProvider.mjs +0 -287
- package/dist/lsp/HoverProvider.mjs +0 -104
- package/dist/lsp/RenameProvider.mjs +0 -6
- package/dist/lsp/SemanticTokenProvider.mjs +0 -276
- package/dist/lsp/index.mjs +0 -7
- package/dist/mcp/MCPServerFactory.mjs +0 -70
- package/dist/mcp/NoopLikeC4MCPServer.mjs +0 -17
- package/dist/mcp/interfaces.mjs +0 -4
- package/dist/mcp/server/StdioLikeC4MCPServer.mjs +0 -46
- package/dist/mcp/server/StreamableLikeC4MCPServer.mjs +0 -153
- package/dist/mcp/server/WithMCPServer.mjs +0 -58
- package/dist/mcp/tools/_common.mjs +0 -42
- package/dist/mcp/tools/find-relationships.mjs +0 -151
- package/dist/mcp/tools/list-projects.mjs +0 -62
- package/dist/mcp/tools/open-view.mjs +0 -52
- package/dist/mcp/tools/read-deployment.mjs +0 -130
- package/dist/mcp/tools/read-element.mjs +0 -198
- package/dist/mcp/tools/read-project-summary.mjs +0 -178
- package/dist/mcp/tools/read-view.mjs +0 -205
- package/dist/mcp/tools/search-element.mjs +0 -171
- package/dist/mcp/utils.mjs +0 -47
- package/dist/model/builder/MergedExtends.mjs +0 -76
- package/dist/model/builder/MergedSpecification.mjs +0 -205
- package/dist/model/builder/assignTagColors.d.ts +0 -7
- package/dist/model/builder/assignTagColors.mjs +0 -51
- package/dist/model/builder/buildModel.mjs +0 -226
- package/dist/model/deployments-index.mjs +0 -100
- package/dist/model/fqn-index.mjs +0 -243
- package/dist/model/index.mjs +0 -6
- package/dist/model/model-builder.mjs +0 -285
- package/dist/model/model-locator.mjs +0 -239
- package/dist/model/model-parser-where.mjs +0 -81
- package/dist/model/model-parser.mjs +0 -127
- package/dist/model/parser/Base.mjs +0 -376
- package/dist/model/parser/DeploymentModelParser.mjs +0 -212
- package/dist/model/parser/DeploymentViewParser.mjs +0 -95
- package/dist/model/parser/FqnRefParser.mjs +0 -398
- package/dist/model/parser/GlobalsParser.mjs +0 -82
- package/dist/model/parser/ImportsParser.mjs +0 -28
- package/dist/model/parser/ModelParser.mjs +0 -190
- package/dist/model/parser/PredicatesParser.mjs +0 -45
- package/dist/model/parser/SpecificationParser.mjs +0 -120
- package/dist/model/parser/ValueConverter.mjs +0 -12
- package/dist/model/parser/ViewsParser.mjs +0 -490
- package/dist/model-change/ModelChanges.mjs +0 -89
- package/dist/model-change/changeElementStyle.mjs +0 -143
- package/dist/model-change/changeViewLayout.mjs +0 -32
- package/dist/model-change/saveManualLayout.d.ts +0 -11
- package/dist/model-change/saveManualLayout.mjs +0 -27
- package/dist/module.mjs +0 -180
- package/dist/protocol.mjs +0 -65
- package/dist/references/index.mjs +0 -3
- package/dist/references/name-provider.mjs +0 -39
- package/dist/references/scope-computation.mjs +0 -312
- package/dist/references/scope-provider.mjs +0 -239
- package/dist/shared/NodeKindProvider.mjs +0 -110
- package/dist/shared/index.mjs +0 -2
- package/dist/test/index.mjs +0 -1
- package/dist/test/testServices.mjs +0 -200
- package/dist/utils/disposable.mjs +0 -25
- package/dist/utils/elementRef.mjs +0 -20
- package/dist/utils/fqnRef.mjs +0 -57
- package/dist/utils/index.mjs +0 -33
- package/dist/utils/printDocs.mjs +0 -1
- package/dist/utils/projectId.mjs +0 -16
- package/dist/utils/stringHash.mjs +0 -5
- package/dist/validation/DocumentValidator.mjs +0 -16
- package/dist/validation/_shared.mjs +0 -25
- package/dist/validation/deployment-checks.mjs +0 -146
- package/dist/validation/dynamic-view.mjs +0 -67
- package/dist/validation/element-ref.mjs +0 -12
- package/dist/validation/element.mjs +0 -50
- package/dist/validation/imports.mjs +0 -25
- package/dist/validation/index.mjs +0 -180
- package/dist/validation/property-checks.mjs +0 -107
- package/dist/validation/relation.mjs +0 -53
- package/dist/validation/specification.mjs +0 -173
- package/dist/validation/view-predicates/fqn-expr-with.mjs +0 -43
- package/dist/validation/view-predicates/fqn-ref-expr.mjs +0 -53
- package/dist/validation/view-predicates/incoming.mjs +0 -16
- package/dist/validation/view-predicates/index.mjs +0 -6
- package/dist/validation/view-predicates/outgoing.mjs +0 -20
- package/dist/validation/view-predicates/relation-expr.mjs +0 -39
- package/dist/validation/view-predicates/relation-with.mjs +0 -16
- package/dist/validation/view.mjs +0 -25
- package/dist/view-utils/assignNavigateTo.mjs +0 -25
- package/dist/view-utils/index.mjs +0 -1
- package/dist/view-utils/manual-layout.mjs +0 -99
- package/dist/views/configurable-layouter.mjs +0 -51
- package/dist/views/index.mjs +0 -1
- package/dist/views/likec4-views.mjs +0 -166
- package/dist/workspace/AstNodeDescriptionProvider.mjs +0 -17
- package/dist/workspace/IndexManager.mjs +0 -17
- package/dist/workspace/LangiumDocuments.mjs +0 -53
- package/dist/workspace/ProjectsManager.mjs +0 -360
- package/dist/workspace/WorkspaceManager.mjs +0 -83
- package/dist/workspace/index.mjs +0 -5
- /package/dist/views/{configurable-layouter.d.ts → ConfigurableLayouter.d.ts} +0 -0
|
@@ -0,0 +1,190 @@
|
|
|
1
|
+
import { AstUtils } from 'langium';
|
|
2
|
+
import { ast } from '../ast';
|
|
3
|
+
import { projectIdFrom } from '../utils';
|
|
4
|
+
import { RESERVED_WORDS, tryOrLog } from './_shared';
|
|
5
|
+
export const checkSpecificationRule = (_) => {
|
|
6
|
+
return tryOrLog((node, accept) => {
|
|
7
|
+
if (node.$containerIndex && node.$containerIndex > 0) {
|
|
8
|
+
accept('warning', `Prefer one specification per document`, {
|
|
9
|
+
node: node,
|
|
10
|
+
property: 'name',
|
|
11
|
+
});
|
|
12
|
+
}
|
|
13
|
+
});
|
|
14
|
+
};
|
|
15
|
+
export const checkModel = (_) => {
|
|
16
|
+
return tryOrLog((node, accept) => {
|
|
17
|
+
if (node.$containerIndex && node.$containerIndex > 0) {
|
|
18
|
+
accept('warning', `Prefer one model per document`, {
|
|
19
|
+
node: node,
|
|
20
|
+
property: 'name',
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
};
|
|
25
|
+
export const checkGlobals = (_) => {
|
|
26
|
+
return tryOrLog((node, accept) => {
|
|
27
|
+
if (node.$containerIndex && node.$containerIndex > 0) {
|
|
28
|
+
accept('warning', `Prefer one global block per document`, {
|
|
29
|
+
node: node,
|
|
30
|
+
property: 'name',
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
});
|
|
34
|
+
};
|
|
35
|
+
export const checkElementKind = (services) => {
|
|
36
|
+
const index = services.shared.workspace.IndexManager;
|
|
37
|
+
return tryOrLog((node, accept) => {
|
|
38
|
+
if (RESERVED_WORDS.includes(node.name)) {
|
|
39
|
+
accept('error', `Reserved word: ${node.name}`, {
|
|
40
|
+
node: node,
|
|
41
|
+
property: 'name',
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
const projectId = projectIdFrom(node);
|
|
45
|
+
const sameKind = index
|
|
46
|
+
.projectElements(projectId, ast.ElementKind)
|
|
47
|
+
.filter(n => n.name === node.name && n.node !== node)
|
|
48
|
+
.head();
|
|
49
|
+
if (sameKind) {
|
|
50
|
+
const isAnotherDoc = sameKind.documentUri !== AstUtils.getDocument(node).uri;
|
|
51
|
+
accept('error', `Duplicate element kind '${node.name}'`, {
|
|
52
|
+
node: node,
|
|
53
|
+
property: 'name',
|
|
54
|
+
...isAnotherDoc && {
|
|
55
|
+
relatedInformation: [
|
|
56
|
+
{
|
|
57
|
+
location: {
|
|
58
|
+
range: sameKind.nameSegment.range,
|
|
59
|
+
uri: sameKind.documentUri.toString(),
|
|
60
|
+
},
|
|
61
|
+
message: `conflicting definition`,
|
|
62
|
+
},
|
|
63
|
+
],
|
|
64
|
+
},
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
});
|
|
68
|
+
};
|
|
69
|
+
export const checkDeploymentNodeKind = (services) => {
|
|
70
|
+
const index = services.shared.workspace.IndexManager;
|
|
71
|
+
return tryOrLog((node, accept) => {
|
|
72
|
+
if (RESERVED_WORDS.includes(node.name)) {
|
|
73
|
+
accept('error', `Reserved word: ${node.name}`, {
|
|
74
|
+
node: node,
|
|
75
|
+
property: 'name',
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
const projectId = projectIdFrom(node);
|
|
79
|
+
const sameKind = index
|
|
80
|
+
.projectElements(projectId, ast.DeploymentNodeKind)
|
|
81
|
+
.filter(n => n.name === node.name && n.node !== node)
|
|
82
|
+
.head();
|
|
83
|
+
if (sameKind) {
|
|
84
|
+
const isAnotherDoc = sameKind.documentUri !== AstUtils.getDocument(node).uri;
|
|
85
|
+
accept('error', `Duplicate deploymentNode kind '${node.name}'`, {
|
|
86
|
+
node: node,
|
|
87
|
+
property: 'name',
|
|
88
|
+
...isAnotherDoc && {
|
|
89
|
+
relatedInformation: [
|
|
90
|
+
{
|
|
91
|
+
location: {
|
|
92
|
+
range: sameKind.nameSegment.range,
|
|
93
|
+
uri: sameKind.documentUri.toString(),
|
|
94
|
+
},
|
|
95
|
+
message: `conflicting definition`,
|
|
96
|
+
},
|
|
97
|
+
],
|
|
98
|
+
},
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
});
|
|
102
|
+
};
|
|
103
|
+
export const checkTag = (services) => {
|
|
104
|
+
const index = services.shared.workspace.IndexManager;
|
|
105
|
+
return tryOrLog((node, accept) => {
|
|
106
|
+
const tagname = node.name;
|
|
107
|
+
const projectId = projectIdFrom(node);
|
|
108
|
+
const sameTag = index
|
|
109
|
+
.projectElements(projectId, ast.Tag)
|
|
110
|
+
.filter(n => n.name === tagname && n.node !== node)
|
|
111
|
+
.head();
|
|
112
|
+
if (sameTag) {
|
|
113
|
+
const isAnotherDoc = sameTag.documentUri !== AstUtils.getDocument(node).uri;
|
|
114
|
+
accept('error', `Duplicate tag '${node.name}'`, {
|
|
115
|
+
node,
|
|
116
|
+
property: 'name',
|
|
117
|
+
...isAnotherDoc && {
|
|
118
|
+
relatedInformation: [
|
|
119
|
+
{
|
|
120
|
+
location: {
|
|
121
|
+
range: sameTag.nameSegment.range,
|
|
122
|
+
uri: sameTag.documentUri.toString(),
|
|
123
|
+
},
|
|
124
|
+
message: `conflicting definition`,
|
|
125
|
+
},
|
|
126
|
+
],
|
|
127
|
+
},
|
|
128
|
+
});
|
|
129
|
+
}
|
|
130
|
+
});
|
|
131
|
+
};
|
|
132
|
+
export const checkRelationshipKind = (services) => {
|
|
133
|
+
const index = services.shared.workspace.IndexManager;
|
|
134
|
+
return tryOrLog((node, accept) => {
|
|
135
|
+
if (RESERVED_WORDS.includes(node.name)) {
|
|
136
|
+
accept('error', `Reserved word: ${node.name}`, {
|
|
137
|
+
node: node,
|
|
138
|
+
property: 'name',
|
|
139
|
+
});
|
|
140
|
+
}
|
|
141
|
+
const projectId = projectIdFrom(node);
|
|
142
|
+
const sameKinds = index
|
|
143
|
+
.projectElements(projectId, ast.RelationshipKind)
|
|
144
|
+
.filter(n => n.name === node.name)
|
|
145
|
+
.limit(2)
|
|
146
|
+
.count();
|
|
147
|
+
if (sameKinds > 1) {
|
|
148
|
+
accept('error', `Duplicate RelationshipKind '${node.name}'`, {
|
|
149
|
+
node: node,
|
|
150
|
+
property: 'name',
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
});
|
|
154
|
+
};
|
|
155
|
+
export const checkGlobalPredicate = (services) => {
|
|
156
|
+
const index = services.shared.workspace.IndexManager;
|
|
157
|
+
return tryOrLog((node, accept) => {
|
|
158
|
+
const projectId = projectIdFrom(node);
|
|
159
|
+
const predicateGroups = index.projectElements(projectId, ast.GlobalPredicateGroup);
|
|
160
|
+
const dynamicPredicateGroups = index.projectElements(projectId, ast.GlobalDynamicPredicateGroup);
|
|
161
|
+
const sameName = predicateGroups
|
|
162
|
+
.concat(dynamicPredicateGroups)
|
|
163
|
+
.filter(s => s.name === node.name)
|
|
164
|
+
.limit(2)
|
|
165
|
+
.count();
|
|
166
|
+
if (sameName > 1) {
|
|
167
|
+
accept('error', `Duplicate GlobalPredicateGroup or GlobalDynamicPredicateGroup name '${node.name}'`, {
|
|
168
|
+
node: node,
|
|
169
|
+
property: 'name',
|
|
170
|
+
});
|
|
171
|
+
}
|
|
172
|
+
});
|
|
173
|
+
};
|
|
174
|
+
export const checkGlobalStyleId = (services) => {
|
|
175
|
+
const index = services.shared.workspace.IndexManager;
|
|
176
|
+
return tryOrLog((node, accept) => {
|
|
177
|
+
const projectId = projectIdFrom(node);
|
|
178
|
+
const sameName = index
|
|
179
|
+
.projectElements(projectId, ast.GlobalStyleId)
|
|
180
|
+
.filter(s => s.name === node.name)
|
|
181
|
+
.limit(2)
|
|
182
|
+
.count();
|
|
183
|
+
if (sameName > 1) {
|
|
184
|
+
accept('error', `Duplicate GlobalStyleId name '${node.name}'`, {
|
|
185
|
+
node: node,
|
|
186
|
+
property: 'name',
|
|
187
|
+
});
|
|
188
|
+
}
|
|
189
|
+
});
|
|
190
|
+
};
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { FqnExpr } from '@likec4/core/types';
|
|
2
|
+
import { nonexhaustive } from '@likec4/core/utils';
|
|
3
|
+
import { AstUtils } from 'langium';
|
|
4
|
+
import { ast, getViewRulePredicateContainer } from '../../ast';
|
|
5
|
+
import { tryOrLog } from '../_shared';
|
|
6
|
+
export const checkFqnExprWith = (services) => {
|
|
7
|
+
const modelParser = services.likec4.ModelParser;
|
|
8
|
+
return tryOrLog((el, accept) => {
|
|
9
|
+
const container = getViewRulePredicateContainer(el);
|
|
10
|
+
if (container?.$type !== 'DynamicViewIncludePredicate' && container?.isInclude !== true) {
|
|
11
|
+
accept('error', 'Invalid usage inside "exclude"', {
|
|
12
|
+
node: el,
|
|
13
|
+
});
|
|
14
|
+
return;
|
|
15
|
+
}
|
|
16
|
+
const isInsideDynamicView = container.$type === 'DynamicViewIncludePredicate';
|
|
17
|
+
const parser = modelParser.forDocument(AstUtils.getDocument(container));
|
|
18
|
+
let expr = FqnExpr.unwrap(parser.parseFqnExprWith(el).custom.expr);
|
|
19
|
+
switch (true) {
|
|
20
|
+
case FqnExpr.isWildcard(expr) && isInsideDynamicView:
|
|
21
|
+
case FqnExpr.isElementKindExpr(expr) && isInsideDynamicView:
|
|
22
|
+
case FqnExpr.isElementTagExpr(expr) && isInsideDynamicView: {
|
|
23
|
+
accept('warning', `Predicate is ignored, as not supported in dynamic views`, {
|
|
24
|
+
node: el,
|
|
25
|
+
});
|
|
26
|
+
return;
|
|
27
|
+
}
|
|
28
|
+
case FqnExpr.isWildcard(expr):
|
|
29
|
+
case FqnExpr.isModelRef(expr):
|
|
30
|
+
case FqnExpr.isDeploymentRef(expr):
|
|
31
|
+
return;
|
|
32
|
+
case FqnExpr.isElementKindExpr(expr):
|
|
33
|
+
case FqnExpr.isElementTagExpr(expr):
|
|
34
|
+
accept('error', 'Invalid target (expect reference to specific element)', {
|
|
35
|
+
node: el,
|
|
36
|
+
property: 'subject',
|
|
37
|
+
});
|
|
38
|
+
return;
|
|
39
|
+
default:
|
|
40
|
+
nonexhaustive(expr);
|
|
41
|
+
}
|
|
42
|
+
});
|
|
43
|
+
};
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { FqnExpr, FqnRef } from '@likec4/core/types';
|
|
2
|
+
import { AstUtils } from 'langium';
|
|
3
|
+
import { isTruthy } from 'remeda';
|
|
4
|
+
import { ast, getViewRulePredicateContainer, isFqnRefInsideDeployment } from '../../ast';
|
|
5
|
+
import { tryOrLog } from '../_shared';
|
|
6
|
+
export const checkFqnRefExpr = (services) => {
|
|
7
|
+
const modelParser = services.likec4.ModelParser;
|
|
8
|
+
return tryOrLog((node, accept) => {
|
|
9
|
+
const parser = modelParser.forDocument(AstUtils.getDocument(node));
|
|
10
|
+
const expr = parser.parseFqnRefExpr(node);
|
|
11
|
+
const viewRulePredicate = getViewRulePredicateContainer(node);
|
|
12
|
+
const isInsideDeploymentButNotStyle = isFqnRefInsideDeployment(node) && !AstUtils.hasContainerOfType(node, n => ast.isDeploymentViewRuleStyle(n) || ast.isViewRuleStyle(n));
|
|
13
|
+
if (viewRulePredicate?.$type === 'DeploymentViewRulePredicate' || isInsideDeploymentButNotStyle) {
|
|
14
|
+
const isPartOfRelationExpr = AstUtils.hasContainerOfType(node, ast.isRelationExpr);
|
|
15
|
+
// This expression is part of element predicate
|
|
16
|
+
if (!isPartOfRelationExpr) {
|
|
17
|
+
if (FqnExpr.isModelRef(expr)) {
|
|
18
|
+
accept('error', 'Deployment view predicate must reference deployment model', {
|
|
19
|
+
node,
|
|
20
|
+
});
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
23
|
+
if (FqnExpr.isDeploymentRef(expr) && FqnRef.isInsideInstanceRef(expr.ref)) {
|
|
24
|
+
accept('error', 'Must reference deployment nodes or instances, but not internals', {
|
|
25
|
+
node,
|
|
26
|
+
});
|
|
27
|
+
return;
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
if (isTruthy(node.selector) && !ast.isDeploymentNode(node.ref.value?.ref)) {
|
|
31
|
+
accept('warning', `Selector '${node.selector}' applies to deployment nodes only, ignored here`, {
|
|
32
|
+
node,
|
|
33
|
+
property: 'selector',
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
if (viewRulePredicate?.$type === 'DynamicViewIncludePredicate') {
|
|
39
|
+
switch (true) {
|
|
40
|
+
case FqnExpr.isElementKindExpr(expr):
|
|
41
|
+
case FqnExpr.isElementTagExpr(expr):
|
|
42
|
+
case FqnExpr.isWildcard(expr): {
|
|
43
|
+
accept('warning', `Predicate is ignored, as not supported in dynamic views`, {
|
|
44
|
+
node,
|
|
45
|
+
});
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
};
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { AstUtils } from 'langium';
|
|
2
|
+
import { isNullish } from 'remeda';
|
|
3
|
+
import { ast } from '../../ast';
|
|
4
|
+
import { tryOrLog } from '../_shared';
|
|
5
|
+
export const checkIncomingRelationExpr = (_services) => {
|
|
6
|
+
return tryOrLog((el, accept) => {
|
|
7
|
+
if (el.to.$type === 'WildcardExpression' && !ast.isInOutRelationExpr(el.$container)) {
|
|
8
|
+
const view = AstUtils.getContainerOfType(el, ast.isElementView);
|
|
9
|
+
if (isNullish(view?.viewOf)) {
|
|
10
|
+
accept('warning', 'Predicate is ignored as it concerns all relationships', {
|
|
11
|
+
node: el,
|
|
12
|
+
});
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
});
|
|
16
|
+
};
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
export { checkFqnExprWith } from './fqn-expr-with';
|
|
2
|
+
export { checkFqnRefExpr } from './fqn-ref-expr';
|
|
3
|
+
export { checkIncomingRelationExpr } from './incoming';
|
|
4
|
+
export { checkOutgoingRelationExpr } from './outgoing';
|
|
5
|
+
export { checkRelationExpr } from './relation-expr';
|
|
6
|
+
export { checkRelationExprWith } from './relation-with';
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { AstUtils } from 'langium';
|
|
2
|
+
import { isNullish } from 'remeda';
|
|
3
|
+
import { ast, getViewRulePredicateContainer } from '../../ast';
|
|
4
|
+
import { tryOrLog } from '../_shared';
|
|
5
|
+
export const checkOutgoingRelationExpr = (_services) => {
|
|
6
|
+
return tryOrLog((el, accept) => {
|
|
7
|
+
const viewRulePredicate = getViewRulePredicateContainer(el);
|
|
8
|
+
if (viewRulePredicate?.$type !== 'ViewRulePredicate') {
|
|
9
|
+
return;
|
|
10
|
+
}
|
|
11
|
+
if (el.$container.$type !== 'DirectedRelationExpr' && el.from.$type === 'WildcardExpression') {
|
|
12
|
+
const view = AstUtils.getContainerOfType(el, ast.isElementView);
|
|
13
|
+
if (view && isNullish(view.viewOf)) {
|
|
14
|
+
accept('warning', 'Predicate is ignored as it concerns all relationships', {
|
|
15
|
+
node: el,
|
|
16
|
+
});
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
});
|
|
20
|
+
};
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { FqnExpr, RelationExpr } from '@likec4/core';
|
|
2
|
+
import { AstUtils } from 'langium';
|
|
3
|
+
import { ast } from '../../ast';
|
|
4
|
+
import { tryOrLog } from '../_shared';
|
|
5
|
+
export const checkRelationExpr = (services) => {
|
|
6
|
+
const ModelParser = services.likec4.ModelParser;
|
|
7
|
+
return tryOrLog((node, accept) => {
|
|
8
|
+
// if (node.$container.$type !== 'DeploymentViewRulePredicateExpression') {
|
|
9
|
+
// // skip validation for this node, validated by container
|
|
10
|
+
// return
|
|
11
|
+
// }
|
|
12
|
+
const predicate = AstUtils.getContainerOfType(node, ast.isDeploymentViewRulePredicate);
|
|
13
|
+
if (!predicate || predicate.isInclude !== true) {
|
|
14
|
+
// no restriction for exclude predicate
|
|
15
|
+
return;
|
|
16
|
+
}
|
|
17
|
+
const doc = AstUtils.getDocument(node);
|
|
18
|
+
const parser = ModelParser.forDocument(doc);
|
|
19
|
+
let relationExpr = RelationExpr.unwrap(parser.parseRelationExpr(node));
|
|
20
|
+
const ModelRefOnlyExclude = 'Model reference is allowed in exclude predicate only';
|
|
21
|
+
if (RelationExpr.isDirect(relationExpr)) {
|
|
22
|
+
if (FqnExpr.isModelRef(relationExpr.source) || FqnExpr.isModelRef(relationExpr.target)) {
|
|
23
|
+
accept('error', ModelRefOnlyExclude, {
|
|
24
|
+
node: node,
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
return;
|
|
28
|
+
}
|
|
29
|
+
let expr;
|
|
30
|
+
if (RelationExpr.isIncoming(relationExpr)) {
|
|
31
|
+
expr = relationExpr.incoming;
|
|
32
|
+
}
|
|
33
|
+
else if (RelationExpr.isOutgoing(relationExpr)) {
|
|
34
|
+
expr = relationExpr.outgoing;
|
|
35
|
+
}
|
|
36
|
+
else {
|
|
37
|
+
expr = relationExpr.inout;
|
|
38
|
+
}
|
|
39
|
+
if (FqnExpr.isModelRef(expr)) {
|
|
40
|
+
accept('error', ModelRefOnlyExclude, {
|
|
41
|
+
node,
|
|
42
|
+
});
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
});
|
|
46
|
+
};
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { ast, getViewRulePredicateContainer } from '../../ast';
|
|
2
|
+
import { tryOrLog } from '../_shared';
|
|
3
|
+
export const checkRelationExprWith = (_services) => {
|
|
4
|
+
return tryOrLog((el, accept) => {
|
|
5
|
+
const container = getViewRulePredicateContainer(el);
|
|
6
|
+
if (!container || container.$type == 'DynamicViewIncludePredicate') {
|
|
7
|
+
return;
|
|
8
|
+
}
|
|
9
|
+
if (!container.isInclude) {
|
|
10
|
+
accept('error', 'Invalid usage inside "exclude"', {
|
|
11
|
+
node: el,
|
|
12
|
+
});
|
|
13
|
+
return;
|
|
14
|
+
}
|
|
15
|
+
});
|
|
16
|
+
};
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import type
|
|
1
|
+
import { type ValidationCheck } from 'langium';
|
|
2
2
|
import { ast } from '../ast';
|
|
3
3
|
import type { LikeC4Services } from '../module';
|
|
4
4
|
export declare const viewChecks: (services: LikeC4Services) => ValidationCheck<ast.LikeC4View>;
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { CstUtils } from 'langium';
|
|
2
|
+
import { ast } from '../ast';
|
|
3
|
+
import { projectIdFrom } from '../utils';
|
|
4
|
+
import { hasManualLayout } from '../view-utils/manual-layout';
|
|
5
|
+
import { RESERVED_WORDS, tryOrLog } from './_shared';
|
|
6
|
+
export const viewChecks = (services) => {
|
|
7
|
+
const index = services.shared.workspace.IndexManager;
|
|
8
|
+
return tryOrLog((el, accept) => {
|
|
9
|
+
const commentNode = CstUtils.findCommentNode(el.$cstNode, ['BLOCK_COMMENT']);
|
|
10
|
+
if (commentNode && hasManualLayout(commentNode.text)) {
|
|
11
|
+
accept('warning', `Migrate to the new manual layout snapshots (run LikeC4: Migrate manual layouts)`, {
|
|
12
|
+
node: el,
|
|
13
|
+
range: commentNode.range,
|
|
14
|
+
code: 'manual-layout-v1',
|
|
15
|
+
// codeDescription: {
|
|
16
|
+
// href: 'https://likec4.dev/docs/guides/manual-layout#migrating-from-v1-to-v2-manual-layout',
|
|
17
|
+
// },
|
|
18
|
+
});
|
|
19
|
+
}
|
|
20
|
+
if (!el.name) {
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
23
|
+
if (RESERVED_WORDS.includes(el.name)) {
|
|
24
|
+
accept('error', `Reserved word: ${el.name}`, {
|
|
25
|
+
node: el,
|
|
26
|
+
property: 'name',
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
const projectId = projectIdFrom(el);
|
|
30
|
+
const anotherViews = index
|
|
31
|
+
.projectElements(projectId, ast.LikeC4View)
|
|
32
|
+
.filter(n => n.name === el.name)
|
|
33
|
+
.limit(2)
|
|
34
|
+
.count();
|
|
35
|
+
if (anotherViews > 1) {
|
|
36
|
+
accept('error', `Duplicate view '${el.name}'`, {
|
|
37
|
+
node: el,
|
|
38
|
+
property: 'name',
|
|
39
|
+
});
|
|
40
|
+
}
|
|
41
|
+
});
|
|
42
|
+
};
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { isElementView } from '@likec4/core';
|
|
2
|
+
import { find, isNullish } from 'remeda';
|
|
3
|
+
export function assignNavigateTo(views) {
|
|
4
|
+
const allElementViews = new Map();
|
|
5
|
+
for (const v of views) {
|
|
6
|
+
if (isElementView(v) && v.viewOf && isNullish(v.extends)) {
|
|
7
|
+
const viewsOf = allElementViews.get(v.viewOf) ?? [];
|
|
8
|
+
viewsOf.push(v.id);
|
|
9
|
+
allElementViews.set(v.viewOf, viewsOf);
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
// set default navigateTo
|
|
13
|
+
for (const { id, nodes } of views) {
|
|
14
|
+
for (const node of nodes) {
|
|
15
|
+
const modelRef = node.modelRef;
|
|
16
|
+
if (node.navigateTo || !modelRef) {
|
|
17
|
+
continue;
|
|
18
|
+
}
|
|
19
|
+
// find first element view that is not the current one
|
|
20
|
+
const navigateTo = find(allElementViews.get(modelRef) ?? [], v => v !== id);
|
|
21
|
+
if (navigateTo) {
|
|
22
|
+
node.navigateTo = navigateTo;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
return views;
|
|
27
|
+
}
|
|
@@ -5,3 +5,9 @@ export declare function serializeToComment(layout: ViewManualLayout): string;
|
|
|
5
5
|
export declare function hasManualLayout(comment: string): boolean;
|
|
6
6
|
export declare function deserializeFromComment(comment: string): ViewManualLayout;
|
|
7
7
|
export declare function parseViewManualLayout(node: ast.LikeC4View): c4.ViewManualLayout | undefined;
|
|
8
|
+
/**
|
|
9
|
+
* Applies styles from the latest layout to the manual layout
|
|
10
|
+
* @param manualLayouted
|
|
11
|
+
* @param latest
|
|
12
|
+
*/
|
|
13
|
+
export declare function applyStylesToManualLayout<A extends c4.AnyAux>(manualLayouted: NoInfer<c4.LayoutedView<A>>, latest: c4.ComputedView<A> | c4.LayoutedView<A>): c4.LayoutedView<A>;
|
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import { exact, isAutoLayoutDirection } from '@likec4/core';
|
|
2
|
+
import { decode, encode } from '@msgpack/msgpack';
|
|
3
|
+
import { fromBase64, toBase64 } from '@smithy/util-base64';
|
|
4
|
+
import { shallowEqual } from 'fast-equals';
|
|
5
|
+
import { AstUtils, CstUtils } from 'langium';
|
|
6
|
+
import { find, mapValues } from 'remeda';
|
|
7
|
+
import { logger, logWarnError } from '../logger';
|
|
8
|
+
const { getDocument } = AstUtils;
|
|
9
|
+
function pack({ nodes, edges, ...rest }) {
|
|
10
|
+
return {
|
|
11
|
+
...rest,
|
|
12
|
+
nodes: mapValues(nodes, ({ x, y, width, height, isCompound, ...n }) => ({
|
|
13
|
+
...n,
|
|
14
|
+
b: [x, y, width, height],
|
|
15
|
+
c: isCompound,
|
|
16
|
+
})),
|
|
17
|
+
edges: mapValues(edges, ({ points, controlPoints, labelBBox, ...e }) => ({
|
|
18
|
+
...!!controlPoints && { cp: controlPoints },
|
|
19
|
+
...!!labelBBox && { l: labelBBox },
|
|
20
|
+
...e,
|
|
21
|
+
p: points,
|
|
22
|
+
})),
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
function unpack({ nodes, edges, autoLayout, ...rest }) {
|
|
26
|
+
return {
|
|
27
|
+
...rest,
|
|
28
|
+
/// Try to parse the old format for backward compatibility
|
|
29
|
+
autoLayout: isAutoLayoutDirection(autoLayout) ? { direction: autoLayout } : autoLayout,
|
|
30
|
+
nodes: mapValues(nodes, ({ b, c, ...n }) => ({
|
|
31
|
+
x: b[0],
|
|
32
|
+
y: b[1],
|
|
33
|
+
width: b[2],
|
|
34
|
+
height: b[3],
|
|
35
|
+
isCompound: c,
|
|
36
|
+
...n,
|
|
37
|
+
})),
|
|
38
|
+
edges: mapValues(edges, ({ p, cp, l, ...e }) => ({
|
|
39
|
+
...!!cp && { controlPoints: cp },
|
|
40
|
+
...!!l && { labelBBox: l },
|
|
41
|
+
...e,
|
|
42
|
+
points: p,
|
|
43
|
+
})),
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
const MAX_LINE_LENGTH = 500;
|
|
47
|
+
export function serializeToComment(layout) {
|
|
48
|
+
const bytes = encode(pack(layout));
|
|
49
|
+
const base64 = toBase64(bytes);
|
|
50
|
+
const lines = [];
|
|
51
|
+
let offset = 0;
|
|
52
|
+
while (offset < base64.length) {
|
|
53
|
+
lines.push(' * ' + base64.slice(offset, Math.min(offset + MAX_LINE_LENGTH, base64.length)));
|
|
54
|
+
offset += MAX_LINE_LENGTH;
|
|
55
|
+
}
|
|
56
|
+
lines.unshift('/**', ' * @likec4-generated(v1)');
|
|
57
|
+
lines.push(' */');
|
|
58
|
+
return lines.join('\n');
|
|
59
|
+
}
|
|
60
|
+
export function hasManualLayout(comment) {
|
|
61
|
+
return comment.includes('@likec4-generated');
|
|
62
|
+
}
|
|
63
|
+
export function deserializeFromComment(comment) {
|
|
64
|
+
if (!hasManualLayout(comment)) {
|
|
65
|
+
throw new Error(`Not a likec4-generated comment: ${comment}`);
|
|
66
|
+
}
|
|
67
|
+
const b64 = comment
|
|
68
|
+
.trim()
|
|
69
|
+
.split('\n')
|
|
70
|
+
.filter(l => !l.includes('**') && !l.includes('@likec4-') && !l.includes('*/'))
|
|
71
|
+
.map(l => l.replaceAll('*', '').trim())
|
|
72
|
+
.join('');
|
|
73
|
+
const decodedb64 = fromBase64(b64);
|
|
74
|
+
return unpack(decode(decodedb64));
|
|
75
|
+
}
|
|
76
|
+
export function parseViewManualLayout(node) {
|
|
77
|
+
const commentNode = CstUtils.findCommentNode(node.$cstNode, ['BLOCK_COMMENT']);
|
|
78
|
+
if (!commentNode || !hasManualLayout(commentNode.text)) {
|
|
79
|
+
return undefined;
|
|
80
|
+
}
|
|
81
|
+
try {
|
|
82
|
+
return deserializeFromComment(commentNode.text);
|
|
83
|
+
}
|
|
84
|
+
catch (e) {
|
|
85
|
+
const doc = getDocument(node);
|
|
86
|
+
logWarnError(e);
|
|
87
|
+
logger.warn(`Ignoring manual layout of "${node.name ?? 'unnamed'}" at ${doc.uri.fsPath}:${1 + (commentNode.range.start.line || 0)}`);
|
|
88
|
+
return undefined;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* Applies styles from the latest layout to the manual layout
|
|
93
|
+
* @param manualLayouted
|
|
94
|
+
* @param latest
|
|
95
|
+
*/
|
|
96
|
+
export function applyStylesToManualLayout(manualLayouted, latest) {
|
|
97
|
+
const nodes = manualLayouted.nodes.map(n => {
|
|
98
|
+
const latestNode = find(latest.nodes, l => l.id === n.id);
|
|
99
|
+
if (!latestNode) {
|
|
100
|
+
return n;
|
|
101
|
+
}
|
|
102
|
+
const color = latestNode.color;
|
|
103
|
+
if (color !== n.color) {
|
|
104
|
+
n = {
|
|
105
|
+
...n,
|
|
106
|
+
color,
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
const { opacity, border } = latestNode.style;
|
|
110
|
+
if (opacity !== n.style.opacity || border !== n.style.border) {
|
|
111
|
+
n = {
|
|
112
|
+
...n,
|
|
113
|
+
style: exact({
|
|
114
|
+
...n.style,
|
|
115
|
+
opacity,
|
|
116
|
+
border,
|
|
117
|
+
}),
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
return n;
|
|
121
|
+
});
|
|
122
|
+
const edges = manualLayouted.edges.map(e => {
|
|
123
|
+
const latestEdge = find(latest.edges, l => l.id === e.id);
|
|
124
|
+
if (!latestEdge) {
|
|
125
|
+
return e;
|
|
126
|
+
}
|
|
127
|
+
if (latestEdge.color !== e.color) {
|
|
128
|
+
e = {
|
|
129
|
+
...e,
|
|
130
|
+
color: latestEdge.color,
|
|
131
|
+
};
|
|
132
|
+
}
|
|
133
|
+
if (latestEdge.line !== e.line) {
|
|
134
|
+
e = {
|
|
135
|
+
...e,
|
|
136
|
+
line: latestEdge.line,
|
|
137
|
+
};
|
|
138
|
+
}
|
|
139
|
+
return e;
|
|
140
|
+
});
|
|
141
|
+
if (!shallowEqual(manualLayouted.nodes, nodes) || !shallowEqual(manualLayouted.edges, edges)) {
|
|
142
|
+
return {
|
|
143
|
+
...manualLayouted,
|
|
144
|
+
nodes,
|
|
145
|
+
edges,
|
|
146
|
+
};
|
|
147
|
+
}
|
|
148
|
+
return manualLayouted;
|
|
149
|
+
}
|