📄 htmltextareaelement.cpp
字号:
/* * Copyright (C) 1999 Lars Knoll (knoll@kde.org) * (C) 1999 Antti Koivisto (koivisto@kde.org) * (C) 2001 Dirk Mueller (mueller@kde.org) * Copyright (C) 2004, 2005, 2006, 2007, 2008 Apple Inc. All rights reserved. * (C) 2006 Alexey Proskuryakov (ap@nypop.com) * Copyright (C) 2007 Samuel Weinig (sam@webkit.org) * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public License * along with this library; see the file COPYING.LIB. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. * */#include "config.h"#include "HTMLTextAreaElement.h"#include "ChromeClient.h"#include "Document.h"#include "Event.h"#include "EventNames.h"#include "FocusController.h"#include "FormDataList.h"#include "Frame.h"#include "HTMLNames.h"#include "Page.h"#include "RenderStyle.h"#include "RenderTextControlMultiLine.h"#include "VisibleSelection.h"#include "Text.h"#include <wtf/StdLibExtras.h>namespace WebCore {using namespace HTMLNames;static const int defaultRows = 2;static const int defaultCols = 20;static inline void notifyFormStateChanged(const HTMLTextAreaElement* element){ Frame* frame = element->document()->frame(); if (!frame) return; frame->page()->chrome()->client()->formStateDidChange(element);}HTMLTextAreaElement::HTMLTextAreaElement(const QualifiedName& tagName, Document* document, HTMLFormElement* form) : HTMLFormControlElementWithState(tagName, document, form) , m_rows(defaultRows) , m_cols(defaultCols) , m_wrap(SoftWrap) , m_cachedSelectionStart(-1) , m_cachedSelectionEnd(-1){ ASSERT(hasTagName(textareaTag)); setValueMatchesRenderer(); notifyFormStateChanged(this);}const AtomicString& HTMLTextAreaElement::type() const{ DEFINE_STATIC_LOCAL(const AtomicString, textarea, ("textarea")); return textarea;}bool HTMLTextAreaElement::saveState(String& result) const{ result = value(); return true;}void HTMLTextAreaElement::restoreState(const String& state){ setDefaultValue(state);}int HTMLTextAreaElement::selectionStart(){ if (!renderer()) return 0; if (document()->focusedNode() != this && m_cachedSelectionStart >= 0) return m_cachedSelectionStart; return toRenderTextControl(renderer())->selectionStart();}int HTMLTextAreaElement::selectionEnd(){ if (!renderer()) return 0; if (document()->focusedNode() != this && m_cachedSelectionEnd >= 0) return m_cachedSelectionEnd; return toRenderTextControl(renderer())->selectionEnd();}void HTMLTextAreaElement::setSelectionStart(int start){ if (!renderer()) return; toRenderTextControl(renderer())->setSelectionStart(start);}void HTMLTextAreaElement::setSelectionEnd(int end){ if (!renderer()) return; toRenderTextControl(renderer())->setSelectionEnd(end);}void HTMLTextAreaElement::select(){ if (!renderer()) return; toRenderTextControl(renderer())->select();}void HTMLTextAreaElement::setSelectionRange(int start, int end){ if (!renderer()) return; toRenderTextControl(renderer())->setSelectionRange(start, end);}void HTMLTextAreaElement::childrenChanged(bool changedByParser, Node* beforeChange, Node* afterChange, int childCountDelta){ setValue(defaultValue()); HTMLElement::childrenChanged(changedByParser, beforeChange, afterChange, childCountDelta);} void HTMLTextAreaElement::parseMappedAttribute(MappedAttribute* attr){ if (attr->name() == rowsAttr) { int rows = attr->value().toInt(); if (rows <= 0) rows = defaultRows; if (m_rows != rows) { m_rows = rows; if (renderer()) renderer()->setNeedsLayoutAndPrefWidthsRecalc(); } } else if (attr->name() == colsAttr) { int cols = attr->value().toInt(); if (cols <= 0) cols = defaultCols; if (m_cols != cols) { m_cols = cols; if (renderer()) renderer()->setNeedsLayoutAndPrefWidthsRecalc(); } } else if (attr->name() == wrapAttr) { // The virtual/physical values were a Netscape extension of HTML 3.0, now deprecated. // The soft/hard /off values are a recommendation for HTML 4 extension by IE and NS 4. WrapMethod wrap; if (equalIgnoringCase(attr->value(), "physical") || equalIgnoringCase(attr->value(), "hard") || equalIgnoringCase(attr->value(), "on")) wrap = HardWrap; else if (equalIgnoringCase(attr->value(), "off")) wrap = NoWrap; else wrap = SoftWrap; if (wrap != m_wrap) { m_wrap = wrap; if (renderer()) renderer()->setNeedsLayoutAndPrefWidthsRecalc(); } } else if (attr->name() == accesskeyAttr) { // ignore for the moment } else if (attr->name() == alignAttr) { // Don't map 'align' attribute. This matches what Firefox, Opera and IE do. // See http://bugs.webkit.org/show_bug.cgi?id=7075 } else if (attr->name() == onfocusAttr) setInlineEventListenerForTypeAndAttribute(eventNames().focusEvent, attr); else if (attr->name() == onblurAttr) setInlineEventListenerForTypeAndAttribute(eventNames().blurEvent, attr); else if (attr->name() == onselectAttr) setInlineEventListenerForTypeAndAttribute(eventNames().selectEvent, attr); else if (attr->name() == onchangeAttr) setInlineEventListenerForTypeAndAttribute(eventNames().changeEvent, attr); else HTMLFormControlElementWithState::parseMappedAttribute(attr);}RenderObject* HTMLTextAreaElement::createRenderer(RenderArena* arena, RenderStyle*){ return new (arena) RenderTextControlMultiLine(this);}bool HTMLTextAreaElement::appendFormData(FormDataList& encoding, bool){ if (name().isEmpty()) return false; // FIXME: It's not acceptable to ignore the HardWrap setting when there is no renderer. // While we have no evidence this has ever been a practical problem, it would be best to fix it some day. RenderTextControl* control = toRenderTextControl(renderer()); const String& text = (m_wrap == HardWrap && control) ? control->textWithHardLineBreaks() : value(); encoding.appendData(name(), text); return true;}void HTMLTextAreaElement::reset(){ setValue(defaultValue());}bool HTMLTextAreaElement::isKeyboardFocusable(KeyboardEvent*) const{ // If a given text area can be focused at all, then it will always be keyboard focusable. return isFocusable();}bool HTMLTextAreaElement::isMouseFocusable() const{ return isFocusable();}void HTMLTextAreaElement::updateFocusAppearance(bool restorePreviousSelection){ ASSERT(renderer()); if (!restorePreviousSelection || m_cachedSelectionStart < 0) {#if ENABLE(ON_FIRST_TEXTAREA_FOCUS_SELECT_ALL) // Devices with trackballs or d-pads may focus on a textarea in route // to another focusable node. By selecting all text, the next movement // can more readily be interpreted as moving to the next node. select();#else // If this is the first focus, set a caret at the beginning of the text. // This matches some browsers' behavior; see bug 11746 Comment #15. // http://bugs.webkit.org/show_bug.cgi?id=11746#c15 setSelectionRange(0, 0);#endif } else { // Restore the cached selection. This matches other browsers' behavior. setSelectionRange(m_cachedSelectionStart, m_cachedSelectionEnd); } if (document()->frame()) document()->frame()->revealSelection();}void HTMLTextAreaElement::defaultEventHandler(Event* event){ if (renderer() && (event->isMouseEvent() || event->isDragEvent() || event->isWheelEvent() || event->type() == eventNames().blurEvent)) static_cast<RenderTextControlMultiLine*>(renderer())->forwardEvent(event); HTMLFormControlElementWithState::defaultEventHandler(event);}void HTMLTextAreaElement::rendererWillBeDestroyed(){ updateValue();}void HTMLTextAreaElement::updateValue() const{ if (valueMatchesRenderer()) return; ASSERT(renderer()); m_value = toRenderTextControl(renderer())->text(); const_cast<HTMLTextAreaElement*>(this)->setValueMatchesRenderer(); notifyFormStateChanged(this);}String HTMLTextAreaElement::value() const{ updateValue(); return m_value;}void HTMLTextAreaElement::setValue(const String& value){ // Code elsewhere normalizes line endings added by the user via the keyboard or pasting. // We normalize line endings coming from JavaScript here. String normalizedValue = value.isNull() ? "" : value; normalizedValue.replace("\r\n", "\n"); normalizedValue.replace('\r', '\n'); // Return early because we don't want to move the caret or trigger other side effects // when the value isn't changing. This matches Firefox behavior, at least. if (normalizedValue == this->value()) return; m_value = normalizedValue; setValueMatchesRenderer(); if (inDocument()) document()->updateRendering(); if (renderer()) renderer()->updateFromElement(); // Set the caret to the end of the text value. if (document()->focusedNode() == this) { unsigned endOfString = m_value.length(); setSelectionRange(endOfString, endOfString); } setChanged(); notifyFormStateChanged(this);}String HTMLTextAreaElement::defaultValue() const{ String value = ""; // Since there may be comments, ignore nodes other than text nodes. for (Node* n = firstChild(); n; n = n->nextSibling()) { if (n->isTextNode()) value += static_cast<Text*>(n)->data(); } UChar firstCharacter = value[0]; if (firstCharacter == '\r' && value[1] == '\n') value.remove(0, 2); else if (firstCharacter == '\r' || firstCharacter == '\n') value.remove(0, 1); return value;}void HTMLTextAreaElement::setDefaultValue(const String& defaultValue){ // To preserve comments, remove only the text nodes, then add a single text node. Vector<RefPtr<Node> > textNodes; for (Node* n = firstChild(); n; n = n->nextSibling()) { if (n->isTextNode()) textNodes.append(n); } ExceptionCode ec; size_t size = textNodes.size(); for (size_t i = 0; i < size; ++i) removeChild(textNodes[i].get(), ec); // Normalize line endings. // Add an extra line break if the string starts with one, since // the code to read default values from the DOM strips the leading one. String value = defaultValue; value.replace("\r\n", "\n"); value.replace('\r', '\n'); if (value[0] == '\n') value = "\n" + value; insertBefore(document()->createTextNode(value), firstChild(), ec); setValue(value);}void HTMLTextAreaElement::accessKeyAction(bool){ focus();}const AtomicString& HTMLTextAreaElement::accessKey() const{ return getAttribute(accesskeyAttr);}void HTMLTextAreaElement::setAccessKey(const String& value){ setAttribute(accesskeyAttr, value);}void HTMLTextAreaElement::setCols(int cols){ setAttribute(colsAttr, String::number(cols));}void HTMLTextAreaElement::setRows(int rows){ setAttribute(rowsAttr, String::number(rows));}VisibleSelection HTMLTextAreaElement::selection() const{ if (!renderer() || m_cachedSelectionStart < 0 || m_cachedSelectionEnd < 0) return VisibleSelection(); return toRenderTextControl(renderer())->selection(m_cachedSelectionStart, m_cachedSelectionEnd);}bool HTMLTextAreaElement::shouldUseInputMethod() const{ return true;}} // namespace
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -