UNPKG

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