qt6windows7/tests/auto/widgets/widgets/qtextbrowser/tst_qtextbrowser.cpp
2023-10-29 23:33:08 +01:00

753 lines
26 KiB
C++

// Copyright (C) 2019 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
#include <QTest>
#include <QSignalSpy>
#include <qtextbrowser.h>
#include <qapplication.h>
#include <qscrollbar.h>
#include <qtextbrowser.h>
#include <qtextobject.h>
#include <QtWidgets/private/qapplication_p.h>
class TestBrowser : public QTextBrowser
{
public:
inline TestBrowser() {
show();
QApplicationPrivate::setActiveWindow(this);
activateWindow();
setFocus();
QVERIFY(QTest::qWaitForWindowActive(this));
QVERIFY(hasFocus());
}
QVariant loadResource(int type, const QUrl &name) override;
int htmlLoadAttempts = 0;
QUrl lastResource;
QUrl sourceInsideLoadResource;
QUrl baseInsideLoadResource;
};
QVariant TestBrowser::loadResource(int type, const QUrl &name)
{
if (type == QTextDocument::HtmlResource)
htmlLoadAttempts++;
lastResource = name;
sourceInsideLoadResource = source();
baseInsideLoadResource = document()->baseUrl();
return QTextBrowser::loadResource(type, name);
}
class tst_QTextBrowser : public QObject
{
Q_OBJECT
private slots:
void initTestCase();
void init();
void cleanup();
void noReloadOnAnchorJump();
void bgColorOnSourceChange();
void forwardButton();
void viewportPositionInHistory();
void relativeLinks();
void anchors();
void resourceAutoDetection();
void forwardBackwardAvailable();
void clearHistory();
void sourceInsideLoadResource();
void textInteractionFlags_vs_readOnly();
void inputMethodAttribute_vs_readOnly();
void anchorsWithSelfBuiltHtml();
void relativeNonLocalUrls();
void adjacentAnchors();
void loadResourceOnRelativeLocalFiles();
void focusIndicator();
void focusHistory();
void urlEncoding();
void sourceType_data();
void sourceType();
void unicode_data();
void unicode();
private:
TestBrowser *browser;
};
void tst_QTextBrowser::initTestCase()
{
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
QSKIP("Wayland: This fails. Figure out why.");
}
void tst_QTextBrowser::init()
{
QString prefix = QFileInfo(QFINDTESTDATA("subdir")).absolutePath();
QVERIFY2(!prefix.isEmpty(), "Test data directory not found");
QDir::setCurrent(prefix);
browser = new TestBrowser;
browser->show();
}
void tst_QTextBrowser::cleanup()
{
delete browser;
browser = nullptr;
}
void tst_QTextBrowser::noReloadOnAnchorJump()
{
QUrl url = QUrl::fromLocalFile(QFINDTESTDATA("anchor.html"));
browser->htmlLoadAttempts = 0;
browser->setSource(url);
QCOMPARE(browser->htmlLoadAttempts, 1);
QVERIFY(!browser->toPlainText().isEmpty());
url.setFragment("jumphere"); // anchor.html#jumphere
browser->setSource(url);
QCOMPARE(browser->htmlLoadAttempts, 1);
QVERIFY(!browser->toPlainText().isEmpty());
QCOMPARE(browser->source(), url);
}
void tst_QTextBrowser::bgColorOnSourceChange()
{
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("pagewithbg.html")));
QVERIFY(browser->document()->rootFrame()->frameFormat().hasProperty(QTextFormat::BackgroundBrush));
QCOMPARE(browser->document()->rootFrame()->frameFormat().background().color(), QColor(Qt::blue));
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("pagewithoutbg.html")));
QVERIFY(!browser->document()->rootFrame()->frameFormat().hasProperty(QTextFormat::BackgroundBrush));
}
void tst_QTextBrowser::forwardButton()
{
QSignalSpy forwardEmissions(browser, SIGNAL(forwardAvailable(bool)));
QSignalSpy backwardEmissions(browser, SIGNAL(backwardAvailable(bool)));
QVERIFY(browser->historyTitle(-1).isEmpty());
QVERIFY(browser->historyTitle(0).isEmpty());
QVERIFY(browser->historyTitle(1).isEmpty());
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("pagewithbg.html")));
QVERIFY(!forwardEmissions.isEmpty());
QVariant val = forwardEmissions.takeLast()[0];
QCOMPARE(val.userType(), QMetaType::Bool);
QVERIFY(!val.toBool());
QVERIFY(!backwardEmissions.isEmpty());
val = backwardEmissions.takeLast()[0];
QCOMPARE(val.userType(), QMetaType::Bool);
QVERIFY(!val.toBool());
QVERIFY(browser->historyTitle(-1).isEmpty());
QCOMPARE(browser->historyUrl(0), QUrl::fromLocalFile(QFINDTESTDATA("pagewithbg.html")));
QCOMPARE(browser->documentTitle(), QString("Page With BG"));
QCOMPARE(browser->historyTitle(0), QString("Page With BG"));
QVERIFY(browser->historyTitle(1).isEmpty());
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("anchor.html")));
QVERIFY(!forwardEmissions.isEmpty());
val = forwardEmissions.takeLast()[0];
QCOMPARE(val.userType(), QMetaType::Bool);
QVERIFY(!val.toBool());
QVERIFY(!backwardEmissions.isEmpty());
val = backwardEmissions.takeLast()[0];
QCOMPARE(val.userType(), QMetaType::Bool);
QVERIFY(val.toBool());
QCOMPARE(browser->historyTitle(-1), QString("Page With BG"));
QCOMPARE(browser->historyTitle(0), QString("Sample Anchor"));
QVERIFY(browser->historyTitle(1).isEmpty());
browser->backward();
QVERIFY(!forwardEmissions.isEmpty());
val = forwardEmissions.takeLast()[0];
QCOMPARE(val.userType(), QMetaType::Bool);
QVERIFY(val.toBool());
QVERIFY(!backwardEmissions.isEmpty());
val = backwardEmissions.takeLast()[0];
QCOMPARE(val.userType(), QMetaType::Bool);
QVERIFY(!val.toBool());
QVERIFY(browser->historyTitle(-1).isEmpty());
QCOMPARE(browser->historyTitle(0), QString("Page With BG"));
QCOMPARE(browser->historyTitle(1), QString("Sample Anchor"));
browser->setSource(QUrl(QFINDTESTDATA("pagewithoutbg.html")));
QVERIFY(!forwardEmissions.isEmpty());
val = forwardEmissions.takeLast()[0];
QCOMPARE(val.userType(), QMetaType::Bool);
QVERIFY(!val.toBool());
QVERIFY(!backwardEmissions.isEmpty());
val = backwardEmissions.takeLast()[0];
QCOMPARE(val.userType(), QMetaType::Bool);
QVERIFY(val.toBool());
}
void tst_QTextBrowser::viewportPositionInHistory()
{
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("bigpage.html")));
browser->scrollToAnchor("bottom");
QVERIFY(browser->verticalScrollBar()->value() > 0);
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("pagewithbg.html")));
QCOMPARE(browser->verticalScrollBar()->value(), 0);
browser->backward();
QVERIFY(browser->verticalScrollBar()->value() > 0);
}
void tst_QTextBrowser::relativeLinks()
{
#ifdef BUILTIN_TESTDATA
QSKIP("Relative links cannot be checked when resources are used to package tests.");
#endif
QSignalSpy sourceChangedSpy(browser, SIGNAL(sourceChanged(QUrl)));
browser->setSource(QUrl("subdir/../qtextbrowser.html"));
QVERIFY(!browser->document()->isEmpty());
QCOMPARE(sourceChangedSpy.size(), 1);
QCOMPARE(sourceChangedSpy.takeFirst()[0].toUrl(), QUrl("subdir/../qtextbrowser.html"));
browser->setSource(QUrl("subdir/index.html"));
QVERIFY(!browser->document()->isEmpty());
QCOMPARE(sourceChangedSpy.size(), 1);
QCOMPARE(sourceChangedSpy.takeFirst()[0].toUrl(), QUrl("subdir/index.html"));
browser->setSource(QUrl("anchor.html"));
QVERIFY(!browser->document()->isEmpty());
QCOMPARE(sourceChangedSpy.size(), 1);
QCOMPARE(sourceChangedSpy.takeFirst()[0].toUrl(), QUrl("anchor.html"));
browser->setSource(QUrl("subdir/index.html"));
QVERIFY(!browser->document()->isEmpty());
QCOMPARE(sourceChangedSpy.size(), 1);
QCOMPARE(sourceChangedSpy.takeFirst()[0].toUrl(), QUrl("subdir/index.html"));
// using QUrl::fromLocalFile()
browser->setSource(QUrl::fromLocalFile("anchor.html"));
QVERIFY(!browser->document()->isEmpty());
QCOMPARE(sourceChangedSpy.size(), 1);
QCOMPARE(sourceChangedSpy.takeFirst()[0].toUrl(), QUrl("file:anchor.html"));
browser->setSource(QUrl("subdir/../qtextbrowser.html"));
QVERIFY(!browser->document()->isEmpty());
QCOMPARE(sourceChangedSpy.size(), 1);
QCOMPARE(sourceChangedSpy.takeFirst()[0].toUrl(), QUrl("subdir/../qtextbrowser.html"));
}
void tst_QTextBrowser::anchors()
{
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("bigpage.html")));
browser->setSource(QUrl("#bottom"));
QVERIFY(browser->verticalScrollBar()->value() > 0);
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("bigpage.html")));
browser->setSource(QUrl("#id-anchor"));
QVERIFY(browser->verticalScrollBar()->value() > 0);
}
void tst_QTextBrowser::resourceAutoDetection()
{
browser->setHtml("<img src=\":/some/resource\"/>");
QCOMPARE(browser->lastResource.toString(), QString("qrc:/some/resource"));
}
void tst_QTextBrowser::forwardBackwardAvailable()
{
QSignalSpy backwardSpy(browser, SIGNAL(backwardAvailable(bool)));
QSignalSpy forwardSpy(browser, SIGNAL(forwardAvailable(bool)));
QVERIFY(!browser->isBackwardAvailable());
QVERIFY(!browser->isForwardAvailable());
browser->setSource(QUrl::fromLocalFile("anchor.html"));
QVERIFY(!browser->isBackwardAvailable());
QVERIFY(!browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(!backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(!forwardSpy.at(0).at(0).toBool());
backwardSpy.clear();
forwardSpy.clear();
browser->setSource(QUrl::fromLocalFile("bigpage.html"));
QVERIFY(browser->isBackwardAvailable());
QVERIFY(!browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(!forwardSpy.at(0).at(0).toBool());
backwardSpy.clear();
forwardSpy.clear();
browser->setSource(QUrl::fromLocalFile("pagewithbg.html"));
QVERIFY(browser->isBackwardAvailable());
QVERIFY(!browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(!forwardSpy.at(0).at(0).toBool());
backwardSpy.clear();
forwardSpy.clear();
browser->backward();
QVERIFY(browser->isBackwardAvailable());
QVERIFY(browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(forwardSpy.at(0).at(0).toBool());
backwardSpy.clear();
forwardSpy.clear();
browser->backward();
QVERIFY(!browser->isBackwardAvailable());
QVERIFY(browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(!backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(forwardSpy.at(0).at(0).toBool());
backwardSpy.clear();
forwardSpy.clear();
browser->forward();
QVERIFY(browser->isBackwardAvailable());
QVERIFY(browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(forwardSpy.at(0).at(0).toBool());
backwardSpy.clear();
forwardSpy.clear();
browser->forward();
QVERIFY(browser->isBackwardAvailable());
QVERIFY(!browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(!forwardSpy.at(0).at(0).toBool());
backwardSpy.clear();
forwardSpy.clear();
}
void tst_QTextBrowser::clearHistory()
{
QSignalSpy backwardSpy(browser, SIGNAL(backwardAvailable(bool)));
QSignalSpy forwardSpy(browser, SIGNAL(forwardAvailable(bool)));
QVERIFY(!browser->isBackwardAvailable());
QVERIFY(!browser->isForwardAvailable());
browser->clearHistory();
QVERIFY(!browser->isBackwardAvailable());
QVERIFY(!browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(!backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(!forwardSpy.at(0).at(0).toBool());
QVERIFY(browser->historyTitle(-1).isEmpty());
QVERIFY(browser->historyTitle(0).isEmpty());
QVERIFY(browser->historyTitle(1).isEmpty());
backwardSpy.clear();
forwardSpy.clear();
browser->setSource(QUrl::fromLocalFile("anchor.html"));
QVERIFY(!browser->isBackwardAvailable());
QVERIFY(!browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(!backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(!forwardSpy.at(0).at(0).toBool());
backwardSpy.clear();
forwardSpy.clear();
browser->setSource(QUrl::fromLocalFile("bigpage.html"));
QVERIFY(browser->isBackwardAvailable());
QVERIFY(!browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(!forwardSpy.at(0).at(0).toBool());
backwardSpy.clear();
forwardSpy.clear();
browser->clearHistory();
QVERIFY(!browser->isBackwardAvailable());
QVERIFY(!browser->isForwardAvailable());
QCOMPARE(backwardSpy.size(), 1);
QVERIFY(!backwardSpy.at(0).at(0).toBool());
QCOMPARE(forwardSpy.size(), 1);
QVERIFY(!forwardSpy.at(0).at(0).toBool());
QVERIFY(browser->historyTitle(-1).isEmpty());
QVERIFY(browser->historyTitle(1).isEmpty());
QCOMPARE(browser->source(), QUrl::fromLocalFile("bigpage.html"));
browser->backward();
QCOMPARE(browser->source(), QUrl::fromLocalFile("bigpage.html"));
browser->home();
QCOMPARE(browser->source(), QUrl::fromLocalFile("bigpage.html"));
}
void tst_QTextBrowser::sourceInsideLoadResource()
{
QUrl url = QUrl::fromLocalFile("pagewithimage.html"); // "file://pagewithimage.html"
browser->setSource(url);
QCOMPARE(browser->lastResource, QUrl::fromLocalFile(QDir::current().filePath("foobar.png")));
// baseUrl was not set because the source URL was a relative one
QCOMPARE(browser->baseInsideLoadResource, QUrl());
QEXPECT_FAIL("", "This is currently not supported", Continue);
QCOMPARE(browser->sourceInsideLoadResource.toString(), url.toString());
url = QUrl::fromLocalFile(QDir::current().filePath("pagewithimage.html")); // "file:///home/user/path/to/pagewithimage.html"
browser->setSource(url);
QCOMPARE(browser->lastResource, QUrl::fromLocalFile(QDir::current().filePath("foobar.png")));
// baseUrl has the full path, and that's where relative-path resources come from
QCOMPARE(browser->baseInsideLoadResource, QUrl::fromLocalFile(QDir::currentPath() + QLatin1Char('/')));
}
void tst_QTextBrowser::textInteractionFlags_vs_readOnly()
{
QVERIFY(browser->isReadOnly());
QCOMPARE(browser->textInteractionFlags(), Qt::TextBrowserInteraction);
browser->setReadOnly(true);
QCOMPARE(browser->textInteractionFlags(), Qt::TextBrowserInteraction);
browser->setReadOnly(false);
QCOMPARE(browser->textInteractionFlags(), Qt::TextEditorInteraction);
browser->setReadOnly(true);
QCOMPARE(browser->textInteractionFlags(), Qt::TextBrowserInteraction);
}
void tst_QTextBrowser::inputMethodAttribute_vs_readOnly()
{
QVERIFY(browser->isReadOnly());
#if defined(Q_OS_ANDROID)
QInputMethodQueryEvent query(Qt::ImReadOnly);
QCoreApplication::sendEvent(browser, &query);
QVERIFY(query.value(Qt::ImReadOnly).toBool());
#else
QVERIFY(!browser->testAttribute(Qt::WA_InputMethodEnabled));
#endif
browser->setReadOnly(false);
#if defined(Q_OS_ANDROID)
QCoreApplication::sendEvent(browser, &query);
QVERIFY(!query.value(Qt::ImReadOnly).toBool());
#else
QVERIFY(browser->testAttribute(Qt::WA_InputMethodEnabled));
#endif
browser->setReadOnly(true);
#if defined(Q_OS_ANDROID)
QCoreApplication::sendEvent(browser, &query);
QVERIFY(query.value(Qt::ImReadOnly).toBool());
#else
QVERIFY(!browser->testAttribute(Qt::WA_InputMethodEnabled));
#endif
}
void tst_QTextBrowser::anchorsWithSelfBuiltHtml()
{
browser->setHtml("<p>Hello <a href=\"#anchor\">Link</a>"
"<p><a name=\"anchor\"/>Blah</p>");
QVERIFY(browser->document()->blockCount() > 1);
browser->setSource(QUrl("#anchor"));
QVERIFY(browser->document()->blockCount() > 1);
}
class HelpBrowser : public QTextBrowser
{
public:
virtual QVariant loadResource(int /*type*/, const QUrl &name) override
{
QString url = name.toString();
if(url == "qhelp://docs/index.html") {
return "index";
} else if (url == "qhelp://docs/classes.html") {
return "classes";
} else if (url == "qhelp://docs/someclass.html") {
return "someclass";
}
return QVariant();
}
};
void tst_QTextBrowser::relativeNonLocalUrls()
{
HelpBrowser browser;
browser.setSource(QUrl("qhelp://docs/index.html"));
QCOMPARE(browser.toPlainText(), QString("index"));
browser.setSource(QUrl("classes.html"));
QCOMPARE(browser.toPlainText(), QString("classes"));
browser.setSource(QUrl("someclass.html"));
QCOMPARE(browser.toPlainText(), QString("someclass"));
}
class HackBrowser : public TestBrowser
{
public:
inline bool focusTheNextChild() { return QTextBrowser::focusNextChild(); }
inline bool focusThePreviousChild() { return QTextBrowser::focusPreviousChild(); }
};
void tst_QTextBrowser::adjacentAnchors()
{
HackBrowser *browser = new HackBrowser;
browser->setHtml("<a href=\"#foo\">foo</a><a href=\"#bar\">bar</a>");
QVERIFY(browser->focusTheNextChild());
QCOMPARE(browser->textCursor().selectedText(), QString("foo"));
QVERIFY(browser->focusTheNextChild());
QCOMPARE(browser->textCursor().selectedText(), QString("bar"));
QVERIFY(!browser->focusTheNextChild());
browser->moveCursor(QTextCursor::End);
QVERIFY(browser->focusThePreviousChild());
QCOMPARE(browser->textCursor().selectedText(), QString("bar"));
QVERIFY(browser->focusThePreviousChild());
QCOMPARE(browser->textCursor().selectedText(), QString("foo"));
delete browser;
}
void tst_QTextBrowser::loadResourceOnRelativeLocalFiles()
{
#ifndef BUILTIN_TESTDATA
browser->setSource(QUrl::fromLocalFile("subdir/index.html"));
#else
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("subdir/index.html")));
#endif
QVERIFY(!browser->toPlainText().isEmpty());
QVariant v = browser->loadResource(QTextDocument::HtmlResource, QUrl("../anchor.html"));
QVERIFY(v.isValid());
QCOMPARE(v.userType(), QMetaType::QByteArray);
QVERIFY(!v.toByteArray().isEmpty());
}
void tst_QTextBrowser::focusIndicator()
{
HackBrowser *browser = new HackBrowser;
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("firstpage.html")));
QVERIFY(!browser->textCursor().hasSelection());
browser->focusTheNextChild();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to second page"));
#ifdef QT_KEYPAD_NAVIGATION
browser->setEditFocus(true);
#endif
QTest::keyClick(browser, Qt::Key_Enter);
QVERIFY(!browser->textCursor().hasSelection());
browser->focusTheNextChild();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to third page from second page"));
QTest::keyClick(browser, Qt::Key_Enter);
QVERIFY(!browser->textCursor().hasSelection());
browser->backward();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to third page from second page"));
browser->backward();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to second page"));
browser->forward();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to third page from second page"));
browser->backward();
browser->backward();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to second page"));
QTest::keyClick(browser, Qt::Key_Enter);
QVERIFY(!browser->textCursor().hasSelection());
delete browser;
}
void tst_QTextBrowser::focusHistory()
{
HackBrowser *browser = new HackBrowser;
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA("firstpage.html")));
QVERIFY(!browser->textCursor().hasSelection());
browser->focusTheNextChild();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to second page"));
#ifdef QT_KEYPAD_NAVIGATION
browser->setEditFocus(true);
#endif
QTest::keyClick(browser, Qt::Key_Enter);
QVERIFY(!browser->textCursor().hasSelection());
browser->focusTheNextChild();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to third page from second page"));
browser->backward();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to second page"));
browser->focusTheNextChild();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to third page"));
// Despite the third page link being highlighted, going forward should go to second,
// and going back after that should still highlight the third link
browser->forward();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to third page from second page"));
browser->backward();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to third page"));
browser->forward();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to third page from second page"));
QTest::keyClick(browser, Qt::Key_Enter);
QVERIFY(!browser->textCursor().hasSelection());
browser->backward();
browser->backward();
QVERIFY(browser->textCursor().hasSelection());
QCOMPARE(browser->textCursor().selectedText(), QString("Link to third page"));
delete browser;
}
void tst_QTextBrowser::urlEncoding()
{
HackBrowser *browser = new HackBrowser;
browser->setOpenLinks(false);
browser->setHtml("<a href=\"http://www.google.com/q=%22\">link</a>");
browser->focusTheNextChild();
QSignalSpy spy(browser, SIGNAL(anchorClicked(QUrl)));
#ifdef QT_KEYPAD_NAVIGATION
browser->setEditFocus(true);
#endif
QTest::keyClick(browser, Qt::Key_Enter);
QCOMPARE(spy.size(), 1);
QUrl url = spy.at(0).at(0).toUrl();
QCOMPARE(url.toEncoded(), QByteArray("http://www.google.com/q=%22"));
delete browser;
}
void tst_QTextBrowser::sourceType_data()
{
QTest::addColumn<QString>("sourceFile");
QTest::addColumn<QTextDocument::ResourceType>("sourceType");
QTest::addColumn<int>("expectedMaxHeadingLevel");
QTest::addColumn<QTextDocument::ResourceType>("expectedSourceType");
#if QT_CONFIG(textmarkdownreader)
const int maxMdHeadingLevel = 3;
const QTextDocument::ResourceType mdExpectedType = QTextDocument::MarkdownResource;
#else
// If Qt doesn't support markdown, and we read a MD document anyway, it won't have any H3's.
const int maxMdHeadingLevel = 0;
const QTextDocument::ResourceType mdExpectedType = QTextDocument::HtmlResource;
#endif
QTest::newRow("markdown detected") << "markdown.md" << QTextDocument::UnknownResource << maxMdHeadingLevel << mdExpectedType;
QTest::newRow("markdown specified") << "markdown.really" << QTextDocument::MarkdownResource << maxMdHeadingLevel << mdExpectedType;
QTest::newRow("markdown not identified") << "markdown.really" << QTextDocument::UnknownResource << 0 << QTextDocument::HtmlResource;
QTest::newRow("html detected") << "heading.html" << QTextDocument::UnknownResource << 3 << QTextDocument::HtmlResource;
QTest::newRow("html specified") << "heading.html" << QTextDocument::HtmlResource << 3 << QTextDocument::HtmlResource;
}
void tst_QTextBrowser::sourceType()
{
QFETCH(QString, sourceFile);
QFETCH(QTextDocument::ResourceType, sourceType);
QFETCH(int, expectedMaxHeadingLevel);
QFETCH(QTextDocument::ResourceType, expectedSourceType);
if (sourceType == QTextDocument::UnknownResource)
// verify that the property setter works, with its default parameter for sourceType
browser->setProperty("source", QUrl::fromLocalFile(QFINDTESTDATA(sourceFile)));
else
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA(sourceFile)), sourceType);
QCOMPARE(browser->sourceType(), expectedSourceType);
QTextFrame::iterator iterator = browser->document()->rootFrame()->begin();
int maxHeadingLevel = -1;
while (!iterator.atEnd())
maxHeadingLevel = qMax(iterator++.currentBlock().blockFormat().intProperty(QTextFormat::HeadingLevel), maxHeadingLevel);
QCOMPARE(maxHeadingLevel, expectedMaxHeadingLevel);
}
void tst_QTextBrowser::unicode_data()
{
QTest::addColumn<QString>("sourceFile");
QTest::addColumn<QTextDocument::ResourceType>("sourceType");
QTest::addColumn<QString>("expectedText");
#if QT_CONFIG(textmarkdownreader)
QTest::newRow("markdown with quotes and fractions") << "quotesAndFractions.md" << QTextDocument::MarkdownResource <<
"you\u2019ll hope to see \u275Dquotes\u275E \uFE601\u00BD \u2154 \u00BC \u2157 \u215A \u215D some \u201Cvulgar\u201D fractions (pardon my \u00ABFrench\u00BB)";
#endif
}
void tst_QTextBrowser::unicode()
{
QFETCH(QString, sourceFile);
QFETCH(QTextDocument::ResourceType, sourceType);
QFETCH(QString, expectedText);
browser->setSource(QUrl::fromLocalFile(QFINDTESTDATA(sourceFile)), sourceType);
QTextFrame::iterator iterator = browser->document()->rootFrame()->begin();
while (!iterator.atEnd()) {
QString blockText = iterator++.currentBlock().text();
if (!blockText.isEmpty())
QCOMPARE(blockText, expectedText);
}
}
QTEST_MAIN(tst_QTextBrowser)
#include "tst_qtextbrowser.moc"