/** * \brief A block structure defined over the variables * * A block structure defined over the variables such that each block contains * 1 or more variables, with the invariant that all constraints inside a block * are satisfied by keeping the variables fixed relative to one another * * Authors: * Tim Dwyer * * Copyright (C) 2005 Authors * * Released under GNU GPL. Read the file 'COPYING' for more information. */ #include "blocks.h" #include "block.h" #include "constraint.h" #ifdef RECTANGLE_OVERLAP_LOGGING #include using std::ios; using std::ofstream; using std::endl; #endif using std::set; using std::vector; using std::iterator; using std::list; using std::copy; long blockTimeCtr; Blocks::Blocks(Variable *vs[], const int n) : vs(vs),nvs(n) { blockTimeCtr=0; for(int i=0;i::iterator i=begin();i!=end();i++) { delete *i; } clear(); } /** * returns a list of variables with total ordering determined by the constraint * DAG */ list *Blocks::totalOrder() { list *order = new list; for(int i=0;ivisited=false; } for(int i=0;iin.size()==0) { dfsVisit(vs[i],order); } } return order; } // Recursive depth first search giving total order by pushing nodes in the DAG // onto the front of the list when we finish searching them void Blocks::dfsVisit(Variable *v, list *order) { v->visited=true; vector::iterator it=v->out.begin(); for(;it!=v->out.end();it++) { Constraint *c=*it; if(!c->right->visited) { dfsVisit(c->right, order); } } #ifdef RECTANGLE_OVERLAP_LOGGING ofstream f(LOGFILE,ios::app); f<<" order="<<*v<push_front(v); } /** * Processes incoming constraints, most violated to least, merging with the * neighbouring (left) block until no more violated constraints are found */ void Blocks::mergeLeft(Block *r) { #ifdef RECTANGLE_OVERLAP_LOGGING ofstream f(LOGFILE,ios::app); f<<"mergeLeft called on "<<*r<timeStamp=++blockTimeCtr; r->setUpInConstraints(); Constraint *c=r->findMinInConstraint(); while (c != NULL && c->slack()<0) { #ifdef RECTANGLE_OVERLAP_LOGGING f<<"mergeLeft on constraint: "<<*c<deleteMinInConstraint(); Block *l = c->left->block; if (l->in==NULL) l->setUpInConstraints(); double dist = c->right->offset - c->left->offset - c->gap; if (r->vars->size() < l->vars->size()) { dist=-dist; std::swap(l, r); } blockTimeCtr++; r->merge(l, c, dist); r->mergeIn(l); r->timeStamp=blockTimeCtr; removeBlock(l); c=r->findMinInConstraint(); } #ifdef RECTANGLE_OVERLAP_LOGGING f<<"merged "<<*r<setUpOutConstraints(); Constraint *c = l->findMinOutConstraint(); while (c != NULL && c->slack()<0) { #ifdef RECTANGLE_OVERLAP_LOGGING f<<"mergeRight on constraint: "<<*c<deleteMinOutConstraint(); Block *r = c->right->block; r->setUpOutConstraints(); double dist = c->left->offset + c->gap - c->right->offset; if (l->vars->size() > r->vars->size()) { dist=-dist; std::swap(l, r); } l->merge(r, c, dist); l->mergeOut(r); removeBlock(r); c=l->findMinOutConstraint(); } #ifdef RECTANGLE_OVERLAP_LOGGING f<<"merged "<<*l<deleted=true; //erase(doomed); } void Blocks::cleanup() { vector bcopy(size()); copy(begin(),end(),bcopy.begin()); for(vector::iterator i=bcopy.begin();i!=bcopy.end();i++) { Block *b=*i; if(b->deleted) { erase(b); delete b; } } } /** * Splits block b across constraint c into two new blocks, l and r (c's left * and right sides respectively) */ void Blocks::split(Block *b, Block *&l, Block *&r, Constraint *c) { b->split(l,r,c); #ifdef RECTANGLE_OVERLAP_LOGGING ofstream f(LOGFILE,ios::app); f<<"Split left: "<<*l<posn = b->posn; r->wposn = r->posn * r->weight; mergeLeft(l); // r may have been merged! r = c->right->block; r->wposn = r->desiredWeightedPosition(); r->posn = r->wposn / r->weight; mergeRight(r); removeBlock(b); insert(l); insert(r); } /** * returns the cost total squared distance of variables from their desired * positions */ double Blocks::cost() { double c = 0; for(set::iterator i=begin();i!=end();i++) { c += (*i)->cost(); } return c; }