2022-05-10 10:06:48 +00:00
|
|
|
// Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2020-11-26 16:31:50 +00:00
|
|
|
#include <QTest>
|
|
|
|
#include <QSignalSpy>
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
#include <qfontdatabase.h>
|
2011-09-16 12:07:05 +00:00
|
|
|
#include <qfontinfo.h>
|
|
|
|
#include <qfontmetrics.h>
|
2014-07-09 07:49:59 +00:00
|
|
|
#include <qtextlayout.h>
|
|
|
|
#include <private/qrawfont_p.h>
|
2021-11-10 07:32:01 +00:00
|
|
|
#include <private/qfont_p.h>
|
|
|
|
#include <private/qfontengine_p.h>
|
2013-07-11 14:23:02 +00:00
|
|
|
#include <qpa/qplatformfontdatabase.h>
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2022-03-18 11:53:51 +00:00
|
|
|
using namespace Qt::StringLiterals;
|
|
|
|
|
2019-04-29 20:45:14 +00:00
|
|
|
Q_LOGGING_CATEGORY(lcTests, "qt.text.tests")
|
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
class tst_QFontDatabase : public QObject
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
public:
|
|
|
|
tst_QFontDatabase();
|
|
|
|
|
|
|
|
private slots:
|
2015-12-03 15:59:30 +00:00
|
|
|
void initTestCase();
|
2011-04-27 10:05:43 +00:00
|
|
|
void styles_data();
|
|
|
|
void styles();
|
|
|
|
|
|
|
|
void fixedPitch_data();
|
|
|
|
void fixedPitch();
|
2019-04-29 20:45:14 +00:00
|
|
|
void systemFixedFont();
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2012-01-16 11:02:14 +00:00
|
|
|
#ifdef Q_OS_MAC
|
2011-04-27 10:05:43 +00:00
|
|
|
void trickyFonts_data();
|
|
|
|
void trickyFonts();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void widthTwoTimes_data();
|
|
|
|
void widthTwoTimes();
|
|
|
|
|
|
|
|
void addAppFont_data();
|
|
|
|
void addAppFont();
|
2013-07-08 14:50:31 +00:00
|
|
|
|
2018-11-09 16:56:08 +00:00
|
|
|
void addTwoAppFontsFromFamily();
|
|
|
|
|
2013-07-08 14:50:31 +00:00
|
|
|
void aliases();
|
2014-07-09 07:49:59 +00:00
|
|
|
void fallbackFonts();
|
2013-07-09 06:22:23 +00:00
|
|
|
|
2016-08-08 13:53:31 +00:00
|
|
|
void condensedFontWidth();
|
macOS: Fix stretch of condensed fonts with NoFontMerging
When showing a condensed font with AnyStretch, we should
not apply any stretch to the font (and if a stretch is
requested, we should calculate the actual stretch based
on how much the font is already stretched or condensed).
This usually works as expected, however, when using
QFont::NoFontMerging as the style strategy, we would
scale the glyph advances by the stretch of the font
since the calculated stretch of the font engine would
be overwritten by the actual stretch. In the case where
we use font merging, this would be done for the multi
engine, so we would not get the same issue, since the
text engine gets the stretch from the actual font engine
and this still has the original, calculated stretch
set.
Note on the test: We can't use testString() for this,
since it contains a space, and the test font does not
have a glyph for this, so we will end up merging a
different font for the space, giving us a slightly
different advance.
[ChangeLog][QtGui][macOS] Fixed display of condensed fonts
when NoFontMerging is in use.
Task-number: QTBUG-63800
Change-Id: I5b05e0dbfc8ae4b5d10c621ecb0975f53fda9483
Reviewed-by: Allan Sandfeld Jensen <allan.jensen@qt.io>
Reviewed-by: Lars Knoll <lars.knoll@qt.io>
2018-04-24 12:33:00 +00:00
|
|
|
void condensedFontWidthNoFontMerging();
|
2016-08-08 13:53:31 +00:00
|
|
|
void condensedFontMatching();
|
2015-08-31 20:13:08 +00:00
|
|
|
|
2016-10-24 13:53:06 +00:00
|
|
|
void rasterFonts();
|
|
|
|
void smoothFonts();
|
|
|
|
|
2018-12-05 11:34:52 +00:00
|
|
|
void registerOpenTypePreferredNamesSystem();
|
|
|
|
void registerOpenTypePreferredNamesApplication();
|
|
|
|
|
2021-09-28 15:00:59 +00:00
|
|
|
void stretchRespected();
|
|
|
|
|
2021-12-02 16:14:18 +00:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
void findCourier();
|
|
|
|
#endif
|
|
|
|
|
2013-07-09 06:22:23 +00:00
|
|
|
private:
|
2016-08-08 13:53:31 +00:00
|
|
|
QString m_ledFont;
|
|
|
|
QString m_testFont;
|
|
|
|
QString m_testFontCondensed;
|
2018-11-09 16:56:08 +00:00
|
|
|
QString m_testFontItalic;
|
2011-04-27 10:05:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
tst_QFontDatabase::tst_QFontDatabase()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2013-07-09 06:22:23 +00:00
|
|
|
void tst_QFontDatabase::initTestCase()
|
|
|
|
{
|
2016-08-08 13:53:31 +00:00
|
|
|
m_ledFont = QFINDTESTDATA("LED_REAL.TTF");
|
|
|
|
m_testFont = QFINDTESTDATA("testfont.ttf");
|
|
|
|
m_testFontCondensed = QFINDTESTDATA("testfont_condensed.ttf");
|
2018-11-09 16:56:08 +00:00
|
|
|
m_testFontItalic = QFINDTESTDATA("testfont_italic.ttf");
|
2016-08-08 13:53:31 +00:00
|
|
|
QVERIFY(!m_ledFont.isEmpty());
|
2013-07-09 06:22:23 +00:00
|
|
|
QVERIFY(!m_testFont.isEmpty());
|
2016-08-08 13:53:31 +00:00
|
|
|
QVERIFY(!m_testFontCondensed.isEmpty());
|
2018-11-09 16:56:08 +00:00
|
|
|
QVERIFY(!m_testFontItalic.isEmpty());
|
2013-07-09 06:22:23 +00:00
|
|
|
}
|
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
void tst_QFontDatabase::styles_data()
|
|
|
|
{
|
|
|
|
QTest::addColumn<QString>("font");
|
|
|
|
|
|
|
|
QTest::newRow( "data0" ) << QString( "Times New Roman" );
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QFontDatabase::styles()
|
|
|
|
{
|
|
|
|
QFETCH( QString, font );
|
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
QStringList styles = QFontDatabase::styles( font );
|
2011-04-27 10:05:43 +00:00
|
|
|
QStringList::Iterator it = styles.begin();
|
|
|
|
while ( it != styles.end() ) {
|
2014-01-13 14:48:44 +00:00
|
|
|
QString style = *it;
|
|
|
|
QString trimmed = style.trimmed();
|
|
|
|
++it;
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2014-01-13 14:48:44 +00:00
|
|
|
QCOMPARE(style, trimmed);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QFontDatabase::fixedPitch_data()
|
|
|
|
{
|
|
|
|
QTest::addColumn<QString>("font");
|
|
|
|
QTest::addColumn<bool>("fixedPitch");
|
|
|
|
|
|
|
|
QTest::newRow( "Times New Roman" ) << QString( "Times New Roman" ) << false;
|
|
|
|
QTest::newRow( "Arial" ) << QString( "Arial" ) << false;
|
|
|
|
QTest::newRow( "Andale Mono" ) << QString( "Andale Mono" ) << true;
|
|
|
|
QTest::newRow( "Courier" ) << QString( "Courier" ) << true;
|
|
|
|
QTest::newRow( "Courier New" ) << QString( "Courier New" ) << true;
|
2012-01-16 11:02:14 +00:00
|
|
|
#ifndef Q_OS_MAC
|
2011-04-27 10:05:43 +00:00
|
|
|
QTest::newRow( "Script" ) << QString( "Script" ) << false;
|
|
|
|
QTest::newRow( "Lucida Console" ) << QString( "Lucida Console" ) << true;
|
Support adding application fonts with QFontconfigDatabase
QFontDatabaseQPA should be able to reinitialize database and
reregister all the fonts when an application font is added, because
it will trigger db->invalidate().
Add cloneWithSize support to QFontEngineBox so that even no usable
font is found in QFontDatabase, QRawFont::setPixelSize can still
work (without making the result rawfont invalid).
Register application fonts with QFontconfigDatabase, the code is
adapted from QFontDatabaseX11.
Reenable QRawFont tests for QPA, these usages are now supported in
QPA.
Fix QStaticText tests, raster in QPA does support transformations.
Translate the text before ZAxix rotation so that it will be visible
in canvas.
Add back fixedPitch support to QPA, and fix QFontDatabase tests.
Fix QGlyphRun tests, ignore non-existence glyphs in alphaMap
locking.
Fix QFontMetrics tests.
Task-number: QTBUG-21415, QTBUG-20754, QTBUG-20977,
QTBUG-20976, QTBUG-20760, QTBUG-20759
Change-Id: I24aea7d6ec6b2ac6342134d1f2591327c23a692b
Reviewed-on: http://codereview.qt-project.org/5384
Reviewed-by: Qt Sanity Bot <qt_sanity_bot@ovi.com>
Reviewed-by: Eskil Abrahamsen Blomfeldt <eskil.abrahamsen-blomfeldt@nokia.com>
2011-09-22 12:23:29 +00:00
|
|
|
QTest::newRow( "DejaVu Sans" ) << QString( "DejaVu Sans" ) << false;
|
|
|
|
QTest::newRow( "DejaVu Sans Mono" ) << QString( "DejaVu Sans Mono" ) << true;
|
2011-04-27 10:05:43 +00:00
|
|
|
#else
|
|
|
|
QTest::newRow( "Menlo" ) << QString( "Menlo" ) << true;
|
|
|
|
QTest::newRow( "Monaco" ) << QString( "Monaco" ) << true;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QFontDatabase::fixedPitch()
|
|
|
|
{
|
|
|
|
QFETCH(QString, font);
|
|
|
|
QFETCH(bool, fixedPitch);
|
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
if (!QFontDatabase::families().contains(font))
|
2014-01-13 14:48:44 +00:00
|
|
|
QSKIP("Font not installed");
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
QCOMPARE(QFontDatabase::isFixedPitch(font), fixedPitch);
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QFont qfont(font);
|
|
|
|
QFontInfo fi(qfont);
|
|
|
|
QCOMPARE(fi.fixedPitch(), fixedPitch);
|
|
|
|
}
|
|
|
|
|
2019-04-29 20:45:14 +00:00
|
|
|
void tst_QFontDatabase::systemFixedFont() // QTBUG-54623
|
|
|
|
{
|
|
|
|
QFont font = QFontDatabase::systemFont(QFontDatabase::FixedFont);
|
|
|
|
QFontInfo fontInfo(font);
|
2020-08-06 06:47:37 +00:00
|
|
|
bool fdbSaysFixed = QFontDatabase::isFixedPitch(fontInfo.family(), fontInfo.styleName());
|
2019-04-29 20:45:14 +00:00
|
|
|
qCDebug(lcTests) << "system fixed font is" << font << "really fixed?" << fdbSaysFixed << fontInfo.fixedPitch();
|
|
|
|
QVERIFY(fdbSaysFixed);
|
|
|
|
QVERIFY(fontInfo.fixedPitch());
|
|
|
|
}
|
|
|
|
|
2012-01-16 11:02:14 +00:00
|
|
|
#ifdef Q_OS_MAC
|
2011-04-27 10:05:43 +00:00
|
|
|
void tst_QFontDatabase::trickyFonts_data()
|
|
|
|
{
|
|
|
|
QTest::addColumn<QString>("font");
|
|
|
|
|
|
|
|
QTest::newRow( "Geeza Pro" ) << QString( "Geeza Pro" );
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QFontDatabase::trickyFonts()
|
|
|
|
{
|
|
|
|
QFETCH(QString, font);
|
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
if (!QFontDatabase::families().contains(font))
|
2011-10-19 02:53:13 +00:00
|
|
|
QSKIP( "Font not installed");
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QFont qfont(font);
|
|
|
|
QFontInfo fi(qfont);
|
|
|
|
QCOMPARE(fi.family(), font);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void tst_QFontDatabase::widthTwoTimes_data()
|
|
|
|
{
|
|
|
|
QTest::addColumn<QString>("font");
|
|
|
|
QTest::addColumn<int>("pixelSize");
|
|
|
|
QTest::addColumn<QString>("text");
|
|
|
|
|
|
|
|
QTest::newRow("Arial") << QString("Arial") << 1000 << QString("Some text");
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QFontDatabase::widthTwoTimes()
|
|
|
|
{
|
|
|
|
QFETCH(QString, font);
|
|
|
|
QFETCH(int, pixelSize);
|
|
|
|
QFETCH(QString, text);
|
|
|
|
|
|
|
|
QFont f;
|
|
|
|
f.setFamily(font);
|
|
|
|
f.setPixelSize(pixelSize);
|
|
|
|
|
|
|
|
QFontMetrics fm(f);
|
2019-05-02 07:23:52 +00:00
|
|
|
int w1 = fm.horizontalAdvance(text, 0);
|
|
|
|
int w2 = fm.horizontalAdvance(text, 0);
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QCOMPARE(w1, w2);
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QFontDatabase::addAppFont_data()
|
|
|
|
{
|
|
|
|
QTest::addColumn<bool>("useMemoryFont");
|
|
|
|
QTest::newRow("font file") << false;
|
|
|
|
QTest::newRow("memory font") << true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QFontDatabase::addAppFont()
|
|
|
|
{
|
|
|
|
QFETCH(bool, useMemoryFont);
|
2011-09-16 12:07:05 +00:00
|
|
|
QSignalSpy fontDbChangedSpy(QGuiApplication::instance(), SIGNAL(fontDatabaseChanged()));
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
const QStringList oldFamilies = QFontDatabase::families();
|
2011-04-27 10:05:43 +00:00
|
|
|
QVERIFY(!oldFamilies.isEmpty());
|
|
|
|
|
|
|
|
fontDbChangedSpy.clear();
|
|
|
|
|
|
|
|
int id;
|
|
|
|
if (useMemoryFont) {
|
2016-08-08 13:53:31 +00:00
|
|
|
QFile fontfile(m_ledFont);
|
2011-04-27 10:05:43 +00:00
|
|
|
fontfile.open(QIODevice::ReadOnly);
|
|
|
|
QByteArray fontdata = fontfile.readAll();
|
|
|
|
QVERIFY(!fontdata.isEmpty());
|
|
|
|
id = QFontDatabase::addApplicationFontFromData(fontdata);
|
|
|
|
} else {
|
2016-08-08 13:53:31 +00:00
|
|
|
id = QFontDatabase::addApplicationFont(m_ledFont);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
#if defined(Q_OS_HPUX) && defined(QT_NO_FONTCONFIG)
|
|
|
|
// Documentation says that X11 systems that don't have fontconfig
|
|
|
|
// don't support application fonts.
|
|
|
|
QCOMPARE(id, -1);
|
|
|
|
return;
|
|
|
|
#endif
|
Port from container::count() and length() to size() - V5
This is a semantic patch using ClangTidyTransformator as in
qtbase/df9d882d41b741fef7c5beeddb0abe9d904443d8, but extended to
handle typedefs and accesses through pointers, too:
const std::string o = "object";
auto hasTypeIgnoringPointer = [](auto type) { return anyOf(hasType(type), hasType(pointsTo(type))); };
auto derivedFromAnyOfClasses = [&](ArrayRef<StringRef> classes) {
auto exprOfDeclaredType = [&](auto decl) {
return expr(hasTypeIgnoringPointer(hasUnqualifiedDesugaredType(recordType(hasDeclaration(decl))))).bind(o);
};
return exprOfDeclaredType(cxxRecordDecl(isSameOrDerivedFrom(hasAnyName(classes))));
};
auto renameMethod = [&] (ArrayRef<StringRef> classes,
StringRef from, StringRef to) {
return makeRule(cxxMemberCallExpr(on(derivedFromAnyOfClasses(classes)),
callee(cxxMethodDecl(hasName(from), parameterCountIs(0)))),
changeTo(cat(access(o, cat(to)), "()")),
cat("use '", to, "' instead of '", from, "'"));
};
renameMethod(<classes>, "count", "size");
renameMethod(<classes>, "length", "size");
except that the on() matcher has been replaced by one that doesn't
ignoreParens().
a.k.a qt-port-to-std-compatible-api V5 with config Scope: 'Container'.
Added two NOLINTNEXTLINEs in tst_qbitarray and tst_qcontiguouscache,
to avoid porting calls that explicitly test count().
Change-Id: Icfb8808c2ff4a30187e9935a51cad26987451c22
Reviewed-by: Ivan Solovev <ivan.solovev@qt.io>
Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org>
2022-09-30 12:09:04 +00:00
|
|
|
QCOMPARE(fontDbChangedSpy.size(), 1);
|
2011-10-07 00:55:06 +00:00
|
|
|
if (id == -1)
|
2011-10-19 02:53:13 +00:00
|
|
|
QSKIP("Skip the test since app fonts are not supported on this system");
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
const QStringList addedFamilies = QFontDatabase::applicationFontFamilies(id);
|
|
|
|
QVERIFY(!addedFamilies.isEmpty());
|
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
const QStringList newFamilies = QFontDatabase::families();
|
2011-04-27 10:05:43 +00:00
|
|
|
QVERIFY(!newFamilies.isEmpty());
|
Port from container::count() and length() to size() - V5
This is a semantic patch using ClangTidyTransformator as in
qtbase/df9d882d41b741fef7c5beeddb0abe9d904443d8, but extended to
handle typedefs and accesses through pointers, too:
const std::string o = "object";
auto hasTypeIgnoringPointer = [](auto type) { return anyOf(hasType(type), hasType(pointsTo(type))); };
auto derivedFromAnyOfClasses = [&](ArrayRef<StringRef> classes) {
auto exprOfDeclaredType = [&](auto decl) {
return expr(hasTypeIgnoringPointer(hasUnqualifiedDesugaredType(recordType(hasDeclaration(decl))))).bind(o);
};
return exprOfDeclaredType(cxxRecordDecl(isSameOrDerivedFrom(hasAnyName(classes))));
};
auto renameMethod = [&] (ArrayRef<StringRef> classes,
StringRef from, StringRef to) {
return makeRule(cxxMemberCallExpr(on(derivedFromAnyOfClasses(classes)),
callee(cxxMethodDecl(hasName(from), parameterCountIs(0)))),
changeTo(cat(access(o, cat(to)), "()")),
cat("use '", to, "' instead of '", from, "'"));
};
renameMethod(<classes>, "count", "size");
renameMethod(<classes>, "length", "size");
except that the on() matcher has been replaced by one that doesn't
ignoreParens().
a.k.a qt-port-to-std-compatible-api V5 with config Scope: 'Container'.
Added two NOLINTNEXTLINEs in tst_qbitarray and tst_qcontiguouscache,
to avoid porting calls that explicitly test count().
Change-Id: Icfb8808c2ff4a30187e9935a51cad26987451c22
Reviewed-by: Ivan Solovev <ivan.solovev@qt.io>
Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org>
2022-09-30 12:09:04 +00:00
|
|
|
QVERIFY(newFamilies.size() >= oldFamilies.size());
|
2011-04-27 10:05:43 +00:00
|
|
|
|
Port from container::count() and length() to size() - V5
This is a semantic patch using ClangTidyTransformator as in
qtbase/df9d882d41b741fef7c5beeddb0abe9d904443d8, but extended to
handle typedefs and accesses through pointers, too:
const std::string o = "object";
auto hasTypeIgnoringPointer = [](auto type) { return anyOf(hasType(type), hasType(pointsTo(type))); };
auto derivedFromAnyOfClasses = [&](ArrayRef<StringRef> classes) {
auto exprOfDeclaredType = [&](auto decl) {
return expr(hasTypeIgnoringPointer(hasUnqualifiedDesugaredType(recordType(hasDeclaration(decl))))).bind(o);
};
return exprOfDeclaredType(cxxRecordDecl(isSameOrDerivedFrom(hasAnyName(classes))));
};
auto renameMethod = [&] (ArrayRef<StringRef> classes,
StringRef from, StringRef to) {
return makeRule(cxxMemberCallExpr(on(derivedFromAnyOfClasses(classes)),
callee(cxxMethodDecl(hasName(from), parameterCountIs(0)))),
changeTo(cat(access(o, cat(to)), "()")),
cat("use '", to, "' instead of '", from, "'"));
};
renameMethod(<classes>, "count", "size");
renameMethod(<classes>, "length", "size");
except that the on() matcher has been replaced by one that doesn't
ignoreParens().
a.k.a qt-port-to-std-compatible-api V5 with config Scope: 'Container'.
Added two NOLINTNEXTLINEs in tst_qbitarray and tst_qcontiguouscache,
to avoid porting calls that explicitly test count().
Change-Id: Icfb8808c2ff4a30187e9935a51cad26987451c22
Reviewed-by: Ivan Solovev <ivan.solovev@qt.io>
Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org>
2022-09-30 12:09:04 +00:00
|
|
|
for (int i = 0; i < addedFamilies.size(); ++i) {
|
2017-03-30 17:29:18 +00:00
|
|
|
QString family = addedFamilies.at(i);
|
|
|
|
QVERIFY(newFamilies.contains(family));
|
|
|
|
QFont qfont(family);
|
|
|
|
QFontInfo fi(qfont);
|
|
|
|
QCOMPARE(fi.family(), family);
|
|
|
|
}
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QVERIFY(QFontDatabase::removeApplicationFont(id));
|
Port from container::count() and length() to size() - V5
This is a semantic patch using ClangTidyTransformator as in
qtbase/df9d882d41b741fef7c5beeddb0abe9d904443d8, but extended to
handle typedefs and accesses through pointers, too:
const std::string o = "object";
auto hasTypeIgnoringPointer = [](auto type) { return anyOf(hasType(type), hasType(pointsTo(type))); };
auto derivedFromAnyOfClasses = [&](ArrayRef<StringRef> classes) {
auto exprOfDeclaredType = [&](auto decl) {
return expr(hasTypeIgnoringPointer(hasUnqualifiedDesugaredType(recordType(hasDeclaration(decl))))).bind(o);
};
return exprOfDeclaredType(cxxRecordDecl(isSameOrDerivedFrom(hasAnyName(classes))));
};
auto renameMethod = [&] (ArrayRef<StringRef> classes,
StringRef from, StringRef to) {
return makeRule(cxxMemberCallExpr(on(derivedFromAnyOfClasses(classes)),
callee(cxxMethodDecl(hasName(from), parameterCountIs(0)))),
changeTo(cat(access(o, cat(to)), "()")),
cat("use '", to, "' instead of '", from, "'"));
};
renameMethod(<classes>, "count", "size");
renameMethod(<classes>, "length", "size");
except that the on() matcher has been replaced by one that doesn't
ignoreParens().
a.k.a qt-port-to-std-compatible-api V5 with config Scope: 'Container'.
Added two NOLINTNEXTLINEs in tst_qbitarray and tst_qcontiguouscache,
to avoid porting calls that explicitly test count().
Change-Id: Icfb8808c2ff4a30187e9935a51cad26987451c22
Reviewed-by: Ivan Solovev <ivan.solovev@qt.io>
Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org>
2022-09-30 12:09:04 +00:00
|
|
|
QCOMPARE(fontDbChangedSpy.size(), 2);
|
2011-04-27 10:05:43 +00:00
|
|
|
|
Port from container::count() and length() to size() - V5
This is a semantic patch using ClangTidyTransformator as in
qtbase/df9d882d41b741fef7c5beeddb0abe9d904443d8, but extended to
handle typedefs and accesses through pointers, too:
const std::string o = "object";
auto hasTypeIgnoringPointer = [](auto type) { return anyOf(hasType(type), hasType(pointsTo(type))); };
auto derivedFromAnyOfClasses = [&](ArrayRef<StringRef> classes) {
auto exprOfDeclaredType = [&](auto decl) {
return expr(hasTypeIgnoringPointer(hasUnqualifiedDesugaredType(recordType(hasDeclaration(decl))))).bind(o);
};
return exprOfDeclaredType(cxxRecordDecl(isSameOrDerivedFrom(hasAnyName(classes))));
};
auto renameMethod = [&] (ArrayRef<StringRef> classes,
StringRef from, StringRef to) {
return makeRule(cxxMemberCallExpr(on(derivedFromAnyOfClasses(classes)),
callee(cxxMethodDecl(hasName(from), parameterCountIs(0)))),
changeTo(cat(access(o, cat(to)), "()")),
cat("use '", to, "' instead of '", from, "'"));
};
renameMethod(<classes>, "count", "size");
renameMethod(<classes>, "length", "size");
except that the on() matcher has been replaced by one that doesn't
ignoreParens().
a.k.a qt-port-to-std-compatible-api V5 with config Scope: 'Container'.
Added two NOLINTNEXTLINEs in tst_qbitarray and tst_qcontiguouscache,
to avoid porting calls that explicitly test count().
Change-Id: Icfb8808c2ff4a30187e9935a51cad26987451c22
Reviewed-by: Ivan Solovev <ivan.solovev@qt.io>
Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org>
2022-09-30 12:09:04 +00:00
|
|
|
QVERIFY(QFontDatabase::families().size() <= oldFamilies.size());
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
2018-11-09 16:56:08 +00:00
|
|
|
void tst_QFontDatabase::addTwoAppFontsFromFamily()
|
|
|
|
{
|
|
|
|
int regularId = QFontDatabase::addApplicationFont(m_testFont);
|
|
|
|
if (regularId == -1)
|
|
|
|
QSKIP("Skip the test since app fonts are not supported on this system");
|
|
|
|
|
|
|
|
int italicId = QFontDatabase::addApplicationFont(m_testFontItalic);
|
|
|
|
QVERIFY(italicId != -1);
|
|
|
|
|
|
|
|
QVERIFY(!QFontDatabase::applicationFontFamilies(regularId).isEmpty());
|
|
|
|
QVERIFY(!QFontDatabase::applicationFontFamilies(italicId).isEmpty());
|
|
|
|
|
|
|
|
QString regularFontName = QFontDatabase::applicationFontFamilies(regularId).first();
|
|
|
|
QString italicFontName = QFontDatabase::applicationFontFamilies(italicId).first();
|
|
|
|
QCOMPARE(regularFontName, italicFontName);
|
|
|
|
|
2020-08-06 06:47:37 +00:00
|
|
|
QFont italicFont = QFontDatabase::font(italicFontName,
|
|
|
|
QString::fromLatin1("Italic"), 14);
|
2018-11-09 16:56:08 +00:00
|
|
|
QVERIFY(italicFont.italic());
|
|
|
|
|
|
|
|
QFontDatabase::removeApplicationFont(regularId);
|
|
|
|
QFontDatabase::removeApplicationFont(italicId);
|
|
|
|
}
|
|
|
|
|
2013-07-08 14:50:31 +00:00
|
|
|
void tst_QFontDatabase::aliases()
|
|
|
|
{
|
2020-11-02 18:48:25 +00:00
|
|
|
const QStringList families = QFontDatabase::families();
|
2013-07-08 14:50:31 +00:00
|
|
|
QVERIFY(!families.isEmpty());
|
2020-12-02 13:42:16 +00:00
|
|
|
|
|
|
|
QString firstFont;
|
|
|
|
for (int i = 0; i < families.size(); ++i) {
|
|
|
|
if (!families.at(i).contains('[')) {
|
|
|
|
firstFont = families.at(i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (firstFont.isEmpty())
|
|
|
|
QSKIP("Skipped because there are no unambiguous font families on the system.");
|
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
QVERIFY(QFontDatabase::hasFamily(firstFont));
|
2013-07-08 14:50:31 +00:00
|
|
|
const QString alias = QStringLiteral("AliasToFirstFont") + firstFont;
|
2020-11-02 18:48:25 +00:00
|
|
|
QVERIFY(!QFontDatabase::hasFamily(alias));
|
2013-07-11 14:23:02 +00:00
|
|
|
QPlatformFontDatabase::registerAliasToFontFamily(firstFont, alias);
|
2020-11-02 18:48:25 +00:00
|
|
|
QVERIFY(QFontDatabase::hasFamily(alias));
|
2013-07-08 14:50:31 +00:00
|
|
|
}
|
|
|
|
|
2014-07-09 07:49:59 +00:00
|
|
|
void tst_QFontDatabase::fallbackFonts()
|
|
|
|
{
|
|
|
|
QTextLayout layout;
|
|
|
|
QString s;
|
|
|
|
s.append(QChar(0x31));
|
|
|
|
s.append(QChar(0x05D0));
|
|
|
|
layout.setText(s);
|
|
|
|
layout.beginLayout();
|
|
|
|
layout.createLine();
|
|
|
|
layout.endLayout();
|
|
|
|
|
|
|
|
QList<QGlyphRun> runs = layout.glyphRuns(0, 1);
|
|
|
|
foreach (QGlyphRun run, runs) {
|
|
|
|
QRawFont rawFont = run.rawFont();
|
|
|
|
QVERIFY(rawFont.isValid());
|
|
|
|
|
|
|
|
QCOMPARE(run.glyphIndexes().size(), 1);
|
|
|
|
QVERIFY(run.glyphIndexes().at(0) != 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-08 13:53:31 +00:00
|
|
|
static QString testString()
|
2015-08-31 20:13:08 +00:00
|
|
|
{
|
2016-08-08 13:53:31 +00:00
|
|
|
return QStringLiteral("foo bar");
|
|
|
|
}
|
2015-08-31 20:13:08 +00:00
|
|
|
|
2021-09-28 15:00:59 +00:00
|
|
|
void tst_QFontDatabase::stretchRespected()
|
|
|
|
{
|
|
|
|
int italicId = QFontDatabase::addApplicationFont(m_testFontItalic);
|
|
|
|
QVERIFY(italicId != -1);
|
|
|
|
|
|
|
|
QVERIFY(!QFontDatabase::applicationFontFamilies(italicId).isEmpty());
|
|
|
|
|
|
|
|
QString italicFontName = QFontDatabase::applicationFontFamilies(italicId).first();
|
|
|
|
|
|
|
|
QFont italicFont = QFontDatabase::font(italicFontName,
|
|
|
|
QString::fromLatin1("Italic"), 14);
|
|
|
|
QVERIFY(italicFont.italic());
|
|
|
|
|
|
|
|
QFont italicStretchedFont = italicFont;
|
|
|
|
italicStretchedFont.setStretch( 400 );
|
|
|
|
|
|
|
|
QVERIFY(QFontMetricsF(italicFont).horizontalAdvance(QStringLiteral("foobar")) <
|
|
|
|
QFontMetricsF(italicStretchedFont).horizontalAdvance(QStringLiteral("foobar")));
|
|
|
|
|
|
|
|
QFontDatabase::removeApplicationFont(italicId);
|
|
|
|
}
|
|
|
|
|
macOS: Fix stretch of condensed fonts with NoFontMerging
When showing a condensed font with AnyStretch, we should
not apply any stretch to the font (and if a stretch is
requested, we should calculate the actual stretch based
on how much the font is already stretched or condensed).
This usually works as expected, however, when using
QFont::NoFontMerging as the style strategy, we would
scale the glyph advances by the stretch of the font
since the calculated stretch of the font engine would
be overwritten by the actual stretch. In the case where
we use font merging, this would be done for the multi
engine, so we would not get the same issue, since the
text engine gets the stretch from the actual font engine
and this still has the original, calculated stretch
set.
Note on the test: We can't use testString() for this,
since it contains a space, and the test font does not
have a glyph for this, so we will end up merging a
different font for the space, giving us a slightly
different advance.
[ChangeLog][QtGui][macOS] Fixed display of condensed fonts
when NoFontMerging is in use.
Task-number: QTBUG-63800
Change-Id: I5b05e0dbfc8ae4b5d10c621ecb0975f53fda9483
Reviewed-by: Allan Sandfeld Jensen <allan.jensen@qt.io>
Reviewed-by: Lars Knoll <lars.knoll@qt.io>
2018-04-24 12:33:00 +00:00
|
|
|
void tst_QFontDatabase::condensedFontWidthNoFontMerging()
|
|
|
|
{
|
|
|
|
int regularFontId = QFontDatabase::addApplicationFont(m_testFont);
|
|
|
|
int condensedFontId = QFontDatabase::addApplicationFont(m_testFontCondensed);
|
|
|
|
|
|
|
|
QVERIFY(!QFontDatabase::applicationFontFamilies(regularFontId).isEmpty());
|
|
|
|
QVERIFY(!QFontDatabase::applicationFontFamilies(condensedFontId).isEmpty());
|
|
|
|
|
|
|
|
QString regularFontName = QFontDatabase::applicationFontFamilies(regularFontId).first();
|
|
|
|
QString condensedFontName = QFontDatabase::applicationFontFamilies(condensedFontId).first();
|
|
|
|
|
|
|
|
QFont condensedFont1(condensedFontName);
|
|
|
|
if (regularFontName == condensedFontName)
|
|
|
|
condensedFont1.setStyleName(QStringLiteral("Condensed"));
|
|
|
|
condensedFont1.setStyleStrategy(QFont::PreferMatch);
|
|
|
|
|
|
|
|
QFont condensedFont2 = condensedFont1;
|
|
|
|
condensedFont2.setStyleStrategy(QFont::StyleStrategy(QFont::NoFontMerging | QFont::PreferMatch));
|
|
|
|
|
|
|
|
QCOMPARE(QFontMetricsF(condensedFont2).horizontalAdvance(QStringLiteral("foobar")),
|
|
|
|
QFontMetricsF(condensedFont1).horizontalAdvance(QStringLiteral("foobar")));
|
|
|
|
}
|
|
|
|
|
2016-08-08 13:53:31 +00:00
|
|
|
void tst_QFontDatabase::condensedFontWidth()
|
|
|
|
{
|
|
|
|
QFontDatabase::addApplicationFont(m_testFont);
|
|
|
|
QFontDatabase::addApplicationFont(m_testFontCondensed);
|
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
QVERIFY(QFontDatabase::hasFamily("QtBidiTestFont"));
|
|
|
|
if (!QFontDatabase::hasFamily("QtBidiTestFontCondensed"))
|
2016-08-08 13:53:31 +00:00
|
|
|
QSKIP("This platform doesn't support font sub-family names (QTBUG-55625)");
|
2015-08-31 20:13:08 +00:00
|
|
|
|
2016-08-08 13:53:31 +00:00
|
|
|
// Test we really get a condensed font, and a not renormalized one (QTBUG-48043):
|
|
|
|
QFont testFont("QtBidiTestFont");
|
|
|
|
QFont testFontCondensed("QtBidiTestFontCondensed");
|
|
|
|
QFontMetrics fmTF(testFont);
|
|
|
|
QFontMetrics fmTFC(testFontCondensed);
|
2017-08-02 09:39:01 +00:00
|
|
|
QVERIFY(fmTF.horizontalAdvance(testString()) > fmTFC.horizontalAdvance(testString()));
|
2015-08-31 20:13:08 +00:00
|
|
|
|
2016-08-08 13:53:31 +00:00
|
|
|
}
|
2015-08-31 20:13:08 +00:00
|
|
|
|
2016-08-08 13:53:31 +00:00
|
|
|
void tst_QFontDatabase::condensedFontMatching()
|
|
|
|
{
|
|
|
|
QFontDatabase::removeAllApplicationFonts();
|
|
|
|
QFontDatabase::addApplicationFont(m_testFontCondensed);
|
2020-11-02 18:48:25 +00:00
|
|
|
if (!QFontDatabase::hasFamily("QtBidiTestFont"))
|
2016-08-08 13:53:31 +00:00
|
|
|
QSKIP("This platform doesn't support preferred font family names (QTBUG-53478)");
|
|
|
|
QFontDatabase::addApplicationFont(m_testFont);
|
|
|
|
|
|
|
|
// Test we correctly get the condensed font using different font matching methods:
|
|
|
|
QFont tfcByStretch("QtBidiTestFont");
|
|
|
|
tfcByStretch.setStretch(QFont::Condensed);
|
|
|
|
QFont tfcByStyleName("QtBidiTestFont");
|
|
|
|
tfcByStyleName.setStyleName("Condensed");
|
|
|
|
|
2016-11-07 13:37:58 +00:00
|
|
|
#ifdef Q_OS_WIN
|
2021-11-11 12:43:08 +00:00
|
|
|
QFont f;
|
|
|
|
f.setStyleStrategy(QFont::NoFontMerging);
|
|
|
|
QFontPrivate *font_d = QFontPrivate::get(f);
|
2021-11-10 07:32:01 +00:00
|
|
|
if (font_d->engineForScript(QChar::Script_Common)->type() != QFontEngine::Freetype)
|
|
|
|
QEXPECT_FAIL("","No matching of sub-family by stretch on Windows", Continue);
|
2016-11-07 13:37:58 +00:00
|
|
|
#endif
|
|
|
|
|
2017-08-02 09:39:01 +00:00
|
|
|
QCOMPARE(QFontMetrics(tfcByStretch).horizontalAdvance(testString()),
|
|
|
|
QFontMetrics(tfcByStyleName).horizontalAdvance(testString()));
|
2016-08-08 13:53:31 +00:00
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
if (!QFontDatabase::hasFamily("QtBidiTestFontCondensed"))
|
2016-08-08 13:53:31 +00:00
|
|
|
QSKIP("This platform doesn't support font sub-family names (QTBUG-55625)");
|
|
|
|
|
|
|
|
QFont tfcBySubfamilyName("QtBidiTestFontCondensed");
|
2017-08-02 09:39:01 +00:00
|
|
|
QCOMPARE(QFontMetrics(tfcByStyleName).horizontalAdvance(testString()),
|
|
|
|
QFontMetrics(tfcBySubfamilyName).horizontalAdvance(testString()));
|
2015-08-31 20:13:08 +00:00
|
|
|
}
|
|
|
|
|
2016-10-24 13:53:06 +00:00
|
|
|
void tst_QFontDatabase::rasterFonts()
|
|
|
|
{
|
|
|
|
QFont font(QLatin1String("Fixedsys"), 1000);
|
|
|
|
QFontInfo fontInfo(font);
|
|
|
|
|
|
|
|
if (fontInfo.family() != font.family())
|
|
|
|
QSKIP("Fixedsys font not available.");
|
|
|
|
|
2020-08-06 06:47:37 +00:00
|
|
|
QVERIFY(!QFontDatabase::isSmoothlyScalable(font.family()));
|
2016-10-24 13:53:06 +00:00
|
|
|
QVERIFY(fontInfo.pointSize() != font.pointSize());
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QFontDatabase::smoothFonts()
|
|
|
|
{
|
|
|
|
QFont font(QLatin1String("Arial"), 1000);
|
|
|
|
QFontInfo fontInfo(font);
|
|
|
|
|
|
|
|
if (fontInfo.family() != font.family())
|
|
|
|
QSKIP("Arial font not available.");
|
|
|
|
|
|
|
|
// Smooth and bitmap scaling are mutually exclusive
|
2020-08-06 06:47:37 +00:00
|
|
|
QVERIFY(QFontDatabase::isSmoothlyScalable(font.family()));
|
|
|
|
QVERIFY(!QFontDatabase::isBitmapScalable(font.family()));
|
2016-10-24 13:53:06 +00:00
|
|
|
}
|
|
|
|
|
2018-12-05 11:34:52 +00:00
|
|
|
void tst_QFontDatabase::registerOpenTypePreferredNamesSystem()
|
|
|
|
{
|
|
|
|
// This font family was picked because it was the only one I had installed which showcased the
|
|
|
|
// problem
|
2020-11-02 18:48:25 +00:00
|
|
|
if (!QFontDatabase::hasFamily(QString::fromLatin1("Source Code Pro ExtraLight")))
|
2018-12-05 11:34:52 +00:00
|
|
|
QSKIP("Source Code Pro ExtraLight is not installed");
|
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
QStringList styles = QFontDatabase::styles(QString::fromLatin1("Source Code Pro"));
|
2018-12-05 11:34:52 +00:00
|
|
|
QVERIFY(styles.contains(QLatin1String("ExtraLight")));
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QFontDatabase::registerOpenTypePreferredNamesApplication()
|
|
|
|
{
|
|
|
|
int id = QFontDatabase::addApplicationFont(QString::fromLatin1(":/testfont_open.otf"));
|
|
|
|
if (id == -1)
|
|
|
|
QSKIP("Skip the test since app fonts are not supported on this system");
|
|
|
|
|
2020-11-02 18:48:25 +00:00
|
|
|
QStringList styles = QFontDatabase::styles(QString::fromLatin1("QtBidiTestFont"));
|
2018-12-05 11:34:52 +00:00
|
|
|
QVERIFY(styles.contains(QLatin1String("Open")));
|
|
|
|
|
|
|
|
QFontDatabase::removeApplicationFont(id);
|
|
|
|
}
|
|
|
|
|
2021-12-02 16:14:18 +00:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
void tst_QFontDatabase::findCourier()
|
|
|
|
{
|
2022-03-18 11:53:51 +00:00
|
|
|
QFont font = QFontDatabase::font(u"Courier"_s, u""_s, 16);
|
2021-12-02 16:14:18 +00:00
|
|
|
QFontInfo info(font);
|
2022-03-18 11:53:51 +00:00
|
|
|
QCOMPARE(info.family(), u"Courier New"_s);
|
2021-12-02 16:14:18 +00:00
|
|
|
QCOMPARE(info.pointSize(), 16);
|
|
|
|
|
|
|
|
font = QFontDatabase::font("Courier", "", 64);
|
|
|
|
info = font;
|
2022-03-18 11:53:51 +00:00
|
|
|
QCOMPARE(info.family(), u"Courier New"_s);
|
2021-12-02 16:14:18 +00:00
|
|
|
QCOMPARE(info.pointSize(), 64);
|
|
|
|
|
|
|
|
// By setting "PreferBitmap" we should get Courier itself.
|
|
|
|
font.setStyleStrategy(QFont::PreferBitmap);
|
|
|
|
info = font;
|
2022-03-18 11:53:51 +00:00
|
|
|
QCOMPARE(info.family(), u"Courier"_s);
|
2021-12-02 16:14:18 +00:00
|
|
|
// Which has an upper bound on point size
|
|
|
|
QCOMPARE(info.pointSize(), 19);
|
|
|
|
|
|
|
|
font.setStyleStrategy(QFont::PreferDefault);
|
|
|
|
info = font;
|
2022-03-18 11:53:51 +00:00
|
|
|
QCOMPARE(info.family(), u"Courier New"_s);
|
2021-12-02 16:14:18 +00:00
|
|
|
QCOMPARE(info.pointSize(), 64);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
QTEST_MAIN(tst_QFontDatabase)
|
|
|
|
#include "tst_qfontdatabase.moc"
|