2006-10-07 16:15:06 +00:00
|
|
|
/**
|
2007-04-26 03:53:02 +00:00
|
|
|
* \file GuiFontMetrics.cpp
|
2006-10-07 16:15:06 +00:00
|
|
|
* This file is part of LyX, the document processor.
|
|
|
|
* Licence details can be found in the file COPYING.
|
|
|
|
*
|
|
|
|
* \author unknown
|
|
|
|
* \author John Levon
|
|
|
|
*
|
|
|
|
* Full author contact details are available in file CREDITS.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "GuiFontMetrics.h"
|
|
|
|
|
|
|
|
#include "qt_helpers.h"
|
|
|
|
|
2007-04-26 04:41:58 +00:00
|
|
|
#include "Dimension.h"
|
2013-06-25 08:18:25 +02:00
|
|
|
#include "Language.h"
|
|
|
|
#include "LyXRC.h"
|
2006-10-07 16:15:06 +00:00
|
|
|
|
2008-02-14 08:49:45 +00:00
|
|
|
#include "insets/Inset.h"
|
|
|
|
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
#include "support/convert.h"
|
2008-04-30 08:26:40 +00:00
|
|
|
#include "support/lassert.h"
|
2007-11-08 00:09:58 +00:00
|
|
|
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
#define DISABLE_PMPROF
|
|
|
|
#include "support/pmprof.h"
|
|
|
|
|
|
|
|
#ifdef CACHE_SOME_METRICS
|
|
|
|
#include <QByteArray>
|
|
|
|
#endif
|
|
|
|
|
2007-12-12 10:16:00 +00:00
|
|
|
using namespace std;
|
2014-05-23 18:59:53 +02:00
|
|
|
using namespace lyx::support;
|
2006-10-07 16:15:06 +00:00
|
|
|
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
#ifdef CACHE_SOME_METRICS
|
|
|
|
namespace std {
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Argument-dependent lookup implies that this function shall be
|
|
|
|
* declared in the namespace of its argument. But this is std
|
|
|
|
* namespace, since lyx::docstring is just std::basic_string<wchar_t>.
|
|
|
|
*/
|
|
|
|
uint qHash(lyx::docstring const & s)
|
|
|
|
{
|
|
|
|
return qHash(QByteArray(reinterpret_cast<char const *>(s.data()),
|
|
|
|
s.size() * sizeof(lyx::docstring::value_type)));
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2006-10-07 16:15:06 +00:00
|
|
|
namespace lyx {
|
|
|
|
namespace frontend {
|
|
|
|
|
2010-11-17 22:00:42 +00:00
|
|
|
namespace {
|
2007-11-08 00:09:58 +00:00
|
|
|
/**
|
|
|
|
* Convert a UCS4 character into a QChar.
|
|
|
|
* This is a hack (it does only make sense for the common part of the UCS4
|
|
|
|
* and UTF16 encodings) and should not be used.
|
|
|
|
* This does only exist because of performance reasons (a real conversion
|
|
|
|
* using iconv is too slow on windows).
|
2010-11-17 22:00:42 +00:00
|
|
|
*
|
|
|
|
* This is no real conversion but a simple cast in reality. This is the reason
|
|
|
|
* why this works well for symbol fonts used in mathed too, even though
|
|
|
|
* these are not real ucs4 characters. These are codepoints in the
|
2012-12-15 13:02:40 +01:00
|
|
|
* computer modern fonts used, nothing unicode related.
|
2014-05-23 18:59:53 +02:00
|
|
|
* See comment in GuiPainter::text() for more explanation.
|
2014-05-07 12:06:56 +02:00
|
|
|
**/
|
2010-11-17 22:00:42 +00:00
|
|
|
inline QChar const ucs4_to_qchar(char_type const ucs4)
|
2007-11-08 00:09:58 +00:00
|
|
|
{
|
2013-04-25 17:27:10 -04:00
|
|
|
LATTEST(is_utf16(ucs4));
|
2007-11-08 00:09:58 +00:00
|
|
|
return QChar(static_cast<unsigned short>(ucs4));
|
|
|
|
}
|
2010-11-17 22:00:42 +00:00
|
|
|
} // anon namespace
|
2007-11-08 00:09:58 +00:00
|
|
|
|
|
|
|
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
/*
|
|
|
|
* Limit (strwidth|breakat)_cache_ size to 512kB of string data.
|
|
|
|
* Limit qtextlayout_cache_ size to 500 elements (we do not know the
|
|
|
|
* size of the QTextLayout objects anyway).
|
|
|
|
* Note that all these numbers are arbitrary.
|
|
|
|
*/
|
2015-11-09 10:11:57 +01:00
|
|
|
GuiFontMetrics::GuiFontMetrics(QFont const & font)
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
: font_(font), metrics_(font, 0)
|
|
|
|
#ifdef CACHE_METRICS_WIDTH
|
|
|
|
, strwidth_cache_(1 << 19)
|
|
|
|
#endif
|
|
|
|
#ifdef CACHE_METRICS_BREAKAT
|
|
|
|
, breakat_cache_(1 << 19)
|
|
|
|
#endif
|
|
|
|
#ifdef CACHE_METRICS_QTEXTLAYOUT
|
|
|
|
, qtextlayout_cache_(500)
|
|
|
|
#endif
|
2006-10-07 16:15:06 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int GuiFontMetrics::maxAscent() const
|
|
|
|
{
|
|
|
|
return metrics_.ascent();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int GuiFontMetrics::maxDescent() const
|
|
|
|
{
|
|
|
|
// We add 1 as the value returned by QT is different than X
|
|
|
|
// See http://doc.trolltech.com/2.3/qfontmetrics.html#200b74
|
|
|
|
return metrics_.descent() + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-03-26 16:55:19 +01:00
|
|
|
int GuiFontMetrics::em() const
|
|
|
|
{
|
|
|
|
return QFontInfo(font_).pixelSize();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-04-14 15:22:11 +02:00
|
|
|
int GuiFontMetrics::lineWidth() const
|
|
|
|
{
|
|
|
|
return metrics_.lineWidth();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int GuiFontMetrics::underlinePos() const
|
|
|
|
{
|
|
|
|
return metrics_.underlinePos();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int GuiFontMetrics::strikeoutPos() const
|
|
|
|
{
|
|
|
|
return metrics_.strikeOutPos();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-10-07 16:15:06 +00:00
|
|
|
int GuiFontMetrics::lbearing(char_type c) const
|
|
|
|
{
|
2007-05-31 12:48:42 +00:00
|
|
|
if (!is_utf16(c))
|
2015-04-14 15:22:11 +02:00
|
|
|
// FIXME: QFontMetrics::leftBearing does not support the
|
2007-05-31 12:48:42 +00:00
|
|
|
// full unicode range. Once it does, we could use:
|
2008-09-08 01:18:33 +00:00
|
|
|
//return metrics_.leftBearing(toqstr(docstring(1, c)));
|
2007-05-31 12:48:42 +00:00
|
|
|
return 0;
|
|
|
|
|
2006-10-07 16:15:06 +00:00
|
|
|
return metrics_.leftBearing(ucs4_to_qchar(c));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-08-13 13:56:54 +00:00
|
|
|
namespace {
|
|
|
|
int const outOfLimitMetric = -10000;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-10-07 16:15:06 +00:00
|
|
|
int GuiFontMetrics::rbearing(char_type c) const
|
|
|
|
{
|
2007-08-13 13:56:54 +00:00
|
|
|
int value = rbearing_cache_.value(c, outOfLimitMetric);
|
|
|
|
if (value != outOfLimitMetric)
|
|
|
|
return value;
|
|
|
|
|
|
|
|
// Qt rbearing is from the right edge of the char's width().
|
|
|
|
if (is_utf16(c)) {
|
|
|
|
QChar sc = ucs4_to_qchar(c);
|
|
|
|
value = width(c) - metrics_.rightBearing(sc);
|
2007-11-13 23:00:36 +00:00
|
|
|
} else {
|
|
|
|
// FIXME: QFontMetrics::leftBearing does not support the
|
2007-08-13 13:56:54 +00:00
|
|
|
// full unicode range. Once it does, we could use:
|
2008-09-08 01:18:33 +00:00
|
|
|
// metrics_.rightBearing(toqstr(docstring(1, c)));
|
2007-08-13 13:56:54 +00:00
|
|
|
value = width(c);
|
2007-11-13 23:00:36 +00:00
|
|
|
}
|
2007-08-13 13:56:54 +00:00
|
|
|
|
|
|
|
rbearing_cache_.insert(c, value);
|
|
|
|
|
|
|
|
return value;
|
2006-10-07 16:15:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-02-26 15:13:08 +00:00
|
|
|
int GuiFontMetrics::width(docstring const & s) const
|
2006-10-07 16:15:06 +00:00
|
|
|
{
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
PROFILE_THIS_BLOCK(width)
|
|
|
|
#ifdef CACHE_METRICS_WIDTH
|
|
|
|
int * pw = strwidth_cache_[s];
|
2015-11-09 10:11:57 +01:00
|
|
|
if (pw)
|
|
|
|
return *pw;
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
PROFILE_CACHE_MISS(width)
|
|
|
|
#endif
|
2017-01-20 10:32:40 +01:00
|
|
|
/* For some reason QMetrics::width returns a wrong value with Qt5
|
|
|
|
* with some arabic text. OTOH, QTextLayout is broken for single
|
2017-01-20 20:30:06 +01:00
|
|
|
* characters with null width (like \not in mathed). Also, as a
|
|
|
|
* safety measure, always use QMetrics::width with our math fonts.
|
2017-01-20 10:32:40 +01:00
|
|
|
*/
|
|
|
|
int w = 0;
|
|
|
|
if (s.length() == 1
|
|
|
|
#if QT_VERSION >= 0x040800
|
|
|
|
|| font_.styleName() == "LyX"
|
|
|
|
#endif
|
|
|
|
)
|
|
|
|
w = metrics_.width(toqstr(s));
|
|
|
|
else {
|
|
|
|
QTextLayout tl;
|
|
|
|
tl.setText(toqstr(s));
|
|
|
|
tl.setFont(font_);
|
|
|
|
tl.beginLayout();
|
|
|
|
QTextLine line = tl.createLine();
|
|
|
|
tl.endLayout();
|
|
|
|
w = int(line.naturalTextWidth());
|
|
|
|
}
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
#ifdef CACHE_METRICS_WIDTH
|
|
|
|
strwidth_cache_.insert(s, new int(w), s.size() * sizeof(char_type));
|
|
|
|
#endif
|
2006-10-27 21:27:03 +00:00
|
|
|
return w;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int GuiFontMetrics::width(QString const & ucs2) const
|
|
|
|
{
|
2007-05-31 12:48:42 +00:00
|
|
|
return width(qstring_to_ucs4(ucs2));
|
2006-10-07 16:15:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int GuiFontMetrics::signedWidth(docstring const & s) const
|
|
|
|
{
|
2006-11-08 11:32:33 +00:00
|
|
|
if (s.empty())
|
|
|
|
return 0;
|
|
|
|
|
2006-10-07 16:15:06 +00:00
|
|
|
if (s[0] == '-')
|
2007-02-26 15:13:08 +00:00
|
|
|
return -width(s.substr(1, s.size() - 1));
|
2006-10-07 16:15:06 +00:00
|
|
|
else
|
2007-02-26 15:13:08 +00:00
|
|
|
return width(s);
|
2006-10-07 16:15:06 +00:00
|
|
|
}
|
|
|
|
|
2015-12-07 10:32:34 +01:00
|
|
|
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
QTextLayout const *
|
|
|
|
GuiFontMetrics::getTextLayout(docstring const & s, bool const rtl,
|
|
|
|
double const wordspacing) const
|
2015-12-11 16:33:34 +01:00
|
|
|
{
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
PROFILE_THIS_BLOCK(getTextLayout)
|
|
|
|
QTextLayout * ptl;
|
|
|
|
#ifdef CACHE_METRICS_QTEXTLAYOUT
|
|
|
|
docstring const s_cache = s + (rtl ? "r" : "l") + convert<docstring>(wordspacing);
|
|
|
|
ptl = qtextlayout_cache_[s_cache];
|
|
|
|
if (!ptl) {
|
|
|
|
PROFILE_CACHE_MISS(getTextLayout)
|
|
|
|
#endif
|
|
|
|
ptl = new QTextLayout();
|
|
|
|
ptl->setCacheEnabled(true);
|
|
|
|
ptl->setText(toqstr(s));
|
|
|
|
QFont copy = font_;
|
|
|
|
copy.setWordSpacing(wordspacing);
|
|
|
|
ptl->setFont(copy);
|
2015-12-07 10:32:34 +01:00
|
|
|
// Note that both setFlags and the enums are undocumented
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
ptl->setFlags(rtl ? Qt::TextForceRightToLeft : Qt::TextForceLeftToRight);
|
|
|
|
ptl->beginLayout();
|
|
|
|
ptl->createLine();
|
|
|
|
ptl->endLayout();
|
|
|
|
#ifdef CACHE_METRICS_QTEXTLAYOUT
|
|
|
|
qtextlayout_cache_.insert(s_cache, ptl);
|
2015-12-07 10:32:34 +01:00
|
|
|
}
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
#endif
|
|
|
|
return ptl;
|
2014-05-14 17:46:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-07-19 01:22:10 +02:00
|
|
|
int GuiFontMetrics::pos2x(docstring const & s, int const pos, bool const rtl,
|
|
|
|
double const wordspacing) const
|
2014-05-14 17:46:43 +02:00
|
|
|
{
|
2016-12-06 16:06:03 +01:00
|
|
|
if (pos <= 0)
|
|
|
|
return rtl ? width(s) : 0;
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
QTextLayout const * tl = getTextLayout(s, rtl, wordspacing);
|
2016-11-02 17:42:56 +01:00
|
|
|
/* Since QString is UTF-16 and docstring is UCS-4, the offsets may
|
|
|
|
* not be the same when there are high-plan unicode characters
|
|
|
|
* (bug #10443).
|
|
|
|
*/
|
|
|
|
int const qpos = toqstr(s.substr(0, pos)).length();
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
return static_cast<int>(tl->lineForTextPosition(qpos).cursorToX(qpos));
|
2014-05-14 17:46:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-07-19 01:22:10 +02:00
|
|
|
int GuiFontMetrics::x2pos(docstring const & s, int & x, bool const rtl,
|
|
|
|
double const wordspacing) const
|
2014-05-14 17:46:43 +02:00
|
|
|
{
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
QTextLayout const * tl = getTextLayout(s, rtl, wordspacing);
|
|
|
|
int const qpos = tl->lineForTextPosition(0).xToCursor(x);
|
2014-05-14 17:46:43 +02:00
|
|
|
// correct x value to the actual cursor position.
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
x = static_cast<int>(tl->lineForTextPosition(0).cursorToX(qpos));
|
2016-11-02 17:42:56 +01:00
|
|
|
/* Since QString is UTF-16 and docstring is UCS-4, the offsets may
|
|
|
|
* not be the same when there are high-plan unicode characters
|
|
|
|
* (bug #10443).
|
|
|
|
*/
|
|
|
|
#if QT_VERSION < 0x040801 || QT_VERSION >= 0x050100
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
return qstring_to_ucs4(tl->text().left(qpos)).length();
|
2016-11-02 17:42:56 +01:00
|
|
|
#else
|
|
|
|
/* Due to QTBUG-25536 in 4.8.1 <= Qt < 5.1.0, the string returned
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
* by QString::toUcs4 (used by qstring_to_ucs4) may have wrong
|
2016-11-02 17:42:56 +01:00
|
|
|
* length. We work around the problem by trying all docstring
|
|
|
|
* positions until the right one is found. This is slow only if
|
|
|
|
* there are many high-plane Unicode characters. It might be
|
|
|
|
* worthwhile to implement a dichotomy search if this shows up
|
|
|
|
* under a profiler.
|
|
|
|
*/
|
|
|
|
int pos = min(qpos, static_cast<int>(s.length()));
|
|
|
|
while (pos >= 0 && toqstr(s.substr(0, pos)).length() != qpos)
|
|
|
|
--pos;
|
|
|
|
LASSERT(pos > 0 || qpos == 0, /**/);
|
2014-05-14 17:46:43 +02:00
|
|
|
return pos;
|
2016-11-02 17:42:56 +01:00
|
|
|
#endif
|
2014-05-14 17:46:43 +02:00
|
|
|
}
|
|
|
|
|
2006-10-07 16:15:06 +00:00
|
|
|
|
2016-08-13 19:03:02 +01:00
|
|
|
int GuiFontMetrics::countExpanders(docstring const & str) const
|
|
|
|
{
|
|
|
|
// Numbers of characters that are expanded by inter-word spacing. These
|
|
|
|
// characters are spaces, except for characters 09-0D which are treated
|
|
|
|
// specially. (From a combination of testing with the notepad found in qt's
|
|
|
|
// examples, and reading the source code.) In addition, consecutive spaces
|
|
|
|
// only count as one expander.
|
|
|
|
bool wasspace = false;
|
|
|
|
int nexp = 0;
|
|
|
|
for (char_type c : str)
|
|
|
|
if (c > 0x0d && QChar(c).isSpace()) {
|
|
|
|
if (!wasspace) {
|
|
|
|
++nexp;
|
|
|
|
wasspace = true;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
wasspace = false;
|
|
|
|
return nexp;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
pair<int, int> *
|
|
|
|
GuiFontMetrics::breakAt_helper(docstring const & s, int const x,
|
|
|
|
bool const rtl, bool const force) const
|
2015-07-19 01:22:10 +02:00
|
|
|
{
|
|
|
|
QTextLayout tl;
|
2016-01-19 12:02:43 +01:00
|
|
|
/* Qt will not break at a leading or trailing space, and we need
|
|
|
|
* that sometimes, see http://www.lyx.org/trac/ticket/9921.
|
|
|
|
*
|
|
|
|
* To work around the problem, we enclose the string between
|
|
|
|
* zero-width characters so that the QTextLayout algorithm will
|
|
|
|
* agree to break the text at these extremal spaces.
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
*/
|
2016-01-19 12:02:43 +01:00
|
|
|
// Unicode character ZERO WIDTH NO-BREAK SPACE
|
|
|
|
QChar const zerow_nbsp(0xfeff);
|
2016-11-02 17:42:56 +01:00
|
|
|
QString qs = zerow_nbsp + toqstr(s) + zerow_nbsp;
|
2016-10-23 20:52:01 +02:00
|
|
|
#if 1
|
|
|
|
/* Use unicode override characters to enforce drawing direction
|
|
|
|
* Source: http://www.iamcal.com/understanding-bidirectional-text/
|
|
|
|
*/
|
|
|
|
if (rtl)
|
|
|
|
// Right-to-left override: forces to draw text right-to-left
|
2016-11-02 17:42:56 +01:00
|
|
|
qs = QChar(0x202E) + qs;
|
2016-10-23 20:52:01 +02:00
|
|
|
else
|
|
|
|
// Left-to-right override: forces to draw text left-to-right
|
2016-11-02 17:42:56 +01:00
|
|
|
qs = QChar(0x202D) + qs;
|
2016-10-23 20:52:01 +02:00
|
|
|
int const offset = 2;
|
|
|
|
#else
|
|
|
|
// Alternative version that breaks with Qt5 and arabic text (#10436)
|
2015-07-19 01:22:10 +02:00
|
|
|
// Note that both setFlags and the enums are undocumented
|
|
|
|
tl.setFlags(rtl ? Qt::TextForceRightToLeft : Qt::TextForceLeftToRight);
|
2016-10-23 20:52:01 +02:00
|
|
|
int const offset = 1;
|
|
|
|
#endif
|
|
|
|
|
2016-11-02 17:42:56 +01:00
|
|
|
tl.setText(qs);
|
2016-10-23 20:52:01 +02:00
|
|
|
tl.setFont(font_);
|
2015-07-19 01:22:10 +02:00
|
|
|
QTextOption to;
|
2016-11-06 15:14:29 +01:00
|
|
|
to.setWrapMode(force ? QTextOption::WrapAtWordBoundaryOrAnywhere
|
|
|
|
: QTextOption::WordWrap);
|
2015-07-19 01:22:10 +02:00
|
|
|
tl.setTextOption(to);
|
|
|
|
tl.beginLayout();
|
|
|
|
QTextLine line = tl.createLine();
|
|
|
|
line.setLineWidth(x);
|
|
|
|
tl.createLine();
|
|
|
|
tl.endLayout();
|
2016-10-23 20:52:01 +02:00
|
|
|
if ((force && line.textLength() == offset) || int(line.naturalTextWidth()) > x)
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
return new pair<int, int>(-1, -1);
|
2016-11-02 17:42:56 +01:00
|
|
|
/* Since QString is UTF-16 and docstring is UCS-4, the offsets may
|
|
|
|
* not be the same when there are high-plan unicode characters
|
|
|
|
* (bug #10443).
|
|
|
|
*/
|
|
|
|
// The variable `offset' is here to account for the extra leading characters.
|
|
|
|
// The ending character zerow_nbsp has to be ignored if the line is complete.
|
|
|
|
int const qlen = line.textLength() - offset - (line.textLength() == qs.length());
|
|
|
|
#if QT_VERSION < 0x040801 || QT_VERSION >= 0x050100
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
int len = qstring_to_ucs4(qs.mid(offset, qlen)).length();
|
2016-11-02 17:42:56 +01:00
|
|
|
#else
|
|
|
|
/* Due to QTBUG-25536 in 4.8.1 <= Qt < 5.1.0, the string returned
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
* by QString::toUcs4 (used by qstring_to_ucs4) may have wrong
|
2016-11-02 17:42:56 +01:00
|
|
|
* length. We work around the problem by trying all docstring
|
|
|
|
* positions until the right one is found. This is slow only if
|
|
|
|
* there are many high-plane Unicode characters. It might be
|
|
|
|
* worthwhile to implement a dichotomy search if this shows up
|
|
|
|
* under a profiler.
|
|
|
|
*/
|
|
|
|
int len = min(qlen, static_cast<int>(s.length()));
|
|
|
|
while (len >= 0 && toqstr(s.substr(0, len)).length() != qlen)
|
|
|
|
--len;
|
|
|
|
LASSERT(len > 0 || qlen == 0, /**/);
|
Add caching for the QTextLayout objects we use
The QTextLayout handling is terribly slow on Qt 4.8.7, but some
caching has been added in Qt5 that makes it much faster. For some
reason, it is not that slow with Qt 4.8.1.
Caches are introduced for the three following methods
* width(doctring), controlled by CACHE_METRICS_WIDTH. This cache already
existed, but the code has been cleaned up
* getTextLayout, controlled by CACHE_METRICS_QTEXTLAYOUT (disabled by
default on Qt5, which does its own caching). This is used for pos2x
and x2pos and now for drawing of text too. The previous code used a
trivial caching scheme of the last used QTextLayout, but now they
are properly kept in a QCache. Moreover, the cacheEnabled() property
is enabled for these QTextLayout object (not sure what this does).
* breakAt, controlled by CACHE_METRICS_BREAKAT. This is the only user
of QTextLayout which did not have some kind of caching already.
For some weird reasons related to Argument-dependent look-up, the
qHash(docstring) function has to be defined in std namespace, since
lyx::docstring is actually std::basic_string<wchar_t>.
[NOTE: this version has profiling hooks, enabled by commenting out the line
#define DISABLE_PMPROF
that should eventually be removed.]
2016-07-05 14:06:22 +02:00
|
|
|
#endif
|
|
|
|
// The -1 is here to account for the leading zerow_nbsp.
|
|
|
|
return new pair<int, int>(len, int(line.naturalTextWidth()));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool GuiFontMetrics::breakAt(docstring & s, int & x, bool const rtl, bool const force) const
|
|
|
|
{
|
|
|
|
PROFILE_THIS_BLOCK(breakAt)
|
|
|
|
if (s.empty())
|
|
|
|
return false;
|
|
|
|
pair<int, int> * pp;
|
|
|
|
#ifdef CACHE_METRICS_BREAKAT
|
|
|
|
docstring const s_cache = s + convert<docstring>(x) + (rtl ? "r" : "l") + (force ? "f" : "w");
|
|
|
|
|
|
|
|
pp = breakat_cache_[s_cache];
|
|
|
|
if (!pp) {
|
|
|
|
PROFILE_CACHE_MISS(breakAt)
|
|
|
|
#endif
|
|
|
|
pp = breakAt_helper(s, x, rtl, force);
|
|
|
|
#ifdef CACHE_METRICS_BREAKAT
|
|
|
|
breakat_cache_.insert(s_cache, pp, s_cache.size() * sizeof(char_type));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if (pp->first == -1)
|
|
|
|
return false;
|
|
|
|
s = s.substr(0, pp->first);
|
|
|
|
x = pp->second;
|
|
|
|
#ifndef CACHE_METRICS_BREAKAT
|
|
|
|
delete pp;
|
2016-11-02 17:42:56 +01:00
|
|
|
#endif
|
2015-07-19 01:22:10 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-10-07 16:15:06 +00:00
|
|
|
void GuiFontMetrics::rectText(docstring const & str,
|
|
|
|
int & w, int & ascent, int & descent) const
|
|
|
|
{
|
2008-03-02 10:20:13 +00:00
|
|
|
static int const d = Inset::TEXT_TO_INSET_OFFSET / 2;
|
|
|
|
|
2008-02-14 08:49:45 +00:00
|
|
|
w = width(str) + Inset::TEXT_TO_INSET_OFFSET;
|
2006-10-07 16:15:06 +00:00
|
|
|
ascent = metrics_.ascent() + d;
|
|
|
|
descent = metrics_.descent() + d;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GuiFontMetrics::buttonText(docstring const & str,
|
|
|
|
int & w, int & ascent, int & descent) const
|
|
|
|
{
|
2008-02-14 09:44:12 +00:00
|
|
|
rectText(str, w, ascent, descent);
|
|
|
|
w += Inset::TEXT_TO_INSET_OFFSET;
|
2006-10-07 16:15:06 +00:00
|
|
|
}
|
|
|
|
|
2006-12-01 16:12:24 +00:00
|
|
|
|
2006-12-04 10:09:22 +00:00
|
|
|
Dimension const GuiFontMetrics::defaultDimension() const
|
2006-10-07 16:15:06 +00:00
|
|
|
{
|
2006-12-04 10:45:43 +00:00
|
|
|
return Dimension(0, maxAscent(), maxDescent());
|
2006-10-07 16:15:06 +00:00
|
|
|
}
|
|
|
|
|
2006-12-01 16:12:24 +00:00
|
|
|
|
2006-12-04 10:09:22 +00:00
|
|
|
Dimension const GuiFontMetrics::dimension(char_type c) const
|
2006-12-01 16:12:24 +00:00
|
|
|
{
|
2006-12-04 10:45:43 +00:00
|
|
|
return Dimension(width(c), ascent(c), descent(c));
|
2006-12-01 16:12:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-08-13 13:56:54 +00:00
|
|
|
GuiFontMetrics::AscendDescend const GuiFontMetrics::fillMetricsCache(
|
|
|
|
char_type c) const
|
2006-12-01 16:12:24 +00:00
|
|
|
{
|
2007-05-31 12:48:42 +00:00
|
|
|
QRect r;
|
|
|
|
if (is_utf16(c))
|
|
|
|
r = metrics_.boundingRect(ucs4_to_qchar(c));
|
|
|
|
else
|
2008-09-08 01:18:33 +00:00
|
|
|
r = metrics_.boundingRect(toqstr(docstring(1, c)));
|
2007-05-31 12:48:42 +00:00
|
|
|
|
2006-12-02 15:54:49 +00:00
|
|
|
AscendDescend ad = { -r.top(), r.bottom() + 1};
|
2006-12-01 16:12:24 +00:00
|
|
|
// We could as well compute the width but this is not really
|
|
|
|
// needed for now as it is done directly in width() below.
|
2006-12-02 15:54:49 +00:00
|
|
|
metrics_cache_.insert(c, ad);
|
2007-08-13 13:56:54 +00:00
|
|
|
|
|
|
|
return ad;
|
2006-12-01 16:12:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int GuiFontMetrics::width(char_type c) const
|
|
|
|
{
|
2007-08-13 13:56:54 +00:00
|
|
|
int value = width_cache_.value(c, outOfLimitMetric);
|
|
|
|
if (value != outOfLimitMetric)
|
|
|
|
return value;
|
|
|
|
|
|
|
|
if (is_utf16(c))
|
|
|
|
value = metrics_.width(ucs4_to_qchar(c));
|
|
|
|
else
|
2007-11-08 00:09:58 +00:00
|
|
|
value = metrics_.width(toqstr(docstring(1, c)));
|
2007-08-13 13:56:54 +00:00
|
|
|
|
|
|
|
width_cache_.insert(c, value);
|
2006-12-01 16:12:24 +00:00
|
|
|
|
2007-08-13 13:56:54 +00:00
|
|
|
return value;
|
2006-12-01 16:12:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int GuiFontMetrics::ascent(char_type c) const
|
|
|
|
{
|
2014-05-07 12:06:56 +02:00
|
|
|
static AscendDescend const outOfLimitAD =
|
2007-08-13 13:56:54 +00:00
|
|
|
{outOfLimitMetric, outOfLimitMetric};
|
|
|
|
AscendDescend value = metrics_cache_.value(c, outOfLimitAD);
|
|
|
|
if (value.ascent != outOfLimitMetric)
|
|
|
|
return value.ascent;
|
2006-12-01 16:12:24 +00:00
|
|
|
|
2007-08-13 13:56:54 +00:00
|
|
|
value = fillMetricsCache(c);
|
|
|
|
return value.ascent;
|
2006-10-07 16:15:06 +00:00
|
|
|
}
|
2006-12-01 16:12:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
int GuiFontMetrics::descent(char_type c) const
|
|
|
|
{
|
2014-05-07 12:06:56 +02:00
|
|
|
static AscendDescend const outOfLimitAD =
|
2007-08-13 13:56:54 +00:00
|
|
|
{outOfLimitMetric, outOfLimitMetric};
|
|
|
|
AscendDescend value = metrics_cache_.value(c, outOfLimitAD);
|
|
|
|
if (value.descent != outOfLimitMetric)
|
|
|
|
return value.descent;
|
2006-12-01 16:12:24 +00:00
|
|
|
|
2007-08-13 13:56:54 +00:00
|
|
|
value = fillMetricsCache(c);
|
|
|
|
return value.descent;
|
2006-10-07 16:15:06 +00:00
|
|
|
}
|
2006-12-01 16:12:24 +00:00
|
|
|
|
2007-11-13 23:00:36 +00:00
|
|
|
} // namespace frontend
|
|
|
|
} // namespace lyx
|