org.metacsp.framework.meta.MetaConstraintSolver Maven / Gradle / Ivy
package org.metacsp.framework.meta;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.NotSerializableException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map.Entry;
import java.util.Set;
import java.util.Vector;
import java.util.logging.Logger;
import org.metacsp.framework.Constraint;
import org.metacsp.framework.ConstraintNetwork;
import org.metacsp.framework.ConstraintSolver;
import org.metacsp.framework.Variable;
import org.metacsp.framework.multi.MultiConstraintSolver;
import org.metacsp.meta.TCSP.TCSPSolver;
import org.metacsp.meta.symbolsAndTime.Scheduler;
import org.metacsp.throwables.NoFocusDefinedException;
import org.metacsp.utility.UI.SearchTreeFrame;
import org.metacsp.utility.logging.MetaCSPLogging;
import edu.uci.ics.jung.graph.DelegateForest;
/**
* A meta-CSP is a high-level CSP whose variables and/or constraints are defined implicitly. These
* variables and constraints are called meta-variables and meta-constraints. They typically represent
* the elements of a higher-level problem defined over a so.called ground-CSP. Many known problems
* can be cast as meta-CSPs. For instance, a resource scheduling problem is a meta-CSP whose
* meta-variables are sets of possibly concurrent activities that over-consume a resource. These
* activities are themselves variables in a ground-CSP which decides their placement in time
* according to temporal constraints. The activities and temporal constraints are, respectively,
* the variables and constraints of the ground-CSP, while the sets of possibly overlapping
* and over-consuming activities are meta-variables in the meta-CSP. The constraints of the meta-CSP
* (meta-constraints) are the resources themselves.
*
* Solving a meta-CSP consists in finding values for the meta-variables such that no meta-constraint
* is violated. The values of meta-variables are {@link ConstraintNetwork}s - i.e., variables and
* constraints which should be posted to the ground-CSP(s) in order to satisfy the meta-constraints.
* In resource scheduling for instance, a value of a meta-variable is a temporal constraint that
* eliminates the temporal overlap of the concurrent over-consuming activities.
*
* This class provides the fundamental mechanisms to define and solve a meta-CSP. It implements
* a backtracking search over the search space defined by meta-variables, meta-values and
* meta-constraints. Examples of concrete {@link MetaConstraintSolver}s that implement this
* class are given in this framework (e.g., the {@link TCSPSolver}, which implements
* a Temporal Constraint Satisfaction Problem solver as a meta-CSP, and the {@link Scheduler}, which
* implements the ESTA algorithm for solving resource scheduling problems).
*
* This class also provides limited support for constraint optimization through branch-and-bound.
*
* @author Federico Pecora
*
*/
public abstract class MetaConstraintSolver extends MultiConstraintSolver {
/**
*
*/
private static final long serialVersionUID = -7343190680692608215L;
protected Vector metaConstraints = null;
protected DelegateForest g;
protected MetaVariable currentVertex = null;
protected boolean breakSearch = false;
protected HashMap metaVarsToMetaCons;
protected HashMap resolvers;
protected HashMap resolversInverseMapping;
protected long animationTime = 0;
protected int counterMoves;
protected FocusConstraint currentFocus = null;
private Vector> backedUpCNs = new Vector>();
//private Vector> statesAlongCurrentBranch = new Vector>();
public MetaConstraint[] getMetaConstraints() {
return this.metaConstraints.toArray(new MetaConstraint[this.metaConstraints.size()]);
}
/**
* Get the final list of resolvers (meta-values) added to the ground-CSP(s)
* for obtaining a solution to the meta-CSP.
* @return List of resolvers that were added to obtain the solution of the
* meta-CSP.
*/
public ConstraintNetwork[] getAddedResolvers() {
Collection ret = resolvers.values();
return ret.toArray(new ConstraintNetwork[ret.size()]);
}
/**
* Retract all resolvers added to the ground-CSP(s) in order to obtain the
* current solution to the meta-CSP. This is useful if one wants to "reset"
* the meta-CSP to its original unsolved state.
*/
public void retractResolvers() {
Set vars = resolvers.keySet();
for (ConstraintNetwork var : vars) {
ConstraintNetwork value = resolvers.get(var);
logger.fine("=== ||| === Retracting value: " + Arrays.toString(value.getConstraints()));
this.retractResolver(var, value);
}
this.resolvers = new HashMap();
this.resolversInverseMapping = new HashMap();
this.metaVarsToMetaCons.clear();
}
/**
* Clear memory of all resolvers added to the ground-CSP(s) in order to obtain the
* current solution to the meta-CSP. This is useful if one wants to restart solving
* from the current solved state. Note that resolvers are not retracted - to "reset"
* the meta-CSP to its original unsolved state, use method {@link MetaConstraintSolver#retractResolvers()}.
*/
public void clearResolvers() {
this.resolvers = new HashMap();
this.metaVarsToMetaCons = new HashMap();
}
protected class TerminalNode extends MetaVariable {
private boolean success;
public TerminalNode(boolean success) {
super(null, null);
this.success = success;
}
public String toString() {
if (success) return "SUCCESS";
return "FAILURE";
}
}
protected MetaConstraintSolver(Class>[] constraintTypes, long animationTime, ConstraintSolver ... internalSolvers) {
super(constraintTypes, MetaVariable.class, internalSolvers, null);
g = new DelegateForest();
this.animationTime = animationTime;
this.resolvers = new HashMap();
this.metaVarsToMetaCons= new HashMap();
this.resolversInverseMapping = new HashMap();
this.counterMoves=0;
}
/**
* Method to add a {@link MetaConstraint} to this meta-CSP solver.
* @param metaConstraint The {@link MetaConstraint} to add.
*/
public void addMetaConstraint(MetaConstraint metaConstraint) {
if (this.metaConstraints == null) this.metaConstraints = new Vector();
// here the metaConstraint is added
this.metaConstraints.add(metaConstraint);
// the meta constraint now refers to the MetaConstraintSolver
metaConstraint.setMetaSolver(this);
boolean found = false;
// Here I simply check if the type of the metaConstraint is already taken into account,
// otherwise the array of types is extended
for (Class> cl : this.constraintTypes) if (cl.equals(metaConstraint.getClass())) found = true;
if (!found) {
Class>[] newConstraintTypes = new Class>[this.constraintTypes.length+1];
for (int i = 0; i < this.constraintTypes.length; i++) newConstraintTypes[i] = this.constraintTypes[i];
newConstraintTypes[this.constraintTypes.length] = metaConstraint.getClass();
this.constraintTypes = newConstraintTypes;
}
}
protected MetaVariable getConflict() {
if(this.metaConstraints==null) return null;
for (MetaConstraint df : this.metaConstraints) {
ConstraintNetwork cn = df.getMetaVariable();
if (cn != null)
return new MetaVariable(df, cn);
}
return null;
}
/**
* Implement this method to define any extra operations that should be performed
* before backtracking over a {@link MetaVariable} (and before it is even chosen).
*/
public abstract void preBacktrack();
/**
* Implement this method to define any extra operations that should be performed
* after branching over a {@link MetaVariable}.
* @param metaVariable The {@link MetaVariable} that has been branched upon.
*/
public abstract void postBacktrack(MetaVariable metaVariable);
/**
* Initiates CSP-style backtracking search on the meta-CSP.
* @return true
iff a set of assignments to all {@link MetaVariable}s which
* satisfies the {@link MetaConstraint}s was found.
*/
public boolean backtrack() {
g = new DelegateForest();
logger.info("Starting search...");
// preBacktrack();
MetaVariable conflict = null;
if ((conflict = this.getConflict()) != null) {
currentVertex = conflict;
if (backtrackHelper(conflict)) {
// postBacktrack();
logger.info("... solution found");
return true;
}
// postBacktrack();
return false;
}
// postBacktrack();
logger.info("... no conflicts found");
return true;
}
//FPA: Is this used? Seems not... please remove! (Iran: it is used for hybrid planner benchmarking)
private boolean timeout = false;
public boolean getTimeOut(){
return timeout;
}
private boolean backtrackHelper(MetaVariable metaVariable) {
preBacktrack();
if (this.g.getRoot() == null) this.g.addVertex(currentVertex);
ConstraintNetwork mostProblematicNetwork = metaVariable.getConstraintNetwork();
logger.fine("Solving conflict: " + metaVariable);
ConstraintNetwork[] values = metaVariable.getMetaConstraint().getMetaValues(metaVariable);
if (values != null) for (ConstraintNetwork value : values) value.setAnnotation(metaVariable);
if (metaVariable.getMetaConstraint().valOH != null && values!=null) {
//System.out.println("SORTING with " + metaVariable.getMetaConstraint().valOH.getClass());
Arrays.sort(values, metaVariable.getMetaConstraint().valOH);
}
if (values == null || values.length == 0) {
this.g.addEdge(new NullConstraintNetwork(null), currentVertex, new TerminalNode(false));
logger.fine("Failure (1)...");
}
else {
for (ConstraintNetwork value : values) {
if (animationTime != 0) {
try { Thread.sleep(animationTime); }
catch (InterruptedException e) { e.printStackTrace(); }
}
String valString = "";
if (value.getVariables().length != 0) valString += "Vars = " + Arrays.toString(value.getVariables());
if (value.getConstraints().length != 0) valString += " Cons = " + Arrays.toString(value.getConstraints());
logger.fine("Trying value: " + valString);
if (this.addResolver(mostProblematicNetwork, value)) {
this.resolvers.put(mostProblematicNetwork, value);
this.metaVarsToMetaCons.put(mostProblematicNetwork, metaVariable.getMetaConstraint());
this.resolversInverseMapping.put(value,mostProblematicNetwork);
this.counterMoves++;
logger.fine("Success...");
metaVariable.getMetaConstraint().markResolvedSub(metaVariable, value);
MetaVariable newConflict = this.getConflict();
if (newConflict == null || breakSearch) {
this.g.addEdge(value, currentVertex, new TerminalNode(true));
breakSearch = false;
return true;
}
// addEdege(e,v,v)
this.g.addEdge(value, currentVertex, newConflict);
currentVertex = newConflict;
if (backtrackHelper(newConflict)) return true;
logger.fine("Retracting value: " + Arrays.toString(value.getConstraints()));
this.retractResolver(mostProblematicNetwork, value);
this.resolvers.remove(mostProblematicNetwork);
this.metaVarsToMetaCons.remove(mostProblematicNetwork);
this.resolversInverseMapping.remove(value);
this.counterMoves--;
}
else {
this.g.addEdge(value, currentVertex, new TerminalNode(false));
logger.fine("Failure... (2)");
}
}
}
logger.fine("Backtracking...");
currentVertex = this.g.getParent(currentVertex);
postBacktrack(metaVariable);
return false;
}
/**
* Service method for backtracking with serialization-based saving of {@link ConstraintNetwork}s.
* This method backs up {@link ConstraintNetwork}s before branching.
*/
private HashMap backupCNs(MultiConstraintSolver conSol) {
//Here we want to save the CNs
ByteArrayOutputStream bos = null;
ObjectOutputStream oos = null;
HashMap currentLevel = new HashMap();
try {
bos = new ByteArrayOutputStream();
oos = new ObjectOutputStream(bos);
for (ConstraintSolver cs : conSol.getConstraintSolvers()) {
logger.finest("Backing up CN of " + cs.getClass().getSimpleName());
ConstraintNetwork cn = cs.getConstraintNetwork();
oos.writeObject(cn);
byte[] backup = bos.toByteArray();
currentLevel.put(cs,backup);
if (cs instanceof MultiConstraintSolver) {
//System.out.println("RECURSIVE on " + cs.getClass().getSimpleName());
HashMap lower = backupCNs((MultiConstraintSolver)cs);
currentLevel.putAll(lower);
}
}
return currentLevel;
}
catch (NotSerializableException e) { e.printStackTrace(); }
catch (IOException e) { e.printStackTrace(); }
return null;
}
/**
* Service method for backtracking with serialization-based saving of {@link ConstraintNetwork}s.
* This method reinstates old {@link ConstraintNetwork}s after backtraking.
*/
private void restoreCNs() {
//REINSTATE OLD CNs
HashMap backup = backedUpCNs.lastElement();
for (Entry entry : backup.entrySet()) {
byte[] backedUpNetwork = entry.getValue();
ConstraintSolver cs = entry.getKey();
logger.finest("Restoring CN of " + cs.getClass().getSimpleName());
ByteArrayInputStream bis = new ByteArrayInputStream(backedUpNetwork);
ObjectInputStream in = null;
try {
in = new ObjectInputStream(bis);
ConstraintNetwork old = (ConstraintNetwork)in.readObject();
cs.setConstraintNetwork(old);
}
catch (IOException e) { e.printStackTrace(); }
catch (ClassNotFoundException e) { e.printStackTrace(); }
}
backedUpCNs.remove(backup);
backup.clear();
logger.info("backup queue: " + (backedUpCNs.size()+1) + " --> " + backedUpCNs.size());
}
/**
* This backtrack method uses serialization to back up {@link ConstraintNetwork}s before branching. This allows to
* backtrack without propagation - but is very memory intensive. In practice, this does not work on reasonably
* sized problems.
*/
private boolean backtrackHelperWithSerialization(MetaVariable metaVariable) {
preBacktrack();
if (this.g.getRoot() == null) this.g.addVertex(currentVertex);
ConstraintNetwork mostProblematicNetwork = metaVariable.getConstraintNetwork();
logger.fine("Solving conflict: " + metaVariable);
ConstraintNetwork[] values = metaVariable.getMetaConstraint().getMetaValues(metaVariable);
if (metaVariable.getMetaConstraint().valOH != null && values!=null) Arrays.sort(values, metaVariable.getMetaConstraint().valOH);
if (values == null || values.length == 0) {
this.g.addEdge(new NullConstraintNetwork(null), currentVertex, new TerminalNode(false));
logger.fine("Failure (1)...");
}
else {
for (ConstraintNetwork value : values) {
if (animationTime != 0) {
try { Thread.sleep(animationTime); }
catch (InterruptedException e) { e.printStackTrace(); }
}
String valString = "";
if (value.getVariables().length != 0) valString += "Vars = " + Arrays.toString(value.getVariables());
if (value.getConstraints().length != 0) valString += " Cons = " + Arrays.toString(value.getConstraints());
logger.fine("Trying value: " + valString);
this.backedUpCNs.add(backupCNs(this));
/*** PRINT INFO ***/
/*
long sizeOfBackup = 0;
for (HashMap oneHM : backedUpCNs) {
for (byte[] oneCN : oneHM.values())
sizeOfBackup += oneCN.length;
}
DecimalFormat df = new DecimalFormat("#.##");
logger.info("Current backup size: " + df.format((sizeOfBackup/1024.00)) + " KB");
*/
/*** END PRINT INFO ***/
if (this.addResolver(mostProblematicNetwork, value)) {
this.resolvers.put(mostProblematicNetwork, value);
this.metaVarsToMetaCons.put(mostProblematicNetwork,metaVariable.getMetaConstraint());
this.resolversInverseMapping.put(value,mostProblematicNetwork);
this.counterMoves++;
logger.fine("Success...");
metaVariable.getMetaConstraint().markResolvedSub(metaVariable, value);
MetaVariable newConflict = this.getConflict();
if (newConflict == null || breakSearch) {
this.g.addEdge(value, currentVertex, new TerminalNode(true));
breakSearch = false;
return true;
}
// addEdege(e,v,v)
this.g.addEdge(value, currentVertex, newConflict);
currentVertex = newConflict;
if (backtrackHelper(newConflict)) return true;
logger.fine("Retracting value: " + Arrays.toString(value.getConstraints()));
//this.retractResolver(mostProblematicNetwork, value);
this.restoreCNs();
this.retractResolverSub(mostProblematicNetwork, value);
this.resolvers.remove(mostProblematicNetwork);
this.metaVarsToMetaCons.remove(mostProblematicNetwork);
this.resolversInverseMapping.remove(value);
this.counterMoves--;
}
else {
this.g.addEdge(value, currentVertex, new TerminalNode(false));
logger.fine("Failure... (2)");
}
}
}
logger.fine("Backtracking...");
currentVertex = this.g.getParent(currentVertex);
postBacktrack(metaVariable);
return false;
}
protected final boolean addResolver(ConstraintNetwork metaVarConstraintNetwork, ConstraintNetwork resolverNetwork) {
if (!this.addResolverSub(metaVarConstraintNetwork, resolverNetwork)) return false;
Constraint[] resolverNetworkConstraints = resolverNetwork.getConstraints();
HashMap> solvers2constraints =
new HashMap>();
for (Constraint c : resolverNetworkConstraints) {
if (!solvers2constraints.containsKey(c.getScope()[0].getConstraintSolver())) {
Vector newVec = new Vector();
solvers2constraints.put(c.getScope()[0].getConstraintSolver(), newVec);
}
solvers2constraints.get(c.getScope()[0].getConstraintSolver()).add(c);
}
Vector addedConstraints = new Vector();
for (ConstraintSolver cs : solvers2constraints.keySet()) {
Constraint[] toAddOneSolver =
solvers2constraints.get(cs).toArray(new Constraint[solvers2constraints.get(cs).size()]);
if (cs.addConstraints(toAddOneSolver)) addedConstraints.add(toAddOneSolver);
else {
for (Constraint[] toDel : addedConstraints) {
toDel[0].getScope()[0].getConstraintSolver().removeConstraints(toDel);
}
this.retractResolverSub(metaVarConstraintNetwork, resolverNetwork);
return false;
}
}
return true;
}
protected final void retractResolver(ConstraintNetwork metaVar, ConstraintNetwork res) {
this.logger.finest("Retracting resolver:");
this.logger.finest(" MetaVariable: " + metaVar.toString());
this.logger.finest(" MetaValue: " + res.toString());
Constraint[] groundConstraints = res.getConstraints();
HashMap> solvers2constraints = new HashMap>();
for (Constraint c : groundConstraints) {
if (!solvers2constraints.containsKey(c.getScope()[0].getConstraintSolver())) {
Vector newVec = new Vector();
solvers2constraints.put(c.getScope()[0].getConstraintSolver(), newVec);
}
solvers2constraints.get(c.getScope()[0].getConstraintSolver()).add(c);
}
for (ConstraintSolver cs : solvers2constraints.keySet()) {
Constraint[] toAddOneSolver = solvers2constraints.get(cs).toArray(new Constraint[solvers2constraints.get(cs).size()]);
cs.removeConstraints(toAddOneSolver);
}
this.retractResolverSub(metaVar, res);
this.logger.finest("Done retracting resolver.");
}
@Override
public boolean propagate() {
// TODO Auto-generated method stub
return false;
}
/**
* Implement this method to define any extra operations that should happen
* after retracting a meta-value in the meta-CSP search (e.g., when backtracking).
* @param metaVariable The {@link MetaVariable} over which backtracking occurs.
* @param metaValue The meta-value that has been retracted.
*/
protected abstract void retractResolverSub(ConstraintNetwork metaVariable, ConstraintNetwork metaValue);
/**
* Implement this method to define any additional operations that should happen before
* adding a meta-value in the meta-CSP search (e.g., when branching).
* @param metaVariable The {@link MetaVariable} over which the search is branching.
* @param metaValue The meta-value that has been selected (the branch).
* @return true
iff the metavalue is consistent with the ground-CSP (Note: if false
is returned, the ground-CSP has not been changed, i.e., it is left in a consistent state.)
*/
protected abstract boolean addResolverSub(ConstraintNetwork metaVariable, ConstraintNetwork metaValue);
/**
* Get all the {@link Variable}s of all ground solvers used by this {@link MetaConstraintSolver}.
* @return The list of variables for each ground solver.
*/
public HashMap getGroundVariables() {
HashMap ret = new HashMap();
for (ConstraintSolver cs : this.constraintSolvers) {
ret.put(cs, cs.getVariables());
}
return ret;
}
/**
* Draws the search space of the meta-CSP.
*/
public void draw() {
SearchTreeFrame.draw(this.g);
}
/**
* Get all the ground constraint solvers of this meta-CSP.
* @return All ground constraint solvers of this meta-CSP.
*/
public ConstraintSolver[] getConstraintSolvers() {
return this.constraintSolvers;
}
/**
* Interrupt the current meta-CSP search.
*/
public void breakSearch() { breakSearch = true; }
/**
* Perform branch-and-bound search for an optimal solution to this meta-CSP. The cost
* of assignments is defined in the implementing class through the upper/lower-bound getter and
* setter methods.
* @return true
iff IRAN: complete this please!
*/
public boolean branchAndBound() {
g = new DelegateForest();
//graph = new ObservableGraph(g);
logger.info("Starting search...");
// preBacktrack();
MetaVariable con = null;
if ((con = this.getConflict()) != null) {
currentVertex = con;
if (branchAndBoundHelper(con)) {
// postBacktrack();
logger.info("... solution found");
return true;
}
// postBacktrack();
return false;
}
// postBacktrack();
logger.info("... no conflicts found");
return true;
}
private boolean branchAndBoundHelper(MetaVariable metaVariable) {
if(metaVariable == null)
return false;
preBacktrack();
if (this.g.getRoot() == null) this.g.addVertex(currentVertex);
ConstraintNetwork cn = metaVariable.getConstraintNetwork();
logger.fine("Solving conflict: " + metaVariable);
ConstraintNetwork[] values = metaVariable.getMetaConstraint().getMetaValues(metaVariable);
if (metaVariable.getMetaConstraint().valOH != null) Arrays.sort(values, metaVariable.getMetaConstraint().valOH);
if (values == null || values.length == 0) {
this.g.addEdge(new NullConstraintNetwork(null), currentVertex, new TerminalNode(false));
logger.fine("Failure... (1)");
}
else {
for (ConstraintNetwork value : values) {
if (animationTime != 0) {
try { Thread.sleep(animationTime); }
catch (InterruptedException e) { e.printStackTrace(); }
}
logger.fine("Trying value: " + Arrays.toString(value.getConstraints()));
if(hasConflictClause(value))
continue;
this.addResolver(cn, value);
setUpperBound();
// System.out.println("test: " + "U: " + getUpperBound() + " L: " + getLowerBound());
if(getUpperBound() <= getLowerBound()){
this.retractResolver(cn, value);
continue;
}
logger.fine("Success...");
metaVariable.getMetaConstraint().markResolvedSub(metaVariable, value);
MetaVariable newCon = this.getConflict();
if(newCon != null ){
this.g.addEdge(value, currentVertex, newCon);
currentVertex = newCon;
}
if(newCon == null)
setLowerBound();
if(branchAndBoundHelper(newCon))
return true;
logger.fine("Retracting value: " + Arrays.toString(value.getConstraints()));
this.retractResolver(cn, value);
logger.fine("Failure... (2)");
}
}
resetFalseClause();
logger.fine("Backtracking...");
currentVertex = this.g.getParent(currentVertex);
postBacktrack(metaVariable);
return false;
}
protected abstract double getUpperBound();
protected abstract void setUpperBound();
protected abstract double getLowerBound();
protected abstract void setLowerBound();
protected abstract boolean hasConflictClause(ConstraintNetwork metaValue);
protected abstract void resetFalseClause();
@Override
public String getDescription() {
String spacer = "";
for (int i = 0; i < nesting; i++) spacer += spacing;
String ret = spacer + "[" + this.getClass().getSimpleName() + " vars: [";
ret += this.variableType.getSimpleName();
ret += "] constraints: [";
for (int i = 0; i < this.constraintTypes.length; i++) {
ret += this.constraintTypes[i].getSimpleName();
if (i != this.constraintTypes.length-1) ret += ",";
}
ret += "]";
nesting++;
// for (MetaConstraintSolver cs : this.nextMetaConstraintSolvers) ret += "\n" + cs.getDescription();
for (ConstraintSolver cs : this.getConstraintSolvers()) ret += "\n" + cs.getDescription();
nesting--;
ret += "]";
return ret;
}
public void failurePruning(int failure_time){
super.failurePruning(failure_time);
this.counterMoves=0;
this.g=new DelegateForest();
this.resolvers.clear();
this.metaVarsToMetaCons.clear();
}
public int getCounterMoves() {
return counterMoves;
}
public FocusConstraint getCurrentFocusConstraint() {
return currentFocus;
}
// private void removeFromConstraintSolvers() {
// if (currentFocus != null) {
// for (ConstraintSolver cs : this.getConstraintSolvers()) {
// cs.removeConstraint(currentFocus);
// }
// }
// }
//
// private void addToConstraintSolvers() {
// for (ConstraintSolver cs : this.getConstraintSolvers()) {
// cs.addConstraint(currentFocus);
// }
// }
public synchronized void setCurrentFocusConstraint(FocusConstraint focus) {
this.currentFocus = focus;
}
public synchronized Variable[] getFocused() {
if (currentFocus != null) return currentFocus.getScope();
return null;
}
public synchronized void setFocus(Variable ... vars) {
currentFocus = new FocusConstraint();
currentFocus.setScope(vars);
}
public synchronized boolean isFocused(Variable var) {
if (currentFocus == null) return false;
for (Variable v : getFocused()) {
if (v.equals(var)) return true;
}
return false;
}
public synchronized void focus(Variable ... vars) {
if (currentFocus == null) {
currentFocus = new FocusConstraint();
}
Vector scopeVars = new Vector();
for (Variable v : currentFocus.getScope()) scopeVars.add(v);
for (Variable v : vars) scopeVars.add(v);
currentFocus.setScope(scopeVars.toArray(new Variable[scopeVars.size()]));
}
public synchronized void removeFromCurrentFocus(Variable ... vars) {
if (currentFocus == null) throw new NoFocusDefinedException(vars);
Vector newScope = new Vector();
for (Variable vOld : currentFocus.getScope()) {
boolean found = false;
for (Variable vToRem : vars) {
if (vOld.equals(vToRem)) found = true;
}
if (!found) newScope.add(vOld);
}
currentFocus.setScope(newScope.toArray(new Variable[newScope.size()]));
}
public void setCounterMoves(int counterMoves) {
this.counterMoves = counterMoves;
}
public HashMap getResolvers() {
return resolvers;
}
public HashMap getResolversInverseMapping() {
return resolversInverseMapping;
}
public void setResolvers(HashMap resolvers) {
this.resolvers = resolvers;
}
public MetaConstraint getMetaConstraint(ConstraintNetwork metaVariable) {
return this.metaVarsToMetaCons.get(metaVariable);
}
@Override
public void registerValueChoiceFunctions() {
// TODO Auto-generated method stub
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy