⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 htmltextareaelement.cpp

📁 linux下开源浏览器WebKit的源码,市面上的很多商用浏览器都是移植自WebKit
💻 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 + -