/* -*- 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): * Blake Ross * * 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 "nsCOMPtr.h" #include "nsTextControlFrame.h" #include "nsIDocument.h" #include "nsIDOMNSHTMLTextAreaElement.h" #include "nsIDOMNSHTMLInputElement.h" #include "nsIFormControl.h" #include "nsIServiceManager.h" #include "nsFrameSelection.h" #include "nsIPlaintextEditor.h" #include "nsEditorCID.h" #include "nsLayoutCID.h" #include "nsIDocumentEncoder.h" #include "nsCaret.h" #include "nsISelectionListener.h" #include "nsISelectionPrivate.h" #include "nsIController.h" #include "nsIControllers.h" #include "nsIControllerContext.h" #include "nsGenericHTMLElement.h" #include "nsIEditorIMESupport.h" #include "nsIPhonetic.h" #include "nsIEditorObserver.h" #include "nsIDOMHTMLTextAreaElement.h" #include "nsINameSpaceManager.h" #include "nsINodeInfo.h" #include "nsIScrollableFrame.h" //to turn off scroll bars #include "nsFormControlFrame.h" //for registering accesskeys #include "nsIDeviceContext.h" // to measure fonts #include "nsIContent.h" #include "nsIAtom.h" #include "nsPresContext.h" #include "nsGkAtoms.h" #include "nsLayoutUtils.h" #include "nsIComponentManager.h" #include "nsIView.h" #include "nsIViewManager.h" #include "nsIDOMHTMLInputElement.h" #include "nsIDOMElement.h" #include "nsIDOMDocument.h" #include "nsIPresShell.h" #include "nsIComponentManager.h" #include "nsBoxLayoutState.h" //for keylistener for "return" check #include "nsIPrivateDOMEvent.h" #include "nsIDOMEventTarget.h" #include "nsIDocument.h" //observe documents to send onchangenotifications #include "nsIStyleSheet.h"//observe documents to send onchangenotifications #include "nsIStyleRule.h"//observe documents to send onchangenotifications #include "nsIDOMEventListener.h"//observe documents to send onchangenotifications #include "nsGUIEvent.h" #include "nsIDOMEventGroup.h" #include "nsIDOM3EventTarget.h" #include "nsIDOMNSEvent.h" #include "nsIDOMNSUIEvent.h" #include "nsIEventStateManager.h" #include "nsIDOMFocusListener.h" //onchange events #include "nsIDOMCharacterData.h" //for selection setting helper func #include "nsIDOMNodeList.h" //for selection setting helper func #include "nsIDOMRange.h" //for selection setting helper func #include "nsPIDOMWindow.h" //needed for notify selection changed to update the menus ect. #ifdef ACCESSIBILITY #include "nsIAccessibilityService.h" #endif #include "nsIServiceManager.h" #include "nsIDOMNode.h" #include "nsITextControlElement.h" #include "nsIEditorObserver.h" #include "nsITransactionManager.h" #include "nsIDOMText.h" //for multiline getselection #include "nsNodeInfoManager.h" #include "nsContentCreatorFunctions.h" #include "nsIDOMKeyListener.h" #include "nsIDOMEventGroup.h" #include "nsIDOM3EventTarget.h" #include "nsINativeKeyBindings.h" #include "nsIJSContextStack.h" #include "nsFocusManager.h" #include "nsTextEditRules.h" #define DEFAULT_COLUMN_WIDTH 20 #include "nsContentCID.h" static NS_DEFINE_IID(kRangeCID, NS_RANGE_CID); static NS_DEFINE_CID(kTextEditorCID, NS_TEXTEDITOR_CID); static NS_DEFINE_CID(kFrameSelectionCID, NS_FRAMESELECTION_CID); static const PRInt32 DEFAULT_COLS = 20; static const PRInt32 DEFAULT_ROWS = 1; static const PRInt32 DEFAULT_ROWS_TEXTAREA = 2; static const PRInt32 DEFAULT_UNDO_CAP = 1000; static nsINativeKeyBindings *sNativeInputBindings = nsnull; static nsINativeKeyBindings *sNativeTextAreaBindings = nsnull; // wrap can be one of these three values. typedef enum { eHTMLTextWrap_Off = 1, // "off" eHTMLTextWrap_Hard = 2, // "hard" eHTMLTextWrap_Soft = 3 // the default } nsHTMLTextWrap; static PRBool GetWrapPropertyEnum(nsIContent* aContent, nsHTMLTextWrap& aWrapProp) { // soft is the default; "physical" defaults to soft as well because all other // browsers treat it that way and there is no real reason to maintain physical // and virtual as separate entities if no one else does. Only hard and off // do anything different. aWrapProp = eHTMLTextWrap_Soft; // the default nsAutoString wrap; if (aContent->IsHTML()) { static nsIContent::AttrValuesArray strings[] = {&nsGkAtoms::HARD, &nsGkAtoms::OFF, nsnull}; switch (aContent->FindAttrValueIn(kNameSpaceID_None, nsGkAtoms::wrap, strings, eIgnoreCase)) { case 0: aWrapProp = eHTMLTextWrap_Hard; break; case 1: aWrapProp = eHTMLTextWrap_Off; break; } return PR_TRUE; } return PR_FALSE; } class nsTextInputListener : public nsISelectionListener, public nsIDOMKeyListener, public nsIEditorObserver, public nsSupportsWeakReference { public: /** the default constructor */ nsTextInputListener(); /** the default destructor. virtual due to the possibility of derivation. */ virtual ~nsTextInputListener(); /** SetEditor gives an address to the editor that will be accessed * @param aEditor the editor this listener calls for editing operations */ void SetFrame(nsTextControlFrame *aFrame){mFrame = aFrame;} NS_DECL_ISUPPORTS NS_DECL_NSISELECTIONLISTENER NS_IMETHOD HandleEvent(nsIDOMEvent* aEvent); // nsIDOMKeyListener NS_IMETHOD KeyDown(nsIDOMEvent *aKeyEvent); NS_IMETHOD KeyPress(nsIDOMEvent *aKeyEvent); NS_IMETHOD KeyUp(nsIDOMEvent *aKeyEvent); NS_DECL_NSIEDITOROBSERVER protected: nsresult UpdateTextInputCommands(const nsAString& commandsToUpdate); NS_HIDDEN_(nsINativeKeyBindings*) GetKeyBindings(); protected: nsTextControlFrame* mFrame; // weak reference PRPackedBool mSelectionWasCollapsed; /** * Whether we had undo items or not the last time we got EditAction() * notification (when this state changes we update undo and redo menus) */ PRPackedBool mHadUndoItems; /** * Whether we had redo items or not the last time we got EditAction() * notification (when this state changes we update undo and redo menus) */ PRPackedBool mHadRedoItems; }; /* * nsTextEditorListener implementation */ nsTextInputListener::nsTextInputListener() : mFrame(nsnull) , mSelectionWasCollapsed(PR_TRUE) , mHadUndoItems(PR_FALSE) , mHadRedoItems(PR_FALSE) { } nsTextInputListener::~nsTextInputListener() { } NS_IMPL_ADDREF(nsTextInputListener) NS_IMPL_RELEASE(nsTextInputListener) NS_INTERFACE_MAP_BEGIN(nsTextInputListener) NS_INTERFACE_MAP_ENTRY(nsISelectionListener) NS_INTERFACE_MAP_ENTRY(nsIEditorObserver) NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference) NS_INTERFACE_MAP_ENTRY(nsIDOMKeyListener) NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsIDOMEventListener, nsIDOMKeyListener) NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIDOMKeyListener) NS_INTERFACE_MAP_END // BEGIN nsIDOMSelectionListener static PRBool IsFocusedContent(nsIContent* aContent) { nsFocusManager* fm = nsFocusManager::GetFocusManager(); return fm && fm->GetFocusedContent() == aContent; } NS_IMETHODIMP nsTextInputListener::NotifySelectionChanged(nsIDOMDocument* aDoc, nsISelection* aSel, PRInt16 aReason) { PRBool collapsed; if (!mFrame || !aDoc || !aSel || NS_FAILED(aSel->GetIsCollapsed(&collapsed))) return NS_OK; // Fire the select event // The specs don't exactly say when we should fire the select event. // IE: Whenever you add/remove a character to/from the selection. Also // each time for select all. Also if you get to the end of the text // field you will get new event for each keypress or a continuous // stream of events if you use the mouse. IE will fire select event // when the selection collapses to nothing if you are holding down // the shift or mouse button. // Mozilla: If we have non-empty selection we will fire a new event for each // keypress (or mouseup) if the selection changed. Mozilla will also // create the event each time select all is called, even if everything // was previously selected, becase technically select all will first collapse // and then extend. Mozilla will never create an event if the selection // collapses to nothing. if (!collapsed && (aReason & (nsISelectionListener::MOUSEUP_REASON | nsISelectionListener::KEYPRESS_REASON | nsISelectionListener::SELECTALL_REASON))) { nsIContent* content = mFrame->GetContent(); if (content) { nsCOMPtr doc = content->GetDocument(); if (doc) { nsCOMPtr presShell = doc->GetPrimaryShell(); if (presShell) { nsEventStatus status = nsEventStatus_eIgnore; nsEvent event(PR_TRUE, NS_FORM_SELECTED); presShell->HandleEventWithTarget(&event, mFrame, content, &status); } } } } // if the collapsed state did not change, don't fire notifications if (collapsed == mSelectionWasCollapsed) return NS_OK; mSelectionWasCollapsed = collapsed; if (!mFrame || !IsFocusedContent(mFrame->GetContent())) return NS_OK; return UpdateTextInputCommands(NS_LITERAL_STRING("select")); } // END nsIDOMSelectionListener // BEGIN nsIDOMKeyListener NS_IMETHODIMP nsTextInputListener::HandleEvent(nsIDOMEvent* aEvent) { return NS_OK; } static void DoCommandCallback(const char *aCommand, void *aData) { nsTextControlFrame *frame = static_cast(aData); nsIContent *content = frame->GetContent(); nsCOMPtr controllers; nsCOMPtr input = do_QueryInterface(content); if (input) { input->GetControllers(getter_AddRefs(controllers)); } else { nsCOMPtr textArea = do_QueryInterface(content); if (textArea) { textArea->GetControllers(getter_AddRefs(controllers)); } } if (!controllers) { NS_WARNING("Could not get controllers"); return; } nsCOMPtr controller; controllers->GetControllerForCommand(aCommand, getter_AddRefs(controller)); if (controller) { controller->DoCommand(aCommand); } } NS_IMETHODIMP nsTextInputListener::KeyDown(nsIDOMEvent *aDOMEvent) { nsCOMPtr keyEvent(do_QueryInterface(aDOMEvent)); NS_ENSURE_TRUE(keyEvent, NS_ERROR_INVALID_ARG); nsNativeKeyEvent nativeEvent; nsINativeKeyBindings *bindings = GetKeyBindings(); if (bindings && nsContentUtils::DOMEventToNativeKeyEvent(keyEvent, &nativeEvent, PR_FALSE)) { if (bindings->KeyDown(nativeEvent, DoCommandCallback, mFrame)) { aDOMEvent->PreventDefault(); } } return NS_OK; } NS_IMETHODIMP nsTextInputListener::KeyPress(nsIDOMEvent *aDOMEvent) { nsCOMPtr keyEvent(do_QueryInterface(aDOMEvent)); NS_ENSURE_TRUE(keyEvent, NS_ERROR_INVALID_ARG); nsNativeKeyEvent nativeEvent; nsINativeKeyBindings *bindings = GetKeyBindings(); if (bindings && nsContentUtils::DOMEventToNativeKeyEvent(keyEvent, &nativeEvent, PR_TRUE)) { if (bindings->KeyPress(nativeEvent, DoCommandCallback, mFrame)) { aDOMEvent->PreventDefault(); } } return NS_OK; } NS_IMETHODIMP nsTextInputListener::KeyUp(nsIDOMEvent *aDOMEvent) { nsCOMPtr keyEvent(do_QueryInterface(aDOMEvent)); NS_ENSURE_TRUE(keyEvent, NS_ERROR_INVALID_ARG); nsNativeKeyEvent nativeEvent; nsINativeKeyBindings *bindings = GetKeyBindings(); if (bindings && nsContentUtils::DOMEventToNativeKeyEvent(keyEvent, &nativeEvent, PR_FALSE)) { if (bindings->KeyUp(nativeEvent, DoCommandCallback, mFrame)) { aDOMEvent->PreventDefault(); } } return NS_OK; } // END nsIDOMKeyListener // BEGIN nsIEditorObserver NS_IMETHODIMP nsTextInputListener::EditAction() { // // Update the undo / redo menus // nsCOMPtr editor; mFrame->GetEditor(getter_AddRefs(editor)); nsCOMPtr manager; editor->GetTransactionManager(getter_AddRefs(manager)); NS_ENSURE_TRUE(manager, NS_ERROR_FAILURE); // Get the number of undo / redo items PRInt32 numUndoItems = 0; PRInt32 numRedoItems = 0; manager->GetNumberOfUndoItems(&numUndoItems); manager->GetNumberOfRedoItems(&numRedoItems); if ((numUndoItems && !mHadUndoItems) || (!numUndoItems && mHadUndoItems) || (numRedoItems && !mHadRedoItems) || (!numRedoItems && mHadRedoItems)) { // Modify the menu if undo or redo items are different UpdateTextInputCommands(NS_LITERAL_STRING("undo")); mHadUndoItems = numUndoItems != 0; mHadRedoItems = numRedoItems != 0; } // Make sure we know we were changed (do NOT set this to false if there are // no undo items; JS could change the value and we'd still need to save it) mFrame->SetValueChanged(PR_TRUE); // Fire input event mFrame->FireOnInput(); return NS_OK; } // END nsIEditorObserver nsresult nsTextInputListener::UpdateTextInputCommands(const nsAString& commandsToUpdate) { NS_ENSURE_STATE(mFrame); nsIContent* content = mFrame->GetContent(); NS_ENSURE_TRUE(content, NS_ERROR_FAILURE); nsCOMPtr doc = content->GetDocument(); NS_ENSURE_TRUE(doc, NS_ERROR_FAILURE); nsPIDOMWindow *domWindow = doc->GetWindow(); NS_ENSURE_TRUE(domWindow, NS_ERROR_FAILURE); return domWindow->UpdateCommands(commandsToUpdate); } nsINativeKeyBindings* nsTextInputListener::GetKeyBindings() { if (mFrame->IsTextArea()) { static PRBool sNoTextAreaBindings = PR_FALSE; if (!sNativeTextAreaBindings && !sNoTextAreaBindings) { CallGetService(NS_NATIVEKEYBINDINGS_CONTRACTID_PREFIX "textarea", &sNativeTextAreaBindings); if (!sNativeTextAreaBindings) { sNoTextAreaBindings = PR_TRUE; } } return sNativeTextAreaBindings; } static PRBool sNoInputBindings = PR_FALSE; if (!sNativeInputBindings && !sNoInputBindings) { CallGetService(NS_NATIVEKEYBINDINGS_CONTRACTID_PREFIX "input", &sNativeInputBindings); if (!sNativeInputBindings) { sNoInputBindings = PR_TRUE; } } return sNativeInputBindings; } // END nsTextInputListener class nsTextInputSelectionImpl : public nsSupportsWeakReference , public nsISelectionController { public: NS_DECL_ISUPPORTS nsTextInputSelectionImpl(nsFrameSelection *aSel, nsIPresShell *aShell, nsIContent *aLimiter); ~nsTextInputSelectionImpl(){} void SetScrollableFrame(nsIScrollableFrame *aScrollableFrame) { mScrollFrame = aScrollableFrame; } //NSISELECTIONCONTROLLER INTERFACES NS_IMETHOD SetDisplaySelection(PRInt16 toggle); NS_IMETHOD GetDisplaySelection(PRInt16 *_retval); NS_IMETHOD SetSelectionFlags(PRInt16 aInEnable); NS_IMETHOD GetSelectionFlags(PRInt16 *aOutEnable); NS_IMETHOD GetSelection(PRInt16 type, nsISelection **_retval); NS_IMETHOD ScrollSelectionIntoView(PRInt16 aType, PRInt16 aRegion, PRBool aIsSynchronous); NS_IMETHOD RepaintSelection(PRInt16 type); NS_IMETHOD RepaintSelection(nsPresContext* aPresContext, SelectionType aSelectionType); NS_IMETHOD SetCaretEnabled(PRBool enabled); NS_IMETHOD SetCaretReadOnly(PRBool aReadOnly); NS_IMETHOD GetCaretEnabled(PRBool *_retval); NS_IMETHOD GetCaretVisible(PRBool *_retval); NS_IMETHOD SetCaretVisibilityDuringSelection(PRBool aVisibility); NS_IMETHOD CharacterMove(PRBool aForward, PRBool aExtend); NS_IMETHOD CharacterExtendForDelete(); NS_IMETHOD WordMove(PRBool aForward, PRBool aExtend); NS_IMETHOD WordExtendForDelete(PRBool aForward); NS_IMETHOD LineMove(PRBool aForward, PRBool aExtend); NS_IMETHOD IntraLineMove(PRBool aForward, PRBool aExtend); NS_IMETHOD PageMove(PRBool aForward, PRBool aExtend); NS_IMETHOD CompleteScroll(PRBool aForward); NS_IMETHOD CompleteMove(PRBool aForward, PRBool aExtend); NS_IMETHOD ScrollPage(PRBool aForward); NS_IMETHOD ScrollLine(PRBool aForward); NS_IMETHOD ScrollHorizontal(PRBool aLeft); NS_IMETHOD SelectAll(void); NS_IMETHOD CheckVisibility(nsIDOMNode *node, PRInt16 startOffset, PRInt16 EndOffset, PRBool *_retval); private: nsCOMPtr mFrameSelection; nsCOMPtr mLimiter; nsIScrollableFrame *mScrollFrame; nsWeakPtr mPresShellWeak; }; // Implement our nsISupports methods NS_IMPL_ISUPPORTS3(nsTextInputSelectionImpl, nsISelectionController, nsISelectionDisplay, nsISupportsWeakReference) // BEGIN nsTextInputSelectionImpl nsTextInputSelectionImpl::nsTextInputSelectionImpl(nsFrameSelection *aSel, nsIPresShell *aShell, nsIContent *aLimiter) : mScrollFrame(nsnull) { if (aSel && aShell) { mFrameSelection = aSel;//we are the owner now! mLimiter = aLimiter; mFrameSelection->Init(aShell, mLimiter); mPresShellWeak = do_GetWeakReference(aShell); } } NS_IMETHODIMP nsTextInputSelectionImpl::SetDisplaySelection(PRInt16 aToggle) { if (!mFrameSelection) return NS_ERROR_NULL_POINTER; mFrameSelection->SetDisplaySelection(aToggle); return NS_OK; } NS_IMETHODIMP nsTextInputSelectionImpl::GetDisplaySelection(PRInt16 *aToggle) { if (!mFrameSelection) return NS_ERROR_NULL_POINTER; *aToggle = mFrameSelection->GetDisplaySelection(); return NS_OK; } NS_IMETHODIMP nsTextInputSelectionImpl::SetSelectionFlags(PRInt16 aToggle) { return NS_OK;//stub this out. not used in input } NS_IMETHODIMP nsTextInputSelectionImpl::GetSelectionFlags(PRInt16 *aOutEnable) { *aOutEnable = nsISelectionDisplay::DISPLAY_TEXT; return NS_OK; } NS_IMETHODIMP nsTextInputSelectionImpl::GetSelection(PRInt16 type, nsISelection **_retval) { if (!mFrameSelection) return NS_ERROR_NULL_POINTER; *_retval = mFrameSelection->GetSelection(type); if (!(*_retval)) return NS_ERROR_FAILURE; NS_ADDREF(*_retval); return NS_OK; } NS_IMETHODIMP nsTextInputSelectionImpl::ScrollSelectionIntoView(PRInt16 aType, PRInt16 aRegion, PRBool aIsSynchronous) { if (!mFrameSelection) return NS_ERROR_FAILURE; return mFrameSelection->ScrollSelectionIntoView(aType, aRegion, aIsSynchronous); } NS_IMETHODIMP nsTextInputSelectionImpl::RepaintSelection(PRInt16 type) { if (!mFrameSelection) return NS_ERROR_FAILURE; return mFrameSelection->RepaintSelection(type); } NS_IMETHODIMP nsTextInputSelectionImpl::RepaintSelection(nsPresContext* aPresContext, SelectionType aSelectionType) { if (!mFrameSelection) return NS_ERROR_FAILURE; return mFrameSelection->RepaintSelection(aSelectionType); } NS_IMETHODIMP nsTextInputSelectionImpl::SetCaretEnabled(PRBool enabled) { if (!mPresShellWeak) return NS_ERROR_NOT_INITIALIZED; nsCOMPtr shell = do_QueryReferent(mPresShellWeak); if (!shell) return NS_ERROR_FAILURE; // tell the pres shell to enable the caret, rather than settings its visibility directly. // this way the presShell's idea of caret visibility is maintained. nsCOMPtr selCon = do_QueryInterface(shell); if (!selCon) return NS_ERROR_NO_INTERFACE; selCon->SetCaretEnabled(enabled); return NS_OK; } NS_IMETHODIMP nsTextInputSelectionImpl::SetCaretReadOnly(PRBool aReadOnly) { if (!mPresShellWeak) return NS_ERROR_NOT_INITIALIZED; nsresult result; nsCOMPtr shell = do_QueryReferent(mPresShellWeak, &result); if (shell) { nsRefPtr caret = shell->GetCaret(); if (caret) { nsISelection* domSel = mFrameSelection-> GetSelection(nsISelectionController::SELECTION_NORMAL); if (domSel) caret->SetCaretReadOnly(aReadOnly); return NS_OK; } } return NS_ERROR_FAILURE; } NS_IMETHODIMP nsTextInputSelectionImpl::GetCaretEnabled(PRBool *_retval) { return GetCaretVisible(_retval); } NS_IMETHODIMP nsTextInputSelectionImpl::GetCaretVisible(PRBool *_retval) { if (!mPresShellWeak) return NS_ERROR_NOT_INITIALIZED; nsresult result; nsCOMPtr shell = do_QueryReferent(mPresShellWeak, &result); if (shell) { nsRefPtr caret = shell->GetCaret(); if (caret) { nsISelection* domSel = mFrameSelection-> GetSelection(nsISelectionController::SELECTION_NORMAL); if (domSel) return caret->GetCaretVisible(_retval); } } return NS_ERROR_FAILURE; } NS_IMETHODIMP nsTextInputSelectionImpl::SetCaretVisibilityDuringSelection(PRBool aVisibility) { if (!mPresShellWeak) return NS_ERROR_NOT_INITIALIZED; nsresult result; nsCOMPtr shell = do_QueryReferent(mPresShellWeak, &result); if (shell) { nsRefPtr caret = shell->GetCaret(); if (caret) { nsISelection* domSel = mFrameSelection-> GetSelection(nsISelectionController::SELECTION_NORMAL); if (domSel) caret->SetVisibilityDuringSelection(aVisibility); return NS_OK; } } return NS_ERROR_FAILURE; } NS_IMETHODIMP nsTextInputSelectionImpl::CharacterMove(PRBool aForward, PRBool aExtend) { if (mFrameSelection) return mFrameSelection->CharacterMove(aForward, aExtend); return NS_ERROR_NULL_POINTER; } NS_IMETHODIMP nsTextInputSelectionImpl::CharacterExtendForDelete() { if (mFrameSelection) return mFrameSelection->CharacterExtendForDelete(); return NS_ERROR_NULL_POINTER; } NS_IMETHODIMP nsTextInputSelectionImpl::WordMove(PRBool aForward, PRBool aExtend) { if (mFrameSelection) return mFrameSelection->WordMove(aForward, aExtend); return NS_ERROR_NULL_POINTER; } NS_IMETHODIMP nsTextInputSelectionImpl::WordExtendForDelete(PRBool aForward) { if (mFrameSelection) return mFrameSelection->WordExtendForDelete(aForward); return NS_ERROR_NULL_POINTER; } NS_IMETHODIMP nsTextInputSelectionImpl::LineMove(PRBool aForward, PRBool aExtend) { if (mFrameSelection) { nsresult result = mFrameSelection->LineMove(aForward, aExtend); if (NS_FAILED(result)) result = CompleteMove(aForward,aExtend); return result; } return NS_ERROR_NULL_POINTER; } NS_IMETHODIMP nsTextInputSelectionImpl::IntraLineMove(PRBool aForward, PRBool aExtend) { if (mFrameSelection) return mFrameSelection->IntraLineMove(aForward, aExtend); return NS_ERROR_NULL_POINTER; } NS_IMETHODIMP nsTextInputSelectionImpl::PageMove(PRBool aForward, PRBool aExtend) { // expected behavior for PageMove is to scroll AND move the caret // and to remain relative position of the caret in view. see Bug 4302. if (mScrollFrame) { mFrameSelection->CommonPageMove(aForward, aExtend, mScrollFrame); } // After ScrollSelectionIntoView(), the pending notifications might be // flushed and PresShell/PresContext/Frames may be dead. See bug 418470. return ScrollSelectionIntoView(nsISelectionController::SELECTION_NORMAL, nsISelectionController::SELECTION_FOCUS_REGION, PR_TRUE); } NS_IMETHODIMP nsTextInputSelectionImpl::CompleteScroll(PRBool aForward) { if (!mScrollFrame) return NS_ERROR_NOT_INITIALIZED; mScrollFrame->ScrollBy(nsIntPoint(0, aForward ? 1 : -1), nsIScrollableFrame::WHOLE, nsIScrollableFrame::INSTANT); return NS_OK; } NS_IMETHODIMP nsTextInputSelectionImpl::CompleteMove(PRBool aForward, PRBool aExtend) { // grab the parent / root DIV for this text widget nsIContent* parentDIV = mFrameSelection->GetLimiter(); if (!parentDIV) return NS_ERROR_UNEXPECTED; // make the caret be either at the very beginning (0) or the very end PRInt32 offset = 0; nsFrameSelection::HINT hint = nsFrameSelection::HINTLEFT; if (aForward) { offset = parentDIV->GetChildCount(); // Prevent the caret from being placed after the last // BR node in the content tree! if (offset > 0) { nsIContent *child = parentDIV->GetChildAt(offset - 1); if (child->Tag() == nsGkAtoms::br) { --offset; hint = nsFrameSelection::HINTRIGHT; // for Bug 106855 } } } mFrameSelection->HandleClick(parentDIV, offset, offset, aExtend, PR_FALSE, hint); // if we got this far, attempt to scroll no matter what the above result is return CompleteScroll(aForward); } NS_IMETHODIMP nsTextInputSelectionImpl::ScrollPage(PRBool aForward) { if (!mScrollFrame) return NS_ERROR_NOT_INITIALIZED; mScrollFrame->ScrollBy(nsIntPoint(0, aForward ? 1 : -1), nsIScrollableFrame::PAGES, nsIScrollableFrame::SMOOTH); return NS_OK; } NS_IMETHODIMP nsTextInputSelectionImpl::ScrollLine(PRBool aForward) { if (!mScrollFrame) return NS_ERROR_NOT_INITIALIZED; mScrollFrame->ScrollBy(nsIntPoint(0, aForward ? 1 : -1), nsIScrollableFrame::LINES, nsIScrollableFrame::SMOOTH); return NS_OK; } NS_IMETHODIMP nsTextInputSelectionImpl::ScrollHorizontal(PRBool aLeft) { if (!mScrollFrame) return NS_ERROR_NOT_INITIALIZED; // will we have bug #7354 because we aren't forcing an update here? mScrollFrame->ScrollBy(nsIntPoint(aLeft ? -1 : 1, 0), nsIScrollableFrame::LINES, nsIScrollableFrame::SMOOTH); return NS_OK; } NS_IMETHODIMP nsTextInputSelectionImpl::SelectAll() { if (mFrameSelection) return mFrameSelection->SelectAll(); return NS_ERROR_NULL_POINTER; } NS_IMETHODIMP nsTextInputSelectionImpl::CheckVisibility(nsIDOMNode *node, PRInt16 startOffset, PRInt16 EndOffset, PRBool *_retval) { if (!mPresShellWeak) return NS_ERROR_NOT_INITIALIZED; nsresult result; nsCOMPtr shell = do_QueryReferent(mPresShellWeak, &result); if (shell) { return shell->CheckVisibility(node,startOffset,EndOffset, _retval); } return NS_ERROR_FAILURE; } nsIFrame* NS_NewTextControlFrame(nsIPresShell* aPresShell, nsStyleContext* aContext) { return new (aPresShell) nsTextControlFrame(aPresShell, aContext); } NS_IMPL_FRAMEARENA_HELPERS(nsTextControlFrame) NS_QUERYFRAME_HEAD(nsTextControlFrame) NS_QUERYFRAME_ENTRY(nsIFormControlFrame) NS_QUERYFRAME_ENTRY(nsIAnonymousContentCreator) NS_QUERYFRAME_ENTRY(nsITextControlFrame) NS_QUERYFRAME_TAIL_INHERITING(nsBoxFrame) #ifdef ACCESSIBILITY NS_IMETHODIMP nsTextControlFrame::GetAccessible(nsIAccessible** aAccessible) { nsCOMPtr accService = do_GetService("@mozilla.org/accessibilityService;1"); if (accService) { return accService->CreateHTMLTextFieldAccessible(static_cast(this), aAccessible); } return NS_ERROR_FAILURE; } #endif #ifdef DEBUG class EditorInitializerEntryTracker { public: explicit EditorInitializerEntryTracker(nsTextControlFrame &frame) : mFrame(frame) , mFirstEntry(PR_FALSE) { if (!mFrame.mInEditorInitialization) { mFrame.mInEditorInitialization = PR_TRUE; mFirstEntry = PR_TRUE; } } ~EditorInitializerEntryTracker() { if (mFirstEntry) { mFrame.mInEditorInitialization = PR_FALSE; } } PRBool EnteredMoreThanOnce() const { return !mFirstEntry; } private: nsTextControlFrame &mFrame; PRBool mFirstEntry; }; #endif nsTextControlFrame::nsTextControlFrame(nsIPresShell* aShell, nsStyleContext* aContext) : nsStackFrame(aShell, aContext) , mUseEditor(PR_FALSE) , mIsProcessing(PR_FALSE) , mNotifyOnInput(PR_TRUE) , mDidPreDestroy(PR_FALSE) , mFireChangeEventState(PR_FALSE) , mInSecureKeyboardInputMode(PR_FALSE) #ifdef DEBUG , mInEditorInitialization(PR_FALSE) #endif , mTextListener(nsnull) { } nsTextControlFrame::~nsTextControlFrame() { NS_IF_RELEASE(mTextListener); } static PRBool SuppressEventHandlers(nsPresContext* aPresContext) { PRBool suppressHandlers = PR_FALSE; if (aPresContext) { // Right now we only suppress event handlers and controller manipulation // when in a print preview or print context! // In the current implementation, we only paginate when // printing or in print preview. suppressHandlers = aPresContext->IsPaginated(); } return suppressHandlers; } void nsTextControlFrame::PreDestroy() { // notify the editor that we are going away if (mEditor) { // If we were in charge of state before, relinquish it back // to the control. if (mUseEditor) { // First get the frame state from the editor nsAutoString value; GetValue(value, PR_TRUE); mUseEditor = PR_FALSE; // Next store the frame state in the control // (now that mUseEditor is false values get stored // in content). SetValue(value); // Reset mUseEditor for now, so that if any of the rest of the operation // leads to an attempt at getting the editor, lazy initialization doesn't // kick in. See bug 557689 for an example of the types of problems this // prevents. mUseEditor = PR_TRUE; } mEditor->PreDestroy(PR_TRUE); } // Clean up the controller if (!SuppressEventHandlers(PresContext())) { nsCOMPtr controllers; nsCOMPtr inputElement = do_QueryInterface(mContent); if (inputElement) inputElement->GetControllers(getter_AddRefs(controllers)); else { nsCOMPtr textAreaElement = do_QueryInterface(mContent); if (textAreaElement) { textAreaElement->GetControllers(getter_AddRefs(controllers)); } } if (controllers) { PRUint32 numControllers; nsresult rv = controllers->GetControllerCount(&numControllers); NS_ASSERTION((NS_SUCCEEDED(rv)), "bad result in gfx text control destructor"); for (PRUint32 i = 0; i < numControllers; i ++) { nsCOMPtr controller; rv = controllers->GetControllerAt(i, getter_AddRefs(controller)); if (NS_SUCCEEDED(rv) && controller) { nsCOMPtr editController = do_QueryInterface(controller); if (editController) { editController->SetCommandContext(nsnull); } } } } } mUseEditor = PR_FALSE; mEditor = nsnull; if (mSelCon) { mSelCon->SetScrollableFrame(nsnull); mSelCon = nsnull; } if (mFrameSel) { mFrameSel->DisconnectFromPresShell(); mFrameSel = nsnull; } nsFormControlFrame::RegUnRegAccessKey(static_cast(this), PR_FALSE); if (mTextListener) { mTextListener->SetFrame(nsnull); nsCOMPtr systemGroup; mContent->GetSystemEventGroup(getter_AddRefs(systemGroup)); nsCOMPtr dom3Targ = do_QueryInterface(mContent); if (dom3Targ) { // cast because of ambiguous base nsIDOMEventListener *listener = static_cast (mTextListener); dom3Targ->RemoveGroupedEventListener(NS_LITERAL_STRING("keydown"), listener, PR_FALSE, systemGroup); dom3Targ->RemoveGroupedEventListener(NS_LITERAL_STRING("keypress"), listener, PR_FALSE, systemGroup); dom3Targ->RemoveGroupedEventListener(NS_LITERAL_STRING("keyup"), listener, PR_FALSE, systemGroup); } } mDidPreDestroy = PR_TRUE; } void nsTextControlFrame::DestroyFrom(nsIFrame* aDestructRoot) { if (mInSecureKeyboardInputMode) { MaybeEndSecureKeyboardInput(); } if (!mDidPreDestroy) { PreDestroy(); } if (mValueDiv && mMutationObserver) { mValueDiv->RemoveMutationObserver(mMutationObserver); } nsContentUtils::DestroyAnonymousContent(&mValueDiv); nsContentUtils::DestroyAnonymousContent(&mPlaceholderDiv); nsBoxFrame::DestroyFrom(aDestructRoot); } nsIAtom* nsTextControlFrame::GetType() const { return nsGkAtoms::textInputFrame; } // XXX: wouldn't it be nice to get this from the style context! PRBool nsTextControlFrame::IsSingleLineTextControl() const { nsCOMPtr formControl = do_QueryInterface(mContent); if (formControl) { PRInt32 type = formControl->GetType(); return (type == NS_FORM_INPUT_TEXT) || (type == NS_FORM_INPUT_PASSWORD); } return PR_FALSE; } PRBool nsTextControlFrame::IsTextArea() const { return mContent && mContent->Tag() == nsGkAtoms::textarea; } // XXX: wouldn't it be nice to get this from the style context! PRBool nsTextControlFrame::IsPlainTextControl() const { // need to check HTML attribute of mContent and/or CSS. return PR_TRUE; } nsresult nsTextControlFrame::MaybeBeginSecureKeyboardInput() { nsresult rv = NS_OK; if (IsPasswordTextControl() && !mInSecureKeyboardInputMode) { nsIWidget* window = GetWindow(); NS_ENSURE_TRUE(window, NS_ERROR_FAILURE); rv = window->BeginSecureKeyboardInput(); mInSecureKeyboardInputMode = NS_SUCCEEDED(rv); } return rv; } void nsTextControlFrame::MaybeEndSecureKeyboardInput() { if (mInSecureKeyboardInputMode) { nsIWidget* window = GetWindow(); if (!window) return; window->EndSecureKeyboardInput(); mInSecureKeyboardInputMode = PR_FALSE; } } PRBool nsTextControlFrame::IsPasswordTextControl() const { nsCOMPtr formControl = do_QueryInterface(mContent); return formControl && formControl->GetType() == NS_FORM_INPUT_PASSWORD; } PRInt32 nsTextControlFrame::GetCols() { nsGenericHTMLElement *content = nsGenericHTMLElement::FromContent(mContent); NS_ASSERTION(content, "Content is not HTML content!"); if (IsTextArea()) { const nsAttrValue* attr = content->GetParsedAttr(nsGkAtoms::cols); if (attr) { PRInt32 cols = attr->Type() == nsAttrValue::eInteger ? attr->GetIntegerValue() : 0; // XXX why a default of 1 char, why hide it return (cols <= 0) ? 1 : cols; } } else { // Else we know (assume) it is an input with size attr const nsAttrValue* attr = content->GetParsedAttr(nsGkAtoms::size); if (attr && attr->Type() == nsAttrValue::eInteger) { PRInt32 cols = attr->GetIntegerValue(); if (cols > 0) { return cols; } } } return DEFAULT_COLS; } PRInt32 nsTextControlFrame::GetRows() { if (IsTextArea()) { nsGenericHTMLElement *content = nsGenericHTMLElement::FromContent(mContent); NS_ASSERTION(content, "Content is not HTML content!"); const nsAttrValue* attr = content->GetParsedAttr(nsGkAtoms::rows); if (attr && attr->Type() == nsAttrValue::eInteger) { PRInt32 rows = attr->GetIntegerValue(); return (rows <= 0) ? DEFAULT_ROWS_TEXTAREA : rows; } return DEFAULT_ROWS_TEXTAREA; } return DEFAULT_ROWS; } nsresult nsTextControlFrame::CalcIntrinsicSize(nsIRenderingContext* aRenderingContext, nsSize& aIntrinsicSize) { // Get leading and the Average/MaxAdvance char width nscoord lineHeight = 0; nscoord charWidth = 0; nscoord charMaxAdvance = 0; nsCOMPtr fontMet; nsresult rv = nsLayoutUtils::GetFontMetricsForFrame(this, getter_AddRefs(fontMet)); NS_ENSURE_SUCCESS(rv, rv); aRenderingContext->SetFont(fontMet); lineHeight = nsHTMLReflowState::CalcLineHeight(GetStyleContext(), NS_AUTOHEIGHT); fontMet->GetAveCharWidth(charWidth); fontMet->GetMaxAdvance(charMaxAdvance); // Set the width equal to the width in characters PRInt32 cols = GetCols(); aIntrinsicSize.width = cols * charWidth; // To better match IE, take the maximum character width(in twips) and remove // 4 pixels add this on as additional padding(internalPadding). But only do // this if charMaxAdvance != charWidth; if they are equal, this is almost // certainly a fixed-width font. if (charWidth != charMaxAdvance) { nscoord internalPadding = NS_MAX(0, charMaxAdvance - nsPresContext::CSSPixelsToAppUnits(4)); nscoord t = nsPresContext::CSSPixelsToAppUnits(1); // Round to a multiple of t nscoord rest = internalPadding % t; if (rest < t - rest) { internalPadding -= rest; } else { internalPadding += t - rest; } // Now add the extra padding on (so that small input sizes work well) aIntrinsicSize.width += internalPadding; } else { // This is to account for the anonymous
having a 1 twip width // in Full Standards mode, see BRFrame::Reflow and bug 228752. if (PresContext()->CompatibilityMode() == eCompatibility_FullStandards) { aIntrinsicSize.width += 1; } // Also add in the padding of our value div child. Note that it hasn't // been reflowed yet, so we can't get its used padding, but it shouldn't be // using percentage padding anyway. nsMargin childPadding; if (GetFirstChild(nsnull)->GetStylePadding()->GetPadding(childPadding)) { aIntrinsicSize.width += childPadding.LeftRight(); } else { NS_ERROR("Percentage padding on value div?"); } } // Increment width with cols * letter-spacing. { const nsStyleCoord& lsCoord = GetStyleText()->mLetterSpacing; if (eStyleUnit_Coord == lsCoord.GetUnit()) { nscoord letterSpacing = lsCoord.GetCoordValue(); if (letterSpacing != 0) { aIntrinsicSize.width += cols * letterSpacing; } } } // Set the height equal to total number of rows (times the height of each // line, of course) aIntrinsicSize.height = lineHeight * GetRows(); // Add in the size of the scrollbars for textarea if (IsTextArea()) { nsIFrame* first = GetFirstChild(nsnull); nsIScrollableFrame *scrollableFrame = do_QueryFrame(first); NS_ASSERTION(scrollableFrame, "Child must be scrollable"); nsMargin scrollbarSizes = scrollableFrame->GetDesiredScrollbarSizes(PresContext(), aRenderingContext); aIntrinsicSize.width += scrollbarSizes.LeftRight(); aIntrinsicSize.height += scrollbarSizes.TopBottom();; } return NS_OK; } PRInt32 nsTextControlFrame::GetWrapCols() { if (IsTextArea()) { // wrap=off means -1 for wrap width no matter what cols is nsHTMLTextWrap wrapProp; ::GetWrapPropertyEnum(mContent, wrapProp); if (wrapProp == eHTMLTextWrap_Off) { // do not wrap when wrap=off return -1; } // Otherwise we just wrap at the given number of columns return GetCols(); } // Never wrap non-textareas return -1; } nsresult nsTextControlFrame::EnsureEditorInitialized() { nsWeakFrame weakFrame(this); nsresult rv = EnsureEditorInitializedInternal(); NS_ENSURE_STATE(weakFrame.IsAlive()); return rv; } nsresult nsTextControlFrame::EnsureEditorInitializedInternal() { // This method initializes our editor, if needed. // This code used to be called from CreateAnonymousContent(), but // when the editor set the initial string, it would trigger a // PresShell listener which called FlushPendingNotifications() // during frame construction. This was causing other form controls // to display wrong values. Additionally, calling this every time // a text frame control is instantiated means that we're effectively // instantiating the editor for all text fields, even if they // never get used. So, now this method is being called lazily only // when we actually need an editor. // Check if this method has been called already. // If so, just return early. if (mUseEditor) return NS_OK; nsIDocument* doc = mContent->GetCurrentDoc(); NS_ENSURE_TRUE(doc, NS_ERROR_FAILURE); nsWeakFrame weakFrame(this); // Flush out content on our document. Have to do this, because script // blockers don't prevent the sink flushing out content and notifying in the // process, which can destroy frames. doc->FlushPendingNotifications(Flush_ContentAndNotify); NS_ENSURE_TRUE(weakFrame.IsAlive(), NS_ERROR_FAILURE); // Make sure that editor init doesn't do things that would kill us off // (especially off the script blockers it'll create for its DOM mutations). nsAutoScriptBlocker scriptBlocker; // Time to mess with our security context... See comments in GetValue() // for why this is needed. nsCxPusher pusher; pusher.PushNull(); // Make sure that we try to focus the content even if the method fails class EnsureSetFocus { public: explicit EnsureSetFocus(nsTextControlFrame* aFrame) : mFrame(aFrame) {} ~EnsureSetFocus() { if (IsFocusedContent(mFrame->GetContent())) mFrame->SetFocus(PR_TRUE, PR_FALSE); } private: nsTextControlFrame *mFrame; }; EnsureSetFocus makeSureSetFocusHappens(this); // Create an editor nsresult rv; mEditor = do_CreateInstance(kTextEditorCID, &rv); NS_ENSURE_SUCCESS(rv, rv); // Setup the editor flags PRUint32 editorFlags = 0; if (IsPlainTextControl()) editorFlags |= nsIPlaintextEditor::eEditorPlaintextMask; if (IsSingleLineTextControl()) editorFlags |= nsIPlaintextEditor::eEditorSingleLineMask; if (IsPasswordTextControl()) editorFlags |= nsIPlaintextEditor::eEditorPasswordMask; // All nsTextControlFrames are widgets editorFlags |= nsIPlaintextEditor::eEditorWidgetMask; // Use async reflow and painting for text widgets to improve // performance. // XXX: Using editor async updates exposes bugs 158782, 151882, // and 165130, so we're disabling it for now, until they // can be addressed. // editorFlags |= nsIPlaintextEditor::eEditorUseAsyncUpdatesMask; // Now initialize the editor. // // NOTE: Conversion of '\n' to
happens inside the // editor's Init() call. nsPresContext *presContext = PresContext(); nsIPresShell *shell = presContext->GetPresShell(); // Get the DOM document nsCOMPtr domdoc = do_QueryInterface(shell->GetDocument()); if (!domdoc) return NS_ERROR_FAILURE; // Make sure we clear out the non-breaking space before we initialize the editor UpdateValueDisplay(PR_FALSE, PR_TRUE); rv = mEditor->Init(domdoc, shell, mValueDiv, mSelCon, editorFlags); NS_ENSURE_SUCCESS(rv, rv); // Initialize the controller for the editor if (!SuppressEventHandlers(presContext)) { nsCOMPtr controllers; nsCOMPtr inputElement = do_QueryInterface(mContent); if (inputElement) { rv = inputElement->GetControllers(getter_AddRefs(controllers)); } else { nsCOMPtr textAreaElement = do_QueryInterface(mContent); if (!textAreaElement) return NS_ERROR_FAILURE; rv = textAreaElement->GetControllers(getter_AddRefs(controllers)); } if (NS_FAILED(rv)) return rv; if (controllers) { PRUint32 numControllers; PRBool found = PR_FALSE; rv = controllers->GetControllerCount(&numControllers); for (PRUint32 i = 0; i < numControllers; i ++) { nsCOMPtr controller; rv = controllers->GetControllerAt(i, getter_AddRefs(controller)); if (NS_SUCCEEDED(rv) && controller) { nsCOMPtr editController = do_QueryInterface(controller); if (editController) { editController->SetCommandContext(mEditor); found = PR_TRUE; } } } if (!found) rv = NS_ERROR_FAILURE; } } // Initialize the plaintext editor nsCOMPtr textEditor(do_QueryInterface(mEditor)); if (textEditor) { // Set up wrapping textEditor->SetWrapColumn(GetWrapCols()); // Set max text field length PRInt32 maxLength; if (GetMaxLength(&maxLength)) { textEditor->SetMaxTextLength(maxLength); } } if (mContent) { rv = mEditor->GetFlags(&editorFlags); if (NS_FAILED(rv)) return nsnull; // Check if the readonly attribute is set. if (mContent->HasAttr(kNameSpaceID_None, nsGkAtoms::readonly)) editorFlags |= nsIPlaintextEditor::eEditorReadonlyMask; // Check if the disabled attribute is set. if (mContent->HasAttr(kNameSpaceID_None, nsGkAtoms::disabled)) editorFlags |= nsIPlaintextEditor::eEditorDisabledMask; // Disable the selection if necessary. if (editorFlags & nsIPlaintextEditor::eEditorDisabledMask) mSelCon->SetDisplaySelection(nsISelectionController::SELECTION_OFF); mEditor->SetFlags(editorFlags); } // Get the current value of the textfield from the content. nsAutoString defaultValue; GetValue(defaultValue, PR_TRUE); // Turn on mUseEditor so that subsequent calls will use the // editor. mUseEditor = PR_TRUE; #ifdef DEBUG // Make sure we are not being called again until we're finished. // If reentrancy happens, just pretend that we don't have an editor. const EditorInitializerEntryTracker tracker(*this); NS_ASSERTION(!tracker.EnteredMoreThanOnce(), "EnsureEditorInitialized has been called while a previous call was in progress"); #endif // If we have a default value, insert it under the div we created // above, but be sure to use the editor so that '*' characters get // displayed for password fields, etc. SetValue() will call the // editor for us. if (!defaultValue.IsEmpty()) { // Avoid causing reentrant painting and reflowing by telling the editor // that we don't want it to force immediate view refreshes or force // immediate reflows during any editor calls. rv = mEditor->SetFlags(editorFlags | nsIPlaintextEditor::eEditorUseAsyncUpdatesMask); if (NS_FAILED(rv)) return rv; // Now call SetValue() which will make the necessary editor calls to set // the default value. Make sure to turn off undo before setting the default // value, and turn it back on afterwards. This will make sure we can't undo // past the default value. rv = mEditor->EnableUndo(PR_FALSE); if (NS_FAILED(rv)) return rv; SetValue(defaultValue); rv = mEditor->EnableUndo(PR_TRUE); NS_ASSERTION(NS_SUCCEEDED(rv),"Transaction Manager must have failed"); // Now restore the original editor flags. rv = mEditor->SetFlags(editorFlags); if (NS_FAILED(rv)) return rv; // By default the placeholder is shown, // we should hide it if the default value is not empty. nsWeakFrame weakFrame(this); HidePlaceholder(); NS_ENSURE_STATE(weakFrame.IsAlive()); } nsCOMPtr transMgr; mEditor->GetTransactionManager(getter_AddRefs(transMgr)); NS_ENSURE_TRUE(transMgr, NS_ERROR_FAILURE); transMgr->SetMaxTransactionCount(DEFAULT_UNDO_CAP); if (IsPasswordTextControl()) { // Disable undo for password textfields. Note that we want to do this at // the very end of InitEditor, so the calls to EnableUndo when setting the // default value don't screw us up. // Since changing the control type does a reframe, we don't have to worry // about dynamic type changes here. mEditor->EnableUndo(PR_FALSE); } mEditor->PostCreate(); if (mTextListener) mEditor->AddEditorObserver(mTextListener); return NS_OK; } nsresult nsTextControlFrame::CreateAnonymousContent(nsTArray& aElements) { mState |= NS_FRAME_INDEPENDENT_SELECTION; nsIPresShell *shell = PresContext()->GetPresShell(); if (!shell) return NS_ERROR_FAILURE; nsIDocument *doc = shell->GetDocument(); if (!doc) return NS_ERROR_FAILURE; // Now create a DIV and add it to the anonymous content child list. nsCOMPtr nodeInfo; nodeInfo = doc->NodeInfoManager()->GetNodeInfo(nsGkAtoms::div, nsnull, kNameSpaceID_XHTML); NS_ENSURE_TRUE(nodeInfo, NS_ERROR_OUT_OF_MEMORY); nsresult rv = NS_NewHTMLElement(getter_AddRefs(mValueDiv), nodeInfo, PR_FALSE); NS_ENSURE_SUCCESS(rv, rv); // Set the necessary classes on the text control. We use class values // instead of a 'style' attribute so that the style comes from a user-agent // style sheet and is still applied even if author styles are disabled. nsAutoString classValue; classValue.AppendLiteral("anonymous-div"); PRInt32 wrapCols = GetWrapCols(); if (wrapCols >= 0) { classValue.AppendLiteral(" wrap"); } if (!IsSingleLineTextControl()) { // We can't just inherit the overflow because setting visible overflow will // crash when the number of lines exceeds the height of the textarea and // setting -moz-hidden-unscrollable overflow (NS_STYLE_OVERFLOW_CLIP) // doesn't paint the caret for some reason. const nsStyleDisplay* disp = GetStyleDisplay(); if (disp->mOverflowX != NS_STYLE_OVERFLOW_VISIBLE && disp->mOverflowX != NS_STYLE_OVERFLOW_CLIP) { classValue.AppendLiteral(" inherit-overflow"); } mMutationObserver = new nsAnonDivObserver(this); NS_ENSURE_TRUE(mMutationObserver, NS_ERROR_OUT_OF_MEMORY); mValueDiv->AddMutationObserver(mMutationObserver); } rv = mValueDiv->SetAttr(kNameSpaceID_None, nsGkAtoms::_class, classValue, PR_FALSE); NS_ENSURE_SUCCESS(rv, rv); if (!aElements.AppendElement(mValueDiv)) return NS_ERROR_OUT_OF_MEMORY; // Now create the placeholder anonymous content rv = CreatePlaceholderDiv(aElements, doc->NodeInfoManager()); NS_ENSURE_SUCCESS(rv, rv); rv = UpdateValueDisplay(PR_FALSE); NS_ENSURE_SUCCESS(rv, rv); // Create selection mFrameSel = do_CreateInstance(kFrameSelectionCID, &rv); if (NS_FAILED(rv)) return rv; // Create a SelectionController mSelCon = new nsTextInputSelectionImpl(mFrameSel, shell, mValueDiv); if (!mSelCon) return NS_ERROR_OUT_OF_MEMORY; mTextListener = new nsTextInputListener(); if (!mTextListener) return NS_ERROR_OUT_OF_MEMORY; NS_ADDREF(mTextListener); mTextListener->SetFrame(this); mSelCon->SetDisplaySelection(nsISelectionController::SELECTION_ON); // Get the caret and make it a selection listener. nsRefPtr domSelection; if (NS_SUCCEEDED(mSelCon->GetSelection(nsISelectionController::SELECTION_NORMAL, getter_AddRefs(domSelection))) && domSelection) { nsCOMPtr selPriv(do_QueryInterface(domSelection)); nsRefPtr caret = shell->GetCaret(); nsCOMPtr listener; if (caret) { listener = do_QueryInterface(caret); if (listener) { selPriv->AddSelectionListener(listener); } } selPriv->AddSelectionListener(static_cast (mTextListener)); } if (!IsSingleLineTextControl()) { // textareas are eagerly initialized NS_ASSERTION(!nsContentUtils::IsSafeToRunScript(), "Someone forgot a script blocker?"); if (!nsContentUtils::AddScriptRunner(new EditorInitializer(this))) { return NS_ERROR_OUT_OF_MEMORY; } } return NS_OK; } void nsTextControlFrame::AppendAnonymousContentTo(nsBaseContentList& aElements) { aElements.MaybeAppendElement(mValueDiv); aElements.MaybeAppendElement(mPlaceholderDiv); } nscoord nsTextControlFrame::GetMinWidth(nsIRenderingContext* aRenderingContext) { // Our min width is just our preferred width if we have auto width. nscoord result; DISPLAY_MIN_WIDTH(this, result); result = GetPrefWidth(aRenderingContext); return result; } nsSize nsTextControlFrame::ComputeAutoSize(nsIRenderingContext *aRenderingContext, nsSize aCBSize, nscoord aAvailableWidth, nsSize aMargin, nsSize aBorder, nsSize aPadding, PRBool aShrinkWrap) { nsSize autoSize; nsresult rv = CalcIntrinsicSize(aRenderingContext, autoSize); if (NS_FAILED(rv)) { // What now? autoSize.SizeTo(0, 0); } #ifdef DEBUG // Note: Ancestor ComputeAutoSize only computes a width if we're auto-width else if (GetStylePosition()->mWidth.GetUnit() == eStyleUnit_Auto) { nsSize ancestorAutoSize = nsStackFrame::ComputeAutoSize(aRenderingContext, aCBSize, aAvailableWidth, aMargin, aBorder, aPadding, aShrinkWrap); NS_ASSERTION(ancestorAutoSize.width == autoSize.width, "Incorrect size computed by ComputeAutoSize?"); } #endif return autoSize; } // We inherit our GetPrefWidth from nsBoxFrame NS_IMETHODIMP nsTextControlFrame::Reflow(nsPresContext* aPresContext, nsHTMLReflowMetrics& aDesiredSize, const nsHTMLReflowState& aReflowState, nsReflowStatus& aStatus) { DO_GLOBAL_REFLOW_COUNT("nsTextControlFrame"); DISPLAY_REFLOW(aPresContext, this, aReflowState, aDesiredSize, aStatus); // make sure the the form registers itself on the initial/first reflow if (mState & NS_FRAME_FIRST_REFLOW) { nsFormControlFrame::RegUnRegAccessKey(this, PR_TRUE); } return nsStackFrame::Reflow(aPresContext, aDesiredSize, aReflowState, aStatus); } nsSize nsTextControlFrame::GetPrefSize(nsBoxLayoutState& aState) { if (!DoesNeedRecalc(mPrefSize)) return mPrefSize; #ifdef DEBUG_LAYOUT PropagateDebug(aState); #endif nsSize pref(0,0); nsresult rv = CalcIntrinsicSize(aState.GetRenderingContext(), pref); NS_ENSURE_SUCCESS(rv, pref); AddBorderAndPadding(pref); PRBool widthSet, heightSet; nsIBox::AddCSSPrefSize(this, pref, widthSet, heightSet); nsSize minSize = GetMinSize(aState); nsSize maxSize = GetMaxSize(aState); mPrefSize = BoundsCheck(minSize, pref, maxSize); #ifdef DEBUG_rods { nsMargin borderPadding(0,0,0,0); GetBorderAndPadding(borderPadding); nsSize size(169, 24); nsSize actual(pref.width/15, pref.height/15); printf("nsGfxText(field) %d,%d %d,%d %d,%d\n", size.width, size.height, actual.width, actual.height, actual.width-size.width, actual.height-size.height); // text field } #endif return mPrefSize; } nsSize nsTextControlFrame::GetMinSize(nsBoxLayoutState& aState) { // XXXbz why? Why not the nsBoxFrame sizes? return nsBox::GetMinSize(aState); } nsSize nsTextControlFrame::GetMaxSize(nsBoxLayoutState& aState) { // XXXbz why? Why not the nsBoxFrame sizes? return nsBox::GetMaxSize(aState); } nscoord nsTextControlFrame::GetBoxAscent(nsBoxLayoutState& aState) { // Return the baseline of the first (nominal) row, with centering for // single-line controls. // First calculate the ascent wrt the client rect nsRect clientRect; GetClientRect(clientRect); nscoord lineHeight = IsSingleLineTextControl() ? clientRect.height : nsHTMLReflowState::CalcLineHeight(GetStyleContext(), NS_AUTOHEIGHT); nsCOMPtr fontMet; nsresult rv = nsLayoutUtils::GetFontMetricsForFrame(this, getter_AddRefs(fontMet)); NS_ENSURE_SUCCESS(rv, 0); nscoord ascent = nsLayoutUtils::GetCenteredFontBaseline(fontMet, lineHeight); // Now adjust for our borders and padding ascent += clientRect.y; return ascent; } PRBool nsTextControlFrame::IsCollapsed(nsBoxLayoutState& aBoxLayoutState) { // We're never collapsed in the box sense. return PR_FALSE; } PRBool nsTextControlFrame::IsLeaf() const { return PR_TRUE; } //IMPLEMENTING NS_IFORMCONTROLFRAME void nsTextControlFrame::SetFocus(PRBool aOn, PRBool aRepaint) { if (!aOn) { nsWeakFrame weakFrame(this); nsAutoString valueString; GetValue(valueString, PR_TRUE); if (valueString.IsEmpty()) ShowPlaceholder(); if (!weakFrame.IsAlive()) { return; } MaybeEndSecureKeyboardInput(); return; } if (!mSelCon) return; nsWeakFrame weakFrame(this); HidePlaceholder(); if (!weakFrame.IsAlive()) { return; } if (NS_SUCCEEDED(InitFocusedValue())) MaybeBeginSecureKeyboardInput(); // Scroll the current selection into view mSelCon->ScrollSelectionIntoView(nsISelectionController::SELECTION_NORMAL, nsISelectionController::SELECTION_FOCUS_REGION, PR_FALSE); // tell the caret to use our selection nsCOMPtr ourSel; mSelCon->GetSelection(nsISelectionController::SELECTION_NORMAL, getter_AddRefs(ourSel)); if (!ourSel) return; nsIPresShell* presShell = PresContext()->GetPresShell(); nsRefPtr caret = presShell->GetCaret(); if (!caret) return; caret->SetCaretDOMSelection(ourSel); // mutual-exclusion: the selection is either controlled by the // document or by the text input/area. Clear any selection in the // document since the focus is now on our independent selection. nsCOMPtr selCon(do_QueryInterface(presShell)); nsCOMPtr docSel; selCon->GetSelection(nsISelectionController::SELECTION_NORMAL, getter_AddRefs(docSel)); if (!docSel) return; PRBool isCollapsed = PR_FALSE; docSel->GetIsCollapsed(&isCollapsed); if (!isCollapsed) docSel->RemoveAllRanges(); } nsresult nsTextControlFrame::SetFormProperty(nsIAtom* aName, const nsAString& aValue) { if (!mIsProcessing)//some kind of lock. { mIsProcessing = PR_TRUE; PRBool isUserInput = (nsGkAtoms::userInput == aName); if (nsGkAtoms::value == aName || isUserInput) { PRBool fireChangeEvent = GetFireChangeEventState(); if (isUserInput) { SetFireChangeEventState(PR_TRUE); } SetValueChanged(PR_TRUE); nsresult rv = SetValue(aValue); // set new text value if (isUserInput) { SetFireChangeEventState(fireChangeEvent); } NS_ENSURE_SUCCESS(rv, rv); } else if (nsGkAtoms::select == aName) { // Select all the text. // // XXX: This is lame, we can't call mEditor->SelectAll() // because that triggers AutoCopies in unix builds. // Instead, we have to call our own homegrown version // of select all which merely builds a range that selects // all of the content and adds that to the selection. SelectAllOrCollapseToEndOfText(PR_TRUE); } mIsProcessing = PR_FALSE; } return NS_OK; } nsresult nsTextControlFrame::GetFormProperty(nsIAtom* aName, nsAString& aValue) const { // Return the value of the property from the widget it is not null. // If widget is null, assume the widget is GFX-rendered and return a member variable instead. if (nsGkAtoms::value == aName) { GetValue(aValue, PR_FALSE); } return NS_OK; } NS_IMETHODIMP nsTextControlFrame::GetEditor(nsIEditor **aEditor) { NS_ENSURE_ARG_POINTER(aEditor); nsresult rv = EnsureEditorInitialized(); NS_ENSURE_SUCCESS(rv, rv); *aEditor = mEditor; NS_IF_ADDREF(*aEditor); return NS_OK; } NS_IMETHODIMP nsTextControlFrame::OwnsValue(PRBool* aOwnsValue) { NS_PRECONDITION(aOwnsValue, "aOwnsValue must be non-null"); *aOwnsValue = mUseEditor; return NS_OK; } NS_IMETHODIMP nsTextControlFrame::GetTextLength(PRInt32* aTextLength) { NS_ENSURE_ARG_POINTER(aTextLength); nsAutoString textContents; GetValue(textContents, PR_FALSE); // this is expensive! *aTextLength = textContents.Length(); return NS_OK; } nsresult nsTextControlFrame::SetSelectionInternal(nsIDOMNode *aStartNode, PRInt32 aStartOffset, nsIDOMNode *aEndNode, PRInt32 aEndOffset) { // Create a new range to represent the new selection. // Note that we use a new range to avoid having to do // isIncreasing checks to avoid possible errors. nsCOMPtr range = do_CreateInstance(kRangeCID); NS_ENSURE_TRUE(range, NS_ERROR_FAILURE); nsresult rv = range->SetStart(aStartNode, aStartOffset); NS_ENSURE_SUCCESS(rv, rv); rv = range->SetEnd(aEndNode, aEndOffset); NS_ENSURE_SUCCESS(rv, rv); // Get the selection, clear it and add the new range to it! nsCOMPtr selection; mSelCon->GetSelection(nsISelectionController::SELECTION_NORMAL, getter_AddRefs(selection)); NS_ENSURE_TRUE(selection, NS_ERROR_FAILURE); rv = selection->RemoveAllRanges(); NS_ENSURE_SUCCESS(rv, rv); rv = selection->AddRange(range); NS_ENSURE_SUCCESS(rv, rv); // Scroll the selection into view (see bug 231389) return mSelCon->ScrollSelectionIntoView(nsISelectionController::SELECTION_NORMAL, nsISelectionController::SELECTION_FOCUS_REGION, PR_FALSE); } nsresult nsTextControlFrame::SelectAllOrCollapseToEndOfText(PRBool aSelect) { if (!mEditor) return NS_OK; nsCOMPtr rootElement; nsresult rv = mEditor->GetRootElement(getter_AddRefs(rootElement)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr rootContent = do_QueryInterface(rootElement); nsCOMPtr rootNode(do_QueryInterface(rootElement)); PRInt32 numChildren = rootContent->GetChildCount(); if (numChildren > 0) { // We never want to place the selection after the last // br under the root node! nsIContent *child = rootContent->GetChildAt(numChildren - 1); if (child) { if (child->Tag() == nsGkAtoms::br) --numChildren; } if (!aSelect && numChildren) { child = rootContent->GetChildAt(numChildren - 1); if (child && child->IsNodeOfType(nsINode::eTEXT)) { rootNode = do_QueryInterface(child); const nsTextFragment* fragment = child->GetText(); numChildren = fragment ? fragment->GetLength() : 0; } } } return SetSelectionInternal(rootNode, aSelect ? 0 : numChildren, rootNode, numChildren); } nsresult nsTextControlFrame::SetSelectionEndPoints(PRInt32 aSelStart, PRInt32 aSelEnd) { NS_ASSERTION(aSelStart <= aSelEnd, "Invalid selection offsets!"); if (aSelStart > aSelEnd) return NS_ERROR_FAILURE; nsCOMPtr startNode, endNode; PRInt32 startOffset, endOffset; // Calculate the selection start point. nsresult rv = OffsetToDOMPoint(aSelStart, getter_AddRefs(startNode), &startOffset); NS_ENSURE_SUCCESS(rv, rv); if (aSelStart == aSelEnd) { // Collapsed selection, so start and end are the same! endNode = startNode; endOffset = startOffset; } else { // Selection isn't collapsed so we have to calculate // the end point too. rv = OffsetToDOMPoint(aSelEnd, getter_AddRefs(endNode), &endOffset); NS_ENSURE_SUCCESS(rv, rv); } return SetSelectionInternal(startNode, startOffset, endNode, endOffset); } NS_IMETHODIMP nsTextControlFrame::SetSelectionRange(PRInt32 aSelStart, PRInt32 aSelEnd) { nsresult rv = EnsureEditorInitialized(); NS_ENSURE_SUCCESS(rv, rv); if (aSelStart > aSelEnd) { // Simulate what we'd see SetSelectionStart() was called, followed // by a SetSelectionEnd(). aSelStart = aSelEnd; } return SetSelectionEndPoints(aSelStart, aSelEnd); } NS_IMETHODIMP nsTextControlFrame::SetSelectionStart(PRInt32 aSelectionStart) { nsresult rv = EnsureEditorInitialized(); NS_ENSURE_SUCCESS(rv, rv); PRInt32 selStart = 0, selEnd = 0; rv = GetSelectionRange(&selStart, &selEnd); NS_ENSURE_SUCCESS(rv, rv); if (aSelectionStart > selEnd) { // Collapse to the new start point. selEnd = aSelectionStart; } selStart = aSelectionStart; return SetSelectionEndPoints(selStart, selEnd); } NS_IMETHODIMP nsTextControlFrame::SetSelectionEnd(PRInt32 aSelectionEnd) { nsresult rv = EnsureEditorInitialized(); NS_ENSURE_SUCCESS(rv, rv); PRInt32 selStart = 0, selEnd = 0; rv = GetSelectionRange(&selStart, &selEnd); NS_ENSURE_SUCCESS(rv, rv); if (aSelectionEnd < selStart) { // Collapse to the new end point. selStart = aSelectionEnd; } selEnd = aSelectionEnd; return SetSelectionEndPoints(selStart, selEnd); } nsresult nsTextControlFrame::DOMPointToOffset(nsIDOMNode* aNode, PRInt32 aNodeOffset, PRInt32* aResult) { NS_ENSURE_ARG_POINTER(aNode && aResult); *aResult = 0; nsresult rv = EnsureEditorInitialized(); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr rootElement; mEditor->GetRootElement(getter_AddRefs(rootElement)); nsCOMPtr rootNode(do_QueryInterface(rootElement)); NS_ENSURE_TRUE(rootNode, NS_ERROR_FAILURE); nsCOMPtr nodeList; rv = rootNode->GetChildNodes(getter_AddRefs(nodeList)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(nodeList, NS_ERROR_FAILURE); PRUint32 length = 0; rv = nodeList->GetLength(&length); NS_ENSURE_SUCCESS(rv, rv); if (!length || aNodeOffset < 0) return NS_OK; PRInt32 i, textOffset = 0; PRInt32 lastIndex = (PRInt32)length - 1; for (i = 0; i < (PRInt32)length; i++) { if (rootNode == aNode && i == aNodeOffset) { *aResult = textOffset; return NS_OK; } nsCOMPtr item; rv = nodeList->Item(i, getter_AddRefs(item)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(item, NS_ERROR_FAILURE); nsCOMPtr domText(do_QueryInterface(item)); if (domText) { PRUint32 textLength = 0; rv = domText->GetLength(&textLength); NS_ENSURE_SUCCESS(rv, rv); if (item == aNode) { NS_ASSERTION((aNodeOffset >= 0 && aNodeOffset <= (PRInt32)textLength), "Invalid aNodeOffset!"); *aResult = textOffset + aNodeOffset; return NS_OK; } textOffset += textLength; } else { // Must be a BR node. If it's not the last BR node // under the root, count it as a newline. if (i != lastIndex) ++textOffset; } } NS_ASSERTION((aNode == rootNode && aNodeOffset == (PRInt32)length), "Invalid node offset!"); *aResult = textOffset; return NS_OK; } nsresult nsTextControlFrame::OffsetToDOMPoint(PRInt32 aOffset, nsIDOMNode** aResult, PRInt32* aPosition) { NS_ENSURE_ARG_POINTER(aResult && aPosition); *aResult = nsnull; *aPosition = 0; nsresult rv = EnsureEditorInitialized(); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr rootElement; mEditor->GetRootElement(getter_AddRefs(rootElement)); nsCOMPtr rootNode(do_QueryInterface(rootElement)); NS_ENSURE_TRUE(rootNode, NS_ERROR_FAILURE); nsCOMPtr nodeList; rv = rootNode->GetChildNodes(getter_AddRefs(nodeList)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(nodeList, NS_ERROR_FAILURE); PRUint32 length = 0; rv = nodeList->GetLength(&length); NS_ENSURE_SUCCESS(rv, rv); if (!length || aOffset < 0) { *aPosition = 0; *aResult = rootNode; NS_ADDREF(*aResult); return NS_OK; } PRInt32 textOffset = 0; PRUint32 lastIndex = length - 1; for (PRUint32 i=0; i item; rv = nodeList->Item(i, getter_AddRefs(item)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(item, NS_ERROR_FAILURE); nsCOMPtr domText(do_QueryInterface(item)); if (domText) { PRUint32 textLength = 0; rv = domText->GetLength(&textLength); NS_ENSURE_SUCCESS(rv, rv); // Check if aOffset falls within this range. if (aOffset >= textOffset && aOffset <= textOffset+(PRInt32)textLength) { *aPosition = aOffset - textOffset; *aResult = item; NS_ADDREF(*aResult); return NS_OK; } textOffset += textLength; // If there aren't any more siblings after this text node, // return the point at the end of this text node! if (i == lastIndex) { *aPosition = textLength; *aResult = item; NS_ADDREF(*aResult); return NS_OK; } } else { // Must be a BR node, count it as a newline. if (aOffset == textOffset || i == lastIndex) { // We've found the correct position, or aOffset takes us // beyond the last child under rootNode, just return the point // under rootNode that is in front of this br. *aPosition = i; *aResult = rootNode; NS_ADDREF(*aResult); return NS_OK; } ++textOffset; } } NS_ERROR("We should never get here!"); return NS_ERROR_FAILURE; } NS_IMETHODIMP nsTextControlFrame::GetSelectionRange(PRInt32* aSelectionStart, PRInt32* aSelectionEnd) { // make sure we have an editor nsresult rv = EnsureEditorInitialized(); NS_ENSURE_SUCCESS(rv, rv); *aSelectionStart = 0; *aSelectionEnd = 0; nsCOMPtr selection; rv = mSelCon->GetSelection(nsISelectionController::SELECTION_NORMAL, getter_AddRefs(selection)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(selection, NS_ERROR_FAILURE); PRInt32 numRanges = 0; selection->GetRangeCount(&numRanges); if (numRanges < 1) return NS_OK; // We only operate on the first range in the selection! nsCOMPtr firstRange; rv = selection->GetRangeAt(0, getter_AddRefs(firstRange)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(firstRange, NS_ERROR_FAILURE); nsCOMPtr startNode, endNode; PRInt32 startOffset = 0, endOffset = 0; // Get the start point of the range. rv = firstRange->GetStartContainer(getter_AddRefs(startNode)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(startNode, NS_ERROR_FAILURE); rv = firstRange->GetStartOffset(&startOffset); NS_ENSURE_SUCCESS(rv, rv); // Get the end point of the range. rv = firstRange->GetEndContainer(getter_AddRefs(endNode)); NS_ENSURE_SUCCESS(rv, rv); NS_ENSURE_TRUE(endNode, NS_ERROR_FAILURE); rv = firstRange->GetEndOffset(&endOffset); NS_ENSURE_SUCCESS(rv, rv); // Convert the start point to a selection offset. rv = DOMPointToOffset(startNode, startOffset, aSelectionStart); NS_ENSURE_SUCCESS(rv, rv); // Convert the end point to a selection offset. return DOMPointToOffset(endNode, endOffset, aSelectionEnd); } nsISelectionController* nsTextControlFrame::GetOwnedSelectionController() { return mSelCon; } /////END INTERFACE IMPLEMENTATIONS ////NSIFRAME NS_IMETHODIMP nsTextControlFrame::AttributeChanged(PRInt32 aNameSpaceID, nsIAtom* aAttribute, PRInt32 aModType) { // First, check for the placeholder attribute, because it doesn't // depend on the editor being present. if (nsGkAtoms::placeholder == aAttribute) { nsWeakFrame weakFrame(this); UpdatePlaceholderText(PR_TRUE); NS_ENSURE_STATE(weakFrame.IsAlive()); } if (!mEditor || !mSelCon) return nsBoxFrame::AttributeChanged(aNameSpaceID, aAttribute, aModType);; nsresult rv = NS_OK; if (nsGkAtoms::maxlength == aAttribute) { PRInt32 maxLength; PRBool maxDefined = GetMaxLength(&maxLength); nsCOMPtr textEditor = do_QueryInterface(mEditor); if (textEditor) { if (maxDefined) { // set the maxLength attribute textEditor->SetMaxTextLength(maxLength); // if maxLength>docLength, we need to truncate the doc content } else { // unset the maxLength attribute textEditor->SetMaxTextLength(-1); } } rv = NS_OK; // don't propagate the error } else if (nsGkAtoms::readonly == aAttribute) { PRUint32 flags; mEditor->GetFlags(&flags); if (AttributeExists(nsGkAtoms::readonly)) { // set readonly flags |= nsIPlaintextEditor::eEditorReadonlyMask; if (IsFocusedContent(mContent)) mSelCon->SetCaretEnabled(PR_FALSE); } else { // unset readonly flags &= ~(nsIPlaintextEditor::eEditorReadonlyMask); if (!(flags & nsIPlaintextEditor::eEditorDisabledMask) && IsFocusedContent(mContent)) mSelCon->SetCaretEnabled(PR_TRUE); } mEditor->SetFlags(flags); } else if (nsGkAtoms::disabled == aAttribute) { PRUint32 flags; mEditor->GetFlags(&flags); if (AttributeExists(nsGkAtoms::disabled)) { // set disabled flags |= nsIPlaintextEditor::eEditorDisabledMask; mSelCon->SetDisplaySelection(nsISelectionController::SELECTION_OFF); if (IsFocusedContent(mContent)) mSelCon->SetCaretEnabled(PR_FALSE); } else { // unset disabled flags &= ~(nsIPlaintextEditor::eEditorDisabledMask); mSelCon->SetDisplaySelection(nsISelectionController::SELECTION_HIDDEN); } mEditor->SetFlags(flags); } else if (!mUseEditor && nsGkAtoms::value == aAttribute) { UpdateValueDisplay(PR_TRUE); } // Allow the base class to handle common attributes supported // by all form elements... else { rv = nsBoxFrame::AttributeChanged(aNameSpaceID, aAttribute, aModType); } return rv; } nsresult nsTextControlFrame::GetText(nsString& aText) { nsresult rv = NS_OK; if (IsSingleLineTextControl()) { // If we're going to remove newlines anyway, ignore the wrap property GetValue(aText, PR_TRUE); nsContentUtils::RemoveNewlines(aText); } else { nsCOMPtr textArea = do_QueryInterface(mContent); if (textArea) { rv = textArea->GetValue(aText); } } return rv; } nsresult nsTextControlFrame::GetPhonetic(nsAString& aPhonetic) { aPhonetic.Truncate(0); nsresult rv = EnsureEditorInitialized(); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr imeSupport = do_QueryInterface(mEditor); if (imeSupport) { nsCOMPtr phonetic = do_QueryInterface(imeSupport); if (phonetic) phonetic->GetPhonetic(aPhonetic); } return NS_OK; } ///END NSIFRAME OVERLOADS /////BEGIN PROTECTED METHODS PRBool nsTextControlFrame::GetMaxLength(PRInt32* aSize) { *aSize = -1; nsGenericHTMLElement *content = nsGenericHTMLElement::FromContent(mContent); if (content) { const nsAttrValue* attr = content->GetParsedAttr(nsGkAtoms::maxlength); if (attr && attr->Type() == nsAttrValue::eInteger) { *aSize = attr->GetIntegerValue(); return PR_TRUE; } } return PR_FALSE; } // this is where we propagate a content changed event void nsTextControlFrame::FireOnInput() { if (!mNotifyOnInput) return; // if notification is turned off, do nothing // Dispatch the "input" event nsEventStatus status = nsEventStatus_eIgnore; nsUIEvent event(PR_TRUE, NS_FORM_INPUT, 0); // Have the content handle the event, propagating it according to normal // DOM rules. nsCOMPtr shell = PresContext()->PresShell(); shell->HandleEventWithTarget(&event, nsnull, mContent, &status); } nsresult nsTextControlFrame::InitFocusedValue() { return GetText(mFocusedValue); } NS_IMETHODIMP nsTextControlFrame::CheckFireOnChange() { nsString value; GetText(value); if (!mFocusedValue.Equals(value)) { mFocusedValue = value; // Dispatch the change event nsEventStatus status = nsEventStatus_eIgnore; nsInputEvent event(PR_TRUE, NS_FORM_CHANGE, nsnull); nsCOMPtr shell = PresContext()->PresShell(); shell->HandleEventWithTarget(&event, nsnull, mContent, &status); } return NS_OK; } //====== //privates NS_IMETHODIMP nsTextControlFrame::GetValue(nsAString& aValue, PRBool aIgnoreWrap) const { aValue.Truncate(); // initialize out param nsresult rv = NS_OK; if (mEditor && mUseEditor) { PRBool canCache = aIgnoreWrap && !IsSingleLineTextControl(); if (canCache && !mCachedValue.IsEmpty()) { aValue = mCachedValue; return NS_OK; } PRUint32 flags = (nsIDocumentEncoder::OutputLFLineBreak | nsIDocumentEncoder::OutputPreformatted | nsIDocumentEncoder::OutputPersistNBSP); if (PR_TRUE==IsPlainTextControl()) { flags |= nsIDocumentEncoder::OutputBodyOnly; } if (!aIgnoreWrap) { nsHTMLTextWrap wrapProp; if (::GetWrapPropertyEnum(mContent, wrapProp) && wrapProp == eHTMLTextWrap_Hard) { flags |= nsIDocumentEncoder::OutputWrap; } } // What follows is a bit of a hack. The problem is that we could be in // this method because we're being destroyed for whatever reason while // script is executing. If that happens, editor will run with the // privileges of the executing script, which means it may not be able to // access its own DOM nodes! Let's try to deal with that by pushing a null // JSContext on the JSContext stack to make it clear that we're native // code. Note that any script that's directly trying to access our value // has to be going through some scriptable object to do that and that // already does the relevant security checks. // XXXbz if we could just get the textContent of our anonymous content (eg // if plaintext editor didn't create
nodes all over), we wouldn't need // this. { /* Scope for context pusher */ nsCxPusher pusher; pusher.PushNull(); rv = mEditor->OutputToString(NS_LITERAL_STRING("text/plain"), flags, aValue); } if (canCache) { const_cast(this)->mCachedValue = aValue; } else { const_cast(this)->mCachedValue.Truncate(); } } else { // Otherwise get the value from content. nsCOMPtr inputControl = do_QueryInterface(mContent); if (inputControl) { rv = inputControl->GetValue(aValue); } else { nsCOMPtr textareaControl = do_QueryInterface(mContent); if (textareaControl) { rv = textareaControl->GetValue(aValue); } } } return rv; } // END IMPLEMENTING NS_IFORMCONTROLFRAME nsresult nsTextControlFrame::SetValue(const nsAString& aValue) { // XXX this method should actually propagate errors! It'd make debugging it // so much easier... if (mEditor && mUseEditor) { // This method isn't used for user-generated changes, except for calls // from nsFileControlFrame which sets mFireChangeEventState==true and // restores it afterwards (ie. we want 'change' events for those changes). // Focused value must be updated to prevent incorrect 'change' events, // but only if user hasn't changed the value. // GetText removes newlines from single line control. nsString currentValue; GetText(currentValue); PRBool focusValueInit = !mFireChangeEventState && mFocusedValue.Equals(currentValue); nsCOMPtr editor = mEditor; nsWeakFrame weakFrame(this); // this is necessary to avoid infinite recursion if (!currentValue.Equals(aValue)) { // \r is an illegal character in the dom, but people use them, // so convert windows and mac platform linebreaks to \n: // Unfortunately aValue is declared const, so we have to copy // in order to do this substitution. nsString newValue(aValue); nsContentUtils::PlatformToDOMLineBreaks(newValue); nsCOMPtr domDoc; editor->GetDocument(getter_AddRefs(domDoc)); NS_ENSURE_STATE(domDoc); PRBool outerTransaction; // Time to mess with our security context... See comments in GetValue() // for why this is needed. Note that we have to do this up here, because // otherwise SelectAll() will fail. { /* Scope for context pusher */ nsCxPusher pusher; pusher.PushNull(); nsCOMPtr domSel; nsCOMPtr selPriv; mSelCon->GetSelection(nsISelectionController::SELECTION_NORMAL, getter_AddRefs(domSel)); if (domSel) { selPriv = do_QueryInterface(domSel); if (selPriv) selPriv->StartBatchChanges(); } nsCOMPtr kungFuDeathGrip = mSelCon.get(); PRUint32 currentLength = currentValue.Length(); PRUint32 newlength = newValue.Length(); if (!currentLength || !StringBeginsWith(newValue, currentValue)) { // Replace the whole text. currentLength = 0; mSelCon->SelectAll(); } else { // Collapse selection to the end so that we can append data. SelectAllOrCollapseToEndOfText(PR_FALSE); } const nsAString& insertValue = StringTail(newValue, newlength - currentLength); nsCOMPtr plaintextEditor = do_QueryInterface(editor); if (!plaintextEditor || !weakFrame.IsAlive()) { NS_WARNING("Somehow not a plaintext editor?"); return NS_ERROR_FAILURE; } // Since this code does not handle user-generated changes to the text, // make sure we don't fire oninput when the editor notifies us. // (mNotifyOnInput must be reset before we return). // To protect against a reentrant call to SetValue, we check whether // another SetValue is already happening for this frame. If it is, // we must wait until we unwind to re-enable oninput events. outerTransaction = mNotifyOnInput; if (outerTransaction) mNotifyOnInput = PR_FALSE; // get the flags, remove readonly and disabled, set the value, // restore flags PRUint32 flags, savedFlags; editor->GetFlags(&savedFlags); flags = savedFlags; flags &= ~(nsIPlaintextEditor::eEditorDisabledMask); flags &= ~(nsIPlaintextEditor::eEditorReadonlyMask); flags |= nsIPlaintextEditor::eEditorUseAsyncUpdatesMask; flags |= nsIPlaintextEditor::eEditorDontEchoPassword; editor->SetFlags(flags); // Also don't enforce max-length here PRInt32 savedMaxLength; plaintextEditor->GetMaxTextLength(&savedMaxLength); plaintextEditor->SetMaxTextLength(-1); if (insertValue.IsEmpty()) { editor->DeleteSelection(nsIEditor::eNone); } else { plaintextEditor->InsertText(insertValue); } if (!IsSingleLineTextControl()) { mCachedValue = newValue; } plaintextEditor->SetMaxTextLength(savedMaxLength); editor->SetFlags(savedFlags); if (selPriv) selPriv->EndBatchChanges(); } NS_ENSURE_STATE(weakFrame.IsAlive()); if (outerTransaction) mNotifyOnInput = PR_TRUE; if (focusValueInit) { // Reset mFocusedValue so the onchange event doesn't fire incorrectly. InitFocusedValue(); } } NS_ENSURE_STATE(weakFrame.IsAlive()); nsIScrollableFrame* scrollableFrame = do_QueryFrame(GetFirstChild(nsnull)); if (scrollableFrame) { // Scroll the upper left corner of the text control's // content area back into view. scrollableFrame->ScrollTo(nsPoint(0, 0), nsIScrollableFrame::INSTANT); } } else { // Otherwise set the value in content. nsCOMPtr textControl = do_QueryInterface(mContent); if (textControl) { textControl->TakeTextFrameValue(aValue); } // The only time mEditor is non-null but mUseEditor is false is when the // frame is being torn down. If that's what's going on, don't bother with // updating the display. if (!mEditor) { UpdateValueDisplay(PR_TRUE, PR_FALSE, &aValue); } } return NS_OK; } NS_IMETHODIMP nsTextControlFrame::SetInitialChildList(nsIAtom* aListName, nsFrameList& aChildList) { nsresult rv = nsBoxFrame::SetInitialChildList(aListName, aChildList); nsIFrame* first = GetFirstChild(nsnull); // Mark the scroll frame as being a reflow root. This will allow // incremental reflows to be initiated at the scroll frame, rather // than descending from the root frame of the frame hierarchy. first->AddStateBits(NS_FRAME_REFLOW_ROOT); //register key listeners nsCOMPtr systemGroup; mContent->GetSystemEventGroup(getter_AddRefs(systemGroup)); nsCOMPtr dom3Targ = do_QueryInterface(mContent); if (dom3Targ) { // cast because of ambiguous base nsIDOMEventListener *listener = static_cast (mTextListener); dom3Targ->AddGroupedEventListener(NS_LITERAL_STRING("keydown"), listener, PR_FALSE, systemGroup); dom3Targ->AddGroupedEventListener(NS_LITERAL_STRING("keypress"), listener, PR_FALSE, systemGroup); dom3Targ->AddGroupedEventListener(NS_LITERAL_STRING("keyup"), listener, PR_FALSE, systemGroup); } mSelCon->SetScrollableFrame(do_QueryFrame(first)); return rv; } PRBool nsTextControlFrame::IsScrollable() const { return !IsSingleLineTextControl(); } void nsTextControlFrame::SetValueChanged(PRBool aValueChanged) { // placeholder management if (!IsFocusedContent(mContent)) { // If the content is focused, we don't care about the changes because // the placeholder is going to be hide/show on blur. nsAutoString valueString; GetValue(valueString, PR_TRUE); if (valueString.IsEmpty()) ShowPlaceholder(); else HidePlaceholder(); } nsCOMPtr elem = do_QueryInterface(mContent); if (elem) { elem->SetValueChanged(aValueChanged); } } nsresult nsTextControlFrame::UpdateValueDisplay(PRBool aNotify, PRBool aBeforeEditorInit, const nsAString *aValue) { if (!IsSingleLineTextControl()) // textareas don't use this return NS_OK; NS_PRECONDITION(mValueDiv, "Must have a div content\n"); NS_PRECONDITION(!mUseEditor, "Do not call this after editor has been initialized"); NS_ASSERTION(mValueDiv->GetChildCount() <= 1, "Cannot have more than one child node"); NS_ASSERTION(mPlaceholderDiv, "A placeholder div must exist"); enum { NO_NODE, TXT_NODE, BR_NODE } childNodeType = NO_NODE; nsIContent* childNode = mValueDiv->GetChildAt(0); #ifdef NS_DEBUG if (aBeforeEditorInit) NS_ASSERTION(childNode, "A child node should exist before initializing the editor"); #endif if (childNode) { if (childNode->IsNodeOfType(nsINode::eELEMENT)) childNodeType = BR_NODE; else if (childNode->IsNodeOfType(nsINode::eTEXT)) childNodeType = TXT_NODE; #ifdef NS_DEBUG else NS_NOTREACHED("Invalid child node found"); #endif } // Get the current value of the textfield from the content. nsAutoString value; if (aValue) { value = *aValue; } else { GetValue(value, PR_TRUE); } // Update the display of the placeholder value if needed. // We don't need to do this if we're about to initialize the // editor, since EnsureEditorInitialized takes care of this. if (!aBeforeEditorInit) { nsWeakFrame weakFrame(this); if (value.IsEmpty()) { ShowPlaceholder(); } else { HidePlaceholder(); } NS_ENSURE_STATE(weakFrame.IsAlive()); } if (aBeforeEditorInit && value.IsEmpty()) { mValueDiv->RemoveChildAt(0, PR_TRUE, PR_FALSE); return NS_OK; } nsTextEditRules::HandleNewLines(value, -1); nsresult rv; if (value.IsEmpty()) { if (childNodeType != BR_NODE) { nsCOMPtr nodeInfo; nodeInfo = mContent->NodeInfo() ->NodeInfoManager() ->GetNodeInfo(nsGkAtoms::br, nsnull, kNameSpaceID_XHTML); NS_ENSURE_TRUE(nodeInfo, NS_ERROR_OUT_OF_MEMORY); nsCOMPtr brNode; rv = NS_NewHTMLElement(getter_AddRefs(brNode), nodeInfo, PR_FALSE); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr brElement = do_QueryInterface(brNode); NS_ENSURE_TRUE(brElement, NS_ERROR_UNEXPECTED); brElement->SetAttribute(kMOZEditorBogusNodeAttr, kMOZEditorBogusNodeValue); mValueDiv->RemoveChildAt(0, aNotify, PR_FALSE); mValueDiv->AppendChildTo(brNode, aNotify); } } else { if (IsPasswordTextControl()) nsTextEditRules::FillBufWithPWChars(&value, value.Length()); // Set up a textnode with our value nsCOMPtr textNode; if (childNodeType != TXT_NODE) { rv = NS_NewTextNode(getter_AddRefs(textNode), mContent->NodeInfo()->NodeInfoManager()); NS_ENSURE_SUCCESS(rv, rv); NS_ASSERTION(textNode, "Must have textcontent!\n"); mValueDiv->RemoveChildAt(0, aNotify, PR_FALSE); mValueDiv->AppendChildTo(textNode, aNotify); } else { textNode = childNode; } textNode->SetText(value, aNotify); } return NS_OK; } /* static */ void nsTextControlFrame::ShutDown() { NS_IF_RELEASE(sNativeTextAreaBindings); NS_IF_RELEASE(sNativeInputBindings); } nsresult nsTextControlFrame::CreatePlaceholderDiv(nsTArray& aElements, nsNodeInfoManager* pNodeInfoManager) { nsresult rv; nsCOMPtr placeholderText; // Create a DIV for the placeholder // and add it to the anonymous content child list nsCOMPtr nodeInfo; nodeInfo = pNodeInfoManager->GetNodeInfo(nsGkAtoms::div, nsnull, kNameSpaceID_XHTML); NS_ENSURE_TRUE(nodeInfo, NS_ERROR_OUT_OF_MEMORY); rv = NS_NewHTMLElement(getter_AddRefs(mPlaceholderDiv), nodeInfo, PR_FALSE); NS_ENSURE_SUCCESS(rv, rv); // Create the text node for the placeholder text before doing anything else rv = NS_NewTextNode(getter_AddRefs(placeholderText), pNodeInfoManager); NS_ENSURE_SUCCESS(rv, rv); rv = mPlaceholderDiv->AppendChildTo(placeholderText, PR_FALSE); NS_ENSURE_SUCCESS(rv, rv); // Set the necessary classes on the text control. We use class values // instead of a 'style' attribute so that the style comes from a user-agent // style sheet and is still applied even if author styles are disabled. SetPlaceholderClass(PR_TRUE, PR_FALSE); if (!aElements.AppendElement(mPlaceholderDiv)) return NS_ERROR_OUT_OF_MEMORY; // initialise the text UpdatePlaceholderText(PR_FALSE); return NS_OK; } nsresult nsTextControlFrame::ShowPlaceholder() { return SetPlaceholderClass(PR_TRUE, PR_TRUE); } nsresult nsTextControlFrame::HidePlaceholder() { return SetPlaceholderClass(PR_FALSE, PR_TRUE); } nsresult nsTextControlFrame::SetPlaceholderClass(PRBool aVisible, PRBool aNotify) { nsresult rv; nsAutoString classValue; classValue.Assign(NS_LITERAL_STRING("anonymous-div placeholder")); if (!aVisible) classValue.AppendLiteral(" hidden"); rv = mPlaceholderDiv->SetAttr(kNameSpaceID_None, nsGkAtoms::_class, classValue, aNotify); NS_ENSURE_SUCCESS(rv, rv); return NS_OK; } nsresult nsTextControlFrame::UpdatePlaceholderText(PRBool aNotify) { nsAutoString placeholderValue; mContent->GetAttr(kNameSpaceID_None, nsGkAtoms::placeholder, placeholderValue); nsContentUtils::RemoveNewlines(placeholderValue); NS_ASSERTION(mPlaceholderDiv->GetChildAt(0), "placeholder div has no child"); mPlaceholderDiv->GetChildAt(0)->SetText(placeholderValue, aNotify); return NS_OK; } NS_IMPL_ISUPPORTS1(nsAnonDivObserver, nsIMutationObserver) void nsAnonDivObserver::CharacterDataChanged(nsIDocument* aDocument, nsIContent* aContent, CharacterDataChangeInfo* aInfo) { mTextControl->ClearValueCache(); } void nsAnonDivObserver::ContentAppended(nsIDocument* aDocument, nsIContent* aContainer, PRInt32 aNewIndexInContainer) { mTextControl->ClearValueCache(); } void nsAnonDivObserver::ContentInserted(nsIDocument* aDocument, nsIContent* aContainer, nsIContent* aChild, PRInt32 aIndexInContainer) { mTextControl->ClearValueCache(); } void nsAnonDivObserver::ContentRemoved(nsIDocument* aDocument, nsIContent* aContainer, nsIContent* aChild, PRInt32 aIndexInContainer) { mTextControl->ClearValueCache(); }