/*! \fn void QTextTable::insertRows(int index, int rows) Inserts a number of \a rows before the row with the specified \a index. \sa resize() insertColumns() removeRows() removeColumns() */ void QTextTable::insertRows(int pos, int num) { Q_D(QTextTable); if (num <= 0) return; if (d->dirty) d->update(); if (pos > d->nRows || pos < 0) pos = d->nRows; // qDebug() << "-------- insertRows" << pos << num; QTextDocumentPrivate *p = d->pieceTable; QTextFormatCollection *c = p->formatCollection(); p->beginEditBlock(); int extended = 0; int insert_before = 0; if (pos > 0 && pos < d->nRows) { for (int i = 0; i < d->nCols; ++i) { int cell = d->grid[pos*d->nCols + i]; if (cell == d->grid[(pos-1)*d->nCols+i]) { // cell spans the insertion place, extend it QTextDocumentPrivate::FragmentIterator it(&p->fragmentMap(), cell); QTextCharFormat fmt = c->charFormat(it->format); fmt.setTableCellRowSpan(fmt.tableCellRowSpan() + num); p->setCharFormat(it.position(), 1, fmt); extended++; } else if (!insert_before) { insert_before = cell; } } } else { insert_before = (pos == 0 ? d->grid[0] : d->fragment_end); } if (extended < d->nCols) { Q_ASSERT(insert_before); QTextDocumentPrivate::FragmentIterator it(&p->fragmentMap(), insert_before); QTextCharFormat fmt = c->charFormat(it->format); fmt.setTableCellRowSpan(1); fmt.setTableCellColumnSpan(1); Q_ASSERT(fmt.objectIndex() == objectIndex()); int pos = it.position(); int cfmt = p->formatCollection()->indexForFormat(fmt); int bfmt = p->formatCollection()->indexForFormat(QTextBlockFormat()); // qDebug("inserting %d cells, nCols=%d extended=%d", num*(d->nCols-extended), d->nCols, extended); for (int i = 0; i < num*(d->nCols-extended); ++i) p->insertBlock(QTextBeginningOfFrame, pos, bfmt, cfmt, QTextUndoCommand::MoveCursor); } // qDebug() << "-------- end insertRows" << pos << num; p->endEditBlock(); }
/*! \fn void QTextTable::insertColumns(int index, int columns) Inserts a number of \a columns before the column with the specified \a index. \sa insertRows() resize() removeRows() removeColumns() appendRows() appendColumns() */ void QTextTable::insertColumns(int pos, int num) { Q_D(QTextTable); if (num <= 0) return; if (d->dirty) d->update(); if (pos > d->nCols || pos < 0) pos = d->nCols; // qDebug() << "-------- insertCols" << pos << num; QTextDocumentPrivate *p = d->pieceTable; QTextFormatCollection *c = p->formatCollection(); p->beginEditBlock(); for (int i = 0; i < d->nRows; ++i) { int cell; if (i == d->nRows - 1 && pos == d->nCols) cell = d->fragment_end; else cell = d->grid[i*d->nCols + pos]; QTextDocumentPrivate::FragmentIterator it(&p->fragmentMap(), cell); QTextCharFormat fmt = c->charFormat(it->format); if (pos > 0 && pos < d->nCols && cell == d->grid[i*d->nCols + pos - 1]) { // cell spans the insertion place, extend it fmt.setTableCellColumnSpan(fmt.tableCellColumnSpan() + num); p->setCharFormat(it.position(), 1, fmt); } else { fmt.setTableCellRowSpan(1); fmt.setTableCellColumnSpan(1); Q_ASSERT(fmt.objectIndex() == objectIndex()); int position = it.position(); int cfmt = p->formatCollection()->indexForFormat(fmt); int bfmt = p->formatCollection()->indexForFormat(QTextBlockFormat()); for (int i = 0; i < num; ++i) p->insertBlock(QTextBeginningOfFrame, position, bfmt, cfmt, QTextUndoCommand::MoveCursor); } } QTextTableFormat tfmt = format(); tfmt.setColumns(tfmt.columns()+num); QVector<QTextLength> columnWidths = tfmt.columnWidthConstraints(); if (! columnWidths.isEmpty()) { for (int i = num; i > 0; --i) columnWidths.insert(pos, columnWidths[qMax(0, pos-1)]); } tfmt.setColumnWidthConstraints (columnWidths); QTextObject::setFormat(tfmt); // qDebug() << "-------- end insertCols" << pos << num; p->endEditBlock(); }
/*! \fn void QTextTable::removeRows(int index, int rows) Removes a number of \a rows starting with the row at the specified \a index. \sa insertRows(), insertColumns(), resize(), removeColumns(), appendRows(), appendColumns() */ void QTextTable::removeRows(int pos, int num) { Q_D(QTextTable); // qDebug() << "-------- removeRows" << pos << num; if (num <= 0 || pos < 0) return; if (d->dirty) d->update(); if (pos >= d->nRows) return; if (pos+num > d->nRows) num = d->nRows - pos; QTextDocumentPrivate *p = d->pieceTable; QTextFormatCollection *collection = p->formatCollection(); p->beginEditBlock(); // delete whole table? if (pos == 0 && num == d->nRows) { const int pos = p->fragmentMap().position(d->fragment_start); p->remove(pos, p->fragmentMap().position(d->fragment_end) - pos + 1); p->endEditBlock(); return; } p->aboutToRemoveCell(cellAt(pos, 0).firstPosition(), cellAt(pos + num - 1, d->nCols - 1).lastPosition()); QList<int> touchedCells; for (int r = pos; r < pos + num; ++r) { for (int c = 0; c < d->nCols; ++c) { int cell = d->grid[r*d->nCols + c]; if (touchedCells.contains(cell)) continue; touchedCells << cell; QTextDocumentPrivate::FragmentIterator it(&p->fragmentMap(), cell); QTextCharFormat fmt = collection->charFormat(it->format); int span = fmt.tableCellRowSpan(); if (span > 1) { fmt.setTableCellRowSpan(span - 1); p->setCharFormat(it.position(), 1, fmt); } else { // remove cell int index = d->cells.indexOf(cell) + 1; int f_end = index < d->cells.size() ? d->cells.at(index) : d->fragment_end; p->remove(it.position(), p->fragmentMap().position(f_end) - it.position()); } } } p->endEditBlock(); // qDebug() << "-------- end removeRows" << pos << num; }
/*! \fn void QTextTable::removeColumns(int index, int columns) Removes a number of \a columns starting with the column at the specified \a index. \sa insertRows() insertColumns() removeRows() resize() */ void QTextTable::removeColumns(int pos, int num) { Q_D(QTextTable); // qDebug() << "-------- removeCols" << pos << num; if (num <= 0 || pos < 0) return; if (d->dirty) d->update(); if (pos >= d->nCols) return; if (pos + num > d->nCols) pos = d->nCols - num; QTextDocumentPrivate *p = d->pieceTable; QTextFormatCollection *collection = p->formatCollection(); p->beginEditBlock(); // delete whole table? if (pos == 0 && num == d->nCols) { const int pos = p->fragmentMap().position(d->fragment_start); p->remove(pos, p->fragmentMap().position(d->fragment_end) - pos + 1); p->endEditBlock(); return; } for (int r = 0; r < d->nRows; ++r) { for (int c = pos; c < pos + num; ++c) { int cell = d->grid[r*d->nCols + c]; QTextDocumentPrivate::FragmentIterator it(&p->fragmentMap(), cell); QTextCharFormat fmt = collection->charFormat(it->format); int span = fmt.tableCellColumnSpan(); if (span > 1) { fmt.setTableCellColumnSpan(span - 1); p->setCharFormat(it.position(), 1, fmt); } else { // remove cell int index = d->cells.indexOf(cell) + 1; int f_end = index < d->cells.size() ? d->cells.at(index) : d->fragment_end; p->remove(it.position(), p->fragmentMap().position(f_end) - it.position()); } } } QTextTableFormat tfmt = format(); tfmt.setColumns(tfmt.columns()-num); QTextObject::setFormat(tfmt); p->endEditBlock(); // qDebug() << "-------- end removeCols" << pos << num; }
void QTextDocumentFragmentPrivate::insert(QTextCursor &_cursor) const { if (_cursor.isNull()) return; QTextDocumentPrivate *destPieceTable = _cursor.d->priv; destPieceTable->beginEditBlock(); QTextCursor sourceCursor(doc); sourceCursor.movePosition(QTextCursor::End, QTextCursor::KeepAnchor); QTextCopyHelper(sourceCursor, _cursor, importedFromPlainText, _cursor.charFormat()).copy(); destPieceTable->endEditBlock(); }
/*! \since 4.1 Merges the cell at the specified \a row and \a column with the adjacent cells into one cell. The new cell will span \a numRows rows and \a numCols columns. If \a numRows or \a numCols is less than the current number of rows or columns the cell spans then this method does nothing. \sa splitCell() */ void QTextTable::mergeCells(int row, int column, int numRows, int numCols) { Q_D(QTextTable); if (d->dirty) d->update(); QTextDocumentPrivate *p = d->pieceTable; QTextFormatCollection *fc = p->formatCollection(); const QTextTableCell cell = cellAt(row, column); if (!cell.isValid() || row != cell.row() || column != cell.column()) return; QTextCharFormat fmt = cell.format(); const int rowSpan = fmt.tableCellRowSpan(); const int colSpan = fmt.tableCellColumnSpan(); numRows = qMin(numRows, rows() - cell.row()); numCols = qMin(numCols, columns() - cell.column()); // nothing to merge? if (numRows < rowSpan || numCols < colSpan) return; // check the edges of the merge rect to make sure no cell spans the edge for (int r = row; r < row + numRows; ++r) { if (cellAt(r, column) == cellAt(r, column - 1)) return; if (cellAt(r, column + numCols) == cellAt(r, column + numCols - 1)) return; } for (int c = column; c < column + numCols; ++c) { if (cellAt(row, c) == cellAt(row - 1, c)) return; if (cellAt(row + numRows, c) == cellAt(row + numRows - 1, c)) return; } p->beginEditBlock(); const int origCellPosition = cell.firstPosition() - 1; const int cellFragment = d->grid[row * d->nCols + column]; // find the position at which to insert the contents of the merged cells QFragmentFindHelper helper(origCellPosition, p->fragmentMap()); QList<int>::Iterator it = qBinaryFind(d->cells.begin(), d->cells.end(), helper); Q_ASSERT(it != d->cells.end()); Q_ASSERT(*it == cellFragment); const int insertCellIndex = it - d->cells.begin(); int insertFragment = d->cells.value(insertCellIndex + 1, d->fragment_end); uint insertPos = p->fragmentMap().position(insertFragment); d->blockFragmentUpdates = true; bool rowHasText = cell.firstCursorPosition().block().length(); bool needsParagraph = rowHasText && colSpan == numCols; // find all cells that will be erased by the merge for (int r = row; r < row + numRows; ++r) { int firstColumn = r < row + rowSpan ? column + colSpan : column; // don't recompute the cell index for the first row int firstCellIndex = r == row ? insertCellIndex + 1 : -1; int cellIndex = firstCellIndex; for (int c = firstColumn; c < column + numCols; ++c) { const int fragment = d->grid[r * d->nCols + c]; // already handled? if (fragment == cellFragment) continue; QTextDocumentPrivate::FragmentIterator it(&p->fragmentMap(), fragment); uint pos = it.position(); if (firstCellIndex == -1) { QFragmentFindHelper helper(pos, p->fragmentMap()); QList<int>::Iterator it = qBinaryFind(d->cells.begin(), d->cells.end(), helper); Q_ASSERT(it != d->cells.end()); Q_ASSERT(*it == fragment); firstCellIndex = cellIndex = it - d->cells.begin(); } ++cellIndex; QTextCharFormat fmt = fc->charFormat(it->format); const int cellRowSpan = fmt.tableCellRowSpan(); const int cellColSpan = fmt.tableCellColumnSpan(); // update the grid for this cell for (int i = r; i < r + cellRowSpan; ++i) for (int j = c; j < c + cellColSpan; ++j) d->grid[i * d->nCols + j] = cellFragment; // erase the cell marker p->remove(pos, 1); const int nextFragment = d->cells.value(cellIndex, d->fragment_end); const uint nextPos = p->fragmentMap().position(nextFragment); Q_ASSERT(nextPos >= pos); // merge the contents of the cell (if not empty) if (nextPos > pos) { if (needsParagraph) { needsParagraph = false; QTextCursor(p, insertPos++).insertBlock(); p->move(pos + 1, insertPos, nextPos - pos); } else if (rowHasText) { QTextCursor(p, insertPos++).insertText(QLatin1String(" ")); p->move(pos + 1, insertPos, nextPos - pos); } else { p->move(pos, insertPos, nextPos - pos); } insertPos += nextPos - pos; rowHasText = true; } } if (rowHasText) { needsParagraph = true; rowHasText = false; } // erase cells from last row if (firstCellIndex >= 0) { d->cellIndices.remove(firstCellIndex, cellIndex - firstCellIndex); d->cells.erase(d->cells.begin() + firstCellIndex, d->cells.begin() + cellIndex); } } d->fragment_start = d->cells.first(); fmt.setTableCellRowSpan(numRows); fmt.setTableCellColumnSpan(numCols); p->setCharFormat(origCellPosition, 1, fmt); d->blockFragmentUpdates = false; d->dirty = false; p->endEditBlock(); }
/*! \since 4.1 Splits the specified cell at \a row and \a column into an array of multiple cells with dimensions specified by \a numRows and \a numCols. \note It is only possible to split cells that span multiple rows or columns, such as rows that have been merged using mergeCells(). \sa mergeCells() */ void QTextTable::splitCell(int row, int column, int numRows, int numCols) { Q_D(QTextTable); if (d->dirty) d->update(); QTextDocumentPrivate *p = d->pieceTable; QTextFormatCollection *c = p->formatCollection(); const QTextTableCell cell = cellAt(row, column); if (!cell.isValid()) return; row = cell.row(); column = cell.column(); QTextCharFormat fmt = cell.format(); const int rowSpan = fmt.tableCellRowSpan(); const int colSpan = fmt.tableCellColumnSpan(); // nothing to split? if (numRows > rowSpan || numCols > colSpan) return; p->beginEditBlock(); const int origCellPosition = cell.firstPosition() - 1; QVarLengthArray<int> rowPositions(rowSpan); rowPositions[0] = cell.lastPosition(); for (int r = row + 1; r < row + rowSpan; ++r) { // find the cell before which to insert the new cell markers int gridIndex = r * d->nCols + column; QVector<int>::iterator it = qUpperBound(d->cellIndices.begin(), d->cellIndices.end(), gridIndex); int cellIndex = it - d->cellIndices.begin(); int fragment = d->cells.value(cellIndex, d->fragment_end); rowPositions[r - row] = p->fragmentMap().position(fragment); } fmt.setTableCellColumnSpan(1); fmt.setTableCellRowSpan(1); const int fmtIndex = c->indexForFormat(fmt); const int blockIndex = p->blockMap().find(cell.lastPosition())->format; int insertAdjustement = 0; for (int i = 0; i < numRows; ++i) { for (int c = 0; c < colSpan - numCols; ++c) p->insertBlock(QTextBeginningOfFrame, rowPositions[i] + insertAdjustement + c, blockIndex, fmtIndex); insertAdjustement += colSpan - numCols; } for (int i = numRows; i < rowSpan; ++i) { for (int c = 0; c < colSpan; ++c) p->insertBlock(QTextBeginningOfFrame, rowPositions[i] + insertAdjustement + c, blockIndex, fmtIndex); insertAdjustement += colSpan; } fmt.setTableCellRowSpan(numRows); fmt.setTableCellColumnSpan(numCols); p->setCharFormat(origCellPosition, 1, fmt); p->endEditBlock(); }
/*! \fn void QTextTable::removeColumns(int index, int columns) Removes a number of \a columns starting with the column at the specified \a index. \sa insertRows(), insertColumns(), removeRows(), resize(), appendRows(), appendColumns() */ void QTextTable::removeColumns(int pos, int num) { Q_D(QTextTable); // qDebug() << "-------- removeCols" << pos << num; if (num <= 0 || pos < 0) return; if (d->dirty) d->update(); if (pos >= d->nCols) return; if (pos + num > d->nCols) pos = d->nCols - num; QTextDocumentPrivate *p = d->pieceTable; QTextFormatCollection *collection = p->formatCollection(); p->beginEditBlock(); // delete whole table? if (pos == 0 && num == d->nCols) { const int pos = p->fragmentMap().position(d->fragment_start); p->remove(pos, p->fragmentMap().position(d->fragment_end) - pos + 1); p->endEditBlock(); return; } p->aboutToRemoveCell(cellAt(0, pos).firstPosition(), cellAt(d->nRows - 1, pos + num - 1).lastPosition()); QList<int> touchedCells; for (int r = 0; r < d->nRows; ++r) { for (int c = pos; c < pos + num; ++c) { int cell = d->grid[r*d->nCols + c]; QTextDocumentPrivate::FragmentIterator it(&p->fragmentMap(), cell); QTextCharFormat fmt = collection->charFormat(it->format); int span = fmt.tableCellColumnSpan(); if (touchedCells.contains(cell) && span <= 1) continue; touchedCells << cell; if (span > 1) { fmt.setTableCellColumnSpan(span - 1); p->setCharFormat(it.position(), 1, fmt); } else { // remove cell int index = d->cells.indexOf(cell) + 1; int f_end = index < d->cells.size() ? d->cells.at(index) : d->fragment_end; p->remove(it.position(), p->fragmentMap().position(f_end) - it.position()); } } } QTextTableFormat tfmt = format(); tfmt.setColumns(tfmt.columns()-num); QVector<QTextLength> columnWidths = tfmt.columnWidthConstraints(); if (columnWidths.count() > pos) { columnWidths.remove(pos, num); tfmt.setColumnWidthConstraints (columnWidths); } QTextObject::setFormat(tfmt); p->endEditBlock(); // qDebug() << "-------- end removeCols" << pos << num; }
/*! \fn void QTextTable::insertColumns(int index, int columns) Inserts a number of \a columns before the column with the specified \a index. \sa insertRows(), resize(), removeRows(), removeColumns(), appendRows(), appendColumns() */ void QTextTable::insertColumns(int pos, int num) { Q_D(QTextTable); if (num <= 0) return; if (d->dirty) d->update(); if (pos > d->nCols || pos < 0) pos = d->nCols; // qDebug() << "-------- insertCols" << pos << num; QTextDocumentPrivate *p = d->pieceTable; QTextFormatCollection *c = p->formatCollection(); p->beginEditBlock(); QList<int> extendedSpans; for (int i = 0; i < d->nRows; ++i) { int cell; if (i == d->nRows - 1 && pos == d->nCols) { cell = d->fragment_end; } else { int logicalGridIndexBeforePosition = pos > 0 ? d->findCellIndex(d->grid[i*d->nCols + pos - 1]) : -1; // Search for the logical insertion point by skipping past cells which are not the first // cell in a rowspan. This means any cell for which the logical grid index is // less than the logical cell index of the cell before the insertion. int logicalGridIndex; int gridArrayOffset = i*d->nCols + pos; do { cell = d->grid[gridArrayOffset]; logicalGridIndex = d->findCellIndex(cell); gridArrayOffset++; } while (logicalGridIndex < logicalGridIndexBeforePosition && gridArrayOffset < d->nRows*d->nCols); if (logicalGridIndex < logicalGridIndexBeforePosition && gridArrayOffset == d->nRows*d->nCols) cell = d->fragment_end; } if (pos > 0 && pos < d->nCols && cell == d->grid[i*d->nCols + pos - 1]) { // cell spans the insertion place, extend it if (!extendedSpans.contains(cell)) { QTextDocumentPrivate::FragmentIterator it(&p->fragmentMap(), cell); QTextCharFormat fmt = c->charFormat(it->format); fmt.setTableCellColumnSpan(fmt.tableCellColumnSpan() + num); p->setCharFormat(it.position(), 1, fmt); d->dirty = true; extendedSpans << cell; } } else { /* If the next cell is spanned from the row above, we need to find the right position to insert to */ if (i > 0 && pos < d->nCols && cell == d->grid[(i-1) * d->nCols + pos]) { int gridIndex = i*d->nCols + pos; const int gridEnd = d->nRows * d->nCols - 1; while (gridIndex < gridEnd && cell == d->grid[gridIndex]) { ++gridIndex; } if (gridIndex == gridEnd) cell = d->fragment_end; else cell = d->grid[gridIndex]; } QTextDocumentPrivate::FragmentIterator it(&p->fragmentMap(), cell); QTextCharFormat fmt = c->charFormat(it->format); fmt.setTableCellRowSpan(1); fmt.setTableCellColumnSpan(1); Q_ASSERT(fmt.objectIndex() == objectIndex()); int position = it.position(); int cfmt = p->formatCollection()->indexForFormat(fmt); int bfmt = p->formatCollection()->indexForFormat(QTextBlockFormat()); for (int i = 0; i < num; ++i) p->insertBlock(QTextBeginningOfFrame, position, bfmt, cfmt, QTextUndoCommand::MoveCursor); } } QTextTableFormat tfmt = format(); tfmt.setColumns(tfmt.columns()+num); QVector<QTextLength> columnWidths = tfmt.columnWidthConstraints(); if (! columnWidths.isEmpty()) { for (int i = num; i > 0; --i) columnWidths.insert(pos, columnWidths[qMax(0, pos-1)]); } tfmt.setColumnWidthConstraints (columnWidths); QTextObject::setFormat(tfmt); // qDebug() << "-------- end insertCols" << pos << num; p->endEditBlock(); }