📄 qtexttable.cpp
字号:
/******************************************************************************** Copyright (C) 1992-2006 Trolltech ASA. All rights reserved.**** This file is part of the QtGui module of the Qt Toolkit.**** This file may be used under the terms of the GNU General Public** License version 2.0 as published by the Free Software Foundation** and appearing in the file LICENSE.GPL included in the packaging of** this file. Please review the following information to ensure GNU** General Public Licensing requirements will be met:** http://www.trolltech.com/products/qt/opensource.html**** If you are unsure which license is appropriate for your use, please** review the following information:** http://www.trolltech.com/products/qt/licensing.html or contact the** sales department at sales@trolltech.com.**** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.******************************************************************************/#include "qtexttable.h"#include "qtextcursor.h"#include "qtextformat.h"#include <qdebug.h>#include "qtexttable_p.h"#include "qvarlengtharray.h"#include <stdlib.h>/*! \class QTextTableCell qtexttable.h \brief The QTextTableCell class represents the properties of a cell in a QTextTable. \ingroup text Table cells are pieces of document structure that belong to a table. The table orders cells into particular rows and columns; cells can also span multiple columns and rows. Cells are usually created when a table is inserted into a document with QTextCursor::insertTable(), but they are also created and destroyed when a table is resized. Cells contain information about their location in a table; you can obtain the row() and column() numbers of a cell, and its rowSpan() and columnSpan(). The format() of a cell describes the default character format of its contents. The firstCursorPosition() and lastCursorPosition() functions are used to obtain the extent of the cell in the document. \sa QTextTable QTextTableFormat*//*! \fn QTextTableCell::QTextTableCell() Constructs an invalid table cell. \sa isValid()*//*! \fn QTextTableCell::QTextTableCell(const QTextTableCell &other) Copy constructor. Creates a new QTextTableCell object based on the \a other cell.*//*! \fn QTextTableCell& QTextTableCell::operator=(const QTextTableCell &other) Assigns the \a other table cell to this table cell.*//*! Returns the cell's character format.*/QTextCharFormat QTextTableCell::format() const{ QTextDocumentPrivate *p = table->docHandle(); QTextFormatCollection *c = p->formatCollection(); return c->charFormat(QTextDocumentPrivate::FragmentIterator(&p->fragmentMap(), fragment)->format);}/*! Returns the number of the row in the table that contains this cell. \sa column()*/int QTextTableCell::row() const{ const QTextTablePrivate *tp = table->d_func(); if (tp->dirty) tp->update(); for (int i = 0; i < tp->nCols*tp->nRows; ++i) { if (tp->grid[i] == fragment) return i/tp->nCols; } return -1;}/*! Returns the number of the column in the table that contains this cell. \sa row()*/int QTextTableCell::column() const{ const QTextTablePrivate *tp = table->d_func(); if (tp->dirty) tp->update(); for (int i = 0; i < tp->nCols*tp->nRows; ++i) { if (tp->grid[i] == fragment) return i%tp->nCols; } return -1;}/*! Returns the number of rows this cell spans. The default is 1. \sa columnSpan()*/int QTextTableCell::rowSpan() const{ return format().tableCellRowSpan();}/*! Returns the number of columns this cell spans. The default is 1. \sa rowSpan()*/int QTextTableCell::columnSpan() const{ return format().tableCellColumnSpan();}/*! \fn bool QTextTableCell::isValid() const Returns true if this is a valid table cell; otherwise returns false.*//*! Returns the first valid cursor position in this cell. \sa lastCursorPosition()*/QTextCursor QTextTableCell::firstCursorPosition() const{ return QTextCursor(table->d_func()->pieceTable, firstPosition());}/*! Returns the last valid cursor position in this cell. \sa firstCursorPosition()*/QTextCursor QTextTableCell::lastCursorPosition() const{ return QTextCursor(table->d_func()->pieceTable, lastPosition());}/*! \internal Returns the first valid position in the document occupied by this cell.*/int QTextTableCell::firstPosition() const{ QTextDocumentPrivate *p = table->docHandle(); return p->fragmentMap().position(fragment) + 1;}/*! \internal Returns the last valid position in the document occupied by this cell.*/int QTextTableCell::lastPosition() const{ QTextDocumentPrivate *p = table->docHandle(); const QTextTablePrivate *td = table->d_func(); int index = table->d_func()->findCellIndex(fragment); int f; if (index != -1) f = td->cells.value(index + 1, td->fragment_end); else f = td->fragment_end; return p->fragmentMap().position(f);}/*! Returns a frame iterator pointing to the beginning of the table's cell. \sa end()*/QTextFrame::iterator QTextTableCell::begin() const{ QTextDocumentPrivate *p = table->docHandle(); int b = p->blockMap().findNode(firstPosition()); int e = p->blockMap().findNode(lastPosition()+1); return QTextFrame::iterator(const_cast<QTextTable *>(table), b, b, e);}/*! Returns a frame iterator pointing to the end of the table's cell. \sa begin()*/QTextFrame::iterator QTextTableCell::end() const{ QTextDocumentPrivate *p = table->docHandle(); int b = p->blockMap().findNode(firstPosition()); int e = p->blockMap().findNode(lastPosition()+1); return QTextFrame::iterator(const_cast<QTextTable *>(table), e, b, e);}/*! \fn QTextCursor QTextTableCell::operator==(const QTextTableCell &other) const Returns true if this cell object and the \a other cell object describe the same cell; otherwise returns false.*//*! \fn QTextCursor QTextTableCell::operator!=(const QTextTableCell &other) const Returns true if this cell object and the \a other cell object describe different cells; otherwise returns false.*//*! \fn QTextTableCell::~QTextTableCell() Destroys the table cell.*/QTextTablePrivate::~QTextTablePrivate(){ if (grid) free(grid);}QTextTable *QTextTablePrivate::createTable(QTextDocumentPrivate *pieceTable, int pos, int rows, int cols, const QTextTableFormat &tableFormat){ QTextTableFormat fmt = tableFormat; fmt.setColumns(cols); QTextTable *table = qobject_cast<QTextTable *>(pieceTable->createObject(fmt)); Q_ASSERT(table); pieceTable->beginEditBlock();// qDebug("---> createTable: rows=%d, cols=%d at %d", rows, cols, pos); // add block after table QTextCharFormat charFmt; charFmt.setObjectIndex(table->objectIndex()); int charIdx = pieceTable->formatCollection()->indexForFormat(charFmt); int cellIdx = pieceTable->formatCollection()->indexForFormat(QTextBlockFormat()); for (int i = 0; i < rows*cols; ++i) { pieceTable->insertBlock(QTextBeginningOfFrame, pos, cellIdx, charIdx);// qDebug(" addCell at %d", pos); ++pos; } pieceTable->insertBlock(QTextEndOfFrame, pos, cellIdx, charIdx);// qDebug(" addEOR at %d", pos); ++pos; pieceTable->endEditBlock(); return table;}struct QFragmentFindHelper{ inline QFragmentFindHelper(int _pos, const QTextDocumentPrivate::FragmentMap &map) : pos(_pos), fragmentMap(map) {} uint pos; const QTextDocumentPrivate::FragmentMap &fragmentMap;};static bool operator<(int fragment, const QFragmentFindHelper &helper){ return helper.fragmentMap.position(fragment) < helper.pos;}static bool operator<(const QFragmentFindHelper &helper, int fragment){ return helper.pos < helper.fragmentMap.position(fragment);}int QTextTablePrivate::findCellIndex(int fragment) const{ QFragmentFindHelper helper(pieceTable->fragmentMap().position(fragment), pieceTable->fragmentMap()); QList<int>::ConstIterator it = qBinaryFind(cells.begin(), cells.end(), helper); if (it == cells.end()) return -1; return it - cells.begin();}void QTextTablePrivate::fragmentAdded(const QChar &type, uint fragment){ dirty = true; if (type == QTextBeginningOfFrame) { Q_ASSERT(cells.indexOf(fragment) == -1); const uint pos = pieceTable->fragmentMap().position(fragment); QFragmentFindHelper helper(pos, pieceTable->fragmentMap()); QList<int>::Iterator it = qLowerBound(cells.begin(), cells.end(), helper); cells.insert(it, fragment); if (!fragment_start || pos < pieceTable->fragmentMap().position(fragment_start)) fragment_start = fragment; return; } QTextFramePrivate::fragmentAdded(type, fragment);}void QTextTablePrivate::fragmentRemoved(const QChar &type, uint fragment){ dirty = true; if (type == QTextBeginningOfFrame) { Q_ASSERT(cells.indexOf(fragment) != -1); cells.removeAll(fragment); if (fragment_start == fragment && cells.size()) { fragment_start = cells.at(0); } if (fragment_start != fragment) return; } QTextFramePrivate::fragmentRemoved(type, fragment);}void QTextTablePrivate::update() const{ Q_Q(const QTextTable); nCols = q->format().columns(); nRows = (cells.size() + nCols-1)/nCols;// qDebug(">>>> QTextTablePrivate::update, nRows=%d, nCols=%d", nRows, nCols); grid = (int *)realloc(grid, nRows*nCols*sizeof(int)); memset(grid, 0, nRows*nCols*sizeof(int)); QTextDocumentPrivate *p = pieceTable; QTextFormatCollection *c = p->formatCollection(); int cell = 0; for (int i = 0; i < cells.size(); ++i) { int fragment = cells.at(i); QTextCharFormat fmt = c->charFormat(QTextDocumentPrivate::FragmentIterator(&p->fragmentMap(), fragment)->format); int rowspan = fmt.tableCellRowSpan(); int colspan = fmt.tableCellColumnSpan(); // skip taken cells while (cell < nRows*nCols && grid[cell]) ++cell; int r = cell/nCols; int c = cell%nCols; if (r + rowspan > nRows) { grid = (int *)realloc(grid, sizeof(int)*(r + rowspan)*nCols); memset(grid + (nRows*nCols), 0, sizeof(int)*(r+rowspan-nRows)*nCols); nRows = r + rowspan; } Q_ASSERT(c + colspan <= nCols); for (int ii = 0; ii < rowspan; ++ii) { for (int jj = 0; jj < colspan; ++jj) { Q_ASSERT(grid[(r+ii)*nCols + c+jj] == 0); grid[(r+ii)*nCols + c+jj] = fragment;// qDebug(" setting cell %d span=%d/%d at %d/%d", fragment, rowspan, colspan, r+ii, c+jj); } } }// qDebug("<<<< end: nRows=%d, nCols=%d", nRows, nCols); dirty = false;}/*! \class QTextTable qtexttable.h \brief The QTextTable class represents a table in a QTextDocument. \ingroup text A table is a group of cells ordered into rows and columns. Each table contains at least one row and one column. Each cell contains a block, and is surrounded by a frame. Tables are usually created and inserted into a document with the QTextCursor::insertTable() function. For example, we can insert a table with three rows and two columns at the current cursor position in an editor using the following lines of code: \quotefromfile snippets/textdocument-tables/mainwindow.cpp \skipto QTextCursor cursor(editor \printuntil cursor.movePosition(QTextCursor::Start); \skipto QTextTable *table = cursor \printuntil QTextTable *table = cursor The table format is either defined when the table is created or changed later with setFormat(). The table currently being edited by the cursor is found with QTextCursor::currentTable(). This allows its format or dimensions to be changed after it has been inserted into a document. A table's size can be changed with resize(), or by using insertRows(), insertColumns(), removeRows(), or removeColumns(). Use cellAt() to retrieve table cells. The starting and ending positions of table rows can be found by moving a cursor within a table, and using the rowStart() and rowEnd() functions to obtain cursors at the start and end of each row. \sa QTextTableFormat*//*! \internal */QTextTable::QTextTable(QTextDocument *doc) : QTextFrame(*new QTextTablePrivate, doc){}/*! \internalDestroys the table. */QTextTable::~QTextTable(){}/*! \fn QTextTableCell QTextTable::cellAt(int row, int column) const Returns the table cell at the given \a row and \a column in the table. \sa columns() rows()*/QTextTableCell QTextTable::cellAt(int row, int col) const{ Q_D(const QTextTable); if (d->dirty) d->update(); if (row < 0 || row >= d->nRows || col < 0 || col >= d->nCols) return QTextTableCell(); return QTextTableCell(this, d->grid[row*d->nCols + col]);}/*! \overload Returns the table cell that contains the character at the given \a position in the document.*/QTextTableCell QTextTable::cellAt(int position) const{ Q_D(const QTextTable); if (d->dirty) d->update(); uint pos = (uint)position; const QTextDocumentPrivate::FragmentMap &m = d->pieceTable->fragmentMap(); if (position < 0 || m.position(d->fragment_start) >= pos || m.position(d->fragment_end) < pos) return QTextTableCell(); QFragmentFindHelper helper(position, m); QList<int>::ConstIterator it = qLowerBound(d->cells.begin(), d->cells.end(), helper); if (it != d->cells.begin()) --it; return QTextTableCell(this, *it);}/*! \fn QTextTableCell QTextTable::cellAt(const QTextCursor &cursor) const \overload Returns the table cell containing the given \a cursor.*/QTextTableCell QTextTable::cellAt(const QTextCursor &c) const{ return cellAt(c.position());}/*! \fn void QTextTable::resize(int rows, int columns)
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -