use of easik.sketch.edge.SketchEdge in project fql by CategoricalData.
the class EntityNode method getShadowEdges.
/**
* This method returns the edges that will be "shadowed" in this entity for
* allowing various types of constraints. The problem arises when we have
* something like: A -> B -> C, where A is the summand of B, but B has
* to be specified. In this case, the B to C edge will be returned as a
* "shadow" edge. We handle this for other constraint types, too. For a
* good, working, technical example, see the shadowEdges.easik sample
* sketch.
*
* @return a set of edges that will be shadowed by this entity node.
*
* Removing shadow edges completely. Started by Sarah Van der Laan
* continued by Federico Mora because a partial solution is worse
* than all or nothing
*
* public LinkedHashSet<SketchEdge> getShadowEdges() { return
* getShadowEdges(new LinkedHashSet<EntityNode>(5), new
* LinkedHashSet<SketchEdge>(5)); }
*/
// Package-only implementation of the above that breaks recursion by
// ignoring
// shadowed nodes that we already know about.
/**
* @param ignore
* @param constraintEdges
*
* @return
*/
LinkedHashSet<SketchEdge> getShadowEdges(final Collection<EntityNode> ignore, final LinkedHashSet<SketchEdge> constraintEdges) {
// These are the other entity node that we (potentially) need to shadow:
final Collection<EntityNode> shadow = new LinkedHashSet<>(10);
CONSTRAINT: for (final ModelConstraint<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge> c : getMModel().getConstraints().values()) {
if (c instanceof SumConstraint) {
final SumConstraint<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge> s = (SumConstraint<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge>) c;
for (final ModelPath<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge> path : s.getPaths()) {
// for this entity, of course):
if (path.getDomain() == this) {
shadow.addAll(path.getEntities());
constraintEdges.addAll(path.getEdges());
continue CONSTRAINT;
}
}
} else if (c instanceof ProductConstraint) {
final ProductConstraint<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge> p = (ProductConstraint<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge>) c;
for (final ModelPath<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge> path : p.getPaths()) {
// codomains), along each product path
if (path.getCoDomain() == this) {
for (final ModelPath<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge> prodPath : p.getPaths()) {
// But we ignore all of the product path edges,
// since they will be automatically generated:
constraintEdges.addAll(prodPath.getEdges());
final Deque<EntityNode> pathNodes = new LinkedList<>();
pathNodes.addAll(prodPath.getEntities());
pathNodes.removeLast();
shadow.addAll(pathNodes);
}
continue CONSTRAINT;
}
}
} else if (c instanceof EqualizerConstraint) {
final EqualizerConstraint<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge> e = (EqualizerConstraint<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge>) c;
// injective *edge* doesn't cause that problem).
if (e.getSourceEntity() == this) {
final ModelPath<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge> projection = e.getProjection();
shadow.addAll(projection.getEntities());
// Ignore the projection edge itself:
constraintEdges.addAll(projection.getEdges());
continue CONSTRAINT;
}
} else if (c instanceof PullbackConstraint) {
final PullbackConstraint<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge> pb = (PullbackConstraint<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge>) c;
// WPBEDIT CF2012
for (int i = 0; i < pb.getWidth(); i++) {
ModelPath<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge> proj = pb.getProjectionPath(i);
if (this == proj.getCoDomain()) {
for (int j = 0; j < pb.getWidth(); j++) {
proj = pb.getProjectionPath(j);
Deque<EntityNode> projNodes = new LinkedList<>(proj.getEntities());
projNodes.removeLast();
shadow.addAll(projNodes);
constraintEdges.addAll(proj.getEdges());
}
continue CONSTRAINT;
}
}
} else if (c instanceof LimitConstraint) {
// TRIANGLES TODO CF2012 incomplete
}
}
final LinkedHashSet<SketchEdge> shadowEdges = new LinkedHashSet<>(20);
// All of the ignore entities, plus everything we just found should be
// ignored by any recursion:
final Collection<EntityNode> toIgnore = new LinkedHashSet<>(3);
toIgnore.add(this);
toIgnore.addAll(ignore);
toIgnore.addAll(shadow);
for (final EntityNode node : shadow) {
// it:
if ((node == this) || ignore.contains(node)) {
continue;
}
// Otherwise, shadow its non-partial edges, and all of its shadow
// edges:
shadowEdges.addAll(node.getShadowEdges(toIgnore, constraintEdges));
shadowEdges.addAll(node.getNonPartialEdges());
// Remove edges already
shadowEdges.removeAll(constraintEdges);
// involved in the sum
}
return shadowEdges;
}
use of easik.sketch.edge.SketchEdge in project fql by CategoricalData.
the class EntityNode method getOutgoingEdges.
/**
* Returns a Set of SketchEdges of the outgoing edges of this EntityNode.
* Note that edges for constraints are <b>not</b> included.
*
* @return Set of outgoing edges from this EntityNode
*/
@Override
public Set<SketchEdge> getOutgoingEdges() {
@SuppressWarnings("rawtypes") final List outEdges = _theModel.getGraphLayoutCache().getOutgoingEdges(this, Collections.emptySet(), false, true);
final Set<SketchEdge> outgoing = new LinkedHashSet<>(outEdges.size());
for (final Object out : outEdges) {
if (out instanceof SketchEdge) {
outgoing.add((SketchEdge) out);
}
}
return outgoing;
}
use of easik.sketch.edge.SketchEdge in project fql by CategoricalData.
the class EditSketchEdgeAction method actionPerformed.
/**
* Called when clicked upon, will popup the edge UI and, if accepted, update
* the name and cascade option.
*
* @param e
* The action event
*/
@Override
public void actionPerformed(ActionEvent e) {
Sketch _ourSketch = _theFrame.getMModel();
Object[] currentSelection = _ourSketch.getSelectionCells();
// non-edges which might be selected
if ((currentSelection.length == 1) && (currentSelection[0] instanceof SketchEdge)) {
// cancel operation
if (_ourSketch.isSynced()) {
if (JOptionPane.showConfirmDialog(_theFrame, "Warning: this sketch is currently synced with a db; delete and break synchronization?", "Warning!", JOptionPane.OK_CANCEL_OPTION, JOptionPane.WARNING_MESSAGE) == JOptionPane.CANCEL_OPTION) {
return;
}
}
SketchEdge edge = (SketchEdge) currentSelection[0];
EdgeOptions opts = new EdgeOptions(_theFrame, edge);
if (opts.isAccepted()) {
if (!opts.getName().equals(edge.getName())) {
edge.setName(opts.getName());
}
edge.setCascading(opts.getCascadeMode());
_theFrame.getInfoTreeUI().refreshTree();
_ourSketch.getGraphLayoutCache().reload();
_ourSketch.clearSelection();
_ourSketch.repaint();
_ourSketch.setDirty();
_ourSketch.setSynced(false);
}
}
}
use of easik.sketch.edge.SketchEdge in project fql by CategoricalData.
the class NewSketchEdgeAction method actionPerformed.
/**
* The action for creating a new edge. Make sure the selection is alright,
* and then create the edge.
*
* @param e
* The action event
*/
@Override
public void actionPerformed(ActionEvent e) {
Sketch _ourSketch = _theFrame.getMModel();
// cancel operation
if (_ourSketch.isSynced()) {
if (JOptionPane.showConfirmDialog(_theFrame, "Warning: this sketch is currently synced with a db; continue and break synchronization?", "Warning!", JOptionPane.OK_CANCEL_OPTION, JOptionPane.WARNING_MESSAGE) == JOptionPane.CANCEL_OPTION) {
return;
}
}
Object[] currentSelection = _ourSketch.getSelectionCells();
EntityNode[] node = new EntityNode[(_edgeType == Edge.SELF) ? 1 : 2];
if (currentSelection.length == node.length) {
for (int i = 0; i < node.length; i++) {
node[i] = null;
if (currentSelection[i] instanceof EntityNode) {
node[i] = (EntityNode) currentSelection[i];
}
}
}
if ((_edgeType == Edge.SELF) && (node[0] == null)) {
JOptionPane.showMessageDialog(_theFrame, "Operation must be performed with one entity selected", "Error", JOptionPane.ERROR_MESSAGE);
} else if ((_edgeType != Edge.SELF) && ((node[0] == null) || (node[1] == null))) {
JOptionPane.showMessageDialog(_theFrame, "Operation must be performed with two entities selected", "Error", JOptionPane.ERROR_MESSAGE);
} else {
EdgeOptions opts = new EdgeOptions(_theFrame, _edgeType, node[0], (_edgeType == Edge.SELF) ? null : node[1]);
if (opts.isAccepted()) {
SketchEdge edge;
if (_edgeType == Edge.SELF) {
edge = new PartialEdge(node[0], node[0], opts.getName(), opts.getCascadeMode());
} else {
// UI:
if (opts.isReversed()) {
node = new EntityNode[] { node[1], node[0] };
}
if (_edgeType == Edge.PARTIAL) {
edge = new PartialEdge(node[0], node[1], opts.getName(), opts.getCascadeMode());
} else if (_edgeType == Edge.INJECTIVE) {
edge = new InjectiveEdge(node[0], node[1], opts.getName(), opts.getCascadeMode());
} else {
edge = new NormalEdge(node[0], node[1], opts.getName(), opts.getCascadeMode());
}
}
_ourSketch.addEdge(edge);
_ourSketch.setDirty();
_ourSketch.setSynced(false);
}
}
}
use of easik.sketch.edge.SketchEdge in project fql by CategoricalData.
the class Sketch method initialiseModel.
/**
* When we initialise the sketch, we flush out all the data concerning the
* sketch itself.
*
* This methods serves as a "new sketch" function.
*/
@Override
public void initialiseModel() {
clearSelection();
model = new SketchGraphModel(this);
final GraphLayoutCache glc = new GraphLayoutCache(model, new ModelViewFactory<SketchFrame, SketchGraphModel, Sketch, EntityNode, SketchEdge>());
setModel(model);
setGraphLayoutCache(glc);
_nodes = new LinkedHashMap<>();
_edges = new LinkedHashMap<>();
if (_Frame.getInfoTreeUI() != null) {
// Wipe
_Frame.setInfoTreeUI(new ModelInfoTreeUI<>(_Frame));
// Tree
_Frame.getInfoTreeUI().refreshTree();
}
_docInfo.reset();
_views = new HashMap<>();
_connParams = new HashMap<>();
model.discardUndo();
}
Aggregations