X-Git-Url: http://git.tdb.fi/?a=blobdiff_plain;f=source%2Flayout.cpp;h=c616ff6fc7caff9833070cf1c30e7c7710e20ad9;hb=8eeb6ee5d40c21150839e24996cc3e9ef308374d;hp=b4c3a0eccd38883ac83beb7b863a62b6d3a52b6b;hpb=6549eff66171d53be91ea7ba3f23339812ef6cad;p=libs%2Fgltk.git diff --git a/source/layout.cpp b/source/layout.cpp index b4c3a0e..c616ff6 100644 --- a/source/layout.cpp +++ b/source/layout.cpp @@ -1,4 +1,8 @@ +#include #include +#include +#include +#include "arrangement.h" #include "container.h" #include "layout.h" #include "widget.h" @@ -44,10 +48,13 @@ public: Row add_row(); Row operator[](unsigned); - Row get_object_row(); - bool solve(); + Row get_objective_row(); float get_variable(unsigned); + bool solve(); private: + void prepare_columns(); + void add_artificial_variables(); + void remove_artificial_variables(); unsigned find_minimal_ratio(unsigned); void make_basic_column(unsigned, unsigned); bool pivot(); @@ -80,10 +87,14 @@ Layout::Pointers Layout::pointers[2] = Layout::Layout(): container(0), + n_active_slots(0), margin(8), row_spacing(5), col_spacing(4) -{ } +{ + n_slack_constraints[0] = 0; + n_slack_constraints[1] = 0; +} Layout::~Layout() { @@ -106,16 +117,67 @@ void Layout::set_margin(const Sides &m) update(); } +void Layout::set_spacing(unsigned s) +{ + row_spacing = s; + col_spacing = s; + if(container) + update(); +} + +void Layout::set_row_spacing(unsigned s) +{ + row_spacing = s; + if(container) + update(); +} + +void Layout::set_column_spacing(unsigned s) +{ + col_spacing = s; + if(container) + update(); +} + +void Layout::push_arrangement(Arrangement &arr) +{ + arrangement_stack.push_back(&arr); +} + +Arrangement *Layout::get_arrangement() const +{ + if(arrangement_stack.empty()) + return 0; + else + return arrangement_stack.back(); +} + +void Layout::pop_arrangement(Arrangement &arr) +{ + list::iterator begin = find(arrangement_stack.begin(), arrangement_stack.end(), &arr); + if(begin==arrangement_stack.end()) + return; + + while(1) + { + Arrangement *top = arrangement_stack.back(); + arrangement_stack.pop_back(); + if(!arrangement_stack.empty()) + arrangement_stack.back()->arrange(*top); + if(top==&arr) + break; + } +} + void Layout::add_widget(Widget &wdg) { if(!container) throw logic_error("!container"); - Slot *slot = create_slot(wdg); - for(list::iterator i=slot->constraints.begin(); i!=slot->constraints.end(); ++i) - i->target.constraints.push_back(Constraint(complement(i->type), *slot)); - slot->index = slots.size(); - slots.push_back(slot); + slots.push_back(new Slot(*this, wdg)); + update_slot_indices(); + if(!arrangement_stack.empty()) + arrangement_stack.back()->arrange(wdg); if(container) update(); } @@ -140,18 +202,32 @@ void Layout::remove_widget(Widget &wdg) delete *i; slots.erase(i); - unsigned n = 0; - for(i=slots.begin(); i!=slots.end(); ++i, ++n) - (*i)->index = n; - + update_slot_indices(); update(); return; } } -Layout::Slot *Layout::create_slot(Widget &wdg) +void Layout::update_slot_indices() { - return new Slot(*this, wdg); + n_active_slots = 0; + for(list::iterator i=slots.begin(); i!=slots.end(); ++i) + { + if((*i)->widget.is_visible()) + (*i)->index = n_active_slots++; + else + (*i)->index = -1; + } + + n_slack_constraints[0] = 0; + n_slack_constraints[1] = 0; + for(list::iterator i=slots.begin(); i!=slots.end(); ++i) + if((*i)->index>=0) + { + for(list::iterator j=(*i)->constraints.begin(); j!=(*i)->constraints.end(); ++j) + if(j->target.index>(*i)->index && (j->type&SLACK)) + ++n_slack_constraints[j->type&1]; + } } Layout::Slot &Layout::get_slot_for_widget(Widget &wdg) @@ -165,19 +241,10 @@ Layout::Slot &Layout::get_slot_for_widget(Widget &wdg) Layout::ConstraintType Layout::complement(ConstraintType type) { - if(type==RIGHT_OF) - return LEFT_OF; - else if(type==LEFT_OF) - return RIGHT_OF; - else if(type==ABOVE) - return BELOW; - else if(type==BELOW) - return ABOVE; - else - return type; + return static_cast((type&~(SELF_MASK|TARGET_MASK)) | ((type&SELF_MASK)<<2) | ((type&TARGET_MASK)>>2)); } -void Layout::add_constraint(Widget &src, ConstraintType type, Widget &tgt) +void Layout::create_constraint(Widget &src, ConstraintType type, Widget &tgt, int sp) { if(&src==&tgt) throw invalid_argument("&src==&tgt"); @@ -190,11 +257,24 @@ void Layout::add_constraint(Widget &src, ConstraintType type, Widget &tgt) return; src_slot.constraints.push_back(Constraint(type, tgt_slot)); + src_slot.constraints.back().spacing = sp; tgt_slot.constraints.push_back(Constraint(complement(type), src_slot)); + tgt_slot.constraints.back().spacing = sp; + update_slot_indices(); update(); } +void Layout::add_constraint(Widget &src, ConstraintType type, Widget &tgt) +{ + create_constraint(src, type, tgt, -1); +} + +void Layout::add_constraint(Widget &src, ConstraintType type, Widget &tgt, unsigned spacing) +{ + create_constraint(src, type, tgt, spacing); +} + void Layout::set_gravity(Widget &wdg, int h, int v) { Slot &slot = get_slot_for_widget(wdg); @@ -217,34 +297,51 @@ void Layout::set_expand(Widget &wdg, bool h, bool v) void Layout::update() { - for(list::iterator i=slots.begin(); i!=slots.end(); ++i) - { - (*i)->widget.autosize(); - (*i)->geom = (*i)->widget.get_geometry(); - } - - solve_constraints(HORIZONTAL); - solve_constraints(VERTICAL); + solve_constraints(HORIZONTAL, UPDATE); + solve_constraints(VERTICAL, UPDATE); for(list::iterator i=slots.begin(); i!=slots.end(); ++i) (*i)->widget.set_geometry((*i)->geom); +} + +void Layout::autosize() +{ + solve_constraints(HORIZONTAL, AUTOSIZE); + solve_constraints(VERTICAL, AUTOSIZE); + container->set_size(autosize_geom.w, autosize_geom.h); } -void Layout::solve_constraints(int dir) +void Layout::solve_constraints(int dir, SolveMode mode) { Pointers &ptrs = pointers[dir&VERTICAL]; + const Geometry &geom = container->get_geometry(); + if(mode==UPDATE && geom.*(ptrs.dim)::iterator i=slots.begin(); i!=slots.end(); ++i) { - linprog.get_object_row()[(*i)->index*5] = ((*i)->*(ptrs.packing)).gravity/weight; - linprog.get_object_row()[(*i)->index*5+1] = (((*i)->*(ptrs.packing)).expand ? weight : -1); + if((*i)->index<0) + continue; + + LinearProgram::Row objective = linprog.get_objective_row(); + if(mode==AUTOSIZE) + { + objective[(*i)->index*5] = -1; + objective[(*i)->index*5+1] = -1; + } + else + { + objective[(*i)->index*5] = ((*i)->*(ptrs.packing)).gravity/weight; + objective[(*i)->index*5+1] = (((*i)->*(ptrs.packing)).expand ? weight : -1); + } { // Prevent the widget from going past the container's low edge. @@ -254,60 +351,87 @@ void Layout::solve_constraints(int dir) row.back() = margin.*(ptrs.low_margin); } + if(mode==UPDATE) { // Prevent the widget from going past the container's high edge. LinearProgram::Row row = linprog.add_row(); row[(*i)->index*5] = 1; row[(*i)->index*5+1] = 1; row[(*i)->index*5+3] = 1; - row.back() = container->get_geometry().*(ptrs.dim)-margin.*(ptrs.high_margin); + row.back() = geom.*(ptrs.dim)-margin.*(ptrs.high_margin); } + if(((*i)->*(ptrs.packing)).gravity==0) { - /* Only allow the widget's dimension to increase. The geometry has - previously been set to the smallest allowable size. */ + /* This forces the widget's distance from the left and right edge of + the container to be equal. It's a bit of a hack, but more time and + thought is needed for a better solution. */ + LinearProgram::Row row = linprog.add_row(); + row[(*i)->index*5+2] = 1; + row[(*i)->index*5+3] = -1; + } + + { + /* Don't allow the widget's dimension to get below that determined + by autosizing. */ LinearProgram::Row row = linprog.add_row(); row[(*i)->index*5+1] = 1; row[(*i)->index*5+4] = -1; - row.back() = (*i)->geom.*(ptrs.dim); + row.back() = (*i)->autosize_geom.*(ptrs.dim); } - /* Add rows for user-defined constraints. Below/above and left/right of - constraints are always added in pairs, so it's only necessary to create a - row for one half. */ + /* Add rows for user-defined constraints. Constraints are always added + in pairs, so it's only necessary to create a row for one half. */ + unsigned k = n_active_slots*5; for(list::iterator j=(*i)->constraints.begin(); j!=(*i)->constraints.end(); ++j) - { - if((j->type&1)==dir && j->type!=BELOW && j->type!=LEFT_OF) + if(j->target.index>(*i)->index && (j->type&1)==dir) { LinearProgram::Row row = linprog.add_row(); + float polarity = ((j->type&SELF_DIM) ? -1 : 1); if(j->type&SELF_POS) - row[(*i)->index*5] = 1; + row[(*i)->index*5] = polarity; if(j->type&SELF_DIM) - row[(*i)->index*5+1] = 1; + row[(*i)->index*5+1] = polarity; if(j->type&TARGET_POS) - row[j->target.index*5] = -1; + row[j->target.index*5] = -polarity; if(j->type&TARGET_DIM) - row[j->target.index*5+1] = -1; + row[j->target.index*5+1] = -polarity; if(j->type&SPACING) - row.back() = this->*(ptrs.spacing); + row.back() = (j->spacing>=0 ? j->spacing : this->*(ptrs.spacing)); + if(j->type&SLACK) + row[k++] = -1; } - } } if(!linprog.solve()) return; - for(list::iterator i=slots.begin(); i!=slots.end(); ++i) + if(mode==AUTOSIZE) { - (*i)->geom.*(ptrs.pos) = linprog.get_variable((*i)->index*5); - (*i)->geom.*(ptrs.dim) = linprog.get_variable((*i)->index*5+1); + autosize_geom.*(ptrs.dim) = 0; + for(list::iterator i=slots.begin(); i!=slots.end(); ++i) + if((*i)->index>=0) + { + int high_edge = linprog.get_variable((*i)->index*5)+linprog.get_variable((*i)->index*5+1); + autosize_geom.*(ptrs.dim) = max(autosize_geom.*(ptrs.dim), high_edge+margin.*(ptrs.high_margin)); + } + } + else + { + for(list::iterator i=slots.begin(); i!=slots.end(); ++i) + if((*i)->index>=0) + { + (*i)->geom.*(ptrs.pos) = linprog.get_variable((*i)->index*5); + (*i)->geom.*(ptrs.dim) = linprog.get_variable((*i)->index*5+1); + } } } Layout::Constraint::Constraint(ConstraintType t, Slot &s): type(t), - target(s) + target(s), + spacing(-1) { } @@ -324,11 +448,141 @@ Layout::Slot::Slot(Layout &l, Widget &w): { vert_pack.gravity = 1; widget.signal_autosize_changed.connect(sigc::mem_fun(this, &Slot::autosize_changed)); + widget.signal_visibility_changed.connect(sigc::mem_fun(this, &Slot::visibility_changed)); + widget.autosize(); + autosize_geom = widget.get_geometry(); } void Layout::Slot::autosize_changed() { - layout.update(); + widget.autosize(); + autosize_geom = widget.get_geometry(); + + if(!widget.is_visible()) + return; + + // If the widget fits in the area it had, just leave it there. + if(autosize_geom.w<=geom.w && autosize_geom.h<=geom.h) + widget.set_geometry(geom); + else + { + layout.container->signal_autosize_changed.emit(); + layout.update(); + } +} + +void Layout::Slot::visibility_changed(bool v) +{ + layout.update_slot_indices(); + if(v) + { + layout.container->signal_autosize_changed.emit(); + layout.update(); + } +} + + +Layout::Loader::Loader(Layout &l, const WidgetMap &wm): + DataFile::ObjectLoader(l), + wdg_map(wm) +{ + add("column_spacing", &Loader::column_spacing); + add("margin", &Loader::margin); + add("row_spacing", &Loader::row_spacing); + add("spacing", &Loader::spacing); + add("widget", &Loader::widget); +} + +void Layout::Loader::column_spacing(unsigned s) +{ + obj.set_column_spacing(s); +} + +void Layout::Loader::margin() +{ + Sides sides; + load_sub(sides); + obj.set_margin(sides); +} + +void Layout::Loader::spacing(unsigned s) +{ + obj.set_spacing(s); +} + +void Layout::Loader::row_spacing(unsigned s) +{ + obj.set_row_spacing(s); +} + +void Layout::Loader::widget(const string &n) +{ + Widget &wdg = *get_item(wdg_map, n); + WidgetLoader ldr(obj, wdg, wdg_map); + load_sub_with(ldr); +} + + +Layout::WidgetLoader::WidgetLoader(Layout &l, Widget &w, const Layout::Loader::WidgetMap &wm): + layout(l), + widget(w), + wdg_map(wm) +{ + add("constraint", &WidgetLoader::constraint); + add("expand", &WidgetLoader::expand); + add("gravity", &WidgetLoader::gravity); +} + +void Layout::WidgetLoader::constraint(ConstraintType type, const string &n) +{ + Widget &target = *get_item(wdg_map, n); + layout.add_constraint(widget, type, target); +} + +void Layout::WidgetLoader::expand(bool h, bool v) +{ + layout.set_expand(widget, h, v); +} + +void Layout::WidgetLoader::gravity(int h, int v) +{ + layout.set_gravity(widget, h, v); +} + + +void operator>>(const LexicalConverter &conv, Layout::ConstraintType &ctype) +{ + const string &str = conv.get(); + if(str=="ABOVE") + ctype = Layout::ABOVE; + else if(str=="BELOW") + ctype = Layout::BELOW; + else if(str=="RIGHT_OF") + ctype = Layout::RIGHT_OF; + else if(str=="LEFT_OF") + ctype = Layout::LEFT_OF; + else if(str=="FAR_ABOVE") + ctype = Layout::FAR_ABOVE; + else if(str=="FAR_BELOW") + ctype = Layout::FAR_BELOW; + else if(str=="FAR_RIGHT_OF") + ctype = Layout::FAR_RIGHT_OF; + else if(str=="FAR_LEFT_OF") + ctype = Layout::FAR_LEFT_OF; + else if(str=="ALIGN_TOP") + ctype = Layout::ALIGN_TOP; + else if(str=="ALIGN_BOTTOM") + ctype = Layout::ALIGN_BOTTOM; + else if(str=="ALIGN_RIGHT") + ctype = Layout::ALIGN_RIGHT; + else if(str=="ALIGN_LEFT") + ctype = Layout::ALIGN_LEFT; + else if(str=="COPY_WIDTH") + ctype = Layout::COPY_WIDTH; + else if(str=="COPY_HEIGHT") + ctype = Layout::COPY_HEIGHT; + else + throw lexical_error(format("conversion of '%s' to ConstraintType", str)); } @@ -353,7 +607,7 @@ Layout::LinearProgram::Row Layout::LinearProgram::operator[](unsigned r) return Row(*this, r); } -Layout::LinearProgram::Row Layout::LinearProgram::get_object_row() +Layout::LinearProgram::Row Layout::LinearProgram::get_objective_row() { return Row(*this, 0); } @@ -365,8 +619,10 @@ float Layout::LinearProgram::get_variable(unsigned i) if(i+1>=n_columns) throw out_of_range("LinearProgram::get_variable"); - unsigned r = columns[i].basic; - return columns.back().values[r]; + if(unsigned r = columns[i].basic) + return columns.back().values[r]; + else + return 0; } bool Layout::LinearProgram::solve() @@ -378,30 +634,9 @@ bool Layout::LinearProgram::solve() objective variable is kept implicit, as it would never change during the execution of the algorithm. */ - /* Force all columns fully into existence and relocate objective row to - bottom in preparation of phase 1. A new objective row is calculated by - pricing out the constraint rows. */ - for(vector::iterator i=columns.begin(); i!=columns.end(); ++i) - { - float objective = i->values.front(); - i->values.front() = 0.0f; - for(vector::iterator j=i->values.begin(); j!=i->values.end(); ++j) - i->values.front() += *j; - i->values.resize(n_rows+1, 0.0f); - i->values.back() = objective; - } + prepare_columns(); - /* Create artificial variables for phase 1. This ensures that each row has - a basic variable associated with it. The original objective row already - contains the implicit objective variable, which is basic. */ - columns.resize(n_columns+n_rows-1); - columns.back() = columns[n_columns-1]; - columns[n_columns-1].values.clear(); - for(unsigned i=1; i obj_coeff(n_rows, 0.0f); + vector row_coeff(n_rows, 1.0f); + const vector &constants = columns.back().values; + for(vector::iterator i=columns.begin(); i!=columns.end(); ++i) + { + if(i->values.size()>=2 && i->values.back()!=0.0f && (constants.size()values.size() || i->values.back()*constants[i->values.size()-1]>=0.0f) && obj_coeff[i->values.size()-1]==0.0f) + { + bool basic = true; + for(unsigned j=1; (basic && j+1values.size()); ++j) + basic = (i->values[j]==0.0f); + if(basic) + { + i->basic = i->values.size()-1; + row_coeff[i->basic] = 1.0f/i->values.back(); + obj_coeff[i->basic] = -i->values.front(); + i->values.clear(); + } + } + } + + // Price out the newly-created basic variables. + for(vector::iterator i=columns.begin(); i!=columns.end(); ++i) + if(!i->values.empty()) + { + for(unsigned j=0; jvalues.size(); ++j) + { + i->values[j] *= row_coeff[j]; + i->values.front() += obj_coeff[j]*i->values[j]; + } + } +} + +void Layout::LinearProgram::add_artificial_variables() +{ + vector artificial_rows(n_rows-1); + for(unsigned i=0; i::iterator i=columns.begin(); i!=columns.end(); ++i) + if(i->basic) + artificial_rows[i->basic-1] = 0; + artificial_rows.erase(std::remove(artificial_rows.begin(), artificial_rows.end(), 0), artificial_rows.end()); + + /* Force all non-basic columns fully into existence and relocate objective + row to bottom in preparation of phase 1. A new objective row is calculated + by pricing out the constraint rows. */ + for(vector::iterator i=columns.begin(); i!=columns.end(); ++i) + if(!i->basic) + { + float objective = 0.0f; + if(!i->values.empty()) + { + objective = i->values.front(); + i->values.front() = 0.0f; + for(vector::iterator j=artificial_rows.begin(); j!=artificial_rows.end(); ++j) + if(*jvalues.size()) + i->values.front() += i->values[*j]; + } + i->values.resize(n_rows+1, 0.0f); + i->values.back() = objective; + } + + if(artificial_rows.empty()) + return; + + /* Create artificial variables for phase 1. This ensures that each row has + a basic variable associated with it. The original objective row already + contains the implicit objective variable, which is basic. */ + columns.resize(n_columns+artificial_rows.size()); + columns.back() = columns[n_columns-1]; + columns[n_columns-1].values.clear(); + for(unsigned i=0; ivalues.front() = i->values.back(); i->values.pop_back(); } - - // Solve the phase 2 problem. We already know it to be feasible. - while(pivot()) ; - - solved = true; - - return true; } unsigned Layout::LinearProgram::find_minimal_ratio(unsigned c) @@ -438,7 +774,7 @@ unsigned Layout::LinearProgram::find_minimal_ratio(unsigned c) /* Pick the row with the minimum ratio between the constant column and the pivot column. This ensures that when the pivot column is made basic, values in the constant column stay positive. - + The use of n_rows instead of the true size of the column is intentional, since the relocated objective row must be ignored in phase 1. */ float best = numeric_limits::infinity(); @@ -453,7 +789,7 @@ unsigned Layout::LinearProgram::find_minimal_ratio(unsigned c) row = i; } } - + return row; } @@ -472,14 +808,11 @@ void Layout::LinearProgram::make_basic_column(unsigned c, unsigned r) } float scale = columns[i].values[r]/columns[c].values[r]; - + + columns[i].values[r] = scale; for(unsigned j=0; j