Files
architype/LayoutGroup.js

116 lines
2.6 KiB
JavaScript
Raw Normal View History

class LayoutGroup {
2019-07-16 05:28:02 +00:00
constructor(graphGroup, nodesByPos, nodes, type) {
2019-07-04 06:42:05 +00:00
this.graphGroup_ = graphGroup;
2019-07-05 06:10:41 +00:00
this.nodesByPos_ = nodesByPos;
this.nodes = new Set(nodes);
2019-07-16 05:28:02 +00:00
this.type = type;
this.tension = 0;
2019-07-09 20:34:36 +00:00
2019-07-13 03:41:36 +00:00
this.label = this.graphGroup_ ? this.graphGroup_.label : null;
2019-07-09 20:34:36 +00:00
for (let node of nodes) {
node.groups.add(this);
2019-07-09 20:34:36 +00:00
}
2019-07-03 01:42:17 +00:00
}
setTension() {
// Groups don't track tension, since we always want to sort last for total
// tension
2019-07-03 01:42:17 +00:00
this.vec = [0, 0];
2019-07-05 06:10:41 +00:00
for (let node of this.nodes) {
2019-07-03 01:42:17 +00:00
for (let i of [0, 1]) {
this.vec[i] += node.vec[i];
2019-07-11 18:14:06 +00:00
}
2019-07-03 01:42:17 +00:00
}
}
offsetCollides(offset) {
2019-07-05 06:10:41 +00:00
for (let node of this.nodes) {
let other = node.offsetCollides(offset);
if (other && !this.nodes.has(other)) {
2019-07-03 01:42:17 +00:00
return other;
}
}
return null;
}
savePos() {
2019-07-16 06:13:53 +00:00
this.savedVec_ = Array.from(this.vec);
2019-07-03 01:42:17 +00:00
}
restorePos() {
2019-07-16 06:13:53 +00:00
this.vec = this.savedVec_;
// Fix up nodesByPos, as intra-group collisions may have corrupted it
for (let node of this.nodes) {
this.nodesByPos_.set(node.pos, node);
}
2019-07-03 01:42:17 +00:00
}
moveBy(offset) {
2019-07-05 06:10:41 +00:00
for (let node of this.nodes) {
node.moveBy(offset);
}
// Fix up nodesByPos, as intra-group collisions may have corrupted it
for (let node of this.nodes) {
2019-07-05 16:18:22 +00:00
this.nodesByPos_.set(node.pos, node);
2019-07-03 01:42:17 +00:00
}
}
2019-07-04 06:42:05 +00:00
getMinMax() {
let min = [Number.MAX_SAFE_INTEGER, Number.MAX_SAFE_INTEGER];
let max = [Number.MIN_SAFE_INTEGER, Number.MIN_SAFE_INTEGER];
for (let node of this.nodes) {
for (let i of [0, 1]) {
min[i] = Math.min(min[i], node.pos[i]);
max[i] = Math.max(max[i], node.pos[i]);
}
}
if (this.graphGroup_ && this.graphGroup_.label) {
// Room for the label
--min[1];
}
return [min, max];
}
2019-07-16 05:28:02 +00:00
// border is [left, right, top, bottom]
isContained(pos, border=[0,0,0,0]) {
2019-07-09 20:34:36 +00:00
let [min, max] = this.getMinMax();
2019-07-16 05:28:02 +00:00
return (pos[0] >= (min[0] - border[0]) && pos[0] <= (max[0] + border[1]) &&
pos[1] >= (min[1] - border[2]) && pos[1] <= (max[1] + border[3]));
}
isContainedWithMargin(pos) {
return this.isContained(pos, [1, 1, this.label ? 2 : 1, 1]);
2019-07-09 20:34:36 +00:00
}
getSteps() {
2019-07-04 06:42:05 +00:00
if (!this.graphGroup_) {
return [];
2019-07-04 06:42:05 +00:00
}
let [min, max] = this.getMinMax();
let steps = [{
2019-07-04 06:42:05 +00:00
type: 'group',
min: min,
max: max,
2019-07-11 05:12:08 +00:00
id: this.graphGroup_.id,
}];
if (this.label) {
steps.push({
type: 'groupLabel',
min: [min[0], min[1]],
max: [max[0], min[1]],
id: this.graphGroup_.labelId,
label: this.graphGroup_.label,
});
}
return steps;
2019-07-04 06:42:05 +00:00
}
2019-07-09 16:43:33 +00:00
2019-07-16 05:28:02 +00:00
isType(type) {
return this.type == type;
2019-07-09 16:43:33 +00:00
}
2019-07-03 01:42:17 +00:00
}