UNPKG

33.6 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3var fp_1 = require("lodash/fp");
4var gpml_utilities_1 = require("../gpml-utilities");
5var geom_utils_1 = require("../geom-utils");
6var calculateAllPoints_1 = require("./calculateAllPoints");
7var VError = require("verror");
8var MarkerMappings = require("./MarkerMappings.json");
9// a stub is a short path segment that is used for the first and/or last segment(s) of a path
10exports.DEFAULT_STUB_LENGTH = 20;
11/**
12 * getOffsetAndOrientationScalarsAlongAxis
13 *
14 * @param relValue {number}
15 * @param axis {string}
16 * @param referencedEntity
17 * @return {OffsetOrientationAndPositionScalarsAlongAxis}
18 */
19function getOffsetAndOrientationScalarsAlongAxis(positionScalar, relativeOffsetScalar, axis,
20// TODO are we correctly handling the case of a group as the referenced
21// entity? Do we have the group width and height yet to properly calculate
22// this?
23referencedEntity) {
24 var offsetScalar = relativeOffsetScalar *
25 (axis === "x" ? referencedEntity.width : referencedEntity.height);
26 // TODO WP536 has a referenced entity that lacks width/height. Why?
27 // The referenced entity was a group.
28 // Is the problem that the group was nested?
29 // Or is it a problem with the order of evaluation of entities (trying to
30 // parse a dependent entity before its dependencies were parsed)?
31 if (!fp_1.isFinite(offsetScalar)) {
32 throw new Error("\n\t\t\tGot non-finite value " + offsetScalar + " for offsetScalar\n\t\t\talong " + axis + " axis for\n\t\t\tgetOffsetAndOrientationScalarsAlongAxis(\n\t\t\t\tpositionScalar=" + positionScalar + ",\n\t\t\t\trelativeOffsetScalar=" + relativeOffsetScalar + ",\n\t\t\t\treferencedEntity=\n\t\t\t\t" + JSON.stringify(referencedEntity, null, " ") + "\n\t\t\t)\n\t\t");
33 }
34 // orientationScalar here refers to the initial direction the edge takes as
35 // it moves away from the entity to which it is attached.
36 var orientationScalar;
37 if (positionScalar === 0) {
38 orientationScalar = -1;
39 }
40 else if (positionScalar === 1) {
41 orientationScalar = 1;
42 }
43 else {
44 orientationScalar = 0;
45 }
46 return { offsetScalar: offsetScalar, orientationScalar: orientationScalar, positionScalar: positionScalar };
47}
48/**
49 * preprocessGPML
50 *
51 * @param edge {GPMLEdge}
52 * @return {GPMLEdge}
53 */
54function preprocessGPML(Edge) {
55 var isAttachedToOrVia = Edge.Graphics.Point
56 .filter(function (p) { return p.GraphRef && gpml_utilities_1.isDefinedCXML(p.GraphRef); })
57 .map(function (p) { return p.GraphRef; });
58 if (isAttachedToOrVia.length > 0) {
59 // In pvjson, an edge attaches directly to another entity (Node, Edge, Group),
60 // not to an anchor.
61 // If the edge attaches to another edge, it does so VIA an anchor.
62 Edge["isAttachedToOrVia"] = isAttachedToOrVia;
63 }
64 return Edge;
65}
66exports.preprocessGPML = preprocessGPML;
67/**
68 * postprocessPVJSON
69 *
70 * @param referencedEntities
71 * @param pvjsonEdge {pvjsonEdge}
72 * @return {pvjsonEdge}
73 */
74function postprocessPVJSON(referencedEntities, pvjsonEdge) {
75 var points = pvjsonEdge.points, drawAs = pvjsonEdge.drawAs;
76 var pointCount = points.length;
77 var index = 0;
78 var pvjsonEdgeIsAttachedTo = [];
79 var providedPvjsonPoints = fp_1.map(function (point) {
80 var marker = point.marker, x = point.x, y = point.y;
81 if (!!marker) {
82 // NOTE: side effects below
83 if (index === 0) {
84 pvjsonEdge.markerStart = marker;
85 }
86 else if (index === pointCount - 1) {
87 pvjsonEdge.markerEnd = marker;
88 }
89 if (MarkerMappings.hasOwnProperty(marker)) {
90 pvjsonEdge.type = fp_1.toPairs(MarkerMappings[marker]).reduce(function (acc, _a) {
91 var namespace = _a[0], moreTypes = _a[1];
92 return gpml_utilities_1.unionLSV(acc, moreTypes);
93 }, pvjsonEdge.type);
94 }
95 }
96 if (gpml_utilities_1.isAttachablePoint(point)) {
97 // NOTE: pvjson allows for expressing one edge attached to another edge.
98 // When we do this, we say that the POINT attaches to an ANCHOR on the other edge,
99 // but the EDGE attaches to the other EDGE, never the anchor.
100 var isAttachedTo = point.isAttachedTo, attachmentDisplay = point.attachmentDisplay;
101 if (!attachmentDisplay.offset) {
102 throw new Error("attachmentDisplay for a Point has no offset property.\n\t\t\t\t\tpostprocessPVJSON(\n\t\t\t\t\t\treferencedEntities=" + JSON.stringify(referencedEntities, null, " ") + ",\n\t\t\t\t\t\tpvjsonEdge=" + JSON.stringify(pvjsonEdge, null, " ") + "\n\t\t\t\t\t)");
103 }
104 // entityReferencedByPoint can be a regular node (DataNode, Shape, Label)
105 // or an Anchor. If connected to an Anchor, the biological meaning is
106 // that the edge is connected to another edge, but in this code, we
107 // implement this by treating the Anchor as a node, as if it were
108 // a "burr" that is always stuck (isAttachedTo) the other edge.
109 var entityReferencedByPoint = referencedEntities &&
110 !!isAttachedTo &&
111 referencedEntities[isAttachedTo];
112 var entityIdReferencedByEdge = gpml_utilities_1.isGPMLAnchor(entityReferencedByPoint)
113 ? entityReferencedByPoint.isAttachedTo
114 : entityReferencedByPoint.id;
115 // WARNING: side effect
116 pvjsonEdgeIsAttachedTo.push(entityIdReferencedByEdge);
117 var entityReferencedByEdge = referencedEntities[entityIdReferencedByEdge];
118 var orientation_1 = (point.orientation =
119 point.orientation || []);
120 // attachmentDisplay: { position: [x: number, y: number], offset: [xOffset: number, yOffset: number], orientation: [dx: number, dy: number] }
121 //
122 // x = xDistance / width (relative: [0,1])
123 // y = yDistance / height (relative: [0,1])
124 // xOffset = distance offset in x direction (absolute)
125 // yOffset = distance offset in y direction (absolute)
126 // dx = x component of edge emanation angle (unit: [0,1])
127 // dy = y component of edge emanation angle (unit: [0,1])
128 //
129 // 0 ----------------- x ------------------->
130 // | ========================================
131 // | || ||
132 // | || ||
133 // | || ||
134 // y || ||
135 // | || ||
136 // | || ||
137 // | || ||
138 // | || ||
139 // v ===================*====================
140 // |
141 // yOffset |
142 // | |
143 // v |
144 // ----------*
145 // xOffset> \
146 // \
147 // \ dx>
148 // dy \
149 // | \
150 // v \
151 // \
152 //
153 // example above is an attachmentDisplay specifying an edge that emanates down and to the right
154 // at a 45 deg. angle (1, 1), offset right 5 x units and down 11 y units from the center (0.5)
155 // of the bottom side (1) of the node: {position: [0.75, 1], offset: [5, 11], orientation: [1, 1]}
156 //
157 //
158 // where x is distance from left side along width axis as a percentage of the total width
159 // y is distance from top side along height axis as a percentage of the total height
160 // offsetX, offsetY are obvious from the name. Notice they are absolute, unlike x,y.
161 // dx, dy are unit vector coordinates of a point that specifies how the edge emanates from the node
162 if (gpml_utilities_1.isPvjsonSingleFreeNode(entityReferencedByEdge) ||
163 gpml_utilities_1.isPvjsonGroup(entityReferencedByEdge) ||
164 gpml_utilities_1.isPvjsonBurr(entityReferencedByEdge)) {
165 var position = attachmentDisplay.position, relativeOffset = attachmentDisplay.relativeOffset;
166 // edge connected to a SingleFreeNode, a Group or a Burr, but NOT another edge or an anchor
167 try {
168 var _a = getOffsetAndOrientationScalarsAlongAxis(position[0], relativeOffset[0], "x", entityReferencedByEdge), offsetScalarX = _a.offsetScalar, orientationScalarX = _a.orientationScalar;
169 var _b = getOffsetAndOrientationScalarsAlongAxis(position[1], relativeOffset[1], "y", entityReferencedByEdge), offsetScalarY = _b.offsetScalar, orientationScalarY = _b.orientationScalar;
170 if (index === 0) {
171 orientation_1[0] = orientationScalarX;
172 orientation_1[1] = orientationScalarY;
173 }
174 else {
175 orientation_1[0] = -1 * orientationScalarX;
176 orientation_1[1] = -1 * orientationScalarY;
177 }
178 // TODO is there a case where we would ever use offset for edges?
179 attachmentDisplay.offset[0] = offsetScalarX;
180 attachmentDisplay.offset[1] = offsetScalarY;
181 point.attachmentDisplay = fp_1.omit(["relativeOffset"], attachmentDisplay);
182 }
183 catch (err) {
184 throw new VError(err, "\n\t\t\t\t\t\tError for:\n\t\t\t\t\t\tpostprocessPVJSON(\n\t\t\t\t\t\t\treferencedEntities=" + JSON.stringify(referencedEntities, null, " ") + ",\n\t\t\t\t\t\t\tpvjsonEdge=" + JSON.stringify(pvjsonEdge, null, " ") + "\n\t\t\t\t\t\t)\n\t\t\t\t\t");
185 /* TODO should we use this?
186 console.warn(`Setting offsetScalar equal to 0.`);
187 offsetScalar = 0;
188 //*/
189 }
190 }
191 else if (gpml_utilities_1.isGPMLAnchor(entityReferencedByPoint)) {
192 // edge is connected to another edge via an anchor
193 point.attachmentDisplay.position =
194 entityReferencedByPoint.attachmentDisplay.position;
195 }
196 else {
197 throw new Error("\n\t\t\t\t\tEdge or Point attached to unexpected entity.\n\t\t\t\t\tPoint is attached to:\n\t\t\t\t\t" + JSON.stringify(entityReferencedByPoint, null, " ") + "\n\t\t\t\t\tPoint is attached to:\n\t\t\t\t\t" + JSON.stringify(entityReferencedByEdge, null, " ") + "\n\t\t\t\t\tfor:\n\t\t\t\t\tpostprocessPVJSON(\n\t\t\t\t\treferencedEntities=" + JSON.stringify(referencedEntities, null, " ") + ",\n\t\t\t\t\tpvjsonEdge=" + JSON.stringify(pvjsonEdge, null, " ") + "\n\t\t\t\t\t)\n\t\t\t\t");
198 }
199 }
200 // NOTE: side effect
201 index += 1;
202 return fp_1.omit(["marker"], point);
203 }, points);
204 var pvjsonEdgeAttachedToCount = pvjsonEdgeIsAttachedTo.length;
205 if (pvjsonEdgeAttachedToCount > 0) {
206 pvjsonEdge.isAttachedTo = pvjsonEdgeIsAttachedTo;
207 }
208 var allPvjsonPoints;
209 if (["StraightLine", "SegmentedLine"].indexOf(drawAs) > -1) {
210 allPvjsonPoints = providedPvjsonPoints;
211 }
212 else if (["ElbowLine", "CurvedLine"].indexOf(drawAs) > -1) {
213 // pvjsonEdge.isAttachedTo refers to what the EDGE is fundamentally attached to.
214 // pvjsonEdge.points[0].isAttachedTo refers to what the POINT is attached to.
215 //
216 // From the perspective of the biological meaning, the edge is always attached to
217 // a regular node like a DataNode or Shape (maybe Label?) but never to an Anchor.
218 //
219 // From the perspective of the implementation of the graphics, we say the edge
220 // has points, one or more of which can be connected to an Anchor.
221 var sourceEntity = void 0;
222 var targetEntity = void 0;
223 if (pvjsonEdgeAttachedToCount === 2) {
224 sourceEntity = referencedEntities[pvjsonEdgeIsAttachedTo[0]];
225 targetEntity = referencedEntities[pvjsonEdgeIsAttachedTo[1]];
226 }
227 else if (pvjsonEdgeAttachedToCount === 1) {
228 var firstPoint = providedPvjsonPoints[0];
229 var lastPoint = providedPvjsonPoints[providedPvjsonPoints.length - 1];
230 if (firstPoint.hasOwnProperty("isAttachedTo")) {
231 sourceEntity = referencedEntities[pvjsonEdgeIsAttachedTo[0]];
232 }
233 else if (lastPoint.hasOwnProperty("isAttachedTo")) {
234 targetEntity = referencedEntities[pvjsonEdgeIsAttachedTo[0]];
235 }
236 else {
237 throw new Error("edge \"" + pvjsonEdge.id + "\" is said to be attached to \"" + pvjsonEdge.isAttachedTo.join() + "\",\n\t\t\t\t\tbut neither first nor last points have \"isAttachedTo\" property");
238 }
239 }
240 allPvjsonPoints = calculateAllPoints_1.calculateAllPoints(providedPvjsonPoints.map(function (point) { return new geom_utils_1.SmartPoint(point); }), sourceEntity, targetEntity);
241 }
242 else {
243 throw new Error("\n\t\t\tUnknown edge drawer \"" + drawAs + "\" for:\n\t\t\tpostprocessPVJSON(\n\t\t\treferencedEntities=" + JSON.stringify(referencedEntities, null, " ") + ",\n\t\t\tpvjsonEdge=" + JSON.stringify(pvjsonEdge, null, " ") + "\n\t\t\t)\n\t\t");
244 // TODO should we use this?
245 // allPvjsonPoints = providedPvjsonPoints;
246 }
247 // TODO how do we distinguish between intermediate (not first or last) points that a user
248 // has explicitly specified vs. intermediate points that are only implied?
249 // Do we need to? I think once a user specifies any implicit points, they may all be
250 // made explicit.
251 // GPML currently does not specify implicit intermediate points, but
252 // pvjson does.
253 pvjsonEdge.points = allPvjsonPoints;
254 // TODO can I get rid of isAttachedToOrVia earlier?
255 return fp_1.omit(["isAttachedToOrVia"], pvjsonEdge);
256}
257exports.postprocessPVJSON = postprocessPVJSON;
258//function recursivelyGetReferencedElements(acc, gpmlElement: GPMLElement) {
259// const { Graphics } = gpmlElement;
260// const graphRefIds: string[] = !!Graphics.Point &&
261// Graphics.Point[0]._exists !== false
262// ? Graphics.Point.filter(P => isString(P.GraphRef)).map(P => P.GraphRef)
263// : gpmlElement.hasOwnProperty("GraphRef")
264// ? arrayify(gpmlElement.GraphRef)
265// : [];
266//
267// const referencedElementIds = arrayify(graphRefIds);
268// //const referencedElementIds = unionLSV(graphRefIds, gpmlElement.GroupRef);
269// return referencedElementIds.length === 0
270// ? acc
271// : hl([
272// acc,
273// hl(referencedElementIds)
274// .flatMap(referencedElementId =>
275// hl(getGPMLElementByGraphId(referencedElementId))
276// )
277// .flatMap(function(referencedElement: GPMLElement) {
278// return recursivelyGetReferencedElements(
279// hl([referencedElement]),
280// referencedElement
281// );
282// })
283// ]).merge();
284//}
285//
286//export function postprocessPVJSON(
287// pvjsonEdge: PvjsonEdge
288//): Highland.Stream<PvjsonEdge> {
289// return hl([
290// hl([pvjsonEdge])
291// .reduce(hl([]), recursivelyGetReferencedElements)
292// .merge()
293// .flatMap(function(referencedGPMLElement: GPMLElement) {
294// return hl(
295// getPvjsonEntityLatestByGraphId(
296// referencedGPMLElement.GraphId
297// )
298// );
299// })
300// ])
301// .merge()
302// .reduce({}, function(
303// acc: {
304// [key: string]: (PvjsonNode | PvjsonEdge);
305// },
306// referencedEntity: (PvjsonNode | PvjsonEdge)
307// ) {
308// acc[referencedEntity.id] = referencedEntity;
309// return acc;
310// })
311// .map(function(referencedEntities) {
312// return process(pvjsonEdge, referencedEntities);
313// });
314// .merge();
315//}
316//
317//export function createEdgeTransformStream(
318// processor,
319// edgeType: "Interaction" | "GraphicalLine"
320//): (
321// s: Highland.Stream<GPML2013a.InteractionType | GPML2013a.GraphicalLineType>
322//) => Highland.Stream<(PvjsonNode | PvjsonEdge)> {
323// const {
324// fillInGPMLPropertiesFromParent,
325// getGPMLElementByGraphId,
326// getPvjsonEntityLatestByGraphId,
327// ensureGraphIdExists,
328// preprocessGPMLElement,
329// processPropertiesAndType
330// } = processor;
331//
332// function recursivelyGetReferencedElements(acc, gpmlElement: GPMLElement) {
333// const { Graphics } = gpmlElement;
334// const graphRefIds: string[] = !!Graphics.Point &&
335// Graphics.Point[0]._exists !== false
336// ? Graphics.Point.filter(P => isString(P.GraphRef)).map(P => P.GraphRef)
337// : gpmlElement.hasOwnProperty("GraphRef")
338// ? arrayify(gpmlElement.GraphRef)
339// : [];
340//
341// const referencedElementIds = arrayify(graphRefIds);
342// //const referencedElementIds = unionLSV(graphRefIds, gpmlElement.GroupRef);
343// return referencedElementIds.length === 0
344// ? acc
345// : hl([
346// acc,
347// hl(referencedElementIds)
348// .flatMap(referencedElementId =>
349// hl(getGPMLElementByGraphId(referencedElementId))
350// )
351// .flatMap(function(referencedElement: GPMLElement) {
352// return recursivelyGetReferencedElements(
353// hl([referencedElement]),
354// referencedElement
355// );
356// })
357// ]).merge();
358// }
359//
360// return function(s) {
361// return s
362// .map(preprocessGPMLElement)
363// .flatMap(function(
364// gpmlEdge: GPMLElement
365// ): Highland.Stream<Highland.Stream<PvjsonNode | PvjsonEdge>> {
366// const { Graphics } = gpmlEdge;
367//
368// const gpmlAnchors = Graphics.hasOwnProperty("Anchor") &&
369// Graphics.Anchor &&
370// Graphics.Anchor[0] &&
371// Graphics.Anchor[0]._exists !== false
372// ? Graphics.Anchor.filter(a => a.hasOwnProperty("GraphId"))
373// : [];
374//
375// const fillInGPMLPropertiesFromEdge = fillInGPMLPropertiesFromParent(
376// gpmlEdge
377// );
378//
379// return hl([
380// hl([gpmlEdge])
381// .map(processPropertiesAndType(edgeType))
382// .flatMap(function(pvjsonEdge: PvjsonEdge) {
383// return hl([
384// hl([gpmlEdge])
385// .reduce(hl([]), recursivelyGetReferencedElements)
386// .merge()
387// .flatMap(function(referencedGPMLElement: GPMLElement) {
388// return hl(
389// getPvjsonEntityLatestByGraphId(
390// referencedGPMLElement.GraphId
391// )
392// );
393// })
394// ])
395// .merge()
396// .reduce({}, function(
397// acc: {
398// [key: string]: (PvjsonNode | PvjsonEdge);
399// },
400// referencedEntity: (PvjsonNode | PvjsonEdge)
401// ) {
402// acc[referencedEntity.id] = referencedEntity;
403// return acc;
404// })
405// .map(function(referencedEntities) {
406// return process(pvjsonEdge, referencedEntities);
407// });
408// }),
409// hl(gpmlAnchors)
410// .map(preprocessGPMLElement)
411// .map(function(gpmlAnchor: GPMLElement) {
412// const filledInAnchor = fillInGPMLPropertiesFromEdge(gpmlAnchor);
413// filledInAnchor.GraphRef = gpmlEdge.GraphId;
414// return filledInAnchor;
415// })
416// .map(processPropertiesAndType("Anchor"))
417// .map(function(pvjsonAnchor: PvjsonNode): PvjsonNode {
418// const drawAnchorAs = pvjsonAnchor.drawAs;
419// if (drawAnchorAs === "None") {
420// defaultsDeep(pvjsonAnchor, {
421// Height: 4,
422// Width: 4
423// });
424// } else if (drawAnchorAs === "Circle") {
425// defaultsDeep(pvjsonAnchor, {
426// Height: 8,
427// Width: 8
428// });
429// }
430// return pvjsonAnchor;
431// })
432// ]);
433// })
434// .merge();
435// };
436//}
437//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"edge.js","sourceRoot":"","sources":["../../src/edge/edge.ts"],"names":[],"mappings":"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"}
\No newline at end of file