/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- * * The contents of this file are subject to the Netscape Public * License Version 1.1 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.mozilla.org/NPL/ * * Software distributed under the License is distributed on an "AS * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or * implied. See the License for the specific language governing * rights and limitations under the License. * * The Original Code is mozilla.org code. * * The Initial Developer of the Original Code is Netscape * Communications Corporation. Portions created by Netscape are * Copyright (C) 1998 Netscape Communications Corporation. All * Rights Reserved. * * Contributor(s): * Pierre Phaneuf */ #include "nsIDOMDocument.h" #include "nsEditor.h" #include "nsIDOMText.h" #include "nsIDOMElement.h" #include "nsIDOMAttr.h" #include "nsIDOMNode.h" #include "nsIDOMNodeList.h" #include "nsIDOMRange.h" #include "nsISelection.h" #include "nsISelectionPrivate.h" #include "nsLayoutCID.h" #include "nsIContent.h" #include "nsIContentIterator.h" #include "nsIAtom.h" #include "nsIDOMHTMLTableElement.h" #include "nsIDOMHTMLTableCellElement.h" #include "nsITableCellLayout.h" // For efficient access to table cell #include "nsITableLayout.h" // data owned by the table and cell frames #include "nsHTMLEditor.h" #include "nsISelectionPrivate.h" // For nsISelectionPrivate::TABLESELECTION_ defines #include "nsVoidArray.h" #include "nsEditorUtils.h" #include "nsTextEditUtils.h" #include "nsHTMLEditUtils.h" static NS_DEFINE_CID(kCContentIteratorCID, NS_CONTENTITERATOR_CID); /*************************************************************************** * stack based helper class for restoring selection after table edit */ class nsSetSelectionAfterTableEdit { private: nsCOMPtr mEd; nsCOMPtr mTable; PRInt32 mCol, mRow, mDirection, mSelected; public: nsSetSelectionAfterTableEdit(nsITableEditor *aEd, nsIDOMElement* aTable, PRInt32 aRow, PRInt32 aCol, PRInt32 aDirection, PRBool aSelected) : mEd(do_QueryInterface(aEd)) { mTable = aTable; mRow = aRow; mCol = aCol; mDirection = aDirection; mSelected = aSelected; } ~nsSetSelectionAfterTableEdit() { if (mEd) mEd->SetSelectionAfterTableEdit(mTable, mRow, mCol, mDirection, mSelected); } // This is needed to abort the caret reset in the destructor // when one method yields control to another void CancelSetCaret() {mEd = nsnull; mTable = nsnull;} }; // Stack-class to turn on/off selection batching for table selection class nsSelectionBatcher { private: nsCOMPtr mSelection; public: nsSelectionBatcher(nsISelection *aSelection) { nsCOMPtr sel(aSelection); mSelection = do_QueryInterface(sel); if (mSelection) mSelection->StartBatchChanges(); } virtual ~nsSelectionBatcher() { if (mSelection) mSelection->EndBatchChanges(); } }; // Table Editing helper utilities (not exposed in IDL) NS_IMETHODIMP nsHTMLEditor::InsertCell(nsIDOMElement *aCell, PRInt32 aRowSpan, PRInt32 aColSpan, PRBool aAfter, PRBool aIsHeader, nsIDOMElement **aNewCell) { if (!aCell) return NS_ERROR_NULL_POINTER; if (aNewCell) *aNewCell = nsnull; // And the parent and offsets needed to do an insert nsCOMPtr cellParent; nsresult res = aCell->GetParentNode(getter_AddRefs(cellParent)); if (NS_FAILED(res)) return res; if (!cellParent) return NS_ERROR_NULL_POINTER; PRInt32 cellOffset; res = GetChildOffset(aCell, cellParent, cellOffset); if (NS_FAILED(res)) return res; nsCOMPtr newCell; if (aIsHeader) res = CreateElementWithDefaults(NS_LITERAL_STRING("th"), getter_AddRefs(newCell)); else res = CreateElementWithDefaults(NS_LITERAL_STRING("td"), getter_AddRefs(newCell)); if(NS_FAILED(res)) return res; if(!newCell) return NS_ERROR_FAILURE; //Optional: return new cell created if (aNewCell) { *aNewCell = newCell.get(); NS_ADDREF(*aNewCell); } if( aRowSpan > 1) { // Note: Do NOT use editor transaction for this nsAutoString newRowSpan; newRowSpan.AppendInt(aRowSpan, 10); newCell->SetAttribute(NS_LITERAL_STRING("rowspan"), newRowSpan); } if( aColSpan > 1) { // Note: Do NOT use editor transaction for this nsAutoString newColSpan; newColSpan.AppendInt(aColSpan, 10); newCell->SetAttribute(NS_LITERAL_STRING("colspan"), newColSpan); } if(aAfter) cellOffset++; //Don't let Rules System change the selection nsAutoTxnsConserveSelection dontChangeSelection(this); return InsertNode(newCell, cellParent, cellOffset); } static PRBool IsRowNode(nsIDOMNode *aNode) { nsCOMPtr atom; nsCOMPtr content = do_QueryInterface(aNode); if (content) { content->GetTag(*getter_AddRefs(atom)); if (atom && atom.get() == nsIEditProperty::tr) return PR_TRUE; } return PR_FALSE; } NS_IMETHODIMP nsHTMLEditor::SetColSpan(nsIDOMElement *aCell, PRInt32 aColSpan) { if (!aCell) return NS_ERROR_NULL_POINTER; nsAutoString newSpan; newSpan.AppendInt(aColSpan, 10); nsAutoString colSpan; colSpan.AssignWithConversion("colspan"); return SetAttribute(aCell, colSpan, newSpan); } NS_IMETHODIMP nsHTMLEditor::SetRowSpan(nsIDOMElement *aCell, PRInt32 aRowSpan) { if (!aCell) return NS_ERROR_NULL_POINTER; nsAutoString newSpan; newSpan.AppendInt(aRowSpan, 10); nsAutoString rowSpan; rowSpan.AssignWithConversion("rowspan"); return SetAttribute(aCell, rowSpan, newSpan); } /****************************************************************/ // Table Editing interface methods NS_IMETHODIMP nsHTMLEditor::InsertTableCell(PRInt32 aNumber, PRBool aAfter) { nsCOMPtr table; nsCOMPtr curCell; nsCOMPtr cellParent; PRInt32 cellOffset, startRowIndex, startColIndex; nsresult res = GetCellContext(nsnull, getter_AddRefs(table), getter_AddRefs(curCell), getter_AddRefs(cellParent), &cellOffset, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; // Don't fail if no cell found if (!curCell) return NS_EDITOR_ELEMENT_NOT_FOUND; // Get more data for current cell in row we are inserting at (we need COLSPAN) PRInt32 curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; res = GetCellDataAt(table, startRowIndex, startColIndex, getter_AddRefs(curCell), curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return res; if (!curCell) return NS_ERROR_FAILURE; PRInt32 newCellIndex = aAfter ? (startColIndex+colSpan) : startColIndex; //We control selection resetting after the insert... nsSetSelectionAfterTableEdit setCaret(this, table, startRowIndex, newCellIndex, ePreviousColumn, PR_FALSE); //...so suppress Rules System selection munging nsAutoTxnsConserveSelection dontChangeSelection(this); PRInt32 i; for (i = 0; i < aNumber; i++) { nsCOMPtr newCell; res = CreateElementWithDefaults(NS_LITERAL_STRING("td"), getter_AddRefs(newCell)); if (NS_SUCCEEDED(res) && newCell) { if (aAfter) cellOffset++; res = InsertNode(newCell, cellParent, cellOffset); if(NS_FAILED(res)) break; } } return res; } NS_IMETHODIMP nsHTMLEditor::GetFirstRow(nsIDOMElement* aTableElement, nsIDOMNode** aRowNode) { if (!aRowNode) return NS_ERROR_NULL_POINTER; *aRowNode = nsnull; if (!aTableElement) return NS_ERROR_NULL_POINTER; nsCOMPtr tableElement; nsresult res = GetElementOrParentByTagName(NS_LITERAL_STRING("table"), aTableElement, getter_AddRefs(tableElement)); if (NS_FAILED(res)) return res; if (!tableElement) return NS_ERROR_NULL_POINTER; nsCOMPtrtableNode = do_QueryInterface(tableElement); if (!tableNode) return NS_ERROR_NULL_POINTER; nsCOMPtr tableChild; res = tableNode->GetFirstChild(getter_AddRefs(tableChild)); if (NS_FAILED(res)) return res; while (tableChild) { nsCOMPtr content = do_QueryInterface(tableChild); if (content) { nsCOMPtr atom; content->GetTag(*getter_AddRefs(atom)); if (atom.get() == nsIEditProperty::tr) { // Found a row directly under *aRowNode = tableChild.get(); NS_ADDREF(*aRowNode); return NS_OK; } // Look for row in one of the row container elements if (atom.get() == nsIEditProperty::tbody || atom.get() == nsIEditProperty::thead || atom.get() == nsIEditProperty::tfoot ) { nsCOMPtr rowNode; res = tableChild->GetFirstChild(getter_AddRefs(rowNode)); if (NS_FAILED(res)) return res; // We can encounter "__moz_text" nodes here -- must find a row while (rowNode && !IsRowNode(rowNode)) { nsCOMPtr nextNode; res = rowNode->GetNextSibling(getter_AddRefs(nextNode)); if (NS_FAILED(res)) return res; rowNode = nextNode; } if(rowNode) { *aRowNode = rowNode.get(); NS_ADDREF(*aRowNode); return NS_OK; } } } // Here if table child was a CAPTION or COLGROUP // or child of a row parent wasn't a row (bad HTML?), // or first child was a "__moz_text" node // Look in next table child nsCOMPtr nextChild; res = tableChild->GetNextSibling(getter_AddRefs(nextChild)); if (NS_FAILED(res)) return res; tableChild = nextChild; }; // If here, row was not found return NS_EDITOR_ELEMENT_NOT_FOUND; } NS_IMETHODIMP nsHTMLEditor::GetNextRow(nsIDOMNode* aCurrentRowNode, nsIDOMNode **aRowNode) { if (!aRowNode) return NS_ERROR_NULL_POINTER; *aRowNode = nsnull; if (!aCurrentRowNode) return NS_ERROR_NULL_POINTER; if (!IsRowNode(aCurrentRowNode)) return NS_ERROR_FAILURE; nsCOMPtr nextRow; nsCOMPtr nextNode; nsresult res = aCurrentRowNode->GetNextSibling(getter_AddRefs(nextRow)); if (NS_FAILED(res)) return res; // Skip over any "__moz_text" nodes here while (nextRow && !IsRowNode(nextRow)) { res = nextRow->GetNextSibling(getter_AddRefs(nextNode)); if (NS_FAILED(res)) return res; nextRow = nextNode; } if(nextRow) { *aRowNode = nextRow.get(); NS_ADDREF(*aRowNode); return NS_OK; } // No row found, search for rows in other table sections nsCOMPtr rowParent; nsCOMPtr parentSibling; res = aCurrentRowNode->GetParentNode(getter_AddRefs(rowParent)); if (NS_FAILED(res)) return res; if (!rowParent) return NS_ERROR_NULL_POINTER; res = rowParent->GetNextSibling(getter_AddRefs(parentSibling)); if (NS_FAILED(res)) return res; while (parentSibling) { res = parentSibling->GetFirstChild(getter_AddRefs(nextRow)); if (NS_FAILED(res)) return res; // We can encounter "__moz_text" nodes here -- must find a row while (nextRow && !IsRowNode(nextRow)) { res = nextRow->GetNextSibling(getter_AddRefs(nextNode)); if (NS_FAILED(res)) return res; nextRow = nextNode; } if(nextRow) { *aRowNode = nextRow.get(); NS_ADDREF(*aRowNode); return NS_OK; } #ifdef DEBUG_cmanske printf("GetNextRow: firstChild of row's parent's sibling is not a TR!\n"); #endif // We arrive here only if a table section has no children // or first child of section is not a row (bad HTML!) res = parentSibling->GetNextSibling(getter_AddRefs(parentSibling)); if (NS_FAILED(res)) return res; } // If here, row was not found return NS_EDITOR_ELEMENT_NOT_FOUND; } NS_IMETHODIMP nsHTMLEditor::GetFirstCellInRow(nsIDOMNode* aRowNode, nsIDOMNode** aCellNode) { if (!aCellNode) return NS_ERROR_NULL_POINTER; *aCellNode = nsnull; if (!aRowNode) return NS_ERROR_NULL_POINTER; nsCOMPtr rowChild; nsresult res = aRowNode->GetFirstChild(getter_AddRefs(rowChild)); if (NS_FAILED(res)) return res; while (rowChild && !nsHTMLEditUtils::IsTableCell(rowChild)) { // Skip over "__moz_text" nodes nsCOMPtr nextChild; res = rowChild->GetNextSibling(getter_AddRefs(nextChild)); if (NS_FAILED(res)) return res; rowChild = nextChild; }; if (rowChild) { *aCellNode = rowChild.get(); NS_ADDREF(*aCellNode); return NS_OK; } // If here, cell was not found return NS_EDITOR_ELEMENT_NOT_FOUND; } NS_IMETHODIMP nsHTMLEditor::GetNextCellInRow(nsIDOMNode* aCurrentCellNode, nsIDOMNode** aCellNode) { if (!aCellNode) return NS_ERROR_NULL_POINTER; *aCellNode = nsnull; if (!aCurrentCellNode) return NS_ERROR_NULL_POINTER; nsCOMPtr nextCell; nsresult res = aCurrentCellNode->GetNextSibling(getter_AddRefs(nextCell)); if (NS_FAILED(res)) return res; while (nextCell && !nsHTMLEditUtils::IsTableCell(nextCell)) { // Skip over "__moz_text" nodes nsCOMPtr nextChild; res = nextCell->GetNextSibling(getter_AddRefs(nextChild)); if (NS_FAILED(res)) return res; nextCell = nextChild; }; if (nextCell) { *aCellNode = nextCell.get(); NS_ADDREF(*aCellNode); return NS_OK; } // If here, cell was not found return NS_EDITOR_ELEMENT_NOT_FOUND; } NS_IMETHODIMP nsHTMLEditor::GetLastCellInRow(nsIDOMNode* aRowNode, nsIDOMNode** aCellNode) { if (!aCellNode) return NS_ERROR_NULL_POINTER; *aCellNode = nsnull; if (!aRowNode) return NS_ERROR_NULL_POINTER; nsCOMPtr rowChild; nsresult res = aRowNode->GetLastChild(getter_AddRefs(rowChild)); if (NS_FAILED(res)) return res; while (rowChild && !nsHTMLEditUtils::IsTableCell(rowChild)) { // Skip over "__moz_text" nodes nsCOMPtr previousChild; res = rowChild->GetPreviousSibling(getter_AddRefs(previousChild)); if (NS_FAILED(res)) return res; rowChild = previousChild; }; if (rowChild) { *aCellNode = rowChild.get(); NS_ADDREF(*aCellNode); return NS_OK; } // If here, cell was not found return NS_EDITOR_ELEMENT_NOT_FOUND; } NS_IMETHODIMP nsHTMLEditor::InsertTableColumn(PRInt32 aNumber, PRBool aAfter) { nsCOMPtr selection; nsCOMPtr table; nsCOMPtr curCell; PRInt32 startRowIndex, startColIndex; nsresult res = GetCellContext(getter_AddRefs(selection), getter_AddRefs(table), getter_AddRefs(curCell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; // Don't fail if no cell found if (!curCell) return NS_EDITOR_ELEMENT_NOT_FOUND; // Get more data for current cell (we need ROWSPAN) PRInt32 curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; res = GetCellDataAt(table, startRowIndex, startColIndex, getter_AddRefs(curCell), curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return res; if (!curCell) return NS_ERROR_FAILURE; nsAutoEditBatch beginBatching(this); // Prevent auto insertion of BR in new cell until we're done nsAutoRules beginRulesSniffing(this, kOpInsertNode, nsIEditor::eNext); // Use column after current cell if requested if (aAfter) { startColIndex += actualColSpan; //Detect when user is adding after a COLSPAN=0 case // Assume they want to stop the "0" behavior and // really add a new column. Thus we set the // colspan to its true value if (colSpan == 0) SetColSpan(curCell, actualColSpan); } PRInt32 rowCount, colCount, rowIndex; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; //We reset caret in destructor... nsSetSelectionAfterTableEdit setCaret(this, table, startRowIndex, startColIndex, ePreviousRow, PR_FALSE); //.. so suppress Rules System selection munging nsAutoTxnsConserveSelection dontChangeSelection(this); // If we are inserting after all existing columns // Make sure table is "well formed" // before appending new column if (startColIndex >= colCount) NormalizeTable(table); nsCOMPtr rowNode; for ( rowIndex = 0; rowIndex < rowCount; rowIndex++) { #ifdef DEBUG_cmanske if (rowIndex == rowCount-1) printf(" ***InsertTableColumn: Inserting cell at last row: %d\n", rowIndex); #endif if (startColIndex < colCount) { // We are inserting before an existing column res = GetCellDataAt(table, rowIndex, startColIndex, getter_AddRefs(curCell), curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return res; // Don't fail entire process if we fail to find a cell // (may fail just in particular rows with < adequate cells per row) if (curCell) { if (curStartColIndex < startColIndex) { // We have a cell spanning this location // Simply increase its colspan to keep table rectangular // Note: we do nothing if colsSpan=0, // since it should automatically span the new column if (colSpan > 0) SetColSpan(curCell, colSpan+aNumber); } else { // Simply set selection to the current cell // so we can let InsertTableCell() do the work // Insert a new cell before current one selection->Collapse(curCell, 0); res = InsertTableCell(aNumber, PR_FALSE); } } } else { // Get current row and append new cells after last cell in row if(rowIndex == 0) res = GetFirstRow(table.get(), getter_AddRefs(rowNode)); else { nsCOMPtr nextRow; res = GetNextRow(rowNode.get(), getter_AddRefs(nextRow)); rowNode = nextRow; } if (NS_FAILED(res)) return res; if (rowNode) { nsCOMPtr lastCell; if (!rowNode) return NS_ERROR_FAILURE; res = GetLastCellInRow(rowNode, getter_AddRefs(lastCell)); if (NS_FAILED(res)) return res; if (!lastCell) return NS_ERROR_FAILURE; curCell = do_QueryInterface(lastCell); if (curCell) { // Simply add same number of cells to each row // Although tempted to check cell indexes for curCell, // the effects of COLSPAN>1 in some cells makes this futile! // We must use NormalizeTable first to assure // that there are cells in each cellmap location selection->Collapse(curCell, 0); res = InsertTableCell(aNumber, PR_TRUE); } } } } return res; } NS_IMETHODIMP nsHTMLEditor::InsertTableRow(PRInt32 aNumber, PRBool aAfter) { nsCOMPtr selection; nsCOMPtr table; nsCOMPtr curCell; nsCOMPtr cellForRowParent; PRInt32 startRowIndex, startColIndex; nsresult res = GetCellContext(nsnull, getter_AddRefs(table), getter_AddRefs(curCell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; // Don't fail if no cell found if (!curCell) return NS_EDITOR_ELEMENT_NOT_FOUND; // Get more data for current cell in row we are inserting at (we need COLSPAN) PRInt32 curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; res = GetCellDataAt(table, startRowIndex, startColIndex, getter_AddRefs(curCell), curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return res; if (!curCell) return NS_ERROR_FAILURE; PRInt32 rowCount, colCount; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; nsAutoEditBatch beginBatching(this); // Prevent auto insertion of BR in new cell until we're done nsAutoRules beginRulesSniffing(this, kOpInsertNode, nsIEditor::eNext); if (aAfter) { // Use row after current cell startRowIndex += actualRowSpan; //Detect when user is adding after a ROWSPAN=0 case // Assume they want to stop the "0" behavior and // really add a new row. Thus we set the // rowspan to its true value if (rowSpan == 0) SetRowSpan(curCell, actualRowSpan); } //We control selection resetting after the insert... nsSetSelectionAfterTableEdit setCaret(this, table, startRowIndex, startColIndex, ePreviousColumn, PR_FALSE); //...so suppress Rules System selection munging nsAutoTxnsConserveSelection dontChangeSelection(this); PRInt32 cellsInRow = 0; if (startRowIndex < rowCount) { // We are inserting above an existing row // Get each cell in the insert row to adjust for COLSPAN effects while we // count how many cells are needed PRInt32 colIndex = 0; // This returns NS_TABLELAYOUT_CELL_NOT_FOUND when we run past end of row, // which passes the NS_SUCCEEDED macro while ( NS_OK == GetCellDataAt(table, startRowIndex, colIndex, getter_AddRefs(curCell), curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected) ) { if (curCell) { if (curStartRowIndex < startRowIndex) { // We have a cell spanning this location // Simply increase its rowspan //Note that if rowSpan == 0, we do nothing, // since that cell should automatically extend into the new row if (rowSpan > 0) SetRowSpan(curCell, rowSpan+aNumber); } else { // We have a cell in the insert row // Count the number of cells we need to add to the new row cellsInRow += actualColSpan; // Save cell we will use below if (!cellForRowParent) cellForRowParent = curCell; } // Next cell in row colIndex += actualColSpan; } else colIndex++; } } else { // We are adding a new row after all others // If it weren't for colspan=0 effect, // we could simply use colCount for number of new cells... cellsInRow = colCount; // ...but we must compensate for all cells with rowSpan = 0 in the last row PRInt32 lastRow = rowCount-1; PRInt32 tempColIndex = 0; while ( NS_OK == GetCellDataAt(table, lastRow, tempColIndex, getter_AddRefs(curCell), curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected) ) { if (rowSpan == 0) cellsInRow -= actualColSpan; tempColIndex += actualColSpan; // Save cell from the last row that we will use below if (!cellForRowParent && curStartRowIndex == lastRow) cellForRowParent = curCell; } } if (cellsInRow > 0) { // The row parent and offset where we will insert new row nsCOMPtr parentOfRow; PRInt32 newRowOffset; if (cellForRowParent) { nsCOMPtr parentRow; res = GetElementOrParentByTagName(NS_LITERAL_STRING("tr"), cellForRowParent, getter_AddRefs(parentRow)); if (NS_FAILED(res)) return res; if (!parentRow) return NS_ERROR_NULL_POINTER; parentRow->GetParentNode(getter_AddRefs(parentOfRow)); if (!parentOfRow) return NS_ERROR_NULL_POINTER; res = GetChildOffset(parentRow, parentOfRow, newRowOffset); if (NS_FAILED(res)) return res; // Adjust for when adding past the end if (aAfter && startRowIndex >= rowCount) newRowOffset++; } else return NS_ERROR_FAILURE; for (PRInt32 row = 0; row < aNumber; row++) { // Create a new row nsCOMPtr newRow; res = CreateElementWithDefaults(NS_LITERAL_STRING("tr"), getter_AddRefs(newRow)); if (NS_SUCCEEDED(res)) { if (!newRow) return NS_ERROR_FAILURE; for (PRInt32 i = 0; i < cellsInRow; i++) { nsCOMPtr newCell; res = CreateElementWithDefaults(NS_LITERAL_STRING("td"), getter_AddRefs(newCell)); if (NS_FAILED(res)) return res; if (!newCell) return NS_ERROR_FAILURE; // Don't use transaction system yet! (not until entire row is inserted) nsCOMPtrresultNode; res = newRow->AppendChild(newCell, getter_AddRefs(resultNode)); if (NS_FAILED(res)) return res; } // Use transaction system to insert the entire row+cells // (Note that rows are inserted at same childoffset each time) res = InsertNode(newRow, parentOfRow, newRowOffset); if (NS_FAILED(res)) return res; } } } return res; } // Editor helper only NS_IMETHODIMP nsHTMLEditor::DeleteTable2(nsIDOMElement *aTable, nsISelection *aSelection) { if (!aTable || !aSelection) return NS_ERROR_NULL_POINTER; nsCOMPtr tableParent; PRInt32 tableOffset; if(!aTable || NS_FAILED(aTable->GetParentNode(getter_AddRefs(tableParent))) || !tableParent) return NS_ERROR_FAILURE; // Save offset we need to restore the selection if(NS_FAILED(GetChildOffset(aTable, tableParent, tableOffset))) return NS_ERROR_FAILURE; nsresult res = DeleteNode(aTable); if (NS_FAILED(res)) return res; // Place selection just before the table aSelection->Collapse(tableParent, tableOffset); return NS_OK; } NS_IMETHODIMP nsHTMLEditor::DeleteTable() { nsCOMPtr selection; nsCOMPtr table; nsresult res = GetCellContext(getter_AddRefs(selection), getter_AddRefs(table), nsnull, nsnull, nsnull, nsnull, nsnull); if (NS_FAILED(res)) return res; nsAutoEditBatch beginBatching(this); res = DeleteTable2(table, selection); if (NS_FAILED(res)) return res; return NS_OK; } NS_IMETHODIMP nsHTMLEditor::DeleteTableCell(PRInt32 aNumber) { nsCOMPtr selection; nsCOMPtr table; nsCOMPtr cell; PRInt32 startRowIndex, startColIndex; nsresult res = GetCellContext(getter_AddRefs(selection), getter_AddRefs(table), getter_AddRefs(cell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; // Don't fail if we didn't find a table or cell if (!table || !cell) return NS_EDITOR_ELEMENT_NOT_FOUND; nsAutoEditBatch beginBatching(this); // Prevent rules testing until we're done nsAutoRules beginRulesSniffing(this, kOpDeleteNode, nsIEditor::eNext); nsCOMPtr firstCell; nsCOMPtr range; res = GetFirstSelectedCell(getter_AddRefs(firstCell), getter_AddRefs(range)); if (NS_FAILED(res)) return res; PRInt32 rangeCount; res = selection->GetRangeCount(&rangeCount); if (NS_FAILED(res)) return res; if (firstCell && rangeCount > 1) { // When > 1 selected cell, // ignore aNumber and use selected cells cell = firstCell; PRInt32 rowCount, colCount; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; // Get indexes -- may be different than original cell res = GetCellIndexes(cell, startRowIndex, startColIndex); if (NS_FAILED(res)) return res; // The setCaret object will call SetSelectionAfterTableEdit in it's destructor nsSetSelectionAfterTableEdit setCaret(this, table, startRowIndex, startColIndex, ePreviousColumn, PR_FALSE); nsAutoTxnsConserveSelection dontChangeSelection(this); PRInt32 currentRow = -1; PRInt32 currentCol = -1; while (cell) { PRBool deleteRow = PR_FALSE; PRBool deleteCol = PR_FALSE; if (startRowIndex != currentRow) { // Optimize to delete an entire row // Remember index so we don't repeat AllCellsInRowSelected within the same row currentRow = startRowIndex; deleteRow = AllCellsInRowSelected(table, startRowIndex, colCount); if (deleteRow) { // First, find the next cell in a different row // to continue after we delete this row PRInt32 nextRow = startRowIndex; while (nextRow == startRowIndex) { res = GetNextSelectedCell(getter_AddRefs(cell), nsnull); if (NS_FAILED(res)) return res; if (!cell) break; res = GetCellIndexes(cell, nextRow, startColIndex); if (NS_FAILED(res)) return res; } // Delete entire row res = DeleteRow(table, startRowIndex); if (NS_FAILED(res)) return res; // For the next cell if (cell) startRowIndex = nextRow; } } if (!deleteRow) { if (startColIndex != currentCol) { // Optimize to delete an entire column // Remember index so we don't repeat AllCellsInColSelected within the same Col currentCol = startColIndex; deleteCol = AllCellsInColumnSelected(table, startColIndex, colCount); if (deleteCol) { // First, find the next cell in a different column // to continue after we delete this column PRInt32 nextCol = startColIndex; while (nextCol == startColIndex) { res = GetNextSelectedCell(getter_AddRefs(cell), nsnull); if (NS_FAILED(res)) return res; if (!cell) break; res = GetCellIndexes(cell, startRowIndex, nextCol); if (NS_FAILED(res)) return res; } // Delete entire Col res = DeleteColumn(table, startColIndex); if (NS_FAILED(res)) return res; // For the next cell if (cell) startColIndex = nextCol; } } if (!deleteCol) { // First get the next cell to delete nsCOMPtr nextCell; res = GetNextSelectedCell(getter_AddRefs(nextCell), getter_AddRefs(range)); if (NS_FAILED(res)) return res; // Then delete the cell res = DeleteNode(cell); if (NS_FAILED(res)) return res; // The next cell to delete cell = nextCell; if (cell) { res = GetCellIndexes(cell, startRowIndex, startColIndex); if (NS_FAILED(res)) return res; } } } } } else for (PRInt32 i = 0; i < aNumber; i++) { res = GetCellContext(getter_AddRefs(selection), getter_AddRefs(table), getter_AddRefs(cell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; // Don't fail if no cell found if (!cell) return NS_EDITOR_ELEMENT_NOT_FOUND; if (1 == GetNumberOfCellsInRow(table, startRowIndex)) { nsCOMPtr parentRow; res = GetElementOrParentByTagName(NS_LITERAL_STRING("tr"), cell, getter_AddRefs(parentRow)); if (NS_FAILED(res)) return res; if (!parentRow) return NS_ERROR_NULL_POINTER; // We should delete the row instead, // but first check if its the only row left // so we can delete the entire table PRInt32 rowCount, colCount; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; if (rowCount == 1) return DeleteTable2(table, selection); // We need to call DeleteTableRow to handle cells with rowspan res = DeleteTableRow(1); if (NS_FAILED(res)) return res; } else { // More than 1 cell in the row // The setCaret object will call SetSelectionAfterTableEdit in it's destructor nsSetSelectionAfterTableEdit setCaret(this, table, startRowIndex, startColIndex, ePreviousColumn, PR_FALSE); nsAutoTxnsConserveSelection dontChangeSelection(this); res = DeleteNode(cell); // If we fail, don't try to delete any more cells??? if (NS_FAILED(res)) return res; } } return NS_OK; } NS_IMETHODIMP nsHTMLEditor::DeleteTableCellContents() { nsCOMPtr selection; nsCOMPtr table; nsCOMPtr cell; PRInt32 startRowIndex, startColIndex; nsresult res = NS_OK; res = GetCellContext(getter_AddRefs(selection), getter_AddRefs(table), getter_AddRefs(cell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; // Don't fail if no cell found if (!cell) return NS_EDITOR_ELEMENT_NOT_FOUND; nsAutoEditBatch beginBatching(this); // Prevent rules testing until we're done nsAutoRules beginRulesSniffing(this, kOpDeleteNode, nsIEditor::eNext); //Don't let Rules System change the selection nsAutoTxnsConserveSelection dontChangeSelection(this); nsCOMPtr firstCell; nsCOMPtr range; res = GetFirstSelectedCell(getter_AddRefs(firstCell), getter_AddRefs(range)); if (NS_FAILED(res)) return res; if (firstCell) { cell = firstCell; res = GetCellIndexes(cell, startRowIndex, startColIndex); if (NS_FAILED(res)) return res; } nsSetSelectionAfterTableEdit setCaret(this, table, startRowIndex, startColIndex, ePreviousColumn, PR_FALSE); while (cell) { DeleteCellContents(cell); if (firstCell) { // We doing a selected cells, so do all of them res = GetNextSelectedCell(getter_AddRefs(cell), nsnull); if (NS_FAILED(res)) return res; } else cell = nsnull; } return NS_OK; } NS_IMETHODIMP nsHTMLEditor::DeleteCellContents(nsIDOMElement *aCell) { if (!aCell) return NS_ERROR_NULL_POINTER; // Prevent rules testing until we're done nsAutoRules beginRulesSniffing(this, kOpDeleteNode, nsIEditor::eNext); nsCOMPtr child; PRBool hasChild; aCell->HasChildNodes(&hasChild); while (hasChild) { aCell->GetLastChild(getter_AddRefs(child)); nsresult res = DeleteNode(child); if (NS_FAILED(res)) return res; aCell->HasChildNodes(&hasChild); } return NS_OK; } NS_IMETHODIMP nsHTMLEditor::DeleteTableColumn(PRInt32 aNumber) { nsCOMPtr selection; nsCOMPtr table; nsCOMPtr cell; PRInt32 startRowIndex, startColIndex, rowCount, colCount; nsresult res = GetCellContext(getter_AddRefs(selection), getter_AddRefs(table), getter_AddRefs(cell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; // Don't fail if no cell found if (!table || !cell) return NS_EDITOR_ELEMENT_NOT_FOUND; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; // Shortcut the case of deleting all columns in table if(startColIndex == 0 && aNumber >= colCount) return DeleteTable2(table, selection); // Check for counts too high aNumber = PR_MIN(aNumber,(colCount-startColIndex)); nsAutoEditBatch beginBatching(this); // Prevent rules testing until we're done nsAutoRules beginRulesSniffing(this, kOpDeleteNode, nsIEditor::eNext); // Test if deletion is controlled by selected cells nsCOMPtr firstCell; nsCOMPtr range; res = GetFirstSelectedCell(getter_AddRefs(firstCell), getter_AddRefs(range)); if (NS_FAILED(res)) return res; PRInt32 rangeCount; res = selection->GetRangeCount(&rangeCount); if (NS_FAILED(res)) return res; if (firstCell && rangeCount > 1) { // Fetch indexes again - may be different for selected cells res = GetCellIndexes(firstCell, startRowIndex, startColIndex); if (NS_FAILED(res)) return res; } //We control selection resetting after the insert... nsSetSelectionAfterTableEdit setCaret(this, table, startRowIndex, startColIndex, ePreviousRow, PR_FALSE); if (firstCell && rangeCount > 1) { // Use selected cells to determine what rows to delete cell = firstCell; while (cell) { if (cell != firstCell) { res = GetCellIndexes(cell, startRowIndex, startColIndex); if (NS_FAILED(res)) return res; } // Find the next cell in a different column // to continue after we delete this column PRInt32 nextCol = startColIndex; while (nextCol == startColIndex) { res = GetNextSelectedCell(getter_AddRefs(cell), getter_AddRefs(range)); if (NS_FAILED(res)) return res; if (!cell) break; res = GetCellIndexes(cell, startRowIndex, nextCol); if (NS_FAILED(res)) return res; } res = DeleteColumn(table, startColIndex); if (NS_FAILED(res)) return res; } } else for (PRInt32 i = 0; i < aNumber; i++) { res = DeleteColumn(table, startColIndex); if (NS_FAILED(res)) return res; } return NS_OK; } NS_IMETHODIMP nsHTMLEditor::DeleteColumn(nsIDOMElement *aTable, PRInt32 aColIndex) { if (!aTable) return NS_ERROR_NULL_POINTER; nsCOMPtr cell; nsCOMPtr cellInDeleteCol; PRInt32 startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; PRInt32 rowIndex = 0; nsresult res = NS_OK; do { res = GetCellDataAt(aTable, rowIndex, aColIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return res; if (cell) { // Find cells that don't start in column we are deleting if (startColIndex < aColIndex || colSpan > 1 || colSpan == 0) { // We have a cell spanning this location // Decrease its colspan to keep table rectangular, // but if colSpan=0, it will adjust automatically if (colSpan > 0) { NS_ASSERTION((colSpan > 1),"Bad COLSPAN in DeleteTableColumn"); SetColSpan(cell, colSpan-1); } if (startColIndex == aColIndex) { // Cell is in column to be deleted, but must have colspan > 1, // so delete contents of cell instead of cell itself // (We must have reset colspan above) DeleteCellContents(cell); } // To next cell in column rowIndex += actualRowSpan; } else { // Delete the cell if (1 == GetNumberOfCellsInRow(aTable, rowIndex)) { // Only 1 cell in row - delete the row nsCOMPtr parentRow; res = GetElementOrParentByTagName(NS_LITERAL_STRING("tr"), cell, getter_AddRefs(parentRow)); if (NS_FAILED(res)) return res; if(!parentRow) return NS_ERROR_NULL_POINTER; // But first check if its the only row left // so we can delete the entire table // (This should never happen but it's the safe thing to do) PRInt32 rowCount, colCount; res = GetTableSize(aTable, rowCount, colCount); if (NS_FAILED(res)) return res; if (rowCount == 1) { nsCOMPtr selection; res = GetSelection(getter_AddRefs(selection)); if (NS_FAILED(res)) return res; if (!selection) return NS_ERROR_FAILURE; return DeleteTable2(aTable, selection); } // Delete the row by placing caret in cell we were to delete // We need to call DeleteTableRow to handle cells with rowspan res = DeleteRow(aTable, startRowIndex); if (NS_FAILED(res)) return res; // Note that we don't incremenet rowIndex // since a row was deleted and "next" // row now has current rowIndex } else { // A more "normal" deletion res = DeleteNode(cell); if (NS_FAILED(res)) return res; //Skip over any rows spanned by this cell rowIndex += actualRowSpan; } } } } while (cell); return NS_OK; } NS_IMETHODIMP nsHTMLEditor::DeleteTableRow(PRInt32 aNumber) { nsCOMPtr selection; nsCOMPtr table; nsCOMPtr cell; PRInt32 startRowIndex, startColIndex; PRInt32 rowCount, colCount; nsresult res = GetCellContext(getter_AddRefs(selection), getter_AddRefs(table), getter_AddRefs(cell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; // Don't fail if no cell found if (!cell) return NS_EDITOR_ELEMENT_NOT_FOUND; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; // Shortcut the case of deleting all rows in table if(startRowIndex == 0 && aNumber >= rowCount) return DeleteTable2(table, selection); nsAutoEditBatch beginBatching(this); // Prevent rules testing until we're done nsAutoRules beginRulesSniffing(this, kOpDeleteNode, nsIEditor::eNext); nsCOMPtr firstCell; nsCOMPtr range; res = GetFirstSelectedCell(getter_AddRefs(firstCell), getter_AddRefs(range)); if (NS_FAILED(res)) return res; PRInt32 rangeCount; res = selection->GetRangeCount(&rangeCount); if (NS_FAILED(res)) return res; if (firstCell && rangeCount > 1) { // Fetch indexes again - may be different for selected cells res = GetCellIndexes(firstCell, startRowIndex, startColIndex); if (NS_FAILED(res)) return res; } //We control selection resetting after the insert... nsSetSelectionAfterTableEdit setCaret(this, table, startRowIndex, startColIndex, ePreviousRow, PR_FALSE); // Don't change selection during deletions nsAutoTxnsConserveSelection dontChangeSelection(this); if (firstCell && rangeCount > 1) { // Use selected cells to determine what rows to delete cell = firstCell; while (cell) { if (cell != firstCell) { res = GetCellIndexes(cell, startRowIndex, startColIndex); if (NS_FAILED(res)) return res; } // Find the next cell in a different row // to continue after we delete this row PRInt32 nextRow = startRowIndex; while (nextRow == startRowIndex) { res = GetNextSelectedCell(getter_AddRefs(cell), getter_AddRefs(range)); if (NS_FAILED(res)) return res; if (!cell) break; res = GetCellIndexes(cell, nextRow, startColIndex); if (NS_FAILED(res)) return res; } // Delete entire row res = DeleteRow(table, startRowIndex); if (NS_FAILED(res)) return res; } } else { // Check for counts too high aNumber = PR_MIN(aNumber,(rowCount-startRowIndex)); for (PRInt32 i = 0; i < aNumber; i++) { res = DeleteRow(table, startRowIndex); // If failed in current row, try the next if (NS_FAILED(res)) startRowIndex++; // Check if there's a cell in the "next" row res = GetCellAt(table, startRowIndex, startColIndex, getter_AddRefs(cell)); if (NS_FAILED(res)) return res; if(!cell) break; } } return NS_OK; } // Helper that doesn't batch or change the selection NS_IMETHODIMP nsHTMLEditor::DeleteRow(nsIDOMElement *aTable, PRInt32 aRowIndex) { if (!aTable) return NS_ERROR_NULL_POINTER; nsCOMPtr cell; nsCOMPtr cellInDeleteRow; PRInt32 startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; PRInt32 colIndex = 0; nsresult res = NS_OK; // Prevent rules testing until we're done nsAutoRules beginRulesSniffing(this, kOpDeleteNode, nsIEditor::eNext); // The list of cells we will change rowspan in // and the new rowspan values for each nsVoidArray spanCellList; nsVoidArray newSpanList; // Scan through cells in row to do rowspan adjustments // Note that after we delete row, startRowIndex will point to the // cells in the next row to be deleted do { res = GetCellDataAt(aTable, aRowIndex, colIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); // We don't fail if we don't find a cell, so this must be real bad if(NS_FAILED(res)) return res; // Compensate for cells that don't start or extend below the row we are deleting if (cell) { if (startRowIndex < aRowIndex) { // Cell starts in row above us // Decrease its rowspan to keep table rectangular // but we don't need to do this if rowspan=0, // since it will automatically adjust if (rowSpan > 0) { // Build list of cells to change rowspan // We can't do it now since it upsets cell map, // so we will do it after deleting the row spanCellList.AppendElement((void*)cell.get()); newSpanList.AppendElement((void*)PR_MAX((aRowIndex - startRowIndex), actualRowSpan-1)); } } else { if (rowSpan > 1) { //Cell spans below row to delete, // so we must insert new cells to keep rows below even // Note that we test "rowSpan" so we don't do this if rowSpan = 0 (automatic readjustment) res = SplitCellIntoRows(aTable, startRowIndex, startColIndex, aRowIndex - startRowIndex + 1, // The row above the row to insert new cell into actualRowSpan - 1, nsnull); // Span remaining below if (NS_FAILED(res)) return res; } if (!cellInDeleteRow) cellInDeleteRow = cell; // Reference cell to find row to delete } // Skip over other columns spanned by this cell colIndex += actualColSpan; } } while (cell); // Things are messed up if we didn't find a cell in the row! if (!cellInDeleteRow) return NS_ERROR_FAILURE; // Delete the entire row nsCOMPtr parentRow; res = GetElementOrParentByTagName(NS_LITERAL_STRING("tr"), cellInDeleteRow, getter_AddRefs(parentRow)); if (NS_FAILED(res)) return res; if (parentRow) { res = DeleteNode(parentRow); if (NS_FAILED(res)) return res; } // Now we can set new rowspans for cells stored above nsIDOMElement *cellPtr; PRInt32 newSpan; PRInt32 count; while ((count = spanCellList.Count())) { // go backwards to keep nsVoidArray from mem-moving everything each time count--; // nsVoidArray is zero based cellPtr = (nsIDOMElement*)spanCellList.ElementAt(count); spanCellList.RemoveElementAt(count); newSpan = (PRInt32)newSpanList.ElementAt(count); newSpanList.RemoveElementAt(count); if (cellPtr) { res = SetRowSpan(cellPtr, newSpan); if (NS_FAILED(res)) return res; } } return NS_OK; } NS_IMETHODIMP nsHTMLEditor::SelectTable() { nsCOMPtr table; nsresult res = NS_ERROR_FAILURE; res = GetElementOrParentByTagName(NS_LITERAL_STRING("table"), nsnull, getter_AddRefs(table)); if (NS_FAILED(res)) return res; // Don't fail if we didn't find a table if (!table) return NS_OK; nsCOMPtr tableNode = do_QueryInterface(table); if (tableNode) { res = ClearSelection(); if (NS_SUCCEEDED(res)) res = AppendNodeToSelectionAsRange(table); } return res; } NS_IMETHODIMP nsHTMLEditor::SelectTableCell() { nsCOMPtr cell; nsresult res = GetElementOrParentByTagName(NS_LITERAL_STRING("td"), nsnull, getter_AddRefs(cell)); if (NS_FAILED(res)) return res; // Don't fail if we didn't find a table if (!cell) return NS_EDITOR_ELEMENT_NOT_FOUND; nsCOMPtr cellNode = do_QueryInterface(cell); if (cellNode) { res = ClearSelection(); if (NS_SUCCEEDED(res)) res = AppendNodeToSelectionAsRange(cellNode); } return res; } NS_IMETHODIMP nsHTMLEditor::SelectBlockOfCells(nsIDOMElement *aStartCell, nsIDOMElement *aEndCell) { if (!aStartCell || !aEndCell) return NS_ERROR_NULL_POINTER; nsCOMPtr selection; nsresult res = GetSelection(getter_AddRefs(selection)); if (NS_FAILED(res)) return res; if (!selection) return NS_ERROR_FAILURE; nsCOMPtr table; res = GetElementOrParentByTagName(NS_LITERAL_STRING("table"), aStartCell, getter_AddRefs(table)); if (NS_FAILED(res)) return res; if (!table) return NS_ERROR_FAILURE; nsCOMPtr endTable; res = GetElementOrParentByTagName(NS_LITERAL_STRING("table"), aEndCell, getter_AddRefs(endTable)); if (NS_FAILED(res)) return res; if (!endTable) return NS_ERROR_FAILURE; // We can only select a block if within the same table, // so do nothing if not within one table if (table != endTable) return NS_OK; PRInt32 startRowIndex, startColIndex, endRowIndex, endColIndex; // Get starting and ending cells' location in the cellmap res = GetCellIndexes(aStartCell, startRowIndex, startColIndex); if(NS_FAILED(res)) return res; res = GetCellIndexes(aEndCell, endRowIndex, endColIndex); if(NS_FAILED(res)) return res; // Suppress nsISelectionListener notification // until all selection changes are finished nsSelectionBatcher selectionBatcher(selection); // Examine all cell nodes in current selection and // remove those outside the new block cell region PRInt32 minColumn = PR_MIN(startColIndex, endColIndex); PRInt32 minRow = PR_MIN(startRowIndex, endRowIndex); PRInt32 maxColumn = PR_MAX(startColIndex, endColIndex); PRInt32 maxRow = PR_MAX(startRowIndex, endRowIndex); nsCOMPtr cell; PRInt32 currentRowIndex, currentColIndex; nsCOMPtr range; res = GetFirstSelectedCell(getter_AddRefs(cell), getter_AddRefs(range)); if (NS_FAILED(res)) return res; if (res == NS_EDITOR_ELEMENT_NOT_FOUND) return NS_OK; while (cell) { res = GetCellIndexes(cell, currentRowIndex, currentColIndex); if (NS_FAILED(res)) return res; if (currentRowIndex < maxRow || currentRowIndex > maxRow || currentColIndex < maxColumn || currentColIndex > maxColumn) { selection->RemoveRange(range); // Since we've removed the range, decrement pointer to next range mSelectedCellIndex--; } res = GetNextSelectedCell(getter_AddRefs(cell), getter_AddRefs(range)); if (NS_FAILED(res)) return res; } PRInt32 rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; for (PRInt32 row = minRow; row <= maxRow; row++) { for(PRInt32 col = minColumn; col <= maxColumn; col += PR_MAX(actualColSpan, 1)) { res = GetCellDataAt(table, row, col, getter_AddRefs(cell), currentRowIndex, currentColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) break; // Skip cells that already selected or are spanned from previous locations if (!isSelected && cell && row == currentRowIndex && col == currentColIndex) { nsCOMPtr cellNode = do_QueryInterface(cell); res = AppendNodeToSelectionAsRange(cellNode); if (NS_FAILED(res)) break; } } } return res; } NS_IMETHODIMP nsHTMLEditor::SelectAllTableCells() { nsCOMPtr cell; nsresult res = GetElementOrParentByTagName(NS_LITERAL_STRING("td"), nsnull, getter_AddRefs(cell)); if (NS_FAILED(res)) return res; // Don't fail if we didn't find a cell if (!cell) return NS_EDITOR_ELEMENT_NOT_FOUND; nsCOMPtr cellNode = do_QueryInterface(cell); if (!cellNode) return NS_ERROR_FAILURE; nsCOMPtr startCell = cell; // Get parent table nsCOMPtr table; res = GetElementOrParentByTagName(NS_LITERAL_STRING("table"), cell, getter_AddRefs(table)); if (NS_FAILED(res)) return res; if(!table) return NS_ERROR_NULL_POINTER; PRInt32 rowCount, colCount; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; nsCOMPtr selection; res = GetSelection(getter_AddRefs(selection)); if (NS_FAILED(res)) return res; if (!selection) return NS_ERROR_FAILURE; // Suppress nsISelectionListener notification // until all selection changes are finished nsSelectionBatcher selectionBatcher(selection); // It is now safe to clear the selection // BE SURE TO RESET IT BEFORE LEAVING! res = ClearSelection(); // Select all cells in the same column as current cell PRBool cellSelected = PR_FALSE; PRInt32 rowSpan, colSpan, actualRowSpan, actualColSpan, currentRowIndex, currentColIndex; PRBool isSelected; for(PRInt32 row = 0; row < rowCount; row++) { for(PRInt32 col = 0; col < colCount; col += PR_MAX(actualColSpan, 1)) { res = GetCellDataAt(table, row, col, getter_AddRefs(cell), currentRowIndex, currentColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) break; // Skip cells that are spanned from previous rows or columns if (cell && row == currentRowIndex && col == currentColIndex) { cellNode = do_QueryInterface(cell); res = AppendNodeToSelectionAsRange(cellNode); if (NS_FAILED(res)) break; cellSelected = PR_TRUE; } } } // Safety code to select starting cell if nothing else was selected if (!cellSelected) { cellNode = do_QueryInterface(startCell); return AppendNodeToSelectionAsRange(cellNode); } return res; } NS_IMETHODIMP nsHTMLEditor::SelectTableRow() { nsCOMPtr cell; nsresult res = GetElementOrParentByTagName(NS_LITERAL_STRING("td"), nsnull, getter_AddRefs(cell)); if (NS_FAILED(res)) return res; // Don't fail if we didn't find a cell if (!cell) return NS_EDITOR_ELEMENT_NOT_FOUND; nsCOMPtr startCell = cell; nsCOMPtr cellNode = do_QueryInterface(cell); if (!cellNode) return NS_ERROR_FAILURE; // Get table and location of cell: nsCOMPtr selection; nsCOMPtr table; PRInt32 startRowIndex, startColIndex; res = GetCellContext(getter_AddRefs(selection), getter_AddRefs(table), getter_AddRefs(cell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; if (!table) return NS_ERROR_FAILURE; PRInt32 rowCount, colCount; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; //Note: At this point, we could get first and last cells in row, // then call SelectBlockOfCells, but that would take just // a little less code, so the following is more efficient // Suppress nsISelectionListener notification // until all selection changes are finished nsSelectionBatcher selectionBatcher(selection); // It is now safe to clear the selection // BE SURE TO RESET IT BEFORE LEAVING! res = ClearSelection(); // Select all cells in the same row as current cell PRBool cellSelected = PR_FALSE; PRInt32 rowSpan, colSpan, actualRowSpan, actualColSpan, currentRowIndex, currentColIndex; PRBool isSelected; for(PRInt32 col = 0; col < colCount; col += PR_MAX(actualColSpan, 1)) { res = GetCellDataAt(table, startRowIndex, col, getter_AddRefs(cell), currentRowIndex, currentColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) break; // Skip cells that are spanned from previous rows or columns if (cell && currentRowIndex == startRowIndex && currentColIndex == col) { cellNode = do_QueryInterface(cell); res = AppendNodeToSelectionAsRange(cellNode); if (NS_FAILED(res)) break; cellSelected = PR_TRUE; } } // Safety code to select starting cell if nothing else was selected if (!cellSelected) { cellNode = do_QueryInterface(startCell); return AppendNodeToSelectionAsRange(cellNode); } return res; } NS_IMETHODIMP nsHTMLEditor::SelectTableColumn() { nsCOMPtr cell; nsresult res = GetElementOrParentByTagName(NS_LITERAL_STRING("td"), nsnull, getter_AddRefs(cell)); if (NS_FAILED(res)) return res; // Don't fail if we didn't find a cell if (!cell) return NS_EDITOR_ELEMENT_NOT_FOUND; nsCOMPtr cellNode = do_QueryInterface(cell); if (!cellNode) return NS_ERROR_FAILURE; nsCOMPtr startCell = cell; // Get location of cell: nsCOMPtr selection; nsCOMPtr table; PRInt32 startRowIndex, startColIndex; res = GetCellContext(getter_AddRefs(selection), getter_AddRefs(table), getter_AddRefs(cell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; if (!table) return NS_ERROR_FAILURE; PRInt32 rowCount, colCount; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; // Suppress nsISelectionListener notification // until all selection changes are finished nsSelectionBatcher selectionBatcher(selection); // It is now safe to clear the selection // BE SURE TO RESET IT BEFORE LEAVING! res = ClearSelection(); // Select all cells in the same column as current cell PRBool cellSelected = PR_FALSE; PRInt32 rowSpan, colSpan, actualRowSpan, actualColSpan, currentRowIndex, currentColIndex; PRBool isSelected; for(PRInt32 row = 0; row < rowCount; row += PR_MAX(actualRowSpan, 1)) { res = GetCellDataAt(table, row, startColIndex, getter_AddRefs(cell), currentRowIndex, currentColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) break; // Skip cells that are spanned from previous rows or columns if (cell && currentRowIndex == row && currentColIndex == startColIndex) { cellNode = do_QueryInterface(cell); res = AppendNodeToSelectionAsRange(cellNode); if (NS_FAILED(res)) break; cellSelected = PR_TRUE; } } // Safety code to select starting cell if nothing else was selected if (!cellSelected) { cellNode = do_QueryInterface(startCell); return AppendNodeToSelectionAsRange(cellNode); } return res; } NS_IMETHODIMP nsHTMLEditor::SplitTableCell() { nsCOMPtr table; nsCOMPtr cell; PRInt32 startRowIndex, startColIndex, actualRowSpan, actualColSpan; // Get cell, table, etc. at selection anchor node nsresult res = GetCellContext(nsnull, getter_AddRefs(table), getter_AddRefs(cell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; if(!table || !cell) return NS_EDITOR_ELEMENT_NOT_FOUND; // We need rowspan and colspan data res = GetCellSpansAt(table, startRowIndex, startColIndex, actualRowSpan, actualColSpan); if (NS_FAILED(res)) return res; // Must have some span to split if (actualRowSpan <= 1 && actualColSpan <= 1) return NS_OK; nsAutoEditBatch beginBatching(this); // Prevent auto insertion of BR in new cell until we're done nsAutoRules beginRulesSniffing(this, kOpInsertNode, nsIEditor::eNext); // We reset selection nsSetSelectionAfterTableEdit setCaret(this, table, startRowIndex, startColIndex, ePreviousColumn, PR_FALSE); //...so suppress Rules System selection munging nsAutoTxnsConserveSelection dontChangeSelection(this); nsCOMPtr newCellInRow; nsCOMPtr newCellInCol; PRInt32 rowIndex = startRowIndex; PRInt32 rowSpanBelow, colSpanAfter; // Split up cell row-wise first into rowspan=1 above, and the rest below, // whittling away at the cell below until no more extra span for (rowSpanBelow = actualRowSpan-1; rowSpanBelow >= 0; rowSpanBelow--) { // We really split row-wise only if we had rowspan > 1 if (rowSpanBelow > 0) { res = SplitCellIntoRows(table, rowIndex, startColIndex, 1, rowSpanBelow, nsnull); if (NS_FAILED(res)) return res; } PRInt32 colIndex = startColIndex; // Now split the cell with rowspan = 1 into cells if it has colSpan > 1 for (colSpanAfter = actualColSpan-1; colSpanAfter > 0; colSpanAfter--) { res = SplitCellIntoColumns(table, rowIndex, colIndex, 1, colSpanAfter, nsnull); if (NS_FAILED(res)) return res; colIndex++; } // Point to the new cell and repeat rowIndex++; } return res; } nsresult nsHTMLEditor::CopyCellBackgroundColor(nsIDOMElement *destCell, nsIDOMElement *sourceCell) { if (!destCell || !sourceCell) return NS_ERROR_NULL_POINTER; // Copy backgournd color to new cell nsAutoString bgcolor; bgcolor.AssignWithConversion("bgcolor"); nsAutoString color; PRBool isSet; nsresult res = GetAttributeValue(sourceCell, bgcolor, color, &isSet); if (NS_SUCCEEDED(res) && isSet) res = SetAttribute(destCell, bgcolor, color); return res; } NS_IMETHODIMP nsHTMLEditor::SplitCellIntoColumns(nsIDOMElement *aTable, PRInt32 aRowIndex, PRInt32 aColIndex, PRInt32 aColSpanLeft, PRInt32 aColSpanRight, nsIDOMElement **aNewCell) { if (!aTable) return NS_ERROR_NULL_POINTER; if (aNewCell) *aNewCell = nsnull; nsCOMPtr cell; PRInt32 startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; nsresult res = GetCellDataAt(aTable, aRowIndex, aColIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return res; if (!cell) return NS_ERROR_NULL_POINTER; // We can't split! if (actualColSpan <= 1 || (aColSpanLeft + aColSpanRight) > actualColSpan) return NS_OK; // Reduce colspan of cell to split res = SetColSpan(cell, aColSpanLeft); if (NS_FAILED(res)) return res; // Insert new cell after using the remaining span // and always get the new cell so we can copy the background color; nsCOMPtr newCell; res = InsertCell(cell, actualRowSpan, aColSpanRight, PR_TRUE, PR_FALSE, getter_AddRefs(newCell)); if (NS_FAILED(res)) return res; if (newCell) { if (aNewCell) { *aNewCell = newCell.get(); NS_ADDREF(*aNewCell); } res = CopyCellBackgroundColor(newCell, cell); } return res; } NS_IMETHODIMP nsHTMLEditor::SplitCellIntoRows(nsIDOMElement *aTable, PRInt32 aRowIndex, PRInt32 aColIndex, PRInt32 aRowSpanAbove, PRInt32 aRowSpanBelow, nsIDOMElement **aNewCell) { if (!aTable) return NS_ERROR_NULL_POINTER; if (aNewCell) *aNewCell = nsnull; nsCOMPtr cell; PRInt32 startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; nsresult res = GetCellDataAt(aTable, aRowIndex, aColIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return res; if (!cell) return NS_ERROR_NULL_POINTER; // We can't split! if (actualRowSpan <= 1 || (aRowSpanAbove + aRowSpanBelow) > actualRowSpan) return NS_OK; nsCOMPtr cell2; nsCOMPtr lastCellFound; PRInt32 startRowIndex2, startColIndex2, rowSpan2, colSpan2, actualRowSpan2, actualColSpan2; PRBool isSelected2; PRInt32 colIndex = 0; PRBool insertAfter = (startColIndex > 0); // This is the row we will insert new cell into PRInt32 rowBelowIndex = startRowIndex+aRowSpanAbove; // Find a cell to insert before or after do { // Search for a cell to insert before res = GetCellDataAt(aTable, rowBelowIndex, colIndex, getter_AddRefs(cell2), startRowIndex2, startColIndex2, rowSpan2, colSpan2, actualRowSpan2, actualColSpan2, isSelected2); // If we fail here, it could be because row has bad rowspan values, // such as all cells having rowspan > 1 (Call FixRowSpan first!) if (NS_FAILED(res) || !cell) return NS_ERROR_FAILURE; // Skip over cells spanned from above (like the one we are splitting!) if (startRowIndex2 == rowBelowIndex) { if (insertAfter) { // New cell isn't first in row, // so stop after we find the cell just before new cell's column if ((startColIndex2 + actualColSpan2) == startColIndex) break; // If cell found is AFTER desired new cell colum, // we have multiple cells with rowspan > 1 that // prevented us from finding a cell to insert after... if (startColIndex2 > startColIndex) { // ... so instead insert before the cell we found insertAfter = PR_FALSE; break; } } else { break; // Inserting before, so stop at first cell in row we want to insert into } lastCellFound = cell2; } // Skip to next available cellmap location colIndex += actualColSpan2; } while(PR_TRUE); if (!cell2 && lastCellFound) { // Edge case where we didn't find a cell to insert after // or before because column(s) before desired column // and all columns after it are spanned from above. // We can insert after the last cell we found cell2 = lastCellFound; insertAfter = PR_TRUE; // Should always be true, but let's be sure } // Reduce rowspan of cell to split res = SetRowSpan(cell, aRowSpanAbove); if (NS_FAILED(res)) return res; // Insert new cell after using the remaining span // and always get the new cell so we can copy the background color; nsCOMPtr newCell; res = InsertCell(cell2, aRowSpanBelow, actualColSpan, insertAfter, PR_FALSE, getter_AddRefs(newCell)); if (NS_FAILED(res)) return res; if (newCell) { if (aNewCell) { *aNewCell = newCell.get(); NS_ADDREF(*aNewCell); } res = CopyCellBackgroundColor(newCell, cell2); } return res; if (NS_FAILED(res)) return res; return CopyCellBackgroundColor(*aNewCell, cell); } NS_IMETHODIMP nsHTMLEditor::SwitchTableCellHeaderType(nsIDOMElement *aSourceCell, nsIDOMElement **aNewCell) { if (!aSourceCell) return NS_ERROR_NULL_POINTER; nsAutoEditBatch beginBatching(this); // Prevent auto insertion of BR in new cell created by ReplaceContainer nsAutoRules beginRulesSniffing(this, kOpInsertNode, nsIEditor::eNext); nsCOMPtr sourceNode = do_QueryInterface(aSourceCell); nsCOMPtr newNode; // Set to the opposite of current type nsAutoString tagName; GetTagString(aSourceCell, tagName); nsString newCellType( (tagName == NS_LITERAL_STRING("td")) ? NS_LITERAL_STRING("th") : NS_LITERAL_STRING("td") ); // Save current selection to restore when done // This is needed so ReplaceContainer can monitor selection // when replacing nodes nsCOMPtrselection; nsresult res = GetSelection(getter_AddRefs(selection)); if (NS_FAILED(res)) return res; if (!selection) return NS_ERROR_FAILURE; nsAutoSelectionReset selectionResetter(selection, this); // This creates new node, moves children, copies attributes (PR_TRUE) // and manages the selection! res = ReplaceContainer(sourceNode, address_of(newNode), newCellType, nsnull, nsnull, PR_TRUE); if (NS_FAILED(res)) return res; if (!newNode) return NS_ERROR_FAILURE; // Return the new cell if (aNewCell) { nsCOMPtr newElement = do_QueryInterface(newNode); *aNewCell = newElement.get(); NS_ADDREF(*aNewCell); } return NS_OK; } NS_IMETHODIMP nsHTMLEditor::JoinTableCells(PRBool aMergeNonContiguousContents) { nsCOMPtr table; nsCOMPtr targetCell; PRInt32 startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; nsCOMPtr cell2; PRInt32 startRowIndex2, startColIndex2, rowSpan2, colSpan2, actualRowSpan2, actualColSpan2; PRBool isSelected2; // Get cell, table, etc. at selection anchor node nsresult res = GetCellContext(nsnull, getter_AddRefs(table), getter_AddRefs(targetCell), nsnull, nsnull, &startRowIndex, &startColIndex); if (NS_FAILED(res)) return res; if(!table || !targetCell) return NS_EDITOR_ELEMENT_NOT_FOUND; nsAutoEditBatch beginBatching(this); //Don't let Rules System change the selection nsAutoTxnsConserveSelection dontChangeSelection(this); // Note: We dont' use nsSetSelectionAfterTableEdit here so the selection // is retained after joining. This leaves the target cell selected // as well as the "non-contiguous" cells, so user can see what happened. nsCOMPtr firstCell; PRInt32 firstRowIndex, firstColIndex; res = GetFirstSelectedCellInTable(getter_AddRefs(firstCell), &firstRowIndex, &firstColIndex); if (NS_FAILED(res)) return res; if (firstCell) { // We have selected cells: Join just contiguous cells // and just merge contents if not contiguous PRInt32 rowCount, colCount; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; // Get spans for cell we will merge into PRInt32 firstRowSpan, firstColSpan; res = GetCellSpansAt( table, firstRowIndex, firstColIndex, firstRowSpan, firstColSpan); if (NS_FAILED(res)) return res; // This defines the last indexes along the "edges" // of the contiguous block of cells, telling us // that we can join adjacent cells to the block // Start with same as the first values, // then expand as we find adjacent selected cells PRInt32 lastRowIndex = firstRowIndex; PRInt32 lastColIndex = firstColIndex; PRInt32 rowIndex, colIndex; // First pass: Determine boundaries of contiguous rectangular block // that we will join into one cell, // favoring adjacent cells in the same row for (rowIndex = firstRowIndex; rowIndex <= lastRowIndex; rowIndex++) { PRInt32 currentRowCount = rowCount; // Be sure each row doesn't have rowspan errors res = FixBadRowSpan(table, rowIndex, rowCount); if (NS_FAILED(res)) return res; // Adjust rowcount by number of rows we removed lastRowIndex -= (currentRowCount-rowCount); PRBool cellFoundInRow = PR_FALSE; PRBool lastRowIsSet = PR_FALSE; PRInt32 lastColInRow = 0; PRInt32 firstColInRow = firstColIndex; for (colIndex = firstColIndex; colIndex < colCount; colIndex += PR_MAX(actualColSpan2, 1)) { res = GetCellDataAt(table, rowIndex, colIndex, getter_AddRefs(cell2), startRowIndex2, startColIndex2, rowSpan2, colSpan2, actualRowSpan2, actualColSpan2, isSelected2); if (NS_FAILED(res)) return res; if (isSelected2) { if (!cellFoundInRow) // We've just found the first selected cell in this row firstColInRow = colIndex; if(rowIndex > firstRowIndex && firstColInRow != firstColIndex) { // We're in at least the second row, // but left boundary is "ragged" (not the same as 1st row's start) //Let's just end block on previous row // and keep previous lastColIndex //TODO: We could try to find the Maximum firstColInRow // so our block can still extend down more rows? lastRowIndex = PR_MAX(0,rowIndex - 1); lastRowIsSet = PR_TRUE; break; } // Save max selected column in this row, including extra colspan lastColInRow = colIndex + (actualColSpan2-1); cellFoundInRow = PR_TRUE; } else if (cellFoundInRow) { // No cell or not selected, but at least one in row was found if (colIndex <= lastColIndex) { // Cell is in a column less than current right border, // so stop block at the previous row lastRowIndex = PR_MAX(0,rowIndex - 1); lastRowIsSet = PR_TRUE; } // We're done with this row break; } } // End of column loop // Done with this row if (cellFoundInRow) { if (rowIndex == firstRowIndex) { // First row always initializes the right boundary lastColIndex = lastColInRow; } // If we didn't determine last row above... if (!lastRowIsSet) { if (colIndex < lastColIndex) { // (don't think we ever get here?) // Cell is in a column less than current right boundary, // so stop block at the previous row lastRowIndex = PR_MAX(0,rowIndex - 1); } else { // Go on to examine next row lastRowIndex = rowIndex+1; // Use the minimun col we found so far for right boundary lastColIndex = PR_MIN(lastColIndex, lastColInRow); } } } else { // No selected cells in this row -- stop at row above // and leave last column at its previous value lastRowIndex = PR_MAX(0,rowIndex - 1); } } // The list of cells we will delete after joining nsVoidArray deleteList; // 2nd pass: Do the joining and merging for (rowIndex = 0; rowIndex < rowCount; rowIndex++) { for (colIndex = 0; colIndex < colCount; colIndex += PR_MAX(actualColSpan2, 1)) { res = GetCellDataAt(table, rowIndex, colIndex, getter_AddRefs(cell2), startRowIndex2, startColIndex2, rowSpan2, colSpan2, actualRowSpan2, actualColSpan2, isSelected2); if (NS_FAILED(res)) return res; // If this is 0, we are past last cell in row, so exit the loop if (actualColSpan2 == 0) break; // Merge only selected cells (skip cell we're merging into, of course) if (isSelected2 && cell2 != firstCell) { if (rowIndex >= firstRowIndex && rowIndex <= lastRowIndex && colIndex >= firstColIndex && colIndex <= lastColIndex) { // We are within the join region // Problem: It is very tricky to delete cells as we merge, // since that will upset the cellmap // Instead, build a list of cells to delete and do it later NS_ASSERTION(startRowIndex2 == rowIndex, "JoinTableCells: StartRowIndex is in row above"); //Check if cell "hangs" off the boundary because of colspan or rowspan > 1 // Use split methods to chop off excess PRInt32 extraColSpan = lastColIndex - (startColIndex2 + actualColSpan2); if ( extraColSpan > 0) { res = SplitCellIntoColumns(table, startRowIndex2, startColIndex2, actualColSpan2-extraColSpan, extraColSpan, nsnull); if (NS_FAILED(res)) return res; } res = MergeCells(firstCell, cell2, PR_FALSE); if (NS_FAILED(res)) return res; // Add cell to list to delete deleteList.AppendElement((void *)cell2.get()); } else if (aMergeNonContiguousContents) { // Cell is outside join region -- just merge the contents res = MergeCells(firstCell, cell2, PR_FALSE); if (NS_FAILED(res)) return res; } } } } // All cell contents are merged. Delete the empty cells we accumulated // Prevent rules testing until we're done nsAutoRules beginRulesSniffing(this, kOpDeleteNode, nsIEditor::eNext); nsIDOMElement *elementPtr; PRInt32 count; while ((count = deleteList.Count())) { // go backwards to keep nsVoidArray from mem-moving everything each time count--; // nsVoidArray is zero based elementPtr = (nsIDOMElement*)deleteList.ElementAt(count); deleteList.RemoveElementAt(count); if (elementPtr) { nsCOMPtr node = do_QueryInterface(elementPtr); res = DeleteNode(node); if (NS_FAILED(res)) return res; // Should we delete this??? //delete elementPtr; } } // Set spans for the cell everthing merged into res = SetRowSpan(firstCell, lastRowIndex-firstRowIndex+1); if (NS_FAILED(res)) return res; res = SetColSpan(firstCell, lastColIndex-firstColIndex+1); if (NS_FAILED(res)) return res; // But check if we merged multiple rows - rowspan shouldn't be > 1 PRInt32 newRowCount; res = FixBadRowSpan(table, firstRowIndex, newRowCount); if (NS_FAILED(res)) return res; } else { // Joining with cell to the right -- get rowspan and colspan data of target cell res = GetCellDataAt(table, startRowIndex, startColIndex, getter_AddRefs(targetCell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return res; if (!targetCell) return NS_ERROR_NULL_POINTER; // Get data for cell to the right res = GetCellDataAt(table, startRowIndex, startColIndex+actualColSpan, getter_AddRefs(cell2), startRowIndex2, startColIndex2, rowSpan2, colSpan2, actualRowSpan2, actualColSpan2, isSelected2); if (NS_FAILED(res)) return res; if(!cell2) return NS_OK; // Don't fail if there's no cell // sanity check NS_ASSERTION((startRowIndex >= startRowIndex2),"JoinCells: startRowIndex < startRowIndex2"); // Figure out span of merged cell starting from target's starting row // to handle case of merged cell starting in a row above PRInt32 spanAboveMergedCell = startRowIndex - startRowIndex2; PRInt32 effectiveRowSpan2 = actualRowSpan2 - spanAboveMergedCell; if (effectiveRowSpan2 > actualRowSpan) { // Cell to the right spans into row below target // Split off portion below target cell's bottom-most row res = SplitCellIntoRows(table, startRowIndex2, startColIndex2, spanAboveMergedCell+actualRowSpan, effectiveRowSpan2-actualRowSpan, nsnull); if (NS_FAILED(res)) return res; } // Move contents from cell to the right // Delete the cell now only if it starts in the same row // and has enough row "height" res = MergeCells(targetCell, cell2, (startRowIndex2 == startRowIndex) && (effectiveRowSpan2 >= actualRowSpan)); if (NS_FAILED(res)) return res; if (effectiveRowSpan2 < actualRowSpan) { // Merged cell is "shorter" // (there are cells(s) below it that are row-spanned by target cell) // We could try splitting those cells, but that's REAL messy, // so the safest thing to do is NOT really join the cells return NS_OK; } if( spanAboveMergedCell > 0 ) { // Cell we merged started in a row above the target cell // Reduce rowspan to give room where target cell will extend it's colspan res = SetRowSpan(cell2, spanAboveMergedCell); if (NS_FAILED(res)) return res; } // Reset target cell's colspan to encompass cell to the right res = SetColSpan(targetCell, actualColSpan+actualColSpan2); if (NS_FAILED(res)) return res; } return res; } NS_IMETHODIMP nsHTMLEditor::MergeCells(nsCOMPtr aTargetCell, nsCOMPtr aCellToMerge, PRBool aDeleteCellToMerge) { nsCOMPtr targetCell = do_QueryInterface(aTargetCell); nsCOMPtr cellToMerge = do_QueryInterface(aCellToMerge); if(!targetCell || !cellToMerge) return NS_ERROR_NULL_POINTER; nsresult res = NS_OK; // Prevent rules testing until we're done nsAutoRules beginRulesSniffing(this, kOpDeleteNode, nsIEditor::eNext); // Get index of last child in target cell nsCOMPtr childNodes; nsCOMPtr cellChild; res = targetCell->GetChildNodes(getter_AddRefs(childNodes)); // If we fail or don't have children, // we insert at index 0 PRInt32 insertIndex = 0; if ((NS_SUCCEEDED(res)) && (childNodes)) { // Start inserting just after last child PRUint32 len; res = childNodes->GetLength(&len); if (NS_FAILED(res)) return res; if (len == 1 && IsEmptyCell(aTargetCell)) { if (IsEmptyCell(aTargetCell)) { // Delete the empty node nsCOMPtr tempNode; res = childNodes->Item(0, getter_AddRefs(cellChild)); if (NS_FAILED(res)) return res; res = DeleteNode(cellChild); if (NS_FAILED(res)) return res; insertIndex = 0; } } else insertIndex = (PRInt32)len; } // Move the contents PRBool hasChild; cellToMerge->HasChildNodes(&hasChild); while (hasChild) { cellToMerge->GetLastChild(getter_AddRefs(cellChild)); res = DeleteNode(cellChild); if (NS_FAILED(res)) return res; res = InsertNode(cellChild, targetCell, insertIndex); if (NS_FAILED(res)) return res; cellToMerge->HasChildNodes(&hasChild); } // Delete cells whose contents were moved if (aDeleteCellToMerge) res = DeleteNode(cellToMerge); return res; } NS_IMETHODIMP nsHTMLEditor::FixBadRowSpan(nsIDOMElement *aTable, PRInt32 aRowIndex, PRInt32& aNewRowCount) { if (!aTable) return NS_ERROR_NULL_POINTER; PRInt32 rowCount, colCount; nsresult res = GetTableSize(aTable, rowCount, colCount); if (NS_FAILED(res)) return res; nsCOMPtrcell; PRInt32 startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; PRInt32 minRowSpan = -1; PRInt32 colIndex; for( colIndex = 0; colIndex < colCount; colIndex += PR_MAX(actualColSpan, 1)) { res = GetCellDataAt(aTable, aRowIndex, colIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); // NOTE: This is a *real* failure. // GetCellDataAt passes if cell is missing from cellmap if(NS_FAILED(res)) return res; if (!cell) break; if(rowSpan > 0 && startRowIndex == aRowIndex && (rowSpan < minRowSpan || minRowSpan == -1)) { minRowSpan = rowSpan; } NS_ASSERTION((actualColSpan > 0),"ActualColSpan = 0 in FixBadRowSpan"); } if(minRowSpan > 1) { // The amount to reduce everyone's rowspan // so at least one cell has rowspan = 1 PRInt32 rowsReduced = minRowSpan - 1; for(colIndex = 0; colIndex < colCount; colIndex += PR_MAX(actualColSpan, 1)) { res = GetCellDataAt(aTable, aRowIndex, colIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if(NS_FAILED(res)) return res; // Fixup rowspans only for cells starting in current row if(cell && rowSpan > 0 && startRowIndex == aRowIndex && startColIndex == colIndex ) { res = SetRowSpan(cell, rowSpan-rowsReduced); if(NS_FAILED(res)) return res; } NS_ASSERTION((actualColSpan > 0),"ActualColSpan = 0 in FixBadRowSpan"); } } return GetTableSize(aTable, aNewRowCount, colCount); } NS_IMETHODIMP nsHTMLEditor::FixBadColSpan(nsIDOMElement *aTable, PRInt32 aColIndex, PRInt32& aNewColCount) { if (!aTable) return NS_ERROR_NULL_POINTER; PRInt32 rowCount, colCount; nsresult res = GetTableSize(aTable, rowCount, colCount); if (NS_FAILED(res)) return res; nsCOMPtr cell; PRInt32 startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; PRInt32 minColSpan = -1; PRInt32 rowIndex; for( rowIndex = 0; rowIndex < rowCount; rowIndex += PR_MAX(actualRowSpan, 1)) { res = GetCellDataAt(aTable, rowIndex, aColIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); // NOTE: This is a *real* failure. // GetCellDataAt passes if cell is missing from cellmap if(NS_FAILED(res)) return res; if (!cell) break; if(colSpan > 0 && startColIndex == aColIndex && (colSpan < minColSpan || minColSpan == -1)) { minColSpan = colSpan; } NS_ASSERTION((actualRowSpan > 0),"ActualRowSpan = 0 in FixBadColSpan"); } if(minColSpan > 1) { // The amount to reduce everyone's colspan // so at least one cell has colspan = 1 PRInt32 colsReduced = minColSpan - 1; for(rowIndex = 0; rowIndex < rowCount; rowIndex += PR_MAX(actualRowSpan, 1)) { res = GetCellDataAt(aTable, rowIndex, aColIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if(NS_FAILED(res)) return res; // Fixup colspans only for cells starting in current column if(cell && colSpan > 0 && startColIndex == aColIndex && startRowIndex == rowIndex ) { res = SetColSpan(cell, colSpan-colsReduced); if(NS_FAILED(res)) return res; } NS_ASSERTION((actualRowSpan > 0),"ActualRowSpan = 0 in FixBadColSpan"); } } return GetTableSize(aTable, rowCount, aNewColCount); } NS_IMETHODIMP nsHTMLEditor::NormalizeTable(nsIDOMElement *aTable) { nsCOMPtrselection; nsresult res = GetSelection(getter_AddRefs(selection)); if (NS_FAILED(res)) return res; if (!selection) return NS_ERROR_FAILURE; nsCOMPtr table; res = GetElementOrParentByTagName(NS_LITERAL_STRING("table"), aTable, getter_AddRefs(table)); if (NS_FAILED(res)) return res; // Don't fail if we didn't find a table if (!table) return NS_OK; PRInt32 rowCount, colCount, rowIndex, colIndex; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; // Save current selection nsAutoSelectionReset selectionResetter(selection, this); nsAutoEditBatch beginBatching(this); // Prevent auto insertion of BR in new cell until we're done nsAutoRules beginRulesSniffing(this, kOpInsertNode, nsIEditor::eNext); nsCOMPtr cell; PRInt32 startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; // Scan all cells in each row to detect bad rowspan values for(rowIndex = 0; rowIndex < rowCount; rowIndex++) { res = FixBadRowSpan(table, rowIndex, rowCount); if (NS_FAILED(res)) return res; } // and same for colspans for(colIndex = 0; colIndex < colCount; colIndex++) { res = FixBadColSpan(table, colIndex, colCount); if (NS_FAILED(res)) return res; } // Fill in missing cellmap locations with empty cells for(rowIndex = 0; rowIndex < rowCount; rowIndex++) { nsCOMPtr previousCellInRow; for(colIndex = 0; colIndex < colCount; colIndex++) { res = GetCellDataAt(table, rowIndex, colIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); // NOTE: This is a *real* failure. // GetCellDataAt passes if cell is missing from cellmap if(NS_FAILED(res)) return res; if (!cell) { //We are missing a cell at a cellmap location #ifdef DEBUG printf("NormalizeTable found missing cell at row=%d, col=%d\n", rowIndex, colIndex); #endif // Add a cell after the previous Cell in the current row if(previousCellInRow) { // Insert a new cell after (PR_TRUE), and return the new cell to us res = InsertCell(previousCellInRow, 1, 1, PR_TRUE, PR_FALSE, getter_AddRefs(cell)); if (NS_FAILED(res)) return res; // Set this so we use returned new "cell" to set previousCellInRow below if(cell) startRowIndex = rowIndex; } else { // We don't have any cells in this row -- We are really messed up! #ifdef DEBUG printf("NormalizeTable found no cells in row=%d, col=%d\n", rowIndex, colIndex); #endif return NS_ERROR_FAILURE; } } // Save the last cell found in the same row we are scanning if(startRowIndex == rowIndex) { previousCellInRow = cell; } } } return res; } NS_IMETHODIMP nsHTMLEditor::GetCellIndexes(nsIDOMElement *aCell, PRInt32 &aRowIndex, PRInt32 &aColIndex) { nsresult res=NS_ERROR_NOT_INITIALIZED; aColIndex=0; // initialize out params aRowIndex=0; if (!aCell) { // Get the selected cell or the cell enclosing the selection anchor nsCOMPtr cell; res = GetElementOrParentByTagName(NS_LITERAL_STRING("td"), nsnull, getter_AddRefs(cell)); if (NS_SUCCEEDED(res) && cell) aCell = cell; else return NS_ERROR_FAILURE; } nsISupports *layoutObject=nsnull; // frames are not ref counted, so don't use an nsCOMPtr res = nsHTMLEditor::GetLayoutObject(aCell, &layoutObject); if (NS_FAILED(res)) return res; if (!layoutObject) return NS_ERROR_FAILURE; nsITableCellLayout *cellLayoutObject=nsnull; // again, frames are not ref-counted res = layoutObject->QueryInterface(NS_GET_IID(nsITableCellLayout), (void**)(&cellLayoutObject)); if (NS_FAILED(res)) return res; if (!cellLayoutObject) return NS_ERROR_FAILURE; return cellLayoutObject->GetCellIndexes(aRowIndex, aColIndex); } NS_IMETHODIMP nsHTMLEditor::GetTableLayoutObject(nsIDOMElement* aTable, nsITableLayout **tableLayoutObject) { *tableLayoutObject=nsnull; if (!aTable) return NS_ERROR_NOT_INITIALIZED; // frames are not ref counted, so don't use an nsCOMPtr nsISupports *layoutObject=nsnull; nsresult res = GetLayoutObject(aTable, &layoutObject); if (NS_FAILED(res)) return res; if (!layoutObject) return NS_ERROR_FAILURE; return layoutObject->QueryInterface(NS_GET_IID(nsITableLayout), (void**)(tableLayoutObject)); } //Return actual number of cells (a cell with colspan > 1 counts as just 1) PRBool nsHTMLEditor::GetNumberOfCellsInRow(nsIDOMElement* aTable, PRInt32 rowIndex) { PRInt32 cellCount = 0; nsCOMPtr cell; PRInt32 colIndex = 0; nsresult res; do { PRInt32 startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; res = GetCellDataAt(aTable, rowIndex, colIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return res; if (cell) { // Only count cells that start in row we are working with if (startRowIndex == rowIndex) cellCount++; //Next possible location for a cell colIndex += actualColSpan; } else colIndex++; } while (cell); return cellCount; } /* Not scriptable: For convenience in C++ Use GetTableRowCount and GetTableColumnCount from JavaScript */ NS_IMETHODIMP nsHTMLEditor::GetTableSize(nsIDOMElement *aTable, PRInt32& aRowCount, PRInt32& aColCount) { nsresult res = NS_ERROR_FAILURE; aRowCount = 0; aColCount = 0; nsCOMPtr table; // Get the selected talbe or the table enclosing the selection anchor res = GetElementOrParentByTagName(NS_LITERAL_STRING("table"), aTable, getter_AddRefs(table)); if (NS_FAILED(res)) return res; if (!table) return NS_ERROR_FAILURE; // frames are not ref counted, so don't use an nsCOMPtr nsITableLayout *tableLayoutObject; res = GetTableLayoutObject(table.get(), &tableLayoutObject); if (NS_FAILED(res)) return res; if (!tableLayoutObject) return NS_ERROR_FAILURE; return tableLayoutObject->GetTableSize(aRowCount, aColCount); } NS_IMETHODIMP nsHTMLEditor::GetCellDataAt(nsIDOMElement* aTable, PRInt32 aRowIndex, PRInt32 aColIndex, nsIDOMElement **aCell, PRInt32& aStartRowIndex, PRInt32& aStartColIndex, PRInt32& aRowSpan, PRInt32& aColSpan, PRInt32& aActualRowSpan, PRInt32& aActualColSpan, PRBool& aIsSelected) { nsresult res=NS_ERROR_FAILURE; aStartRowIndex = 0; aStartColIndex = 0; aRowSpan = 0; aColSpan = 0; aActualRowSpan = 0; aActualColSpan = 0; aIsSelected = PR_FALSE; if (!aCell) return NS_ERROR_NULL_POINTER; *aCell = nsnull; if (!aTable) { // Get the selected table or the table enclosing the selection anchor nsCOMPtr table; res = GetElementOrParentByTagName(NS_LITERAL_STRING("table"), nsnull, getter_AddRefs(table)); if (NS_FAILED(res)) return res; if (table) aTable = table; else return NS_ERROR_FAILURE; } // frames are not ref counted, so don't use an nsCOMPtr nsITableLayout *tableLayoutObject; res = GetTableLayoutObject(aTable, &tableLayoutObject); if (NS_FAILED(res)) return res; if (!tableLayoutObject) return NS_ERROR_FAILURE; // Note that this returns NS_TABLELAYOUT_CELL_NOT_FOUND when // the index(es) are out of bounds nsCOMPtr cell; res = tableLayoutObject->GetCellDataAt(aRowIndex, aColIndex, *getter_AddRefs(cell), aStartRowIndex, aStartColIndex, aRowSpan, aColSpan, aActualRowSpan, aActualColSpan, aIsSelected); if (cell) { *aCell = cell.get(); NS_ADDREF(*aCell); } // Convert to editor's generic "not found" return value if (res == NS_TABLELAYOUT_CELL_NOT_FOUND) res = NS_EDITOR_ELEMENT_NOT_FOUND; return res; } // When all you want is the cell NS_IMETHODIMP nsHTMLEditor::GetCellAt(nsIDOMElement* aTable, PRInt32 aRowIndex, PRInt32 aColIndex, nsIDOMElement **aCell) { PRInt32 startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; return GetCellDataAt(aTable, aRowIndex, aColIndex, aCell, startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); } // When all you want are the rowspan and colspan (not exposed in nsITableEditor) NS_IMETHODIMP nsHTMLEditor::GetCellSpansAt(nsIDOMElement* aTable, PRInt32 aRowIndex, PRInt32 aColIndex, PRInt32& aActualRowSpan, PRInt32& aActualColSpan) { nsCOMPtr cell; PRInt32 startRowIndex, startColIndex, rowSpan, colSpan; PRBool isSelected; return GetCellDataAt(aTable, aRowIndex, aColIndex, getter_AddRefs(cell), startRowIndex, startColIndex, rowSpan, colSpan, aActualRowSpan, aActualColSpan, isSelected); } NS_IMETHODIMP nsHTMLEditor::GetCellContext(nsISelection **aSelection, nsIDOMElement **aTable, nsIDOMElement **aCell, nsIDOMNode **aCellParent, PRInt32 *aCellOffset, PRInt32 *aRowIndex, PRInt32 *aColIndex) { // Initialize return pointers if (aSelection) *aSelection = nsnull; if (aTable) *aTable = nsnull; if (aCell) *aCell = nsnull; if (aCellParent) *aCellParent = nsnull; if (aCellOffset) *aCellOffset = 0; if (aRowIndex) *aRowIndex = 0; if (aColIndex) *aColIndex = 0; nsCOMPtr selection; nsresult res = GetSelection(getter_AddRefs(selection)); if (NS_FAILED(res)) return res; if (!selection) return NS_ERROR_FAILURE; if (aSelection) { *aSelection = selection.get(); NS_ADDREF(*aSelection); } nsCOMPtr table; nsCOMPtr cell; // Caller may supply the cell... if (aCell && *aCell) cell = *aCell; // ...but if not supplied, // get cell if it's the child of selection anchor node, // or get the enclosing by a cell if (!cell) { // Find a selected or enclosing table element nsCOMPtr cellOrTableElement; PRInt32 selectedCount; nsAutoString tagName; res = GetSelectedOrParentTableElement(*getter_AddRefs(cellOrTableElement), tagName, selectedCount); if (NS_FAILED(res)) return res; if (tagName == NS_LITERAL_STRING("table")) { // We have a selected table, not a cell if (aTable) { *aTable = cellOrTableElement.get(); NS_ADDREF(*aTable); } return NS_OK; } // Don't fail if we are not in a cell if (tagName != NS_LITERAL_STRING("td")) return NS_EDITOR_ELEMENT_NOT_FOUND; // We found a cell cell = cellOrTableElement; } if (aCell) { *aCell = cell.get(); NS_ADDREF(*aCell); } // Get containing table res = GetElementOrParentByTagName(NS_LITERAL_STRING("table"), cell, getter_AddRefs(table)); if (NS_FAILED(res)) return res; // Cell must be in a table, so fail if not found if (!table) return NS_ERROR_FAILURE; if (aTable) { *aTable = table.get(); NS_ADDREF(*aTable); } // Get the rest of the related data only if requested if (aRowIndex || aColIndex) { PRInt32 rowIndex, colIndex; // Get current cell location so we can put caret back there when done res = GetCellIndexes(cell, rowIndex, colIndex); if(NS_FAILED(res)) return res; if (aRowIndex) *aRowIndex = rowIndex; if (aColIndex) *aColIndex = colIndex; } if (aCellParent) { nsCOMPtr cellParent; // Get the immediate parent of the cell res = cell->GetParentNode(getter_AddRefs(cellParent)); if (NS_FAILED(res)) return res; // Cell has to have a parent, so fail if not found if (!cellParent) return NS_ERROR_FAILURE; *aCellParent = cellParent.get(); NS_ADDREF(*aCellParent); if (aCellOffset) res = GetChildOffset(cell, cellParent, *aCellOffset); } return res; } nsresult nsHTMLEditor::GetCellFromRange(nsIDOMRange *aRange, nsIDOMElement **aCell) { // Note: this might return a node that is outside of the range. // Use carefully. if (!aRange || !aCell) return NS_ERROR_NULL_POINTER; *aCell = nsnull; nsCOMPtr startParent; nsresult res = aRange->GetStartContainer(getter_AddRefs(startParent)); if (NS_FAILED(res)) return res; if (!startParent) return NS_ERROR_FAILURE; PRInt32 startOffset; res = aRange->GetStartOffset(&startOffset); if (NS_FAILED(res)) return res; nsCOMPtr childNode = GetChildAt(startParent, startOffset); // This means selection is probably at a text node (or end of doc?) if (!childNode) return NS_ERROR_FAILURE; nsCOMPtr endParent; res = aRange->GetEndContainer(getter_AddRefs(endParent)); if (NS_FAILED(res)) return res; if (!startParent) return NS_ERROR_FAILURE; PRInt32 endOffset; res = aRange->GetEndOffset(&endOffset); if (NS_FAILED(res)) return res; // If a cell is deleted, the range is collapse // (startOffset == endOffset) // so tell caller the cell wasn't found if (startParent == endParent && endOffset == startOffset+1 && nsHTMLEditUtils::IsTableCell(childNode)) { // Should we also test if frame is selected? (Use GetCellDataAt()) // (Let's not for now -- more efficient) nsCOMPtr cellElement = do_QueryInterface(childNode); *aCell = cellElement.get(); NS_ADDREF(*aCell); return NS_OK; } return NS_EDITOR_ELEMENT_NOT_FOUND; } NS_IMETHODIMP nsHTMLEditor::GetFirstSelectedCell(nsIDOMElement **aCell, nsIDOMRange **aRange) { if (!aCell) return NS_ERROR_NULL_POINTER; *aCell = nsnull; if (aRange) *aRange = nsnull; nsCOMPtr selection; nsresult res = GetSelection(getter_AddRefs(selection)); if (NS_FAILED(res)) return res; if (!selection) return NS_ERROR_FAILURE; nsCOMPtr range; res = selection->GetRangeAt(0, getter_AddRefs(range)); if (NS_FAILED(res)) return res; if (!range) return NS_ERROR_FAILURE; mSelectedCellIndex = 0; nsCOMPtr cellNode; res = GetCellFromRange(range, aCell); // Failure here probably means selection is in a text node, // so there's no selected cell if (NS_FAILED(res)) return NS_EDITOR_ELEMENT_NOT_FOUND; // No cell means range was collapsed (cell was deleted) if (!*aCell) return NS_EDITOR_ELEMENT_NOT_FOUND; if (aRange) { *aRange = range.get(); NS_ADDREF(*aRange); } // Setup for next cell mSelectedCellIndex = 1; return res; } NS_IMETHODIMP nsHTMLEditor::GetNextSelectedCell(nsIDOMElement **aCell, nsIDOMRange **aRange) { if (!aCell) return NS_ERROR_NULL_POINTER; *aCell = nsnull; if (aRange) *aRange = nsnull; nsCOMPtr selection; nsresult res = GetSelection(getter_AddRefs(selection)); if (NS_FAILED(res)) return res; if (!selection) return NS_ERROR_FAILURE; PRInt32 rangeCount; res = selection->GetRangeCount(&rangeCount); if (NS_FAILED(res)) return res; // Don't even try if index exceeds range count if (mSelectedCellIndex >= rangeCount) return NS_EDITOR_ELEMENT_NOT_FOUND; // Scan through ranges to find next valid selected cell nsCOMPtr range; for (; mSelectedCellIndex < rangeCount; mSelectedCellIndex++) { res = selection->GetRangeAt(mSelectedCellIndex, getter_AddRefs(range)); if (NS_FAILED(res)) return res; if (!range) return NS_ERROR_FAILURE; res = GetCellFromRange(range, aCell); // Failure here means the range doesn't contain a cell if (NS_FAILED(res)) return NS_EDITOR_ELEMENT_NOT_FOUND; // We found a selected cell if (*aCell) break; #ifdef DEBUG_cmanske else printf("GetNextSelectedCell: Collapsed range found\n"); #endif // If we didn't find a cell, continue to next range in selection } // No cell means all remaining ranges were collapsed (cells were deleted) if (!*aCell) return NS_EDITOR_ELEMENT_NOT_FOUND; if (aRange) { *aRange = range.get(); NS_ADDREF(*aRange); } // Setup for next cell mSelectedCellIndex++; return res; } NS_IMETHODIMP nsHTMLEditor::GetFirstSelectedCellInTable(nsIDOMElement **aCell, PRInt32 *aRowIndex, PRInt32 *aColIndex) { if (!aCell) return NS_ERROR_NULL_POINTER; *aCell = nsnull; if (aRowIndex) *aRowIndex = 0; if (aColIndex) *aColIndex = 0; nsCOMPtr cell; nsresult res = GetFirstSelectedCell(getter_AddRefs(cell), nsnull); if (NS_FAILED(res)) return res; if (!cell) return NS_EDITOR_ELEMENT_NOT_FOUND; PRInt32 startRowIndex, startColIndex; res = GetCellIndexes(cell, startRowIndex, startColIndex); if(NS_FAILED(res)) return res; // Start with first cell selected nsCOMPtr firstCell = cell; PRInt32 firstRowIndex = startRowIndex; PRInt32 firstColIndex = startColIndex; while (cell) { res = GetNextSelectedCell(getter_AddRefs(cell), nsnull); if (NS_FAILED(res)) return res; if (cell) { res = GetCellIndexes(cell, startRowIndex, startColIndex); if(NS_FAILED(res)) return res; // Find the topmost row if (startRowIndex <= firstRowIndex) { // Then save the left-most cell in that row if (startRowIndex < firstRowIndex || startColIndex < firstColIndex) { firstCell = cell; firstRowIndex = startRowIndex; firstColIndex = startColIndex; } } } } if (NS_SUCCEEDED(res)) { *aCell = firstCell.get(); NS_ADDREF(*aCell); if (aRowIndex) *aRowIndex = firstRowIndex; if (aColIndex) *aColIndex = firstColIndex; } return res; } NS_IMETHODIMP nsHTMLEditor::SetSelectionAfterTableEdit(nsIDOMElement* aTable, PRInt32 aRow, PRInt32 aCol, PRInt32 aDirection, PRBool aSelected) { nsresult res = NS_ERROR_NOT_INITIALIZED; if (!aTable) return res; nsCOMPtrselection; res = GetSelection(getter_AddRefs(selection)); if (NS_FAILED(res)) return res; if (!selection) { #ifdef DEBUG_cmanske printf("Selection not found after table manipulation!\n"); #endif return NS_ERROR_FAILURE; } nsCOMPtr cell; PRBool done = PR_FALSE; do { res = GetCellAt(aTable, aRow, aCol, getter_AddRefs(cell)); nsCOMPtr cellNode = do_QueryInterface(cell); if (NS_SUCCEEDED(res)) { if (cell) { if (aSelected) { // Reselect the cell return SelectElement(cell); } else { // Set the caret to deepest first child // but don't go into nested tables // TODO: Should we really be placing the caret at the END // of the cell content? return CollapseSelectionToDeepestNonTableFirstChild(selection, cellNode); } } else { // Setup index to find another cell in the // direction requested, but move in // other direction if already at beginning of row or column switch (aDirection) { case ePreviousColumn: if (aCol == 0) { if (aRow > 0) aRow--; else done = PR_TRUE; } else aCol--; break; case ePreviousRow: if (aRow == 0) { if (aCol > 0) aCol--; else done = PR_TRUE; } else aRow--; break; default: done = PR_TRUE; } } } else break; } while (!done); // We didn't find a cell // Set selection to just before the table nsCOMPtr tableParent; PRInt32 tableOffset; res = aTable->GetParentNode(getter_AddRefs(tableParent)); if(NS_SUCCEEDED(res) && tableParent) { if(NS_SUCCEEDED(GetChildOffset(aTable, tableParent, tableOffset))) return selection->Collapse(tableParent, tableOffset); } // Last resort: Set selection to start of doc // (it's very bad to not have a valid selection!) return SetSelectionAtDocumentStart(selection); } NS_IMETHODIMP nsHTMLEditor::GetSelectedOrParentTableElement(nsIDOMElement* &aTableElement, nsString& aTagName, PRInt32 &aSelectedCount) { aTableElement = nsnull; aTagName.SetLength(0); aSelectedCount = 0; nsCOMPtr selection; nsresult res = GetSelection(getter_AddRefs(selection)); if (NS_FAILED(res)) return res; if (!selection) return NS_ERROR_FAILURE; nsAutoString tdName; tdName.AssignWithConversion("td"); // Try to get the first selected cell nsCOMPtr tableOrCellElement; res = GetFirstSelectedCell(getter_AddRefs(tableOrCellElement), nsnull); if (NS_FAILED(res)) return res; if (tableOrCellElement) { // Each cell is in its own selection range, // so count signals multiple-cell selection res = selection->GetRangeCount(&aSelectedCount); if (NS_FAILED(res)) return res; aTagName = tdName; } else { nsAutoString tableName; tableName.AssignWithConversion("table"); nsAutoString trName; trName.AssignWithConversion("tr"); nsCOMPtr anchorNode; res = selection->GetAnchorNode(getter_AddRefs(anchorNode)); if(NS_FAILED(res)) return res; if (!anchorNode) return NS_ERROR_FAILURE; nsCOMPtr selectedNode; // Get child of anchor node, if exists PRBool hasChildren; anchorNode->HasChildNodes(&hasChildren); if (hasChildren) { PRInt32 anchorOffset; res = selection->GetAnchorOffset(&anchorOffset); if (NS_FAILED(res)) return res; selectedNode = GetChildAt(anchorNode, anchorOffset); if (!selectedNode) { selectedNode = anchorNode; // If anchor doesn't have a child, we can't be selecting a table element, // so don't do the following: } else { nsAutoString tag; GetTagString(selectedNode,tag); if (tag == tdName) { tableOrCellElement = do_QueryInterface(selectedNode); aTagName = tdName; // Each cell is in its own selection range, // so count signals multiple-cell selection res = selection->GetRangeCount(&aSelectedCount); if (NS_FAILED(res)) return res; } else if(tag == tableName) { tableOrCellElement = do_QueryInterface(selectedNode); aTagName = tableName; aSelectedCount = 1; } else if(tag == trName) { tableOrCellElement = do_QueryInterface(selectedNode); aTagName = trName; aSelectedCount = 1; } } } if (!tableOrCellElement) { // Didn't find a table element -- find a cell parent res = GetElementOrParentByTagName(tdName, anchorNode, getter_AddRefs(tableOrCellElement)); if(NS_FAILED(res)) return res; if (tableOrCellElement) aTagName = tdName; } } if (tableOrCellElement) { aTableElement = tableOrCellElement.get(); NS_ADDREF(aTableElement); } return res; } static PRBool IndexNotTested(nsVoidArray *aArray, PRInt32 aIndex) { if (aArray) { PRInt32 count = aArray->Count(); for (PRInt32 i = 0; i < count; i++) { if(aIndex == (PRInt32)(aArray->ElementAt(i))) return PR_FALSE; } } return PR_TRUE; } NS_IMETHODIMP nsHTMLEditor::GetSelectedCellsType(nsIDOMElement *aElement, PRUint32 &aSelectionType) { aSelectionType = 0; // Be sure we have a table element // (if aElement is null, this uses selection's anchor node) nsCOMPtr table; nsresult res = GetElementOrParentByTagName(NS_LITERAL_STRING("table"), aElement, getter_AddRefs(table)); if (NS_FAILED(res)) return res; PRInt32 rowCount, colCount; res = GetTableSize(table, rowCount, colCount); if (NS_FAILED(res)) return res; // Traverse all selected cells nsCOMPtr selectedCell; res = GetFirstSelectedCell(getter_AddRefs(selectedCell), nsnull); if (NS_FAILED(res)) return res; if (res == NS_EDITOR_ELEMENT_NOT_FOUND) return NS_OK; // We have at least one selected cell, so set return value aSelectionType = nsISelectionPrivate::TABLESELECTION_CELL; // Store indexes of each row/col to avoid duplication of searches nsVoidArray indexArray; PRBool allCellsInRowAreSelected = PR_FALSE; PRBool allCellsInColAreSelected = PR_FALSE; while (NS_SUCCEEDED(res) && selectedCell) { // Get the cell's location in the cellmap PRInt32 startRowIndex, startColIndex; res = GetCellIndexes(selectedCell, startRowIndex, startColIndex); if(NS_FAILED(res)) return res; if (IndexNotTested(&indexArray, startColIndex)) { indexArray.AppendElement((void*)startColIndex); allCellsInRowAreSelected = AllCellsInRowSelected(table, startRowIndex, colCount); // We're done as soon as we fail for any row if (!allCellsInRowAreSelected) break; } res = GetNextSelectedCell(getter_AddRefs(selectedCell), nsnull); } if (allCellsInRowAreSelected) { aSelectionType = nsISelectionPrivate::TABLESELECTION_ROW; return NS_OK; } // Test for columns // Empty the indexArray indexArray.Clear(); // Start at first cell again res = GetFirstSelectedCell(getter_AddRefs(selectedCell), nsnull); while (NS_SUCCEEDED(res) && selectedCell) { // Get the cell's location in the cellmap PRInt32 startRowIndex, startColIndex; res = GetCellIndexes(selectedCell, startRowIndex, startColIndex); if(NS_FAILED(res)) return res; if (IndexNotTested(&indexArray, startRowIndex)) { indexArray.AppendElement((void*)startColIndex); allCellsInColAreSelected = AllCellsInColumnSelected(table, startColIndex, rowCount); // We're done as soon as we fail for any column if (!allCellsInRowAreSelected) break; } res = GetNextSelectedCell(getter_AddRefs(selectedCell), nsnull); } if (allCellsInColAreSelected) aSelectionType = nsISelectionPrivate::TABLESELECTION_COLUMN; return NS_OK; } PRBool nsHTMLEditor::AllCellsInRowSelected(nsIDOMElement *aTable, PRInt32 aRowIndex, PRInt32 aNumberOfColumns) { if (!aTable) return PR_FALSE; PRInt32 curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; for( PRInt32 col = 0; col < aNumberOfColumns; col += PR_MAX(actualColSpan, 1)) { nsCOMPtr cell; nsresult res = GetCellDataAt(aTable, aRowIndex, col, getter_AddRefs(cell), curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return PR_FALSE; // If no cell, we may have a "ragged" right edge, // so return TRUE only if we already found a cell in the row if (!cell) return (col > 0) ? PR_TRUE : PR_FALSE; // Return as soon as a non-selected cell is found if (!isSelected) return PR_FALSE; NS_ASSERTION((actualColSpan > 0),"ActualColSpan = 0 in AllCellsInRowSelected"); } return PR_TRUE; } PRBool nsHTMLEditor::AllCellsInColumnSelected(nsIDOMElement *aTable, PRInt32 aColIndex, PRInt32 aNumberOfRows) { if (!aTable) return PR_FALSE; PRInt32 curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; for( PRInt32 row = 0; row < aNumberOfRows; row += PR_MAX(actualRowSpan, 1)) { nsCOMPtr cell; nsresult res = GetCellDataAt(aTable, row, aColIndex, getter_AddRefs(cell), curStartRowIndex, curStartColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); if (NS_FAILED(res)) return PR_FALSE; // If no cell, we must have a "ragged" right edge on the last column // so return TRUE only if we already found a cell in the row if (!cell) return (row > 0) ? PR_TRUE : PR_FALSE; // Return as soon as a non-selected cell is found if (!isSelected) return PR_FALSE; } return PR_TRUE; } PRBool nsHTMLEditor::IsEmptyCell(nsIDOMElement *aCell) { nsCOMPtr cellChild; // Check if target only contains empty text node or
nsresult res = aCell->GetFirstChild(getter_AddRefs(cellChild)); if (NS_FAILED(res)) return res; if (cellChild) { nsCOMPtr nextChild; res = cellChild->GetNextSibling(getter_AddRefs(nextChild)); if (!nextChild) { // We insert a single break into a cell by default // to have some place to locate a cursor -- it is dispensable PRBool isEmpty = nsTextEditUtils::IsBreak(cellChild); // Or check if no real content if (!isEmpty) IsEmptyNode(cellChild, &isEmpty, PR_FALSE, PR_FALSE); return isEmpty; } } return PR_FALSE; }