mirror of
https://git.lyx.org/repos/lyx.git
synced 2024-11-17 00:30:55 +00:00
66fa801e74
The computation of length on screen depend in particular of the computation of the size of an em. Many places of the code used to rely on the width of the M character, which is not really correct: http://en.wikipedia.org/wiki/Em_%28typography%29 In digital typography, the best value to use is the point size of the font. * Implement FontMetrics::em(), which returns the value in pixels of the EM unit. Convert code to use it. * Introduce Length::inPixel(MetricsBase const &), which takes the textwidth and em information from the MetricsBase object. Convert code to use it. * Fix several places where Length::inPixel is used without a proper em value. * add mathed_font_em() helper function. It should eventually be removed like some other functions in MathSupport. * Add dummy implementation of FontMetrics to tex2lyx for linking purposes.
469 lines
8.6 KiB
C++
469 lines
8.6 KiB
C++
/**
|
|
* \file Length.cpp
|
|
* This file is part of LyX, the document processor.
|
|
* Licence details can be found in the file COPYING.
|
|
*
|
|
* \author Matthias Ettrich
|
|
* \author Lars Gullik Bjønnes
|
|
* \author Jean-Marc Lasgouttes
|
|
* \author Angus Leeming
|
|
* \author John Levon
|
|
* \author Dekel Tsur
|
|
*
|
|
* Full author contact details are available in file CREDITS.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "Length.h"
|
|
#include "LyXRC.h"
|
|
#include "MetricsInfo.h"
|
|
|
|
#include "frontends/FontMetrics.h"
|
|
|
|
#include "support/docstream.h"
|
|
|
|
#include <sstream>
|
|
#include <iomanip>
|
|
|
|
using namespace std;
|
|
|
|
namespace lyx {
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Length
|
|
//
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
Length::Length()
|
|
: val_(0), unit_(Length::UNIT_NONE)
|
|
{}
|
|
|
|
|
|
Length::Length(double v, Length::UNIT u)
|
|
: val_(v), unit_(u)
|
|
{}
|
|
|
|
|
|
Length::Length(string const & data)
|
|
: val_(0), unit_(Length::PT)
|
|
{
|
|
Length tmp;
|
|
|
|
if (!isValidLength(data, &tmp))
|
|
return; // should raise an exception
|
|
|
|
val_ = tmp.val_;
|
|
unit_ = tmp.unit_;
|
|
}
|
|
|
|
|
|
string const Length::asString() const
|
|
{
|
|
ostringstream os;
|
|
if (unit_ != UNIT_NONE)
|
|
os << val_ << unit_name[unit_]; // setw?
|
|
return os.str();
|
|
}
|
|
|
|
|
|
docstring const Length::asDocstring() const
|
|
{
|
|
odocstringstream os;
|
|
if (unit_ != UNIT_NONE)
|
|
os << val_ << unit_name[unit_]; // setw?
|
|
return os.str();
|
|
}
|
|
|
|
|
|
string const Length::asLatexString() const
|
|
{
|
|
ostringstream os;
|
|
switch (unit_) {
|
|
case PTW:
|
|
os << val_ / 100.0 << "\\textwidth";
|
|
break;
|
|
case PCW:
|
|
os << val_ / 100.0 << "\\columnwidth";
|
|
break;
|
|
case PPW:
|
|
os << val_ / 100.0 << "\\paperwidth";
|
|
break;
|
|
case PLW:
|
|
os << val_ / 100.0 << "\\linewidth";
|
|
break;
|
|
case PTH:
|
|
os << val_ / 100.0 << "\\textheight";
|
|
break;
|
|
case PPH:
|
|
os << val_ / 100.0 << "\\paperheight";
|
|
break;
|
|
case UNIT_NONE:
|
|
break;
|
|
default:
|
|
os << val_ << unit_name[unit_];
|
|
break;
|
|
}
|
|
return os.str();
|
|
}
|
|
|
|
|
|
string const Length::asHTMLString() const
|
|
{
|
|
ostringstream os;
|
|
switch (unit_) {
|
|
case PT:
|
|
case BP:
|
|
case DD:
|
|
// close enough
|
|
os << val_ << "pt";
|
|
break;
|
|
case MM:
|
|
case CM:
|
|
case PC:
|
|
case IN:
|
|
case EX:
|
|
case EM:
|
|
os << val_ << unit_name[unit_];
|
|
break;
|
|
case CC:
|
|
os << val_/12.0 << "pt";
|
|
break;
|
|
case MU:
|
|
os << val_/18.0 << "em";
|
|
break;
|
|
case PTW:
|
|
case PPW:
|
|
case PLW:
|
|
case PCW:
|
|
case PTH:
|
|
case PPH:
|
|
// what it's a percentage of probably won't make sense for HTML,
|
|
// so we'll assume people have chosen these appropriately
|
|
os << val_ << '%';
|
|
break;
|
|
case SP:
|
|
case UNIT_NONE:
|
|
break;
|
|
}
|
|
return os.str();
|
|
}
|
|
|
|
|
|
double Length::value() const
|
|
{
|
|
return val_;
|
|
}
|
|
|
|
|
|
Length::UNIT Length::unit() const
|
|
{
|
|
return unit_;
|
|
}
|
|
|
|
|
|
void Length::value(double v)
|
|
{
|
|
val_ = v;
|
|
}
|
|
|
|
|
|
void Length::unit(Length::UNIT u)
|
|
{
|
|
unit_ = u;
|
|
}
|
|
|
|
|
|
bool Length::zero() const
|
|
{
|
|
return val_ == 0.0;
|
|
}
|
|
|
|
|
|
bool Length::empty() const
|
|
{
|
|
return unit_ == Length::UNIT_NONE;
|
|
}
|
|
|
|
|
|
int Length::inPixels(int text_width, int em_width_base) const
|
|
{
|
|
// Zoom factor specified by user in percent
|
|
double const zoom = lyxrc.zoom / 100.0; // [percent]
|
|
|
|
// DPI setting for monitor: pixels/inch
|
|
double const dpi = lyxrc.dpi; // screen resolution [pixels/inch]
|
|
|
|
double const em_width = (em_width_base > 0)
|
|
? em_width_base
|
|
: 10*(dpi/72.27)*zoom;
|
|
// A different estimate for em_width is
|
|
// theFontMetrics(FontInfo(sane_font)).em()
|
|
// but this estimate might not be more accurate as the screen font
|
|
// is different then the latex font.
|
|
|
|
// Pixel values are scaled so that the ratio
|
|
// between lengths and font sizes on the screen
|
|
// is the same as on paper.
|
|
|
|
double result = 0.0;
|
|
|
|
switch (unit_) {
|
|
case Length::SP:
|
|
// Scaled point: sp = 1/65536 pt
|
|
result = zoom * dpi * val_
|
|
/ (72.27 * 65536); // 4736286.7
|
|
break;
|
|
case Length::PT:
|
|
// Point: 1 pt = 1/72.27 inch
|
|
result = zoom * dpi * val_
|
|
/ 72.27; // 72.27
|
|
break;
|
|
case Length::BP:
|
|
// Big point: 1 bp = 1/72 inch
|
|
result = zoom * dpi * val_
|
|
/ 72; // 72
|
|
break;
|
|
case Length::DD:
|
|
// Didot: 1157dd = 1238 pt?
|
|
result = zoom * dpi * val_
|
|
/ (72.27 / (0.376 * 2.845)); // 67.559735
|
|
break;
|
|
case Length::MM:
|
|
// Millimeter: 1 mm = 1/25.4 inch
|
|
result = zoom * dpi * val_
|
|
/ 25.4; // 25.4
|
|
break;
|
|
case Length::PC:
|
|
// Pica: 1 pc = 12 pt
|
|
result = zoom * dpi * val_
|
|
/ (72.27 / 12); // 6.0225
|
|
break;
|
|
case Length::CC:
|
|
// Cicero: 1 cc = 12 dd
|
|
result = zoom * dpi * val_
|
|
/ (72.27 / (12 * 0.376 * 2.845)); // 5.6299779
|
|
break;
|
|
case Length::CM:
|
|
// Centimeter: 1 cm = 1/2.54 inch
|
|
result = zoom * dpi * val_
|
|
/ 2.54; // 2.54
|
|
break;
|
|
case Length::IN:
|
|
// Inch
|
|
result = zoom * dpi * val_;
|
|
break;
|
|
case Length::EX:
|
|
// Ex: The height of an "x"
|
|
// 0.4305 is the ration between 1ex and 1em in cmr10
|
|
result = val_ * em_width * 0.4305;
|
|
break;
|
|
case Length::EM:
|
|
// Em: The width of an "m"
|
|
result = val_ * em_width;
|
|
break;
|
|
case Length::MU:
|
|
// math unit = 1/18em
|
|
result = val_ * em_width / 18;
|
|
break;
|
|
case Length::PCW: // Always % of workarea
|
|
case Length::PTW:
|
|
case Length::PLW:
|
|
result = val_ * text_width / 100;
|
|
break;
|
|
case Length::PPW:
|
|
// paperwidth/textwidth is 1.7 for A4 paper with default margins
|
|
result = val_ * text_width * 1.7 / 100;
|
|
break;
|
|
case Length::PTH:
|
|
result = val_ * text_width * 1.787 / 100;
|
|
break;
|
|
case Length::PPH:
|
|
result = val_ * text_width * 2.2 / 100;
|
|
break;
|
|
case Length::UNIT_NONE:
|
|
result = 0; // this cannot happen
|
|
break;
|
|
}
|
|
return static_cast<int>(result + ((result >= 0) ? 0.5 : -0.5));
|
|
}
|
|
|
|
|
|
int Length::inPixels(MetricsBase const & base) const
|
|
{
|
|
return inPixels(base.textwidth, theFontMetrics(base.font).em());
|
|
}
|
|
|
|
|
|
int Length::inBP() const
|
|
{
|
|
// return any Length value as a one with
|
|
// the PostScript point, called bp (big points)
|
|
double result = 0.0;
|
|
switch (unit_) {
|
|
case Length::CM:
|
|
// 1bp = 0.2835cm
|
|
result = val_ * 28.346;
|
|
break;
|
|
case Length::MM:
|
|
// 1bp = 0.02835mm
|
|
result = val_ * 2.8346;
|
|
break;
|
|
case Length::IN:
|
|
// 1pt = 1/72in
|
|
result = val_ * 72.0;
|
|
break;
|
|
default:
|
|
// no other than bp possible
|
|
result = val_;
|
|
break;
|
|
}
|
|
return static_cast<int>(result + 0.5);
|
|
}
|
|
|
|
|
|
Length::UNIT Length::defaultUnit()
|
|
{
|
|
return lyxrc.default_length_unit;
|
|
}
|
|
|
|
|
|
|
|
bool operator==(Length const & l1, Length const & l2)
|
|
{
|
|
return l1.value() == l2.value() && l1.unit() == l2.unit();
|
|
}
|
|
|
|
|
|
bool operator!=(Length const & l1, Length const & l2)
|
|
{
|
|
return !(l1 == l2);
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GlueLength
|
|
//
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
GlueLength::GlueLength(Length const & len)
|
|
: len_(len)
|
|
{}
|
|
|
|
|
|
GlueLength::GlueLength(Length const & len, Length const & plus,
|
|
Length const & minus)
|
|
: len_(len), plus_(plus), minus_(minus)
|
|
{}
|
|
|
|
|
|
GlueLength::GlueLength(string const & data)
|
|
{
|
|
isValidGlueLength(data, this);
|
|
}
|
|
|
|
|
|
string const GlueLength::asString() const
|
|
{
|
|
if (len_.empty())
|
|
return string();
|
|
|
|
ostringstream buffer;
|
|
|
|
buffer << len_.value();
|
|
|
|
if (plus_.zero() && minus_.zero()) {
|
|
buffer << unit_name[len_.unit()];
|
|
return buffer.str();
|
|
}
|
|
|
|
// just len and plus
|
|
if (minus_.zero()) {
|
|
if (len_.unit() != plus_.unit())
|
|
buffer << unit_name[len_.unit()];
|
|
buffer << '+' << plus_.value();
|
|
buffer << unit_name[plus_.unit()];
|
|
return buffer.str();
|
|
}
|
|
|
|
// just len and minus
|
|
if (plus_.zero()) {
|
|
if (len_.unit() != minus_.unit())
|
|
buffer << unit_name[len_.unit()];
|
|
buffer << '-' << minus_.value();
|
|
buffer << unit_name[minus_.unit()];
|
|
return buffer.str();
|
|
}
|
|
|
|
// ok, len, plus AND minus
|
|
|
|
// len+-
|
|
if (minus_ == plus_) {
|
|
if (len_.unit() != minus_.unit())
|
|
buffer << unit_name[len_.unit()];
|
|
buffer << "+-" << minus_.value();
|
|
buffer << unit_name[minus_.unit()];
|
|
return buffer.str();
|
|
}
|
|
|
|
// this is so rare a case, why bother minimising units ?
|
|
|
|
buffer << unit_name[len_.unit()];
|
|
buffer << '+' << plus_.value() << unit_name[plus_.unit()];
|
|
buffer << '-' << minus_.value() << unit_name[minus_.unit()];
|
|
|
|
return buffer.str();
|
|
}
|
|
|
|
|
|
string const GlueLength::asLatexString() const
|
|
{
|
|
ostringstream buffer;
|
|
// use Length::asLatexString() to handle also the percent lengths
|
|
buffer << len_.Length::asLatexString();
|
|
if (!plus_.zero())
|
|
buffer << " plus " << plus_.Length::asLatexString();
|
|
if (!minus_.zero())
|
|
buffer << " minus " << minus_.Length::asLatexString();
|
|
return buffer.str();
|
|
}
|
|
|
|
|
|
Length const & GlueLength::len() const
|
|
{
|
|
return len_;
|
|
}
|
|
|
|
|
|
Length const & GlueLength::plus() const
|
|
{
|
|
return plus_;
|
|
}
|
|
|
|
|
|
Length const & GlueLength::minus() const
|
|
{
|
|
return minus_;
|
|
}
|
|
|
|
|
|
bool operator==(GlueLength const & l1, GlueLength const & l2)
|
|
{
|
|
return l1.len() == l2.len()
|
|
&& l1.plus() == l2.plus()
|
|
&& l1.minus() == l2.minus();
|
|
}
|
|
|
|
|
|
bool operator!=(GlueLength const & l1, GlueLength const & l2)
|
|
{
|
|
return !(l1 == l2);
|
|
}
|
|
|
|
|
|
} // namespace lyx
|