X-Git-Url: https://git.lyx.org/gitweb/?a=blobdiff_plain;f=src%2FRow.cpp;h=9df572441cae0e735f0f6f9678c9f5c4c2d2467c;hb=2de30c62f8d671a8c8d4d52a6a7310e2c5ca84de;hp=58a23805f2ff35491df7fe56d265fcda40e85b52;hpb=b917c4e40f9f5cd3d101444600eddafcca54d6e3;p=lyx.git diff --git a/src/Row.cpp b/src/Row.cpp index 58a23805f2..9df572441c 100644 --- a/src/Row.cpp +++ b/src/Row.cpp @@ -37,23 +37,39 @@ using support::rtrim; using frontend::FontMetrics; +// Maximum length that a space can be stretched when justifying text +static double const MAX_SPACE_STRETCH = 1.5; //em + + int Row::Element::countSeparators() const { if (type != STRING) return 0; - // Consecutive spaces count as only one separator. - bool wasspace = false; - int nsep = 0; - for (size_t i = 0 ; i < str.size() ; ++i) { - if (str[i] == ' ') { - if (!wasspace) { - ++nsep; - wasspace = true; - } - } else - wasspace = false; - } - return nsep; + return count(str.begin(), str.end(), ' '); +} + + +int Row::Element::countExpanders() const +{ + if (type != STRING) + return 0; + return theFontMetrics(font).countExpanders(str); +} + + +int Row::Element::expansionAmount() const +{ + if (type != STRING) + return 0; + return countExpanders() * theFontMetrics(font).em(); +} + + +void Row::Element::setExtra(double extra_per_em) +{ + if (type != STRING) + return; + extra = extra_per_em * theFontMetrics(font).em(); } @@ -234,7 +250,7 @@ ostream & operator<<(ostream & os, Row::Element const & e) switch (e.type) { case Row::STRING: os << "STRING: `" << to_utf8(e.str) << "' (" - << e.countSeparators() << " sep.), "; + << e.countExpanders() << " expanders.), "; break; case Row::VIRTUAL: os << "VIRTUAL: `" << to_utf8(e.str) << "', "; @@ -311,13 +327,30 @@ int Row::countSeparators() const } -void Row::setSeparatorExtraWidth(double w) +bool Row::setExtraWidth(int w) { - separator = w; - iterator const end = elements_.end(); - for (iterator it = elements_.begin() ; it != end ; ++it) - if (it->type == Row::STRING) - it->extra = w; + if (w < 0) + // this is not expected to happen (but it does) + return false; + // amount of expansion: number of expanders time the em value for each + // string element + int exp_amount = 0; + for (Row::Element const & e : elements_) + exp_amount += e.expansionAmount(); + if (!exp_amount) + return false; + // extra length per expander per em + double extra_per_em = double(w) / exp_amount; + if (extra_per_em > MAX_SPACE_STRETCH) + // do not stretch more than MAX_SPACE_STRETCH em per expander + return false; + // add extra length to each element proportionally to its em. + for (Row::Element & e : elements_) + if (e.type == Row::STRING) + e.setExtra(extra_per_em); + // update row dimension + dim_.wid += w; + return true; }