d8602ce58b
By depending on setFamilies() then we can be sure that font names with spaces, commas, quotes and so on are correctly handled without being misinterpreted. For now it will split on the comma when a string containing one is passed to setFamily. But from Qt 6.2 this will be removed to preserve the family string as a convenience function. [ChangeLog][QtGui][QFont] Indicated that setFamilies/families is preferred over setFamily/family to ensure that font family names are preserved when spaces, commas and so on are used in the name. Change-Id: Id3c1a4e827756a4c928fed461a4aafa5a0f06633 Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org> Reviewed-by: Eskil Abrahamsen Blomfeldt <eskil.abrahamsen-blomfeldt@qt.io>
1342 lines
45 KiB
C++
1342 lines
45 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** This file is part of the test suite of the Qt Toolkit.
|
|
**
|
|
** $QT_BEGIN_LICENSE:GPL-EXCEPT$
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3 as published by the Free Software
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
** included in the packaging of this file. Please review the following
|
|
** information to ensure the GNU General Public License requirements will
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
**
|
|
** $QT_END_LICENSE$
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include <QtTest/QtTest>
|
|
#include <private/qfontengine_p.h>
|
|
#include <private/qtextengine_p.h>
|
|
#include <qtextlayout.h>
|
|
#include <qfontdatabase.h>
|
|
#include <qfontinfo.h>
|
|
|
|
class tst_QTextScriptEngine : public QObject
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
tst_QTextScriptEngine();
|
|
|
|
private slots:
|
|
void initTestCase();
|
|
void devanagari_data();
|
|
void devanagari();
|
|
void bengali_data();
|
|
void bengali();
|
|
void gurmukhi_data();
|
|
void gurmukhi();
|
|
// gujarati missing
|
|
void oriya_data();
|
|
void oriya();
|
|
void tamil_data();
|
|
void tamil();
|
|
void telugu_data();
|
|
void telugu();
|
|
void kannada_data();
|
|
void kannada();
|
|
void malayalam_data();
|
|
void malayalam();
|
|
void sinhala_data();
|
|
void sinhala();
|
|
void khmer_data();
|
|
void khmer();
|
|
void linearB_data();
|
|
void linearB();
|
|
void greek_data();
|
|
void greek();
|
|
|
|
void mirroredChars_data();
|
|
void mirroredChars();
|
|
|
|
void controlInSyllable_qtbug14204();
|
|
void combiningMarks_qtbug15675_data();
|
|
void combiningMarks_qtbug15675();
|
|
|
|
void thaiIsolatedSaraAm();
|
|
void thaiWithZWJ();
|
|
void thaiMultipleVowels();
|
|
|
|
void shapingDisabledDevanagari();
|
|
void shapingDisabledLatin();
|
|
private:
|
|
bool haveTestFonts;
|
|
};
|
|
|
|
tst_QTextScriptEngine::tst_QTextScriptEngine()
|
|
: haveTestFonts(qgetenv("QT_HAVE_TEST_FONTS") == QByteArray("1"))
|
|
{
|
|
}
|
|
|
|
void tst_QTextScriptEngine::initTestCase()
|
|
{
|
|
if (!haveTestFonts) {
|
|
qWarning(
|
|
"Some of these tests depend on the internals of some test fonts which are not freely "
|
|
"distributable.\n"
|
|
"These tests will be skipped.\n"
|
|
"If you have the fonts available, set QT_HAVE_TEST_FONTS=1 in your environment and "
|
|
"run the test again."
|
|
);
|
|
}
|
|
}
|
|
|
|
struct ShapeTable {
|
|
unsigned short unicode[16];
|
|
unsigned short glyphs[16];
|
|
};
|
|
|
|
static void prepareShapingTest(const QFont &font, const ShapeTable *shape_table)
|
|
{
|
|
for (const ShapeTable *s = shape_table; s->unicode[0]; ++s) {
|
|
QByteArray testName = font.family().toLatin1() + ':';
|
|
QString string;
|
|
for (const ushort *u = s->unicode; *u; ++u) {
|
|
string.append(QChar(*u));
|
|
testName.append(" 0x" + QByteArray::number(*u, 16));
|
|
}
|
|
QList<ushort> glyphs;
|
|
for (const ushort *g = s->glyphs; *g; ++g)
|
|
glyphs.append(*g);
|
|
|
|
QTest::newRow(testName.constData()) << font << string << glyphs;
|
|
}
|
|
}
|
|
|
|
static void doShapingTests()
|
|
{
|
|
QFETCH(QFont, font);
|
|
QFETCH(QString, string);
|
|
QFETCH(QList<ushort>, glyphs);
|
|
|
|
QVERIFY(!string.isEmpty());
|
|
|
|
QTextLayout layout(string, font);
|
|
QTextEngine *e = layout.engine();
|
|
e->itemize();
|
|
e->shape(0);
|
|
|
|
QVERIFY(!e->layoutData->items.isEmpty());
|
|
if (e->fontEngine(e->layoutData->items[0])->type() == QFontEngine::Box)
|
|
QSKIP("OpenType support missing for script");
|
|
|
|
QCOMPARE(e->fontEngine(e->layoutData->items[0])->fontDef.families.first(), font.family());
|
|
|
|
ushort nglyphs = glyphs.size();
|
|
if (!glyphs.isEmpty()) {
|
|
QCOMPARE(e->layoutData->items[0].num_glyphs, nglyphs);
|
|
for (ushort i = 0; i < glyphs.size(); ++i) {
|
|
ushort glyph = (e->layoutData->glyphLayout.glyphs[i] & 0xffffff);
|
|
QCOMPARE(glyph, glyphs.at(i));
|
|
}
|
|
} else {
|
|
// decomposed shaping
|
|
if (string.at(0) == QChar(0x1fc1) || string.at(0) == QChar(0x1fed))
|
|
return;
|
|
if (string.normalized(QString::NormalizationForm_D).normalized(QString::NormalizationForm_C) != string)
|
|
return;
|
|
|
|
QTextLayout decomposedLayout(string.normalized(QString::NormalizationForm_D), font);
|
|
QTextEngine *de = decomposedLayout.engine();
|
|
de->itemize();
|
|
de->shape(0);
|
|
|
|
QCOMPARE(de->layoutData->items[0].num_glyphs, e->layoutData->items[0].num_glyphs);
|
|
for (ushort i = 0; i < nglyphs; ++i) {
|
|
ushort glyph = (e->layoutData->glyphLayout.glyphs[i] & 0xffffff);
|
|
ushort glyph2 = (de->layoutData->glyphLayout.glyphs[i] & 0xffffff);
|
|
QCOMPARE(glyph2, glyph);
|
|
}
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::devanagari_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Devanagari).contains("Raghindi")) {
|
|
QFont f("Raghindi");
|
|
const ShapeTable shape_table [] = {
|
|
// Ka
|
|
{ { 0x0915, 0x0 },
|
|
{ 0x0080, 0x0 } },
|
|
// Ka Halant
|
|
{ { 0x0915, 0x094d, 0x0 },
|
|
{ 0x0080, 0x0051, 0x0 } },
|
|
// Ka Halant Ka
|
|
{ { 0x0915, 0x094d, 0x0915, 0x0 },
|
|
{ 0x00c8, 0x0080, 0x0 } },
|
|
// Ka MatraI
|
|
{ { 0x0915, 0x093f, 0x0 },
|
|
{ 0x01d1, 0x0080, 0x0 } },
|
|
// Ra Halant Ka
|
|
{ { 0x0930, 0x094d, 0x0915, 0x0 },
|
|
{ 0x0080, 0x005b, 0x0 } },
|
|
// Ra Halant Ka MatraI
|
|
{ { 0x0930, 0x094d, 0x0915, 0x093f, 0x0 },
|
|
{ 0x01d1, 0x0080, 0x005b, 0x0 } },
|
|
// MatraI
|
|
{ { 0x093f, 0x0 },
|
|
{ 0x01d4, 0x029c, 0x0 } },
|
|
// Ka Nukta
|
|
{ { 0x0915, 0x093c, 0x0 },
|
|
{ 0x00a4, 0x0 } },
|
|
// Ka Halant Ra
|
|
{ { 0x0915, 0x094d, 0x0930, 0x0 },
|
|
{ 0x0110, 0x0 } },
|
|
// Ka Halant Ra Halant Ka
|
|
{ { 0x0915, 0x094d, 0x0930, 0x094d, 0x0915, 0x0 },
|
|
{ 0x0158, 0x0080, 0x0 } },
|
|
{ { 0x0930, 0x094d, 0x200d, 0x0 },
|
|
{ 0x00e2, 0x0 } },
|
|
{ { 0x0915, 0x094d, 0x0930, 0x094d, 0x200d, 0x0 },
|
|
{ 0x0158, 0x0 } },
|
|
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Raghindi");
|
|
}
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Devanagari).contains("Mangal")) {
|
|
QFont f("Mangal");
|
|
const ShapeTable shape_table [] = {
|
|
// Ka
|
|
{ { 0x0915, 0x0 },
|
|
{ 0x0080, 0x0 } },
|
|
// Ka Halant
|
|
{ { 0x0915, 0x094d, 0x0 },
|
|
{ 0x0080, 0x0051, 0x0 } },
|
|
// Ka Halant Ka
|
|
{ { 0x0915, 0x094d, 0x0915, 0x0 },
|
|
{ 0x00c8, 0x0080, 0x0 } },
|
|
// Ka MatraI
|
|
{ { 0x0915, 0x093f, 0x0 },
|
|
{ 0x01d1, 0x0080, 0x0 } },
|
|
// Ra Halant Ka
|
|
{ { 0x0930, 0x094d, 0x0915, 0x0 },
|
|
{ 0x0080, 0x005b, 0x0 } },
|
|
// Ra Halant Ka MatraI
|
|
{ { 0x0930, 0x094d, 0x0915, 0x093f, 0x0 },
|
|
{ 0x01d1, 0x0080, 0x005b, 0x0 } },
|
|
// MatraI
|
|
{ { 0x093f, 0x0 },
|
|
{ 0x01d4, 0x029c, 0x0 } },
|
|
// Ka Nukta
|
|
{ { 0x0915, 0x093c, 0x0 },
|
|
{ 0x00a4, 0x0 } },
|
|
// Ka Halant Ra
|
|
{ { 0x0915, 0x094d, 0x0930, 0x0 },
|
|
{ 0x0110, 0x0 } },
|
|
// Ka Halant Ra Halant Ka
|
|
{ { 0x0915, 0x094d, 0x0930, 0x094d, 0x0915, 0x0 },
|
|
{ 0x0158, 0x0080, 0x0 } },
|
|
|
|
{ { 0x92b, 0x94d, 0x930, 0x0 },
|
|
{ 0x125, 0x0 } },
|
|
{ { 0x92b, 0x93c, 0x94d, 0x930, 0x0 },
|
|
{ 0x149, 0x0 } },
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find mangal");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::devanagari()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::bengali_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Bengali).contains("Akaash")) {
|
|
QFont f("Akaash");
|
|
const ShapeTable shape_table [] = {
|
|
// Ka
|
|
{ { 0x0995, 0x0 },
|
|
{ 0x0151, 0x0 } },
|
|
// Ka Halant
|
|
{ { 0x0995, 0x09cd, 0x0 },
|
|
{ 0x0151, 0x017d, 0x0 } },
|
|
// Ka Halant Ka
|
|
{ { 0x0995, 0x09cd, 0x0995, 0x0 },
|
|
{ 0x019b, 0x0 } },
|
|
// Ka MatraI
|
|
{ { 0x0995, 0x09bf, 0x0 },
|
|
{ 0x0173, 0x0151, 0x0 } },
|
|
// Ra Halant Ka
|
|
{ { 0x09b0, 0x09cd, 0x0995, 0x0 },
|
|
{ 0x0151, 0x0276, 0x0 } },
|
|
// Ra Halant Ka MatraI
|
|
{ { 0x09b0, 0x09cd, 0x0995, 0x09bf, 0x0 },
|
|
{ 0x0173, 0x0151, 0x0276, 0x0 } },
|
|
// Ka Nukta
|
|
{ { 0x0995, 0x09bc, 0x0 },
|
|
{ 0x0151, 0x0171, 0x0 } },
|
|
// Ka Halant Ra
|
|
{ { 0x0995, 0x09cd, 0x09b0, 0x0 },
|
|
{ 0x01f4, 0x0 } },
|
|
// Ka Halant Ra Halant Ka
|
|
{ { 0x0995, 0x09cd, 0x09b0, 0x09cd, 0x0995, 0x0 },
|
|
{ 0x025c, 0x0276, 0x0151, 0x0 } },
|
|
// Ya + Halant
|
|
{ { 0x09af, 0x09cd, 0x0 },
|
|
{ 0x016a, 0x017d, 0x0 } },
|
|
// Da Halant Ya -> Da Ya-Phala
|
|
{ { 0x09a6, 0x09cd, 0x09af, 0x0 },
|
|
{ 0x01e5, 0x0 } },
|
|
// A Halant Ya -> A Ya-phala
|
|
{ { 0x0985, 0x09cd, 0x09af, 0x0 },
|
|
{ 0x0145, 0x01cf, 0x0 } },
|
|
// Na Halant Ka
|
|
{ { 0x09a8, 0x09cd, 0x0995, 0x0 },
|
|
{ 0x026f, 0x0151, 0x0 } },
|
|
// Na Halant ZWNJ Ka
|
|
{ { 0x09a8, 0x09cd, 0x200c, 0x0995, 0x0 },
|
|
{ 0x0164, 0x017d, 0x0151, 0x0 } },
|
|
// Na Halant ZWJ Ka
|
|
{ { 0x09a8, 0x09cd, 0x200d, 0x0995, 0x0 },
|
|
{ 0x026f, 0x0151, 0x0 } },
|
|
// Ka Halant ZWNJ Ka
|
|
{ { 0x0995, 0x09cd, 0x200c, 0x0995, 0x0 },
|
|
{ 0x0151, 0x017d, 0x0151, 0x0 } },
|
|
// Ka Halant ZWJ Ka
|
|
{ { 0x0995, 0x09cd, 0x200d, 0x0995, 0x0 },
|
|
{ 0x025c, 0x0151, 0x0 } },
|
|
// Na Halant Ra
|
|
{ { 0x09a8, 0x09cd, 0x09b0, 0x0 },
|
|
{ 0x0207, 0x0 } },
|
|
// Na Halant ZWNJ Ra
|
|
{ { 0x09a8, 0x09cd, 0x200c, 0x09b0, 0x0 },
|
|
{ 0x0164, 0x017d, 0x016b, 0x0 } },
|
|
// Na Halant ZWJ Ra
|
|
{ { 0x09a8, 0x09cd, 0x200d, 0x09b0, 0x0 },
|
|
{ 0x026f, 0x016b, 0x0 } },
|
|
// Na Halant Ba
|
|
{ { 0x09a8, 0x09cd, 0x09ac, 0x0 },
|
|
{ 0x022f, 0x0 } },
|
|
// Na Halant ZWNJ Ba
|
|
{ { 0x09a8, 0x09cd, 0x200c, 0x09ac, 0x0 },
|
|
{ 0x0164, 0x017d, 0x0167, 0x0 } },
|
|
// Na Halant ZWJ Ba
|
|
{ { 0x09a8, 0x09cd, 0x200d, 0x09ac, 0x0 },
|
|
{ 0x026f, 0x0167, 0x0 } },
|
|
// Na Halant Dha
|
|
{ { 0x09a8, 0x09cd, 0x09a7, 0x0 },
|
|
{ 0x01d3, 0x0 } },
|
|
// Na Halant ZWNJ Dha
|
|
{ { 0x09a8, 0x09cd, 0x200c, 0x09a7, 0x0 },
|
|
{ 0x0164, 0x017d, 0x0163, 0x0 } },
|
|
// Na Halant ZWJ Dha
|
|
{ { 0x09a8, 0x09cd, 0x200d, 0x09a7, 0x0 },
|
|
{ 0x026f, 0x0163, 0x0 } },
|
|
// Ra Halant Ka MatraAU
|
|
{ { 0x09b0, 0x09cd, 0x0995, 0x09cc, 0x0 },
|
|
{ 0x0179, 0x0151, 0x0276, 0x017e, 0x0 } },
|
|
// Ra Halant Ba Halant Ba
|
|
{ { 0x09b0, 0x09cd, 0x09ac, 0x09cd, 0x09ac, 0x0 },
|
|
{ 0x0232, 0x0276, 0x0 } },
|
|
{ { 0x9b0, 0x9cd, 0x995, 0x9be, 0x982, 0x0 },
|
|
{ 0x151, 0x276, 0x172, 0x143, 0x0 } },
|
|
{ { 0x9b0, 0x9cd, 0x995, 0x9be, 0x983, 0x0 },
|
|
{ 0x151, 0x276, 0x172, 0x144, 0x0 } },
|
|
// test decomposed two parts matras
|
|
{ { 0x995, 0x9c7, 0x9be, 0x0 },
|
|
{ 0x179, 0x151, 0x172, 0x0 } },
|
|
{ { 0x995, 0x9c7, 0x9d7, 0x0 },
|
|
{ 0x179, 0x151, 0x17e, 0x0 } },
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Akaash");
|
|
}
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Bengali).contains("Mukti Narrow")) {
|
|
QFont f("Mukti Narrow");
|
|
const ShapeTable shape_table [] = {
|
|
// Ka
|
|
{ { 0x0995, 0x0 },
|
|
{ 0x0073, 0x0 } },
|
|
// Ka Halant
|
|
{ { 0x0995, 0x09cd, 0x0 },
|
|
{ 0x00b9, 0x0 } },
|
|
// Ka Halant Ka
|
|
{ { 0x0995, 0x09cd, 0x0995, 0x0 },
|
|
{ 0x0109, 0x0 } },
|
|
// Ka MatraI
|
|
{ { 0x0995, 0x09bf, 0x0 },
|
|
{ 0x0095, 0x0073, 0x0 } },
|
|
// Ra Halant Ka
|
|
{ { 0x09b0, 0x09cd, 0x0995, 0x0 },
|
|
{ 0x0073, 0x00e1, 0x0 } },
|
|
// Ra Halant Ka MatraI
|
|
{ { 0x09b0, 0x09cd, 0x0995, 0x09bf, 0x0 },
|
|
{ 0x0095, 0x0073, 0x00e1, 0x0 } },
|
|
// MatraI
|
|
{ { 0x09bf, 0x0 },
|
|
{ 0x0095, 0x01c8, 0x0 } },
|
|
// Ka Nukta
|
|
{ { 0x0995, 0x09bc, 0x0 },
|
|
{ 0x0073, 0x0093, 0x0 } },
|
|
// Ka Halant Ra
|
|
{ { 0x0995, 0x09cd, 0x09b0, 0x0 },
|
|
{ 0x00e5, 0x0 } },
|
|
// Ka Halant Ra Halant Ka
|
|
{ { 0x995, 0x9cd, 0x9b0, 0x9cd, 0x995, 0x0 },
|
|
{ 0x234, 0x24e, 0x73, 0x0 } },
|
|
// Ya + Halant
|
|
{ { 0x09af, 0x09cd, 0x0 },
|
|
{ 0x00d2, 0x0 } },
|
|
// Da Halant Ya -> Da Ya-Phala
|
|
{ { 0x09a6, 0x09cd, 0x09af, 0x0 },
|
|
{ 0x0084, 0x00e2, 0x0 } },
|
|
// A Halant Ya -> A Ya-phala
|
|
{ { 0x0985, 0x09cd, 0x09af, 0x0 },
|
|
{ 0x0067, 0x00e2, 0x0 } },
|
|
// Na Halant Ka
|
|
{ { 0x09a8, 0x09cd, 0x0995, 0x0 },
|
|
{ 0x0188, 0x0 } },
|
|
// Na Halant ZWNJ Ka
|
|
{ { 0x9a8, 0x9cd, 0x200c, 0x995, 0x0 },
|
|
{ 0xcc, 0x73, 0x0 } },
|
|
// Na Halant ZWJ Ka
|
|
{ { 0x9a8, 0x9cd, 0x200d, 0x995, 0x0 },
|
|
{ 0x247, 0x73, 0x0 } },
|
|
// Ka Halant ZWNJ Ka
|
|
{ { 0x9a8, 0x9cd, 0x200d, 0x995, 0x0 },
|
|
{ 0x247, 0x73, 0x0 } },
|
|
// Ka Halant ZWJ Ka
|
|
{ { 0x9a8, 0x9cd, 0x200d, 0x995, 0x0 },
|
|
{ 0x247, 0x73, 0x0 } },
|
|
// Na Halant Ra
|
|
{ { 0x09a8, 0x09cd, 0x09b0, 0x0 },
|
|
{ 0x00f8, 0x0 } },
|
|
// Na Halant ZWNJ Ra
|
|
{ { 0x09a8, 0x09cd, 0x200c, 0x09b0, 0x0 },
|
|
{ 0xcc, 0x8d, 0x0 } },
|
|
// Na Halant ZWJ Ra
|
|
{ { 0x9a8, 0x9cd, 0x200d, 0x9b0, 0x0 },
|
|
{ 0x247, 0x8d, 0x0 } },
|
|
// Na Halant Ba
|
|
{ { 0x09a8, 0x09cd, 0x09ac, 0x0 },
|
|
{ 0x0139, 0x0 } },
|
|
// Na Halant ZWNJ Ba
|
|
{ { 0x9a8, 0x9cd, 0x200c, 0x9ac, 0x0 },
|
|
{ 0xcc, 0x89, 0x0 } },
|
|
// Na Halant ZWJ Ba
|
|
{ { 0x9a8, 0x9cd, 0x200d, 0x9ac, 0x0 },
|
|
{ 0x247, 0x89, 0x0 } },
|
|
// Na Halant Dha
|
|
{ { 0x09a8, 0x09cd, 0x09a7, 0x0 },
|
|
{ 0x0145, 0x0 } },
|
|
// Na Halant ZWNJ Dha
|
|
{ { 0x09a8, 0x09cd, 0x200c, 0x09a7, 0x0 },
|
|
{ 0xcc, 0x85, 0x0 } },
|
|
// Na Halant ZWJ Dha
|
|
{ { 0x09a8, 0x09cd, 0x200d, 0x09a7, 0x0 },
|
|
{ 0x247, 0x85, 0x0 } },
|
|
// Ra Halant Ka MatraAU
|
|
{ { 0x9b0, 0x9cd, 0x995, 0x9cc, 0x0 },
|
|
{ 0x232, 0x73, 0xe1, 0xa0, 0x0 } },
|
|
// Ra Halant Ba Halant Ba
|
|
{ { 0x09b0, 0x09cd, 0x09ac, 0x09cd, 0x09ac, 0x0 },
|
|
{ 0x013b, 0x00e1, 0x0 } },
|
|
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Mukti");
|
|
}
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Bengali).contains("Likhan")) {
|
|
QFont f("Likhan");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0x9a8, 0x9cd, 0x9af, 0x0 },
|
|
{ 0x1ca, 0x0 } },
|
|
{ { 0x09b8, 0x09cd, 0x09af, 0x0 },
|
|
{ 0x020e, 0x0 } },
|
|
{ { 0x09b6, 0x09cd, 0x09af, 0x0 },
|
|
{ 0x01f4, 0x0 } },
|
|
{ { 0x09b7, 0x09cd, 0x09af, 0x0 },
|
|
{ 0x01fe, 0x0 } },
|
|
{ { 0x09b0, 0x09cd, 0x09a8, 0x09cd, 0x200d, 0x0 },
|
|
{ 0x10b, 0x167, 0x0 } },
|
|
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Likhan");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::bengali()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::gurmukhi_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Gurmukhi).contains("Lohit Punjabi")) {
|
|
QFont f("Lohit Punjabi");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0xA15, 0xA4D, 0xa39, 0x0 },
|
|
{ 0x3b, 0x8b, 0x0 } },
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Lohit Punjabi");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::gurmukhi()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::oriya_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Oriya).contains("utkal")) {
|
|
QFont f("utkal");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0xb15, 0xb4d, 0xb24, 0xb4d, 0xb30, 0x0 },
|
|
{ 0x150, 0x125, 0x0 } },
|
|
{ { 0xb24, 0xb4d, 0xb24, 0xb4d, 0xb2c, 0x0 },
|
|
{ 0x151, 0x120, 0x0 } },
|
|
{ { 0xb28, 0xb4d, 0xb24, 0xb4d, 0xb2c, 0x0 },
|
|
{ 0x152, 0x120, 0x0 } },
|
|
{ { 0xb28, 0xb4d, 0xb24, 0xb4d, 0xb2c, 0x0 },
|
|
{ 0x152, 0x120, 0x0 } },
|
|
{ { 0xb28, 0xb4d, 0xb24, 0xb4d, 0xb30, 0x0 },
|
|
{ 0x176, 0x0 } },
|
|
{ { 0xb38, 0xb4d, 0xb24, 0xb4d, 0xb30, 0x0 },
|
|
{ 0x177, 0x0 } },
|
|
{ { 0xb28, 0xb4d, 0xb24, 0xb4d, 0xb30, 0xb4d, 0xb2f, 0x0 },
|
|
{ 0x176, 0x124, 0x0 } },
|
|
{ {0}, {0} }
|
|
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find utkal");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::oriya()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::tamil_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Tamil).contains("AkrutiTml1")) {
|
|
QFont f("AkrutiTml1");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0x0b95, 0x0bc2, 0x0 },
|
|
{ 0x004e, 0x0 } },
|
|
{ { 0x0bae, 0x0bc2, 0x0 },
|
|
{ 0x009e, 0x0 } },
|
|
{ { 0x0b9a, 0x0bc2, 0x0 },
|
|
{ 0x0058, 0x0 } },
|
|
{ { 0x0b99, 0x0bc2, 0x0 },
|
|
{ 0x0053, 0x0 } },
|
|
{ { 0x0bb0, 0x0bc2, 0x0 },
|
|
{ 0x00a8, 0x0 } },
|
|
{ { 0x0ba4, 0x0bc2, 0x0 },
|
|
{ 0x008e, 0x0 } },
|
|
{ { 0x0b9f, 0x0bc2, 0x0 },
|
|
{ 0x0062, 0x0 } },
|
|
{ { 0x0b95, 0x0bc6, 0x0 },
|
|
{ 0x000a, 0x0031, 0x0 } },
|
|
{ { 0x0b95, 0x0bca, 0x0 },
|
|
{ 0x000a, 0x0031, 0x0007, 0x0 } },
|
|
{ { 0x0b95, 0x0bc6, 0x0bbe, 0x0 },
|
|
{ 0x000a, 0x0031, 0x007, 0x0 } },
|
|
{ { 0x0b95, 0x0bcd, 0x0bb7, 0x0 },
|
|
{ 0x0049, 0x0 } },
|
|
{ { 0x0b95, 0x0bcd, 0x0bb7, 0x0bca, 0x0 },
|
|
{ 0x000a, 0x0049, 0x007, 0x0 } },
|
|
{ { 0x0b95, 0x0bcd, 0x0bb7, 0x0bc6, 0x0bbe, 0x0 },
|
|
{ 0x000a, 0x0049, 0x007, 0x0 } },
|
|
{ { 0x0b9f, 0x0bbf, 0x0 },
|
|
{ 0x005f, 0x0 } },
|
|
{ { 0x0b9f, 0x0bc0, 0x0 },
|
|
{ 0x0060, 0x0 } },
|
|
{ { 0x0bb2, 0x0bc0, 0x0 },
|
|
{ 0x00ab, 0x0 } },
|
|
{ { 0x0bb2, 0x0bbf, 0x0 },
|
|
{ 0x00aa, 0x0 } },
|
|
{ { 0x0bb0, 0x0bcd, 0x0 },
|
|
{ 0x00a4, 0x0 } },
|
|
{ { 0x0bb0, 0x0bbf, 0x0 },
|
|
{ 0x00a5, 0x0 } },
|
|
{ { 0x0bb0, 0x0bc0, 0x0 },
|
|
{ 0x00a6, 0x0 } },
|
|
{ { 0x0b83, 0x0 },
|
|
{ 0x0025, 0x0 } },
|
|
{ { 0x0b83, 0x0b95, 0x0 },
|
|
{ 0x0025, 0x0031, 0x0 } },
|
|
{ { 0xb95, 0xbc6, 0xbbe, 0x0 },
|
|
{ 0xa, 0x31, 0x7, 0x0 } },
|
|
{ { 0xb95, 0xbc7, 0xbbe, 0x0 },
|
|
{ 0xb, 0x31, 0x7, 0x0 } },
|
|
{ { 0xb95, 0xbc6, 0xbd7, 0x0 },
|
|
{ 0xa, 0x31, 0x40, 0x0 } },
|
|
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find AkrutiTml1");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::tamil()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::telugu_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Telugu).contains("Pothana2000")) {
|
|
QFont f("Pothana2000");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0xc15, 0xc4d, 0x0 },
|
|
{ 0xbb, 0x0 } },
|
|
{ { 0xc15, 0xc4d, 0xc37, 0x0 },
|
|
{ 0x4b, 0x0 } },
|
|
{ { 0xc15, 0xc4d, 0xc37, 0xc4d, 0x0 },
|
|
{ 0xe0, 0x0 } },
|
|
{ { 0xc15, 0xc4d, 0xc37, 0xc4d, 0xc23, 0x0 },
|
|
{ 0x4b, 0x91, 0x0 } },
|
|
{ { 0xc15, 0xc4d, 0xc30, 0x0 },
|
|
{ 0x5a, 0xb2, 0x0 } },
|
|
{ { 0xc15, 0xc4d, 0xc30, 0xc4d, 0x0 },
|
|
{ 0xbb, 0xb2, 0x0 } },
|
|
{ { 0xc15, 0xc4d, 0xc30, 0xc4d, 0xc15, 0x0 },
|
|
{ 0x5a, 0xb2, 0x83, 0x0 } },
|
|
{ { 0xc15, 0xc4d, 0xc30, 0xc3f, 0x0 },
|
|
{ 0xe2, 0xb2, 0x0 } },
|
|
{ { 0xc15, 0xc4d, 0xc15, 0xc48, 0x0 },
|
|
{ 0xe6, 0xb3, 0x83, 0x0 } },
|
|
{ { 0xc15, 0xc4d, 0xc30, 0xc48, 0x0 },
|
|
{ 0xe6, 0xb3, 0x9f, 0x0 } },
|
|
{ { 0xc15, 0xc46, 0xc56, 0x0 },
|
|
{ 0xe6, 0xb3, 0x0 } },
|
|
{ {0}, {0} }
|
|
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Pothana2000");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::telugu()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::kannada_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Kannada).contains("Sampige")) {
|
|
QFont f("Sampige");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0x0ca8, 0x0ccd, 0x0ca8, 0x0 },
|
|
{ 0x0049, 0x00ba, 0x0 } },
|
|
{ { 0x0ca8, 0x0ccd, 0x0ca1, 0x0 },
|
|
{ 0x0049, 0x00b3, 0x0 } },
|
|
{ { 0x0caf, 0x0cc2, 0x0 },
|
|
{ 0x004f, 0x005d, 0x0 } },
|
|
{ { 0x0ce0, 0x0 },
|
|
{ 0x006a, 0x0 } },
|
|
{ { 0x0ce6, 0x0ce7, 0x0ce8, 0x0 },
|
|
{ 0x006b, 0x006c, 0x006d, 0x0 } },
|
|
{ { 0x0cb5, 0x0ccb, 0x0 },
|
|
{ 0x015f, 0x0067, 0x0 } },
|
|
{ { 0x0cb0, 0x0ccd, 0x0cae, 0x0 },
|
|
{ 0x004e, 0x0082, 0x0 } },
|
|
{ { 0x0cb0, 0x0ccd, 0x0c95, 0x0 },
|
|
{ 0x0036, 0x0082, 0x0 } },
|
|
{ { 0x0c95, 0x0ccd, 0x0cb0, 0x0 },
|
|
{ 0x0036, 0x00c1, 0x0 } },
|
|
{ { 0x0cb0, 0x0ccd, 0x200d, 0x0c95, 0x0 },
|
|
{ 0x0050, 0x00a7, 0x0 } },
|
|
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Sampige");
|
|
}
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Kannada).contains("Tunga")) {
|
|
QFont f("Tunga");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0x0cb7, 0x0cc6, 0x0 },
|
|
{ 0x00b0, 0x006c, 0x0 } },
|
|
{ { 0x0cb7, 0x0ccd, 0x0 },
|
|
{ 0x0163, 0x0 } },
|
|
{ { 0xc95, 0xcbf, 0xcd5, 0x0 },
|
|
{ 0x114, 0x73, 0x0 } },
|
|
{ { 0xc95, 0xcc6, 0xcd5, 0x0 },
|
|
{ 0x90, 0x6c, 0x73, 0x0 } },
|
|
{ { 0xc95, 0xcc6, 0xcd6, 0x0 },
|
|
{ 0x90, 0x6c, 0x74, 0x0 } },
|
|
{ { 0xc95, 0xcc6, 0xcc2, 0x0 },
|
|
{ 0x90, 0x6c, 0x69, 0x0 } },
|
|
{ { 0xc95, 0xcca, 0xcd5, 0x0 },
|
|
{ 0x90, 0x6c, 0x69, 0x73, 0x0 } },
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Tunga");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::kannada()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::malayalam_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Malayalam).contains("AkrutiMal2")) {
|
|
QFont f("AkrutiMal2");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0x0d15, 0x0d46, 0x0 },
|
|
{ 0x005e, 0x0034, 0x0 } },
|
|
{ { 0x0d15, 0x0d47, 0x0 },
|
|
{ 0x005f, 0x0034, 0x0 } },
|
|
{ { 0x0d15, 0x0d4b, 0x0 },
|
|
{ 0x005f, 0x0034, 0x0058, 0x0 } },
|
|
{ { 0x0d15, 0x0d48, 0x0 },
|
|
{ 0x0060, 0x0034, 0x0 } },
|
|
{ { 0x0d15, 0x0d4a, 0x0 },
|
|
{ 0x005e, 0x0034, 0x0058, 0x0 } },
|
|
{ { 0x0d30, 0x0d4d, 0x0d15, 0x0 },
|
|
{ 0x009e, 0x0034, 0x0 } },
|
|
{ { 0x0d15, 0x0d4d, 0x0d35, 0x0 },
|
|
{ 0x0034, 0x007a, 0x0 } },
|
|
{ { 0x0d15, 0x0d4d, 0x0d2f, 0x0 },
|
|
{ 0x0034, 0x00a2, 0x0 } },
|
|
{ { 0x0d1f, 0x0d4d, 0x0d1f, 0x0 },
|
|
{ 0x0069, 0x0 } },
|
|
{ { 0x0d26, 0x0d4d, 0x0d26, 0x0 },
|
|
{ 0x0074, 0x0 } },
|
|
{ { 0x0d30, 0x0d4d, 0x0 },
|
|
{ 0x009e, 0x0 } },
|
|
{ { 0x0d30, 0x0d4d, 0x200c, 0x0 },
|
|
{ 0x009e, 0x0 } },
|
|
{ { 0x0d30, 0x0d4d, 0x200d, 0x0 },
|
|
{ 0x009e, 0x0 } },
|
|
{ { 0xd15, 0xd46, 0xd3e, 0x0 },
|
|
{ 0x5e, 0x34, 0x58, 0x0 } },
|
|
{ { 0xd15, 0xd47, 0xd3e, 0x0 },
|
|
{ 0x5f, 0x34, 0x58, 0x0 } },
|
|
{ { 0xd15, 0xd46, 0xd57, 0x0 },
|
|
{ 0x5e, 0x34, 0x65, 0x0 } },
|
|
{ { 0xd15, 0xd57, 0x0 },
|
|
{ 0x34, 0x65, 0x0 } },
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find AkrutiMal2");
|
|
}
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Malayalam).contains("Rachana")) {
|
|
QFont f("Rachana");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0xd37, 0xd4d, 0xd1f, 0xd4d, 0xd30, 0xd40, 0x0 },
|
|
{ 0x385, 0xa3, 0x0 } },
|
|
{ { 0xd2f, 0xd4d, 0xd15, 0xd4d, 0xd15, 0xd41, 0x0 },
|
|
{ 0x2ff, 0x0 } },
|
|
{ { 0xd33, 0xd4d, 0xd33, 0x0 },
|
|
{ 0x3f8, 0x0 } },
|
|
{ { 0xd2f, 0xd4d, 0xd15, 0xd4d, 0xd15, 0xd41, 0x0 },
|
|
{ 0x2ff, 0x0 } },
|
|
{ { 0xd30, 0xd4d, 0x200d, 0xd35, 0xd4d, 0xd35, 0x0 },
|
|
{ 0xf3, 0x350, 0x0 } },
|
|
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Rachana");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::malayalam()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::sinhala_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Sinhala).contains("Malithi Web")) {
|
|
QFont f("Malithi Web");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0xd9a, 0xdd9, 0xdcf, 0x0 },
|
|
{ 0x4a, 0x61, 0x42, 0x0 } },
|
|
{ { 0xd9a, 0xdd9, 0xddf, 0x0 },
|
|
{ 0x4a, 0x61, 0x50, 0x0 } },
|
|
{ { 0xd9a, 0xdd9, 0xdca, 0x0 },
|
|
{ 0x4a, 0x62, 0x0 } },
|
|
{ { 0xd9a, 0xddc, 0xdca, 0x0 },
|
|
{ 0x4a, 0x61, 0x42, 0x41, 0x0 } },
|
|
{ { 0xd9a, 0xdda, 0x0 },
|
|
{ 0x4a, 0x62, 0x0 } },
|
|
{ { 0xd9a, 0xddd, 0x0 },
|
|
{ 0x4a, 0x61, 0x42, 0x41, 0x0 } },
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Malithi Web");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::sinhala()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::khmer_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Khmer).contains("Khmer OS")) {
|
|
QFont f("Khmer OS");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0x179a, 0x17cd, 0x0 },
|
|
{ 0x24c, 0x27f, 0x0 } },
|
|
{ { 0x179f, 0x17c5, 0x0 },
|
|
{ 0x273, 0x203, 0x0 } },
|
|
{ { 0x1790, 0x17d2, 0x1784, 0x17c3, 0x0 },
|
|
{ 0x275, 0x242, 0x182, 0x0 } },
|
|
{ { 0x179a, 0x0 },
|
|
{ 0x24c, 0x0 } },
|
|
{ { 0x1781, 0x17d2, 0x1798, 0x17c2, 0x0 },
|
|
{ 0x274, 0x233, 0x197, 0x0 } },
|
|
{ { 0x1798, 0x17b6, 0x0 },
|
|
{ 0x1cb, 0x0 } },
|
|
{ { 0x179a, 0x17b8, 0x0 },
|
|
{ 0x24c, 0x26a, 0x0 } },
|
|
{ { 0x1787, 0x17b6, 0x0 },
|
|
{ 0x1ba, 0x0 } },
|
|
{ { 0x1798, 0x17d2, 0x1796, 0x17bb, 0x0 },
|
|
{ 0x24a, 0x195, 0x26d, 0x0 } },
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Khmer OS");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::khmer()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::linearB_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Any).contains("Penuturesu")) {
|
|
QFont f("Penuturesu");
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0xd800, 0xdc01, 0xd800, 0xdc02, 0xd800, 0xdc03, 0 },
|
|
{ 0x5, 0x6, 0x7, 0 } },
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find Penuturesu");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::linearB()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::greek_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
QTest::addColumn<QList<ushort>>("glyphs");
|
|
|
|
if (!haveTestFonts)
|
|
QSKIP("Test fonts are not available");
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Any).contains("DejaVu Sans")) {
|
|
QFont f("DejaVu Sans");
|
|
for (int uc = 0x1f00; uc <= 0x1fff; ++uc) {
|
|
QString string;
|
|
string.append(QChar(uc));
|
|
QByteArray testName = f.family().toLatin1() + ": 0x" + QByteArray::number(uc, 16);
|
|
QTest::newRow(testName.constData()) << f << string << QList<ushort>();
|
|
}
|
|
} else
|
|
QSKIP("couldn't find DejaVu Sans");
|
|
}
|
|
|
|
{
|
|
if (QFontDatabase::families(QFontDatabase::Any).contains("SBL Greek")) {
|
|
QFont f("SBL Greek");
|
|
for (int uc = 0x1f00; uc <= 0x1fff; ++uc) {
|
|
QString string;
|
|
string.append(QChar(uc));
|
|
QByteArray testName = f.family().toLatin1() + ": 0x" + QByteArray::number(uc, 16);
|
|
QTest::newRow(testName.constData()) << f << string << QList<ushort>();
|
|
}
|
|
|
|
const ShapeTable shape_table [] = {
|
|
{ { 0x3b1, 0x300, 0x313, 0x0 },
|
|
{ 0xb8, 0x3d3, 0x3c7, 0x0 } },
|
|
{ { 0x3b1, 0x313, 0x300, 0x0 },
|
|
{ 0xd4, 0x0 } },
|
|
|
|
{ {0}, {0} }
|
|
};
|
|
prepareShapingTest(f, shape_table);
|
|
} else
|
|
QSKIP("couldn't find SBL_grk");
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::greek()
|
|
{
|
|
doShapingTests();
|
|
}
|
|
|
|
void tst_QTextScriptEngine::mirroredChars_data()
|
|
{
|
|
QTest::addColumn<QString>("s");
|
|
|
|
QTest::newRow("()") << QStringLiteral("()");
|
|
QTest::newRow("[]") << QStringLiteral("[]");
|
|
QTest::newRow("{}") << QStringLiteral("{}");
|
|
}
|
|
|
|
void tst_QTextScriptEngine::mirroredChars()
|
|
{
|
|
QFETCH(QString, s);
|
|
|
|
glyph_t leftParenthesis;
|
|
glyph_t rightParenthesis;
|
|
{
|
|
QTextLayout layout(s);
|
|
layout.setCacheEnabled(true);
|
|
layout.beginLayout();
|
|
layout.createLine();
|
|
layout.endLayout();
|
|
|
|
QTextEngine *e = layout.engine();
|
|
e->itemize();
|
|
QCOMPARE(e->layoutData->items.size(), 1);
|
|
|
|
e->shape(0);
|
|
QCOMPARE(e->layoutData->items[0].num_glyphs, ushort(2));
|
|
|
|
const QGlyphLayout glyphLayout = e->shapedGlyphs(&e->layoutData->items[0]);
|
|
leftParenthesis = glyphLayout.glyphs[0];
|
|
rightParenthesis = glyphLayout.glyphs[1];
|
|
}
|
|
|
|
{
|
|
QTextLayout layout(s);
|
|
layout.setFlags(Qt::TextForceRightToLeft);
|
|
|
|
QTextEngine *e = layout.engine();
|
|
e->itemize();
|
|
QCOMPARE(e->layoutData->items.size(), 1);
|
|
|
|
e->shape(0);
|
|
QCOMPARE(e->layoutData->items[0].num_glyphs, ushort(2));
|
|
|
|
const QGlyphLayout glyphLayout = e->shapedGlyphs(&e->layoutData->items[0]);
|
|
QCOMPARE(glyphLayout.glyphs[0], rightParenthesis);
|
|
QCOMPARE(glyphLayout.glyphs[1], leftParenthesis);
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::controlInSyllable_qtbug14204()
|
|
{
|
|
if (!QFontDatabase::families().contains(QStringLiteral("Aparajita")))
|
|
QSKIP("couldn't find 'Aparajita' font");
|
|
|
|
QFont font(QStringLiteral("Aparajita"));
|
|
font.setStyleStrategy(QFont::NoFontMerging);
|
|
|
|
QString s;
|
|
s.append(QChar(0x0915));
|
|
s.append(QChar(0x094d));
|
|
s.append(QChar(0x200d));
|
|
s.append(QChar(0x0915));
|
|
|
|
QTextLayout layout(s, font);
|
|
QTextEngine *e = layout.engine();
|
|
e->itemize();
|
|
QCOMPARE(e->layoutData->items.size(), 1);
|
|
|
|
QFontEngine *fe = e->fontEngine(e->layoutData->items[0]);
|
|
if (fe->type() == QFontEngine::Box)
|
|
QSKIP("OpenType support missing for script");
|
|
QCOMPARE(fe->fontDef.families.first(), font.family());
|
|
|
|
e->shape(0);
|
|
QCOMPARE(e->layoutData->items[0].num_glyphs, ushort(3));
|
|
|
|
const ushort *log_clusters = e->logClusters(&e->layoutData->items[0]);
|
|
QCOMPARE(log_clusters[0], ushort(0));
|
|
QCOMPARE(log_clusters[1], ushort(0));
|
|
QCOMPARE(log_clusters[2], ushort(0));
|
|
QCOMPARE(log_clusters[3], ushort(2));
|
|
}
|
|
|
|
void tst_QTextScriptEngine::combiningMarks_qtbug15675_data()
|
|
{
|
|
QTest::addColumn<QFont>("font");
|
|
QTest::addColumn<QString>("string");
|
|
|
|
QSKIP("Result differs for HarfBuzz-NG, skip test.");
|
|
|
|
bool hasTests = false;
|
|
|
|
QStringList families;
|
|
families << QStringLiteral("Monaco");
|
|
families << QStringLiteral("DejaVu Sans Mono");
|
|
|
|
foreach (const QString &family, families) {
|
|
QFont font(family);
|
|
font.setStyleStrategy(QFont::NoFontMerging);
|
|
if (QFontInfo(font).family() != family)
|
|
continue;
|
|
|
|
hasTests = true;
|
|
|
|
QString s(QStringLiteral("ab cd"));
|
|
for (ushort uc = 0x0300; uc < 0x0370; ++uc) {
|
|
s[2] = QChar(uc);
|
|
QByteArray testName = family.toLatin1() + ": ab<U+" + QByteArray::number(uc, 16).rightJustified(4, '0') + ">cd";
|
|
QTest::newRow(testName.constData()) << font << s;
|
|
}
|
|
}
|
|
|
|
if (!hasTests)
|
|
QSKIP("Couldn't find required fonts, skip test.");
|
|
}
|
|
|
|
void tst_QTextScriptEngine::combiningMarks_qtbug15675()
|
|
{
|
|
QFETCH(QFont, font);
|
|
QFETCH(QString, string);
|
|
|
|
QTextLayout layout(string, font);
|
|
QTextEngine *e = layout.engine();
|
|
e->itemize();
|
|
QCOMPARE(e->layoutData->items.size(), 1);
|
|
|
|
QFontEngine *fe = e->fontEngine(e->layoutData->items[0]);
|
|
if (fe->type() == QFontEngine::Box)
|
|
QSKIP("OpenType support missing for script");
|
|
QCOMPARE(fe->fontDef.families.first(), font.family());
|
|
|
|
e->shape(0);
|
|
const int diff = e->layoutData->items[0].num_glyphs - string.size();
|
|
QVERIFY(diff >= -1 && diff <= 1); // could compose or decompose exactly one character
|
|
|
|
const ushort *log_clusters = e->logClusters(&e->layoutData->items[0]);
|
|
QCOMPARE(log_clusters[0], ushort(0));
|
|
QCOMPARE(log_clusters[1], ushort(1));
|
|
QCOMPARE(log_clusters[2], ushort(1));
|
|
QCOMPARE(log_clusters[3], ushort(3 + diff));
|
|
QCOMPARE(log_clusters[4], ushort(4 + diff));
|
|
|
|
const QGlyphLayout glyphLayout = e->shapedGlyphs(&e->layoutData->items[0]);
|
|
for (int i = 0; i < glyphLayout.numGlyphs; ++i) {
|
|
if ((diff >= 0 && i == 2) || (diff > 0 && i == 2 + diff))
|
|
QCOMPARE(glyphLayout.advances[i].toInt(), 0);
|
|
else
|
|
QVERIFY(glyphLayout.advances[i].toInt() != 0);
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::thaiIsolatedSaraAm()
|
|
{
|
|
if (!QFontDatabase::families().contains("Waree"))
|
|
QSKIP("couldn't find 'Waree' font");
|
|
|
|
QFont font(QStringLiteral("Waree"));
|
|
font.setStyleStrategy(QFont::NoFontMerging);
|
|
|
|
QString s;
|
|
s.append(QChar(0x0e33));
|
|
|
|
QTextLayout layout(s, font);
|
|
QTextEngine *e = layout.engine();
|
|
e->itemize();
|
|
QCOMPARE(e->layoutData->items.size(), 1);
|
|
|
|
QFontEngine *fe = e->fontEngine(e->layoutData->items[0]);
|
|
if (fe->type() == QFontEngine::Box)
|
|
QSKIP("OpenType support missing for script");
|
|
QCOMPARE(fe->fontDef.families.first(), font.family());
|
|
|
|
e->shape(0);
|
|
QVERIFY(e->layoutData->items[0].num_glyphs > 0);
|
|
|
|
const ushort *log_clusters = e->logClusters(&e->layoutData->items[0]);
|
|
QCOMPARE(log_clusters[0], ushort(0));
|
|
}
|
|
|
|
void tst_QTextScriptEngine::thaiWithZWJ()
|
|
{
|
|
#if QT_CONFIG(system_harfbuzz)
|
|
QSKIP("Requires up-to-date Harfbuzz");
|
|
#endif
|
|
|
|
if (!QFontDatabase::families().contains("Waree"))
|
|
QSKIP("couldn't find 'Waree' font");
|
|
|
|
QFont font(QStringLiteral("Waree"));
|
|
font.setStyleStrategy(QFont::NoFontMerging);
|
|
|
|
QString s(QString::fromUtf8("\xe0\xb8\xa3\xe2\x80\x8d\xe0\xb8\xa3\xe2\x80"
|
|
"\x8c\x2e\xe0\xb8\xa3\x2e\xe2\x80\x9c\xe0\xb8"
|
|
"\xa3\xe2\x80\xa6\xe0\xb8\xa3\xe2\x80\x9d\xe0"
|
|
"\xb8\xa3\xa0\xe0\xb8\xa3\xe6\x9c\xac\xe0\xb8\xa3")
|
|
+ QChar(0x0363)/*superscript 'a', for testing Inherited class*/);
|
|
|
|
QTextLayout layout(s, font);
|
|
QTextEngine *e = layout.engine();
|
|
e->itemize();
|
|
QCOMPARE(e->layoutData->items.size(), 3);
|
|
|
|
for (int item = 0; item < e->layoutData->items.size(); ++item)
|
|
e->shape(item);
|
|
|
|
QCOMPARE(e->layoutData->items[0].num_glyphs, ushort(15)); // Thai, Inherited and Common
|
|
QCOMPARE(e->layoutData->items[1].num_glyphs, ushort(1)); // Japanese: Kanji for tree
|
|
QCOMPARE(e->layoutData->items[2].num_glyphs, ushort(2)); // Thai: Thai character followed by superscript "a" which is of inherited type
|
|
|
|
//A quick sanity check - check all the characters are individual clusters
|
|
// A thai implementation could either remove the ZWJ character, or hide it.
|
|
// The current implementation merges the cluster for ZWJ and keeps ZWNJ, so we test for that.
|
|
unsigned short *logClusters = e->layoutData->logClustersPtr;
|
|
QCOMPARE(logClusters[0], ushort(0));
|
|
QCOMPARE(logClusters[1], ushort(0));
|
|
QCOMPARE(logClusters[2], ushort(2));
|
|
for (int i = 3; i < 15; i++)
|
|
QCOMPARE(logClusters[i], ushort(i));
|
|
for (int i = 0; i < 3; i++)
|
|
QCOMPARE(logClusters[i+15], ushort(0));
|
|
|
|
// The only characters that we should be hiding are the ZWJ and ZWNJ characters in position 1 and 3.
|
|
const QGlyphLayout glyphLayout = e->layoutData->glyphLayout;
|
|
for (int i = 0; i < 18; i++) {
|
|
if (i == 1 || i == 3)
|
|
QCOMPARE(glyphLayout.advances[i].toInt(), 0);
|
|
else
|
|
QVERIFY(glyphLayout.advances[i].toInt() != 0);
|
|
}
|
|
}
|
|
|
|
void tst_QTextScriptEngine::thaiMultipleVowels()
|
|
{
|
|
QString s(QString::fromUtf8("\xe0\xb8\xaa"));
|
|
for (int i = 0; i < 100; i++)
|
|
s += QChar(0x0E47); // Add lots of "VOWEL SIGN MAI TAI KHU N/S-T" stacked on top of the character
|
|
s += QChar(0x200D); // Now add a zero width joiner (which adds a circle which is hidden)
|
|
for (int i = 0; i < 100; i++)
|
|
s += QChar(0x0E47); //Add lots of "VOWEL SIGN MAI TAI KHU N/S-T" stacked on top of the ZWJ
|
|
|
|
for (int i = 0; i < 10; i++)
|
|
s += s; //Repeat the string to make it more likely to crash if we have a buffer overflow
|
|
|
|
QTextLayout layout(s);
|
|
QTextEngine *e = layout.engine();
|
|
e->itemize();
|
|
|
|
for (int item = 0; item < e->layoutData->items.size(); ++item)
|
|
e->shape(item);
|
|
|
|
// If we haven't crashed at this point, then the test has passed.
|
|
}
|
|
|
|
void tst_QTextScriptEngine::shapingDisabledLatin()
|
|
{
|
|
QString s("fi");
|
|
|
|
QFont font("Calibri");
|
|
font.setStyleStrategy(QFont::PreferNoShaping);
|
|
|
|
QTextLayout layout(s);
|
|
layout.setFont(font);
|
|
layout.beginLayout();
|
|
layout.createLine();
|
|
layout.endLayout();
|
|
|
|
QList<QGlyphRun> runs = layout.glyphRuns();
|
|
|
|
QCOMPARE(runs.size(), 1);
|
|
QCOMPARE(runs.first().glyphIndexes().size(), 2);
|
|
}
|
|
|
|
void tst_QTextScriptEngine::shapingDisabledDevanagari()
|
|
{
|
|
QString s;
|
|
s += QChar(0x0915); // KA
|
|
s += QChar(0x094D); // VIRAMA
|
|
s += QChar(0x0915); // KA
|
|
|
|
|
|
QList<QGlyphRun> normalRuns;
|
|
{
|
|
QTextLayout layout(s);
|
|
layout.beginLayout();
|
|
layout.createLine();
|
|
layout.endLayout();
|
|
|
|
normalRuns = layout.glyphRuns();
|
|
}
|
|
|
|
QFont font;
|
|
font.setStyleStrategy(QFont::PreferNoShaping);
|
|
|
|
QList<QGlyphRun> noShapingRuns;
|
|
{
|
|
QTextLayout layout(s);
|
|
layout.setFont(font);
|
|
layout.beginLayout();
|
|
layout.createLine();
|
|
layout.endLayout();
|
|
|
|
noShapingRuns = layout.glyphRuns();
|
|
}
|
|
|
|
// Even though shaping is disabled, Devanagari requires it, so the flag should be ignored.
|
|
QCOMPARE(normalRuns.size(), 1);
|
|
QCOMPARE(noShapingRuns.size(), 1);
|
|
QCOMPARE(noShapingRuns.first().glyphIndexes().size(), normalRuns.first().glyphIndexes().size());
|
|
}
|
|
|
|
QTEST_MAIN(tst_QTextScriptEngine)
|
|
#include "tst_qtextscriptengine.moc"
|