/* * This file is part of the DOM implementation for KDE. * * 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 Apple Computer, Inc. * (C) 2006 Alexey Proskuryakov (ap@nypop.com) * * 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 "HTMLOptionElement.h" #include "CSSStyleSelector.h" #include "Document.h" #include "ExceptionCode.h" #include "HTMLNames.h" #include "HTMLSelectElement.h" #include "RenderMenuList.h" #include "Text.h" #include "NodeRenderStyle.h" #include namespace WebCore { using namespace HTMLNames; HTMLOptionElement::HTMLOptionElement(Document* doc, HTMLFormElement* f) : HTMLFormControlElement(optionTag, doc, f) , m_selected(false) , m_style(0) { } bool HTMLOptionElement::checkDTD(const Node* newChild) { return newChild->isTextNode() || newChild->hasTagName(scriptTag); } void HTMLOptionElement::attach() { if (parentNode()->renderStyle()) setRenderStyle(styleForRenderer()); HTMLFormControlElement::attach(); } void HTMLOptionElement::detach() { m_style.clear(); HTMLFormControlElement::detach(); } bool HTMLOptionElement::isFocusable() const { return HTMLElement::isFocusable(); } const AtomicString& HTMLOptionElement::type() const { static const AtomicString option("option"); return option; } String HTMLOptionElement::text() const { String text; // WinIE does not use the label attribute, so as a quirk, we ignore it. if (!document()->inCompatMode()) text = getAttribute(labelAttr); if (text.isEmpty()) { const Node* n = firstChild(); while (n) { if (n->nodeType() == TEXT_NODE || n->nodeType() == CDATA_SECTION_NODE) text += n->nodeValue(); // skip script content if (n->isElementNode() && n->hasTagName(HTMLNames::scriptTag)) n = n->traverseNextSibling(this); else n = n->traverseNextNode(this); } } text.replace('\\', document()->backslashAsCurrencySymbol()); // In WinIE, leading and trailing whitespace is ignored in options and optgroups. We match this behavior. text = text.stripWhiteSpace(); // We want to collapse our whitespace too. This will match other browsers. text = text.simplifyWhiteSpace(); return text; } void HTMLOptionElement::setText(const String &text, ExceptionCode& ec) { // Handle the common special case where there's exactly 1 child node, and it's a text node. Node* child = firstChild(); if (child && child->isTextNode() && !child->nextSibling()) { static_cast(child)->setData(text, ec); return; } removeChildren(); appendChild(new Text(document(), text), ec); } void HTMLOptionElement::accessKeyAction(bool sendToAnyElement) { HTMLSelectElement* select = ownerSelectElement(); if (select) select->accessKeySetSelectedIndex(index()); } int HTMLOptionElement::index() const { // Let's do this dynamically. Might be a bit slow, but we're sure // we won't forget to update a member variable in some cases... HTMLSelectElement* select = ownerSelectElement(); if (select) { const Vector& items = select->listItems(); int l = items.size(); int optionIndex = 0; for(int i = 0; i < l; i++) { if (items[i]->hasLocalName(optionTag)) { if (static_cast(items[i]) == this) return optionIndex; optionIndex++; } } } return 0; } void HTMLOptionElement::parseMappedAttribute(MappedAttribute *attr) { if (attr->name() == selectedAttr) m_selected = (!attr->isNull()); else if (attr->name() == valueAttr) m_value = attr->value(); else HTMLFormControlElement::parseMappedAttribute(attr); } String HTMLOptionElement::value() const { if ( !m_value.isNull() ) return m_value; // Use the text if the value wasn't set. return text().stripWhiteSpace(); } void HTMLOptionElement::setValue(const String& value) { setAttribute(valueAttr, value); } void HTMLOptionElement::setSelected(bool selected) { if (m_selected == selected) return; if (HTMLSelectElement* select = ownerSelectElement()) select->setSelectedIndex(selected ? index() : -1, false); m_selected = selected; } void HTMLOptionElement::setSelectedState(bool selected) { if (m_selected == selected) return; m_selected = selected; setChanged(); } void HTMLOptionElement::childrenChanged(bool changedByParser, Node* beforeChange, Node* afterChange, int childCountDelta) { HTMLSelectElement* select = ownerSelectElement(); if (select) select->childrenChanged(changedByParser); HTMLFormControlElement::childrenChanged(changedByParser, beforeChange, afterChange, childCountDelta); } HTMLSelectElement* HTMLOptionElement::ownerSelectElement() const { Node* select = parentNode(); while (select && !select->hasTagName(selectTag)) select = select->parentNode(); if (!select) return 0; return static_cast(select); } bool HTMLOptionElement::defaultSelected() const { return !getAttribute(selectedAttr).isNull(); } void HTMLOptionElement::setDefaultSelected(bool b) { setAttribute(selectedAttr, b ? "" : 0); } String HTMLOptionElement::label() const { return getAttribute(labelAttr); } void HTMLOptionElement::setLabel(const String& value) { setAttribute(labelAttr, value); } void HTMLOptionElement::setRenderStyle(PassRefPtr newStyle) { m_style = newStyle; } RenderStyle* HTMLOptionElement::nonRendererRenderStyle() const { return m_style.get(); } String HTMLOptionElement::optionText() { if (parentNode() && parentNode()->hasTagName(optgroupTag)) return " " + text(); return text(); } bool HTMLOptionElement::disabled() const { return HTMLFormControlElement::disabled() || (parentNode() && static_cast(parentNode())->disabled()); } void HTMLOptionElement::insertedIntoDocument() { HTMLSelectElement* select; if (selected() && (select = ownerSelectElement())) select->scrollToSelection(); HTMLFormControlElement::insertedIntoDocument(); } } // namespace