1 | import { isFinite, map, omit, toPairs } from "lodash/fp";
|
2 | import { isAttachablePoint, isDefinedCXML, isGPMLAnchor, isPvjsonBurr, isPvjsonSingleFreeNode, isPvjsonGroup, unionLSV } from "../gpml-utilities";
|
3 | import { SmartPoint } from "../geom-utils";
|
4 | import { calculateAllPoints } from "./calculateAllPoints";
|
5 | import * as VError from "verror";
|
6 | import * 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
|
8 | export const DEFAULT_STUB_LENGTH = 20;
|
9 | /**
|
10 | * getOffsetAndOrientationScalarsAlongAxis
|
11 | *
|
12 | * @param relValue {number}
|
13 | * @param axis {string}
|
14 | * @param referencedEntity
|
15 | * @return {OffsetOrientationAndPositionScalarsAlongAxis}
|
16 | */
|
17 | function 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?
|
21 | referencedEntity) {
|
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 | */
|
61 | export 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 | */
|
80 | export 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 |