mirror of
https://github.com/crystalidea/qt6windows7.git
synced 2024-11-24 12:30:32 +08:00
1345 lines
46 KiB
C++
1345 lines
46 KiB
C++
|
// Copyright (C) 2016 The Qt Company Ltd.
|
||
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
||
|
|
||
|
|
||
|
#include <QTest>
|
||
|
|
||
|
#include <qbuffer.h>
|
||
|
#include <qtextdocument.h>
|
||
|
#include <qtextdocumentfragment.h>
|
||
|
#include <qtexttable.h>
|
||
|
#include <qdebug.h>
|
||
|
#include <qtextcursor.h>
|
||
|
#include <qtextdocument.h>
|
||
|
#ifndef QT_NO_WIDGETS
|
||
|
#include <qtextedit.h>
|
||
|
#endif
|
||
|
#ifndef QT_NO_PRINTER
|
||
|
#include <QPagedPaintDevice>
|
||
|
#include <QPainter>
|
||
|
#include <QPaintEngine>
|
||
|
#endif
|
||
|
#include <private/qtextdocumentlayout_p.h>
|
||
|
#include <private/qpagedpaintdevice_p.h>
|
||
|
|
||
|
typedef QList<int> IntList;
|
||
|
|
||
|
QT_FORWARD_DECLARE_CLASS(QTextDocument)
|
||
|
|
||
|
Q_DECLARE_METATYPE(QTextFrameFormat::BorderStyle);
|
||
|
|
||
|
class tst_QTextTable : public QObject
|
||
|
{
|
||
|
Q_OBJECT
|
||
|
|
||
|
private slots:
|
||
|
void init();
|
||
|
void cleanup();
|
||
|
void cursorPositioning();
|
||
|
void variousTableModifications();
|
||
|
void tableShrinking();
|
||
|
void spans();
|
||
|
void variousModifications2();
|
||
|
void tableManager_undo();
|
||
|
void tableManager_removeCell();
|
||
|
void rowAt();
|
||
|
void rowAtWithSpans();
|
||
|
void multiBlockCells();
|
||
|
void insertRows();
|
||
|
void deleteInTable();
|
||
|
void mergeCells();
|
||
|
void mergeAndInsert();
|
||
|
void splitCells();
|
||
|
void blocksForTableShouldHaveEmptyFormat();
|
||
|
void removeTableByRemoveRows();
|
||
|
void removeTableByRemoveColumns();
|
||
|
void setCellFormat();
|
||
|
void removeRows1();
|
||
|
void removeRows2();
|
||
|
void removeRows3();
|
||
|
void removeRows4();
|
||
|
void removeRows5();
|
||
|
void removeColumns1();
|
||
|
void removeColumns2();
|
||
|
void removeColumns3();
|
||
|
void removeColumns4();
|
||
|
void removeColumns5();
|
||
|
void removeColumnsInTableWithMergedRows();
|
||
|
#ifndef QT_NO_WIDGETS
|
||
|
void QTBUG11282_insertBeforeMergedEnding_data();
|
||
|
void QTBUG11282_insertBeforeMergedEnding();
|
||
|
#endif
|
||
|
void QTBUG22011_insertBeforeRowSpan();
|
||
|
#if !defined(QT_NO_PRINTER) && defined(QT_BUILD_INTERNAL)
|
||
|
void QTBUG31330_renderBackground();
|
||
|
#endif
|
||
|
void checkBorderAttributes_data();
|
||
|
void checkBorderAttributes();
|
||
|
|
||
|
#ifndef QT_NO_WIDGETS
|
||
|
void columnWidthWithSpans();
|
||
|
|
||
|
void columnWidthWithImage_data();
|
||
|
void columnWidthWithImage();
|
||
|
#endif
|
||
|
|
||
|
private:
|
||
|
QTextTable *create2x2Table();
|
||
|
QTextTable *create4x4Table();
|
||
|
|
||
|
QTextTable *createTable(int rows, int cols);
|
||
|
|
||
|
QTextDocument *doc;
|
||
|
QTextCursor cursor;
|
||
|
};
|
||
|
|
||
|
void tst_QTextTable::init()
|
||
|
{
|
||
|
doc = new QTextDocument;
|
||
|
cursor = QTextCursor(doc);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::cleanup()
|
||
|
{
|
||
|
cursor = QTextCursor();
|
||
|
delete doc;
|
||
|
doc = 0;
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::cursorPositioning()
|
||
|
{
|
||
|
// ensure the cursor is placed at the beginning of the first cell upon
|
||
|
// table creation
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
|
||
|
QVERIFY(cursor == table->cellAt(0, 0).firstCursorPosition());
|
||
|
QVERIFY(table->cellAt(0, 0).firstPosition() == table->firstPosition());
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::variousTableModifications()
|
||
|
{
|
||
|
QTextTableFormat tableFmt;
|
||
|
|
||
|
QTextTable *tab = cursor.insertTable(2, 2, tableFmt);
|
||
|
QCOMPARE(doc->toPlainText().size(), 5);
|
||
|
QCOMPARE(tab, cursor.currentTable());
|
||
|
QCOMPARE(tab->columns(), 2);
|
||
|
QCOMPARE(tab->rows(), 2);
|
||
|
|
||
|
QCOMPARE(cursor.position(), 1);
|
||
|
QTextCharFormat fmt = cursor.charFormat();
|
||
|
QCOMPARE(fmt.objectIndex(), -1);
|
||
|
QTextTableCell cell = tab->cellAt(cursor);
|
||
|
QVERIFY(cell.isValid());
|
||
|
QCOMPARE(cell.row(), 0);
|
||
|
QCOMPARE(cell.column(), 0);
|
||
|
|
||
|
cursor.movePosition(QTextCursor::NextBlock);
|
||
|
QCOMPARE(cursor.position(), 2);
|
||
|
fmt = cursor.charFormat();
|
||
|
QCOMPARE(fmt.objectIndex(), -1);
|
||
|
cell = tab->cellAt(cursor);
|
||
|
QVERIFY(cell.isValid());
|
||
|
QCOMPARE(cell.row(), 0);
|
||
|
QCOMPARE(cell.column(), 1);
|
||
|
|
||
|
cursor.movePosition(QTextCursor::NextBlock);
|
||
|
QCOMPARE(cursor.position(), 3);
|
||
|
fmt = cursor.charFormat();
|
||
|
QCOMPARE(fmt.objectIndex(), -1);
|
||
|
cell = tab->cellAt(cursor);
|
||
|
QVERIFY(cell.isValid());
|
||
|
QCOMPARE(cell.row(), 1);
|
||
|
QCOMPARE(cell.column(), 0);
|
||
|
|
||
|
cursor.movePosition(QTextCursor::NextBlock);
|
||
|
QCOMPARE(cursor.position(), 4);
|
||
|
fmt = cursor.charFormat();
|
||
|
QCOMPARE(fmt.objectIndex(), -1);
|
||
|
cell = tab->cellAt(cursor);
|
||
|
QVERIFY(cell.isValid());
|
||
|
QCOMPARE(cell.row(), 1);
|
||
|
QCOMPARE(cell.column(), 1);
|
||
|
|
||
|
cursor.movePosition(QTextCursor::NextBlock);
|
||
|
QCOMPARE(cursor.position(), 5);
|
||
|
fmt = cursor.charFormat();
|
||
|
QCOMPARE(fmt.objectIndex(), -1);
|
||
|
cell = tab->cellAt(cursor);
|
||
|
QVERIFY(!cell.isValid());
|
||
|
|
||
|
cursor.movePosition(QTextCursor::NextBlock);
|
||
|
QCOMPARE(cursor.position(), 5);
|
||
|
|
||
|
// check we can't delete the cells with the cursor
|
||
|
cursor.movePosition(QTextCursor::Start);
|
||
|
cursor.movePosition(QTextCursor::NextBlock);
|
||
|
QCOMPARE(cursor.position(), 1);
|
||
|
cursor.deleteChar();
|
||
|
QCOMPARE(doc->toPlainText().size(), 5);
|
||
|
cursor.movePosition(QTextCursor::NextBlock);
|
||
|
QCOMPARE(cursor.position(), 2);
|
||
|
cursor.deleteChar();
|
||
|
QCOMPARE(doc->toPlainText().size(), 5);
|
||
|
cursor.deletePreviousChar();
|
||
|
QCOMPARE(cursor.position(), 2);
|
||
|
QCOMPARE(doc->toPlainText().size(), 5);
|
||
|
|
||
|
QTextTable *table = cursor.currentTable();
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
|
||
|
table->insertRows(2, 1);
|
||
|
QCOMPARE(table->rows(), 3);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QCOMPARE(doc->toPlainText().size(), 7);
|
||
|
table->insertColumns(2, 2);
|
||
|
QCOMPARE(table->rows(), 3);
|
||
|
QCOMPARE(table->columns(), 4);
|
||
|
QCOMPARE(doc->toPlainText().size(), 13);
|
||
|
|
||
|
table->resize(4, 5);
|
||
|
QCOMPARE(table->rows(), 4);
|
||
|
QCOMPARE(table->columns(), 5);
|
||
|
QCOMPARE(doc->toPlainText().size(), 21);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::tableShrinking()
|
||
|
{
|
||
|
QTextTableFormat tableFmt;
|
||
|
|
||
|
cursor.insertTable(3, 4, tableFmt);
|
||
|
QCOMPARE(doc->toPlainText().size(), 13);
|
||
|
|
||
|
QTextTable *table = cursor.currentTable();
|
||
|
QCOMPARE(table->rows(), 3);
|
||
|
QCOMPARE(table->columns(), 4);
|
||
|
|
||
|
table->removeRows(1, 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 4);
|
||
|
QCOMPARE(doc->toPlainText().size(), 9);
|
||
|
table->removeColumns(1, 2);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QCOMPARE(doc->toPlainText().size(), 5);
|
||
|
|
||
|
table->resize(1, 1);
|
||
|
QCOMPARE(table->rows(), 1);
|
||
|
QCOMPARE(table->columns(), 1);
|
||
|
QCOMPARE(doc->toPlainText().size(), 2);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::spans()
|
||
|
{
|
||
|
QTextTableFormat tableFmt;
|
||
|
|
||
|
cursor.insertTable(2, 2, tableFmt);
|
||
|
|
||
|
QTextTable *table = cursor.currentTable();
|
||
|
QVERIFY(table->cellAt(0, 0) != table->cellAt(0, 1));
|
||
|
table->mergeCells(0, 0, 1, 2);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QVERIFY(table->cellAt(0, 0) == table->cellAt(0, 1));
|
||
|
table->mergeCells(0, 0, 2, 2);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::variousModifications2()
|
||
|
{
|
||
|
QTextTableFormat tableFmt;
|
||
|
|
||
|
cursor.insertTable(2, 5, tableFmt);
|
||
|
QCOMPARE(doc->toPlainText().size(), 11);
|
||
|
QTextTable *table = cursor.currentTable();
|
||
|
QCOMPARE(cursor.position(), 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 5);
|
||
|
|
||
|
table->insertColumns(0, 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 6);
|
||
|
table->insertColumns(6, 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 7);
|
||
|
|
||
|
table->insertRows(0, 1);
|
||
|
QCOMPARE(table->rows(), 3);
|
||
|
QCOMPARE(table->columns(), 7);
|
||
|
table->insertRows(3, 1);
|
||
|
QCOMPARE(table->rows(), 4);
|
||
|
QCOMPARE(table->columns(), 7);
|
||
|
|
||
|
table->removeRows(0, 1);
|
||
|
QCOMPARE(table->rows(), 3);
|
||
|
QCOMPARE(table->columns(), 7);
|
||
|
table->removeRows(2, 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 7);
|
||
|
|
||
|
table->removeColumns(0, 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 6);
|
||
|
table->removeColumns(5, 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 5);
|
||
|
|
||
|
tableFmt = table->format();
|
||
|
table->insertColumns(2, 1);
|
||
|
table->setFormat(tableFmt);
|
||
|
table->insertColumns(2, 1);
|
||
|
QCOMPARE(table->columns(), 7);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::tableManager_undo()
|
||
|
{
|
||
|
QTextTableFormat fmt;
|
||
|
fmt.setBorder(10);
|
||
|
QTextTable *table = cursor.insertTable(2, 2, fmt);
|
||
|
QVERIFY(table);
|
||
|
|
||
|
QCOMPARE(table->format().border(), qreal(10));
|
||
|
|
||
|
fmt.setBorder(20);
|
||
|
table->setFormat(fmt);
|
||
|
|
||
|
QCOMPARE(table->format().border(), qreal(20));
|
||
|
|
||
|
doc->undo();
|
||
|
|
||
|
QCOMPARE(table->format().border(), qreal(10));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::tableManager_removeCell()
|
||
|
{
|
||
|
// essentially a test for TableManager::removeCell, in particular to remove empty items from the rowlist.
|
||
|
// If it fails it'll triger assertions inside TableManager. Yeah, not pretty, should VERIFY here ;(
|
||
|
cursor.insertTable(2, 2, QTextTableFormat());
|
||
|
doc->undo();
|
||
|
// ###
|
||
|
QVERIFY(true);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::rowAt()
|
||
|
{
|
||
|
// test TablePrivate::rowAt
|
||
|
QTextTable *table = cursor.insertTable(4, 2);
|
||
|
|
||
|
QCOMPARE(table->rows(), 4);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
|
||
|
QTextCursor cell00Cursor = table->cellAt(0, 0).firstCursorPosition();
|
||
|
QTextCursor cell10Cursor = table->cellAt(1, 0).firstCursorPosition();
|
||
|
QTextCursor cell20Cursor = table->cellAt(2, 0).firstCursorPosition();
|
||
|
QTextCursor cell21Cursor = table->cellAt(2, 1).firstCursorPosition();
|
||
|
QTextCursor cell30Cursor = table->cellAt(3, 0).firstCursorPosition();
|
||
|
QCOMPARE(table->cellAt(cell00Cursor).firstCursorPosition(), cell00Cursor);
|
||
|
QCOMPARE(table->cellAt(cell10Cursor).firstCursorPosition(), cell10Cursor);
|
||
|
QCOMPARE(table->cellAt(cell20Cursor).firstCursorPosition(), cell20Cursor);
|
||
|
QCOMPARE(table->cellAt(cell30Cursor).firstCursorPosition(), cell30Cursor);
|
||
|
|
||
|
table->mergeCells(1, 0, 2, 1);
|
||
|
|
||
|
QCOMPARE(table->rows(), 4);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
|
||
|
QVERIFY(cell00Cursor == table->cellAt(0, 0).firstCursorPosition());
|
||
|
QVERIFY(cell10Cursor == table->cellAt(1, 0).firstCursorPosition());
|
||
|
QVERIFY(cell10Cursor == table->cellAt(2, 0).firstCursorPosition());
|
||
|
QVERIFY(cell21Cursor == table->cellAt(2, 1).firstCursorPosition());
|
||
|
QVERIFY(cell30Cursor == table->cellAt(3, 0).firstCursorPosition());
|
||
|
|
||
|
table->mergeCells(1, 0, 2, 2);
|
||
|
|
||
|
QCOMPARE(table->rows(), 4);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
|
||
|
QVERIFY(cell00Cursor == table->cellAt(0, 0).firstCursorPosition());
|
||
|
QVERIFY(cell00Cursor == table->cellAt(0, 0).firstCursorPosition());
|
||
|
QVERIFY(cell10Cursor == table->cellAt(1, 0).firstCursorPosition());
|
||
|
QVERIFY(cell10Cursor == table->cellAt(1, 1).firstCursorPosition());
|
||
|
QVERIFY(cell10Cursor == table->cellAt(2, 0).firstCursorPosition());
|
||
|
QVERIFY(cell10Cursor == table->cellAt(2, 1).firstCursorPosition());
|
||
|
QVERIFY(cell30Cursor == table->cellAt(3, 0).firstCursorPosition());
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::rowAtWithSpans()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
|
||
|
table->mergeCells(0, 0, 2, 1);
|
||
|
QVERIFY(table->cellAt(0, 0).rowSpan() == 2);
|
||
|
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
|
||
|
table->mergeCells(0, 0, 2, 2);
|
||
|
QVERIFY(table->cellAt(0, 0).columnSpan() == 2);
|
||
|
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::multiBlockCells()
|
||
|
{
|
||
|
// little testcase for multi-block cells
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
|
||
|
QVERIFY(cursor == table->cellAt(0, 0).firstCursorPosition());
|
||
|
|
||
|
cursor.insertText("Hello");
|
||
|
cursor.insertBlock(QTextBlockFormat());
|
||
|
cursor.insertText("World");
|
||
|
|
||
|
cursor.movePosition(QTextCursor::Left);
|
||
|
QVERIFY(table->cellAt(0, 0) == table->cellAt(cursor));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::insertRows()
|
||
|
{
|
||
|
// little testcase for multi-block cells
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
|
||
|
QVERIFY(cursor == table->cellAt(0, 0).firstCursorPosition());
|
||
|
|
||
|
table->insertRows(0, 1);
|
||
|
QCOMPARE(table->rows(), 3);
|
||
|
|
||
|
table->insertRows(1, 1);
|
||
|
QCOMPARE(table->rows(), 4);
|
||
|
|
||
|
table->insertRows(-1, 1);
|
||
|
QCOMPARE(table->rows(), 5);
|
||
|
|
||
|
table->insertRows(5, 2);
|
||
|
QCOMPARE(table->rows(), 7);
|
||
|
|
||
|
table = cursor.insertTable(5,5);
|
||
|
table->mergeCells(0,0,3,3);
|
||
|
table->insertRows(2,1);
|
||
|
|
||
|
QCOMPARE(table->rows(), 6);
|
||
|
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::deleteInTable()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("Blah");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Foo");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Bar");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Hah");
|
||
|
|
||
|
cursor = table->cellAt(1, 1).firstCursorPosition();
|
||
|
cursor.movePosition(QTextCursor::PreviousBlock, QTextCursor::KeepAnchor);
|
||
|
|
||
|
QCOMPARE(table->cellAt(cursor.position()).row(), 1);
|
||
|
QCOMPARE(table->cellAt(cursor.position()).column(), 0);
|
||
|
|
||
|
cursor.removeSelectedText();
|
||
|
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
|
||
|
// verify table is still all in shape. Only the text inside should get deleted
|
||
|
for (int row = 0; row < table->rows(); ++row)
|
||
|
for (int col = 0; col < table->columns(); ++col) {
|
||
|
const QTextTableCell cell = table->cellAt(row, col);
|
||
|
QVERIFY(cell.isValid());
|
||
|
QCOMPARE(cell.rowSpan(), 1);
|
||
|
QCOMPARE(cell.columnSpan(), 1);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
QTextTable *tst_QTextTable::create2x2Table()
|
||
|
{
|
||
|
cleanup();
|
||
|
init();
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("Blah");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Foo");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Bar");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Hah");
|
||
|
return table;
|
||
|
}
|
||
|
|
||
|
QTextTable *tst_QTextTable::create4x4Table()
|
||
|
{
|
||
|
cleanup();
|
||
|
init();
|
||
|
QTextTable *table = cursor.insertTable(4, 4);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("Blah");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Foo");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Bar");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Hah");
|
||
|
return table;
|
||
|
}
|
||
|
|
||
|
QTextTable *tst_QTextTable::createTable(int rows, int cols)
|
||
|
{
|
||
|
cleanup();
|
||
|
init();
|
||
|
QTextTable *table = cursor.insertTable(rows, cols);
|
||
|
return table;
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::mergeCells()
|
||
|
{
|
||
|
QTextTable *table = create4x4Table();
|
||
|
|
||
|
table->mergeCells(1, 1, 1, 2);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(1, 2));
|
||
|
|
||
|
table->mergeCells(1, 1, 2, 2);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 2));
|
||
|
|
||
|
table = create4x4Table();
|
||
|
|
||
|
table->mergeCells(1, 1, 2, 1);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 1));
|
||
|
|
||
|
table->mergeCells(1, 1, 2, 2);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 2));
|
||
|
|
||
|
table = create4x4Table();
|
||
|
|
||
|
table->mergeCells(1, 1, 2, 2);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 2));
|
||
|
|
||
|
// should do nothing
|
||
|
table->mergeCells(1, 1, 1, 1);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 2));
|
||
|
|
||
|
table = create2x2Table();
|
||
|
|
||
|
table->mergeCells(0, 1, 2, 1);
|
||
|
table->mergeCells(0, 0, 2, 2);
|
||
|
QVERIFY(table->cellAt(0, 0) == table->cellAt(0, 1));
|
||
|
QVERIFY(table->cellAt(0, 0) == table->cellAt(1, 0));
|
||
|
QVERIFY(table->cellAt(0, 0) == table->cellAt(1, 1));
|
||
|
|
||
|
QTextBlock block = table->cellAt(0, 0).firstCursorPosition().block();
|
||
|
|
||
|
QCOMPARE(block.text(), QLatin1String("Blah Foo"));
|
||
|
QCOMPARE(block.next().text(), QLatin1String("Hah"));
|
||
|
QCOMPARE(block.next().next().text(), QLatin1String("Bar"));
|
||
|
|
||
|
table = create4x4Table();
|
||
|
|
||
|
QTextCursor cursor = table->cellAt(3, 3).firstCursorPosition();
|
||
|
QTextTable *t2 = cursor.insertTable(2, 2);
|
||
|
t2->cellAt(0, 0).firstCursorPosition().insertText("Test");
|
||
|
|
||
|
table->mergeCells(2, 2, 2, 2);
|
||
|
cursor = table->cellAt(2, 2).firstCursorPosition();
|
||
|
|
||
|
QTextFrame *frame = cursor.currentFrame();
|
||
|
|
||
|
QTextFrame::iterator it = frame->begin();
|
||
|
|
||
|
// find the embedded table
|
||
|
while (it != frame->end() && !it.currentFrame())
|
||
|
++it;
|
||
|
|
||
|
table = qobject_cast<QTextTable *>(it.currentFrame());
|
||
|
|
||
|
QVERIFY(table);
|
||
|
|
||
|
if (table) {
|
||
|
cursor = table->cellAt(0, 0).firstCursorPosition();
|
||
|
|
||
|
QCOMPARE(cursor.block().text(), QLatin1String("Test"));
|
||
|
}
|
||
|
|
||
|
table = create2x2Table();
|
||
|
|
||
|
table->mergeCells(0, 1, 2, 1);
|
||
|
|
||
|
QVERIFY(table->cellAt(0, 0) != table->cellAt(0, 1));
|
||
|
QVERIFY(table->cellAt(0, 1) == table->cellAt(1, 1));
|
||
|
|
||
|
// should do nothing
|
||
|
table->mergeCells(0, 0, 1, 2);
|
||
|
|
||
|
QVERIFY(table->cellAt(0, 0) != table->cellAt(0, 1));
|
||
|
QVERIFY(table->cellAt(0, 1) == table->cellAt(1, 1));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::mergeAndInsert()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(4,3);
|
||
|
table->mergeCells(0,1,3,2);
|
||
|
table->mergeCells(3,0,1,3);
|
||
|
//Don't crash !
|
||
|
table->insertColumns(1,2);
|
||
|
QCOMPARE(table->columns(), 5);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::splitCells()
|
||
|
{
|
||
|
QTextTable *table = create4x4Table();
|
||
|
table->mergeCells(1, 1, 2, 2);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 2));
|
||
|
|
||
|
table->splitCell(1, 1, 1, 2);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(2, 2));
|
||
|
|
||
|
table->splitCell(1, 1, 1, 1);
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(2, 2));
|
||
|
|
||
|
|
||
|
table = create4x4Table();
|
||
|
table->mergeCells(1, 1, 2, 2);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 2));
|
||
|
|
||
|
table->splitCell(1, 1, 2, 1);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(2, 2));
|
||
|
|
||
|
table->splitCell(1, 1, 1, 1);
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(2, 2));
|
||
|
|
||
|
|
||
|
table = create4x4Table();
|
||
|
table->mergeCells(1, 1, 2, 2);
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) == table->cellAt(2, 2));
|
||
|
|
||
|
table->splitCell(1, 1, 1, 1);
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(1, 2));
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(2, 1));
|
||
|
QVERIFY(table->cellAt(1, 1) != table->cellAt(2, 2));
|
||
|
|
||
|
table = createTable(2, 5);
|
||
|
table->mergeCells(0, 0, 2, 1);
|
||
|
table->mergeCells(0, 1, 2, 1);
|
||
|
QVERIFY(table->cellAt(0, 0) == table->cellAt(1, 0));
|
||
|
QVERIFY(table->cellAt(0, 1) == table->cellAt(1, 1));
|
||
|
table->splitCell(0, 0, 1, 1);
|
||
|
QVERIFY(table->cellAt(0, 0) != table->cellAt(1, 0));
|
||
|
QVERIFY(table->cellAt(0, 1) == table->cellAt(1, 1));
|
||
|
|
||
|
table = createTable(2, 5);
|
||
|
table->mergeCells(0, 4, 2, 1);
|
||
|
QVERIFY(table->cellAt(0, 4) == table->cellAt(1, 4));
|
||
|
|
||
|
table->splitCell(0, 4, 1, 1);
|
||
|
QVERIFY(table->cellAt(0, 4) != table->cellAt(1, 4));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::blocksForTableShouldHaveEmptyFormat()
|
||
|
{
|
||
|
QTextBlockFormat fmt;
|
||
|
fmt.setProperty(QTextFormat::UserProperty, true);
|
||
|
cursor.insertBlock(fmt);
|
||
|
QVERIFY(cursor.blockFormat().hasProperty(QTextFormat::UserProperty));
|
||
|
|
||
|
QTextTable *table = cursor.insertTable(1, 1);
|
||
|
QVERIFY(!table->cellAt(0, 0).firstCursorPosition().blockFormat().hasProperty(QTextFormat::UserProperty));
|
||
|
|
||
|
int userPropCount = 0;
|
||
|
for (QTextBlock block = doc->begin();
|
||
|
block.isValid(); block = block.next()) {
|
||
|
if (block.blockFormat().hasProperty(QTextFormat::UserProperty))
|
||
|
userPropCount++;
|
||
|
}
|
||
|
QCOMPARE(userPropCount, 1);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeTableByRemoveRows()
|
||
|
{
|
||
|
QPointer<QTextTable> table1 = QTextCursor(cursor).insertTable(4, 4);
|
||
|
QPointer<QTextTable> table2 = QTextCursor(cursor).insertTable(4, 4);
|
||
|
QPointer<QTextTable> table3 = QTextCursor(cursor).insertTable(4, 4);
|
||
|
|
||
|
QVERIFY(table1);
|
||
|
QVERIFY(table2);
|
||
|
QVERIFY(table3);
|
||
|
|
||
|
table2->removeRows(1, 1);
|
||
|
|
||
|
QVERIFY(table1);
|
||
|
QVERIFY(table2);
|
||
|
QVERIFY(table3);
|
||
|
|
||
|
table2->removeRows(0, table2->rows());
|
||
|
|
||
|
QVERIFY(table1);
|
||
|
QVERIFY(!table2);
|
||
|
QVERIFY(table3);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeTableByRemoveColumns()
|
||
|
{
|
||
|
QPointer<QTextTable> table1 = QTextCursor(cursor).insertTable(4, 4);
|
||
|
QPointer<QTextTable> table2 = QTextCursor(cursor).insertTable(4, 4);
|
||
|
QPointer<QTextTable> table3 = QTextCursor(cursor).insertTable(4, 4);
|
||
|
|
||
|
QVERIFY(table1);
|
||
|
QVERIFY(table2);
|
||
|
QVERIFY(table3);
|
||
|
|
||
|
table2->removeColumns(1, 1);
|
||
|
|
||
|
QVERIFY(table1);
|
||
|
QVERIFY(table2);
|
||
|
QVERIFY(table3);
|
||
|
|
||
|
table2->removeColumns(0, table2->columns());
|
||
|
|
||
|
QVERIFY(table1);
|
||
|
QVERIFY(!table2);
|
||
|
QVERIFY(table3);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::setCellFormat()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("First");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Second");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Third");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Fourth");
|
||
|
QTextTableCell cell = table->cellAt(0, 0);
|
||
|
QTextCharFormat fmt;
|
||
|
fmt.setObjectIndex(23);
|
||
|
fmt.setBackground(Qt::blue);
|
||
|
fmt.setTableCellColumnSpan(25);
|
||
|
fmt.setTableCellRowSpan(42);
|
||
|
cell.setFormat(fmt);
|
||
|
QCOMPARE(cell.format().background().color(), QColor(Qt::blue));
|
||
|
QCOMPARE(cell.format().tableCellColumnSpan(), 1);
|
||
|
QCOMPARE(cell.format().tableCellRowSpan(), 1);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeRows1()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("First");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Second");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Third");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Fourth");
|
||
|
table->removeRows(0, 1);
|
||
|
QCOMPARE(table->rows(), 1);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstCursorPosition().block().text(), QString("Third"));
|
||
|
QCOMPARE(table->cellAt(0, 1).firstCursorPosition().block().text(), QString("Fourth"));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeRows2()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("First");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Second");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Third");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Fourth");
|
||
|
table->removeRows(1, 1);
|
||
|
QCOMPARE(table->rows(), 1);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstCursorPosition().block().text(), QString("First"));
|
||
|
QCOMPARE(table->cellAt(0, 1).firstCursorPosition().block().text(), QString("Second"));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeRows3()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(3, 2);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("First");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Second");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Third");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Fourth");
|
||
|
table->cellAt(2, 0).firstCursorPosition().insertText("Fifth");
|
||
|
table->cellAt(2, 1).firstCursorPosition().insertText("Sixth");
|
||
|
table->removeRows(1, 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstCursorPosition().block().text(), QString("First"));
|
||
|
QCOMPARE(table->cellAt(0, 1).firstCursorPosition().block().text(), QString("Second"));
|
||
|
QCOMPARE(table->cellAt(1, 0).firstCursorPosition().block().text(), QString("Fifth"));
|
||
|
QCOMPARE(table->cellAt(1, 1).firstCursorPosition().block().text(), QString("Sixth"));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeRows4()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(4, 2);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("First");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Second");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Third");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Fourth");
|
||
|
table->cellAt(2, 0).firstCursorPosition().insertText("Fifth");
|
||
|
table->cellAt(2, 1).firstCursorPosition().insertText("Sixth");
|
||
|
table->cellAt(3, 0).firstCursorPosition().insertText("Seventh");
|
||
|
table->cellAt(3, 1).firstCursorPosition().insertText("Eighth");
|
||
|
table->removeRows(1, 2);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstCursorPosition().block().text(), QString("First"));
|
||
|
QCOMPARE(table->cellAt(0, 1).firstCursorPosition().block().text(), QString("Second"));
|
||
|
QCOMPARE(table->cellAt(1, 0).firstCursorPosition().block().text(), QString("Seventh"));
|
||
|
QCOMPARE(table->cellAt(1, 1).firstCursorPosition().block().text(), QString("Eighth"));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeRows5()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(2,2);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("First");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Second");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Third");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Fourth");
|
||
|
table->insertRows(1,1);
|
||
|
table->mergeCells(1,0,1,2);
|
||
|
table->removeRows(1,1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstCursorPosition().block().text(), QString("First"));
|
||
|
QCOMPARE(table->cellAt(0, 1).firstCursorPosition().block().text(), QString("Second"));
|
||
|
QCOMPARE(table->cellAt(1, 0).firstCursorPosition().block().text(), QString("Third"));
|
||
|
QCOMPARE(table->cellAt(1, 1).firstCursorPosition().block().text(), QString("Fourth"));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeColumns1()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("First");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Second");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Third");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Fourth");
|
||
|
table->removeColumns(0, 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 1);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstCursorPosition().block().text(), QString("Second"));
|
||
|
QCOMPARE(table->cellAt(1, 0).firstCursorPosition().block().text(), QString("Fourth"));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeColumns2()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(2, 2);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("First");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Second");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Third");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Fourth");
|
||
|
table->removeColumns(1, 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 1);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstCursorPosition().block().text(), QString("First"));
|
||
|
QCOMPARE(table->cellAt(1, 0).firstCursorPosition().block().text(), QString("Third"));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeColumns3()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(2, 3);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("First");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Second");
|
||
|
table->cellAt(0, 2).firstCursorPosition().insertText("Third");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Fourth");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Fifth");
|
||
|
table->cellAt(1, 2).firstCursorPosition().insertText("Sixth");
|
||
|
table->removeColumns(1, 1);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstCursorPosition().block().text(), QString("First"));
|
||
|
QCOMPARE(table->cellAt(0, 1).firstCursorPosition().block().text(), QString("Third"));
|
||
|
QCOMPARE(table->cellAt(1, 0).firstCursorPosition().block().text(), QString("Fourth"));
|
||
|
QCOMPARE(table->cellAt(1, 1).firstCursorPosition().block().text(), QString("Sixth"));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeColumns4()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(2, 4);
|
||
|
table->cellAt(0, 0).firstCursorPosition().insertText("First");
|
||
|
table->cellAt(0, 1).firstCursorPosition().insertText("Second");
|
||
|
table->cellAt(0, 2).firstCursorPosition().insertText("Third");
|
||
|
table->cellAt(0, 3).firstCursorPosition().insertText("Fourth");
|
||
|
table->cellAt(1, 0).firstCursorPosition().insertText("Fifth");
|
||
|
table->cellAt(1, 1).firstCursorPosition().insertText("Sixth");
|
||
|
table->cellAt(1, 2).firstCursorPosition().insertText("Seventh");
|
||
|
table->cellAt(1, 3).firstCursorPosition().insertText("Eighth");
|
||
|
table->removeColumns(1, 2);
|
||
|
QCOMPARE(table->rows(), 2);
|
||
|
QCOMPARE(table->columns(), 2);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstCursorPosition().block().text(), QString("First"));
|
||
|
QCOMPARE(table->cellAt(0, 1).firstCursorPosition().block().text(), QString("Fourth"));
|
||
|
QCOMPARE(table->cellAt(1, 0).firstCursorPosition().block().text(), QString("Fifth"));
|
||
|
QCOMPARE(table->cellAt(1, 1).firstCursorPosition().block().text(), QString("Eighth"));
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeColumns5()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(4, 4);
|
||
|
QTextCursor tc (doc);
|
||
|
tc.setPosition(table->cellAt(2,0).firstPosition());
|
||
|
tc.setPosition(table->cellAt(3,1).firstPosition(), QTextCursor::KeepAnchor);
|
||
|
table->mergeCells(tc);
|
||
|
QCOMPARE(table->rows(), 4);
|
||
|
QCOMPARE(table->columns(), 4);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstPosition(), 1);
|
||
|
QCOMPARE(table->cellAt(0, 1).firstPosition(), 2);
|
||
|
QCOMPARE(table->cellAt(0, 2).firstPosition(), 3);
|
||
|
QCOMPARE(table->cellAt(0, 3).firstPosition(), 4);
|
||
|
QCOMPARE(table->cellAt(1, 0).firstPosition(), 5);
|
||
|
QCOMPARE(table->cellAt(1, 1).firstPosition(), 6);
|
||
|
QCOMPARE(table->cellAt(1, 2).firstPosition(), 7);
|
||
|
QCOMPARE(table->cellAt(1, 3).firstPosition(), 8);
|
||
|
QCOMPARE(table->cellAt(2, 0).firstPosition(), 9);
|
||
|
QCOMPARE(table->cellAt(2, 0).rowSpan(), 2);
|
||
|
QCOMPARE(table->cellAt(2, 0).columnSpan(), 2);
|
||
|
QCOMPARE(table->cellAt(2, 1).firstPosition(), 9);
|
||
|
QCOMPARE(table->cellAt(2, 2).firstPosition(), 10);
|
||
|
QCOMPARE(table->cellAt(2, 3).firstPosition(), 11);
|
||
|
QCOMPARE(table->cellAt(3, 0).firstPosition(), 9);
|
||
|
QCOMPARE(table->cellAt(3, 1).firstPosition(), 9);
|
||
|
QCOMPARE(table->cellAt(3, 2).firstPosition(), 12);
|
||
|
QCOMPARE(table->cellAt(3, 3).firstPosition(), 13);
|
||
|
|
||
|
table->removeColumns(1, 1);
|
||
|
QCOMPARE(table->rows(), 4);
|
||
|
QCOMPARE(table->columns(), 3);
|
||
|
QCOMPARE(table->cellAt(0, 0).firstPosition(), 1);
|
||
|
QCOMPARE(table->cellAt(0, 1).firstPosition(), 2);
|
||
|
QCOMPARE(table->cellAt(0, 2).firstPosition(), 3);
|
||
|
QCOMPARE(table->cellAt(1, 0).firstPosition(), 4);
|
||
|
QCOMPARE(table->cellAt(1, 1).firstPosition(), 5);
|
||
|
QCOMPARE(table->cellAt(1, 2).firstPosition(), 6);
|
||
|
QCOMPARE(table->cellAt(2, 0).firstPosition(), 7);
|
||
|
QCOMPARE(table->cellAt(2, 0).rowSpan(), 2);
|
||
|
QCOMPARE(table->cellAt(2, 0).columnSpan(), 1);
|
||
|
QCOMPARE(table->cellAt(2, 1).firstPosition(), 8);
|
||
|
QCOMPARE(table->cellAt(2, 2).firstPosition(), 9);
|
||
|
QCOMPARE(table->cellAt(3, 0).firstPosition(), 7);
|
||
|
QCOMPARE(table->cellAt(3, 1).firstPosition(), 10);
|
||
|
QCOMPARE(table->cellAt(3, 2).firstPosition(), 11);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::removeColumnsInTableWithMergedRows()
|
||
|
{
|
||
|
QTextTable *table = cursor.insertTable(3, 4);
|
||
|
table->mergeCells(0, 0, 1, 4);
|
||
|
QCOMPARE(table->rows(), 3);
|
||
|
QCOMPARE(table->columns(), 4);
|
||
|
|
||
|
table->removeColumns(0, table->columns() - 1);
|
||
|
|
||
|
QCOMPARE(table->rows(), 3);
|
||
|
QCOMPARE(table->columns(), 1);
|
||
|
}
|
||
|
|
||
|
#ifndef QT_NO_WIDGETS
|
||
|
void tst_QTextTable::QTBUG11282_insertBeforeMergedEnding_data()
|
||
|
{
|
||
|
QTest::addColumn<int>("rows");
|
||
|
QTest::addColumn<int>("columns");
|
||
|
QTest::addColumn<QList<int> >("merge");
|
||
|
QTest::addColumn<QList<int> >("insert");
|
||
|
|
||
|
QTest::newRow("2x3, merge two, insert one") << 2 << 3 << (QList<int>() << 1 << 2 << 2)
|
||
|
<< (QList<int>() << 1 << 1) ;
|
||
|
QTest::newRow("3x4, merge three, insert one") << 3 << 4 << (QList<int>() << 1 << 3 << 3)
|
||
|
<< (QList<int>() << 1 << 1) ;
|
||
|
QTest::newRow("4x3, merge two, insert two") << 4 << 3 << (QList<int>() << 1 << 4 << 2)
|
||
|
<< (QList<int>() << 1 << 2) ;
|
||
|
QTest::newRow("4x4, merge middle two, insert one") << 4 << 4 << (QList<int>() << 1 << 4 << 2)
|
||
|
<< (QList<int>() << 1 << 1) ;
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::QTBUG11282_insertBeforeMergedEnding()
|
||
|
{
|
||
|
QFETCH(int, rows);
|
||
|
QFETCH(int, columns);
|
||
|
QFETCH(QList<int>, merge);
|
||
|
QFETCH(QList<int>, insert);
|
||
|
QTextTable *table = cursor.insertTable(rows, columns);
|
||
|
QTextEdit *textEdit = new QTextEdit;
|
||
|
textEdit->setDocument(doc);
|
||
|
textEdit->show();
|
||
|
QVERIFY(QTest::qWaitForWindowExposed(textEdit));
|
||
|
table->mergeCells(0,merge.at(0), merge.at(1), merge.at(2));
|
||
|
//Don't crash !
|
||
|
table->insertColumns(insert.at(0), insert.at(1));
|
||
|
//Check that the final size is what we expected
|
||
|
QCOMPARE(table->rows(), rows);
|
||
|
QCOMPARE(table->columns(), columns + insert.at(1));
|
||
|
delete textEdit;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
void tst_QTextTable::QTBUG22011_insertBeforeRowSpan()
|
||
|
{
|
||
|
QTextDocument doc;
|
||
|
QTextCursor cursor(&doc);
|
||
|
QTextTable *table = cursor.insertTable(1,1); // 1x1
|
||
|
|
||
|
table->appendColumns(1); // 1x2
|
||
|
table->appendRows(1); // 2x2
|
||
|
table->mergeCells(0, 0, 2, 1); // 2x2
|
||
|
table->insertColumns(1, 1); // 2x3
|
||
|
table->mergeCells(0, 1, 1, 2); // 2x3
|
||
|
table->appendRows(1); // 3x3
|
||
|
table->mergeCells(0, 0, 3, 1); // 3x3
|
||
|
table->appendRows(1); // 4x3
|
||
|
table->insertColumns(1, 1); // 4x4
|
||
|
table->mergeCells(0, 1, 1, 3);
|
||
|
table->mergeCells(1, 1, 1, 2);
|
||
|
table->mergeCells(2, 1, 1, 2);
|
||
|
table->mergeCells(3, 0, 1, 2);
|
||
|
table->insertColumns(3, 1); // 4x5
|
||
|
table->mergeCells(0, 1, 1, 4);
|
||
|
|
||
|
table->appendColumns(1); // 4x6
|
||
|
|
||
|
QCOMPARE(table->rows(), 4);
|
||
|
QCOMPARE(table->columns(), 6);
|
||
|
}
|
||
|
|
||
|
#if !defined(QT_NO_PRINTER) && defined(QT_BUILD_INTERNAL)
|
||
|
namespace {
|
||
|
class QTBUG31330_PaintDevice : public QPagedPaintDevice
|
||
|
{
|
||
|
public:
|
||
|
class PaintEngine : public QPaintEngine
|
||
|
{
|
||
|
public:
|
||
|
QList<QRectF> rects;
|
||
|
|
||
|
PaintEngine()
|
||
|
: QPaintEngine(QPaintEngine::PaintEngineFeatures{ })
|
||
|
{}
|
||
|
virtual Type type() const override
|
||
|
{
|
||
|
return User;
|
||
|
}
|
||
|
virtual bool begin(QPaintDevice *) override
|
||
|
{
|
||
|
return true;
|
||
|
}
|
||
|
virtual bool end() override
|
||
|
{
|
||
|
return true;
|
||
|
}
|
||
|
virtual void updateState(const QPaintEngineState &) override
|
||
|
{}
|
||
|
virtual void drawRects(const QRect *, int) override
|
||
|
{}
|
||
|
virtual void drawRects(const QRectF *r, int) override
|
||
|
{
|
||
|
if (painter()->brush() == QBrush(Qt::green))
|
||
|
{
|
||
|
rects.append(*r);
|
||
|
}
|
||
|
}
|
||
|
virtual void drawPixmap(const QRectF &, const QPixmap &, const QRectF &) override
|
||
|
{}
|
||
|
};
|
||
|
|
||
|
class QDummyPagedPaintDevicePrivate : public QPagedPaintDevicePrivate
|
||
|
{
|
||
|
bool setPageLayout(const QPageLayout &newPageLayout) override
|
||
|
{
|
||
|
m_pageLayout = newPageLayout;
|
||
|
return m_pageLayout.isEquivalentTo(newPageLayout);
|
||
|
}
|
||
|
|
||
|
bool setPageSize(const QPageSize &pageSize) override
|
||
|
{
|
||
|
m_pageLayout.setPageSize(pageSize);
|
||
|
return m_pageLayout.pageSize().isEquivalentTo(pageSize);
|
||
|
}
|
||
|
|
||
|
bool setPageOrientation(QPageLayout::Orientation orientation) override
|
||
|
{
|
||
|
m_pageLayout.setOrientation(orientation);
|
||
|
return m_pageLayout.orientation() == orientation;
|
||
|
}
|
||
|
|
||
|
bool setPageMargins(const QMarginsF &margins, QPageLayout::Unit units) override
|
||
|
{
|
||
|
m_pageLayout.setUnits(units);
|
||
|
m_pageLayout.setMargins(margins);
|
||
|
return m_pageLayout.margins() == margins && m_pageLayout.units() == units;
|
||
|
}
|
||
|
|
||
|
QPageLayout pageLayout() const override
|
||
|
{
|
||
|
return m_pageLayout;
|
||
|
}
|
||
|
|
||
|
QPageLayout m_pageLayout;
|
||
|
};
|
||
|
|
||
|
int pages;
|
||
|
QPaintEngine* engine;
|
||
|
|
||
|
QTBUG31330_PaintDevice(QPaintEngine* engine)
|
||
|
: QPagedPaintDevice(new QDummyPagedPaintDevicePrivate), pages(1), engine(engine)
|
||
|
{
|
||
|
QPageLayout layout = pageLayout();
|
||
|
layout.setUnits(QPageLayout::Point);
|
||
|
setPageLayout(layout);
|
||
|
}
|
||
|
virtual int metric(PaintDeviceMetric metric) const override
|
||
|
{
|
||
|
if (PdmDevicePixelRatio == metric)
|
||
|
return 1;
|
||
|
if (PdmDevicePixelRatioScaled == metric)
|
||
|
return 1 * QPaintDevice::devicePixelRatioFScale();
|
||
|
if (PdmDpiY == metric)
|
||
|
return 96;
|
||
|
if (PdmDpiX == metric)
|
||
|
return 96;
|
||
|
if (PdmHeight == metric)
|
||
|
return 1000;
|
||
|
if (PdmWidth == metric)
|
||
|
return 700;
|
||
|
return 900;
|
||
|
}
|
||
|
virtual QPaintEngine *paintEngine() const override
|
||
|
{
|
||
|
return engine;
|
||
|
}
|
||
|
bool newPage() override
|
||
|
{
|
||
|
++pages;
|
||
|
return true;
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::QTBUG31330_renderBackground()
|
||
|
{
|
||
|
QTextDocument doc;
|
||
|
QTextCursor cursor(&doc);
|
||
|
QTextTable* table = cursor.insertTable(4, 2);
|
||
|
|
||
|
QTextTableCell cell = table->cellAt(3, 0);
|
||
|
|
||
|
QTextCharFormat cellFormat = cell.format();
|
||
|
cellFormat.setBackground(QBrush(Qt::green));
|
||
|
cell.setFormat(cellFormat);
|
||
|
|
||
|
QTextCursor tc = cell.firstCursorPosition();
|
||
|
for (int i = 0; i < 100; ++i) {
|
||
|
tc.insertBlock();
|
||
|
}
|
||
|
QTBUG31330_PaintDevice::PaintEngine engine;
|
||
|
QTBUG31330_PaintDevice paintDevice(&engine);
|
||
|
paintDevice.setPageSize(QPageSize(QPageSize::A4));
|
||
|
doc.print(&paintDevice);
|
||
|
|
||
|
QVERIFY(paintDevice.pages >= 2);
|
||
|
QCOMPARE(engine.rects.size(), paintDevice.pages);
|
||
|
for (int i = 0; i < engine.rects.size(); ++i) {
|
||
|
QRectF rect = engine.rects[i];
|
||
|
QVERIFY(rect.top() > 0);
|
||
|
QVERIFY(rect.bottom() < 1000);
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
void tst_QTextTable::checkBorderAttributes_data()
|
||
|
{
|
||
|
QTest::addColumn<QString>("html");
|
||
|
QTest::addColumn<qreal>("topBorderWidth");
|
||
|
QTest::addColumn<qreal>("bottomBorderWidth");
|
||
|
QTest::addColumn<qreal>("leftBorderWidth");
|
||
|
QTest::addColumn<qreal>("rightBorderWidth");
|
||
|
QTest::addColumn<QTextFrameFormat::BorderStyle>("topBorderStyle");
|
||
|
QTest::addColumn<QTextFrameFormat::BorderStyle>("bottomBorderStyle");
|
||
|
QTest::addColumn<QTextFrameFormat::BorderStyle>("leftBorderStyle");
|
||
|
QTest::addColumn<QTextFrameFormat::BorderStyle>("rightBorderStyle");
|
||
|
QTest::addColumn<QBrush>("topBorderBrush");
|
||
|
QTest::addColumn<QBrush>("bottomBorderBrush");
|
||
|
QTest::addColumn<QBrush>("leftBorderBrush");
|
||
|
QTest::addColumn<QBrush>("rightBorderBrush");
|
||
|
|
||
|
const QString tableHtmlStart = QStringLiteral("<html><head><style>");
|
||
|
const QString tableHtmlEnd = QStringLiteral("</style></head><body>"
|
||
|
"<table border=\"1\"><tr><td>One</td><td>Two</td></tr>"
|
||
|
"<tr><td>Three</td><td>Four</td></tr></table></body></html>");
|
||
|
QTest::newRow("1px-solid-colors")
|
||
|
<< QString("%1"
|
||
|
"td {"
|
||
|
"border-top: 1px solid red;"
|
||
|
"border-bottom: 1px solid blue;"
|
||
|
"border-left: 1px solid green;"
|
||
|
"border-right: 1px solid yellow;"
|
||
|
"}"
|
||
|
"%2").arg(tableHtmlStart).arg(tableHtmlEnd)
|
||
|
<< 1.0 << 1.0 << 1.0 << 1.0
|
||
|
<< QTextFrameFormat::BorderStyle_Solid << QTextFrameFormat::BorderStyle_Solid
|
||
|
<< QTextFrameFormat::BorderStyle_Solid << QTextFrameFormat::BorderStyle_Solid
|
||
|
<< QBrush(Qt::red) << QBrush(Qt::blue) << QBrush(Qt::darkGreen) << QBrush(Qt::yellow);
|
||
|
QTest::newRow("MixedWidth-solid-colors")
|
||
|
<< QString("%1"
|
||
|
"td {"
|
||
|
"border-top: 1px solid red;"
|
||
|
"border-bottom: 2px solid blue;"
|
||
|
"border-left: 3px solid green;"
|
||
|
"border-right: 4px solid yellow;"
|
||
|
"}"
|
||
|
"%2").arg(tableHtmlStart).arg(tableHtmlEnd)
|
||
|
<< 1.0 << 2.0 << 3.0 << 4.0
|
||
|
<< QTextFrameFormat::BorderStyle_Solid << QTextFrameFormat::BorderStyle_Solid
|
||
|
<< QTextFrameFormat::BorderStyle_Solid << QTextFrameFormat::BorderStyle_Solid
|
||
|
<< QBrush(Qt::red) << QBrush(Qt::blue) << QBrush(Qt::darkGreen) << QBrush(Qt::yellow);
|
||
|
QTest::newRow("MixedWidth-MixedStyle-colors")
|
||
|
<< QString("%1"
|
||
|
"td {"
|
||
|
"border-top: 1px solid red;"
|
||
|
"border-bottom: 2px dotted blue;"
|
||
|
"border-left: 3px dashed green;"
|
||
|
"border-right: 4px inset yellow;"
|
||
|
"}"
|
||
|
"%2").arg(tableHtmlStart).arg(tableHtmlEnd)
|
||
|
<< 1.0 << 2.0 << 3.0 << 4.0
|
||
|
<< QTextFrameFormat::BorderStyle_Solid << QTextFrameFormat::BorderStyle_Dotted
|
||
|
<< QTextFrameFormat::BorderStyle_Dashed << QTextFrameFormat::BorderStyle_Inset
|
||
|
<< QBrush(Qt::red) << QBrush(Qt::blue) << QBrush(Qt::darkGreen) << QBrush(Qt::yellow);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::checkBorderAttributes()
|
||
|
{
|
||
|
QFETCH(QString, html);
|
||
|
QFETCH(qreal, topBorderWidth);
|
||
|
QFETCH(qreal, bottomBorderWidth);
|
||
|
QFETCH(qreal, leftBorderWidth);
|
||
|
QFETCH(qreal, rightBorderWidth);
|
||
|
QFETCH(QTextFrameFormat::BorderStyle, topBorderStyle);
|
||
|
QFETCH(QTextFrameFormat::BorderStyle, bottomBorderStyle);
|
||
|
QFETCH(QTextFrameFormat::BorderStyle, leftBorderStyle);
|
||
|
QFETCH(QTextFrameFormat::BorderStyle, rightBorderStyle);
|
||
|
QFETCH(QBrush, topBorderBrush);
|
||
|
QFETCH(QBrush, bottomBorderBrush);
|
||
|
QFETCH(QBrush, leftBorderBrush);
|
||
|
QFETCH(QBrush, rightBorderBrush);
|
||
|
|
||
|
QTextDocument doc;
|
||
|
doc.setHtml(html);
|
||
|
QTextCursor cursor(doc.firstBlock());
|
||
|
cursor.movePosition(QTextCursor::Right);
|
||
|
|
||
|
QTextTable *currentTable = cursor.currentTable();
|
||
|
QVERIFY(currentTable);
|
||
|
for (int row = 0; row < 2; row++) {
|
||
|
for (int column = 0; column < 2; column++) {
|
||
|
QTextTableCell cell = currentTable->cellAt(row, column);
|
||
|
QTextCharFormat cellFormat = cell.format();
|
||
|
QCOMPARE(cellFormat.doubleProperty(QTextFormat::TableCellTopBorder), topBorderWidth);
|
||
|
QCOMPARE(cellFormat.doubleProperty(QTextFormat::TableCellBottomBorder), bottomBorderWidth);
|
||
|
QCOMPARE(cellFormat.doubleProperty(QTextFormat::TableCellLeftBorder), leftBorderWidth);
|
||
|
QCOMPARE(cellFormat.doubleProperty(QTextFormat::TableCellRightBorder), rightBorderWidth);
|
||
|
QCOMPARE(cellFormat.property(QTextFormat::TableCellTopBorderStyle), topBorderStyle);
|
||
|
QCOMPARE(cellFormat.property(QTextFormat::TableCellBottomBorderStyle), bottomBorderStyle);
|
||
|
QCOMPARE(cellFormat.property(QTextFormat::TableCellLeftBorderStyle), leftBorderStyle);
|
||
|
QCOMPARE(cellFormat.property(QTextFormat::TableCellRightBorderStyle), rightBorderStyle);
|
||
|
QCOMPARE(cellFormat.brushProperty(QTextFormat::TableCellTopBorderBrush), topBorderBrush);
|
||
|
QCOMPARE(cellFormat.brushProperty(QTextFormat::TableCellBottomBorderBrush), bottomBorderBrush);
|
||
|
QCOMPARE(cellFormat.brushProperty(QTextFormat::TableCellLeftBorderBrush), leftBorderBrush);
|
||
|
QCOMPARE(cellFormat.brushProperty(QTextFormat::TableCellRightBorderBrush), rightBorderBrush);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#ifndef QT_NO_WIDGETS
|
||
|
void tst_QTextTable::columnWidthWithSpans()
|
||
|
{
|
||
|
cleanup();
|
||
|
init();
|
||
|
QTextTable *table = cursor.insertTable(4, 4);
|
||
|
QTextEdit textEdit;
|
||
|
textEdit.setDocument(doc);
|
||
|
textEdit.show();
|
||
|
QVERIFY(QTest::qWaitForWindowExposed(&textEdit));
|
||
|
|
||
|
for (int i = 0; i < table->columns(); ++i)
|
||
|
table->cellAt(0, i).firstCursorPosition().insertText(QString("Header %1").arg(i));
|
||
|
|
||
|
QTextBlock block = table->cellAt(0, 0).firstCursorPosition().block();
|
||
|
const QRectF beforeRect = table->document()->documentLayout()->blockBoundingRect(block);
|
||
|
table->mergeCells(1, 0, 1, table->columns());
|
||
|
block = table->cellAt(0, 0).firstCursorPosition().block();
|
||
|
const QRectF afterRect = table->document()->documentLayout()->blockBoundingRect(block);
|
||
|
QCOMPARE(afterRect, beforeRect);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::columnWidthWithImage_data()
|
||
|
{
|
||
|
const auto imageHtml = [](int width, int height) {
|
||
|
QImage image(width, height, QImage::Format_RGB32);
|
||
|
image.fill(Qt::red);
|
||
|
QByteArray imageBytes;
|
||
|
QBuffer buffer(&imageBytes);
|
||
|
buffer.open(QIODevice::WriteOnly);
|
||
|
image.save(&buffer, "png");
|
||
|
return QString("<td><img src='data:image/png;base64,%1'/></td>").arg(imageBytes.toBase64());
|
||
|
};
|
||
|
|
||
|
QTest::addColumn<QString>("leftHtml");
|
||
|
QTest::addColumn<QString>("rightHtml");
|
||
|
QTest::addColumn<QSize>("imageSize");
|
||
|
QTest::addRow("image")
|
||
|
<< imageHtml(500, 32) << "<td></td>" << QSize(500, 32);
|
||
|
QTest::addRow("image, text")
|
||
|
<< imageHtml(32, 32) << "<td>abc</td>" << QSize(32, 32);
|
||
|
QTest::addRow("image, 100%% text")
|
||
|
<< imageHtml(32, 32) << "<td style='background-color: grey' width='100%'>abc</td>"
|
||
|
<< QSize(32, 32);
|
||
|
QTest::addRow("image, image")
|
||
|
<< imageHtml(256, 32) << imageHtml(256, 32) << QSize(256, 32);
|
||
|
}
|
||
|
|
||
|
void tst_QTextTable::columnWidthWithImage()
|
||
|
{
|
||
|
const QString tableTemplate = "<table><tr>%1 %2</tr></table>";
|
||
|
|
||
|
QFETCH(QString, leftHtml);
|
||
|
QFETCH(QString, rightHtml);
|
||
|
QFETCH(QSize, imageSize);
|
||
|
|
||
|
QTextDocument doc;
|
||
|
doc.setHtml(tableTemplate.arg(leftHtml).arg(rightHtml));
|
||
|
QTextEdit textEdit;
|
||
|
textEdit.setDocument(&doc);
|
||
|
textEdit.show();
|
||
|
QVERIFY(QTest::qWaitForWindowExposed(&textEdit));
|
||
|
|
||
|
QTextCursor cursor(doc.firstBlock());
|
||
|
cursor.movePosition(QTextCursor::Right);
|
||
|
|
||
|
QTextTable *currentTable = cursor.currentTable();
|
||
|
QVERIFY(currentTable);
|
||
|
|
||
|
QTextBlock block = currentTable->cellAt(0, 0).firstCursorPosition().block();
|
||
|
const QRectF leftRect = currentTable->document()->documentLayout()->blockBoundingRect(block);
|
||
|
block = currentTable->cellAt(0, 1).firstCursorPosition().block();
|
||
|
const QRectF rightRect = currentTable->document()->documentLayout()->blockBoundingRect(block);
|
||
|
QCOMPARE(leftRect.size().toSize(), imageSize);
|
||
|
QVERIFY(rightRect.left() > leftRect.right());
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
|
||
|
QTEST_MAIN(tst_QTextTable)
|
||
|
#include "tst_qtexttable.moc"
|