2011-04-27 10:05:43 +00:00
/****************************************************************************
* *
2016-01-15 12:36:27 +00:00
* * Copyright ( C ) 2016 The Qt Company Ltd .
* * Contact : https : //www.qt.io/licensing/
2011-04-27 10:05:43 +00:00
* *
* * This file is part of the test suite of the Qt Toolkit .
* *
2016-01-15 12:36:27 +00:00
* * $ QT_BEGIN_LICENSE : GPL - EXCEPT $
2012-09-19 12:28:29 +00:00
* * 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
2015-01-28 08:44:43 +00:00
* * a written agreement between you and The Qt Company . For licensing terms
2016-01-15 12:36:27 +00:00
* * and conditions see https : //www.qt.io/terms-conditions. For further
* * information use the contact form at https : //www.qt.io/contact-us.
2012-09-19 12:28:29 +00:00
* *
2016-01-15 12:36:27 +00:00
* * 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.
2011-04-27 10:05:43 +00:00
* *
* * $ QT_END_LICENSE $
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2020-11-26 16:31:50 +00:00
# include <QTest>
2011-04-27 10:05:43 +00:00
# include <qfont.h>
# include <qfontmetrics.h>
# include <qfontdatabase.h>
2012-10-29 10:03:34 +00:00
# include <private/qfontengine_p.h>
2011-04-27 10:05:43 +00:00
# include <qstringlist.h>
# include <qlist.h>
class tst_QFontMetrics : public QObject
{
Q_OBJECT
private slots :
void same ( ) ;
void metrics ( ) ;
void boundingRect ( ) ;
2021-10-12 10:52:39 +00:00
void boundingRect2 ( ) ;
2011-04-27 10:05:43 +00:00
void elidedText_data ( ) ;
void elidedText ( ) ;
void veryNarrowElidedText ( ) ;
void averageCharWidth ( ) ;
void bypassShaping ( ) ;
void elidedMultiLength ( ) ;
void elidedMultiLengthF ( ) ;
void inFontUcs4 ( ) ;
void lineWidth ( ) ;
2014-09-26 10:36:43 +00:00
void mnemonicTextWidth ( ) ;
2015-09-25 16:54:46 +00:00
void leadingBelowLine ( ) ;
2018-12-20 08:27:09 +00:00
void elidedMetrics ( ) ;
2020-02-25 14:59:31 +00:00
void zeroWidthMetrics ( ) ;
2011-04-27 10:05:43 +00:00
} ;
void tst_QFontMetrics : : same ( )
{
QFont font ;
font . setBold ( true ) ;
QFontMetrics fm ( font ) ;
const QString text = QLatin1String ( " Some stupid STRING " ) ;
QCOMPARE ( fm . size ( 0 , text ) , fm . size ( 0 , text ) ) ;
2014-02-19 17:00:24 +00:00
for ( int i = 10 ; i < = 32 ; + + i ) {
font . setPixelSize ( i ) ;
QFontMetrics fm1 ( font ) ;
QCOMPARE ( fm1 . size ( 0 , text ) , fm1 . size ( 0 , text ) ) ;
}
2011-04-27 10:05:43 +00:00
{
QImage image ;
QFontMetrics fm2 ( font , & image ) ;
QString text2 = QLatin1String ( " Foo Foo " ) ;
QCOMPARE ( fm2 . size ( 0 , text2 ) , fm2 . size ( 0 , text2 ) ) ; //used to crash
}
{
QImage image ;
QFontMetricsF fm3 ( font , & image ) ;
QString text2 = QLatin1String ( " Foo Foo " ) ;
QCOMPARE ( fm3 . size ( 0 , text2 ) , fm3 . size ( 0 , text2 ) ) ; //used to crash
}
}
void tst_QFontMetrics : : metrics ( )
{
QFont font ;
// Query the QFontDatabase for a specific font, store the
// result in family, style and size.
2020-11-02 18:48:25 +00:00
QStringList families = QFontDatabase : : families ( ) ;
2011-04-27 10:05:43 +00:00
if ( families . isEmpty ( ) )
return ;
QStringList : : ConstIterator f_it , f_end = families . end ( ) ;
for ( f_it = families . begin ( ) ; f_it ! = f_end ; + + f_it ) {
const QString & family = * f_it ;
2020-11-02 18:48:25 +00:00
QStringList styles = QFontDatabase : : styles ( family ) ;
2011-04-27 10:05:43 +00:00
QStringList : : ConstIterator s_it , s_end = styles . end ( ) ;
for ( s_it = styles . begin ( ) ; s_it ! = s_end ; + + s_it ) {
const QString & style = * s_it ;
2020-11-02 18:48:25 +00:00
if ( QFontDatabase : : isSmoothlyScalable ( family , style ) ) {
2011-04-27 10:05:43 +00:00
// smoothly scalable font... don't need to load every pointsize
2020-11-02 18:48:25 +00:00
font = QFontDatabase : : font ( family , style , 12 ) ;
2011-04-27 10:05:43 +00:00
QFontMetrics fontmetrics ( font ) ;
2012-02-03 13:28:16 +00:00
QCOMPARE ( fontmetrics . ascent ( ) + fontmetrics . descent ( ) ,
2011-04-27 10:05:43 +00:00
fontmetrics . height ( ) ) ;
QCOMPARE ( fontmetrics . height ( ) + fontmetrics . leading ( ) ,
fontmetrics . lineSpacing ( ) ) ;
} else {
2020-11-02 18:48:25 +00:00
QList < int > sizes = QFontDatabase : : pointSizes ( family , style ) ;
2011-04-27 10:05:43 +00:00
QVERIFY ( ! sizes . isEmpty ( ) ) ;
QList < int > : : ConstIterator z_it , z_end = sizes . end ( ) ;
for ( z_it = sizes . begin ( ) ; z_it ! = z_end ; + + z_it ) {
const int size = * z_it ;
// Initialize the font, and check if it is an exact match
2020-11-02 18:48:25 +00:00
font = QFontDatabase : : font ( family , style , size ) ;
2011-04-27 10:05:43 +00:00
QFontMetrics fontmetrics ( font ) ;
2012-02-03 13:28:16 +00:00
QCOMPARE ( fontmetrics . ascent ( ) + fontmetrics . descent ( ) ,
2011-04-27 10:05:43 +00:00
fontmetrics . height ( ) ) ;
QCOMPARE ( fontmetrics . height ( ) + fontmetrics . leading ( ) ,
fontmetrics . lineSpacing ( ) ) ;
}
}
2014-01-13 14:48:44 +00:00
}
2011-04-27 10:05:43 +00:00
}
}
void tst_QFontMetrics : : boundingRect ( )
{
QFont f ;
f . setPointSize ( 24 ) ;
QFontMetrics fm ( f ) ;
QRect r = fm . boundingRect ( QChar ( ' Y ' ) ) ;
QVERIFY ( r . top ( ) < 0 ) ;
r = fm . boundingRect ( QString ( " Y " ) ) ;
QVERIFY ( r . top ( ) < 0 ) ;
}
2021-10-12 10:52:39 +00:00
void tst_QFontMetrics : : boundingRect2 ( )
{
QFont f ;
f . setPixelSize ( 16 ) ;
QFontMetricsF fm ( f ) ;
QString str ( " AVAVAVA vvvvvvvvvv fffffffff file " ) ;
QRectF br = fm . boundingRect ( str ) ;
QRectF tbr = fm . tightBoundingRect ( str ) ;
qreal advance = fm . horizontalAdvance ( str ) ;
// Bounding rect plus bearings should be similar to advance
qreal bearings = fm . leftBearing ( QChar ( ' A ' ) ) + fm . rightBearing ( QChar ( ' e ' ) ) ;
QVERIFY ( qAbs ( br . width ( ) + bearings - advance ) < fm . averageCharWidth ( ) / 2.0 ) ;
QVERIFY ( qAbs ( tbr . width ( ) + bearings - advance ) < fm . averageCharWidth ( ) / 2.0 ) ;
}
2011-04-27 10:05:43 +00:00
void tst_QFontMetrics : : elidedText_data ( )
{
QTest : : addColumn < QFont > ( " font " ) ;
QTest : : addColumn < QString > ( " text " ) ;
QTest : : newRow ( " helvetica hello " ) < < QFont ( " helvetica " , 10 ) < < QString ( " hello " ) ;
QTest : : newRow ( " helvetica hello &Bye " ) < < QFont ( " helvetica " , 10 ) < < QString ( " hello&Bye " ) ;
}
void tst_QFontMetrics : : elidedText ( )
{
QFETCH ( QFont , font ) ;
QFETCH ( QString , text ) ;
QFontMetrics fm ( font ) ;
2017-08-02 09:39:01 +00:00
int w = fm . horizontalAdvance ( text ) ;
2011-04-27 10:05:43 +00:00
QString newtext = fm . elidedText ( text , Qt : : ElideRight , w + 1 , 0 ) ;
QCOMPARE ( text , newtext ) ; // should not elide
newtext = fm . elidedText ( text , Qt : : ElideRight , w - 1 , 0 ) ;
QVERIFY ( text ! = newtext ) ; // should elide
}
void tst_QFontMetrics : : veryNarrowElidedText ( )
{
QFont f ;
QFontMetrics fm ( f ) ;
QString text ( " hello " ) ;
QCOMPARE ( fm . elidedText ( text , Qt : : ElideRight , 0 ) , QString ( ) ) ;
}
void tst_QFontMetrics : : averageCharWidth ( )
{
QFont f ;
QFontMetrics fm ( f ) ;
QVERIFY ( fm . averageCharWidth ( ) ! = 0 ) ;
QFontMetricsF fmf ( f ) ;
QVERIFY ( fmf . averageCharWidth ( ) ! = 0 ) ;
}
void tst_QFontMetrics : : bypassShaping ( )
{
QFont f ;
2020-06-03 09:25:03 +00:00
f . setStyleStrategy ( QFont : : PreferNoShaping ) ;
f . setKerning ( false ) ;
QFontMetricsF fm ( f ) ;
2011-04-27 10:05:43 +00:00
QString text = " A, B, C, D, E, F, G, H, I, J, K, L, M, N, O, P, Q, R, S, T, U, V, W, X, Y, Z, a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p, q, r, s, t, u, v, w, x, y, z " ;
2020-06-03 09:25:03 +00:00
qreal textWidth = fm . horizontalAdvance ( text ) ;
2011-04-27 10:05:43 +00:00
QVERIFY ( textWidth ! = 0 ) ;
2020-06-03 09:25:03 +00:00
qreal charsWidth = 0 ;
2011-04-27 10:05:43 +00:00
for ( int i = 0 ; i < text . size ( ) ; + + i )
2017-08-02 09:39:01 +00:00
charsWidth + = fm . horizontalAdvance ( text [ i ] ) ;
2011-04-27 10:05:43 +00:00
QCOMPARE ( textWidth , charsWidth ) ;
}
2017-02-01 10:36:17 +00:00
template < class FontMetrics , typename PrimitiveType > void elidedMultiLength_helper ( )
2011-04-27 10:05:43 +00:00
{
2012-04-23 14:38:17 +00:00
QString text1 = QLatin1String ( " Long Text 1 \x9c Shorter \x9c small " ) ;
2011-04-27 10:05:43 +00:00
QString text1_long = " Long Text 1 " ;
QString text1_short = " Shorter " ;
QString text1_small = " small " ;
FontMetrics fm = FontMetrics ( QFont ( ) ) ;
2017-02-01 10:36:17 +00:00
PrimitiveType width_long = fm . size ( 0 , text1_long ) . width ( ) ;
2011-04-27 10:05:43 +00:00
QCOMPARE ( fm . elidedText ( text1 , Qt : : ElideRight , 8000 ) , text1_long ) ;
QCOMPARE ( fm . elidedText ( text1 , Qt : : ElideRight , width_long + 1 ) , text1_long ) ;
QCOMPARE ( fm . elidedText ( text1 , Qt : : ElideRight , width_long - 1 ) , text1_short ) ;
2017-02-01 10:36:17 +00:00
PrimitiveType width_short = fm . size ( 0 , text1_short ) . width ( ) ;
2011-04-27 10:05:43 +00:00
QCOMPARE ( fm . elidedText ( text1 , Qt : : ElideRight , width_short + 1 ) , text1_short ) ;
QCOMPARE ( fm . elidedText ( text1 , Qt : : ElideRight , width_short - 1 ) , text1_small ) ;
// Not even wide enough for "small" - should use ellipsis
QChar ellipsisChar ( 0x2026 ) ;
QString text1_el = QString : : fromLatin1 ( " s " ) + ellipsisChar ;
2017-08-02 09:39:01 +00:00
PrimitiveType width_small = fm . horizontalAdvance ( text1_el ) ;
2011-04-27 10:05:43 +00:00
QCOMPARE ( fm . elidedText ( text1 , Qt : : ElideRight , width_small + 1 ) , text1_el ) ;
}
void tst_QFontMetrics : : elidedMultiLength ( )
{
2017-02-01 10:36:17 +00:00
elidedMultiLength_helper < QFontMetrics , int > ( ) ;
2011-04-27 10:05:43 +00:00
}
void tst_QFontMetrics : : elidedMultiLengthF ( )
{
2017-02-01 10:36:17 +00:00
elidedMultiLength_helper < QFontMetricsF , qreal > ( ) ;
2011-04-27 10:05:43 +00:00
}
void tst_QFontMetrics : : inFontUcs4 ( )
{
int id = QFontDatabase : : addApplicationFont ( " :/fonts/ucs4font.ttf " ) ;
QVERIFY ( id > = 0 ) ;
QFont font ( " QtTestUcs4 " ) ;
{
QFontMetrics fm ( font ) ;
QVERIFY ( fm . inFontUcs4 ( 0x1D7FF ) ) ;
}
{
QFontMetricsF fm ( font ) ;
QVERIFY ( fm . inFontUcs4 ( 0x1D7FF ) ) ;
}
2012-10-29 10:03:34 +00:00
{
2012-12-21 16:35:58 +00:00
QFontEngine * engine = QFontPrivate : : get ( font ) - > engineForScript ( QChar : : Script_Common ) ;
2012-10-29 10:03:34 +00:00
QGlyphLayout glyphs ;
glyphs . numGlyphs = 3 ;
uint buf [ 3 ] ;
glyphs . glyphs = buf ;
QString string ;
{
string . append ( QChar : : highSurrogate ( 0x1D7FF ) ) ;
string . append ( QChar : : lowSurrogate ( 0x1D7FF ) ) ;
glyphs . numGlyphs = 3 ;
glyphs . glyphs [ 0 ] = 0 ;
QVERIFY ( engine - > stringToCMap ( string . constData ( ) , string . size ( ) ,
& glyphs , & glyphs . numGlyphs ,
QFontEngine : : GlyphIndicesOnly ) ) ;
QCOMPARE ( glyphs . numGlyphs , 1 ) ;
QCOMPARE ( glyphs . glyphs [ 0 ] , uint ( 1 ) ) ;
}
{
string . clear ( ) ;
string . append ( QChar : : ObjectReplacementCharacter ) ;
glyphs . numGlyphs = 3 ;
glyphs . glyphs [ 0 ] = 0 ;
QVERIFY ( engine - > stringToCMap ( string . constData ( ) , string . size ( ) ,
& glyphs , & glyphs . numGlyphs ,
QFontEngine : : GlyphIndicesOnly ) ) ;
QVERIFY ( glyphs . glyphs [ 0 ] ! = 1 ) ;
}
}
2011-04-27 10:05:43 +00:00
QFontDatabase : : removeApplicationFont ( id ) ;
}
void tst_QFontMetrics : : lineWidth ( )
{
// QTBUG-13009, QTBUG-13011
QFont smallFont ;
smallFont . setPointSize ( 8 ) ;
smallFont . setWeight ( QFont : : Light ) ;
const QFontMetrics smallFontMetrics ( smallFont ) ;
QFont bigFont ;
bigFont . setPointSize ( 40 ) ;
bigFont . setWeight ( QFont : : Black ) ;
const QFontMetrics bigFontMetrics ( bigFont ) ;
QVERIFY ( smallFontMetrics . lineWidth ( ) > = 1 ) ;
QVERIFY ( smallFontMetrics . lineWidth ( ) < bigFontMetrics . lineWidth ( ) ) ;
}
2014-09-26 10:36:43 +00:00
void tst_QFontMetrics : : mnemonicTextWidth ( )
{
// QTBUG-41593
QFont f ;
QFontMetrics fm ( f ) ;
const QString f1 = " File " ;
const QString f2 = " &File " ;
QCOMPARE ( fm . size ( Qt : : TextShowMnemonic , f1 ) , fm . size ( Qt : : TextShowMnemonic , f2 ) ) ;
QCOMPARE ( fm . size ( Qt : : TextHideMnemonic , f1 ) , fm . size ( Qt : : TextHideMnemonic , f2 ) ) ;
}
2015-09-25 16:54:46 +00:00
void tst_QFontMetrics : : leadingBelowLine ( )
{
QScriptLine line ;
line . leading = 10 ;
line . leadingIncluded = true ;
line . ascent = 5 ;
QCOMPARE ( line . height ( ) , line . ascent + line . descent + line . leading ) ;
QCOMPARE ( line . base ( ) , line . ascent ) ;
}
2018-12-20 08:27:09 +00:00
void tst_QFontMetrics : : elidedMetrics ( )
{
QString testFont = QFINDTESTDATA ( " fonts/testfont.ttf " ) ;
QVERIFY ( ! testFont . isEmpty ( ) ) ;
int id = QFontDatabase : : addApplicationFont ( testFont ) ;
QVERIFY ( id > = 0 ) ;
QFont font ( QFontDatabase : : applicationFontFamilies ( id ) . at ( 0 ) ) ;
font . setPixelSize ( 12.0 ) ;
QFontMetricsF metrics ( font ) ;
QString s = QStringLiteral ( " VeryLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongLongText " ) ;
QRectF boundingRect = metrics . boundingRect ( s ) ;
QString elided = metrics . elidedText ( s , Qt : : ElideRight , boundingRect . width ( ) / 2.0 ) ;
QRectF elidedBoundingRect = metrics . boundingRect ( elided ) ;
QCOMPARE ( boundingRect . height ( ) , elidedBoundingRect . height ( ) ) ;
QCOMPARE ( boundingRect . y ( ) , elidedBoundingRect . y ( ) ) ;
QFontDatabase : : removeApplicationFont ( id ) ;
}
2020-02-25 14:59:31 +00:00
void tst_QFontMetrics : : zeroWidthMetrics ( )
{
QString zwnj ( QChar ( 0x200c ) ) ;
QString zwsp ( QChar ( 0x200b ) ) ;
QFont font ;
QFontMetricsF fm ( font ) ;
QCOMPARE ( fm . horizontalAdvance ( zwnj ) , 0 ) ;
QCOMPARE ( fm . horizontalAdvance ( zwsp ) , 0 ) ;
QCOMPARE ( fm . boundingRect ( zwnj ) . width ( ) , 0 ) ;
QCOMPARE ( fm . boundingRect ( zwsp ) . width ( ) , 0 ) ;
QString string1 = QStringLiteral ( " ( " ) + zwnj + QStringLiteral ( " ) " ) ;
QString string2 = QStringLiteral ( " ( " ) + zwnj + zwnj + QStringLiteral ( " ) " ) ;
QString string3 = QStringLiteral ( " ( " ) + zwsp + QStringLiteral ( " ) " ) ;
QString string4 = QStringLiteral ( " ( " ) + zwsp + zwsp + QStringLiteral ( " ) " ) ;
QCOMPARE ( fm . horizontalAdvance ( string1 ) , fm . horizontalAdvance ( string2 ) ) ;
QCOMPARE ( fm . horizontalAdvance ( string3 ) , fm . horizontalAdvance ( string4 ) ) ;
QCOMPARE ( fm . boundingRect ( string1 ) . width ( ) , fm . boundingRect ( string2 ) . width ( ) ) ;
QCOMPARE ( fm . boundingRect ( string3 ) . width ( ) , fm . boundingRect ( string4 ) . width ( ) ) ;
2021-10-12 10:52:39 +00:00
QCOMPARE ( fm . tightBoundingRect ( string1 ) . width ( ) , fm . tightBoundingRect ( string2 ) . width ( ) ) ;
QCOMPARE ( fm . tightBoundingRect ( string3 ) . width ( ) , fm . tightBoundingRect ( string4 ) . width ( ) ) ;
2020-02-25 14:59:31 +00:00
}
2011-04-27 10:05:43 +00:00
QTEST_MAIN ( tst_QFontMetrics )
# include "tst_qfontmetrics.moc"