/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla 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/MPL/ * * 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 the Initial Developer are Copyright (C) 1998 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Daniel Glazman * Mats Palmgren * * Alternatively, the contents of this file may be used under the terms of * either of the GNU General Public License Version 2 or later (the "GPL"), * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include "nsUnicharUtils.h" #include "nsHTMLEditor.h" #include "nsHTMLEditRules.h" #include "nsTextEditUtils.h" #include "nsHTMLEditUtils.h" #include "nsIDOMNodeList.h" #include "nsIDOMAttr.h" #include "nsIDOMMouseEvent.h" #include "nsISelection.h" #include "nsISelectionPrivate.h" #include "nsIDOMHTMLImageElement.h" #include "nsISelectionController.h" #include "nsIDocumentObserver.h" #include "TypeInState.h" #include "nsIEnumerator.h" #include "nsIContent.h" #include "nsIContentIterator.h" #include "nsAttrName.h" #include "mozilla/dom/Element.h" using namespace mozilla; NS_IMETHODIMP nsHTMLEditor::AddDefaultProperty(nsIAtom *aProperty, const nsAString & aAttribute, const nsAString & aValue) { nsString outValue; PRInt32 index; nsString attr(aAttribute); if (TypeInState::FindPropInList(aProperty, attr, &outValue, mDefaultStyles, index)) { PropItem *item = mDefaultStyles[index]; item->value = aValue; } else { nsString value(aValue); PropItem *propItem = new PropItem(aProperty, attr, value); mDefaultStyles.AppendElement(propItem); } return NS_OK; } NS_IMETHODIMP nsHTMLEditor::RemoveDefaultProperty(nsIAtom *aProperty, const nsAString & aAttribute, const nsAString & aValue) { nsString outValue; PRInt32 index; nsString attr(aAttribute); if (TypeInState::FindPropInList(aProperty, attr, &outValue, mDefaultStyles, index)) { delete mDefaultStyles[index]; mDefaultStyles.RemoveElementAt(index); } return NS_OK; } NS_IMETHODIMP nsHTMLEditor::RemoveAllDefaultProperties() { PRUint32 j, defcon = mDefaultStyles.Length(); for (j=0; j selection; nsresult res = GetSelection(getter_AddRefs(selection)); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(selection, NS_ERROR_NULL_POINTER); nsCOMPtr selPriv(do_QueryInterface(selection)); bool isCollapsed; selection->GetIsCollapsed(&isCollapsed); if (isCollapsed) { // manipulating text attributes on a collapsed selection only sets state // for the next text insertion nsString tAttr(aAttribute);//MJUDGE SCC NEED HELP nsString tVal(aValue);//MJUDGE SCC NEED HELP return mTypeInState->SetProp(aProperty, tAttr, tVal); } nsAutoEditBatch batchIt(this); nsAutoRules beginRulesSniffing(this, kOpInsertElement, nsIEditor::eNext); nsAutoSelectionReset selectionResetter(selection, this); nsAutoTxnsConserveSelection dontSpazMySelection(this); bool cancel, handled; nsTextRulesInfo ruleInfo(nsTextEditRules::kSetTextProperty); res = mRules->WillDoAction(selection, &ruleInfo, &cancel, &handled); NS_ENSURE_SUCCESS(res, res); if (!cancel && !handled) { // get selection range enumerator nsCOMPtr enumerator; res = selPriv->GetEnumerator(getter_AddRefs(enumerator)); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(enumerator, NS_ERROR_FAILURE); // loop thru the ranges in the selection nsCOMPtr currentItem; for (enumerator->First(); NS_ENUMERATOR_FALSE == enumerator->IsDone(); enumerator->Next()) { res = enumerator->CurrentItem(getter_AddRefs(currentItem)); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(currentItem, NS_ERROR_FAILURE); nsCOMPtr range(do_QueryInterface(currentItem)); // adjust range to include any ancestors whose children are entirely // selected res = PromoteInlineRange(range); NS_ENSURE_SUCCESS(res, res); // check for easy case: both range endpoints in same text node nsCOMPtr startNode, endNode; res = range->GetStartContainer(getter_AddRefs(startNode)); NS_ENSURE_SUCCESS(res, res); res = range->GetEndContainer(getter_AddRefs(endNode)); NS_ENSURE_SUCCESS(res, res); if (startNode == endNode && IsTextNode(startNode)) { PRInt32 startOffset, endOffset; range->GetStartOffset(&startOffset); range->GetEndOffset(&endOffset); nsCOMPtr nodeAsText = do_QueryInterface(startNode); res = SetInlinePropertyOnTextNode(nodeAsText, startOffset, endOffset, aProperty, &aAttribute, &aValue); NS_ENSURE_SUCCESS(res, res); continue; } // Not the easy case. Range not contained in single text node. There // are up to three phases here. There are all the nodes reported by the // subtree iterator to be processed. And there are potentially a // starting textnode and an ending textnode which are only partially // contained by the range. // Let's handle the nodes reported by the iterator. These nodes are // entirely contained in the selection range. We build up a list of them // (since doing operations on the document during iteration would perturb // the iterator). nsCOMPtr iter = do_CreateInstance("@mozilla.org/content/subtree-content-iterator;1", &res); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(iter, NS_ERROR_FAILURE); nsCOMArray arrayOfNodes; // iterate range and build up array res = iter->Init(range); // Init returns an error if there are no nodes in range. This can easily // happen with the subtree iterator if the selection doesn't contain any // *whole* nodes. if (NS_SUCCEEDED(res)) { nsCOMPtr node; for (; !iter->IsDone(); iter->Next()) { node = do_QueryInterface(iter->GetCurrentNode()); NS_ENSURE_TRUE(node, NS_ERROR_FAILURE); if (IsEditable(node)) { arrayOfNodes.AppendObject(node); } } } // first check the start parent of the range to see if it needs to // be separately handled (it does if it's a text node, due to how the // subtree iterator works - it will not have reported it). if (IsTextNode(startNode) && IsEditable(startNode)) { nsCOMPtr nodeAsText = do_QueryInterface(startNode); PRInt32 startOffset; PRUint32 textLen; range->GetStartOffset(&startOffset); nodeAsText->GetLength(&textLen); res = SetInlinePropertyOnTextNode(nodeAsText, startOffset, textLen, aProperty, &aAttribute, &aValue); NS_ENSURE_SUCCESS(res, res); } // then loop through the list, set the property on each node PRInt32 listCount = arrayOfNodes.Count(); PRInt32 j; for (j = 0; j < listCount; j++) { res = SetInlinePropertyOnNode(arrayOfNodes[j], aProperty, &aAttribute, &aValue); NS_ENSURE_SUCCESS(res, res); } // last check the end parent of the range to see if it needs to // be separately handled (it does if it's a text node, due to how the // subtree iterator works - it will not have reported it). if (IsTextNode(endNode) && IsEditable(endNode)) { nsCOMPtr nodeAsText = do_QueryInterface(endNode); PRInt32 endOffset; range->GetEndOffset(&endOffset); res = SetInlinePropertyOnTextNode(nodeAsText, 0, endOffset, aProperty, &aAttribute, &aValue); NS_ENSURE_SUCCESS(res, res); } } } if (!cancel) { // post-process return mRules->DidDoAction(selection, &ruleInfo, res); } return NS_OK; } nsresult nsHTMLEditor::SetInlinePropertyOnTextNode( nsIDOMCharacterData *aTextNode, PRInt32 aStartOffset, PRInt32 aEndOffset, nsIAtom *aProperty, const nsAString *aAttribute, const nsAString *aValue) { NS_ENSURE_TRUE(aTextNode, NS_ERROR_NULL_POINTER); nsCOMPtr parent; nsresult res = aTextNode->GetParentNode(getter_AddRefs(parent)); NS_ENSURE_SUCCESS(res, res); if (!CanContainTag(parent, aProperty)) { return NS_OK; } // don't need to do anything if no characters actually selected if (aStartOffset == aEndOffset) return NS_OK; nsCOMPtr node = do_QueryInterface(aTextNode); // don't need to do anything if property already set on node bool bHasProp; if (IsCSSEnabled() && mHTMLCSSUtils->IsCSSEditableProperty(node, aProperty, aAttribute, aValue)) { // the HTML styles defined by aProperty/aAttribute has a CSS equivalence // in this implementation for node; let's check if it carries those css styles nsAutoString value; if (aValue) value.Assign(*aValue); mHTMLCSSUtils->IsCSSEquivalentToHTMLInlineStyleSet(node, aProperty, aAttribute, bHasProp, value, COMPUTED_STYLE_TYPE); } else { IsTextPropertySetByContent(node, aProperty, aAttribute, aValue, bHasProp); } if (bHasProp) return NS_OK; // do we need to split the text node? PRUint32 textLen; aTextNode->GetLength(&textLen); nsCOMPtr tmp; if ( (PRUint32)aEndOffset != textLen ) { // we need to split off back of text node res = SplitNode(node, aEndOffset, getter_AddRefs(tmp)); NS_ENSURE_SUCCESS(res, res); node = tmp; // remember left node } if ( aStartOffset ) { // we need to split off front of text node res = SplitNode(node, aStartOffset, getter_AddRefs(tmp)); NS_ENSURE_SUCCESS(res, res); } // look for siblings that are correct type of node nsCOMPtr sibling; GetPriorHTMLSibling(node, address_of(sibling)); if (sibling && NodeIsType(sibling, aProperty) && HasAttrVal(sibling, aAttribute, aValue) && IsOnlyAttribute(sibling, aAttribute) ) { // previous sib is already right kind of inline node; slide this over into it res = MoveNode(node, sibling, -1); return res; } sibling = nsnull; GetNextHTMLSibling(node, address_of(sibling)); if (sibling && NodeIsType(sibling, aProperty) && HasAttrVal(sibling, aAttribute, aValue) && IsOnlyAttribute(sibling, aAttribute) ) { // following sib is already right kind of inline node; slide this over into it res = MoveNode(node, sibling, 0); return res; } // reparent the node inside inline node with appropriate {attribute,value} return SetInlinePropertyOnNode(node, aProperty, aAttribute, aValue); } nsresult nsHTMLEditor::SetInlinePropertyOnNodeImpl(nsIDOMNode *aNode, nsIAtom *aProperty, const nsAString *aAttribute, const nsAString *aValue) { MOZ_ASSERT(aNode && aProperty); nsresult res; nsCOMPtr tmp; nsAutoString tag; aProperty->ToString(tag); ToLowerCase(tag); // If this is an element that can't be contained in a span, we have to // recurse to its children. if (!TagCanContain(nsGkAtoms::span, aNode)) { nsCOMPtr childNodes; res = aNode->GetChildNodes(getter_AddRefs(childNodes)); NS_ENSURE_SUCCESS(res, res); if (childNodes) { PRInt32 j; PRUint32 childCount; childNodes->GetLength(&childCount); if (childCount) { nsCOMArray arrayOfNodes; // populate the list for (j = 0; j < (PRInt32)childCount; j++) { nsCOMPtr childNode; res = childNodes->Item(j, getter_AddRefs(childNode)); if (NS_SUCCEEDED(res) && childNode && IsEditable(childNode)) { arrayOfNodes.AppendObject(childNode); } } // then loop through the list, set the property on each node PRInt32 listCount = arrayOfNodes.Count(); for (j = 0; j < listCount; j++) { res = SetInlinePropertyOnNode(arrayOfNodes[j], aProperty, aAttribute, aValue); NS_ENSURE_SUCCESS(res, res); } } } return NS_OK; } bool useCSS = (IsCSSEnabled() && mHTMLCSSUtils->IsCSSEditableProperty(aNode, aProperty, aAttribute, aValue)) || // bgcolor is always done using CSS aAttribute->EqualsLiteral("bgcolor"); if (useCSS) { tmp = aNode; // We only add style="" to s with no attributes (bug 746515). If we // don't have one, we need to make one. nsCOMPtr element = do_QueryInterface(tmp); if (!element || !element->IsHTML(nsGkAtoms::span) || element->GetAttrCount()) { res = InsertContainerAbove(aNode, address_of(tmp), NS_LITERAL_STRING("span"), nsnull, nsnull); NS_ENSURE_SUCCESS(res, res); } // Add the CSS styles corresponding to the HTML style request PRInt32 count; res = mHTMLCSSUtils->SetCSSEquivalentToHTMLStyle(tmp, aProperty, aAttribute, aValue, &count, false); NS_ENSURE_SUCCESS(res, res); nsCOMPtr nextSibling, previousSibling; GetNextHTMLSibling(tmp, address_of(nextSibling)); GetPriorHTMLSibling(tmp, address_of(previousSibling)); if (nextSibling || previousSibling) { nsCOMPtr mergeParent; res = tmp->GetParentNode(getter_AddRefs(mergeParent)); NS_ENSURE_SUCCESS(res, res); if (previousSibling && nsEditor::NodeIsType(previousSibling, nsEditProperty::span) && NodesSameType(tmp, previousSibling)) { res = JoinNodes(previousSibling, tmp, mergeParent); NS_ENSURE_SUCCESS(res, res); } if (nextSibling && nsEditor::NodeIsType(nextSibling, nsEditProperty::span) && NodesSameType(tmp, nextSibling)) { res = JoinNodes(tmp, nextSibling, mergeParent); NS_ENSURE_SUCCESS(res, res); } } return NS_OK; } // is it already the right kind of node, but with wrong attribute? if (NodeIsType(aNode, aProperty)) { // Just set the attribute on it. nsCOMPtr elem = do_QueryInterface(aNode); return SetAttribute(elem, *aAttribute, *aValue); } // Either put it inside a neighboring node, or make a new one. nsCOMPtr priorNode, nextNode; // is either of its neighbors the right kind of node? GetPriorHTMLSibling(aNode, address_of(priorNode)); GetNextHTMLSibling(aNode, address_of(nextNode)); if (priorNode && NodeIsType(priorNode, aProperty) && HasAttrVal(priorNode, aAttribute, aValue) && IsOnlyAttribute(priorNode, aAttribute)) { // previous sib is already right kind of inline node; slide this over into it return MoveNode(aNode, priorNode, -1); } if (nextNode && NodeIsType(nextNode, aProperty) && HasAttrVal(nextNode, aAttribute, aValue) && IsOnlyAttribute(priorNode, aAttribute)) { // following sib is already right kind of inline node; slide this over into it return MoveNode(aNode, nextNode, 0); } // ok, chuck it in its very own container return InsertContainerAbove(aNode, address_of(tmp), tag, aAttribute, aValue); } nsresult nsHTMLEditor::SetInlinePropertyOnNode(nsIDOMNode *aNode, nsIAtom *aProperty, const nsAString *aAttribute, const nsAString *aValue) { // Before setting the property, we remove it if it's already set. // RemoveStyleInside might remove the node we're looking at or some of its // descendants, however, in which case we want to set the property on // whatever wound up in its place. We have to save the original siblings and // parent to figure this out. NS_ENSURE_TRUE(aNode && aProperty, NS_ERROR_NULL_POINTER); nsCOMPtr node = do_QueryInterface(aNode); NS_ENSURE_STATE(node); nsCOMPtr previousSibling = node->GetPreviousSibling(), nextSibling = node->GetNextSibling(); nsCOMPtr parent = node->GetNodeParent(); NS_ENSURE_STATE(parent); nsresult res = RemoveStyleInside(aNode, aProperty, aAttribute); NS_ENSURE_SUCCESS(res, res); if (node->GetNodeParent()) { // The node is still where it was return SetInlinePropertyOnNodeImpl(aNode, aProperty, aAttribute, aValue); } // It's vanished. Use the old siblings for reference to construct a // list. But first, verify that the previous/next siblings are still // where we expect them; otherwise we have to give up. if ((previousSibling && previousSibling->GetNodeParent() != parent) || (nextSibling && nextSibling->GetNodeParent() != parent)) { return NS_ERROR_UNEXPECTED; } nsCOMArray nodesToSet; nsCOMPtr cur = previousSibling ? previousSibling->GetNextSibling() : parent->GetFirstChild(); while (cur && cur != nextSibling) { if (IsEditable(cur)) { nodesToSet.AppendObject(cur); } cur = cur->GetNextSibling(); } PRInt32 nodesToSetCount = nodesToSet.Count(); for (PRInt32 k = 0; k < nodesToSetCount; k++) { nsCOMPtr nodeToSet = do_QueryInterface(nodesToSet[k]); res = SetInlinePropertyOnNodeImpl(nodeToSet, aProperty, aAttribute, aValue); NS_ENSURE_SUCCESS(res, res); } return NS_OK; } nsresult nsHTMLEditor::SplitStyleAboveRange(nsIDOMRange *inRange, nsIAtom *aProperty, const nsAString *aAttribute) { NS_ENSURE_TRUE(inRange, NS_ERROR_NULL_POINTER); nsresult res; nsCOMPtr startNode, endNode, origStartNode; PRInt32 startOffset, endOffset; res = inRange->GetStartContainer(getter_AddRefs(startNode)); NS_ENSURE_SUCCESS(res, res); res = inRange->GetStartOffset(&startOffset); NS_ENSURE_SUCCESS(res, res); res = inRange->GetEndContainer(getter_AddRefs(endNode)); NS_ENSURE_SUCCESS(res, res); res = inRange->GetEndOffset(&endOffset); NS_ENSURE_SUCCESS(res, res); origStartNode = startNode; // split any matching style nodes above the start of range { nsAutoTrackDOMPoint tracker(mRangeUpdater, address_of(endNode), &endOffset); res = SplitStyleAbovePoint(address_of(startNode), &startOffset, aProperty, aAttribute); NS_ENSURE_SUCCESS(res, res); } // second verse, same as the first... res = SplitStyleAbovePoint(address_of(endNode), &endOffset, aProperty, aAttribute); NS_ENSURE_SUCCESS(res, res); // reset the range res = inRange->SetStart(startNode, startOffset); NS_ENSURE_SUCCESS(res, res); res = inRange->SetEnd(endNode, endOffset); return res; } nsresult nsHTMLEditor::SplitStyleAbovePoint(nsCOMPtr *aNode, PRInt32 *aOffset, nsIAtom *aProperty, // null here means we split all properties const nsAString *aAttribute, nsCOMPtr *outLeftNode, nsCOMPtr *outRightNode) { NS_ENSURE_TRUE(aNode && *aNode && aOffset, NS_ERROR_NULL_POINTER); if (outLeftNode) *outLeftNode = nsnull; if (outRightNode) *outRightNode = nsnull; // split any matching style nodes above the node/offset nsCOMPtr parent, tmp = *aNode; PRInt32 offset; bool useCSS = IsCSSEnabled(); bool isSet; while (tmp && !IsBlockNode(tmp)) { isSet = false; if (useCSS && mHTMLCSSUtils->IsCSSEditableProperty(tmp, aProperty, aAttribute)) { // the HTML style defined by aProperty/aAttribute has a CSS equivalence // in this implementation for the node tmp; let's check if it carries those css styles nsAutoString firstValue; mHTMLCSSUtils->IsCSSEquivalentToHTMLInlineStyleSet(tmp, aProperty, aAttribute, isSet, firstValue, SPECIFIED_STYLE_TYPE); } if ( (aProperty && NodeIsType(tmp, aProperty)) || // node is the correct inline prop (aProperty == nsEditProperty::href && nsHTMLEditUtils::IsLink(tmp)) || // node is href - test if really GetParentNode(getter_AddRefs(*aNode)); *aOffset = offset; } tmp->GetParentNode(getter_AddRefs(parent)); tmp = parent; } return NS_OK; } bool nsHTMLEditor::NodeIsProperty(nsIDOMNode *aNode) { NS_ENSURE_TRUE(aNode, false); if (!IsContainer(aNode)) return false; if (!IsEditable(aNode)) return false; if (IsBlockNode(aNode)) return false; if (NodeIsType(aNode, nsEditProperty::a)) return false; return true; } nsresult nsHTMLEditor::ApplyDefaultProperties() { nsresult res = NS_OK; PRUint32 j, defcon = mDefaultStyles.Length(); for (j=0; jtag, propItem->attr, propItem->value); NS_ENSURE_SUCCESS(res, res); } return res; } nsresult nsHTMLEditor::RemoveStyleInside(nsIDOMNode *aNode, // null here means remove all properties nsIAtom *aProperty, const nsAString *aAttribute, const bool aChildrenOnly) { NS_ENSURE_TRUE(aNode, NS_ERROR_NULL_POINTER); if (IsTextNode(aNode)) { return NS_OK; } nsresult res; // first process the children nsCOMPtr child, tmp; aNode->GetFirstChild(getter_AddRefs(child)); while (child) { // cache next sibling since we might remove child child->GetNextSibling(getter_AddRefs(tmp)); res = RemoveStyleInside(child, aProperty, aAttribute); NS_ENSURE_SUCCESS(res, res); child = tmp; } // then process the node itself if ( (!aChildrenOnly && ( // node is prop we asked for (aProperty && NodeIsType(aNode, aProperty)) || // but check for link (IsEmpty()) { NS_NAMED_LITERAL_STRING(styleAttr, "style"); NS_NAMED_LITERAL_STRING(classAttr, "class"); bool hasStyleAttr = HasAttr(aNode, &styleAttr); bool hasClassAttr = HasAttr(aNode, &classAttr); if (aProperty && (hasStyleAttr || hasClassAttr)) { // aNode carries inline styles or a class attribute so we can't // just remove the element... We need to create above the element // a span that will carry those styles or class, then we can delete // the node. nsCOMPtr spanNode; res = InsertContainerAbove(aNode, address_of(spanNode), NS_LITERAL_STRING("span")); NS_ENSURE_SUCCESS(res, res); res = CloneAttribute(styleAttr, spanNode, aNode); NS_ENSURE_SUCCESS(res, res); res = CloneAttribute(classAttr, spanNode, aNode); NS_ENSURE_SUCCESS(res, res); } res = RemoveContainer(aNode); NS_ENSURE_SUCCESS(res, res); } else { // otherwise we just want to eliminate the attribute if (HasAttr(aNode, aAttribute)) { // if this matching attribute is the ONLY one on the node, // then remove the whole node. Otherwise just nix the attribute. if (IsOnlyAttribute(aNode, aAttribute)) { res = RemoveContainer(aNode); } else { nsCOMPtr elem = do_QueryInterface(aNode); NS_ENSURE_TRUE(elem, NS_ERROR_NULL_POINTER); res = RemoveAttribute(elem, *aAttribute); } NS_ENSURE_SUCCESS(res, res); } } } if (!aChildrenOnly && mHTMLCSSUtils->IsCSSEditableProperty(aNode, aProperty, aAttribute)) { // the HTML style defined by aProperty/aAttribute has a CSS equivalence in // this implementation for the node aNode; let's check if it carries those // css styles nsAutoString propertyValue; bool isSet; mHTMLCSSUtils->IsCSSEquivalentToHTMLInlineStyleSet(aNode, aProperty, aAttribute, isSet, propertyValue, SPECIFIED_STYLE_TYPE); if (isSet) { // yes, tmp has the corresponding css declarations in its style attribute // let's remove them mHTMLCSSUtils->RemoveCSSEquivalentToHTMLStyle(aNode, aProperty, aAttribute, &propertyValue, false); // remove the node if it is a span or font, if its style attribute is // empty or absent, and if it does not have a class nor an id RemoveElementIfNoStyleOrIdOrClass(aNode); } } if (aProperty == nsEditProperty::font && // or node is big or small and we are setting font size (nsHTMLEditUtils::IsBig(aNode) || nsHTMLEditUtils::IsSmall(aNode)) && aAttribute && aAttribute->LowerCaseEqualsLiteral("size")) { return RemoveContainer(aNode); // if we are setting font size, remove any nested bigs and smalls } return NS_OK; } bool nsHTMLEditor::IsOnlyAttribute(nsIDOMNode *aNode, const nsAString *aAttribute) { NS_ENSURE_TRUE(aNode && aAttribute, false); // ooops nsCOMPtr content = do_QueryInterface(aNode); NS_ENSURE_TRUE(content, false); // ooops PRUint32 i, attrCount = content->GetAttrCount(); for (i = 0; i < attrCount; ++i) { nsAutoString attrString; const nsAttrName* name = content->GetAttrNameAt(i); if (!name->NamespaceEquals(kNameSpaceID_None)) { return false; } name->LocalName()->ToString(attrString); // if it's the attribute we know about, or a special _moz attribute, // keep looking if (!attrString.Equals(*aAttribute, nsCaseInsensitiveStringComparator()) && !StringBeginsWith(attrString, NS_LITERAL_STRING("_moz"))) { return false; } } // if we made it through all of them without finding a real attribute // other than aAttribute, then return true return true; } bool nsHTMLEditor::HasAttr(nsIDOMNode* aNode, const nsAString* aAttribute) { NS_ENSURE_TRUE(aNode, false); if (!aAttribute || aAttribute->IsEmpty()) { // everybody has the 'null' attribute return true; } // get element nsCOMPtr element = do_QueryInterface(aNode); NS_ENSURE_TRUE(element, false); nsCOMPtr atom = do_GetAtom(*aAttribute); NS_ENSURE_TRUE(atom, false); return element->HasAttr(kNameSpaceID_None, atom); } bool nsHTMLEditor::HasAttrVal(nsIDOMNode* aNode, const nsAString* aAttribute, const nsAString* aValue) { NS_ENSURE_TRUE(aNode, false); if (!aAttribute || aAttribute->IsEmpty()) { // everybody has the 'null' attribute return true; } // get element nsCOMPtr element = do_QueryInterface(aNode); NS_ENSURE_TRUE(element, false); nsCOMPtr atom = do_GetAtom(*aAttribute); NS_ENSURE_TRUE(atom, false); return element->AttrValueIs(kNameSpaceID_None, atom, *aValue, eIgnoreCase); } nsresult nsHTMLEditor::PromoteRangeIfStartsOrEndsInNamedAnchor(nsIDOMRange *inRange) { NS_ENSURE_TRUE(inRange, NS_ERROR_NULL_POINTER); nsresult res; nsCOMPtr startNode, endNode, parent, tmp; PRInt32 startOffset, endOffset, tmpOffset; res = inRange->GetStartContainer(getter_AddRefs(startNode)); NS_ENSURE_SUCCESS(res, res); res = inRange->GetStartOffset(&startOffset); NS_ENSURE_SUCCESS(res, res); res = inRange->GetEndContainer(getter_AddRefs(endNode)); NS_ENSURE_SUCCESS(res, res); res = inRange->GetEndOffset(&endOffset); NS_ENSURE_SUCCESS(res, res); tmp = startNode; while ( tmp && !nsTextEditUtils::IsBody(tmp) && !nsHTMLEditUtils::IsNamedAnchor(tmp)) { res = GetNodeLocation(tmp, address_of(parent), &tmpOffset); NS_ENSURE_SUCCESS(res, res); tmp = parent; } NS_ENSURE_TRUE(tmp, NS_ERROR_NULL_POINTER); if (nsHTMLEditUtils::IsNamedAnchor(tmp)) { res = GetNodeLocation(tmp, address_of(parent), &tmpOffset); NS_ENSURE_SUCCESS(res, res); startNode = parent; startOffset = tmpOffset; } tmp = endNode; while ( tmp && !nsTextEditUtils::IsBody(tmp) && !nsHTMLEditUtils::IsNamedAnchor(tmp)) { res = GetNodeLocation(tmp, address_of(parent), &tmpOffset); NS_ENSURE_SUCCESS(res, res); tmp = parent; } NS_ENSURE_TRUE(tmp, NS_ERROR_NULL_POINTER); if (nsHTMLEditUtils::IsNamedAnchor(tmp)) { res = GetNodeLocation(tmp, address_of(parent), &tmpOffset); NS_ENSURE_SUCCESS(res, res); endNode = parent; endOffset = tmpOffset + 1; } res = inRange->SetStart(startNode, startOffset); NS_ENSURE_SUCCESS(res, res); res = inRange->SetEnd(endNode, endOffset); return res; } nsresult nsHTMLEditor::PromoteInlineRange(nsIDOMRange *inRange) { NS_ENSURE_TRUE(inRange, NS_ERROR_NULL_POINTER); nsresult res; nsCOMPtr startNode, endNode, parent; PRInt32 startOffset, endOffset; res = inRange->GetStartContainer(getter_AddRefs(startNode)); NS_ENSURE_SUCCESS(res, res); res = inRange->GetStartOffset(&startOffset); NS_ENSURE_SUCCESS(res, res); res = inRange->GetEndContainer(getter_AddRefs(endNode)); NS_ENSURE_SUCCESS(res, res); res = inRange->GetEndOffset(&endOffset); NS_ENSURE_SUCCESS(res, res); while ( startNode && !nsTextEditUtils::IsBody(startNode) && IsEditable(startNode) && IsAtFrontOfNode(startNode, startOffset) ) { res = GetNodeLocation(startNode, address_of(parent), &startOffset); NS_ENSURE_SUCCESS(res, res); startNode = parent; } NS_ENSURE_TRUE(startNode, NS_ERROR_NULL_POINTER); while ( endNode && !nsTextEditUtils::IsBody(endNode) && IsEditable(endNode) && IsAtEndOfNode(endNode, endOffset) ) { res = GetNodeLocation(endNode, address_of(parent), &endOffset); NS_ENSURE_SUCCESS(res, res); endNode = parent; endOffset++; // we are AFTER this node } NS_ENSURE_TRUE(endNode, NS_ERROR_NULL_POINTER); res = inRange->SetStart(startNode, startOffset); NS_ENSURE_SUCCESS(res, res); res = inRange->SetEnd(endNode, endOffset); return res; } bool nsHTMLEditor::IsAtFrontOfNode(nsIDOMNode *aNode, PRInt32 aOffset) { NS_ENSURE_TRUE(aNode, false); // oops if (!aOffset) { return true; } if (IsTextNode(aNode)) { return false; } else { nsCOMPtr firstNode; GetFirstEditableChild(aNode, address_of(firstNode)); NS_ENSURE_TRUE(firstNode, true); PRInt32 offset; nsEditor::GetChildOffset(firstNode, aNode, offset); if (offset < aOffset) return false; return true; } } bool nsHTMLEditor::IsAtEndOfNode(nsIDOMNode *aNode, PRInt32 aOffset) { NS_ENSURE_TRUE(aNode, false); // oops PRUint32 len; GetLengthOfDOMNode(aNode, len); if (aOffset == (PRInt32)len) return true; if (IsTextNode(aNode)) { return false; } else { nsCOMPtr lastNode; GetLastEditableChild(aNode, address_of(lastNode)); NS_ENSURE_TRUE(lastNode, true); PRInt32 offset; nsEditor::GetChildOffset(lastNode, aNode, offset); if (offset < aOffset) return true; return false; } } nsresult nsHTMLEditor::GetInlinePropertyBase(nsIAtom *aProperty, const nsAString *aAttribute, const nsAString *aValue, bool *aFirst, bool *aAny, bool *aAll, nsAString *outValue, bool aCheckDefaults) { NS_ENSURE_TRUE(aProperty, NS_ERROR_NULL_POINTER); nsresult result; *aAny = false; *aAll = true; *aFirst = false; bool first = true; nsCOMPtr selection; result = GetSelection(getter_AddRefs(selection)); NS_ENSURE_SUCCESS(result, result); NS_ENSURE_TRUE(selection, NS_ERROR_NULL_POINTER); nsCOMPtr selPriv(do_QueryInterface(selection)); bool isCollapsed; selection->GetIsCollapsed(&isCollapsed); nsCOMPtr collapsedNode; nsCOMPtr enumerator; result = selPriv->GetEnumerator(getter_AddRefs(enumerator)); NS_ENSURE_SUCCESS(result, result); NS_ENSURE_TRUE(enumerator, NS_ERROR_NULL_POINTER); enumerator->First(); nsCOMPtr currentItem; result = enumerator->CurrentItem(getter_AddRefs(currentItem)); // XXX: should be a while loop, to get each separate range // XXX: ERROR_HANDLING can currentItem be null? if (NS_SUCCEEDED(result) && currentItem) { bool firstNodeInRange = true; // for each range, set a flag nsCOMPtr range(do_QueryInterface(currentItem)); if (isCollapsed) { range->GetStartContainer(getter_AddRefs(collapsedNode)); NS_ENSURE_TRUE(collapsedNode, NS_ERROR_FAILURE); bool isSet, theSetting; nsString tOutString; if (aAttribute) { nsString tString(*aAttribute); mTypeInState->GetTypingState(isSet, theSetting, aProperty, tString, &tOutString); if (outValue) { outValue->Assign(tOutString); } } else { mTypeInState->GetTypingState(isSet, theSetting, aProperty); } if (isSet) { *aFirst = *aAny = *aAll = theSetting; return NS_OK; } // Bug 747889: we don't support CSS for fontSize values if ((aProperty != nsEditProperty::font || !aAttribute->EqualsLiteral("size")) && mHTMLCSSUtils->IsCSSEditableProperty(collapsedNode, aProperty, aAttribute)) { mHTMLCSSUtils->IsCSSEquivalentToHTMLInlineStyleSet( collapsedNode, aProperty, aAttribute, isSet, tOutString, COMPUTED_STYLE_TYPE); if (outValue) { outValue->Assign(tOutString); } *aFirst = *aAny = *aAll = isSet; return NS_OK; } IsTextPropertySetByContent(collapsedNode, aProperty, aAttribute, aValue, isSet, outValue); *aFirst = *aAny = *aAll = isSet; if (!isSet && aCheckDefaults) { // style not set, but if it is a default then it will appear if // content is inserted, so we should report it as set (analogous to // TypeInState). PRInt32 index; if (aAttribute && TypeInState::FindPropInList(aProperty, *aAttribute, outValue, mDefaultStyles, index)) { *aFirst = *aAny = *aAll = true; if (outValue) { outValue->Assign(mDefaultStyles[index]->value); } } } return NS_OK; } // non-collapsed selection nsCOMPtr iter = do_CreateInstance("@mozilla.org/content/post-content-iterator;1"); NS_ENSURE_TRUE(iter, NS_ERROR_NULL_POINTER); nsAutoString firstValue, theValue; nsCOMPtr endNode; PRInt32 endOffset; result = range->GetEndContainer(getter_AddRefs(endNode)); NS_ENSURE_SUCCESS(result, result); result = range->GetEndOffset(&endOffset); NS_ENSURE_SUCCESS(result, result); for (iter->Init(range); !iter->IsDone(); iter->Next()) { nsCOMPtr content = do_QueryInterface(iter->GetCurrentNode()); nsCOMPtr node = do_QueryInterface(content); if (node && nsTextEditUtils::IsBody(node)) { break; } nsCOMPtr text; text = do_QueryInterface(content); // just ignore any non-editable nodes if (text && !IsEditable(text)) { continue; } if (text) { if (!isCollapsed && first && firstNodeInRange) { firstNodeInRange = false; PRInt32 startOffset; range->GetStartOffset(&startOffset); PRUint32 count; text->GetLength(&count); if (startOffset == (PRInt32)count) { continue; } } else if (node == endNode && !endOffset) { continue; } } else if (content->IsElement()) { // handle non-text leaf nodes here continue; } if (node) { bool isSet = false; if (first) { if (mHTMLCSSUtils->IsCSSEditableProperty(node, aProperty, aAttribute) && // Bug 747889: we don't support CSS for fontSize values (aProperty != nsEditProperty::font || !aAttribute->EqualsLiteral("size"))) { // the HTML styles defined by aProperty/aAttribute has a CSS // equivalence in this implementation for node; let's check if it // carries those css styles if (aValue) { firstValue.Assign(*aValue); } mHTMLCSSUtils->IsCSSEquivalentToHTMLInlineStyleSet(node, aProperty, aAttribute, isSet, firstValue, COMPUTED_STYLE_TYPE); } else { IsTextPropertySetByContent(node, aProperty, aAttribute, aValue, isSet, &firstValue); } *aFirst = isSet; first = false; if (outValue) { *outValue = firstValue; } } else { if (mHTMLCSSUtils->IsCSSEditableProperty(node, aProperty, aAttribute) && // Bug 747889: we don't support CSS for fontSize values (aProperty != nsEditProperty::font || !aAttribute->EqualsLiteral("size"))) { // the HTML styles defined by aProperty/aAttribute has a CSS equivalence // in this implementation for node; let's check if it carries those css styles if (aValue) { theValue.Assign(*aValue); } mHTMLCSSUtils->IsCSSEquivalentToHTMLInlineStyleSet(node, aProperty, aAttribute, isSet, theValue, COMPUTED_STYLE_TYPE); } else { IsTextPropertySetByContent(node, aProperty, aAttribute, aValue, isSet, &theValue); } if (firstValue != theValue) { *aAll = false; } } if (isSet) { *aAny = true; } else { *aAll = false; } } } } if (!*aAny) { // make sure that if none of the selection is set, we don't report all is // set *aAll = false; } return result; } NS_IMETHODIMP nsHTMLEditor::GetInlineProperty(nsIAtom *aProperty, const nsAString &aAttribute, const nsAString &aValue, bool *aFirst, bool *aAny, bool *aAll) { NS_ENSURE_TRUE(aProperty && aFirst && aAny && aAll, NS_ERROR_NULL_POINTER); const nsAString *att = nsnull; if (!aAttribute.IsEmpty()) att = &aAttribute; const nsAString *val = nsnull; if (!aValue.IsEmpty()) val = &aValue; return GetInlinePropertyBase( aProperty, att, val, aFirst, aAny, aAll, nsnull); } NS_IMETHODIMP nsHTMLEditor::GetInlinePropertyWithAttrValue(nsIAtom *aProperty, const nsAString &aAttribute, const nsAString &aValue, bool *aFirst, bool *aAny, bool *aAll, nsAString &outValue) { NS_ENSURE_TRUE(aProperty && aFirst && aAny && aAll, NS_ERROR_NULL_POINTER); const nsAString *att = nsnull; if (!aAttribute.IsEmpty()) att = &aAttribute; const nsAString *val = nsnull; if (!aValue.IsEmpty()) val = &aValue; return GetInlinePropertyBase( aProperty, att, val, aFirst, aAny, aAll, &outValue); } NS_IMETHODIMP nsHTMLEditor::RemoveAllInlineProperties() { nsAutoEditBatch batchIt(this); nsAutoRules beginRulesSniffing(this, kOpResetTextProperties, nsIEditor::eNext); nsresult res = RemoveInlinePropertyImpl(nsnull, nsnull); NS_ENSURE_SUCCESS(res, res); return ApplyDefaultProperties(); } NS_IMETHODIMP nsHTMLEditor::RemoveInlineProperty(nsIAtom *aProperty, const nsAString &aAttribute) { return RemoveInlinePropertyImpl(aProperty, &aAttribute); } nsresult nsHTMLEditor::RemoveInlinePropertyImpl(nsIAtom *aProperty, const nsAString *aAttribute) { NS_ENSURE_TRUE(mRules, NS_ERROR_NOT_INITIALIZED); ForceCompositionEnd(); nsresult res; nsCOMPtrselection; res = GetSelection(getter_AddRefs(selection)); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(selection, NS_ERROR_NULL_POINTER); nsCOMPtr selPriv(do_QueryInterface(selection)); bool isCollapsed; selection->GetIsCollapsed(&isCollapsed); bool useCSS = IsCSSEnabled(); if (isCollapsed) { // manipulating text attributes on a collapsed selection only sets state for the next text insertion // For links, aProperty uses "href", use "a" instead if (aProperty == nsEditProperty::href || aProperty == nsEditProperty::name) aProperty = nsEditProperty::a; if (aProperty) return mTypeInState->ClearProp(aProperty, nsAutoString(*aAttribute)); else return mTypeInState->ClearAllProps(); } nsAutoEditBatch batchIt(this); nsAutoRules beginRulesSniffing(this, kOpRemoveTextProperty, nsIEditor::eNext); nsAutoSelectionReset selectionResetter(selection, this); nsAutoTxnsConserveSelection dontSpazMySelection(this); bool cancel, handled; nsTextRulesInfo ruleInfo(nsTextEditRules::kRemoveTextProperty); res = mRules->WillDoAction(selection, &ruleInfo, &cancel, &handled); NS_ENSURE_SUCCESS(res, res); if (!cancel && !handled) { // get selection range enumerator nsCOMPtr enumerator; res = selPriv->GetEnumerator(getter_AddRefs(enumerator)); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(enumerator, NS_ERROR_FAILURE); // loop thru the ranges in the selection enumerator->First(); nsCOMPtr currentItem; while ((NS_ENUMERATOR_FALSE == enumerator->IsDone())) { res = enumerator->CurrentItem(getter_AddRefs(currentItem)); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(currentItem, NS_ERROR_FAILURE); nsCOMPtr range( do_QueryInterface(currentItem) ); if (aProperty == nsEditProperty::name) { // promote range if it starts or end in a named anchor and we // want to remove named anchors res = PromoteRangeIfStartsOrEndsInNamedAnchor(range); } else { // adjust range to include any ancestors who's children are entirely selected res = PromoteInlineRange(range); } NS_ENSURE_SUCCESS(res, res); // remove this style from ancestors of our range endpoints, // splitting them as appropriate res = SplitStyleAboveRange(range, aProperty, aAttribute); NS_ENSURE_SUCCESS(res, res); // check for easy case: both range endpoints in same text node nsCOMPtr startNode, endNode; res = range->GetStartContainer(getter_AddRefs(startNode)); NS_ENSURE_SUCCESS(res, res); res = range->GetEndContainer(getter_AddRefs(endNode)); NS_ENSURE_SUCCESS(res, res); if ((startNode == endNode) && IsTextNode(startNode)) { // we're done with this range! if (useCSS && mHTMLCSSUtils->IsCSSEditableProperty(startNode, aProperty, aAttribute)) { // the HTML style defined by aProperty/aAttribute has a CSS equivalence // in this implementation for startNode nsAutoString cssValue; bool isSet = false; mHTMLCSSUtils->IsCSSEquivalentToHTMLInlineStyleSet(startNode, aProperty, aAttribute, isSet , cssValue, COMPUTED_STYLE_TYPE); if (isSet) { // startNode's computed style indicates the CSS equivalence to the HTML style to // remove is applied; but we found no element in the ancestors of startNode // carrying specified styles; assume it comes from a rule and let's try to // insert a span "inverting" the style nsAutoString value; value.AssignLiteral("-moz-editor-invert-value"); PRInt32 startOffset, endOffset; range->GetStartOffset(&startOffset); range->GetEndOffset(&endOffset); nsCOMPtr nodeAsText = do_QueryInterface(startNode); if (mHTMLCSSUtils->IsCSSInvertable(aProperty, aAttribute)) { SetInlinePropertyOnTextNode(nodeAsText, startOffset, endOffset, aProperty, aAttribute, &value); } } } } else { // not the easy case. range not contained in single text node. nsCOMPtr iter = do_CreateInstance("@mozilla.org/content/subtree-content-iterator;1", &res); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(iter, NS_ERROR_FAILURE); nsCOMArray arrayOfNodes; nsCOMPtr node; // iterate range and build up array iter->Init(range); while (!iter->IsDone()) { node = do_QueryInterface(iter->GetCurrentNode()); NS_ENSURE_TRUE(node, NS_ERROR_FAILURE); if (IsEditable(node)) { arrayOfNodes.AppendObject(node); } iter->Next(); } // loop through the list, remove the property on each node PRInt32 listCount = arrayOfNodes.Count(); PRInt32 j; for (j = 0; j < listCount; j++) { node = arrayOfNodes[j]; res = RemoveStyleInside(node, aProperty, aAttribute); NS_ENSURE_SUCCESS(res, res); if (useCSS && mHTMLCSSUtils->IsCSSEditableProperty(node, aProperty, aAttribute)) { // the HTML style defined by aProperty/aAttribute has a CSS equivalence // in this implementation for node nsAutoString cssValue; bool isSet = false; mHTMLCSSUtils->IsCSSEquivalentToHTMLInlineStyleSet(node, aProperty, aAttribute, isSet , cssValue, COMPUTED_STYLE_TYPE); if (isSet) { // startNode's computed style indicates the CSS equivalence to the HTML style to // remove is applied; but we found no element in the ancestors of startNode // carrying specified styles; assume it comes from a rule and let's try to // insert a span "inverting" the style if (mHTMLCSSUtils->IsCSSInvertable(aProperty, aAttribute)) { nsAutoString value; value.AssignLiteral("-moz-editor-invert-value"); SetInlinePropertyOnNode(node, aProperty, aAttribute, &value); } } } } arrayOfNodes.Clear(); } enumerator->Next(); } } if (!cancel) { // post-process res = mRules->DidDoAction(selection, &ruleInfo, res); } return res; } NS_IMETHODIMP nsHTMLEditor::IncreaseFontSize() { return RelativeFontChange(1); } NS_IMETHODIMP nsHTMLEditor::DecreaseFontSize() { return RelativeFontChange(-1); } nsresult nsHTMLEditor::RelativeFontChange( PRInt32 aSizeChange) { // Can only change font size by + or - 1 if ( !( (aSizeChange==1) || (aSizeChange==-1) ) ) return NS_ERROR_ILLEGAL_VALUE; ForceCompositionEnd(); // Get the selection nsCOMPtrselection; nsresult res = GetSelection(getter_AddRefs(selection)); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(selection, NS_ERROR_FAILURE); nsCOMPtr selPriv(do_QueryInterface(selection)); // Is the selection collapsed? bool bCollapsed; res = selection->GetIsCollapsed(&bCollapsed); NS_ENSURE_SUCCESS(res, res); // if it's collapsed set typing state if (bCollapsed) { nsCOMPtr atom; if (aSizeChange==1) atom = nsEditProperty::big; else atom = nsEditProperty::small; // Let's see in what kind of element the selection is PRInt32 offset; nsCOMPtr selectedNode; res = GetStartNodeAndOffset(selection, getter_AddRefs(selectedNode), &offset); if (IsTextNode(selectedNode)) { nsCOMPtr parent; res = selectedNode->GetParentNode(getter_AddRefs(parent)); NS_ENSURE_SUCCESS(res, res); selectedNode = parent; } if (!CanContainTag(selectedNode, atom)) { return NS_OK; } // manipulating text attributes on a collapsed selection only sets state for the next text insertion return mTypeInState->SetProp(atom, EmptyString(), EmptyString()); } // wrap with txn batching, rules sniffing, and selection preservation code nsAutoEditBatch batchIt(this); nsAutoRules beginRulesSniffing(this, kOpSetTextProperty, nsIEditor::eNext); nsAutoSelectionReset selectionResetter(selection, this); nsAutoTxnsConserveSelection dontSpazMySelection(this); // get selection range enumerator nsCOMPtr enumerator; res = selPriv->GetEnumerator(getter_AddRefs(enumerator)); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(enumerator, NS_ERROR_FAILURE); // loop thru the ranges in the selection enumerator->First(); nsCOMPtr currentItem; while ((NS_ENUMERATOR_FALSE == enumerator->IsDone())) { res = enumerator->CurrentItem(getter_AddRefs(currentItem)); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(currentItem, NS_ERROR_FAILURE); nsCOMPtr range( do_QueryInterface(currentItem) ); // adjust range to include any ancestors who's children are entirely selected res = PromoteInlineRange(range); NS_ENSURE_SUCCESS(res, res); // check for easy case: both range endpoints in same text node nsCOMPtr startNode, endNode; res = range->GetStartContainer(getter_AddRefs(startNode)); NS_ENSURE_SUCCESS(res, res); res = range->GetEndContainer(getter_AddRefs(endNode)); NS_ENSURE_SUCCESS(res, res); if ((startNode == endNode) && IsTextNode(startNode)) { PRInt32 startOffset, endOffset; range->GetStartOffset(&startOffset); range->GetEndOffset(&endOffset); nsCOMPtr nodeAsText = do_QueryInterface(startNode); res = RelativeFontChangeOnTextNode(aSizeChange, nodeAsText, startOffset, endOffset); NS_ENSURE_SUCCESS(res, res); } else { // not the easy case. range not contained in single text node. // there are up to three phases here. There are all the nodes // reported by the subtree iterator to be processed. And there // are potentially a starting textnode and an ending textnode // which are only partially contained by the range. // lets handle the nodes reported by the iterator. These nodes // are entirely contained in the selection range. We build up // a list of them (since doing operations on the document during // iteration would perturb the iterator). nsCOMPtr iter = do_CreateInstance("@mozilla.org/content/subtree-content-iterator;1", &res); NS_ENSURE_SUCCESS(res, res); NS_ENSURE_TRUE(iter, NS_ERROR_FAILURE); nsCOMArray arrayOfNodes; nsCOMPtr node; // iterate range and build up array res = iter->Init(range); if (NS_SUCCEEDED(res)) { while (!iter->IsDone()) { node = do_QueryInterface(iter->GetCurrentNode()); NS_ENSURE_TRUE(node, NS_ERROR_FAILURE); if (IsEditable(node)) { arrayOfNodes.AppendObject(node); } iter->Next(); } // now that we have the list, do the font size change on each node PRInt32 listCount = arrayOfNodes.Count(); PRInt32 j; for (j = 0; j < listCount; j++) { node = arrayOfNodes[j]; res = RelativeFontChangeOnNode(aSizeChange, node); NS_ENSURE_SUCCESS(res, res); } arrayOfNodes.Clear(); } // now check the start and end parents of the range to see if they need to // be separately handled (they do if they are text nodes, due to how the // subtree iterator works - it will not have reported them). if (IsTextNode(startNode) && IsEditable(startNode)) { nsCOMPtr nodeAsText = do_QueryInterface(startNode); PRInt32 startOffset; PRUint32 textLen; range->GetStartOffset(&startOffset); nodeAsText->GetLength(&textLen); res = RelativeFontChangeOnTextNode(aSizeChange, nodeAsText, startOffset, textLen); NS_ENSURE_SUCCESS(res, res); } if (IsTextNode(endNode) && IsEditable(endNode)) { nsCOMPtr nodeAsText = do_QueryInterface(endNode); PRInt32 endOffset; range->GetEndOffset(&endOffset); res = RelativeFontChangeOnTextNode(aSizeChange, nodeAsText, 0, endOffset); NS_ENSURE_SUCCESS(res, res); } } enumerator->Next(); } return res; } nsresult nsHTMLEditor::RelativeFontChangeOnTextNode( PRInt32 aSizeChange, nsIDOMCharacterData *aTextNode, PRInt32 aStartOffset, PRInt32 aEndOffset) { // Can only change font size by + or - 1 if ( !( (aSizeChange==1) || (aSizeChange==-1) ) ) return NS_ERROR_ILLEGAL_VALUE; NS_ENSURE_TRUE(aTextNode, NS_ERROR_NULL_POINTER); // don't need to do anything if no characters actually selected if (aStartOffset == aEndOffset) return NS_OK; nsresult res = NS_OK; nsCOMPtr parent; res = aTextNode->GetParentNode(getter_AddRefs(parent)); NS_ENSURE_SUCCESS(res, res); if (!CanContainTag(parent, nsGkAtoms::big)) { return NS_OK; } nsCOMPtr tmp, node = do_QueryInterface(aTextNode); // do we need to split the text node? PRUint32 textLen; aTextNode->GetLength(&textLen); // -1 is a magic value meaning to the end of node if (aEndOffset == -1) aEndOffset = textLen; if ( (PRUint32)aEndOffset != textLen ) { // we need to split off back of text node res = SplitNode(node, aEndOffset, getter_AddRefs(tmp)); NS_ENSURE_SUCCESS(res, res); node = tmp; // remember left node } if ( aStartOffset ) { // we need to split off front of text node res = SplitNode(node, aStartOffset, getter_AddRefs(tmp)); NS_ENSURE_SUCCESS(res, res); } NS_NAMED_LITERAL_STRING(bigSize, "big"); NS_NAMED_LITERAL_STRING(smallSize, "small"); const nsAString& nodeType = (aSizeChange==1) ? static_cast(bigSize) : static_cast(smallSize); // look for siblings that are correct type of node nsCOMPtr sibling; GetPriorHTMLSibling(node, address_of(sibling)); if (sibling && NodeIsType(sibling, (aSizeChange==1) ? nsEditProperty::big : nsEditProperty::small)) { // previous sib is already right kind of inline node; slide this over into it res = MoveNode(node, sibling, -1); return res; } sibling = nsnull; GetNextHTMLSibling(node, address_of(sibling)); if (sibling && NodeIsType(sibling, (aSizeChange==1) ? nsEditProperty::big : nsEditProperty::small)) { // following sib is already right kind of inline node; slide this over into it res = MoveNode(node, sibling, 0); return res; } // else reparent the node inside font node with appropriate relative size res = InsertContainerAbove(node, address_of(tmp), nodeType); return res; } nsresult nsHTMLEditor::RelativeFontChangeHelper( PRInt32 aSizeChange, nsIDOMNode *aNode) { /* This routine looks for all the font nodes in the tree rooted by aNode, including aNode itself, looking for font nodes that have the size attr set. Any such nodes need to have big or small put inside them, since they override any big/small that are above them. */ // Can only change font size by + or - 1 if ( !( (aSizeChange==1) || (aSizeChange==-1) ) ) return NS_ERROR_ILLEGAL_VALUE; NS_ENSURE_TRUE(aNode, NS_ERROR_NULL_POINTER); nsresult res = NS_OK; nsAutoString tag; if (aSizeChange == 1) tag.AssignLiteral("big"); else tag.AssignLiteral("small"); nsCOMPtr childNodes; PRInt32 j; PRUint32 childCount; nsCOMPtr childNode; // if this is a font node with size, put big/small inside it NS_NAMED_LITERAL_STRING(attr, "size"); if (NodeIsType(aNode, nsEditProperty::font) && HasAttr(aNode, &attr)) { // cycle through children and adjust relative font size res = aNode->GetChildNodes(getter_AddRefs(childNodes)); NS_ENSURE_SUCCESS(res, res); if (childNodes) { childNodes->GetLength(&childCount); for (j=childCount-1; j>=0; j--) { res = childNodes->Item(j, getter_AddRefs(childNode)); if ((NS_SUCCEEDED(res)) && (childNode)) { res = RelativeFontChangeOnNode(aSizeChange, childNode); NS_ENSURE_SUCCESS(res, res); } } } } childNodes = nsnull; // now cycle through the children. res = aNode->GetChildNodes(getter_AddRefs(childNodes)); NS_ENSURE_SUCCESS(res, res); if (childNodes) { childNodes->GetLength(&childCount); for (j=childCount-1; j>=0; j--) { res = childNodes->Item(j, getter_AddRefs(childNode)); if ((NS_SUCCEEDED(res)) && (childNode)) { res = RelativeFontChangeHelper(aSizeChange, childNode); NS_ENSURE_SUCCESS(res, res); } } } return res; } nsresult nsHTMLEditor::RelativeFontChangeOnNode( PRInt32 aSizeChange, nsIDOMNode *aNode) { // Can only change font size by + or - 1 if ( !( (aSizeChange==1) || (aSizeChange==-1) ) ) return NS_ERROR_ILLEGAL_VALUE; NS_ENSURE_TRUE(aNode, NS_ERROR_NULL_POINTER); nsresult res = NS_OK; nsCOMPtr tmp; nsIAtom* atom; if (aSizeChange == 1) { atom = nsGkAtoms::big; } else { atom = nsGkAtoms::small; } // is it the opposite of what we want? if ( ((aSizeChange == 1) && nsHTMLEditUtils::IsSmall(aNode)) || ((aSizeChange == -1) && nsHTMLEditUtils::IsBig(aNode)) ) { // first populate any nested font tags that have the size attr set res = RelativeFontChangeHelper(aSizeChange, aNode); NS_ENSURE_SUCCESS(res, res); // in that case, just remove this node and pull up the children res = RemoveContainer(aNode); return res; } // can it be put inside a "big" or "small"? if (TagCanContain(atom, aNode)) { // first populate any nested font tags that have the size attr set res = RelativeFontChangeHelper(aSizeChange, aNode); NS_ENSURE_SUCCESS(res, res); // ok, chuck it in. // first look at siblings of aNode for matching bigs or smalls. // if we find one, move aNode into it. nsCOMPtr sibling; GetPriorHTMLSibling(aNode, address_of(sibling)); if (sibling && nsEditor::NodeIsType(sibling, (aSizeChange==1 ? nsEditProperty::big : nsEditProperty::small))) { // previous sib is already right kind of inline node; slide this over into it res = MoveNode(aNode, sibling, -1); return res; } sibling = nsnull; GetNextHTMLSibling(aNode, address_of(sibling)); if (sibling && nsEditor::NodeIsType(sibling, (aSizeChange==1 ? nsEditProperty::big : nsEditProperty::small))) { // following sib is already right kind of inline node; slide this over into it res = MoveNode(aNode, sibling, 0); return res; } // else insert it above aNode res = InsertContainerAbove(aNode, address_of(tmp), nsAtomString(atom)); return res; } // none of the above? then cycle through the children. // MOOSE: we should group the children together if possible // into a single "big" or "small". For the moment they are // each getting their own. nsCOMPtr childNodes; res = aNode->GetChildNodes(getter_AddRefs(childNodes)); NS_ENSURE_SUCCESS(res, res); if (childNodes) { PRInt32 j; PRUint32 childCount; childNodes->GetLength(&childCount); for (j=childCount-1; j>=0; j--) { nsCOMPtr childNode; res = childNodes->Item(j, getter_AddRefs(childNode)); if ((NS_SUCCEEDED(res)) && (childNode)) { res = RelativeFontChangeOnNode(aSizeChange, childNode); NS_ENSURE_SUCCESS(res, res); } } } return res; } NS_IMETHODIMP nsHTMLEditor::GetFontFaceState(bool *aMixed, nsAString &outFace) { NS_ENSURE_TRUE(aMixed, NS_ERROR_FAILURE); *aMixed = true; outFace.Truncate(); nsresult res; bool first, any, all; NS_NAMED_LITERAL_STRING(attr, "face"); res = GetInlinePropertyBase(nsEditProperty::font, &attr, nsnull, &first, &any, &all, &outFace); NS_ENSURE_SUCCESS(res, res); if (any && !all) return res; // mixed if (all) { *aMixed = false; return res; } // if there is no font face, check for tt res = GetInlinePropertyBase(nsEditProperty::tt, nsnull, nsnull, &first, &any, &all,nsnull); NS_ENSURE_SUCCESS(res, res); if (any && !all) return res; // mixed if (all) { *aMixed = false; nsEditProperty::tt->ToString(outFace); } if (!any) { // there was no font face attrs of any kind. We are in normal font. outFace.Truncate(); *aMixed = false; } return res; } NS_IMETHODIMP nsHTMLEditor::GetFontColorState(bool *aMixed, nsAString &aOutColor) { NS_ENSURE_TRUE(aMixed, NS_ERROR_NULL_POINTER); *aMixed = true; aOutColor.Truncate(); nsresult res; NS_NAMED_LITERAL_STRING(colorStr, "color"); bool first, any, all; res = GetInlinePropertyBase(nsEditProperty::font, &colorStr, nsnull, &first, &any, &all, &aOutColor); NS_ENSURE_SUCCESS(res, res); if (any && !all) return res; // mixed if (all) { *aMixed = false; return res; } if (!any) { // there was no font color attrs of any kind.. aOutColor.Truncate(); *aMixed = false; } return res; } // the return value is true only if the instance of the HTML editor we created // can handle CSS styles (for instance, Composer can, Messenger can't) and if // the CSS preference is checked nsresult nsHTMLEditor::GetIsCSSEnabled(bool *aIsCSSEnabled) { *aIsCSSEnabled = IsCSSEnabled(); return NS_OK; } static bool HasNonEmptyAttribute(dom::Element* aElement, nsIAtom* aName) { MOZ_ASSERT(aElement); nsAutoString value; return aElement->GetAttr(kNameSpaceID_None, aName, value) && !value.IsEmpty(); } bool nsHTMLEditor::HasStyleOrIdOrClass(dom::Element* aElement) { MOZ_ASSERT(aElement); // remove the node if its style attribute is empty or absent, // and if it does not have a class nor an id return HasNonEmptyAttribute(aElement, nsGkAtoms::style) || HasNonEmptyAttribute(aElement, nsGkAtoms::_class) || HasNonEmptyAttribute(aElement, nsGkAtoms::id); } nsresult nsHTMLEditor::RemoveElementIfNoStyleOrIdOrClass(nsIDOMNode* aElement) { nsCOMPtr element = do_QueryInterface(aElement); NS_ENSURE_TRUE(element, NS_ERROR_NULL_POINTER); // early way out if node is not the right kind of element if ((!element->IsHTML(nsGkAtoms::span) && !element->IsHTML(nsGkAtoms::font)) || HasStyleOrIdOrClass(element)) { return NS_OK; } return RemoveContainer(element); }