/* * Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies) * * 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 "HTMLSummaryElement.h" #if ENABLE(DETAILS_ELEMENT) #include "DetailsMarkerControl.h" #include "ElementShadow.h" #include "HTMLContentElement.h" #include "HTMLDetailsElement.h" #include "HTMLNames.h" #include "KeyboardEvent.h" #include "MouseEvent.h" #include "NodeRenderingContext.h" #include "PlatformMouseEvent.h" #include "RenderBlock.h" #include "ShadowRoot.h" namespace WebCore { using namespace HTMLNames; class SummaryContentElement : public HTMLContentElement { public: static PassRefPtr create(Document*); private: SummaryContentElement(Document* document) : HTMLContentElement(HTMLNames::webkitShadowContentTag, document) { } }; PassRefPtr SummaryContentElement::create(Document* document) { return adoptRef(new SummaryContentElement(document)); } PassRefPtr HTMLSummaryElement::create(const QualifiedName& tagName, Document* document) { RefPtr result = adoptRef(new HTMLSummaryElement(tagName, document)); result->createShadowSubtree(); return result; } HTMLSummaryElement::HTMLSummaryElement(const QualifiedName& tagName, Document* document) : HTMLElement(tagName, document) { ASSERT(hasTagName(summaryTag)); } RenderObject* HTMLSummaryElement::createRenderer(RenderArena* arena, RenderStyle*) { return new (arena) RenderBlock(this); } bool HTMLSummaryElement::childShouldCreateRenderer(const NodeRenderingContext& childContext) const { return childContext.isOnEncapsulationBoundary() && HTMLElement::childShouldCreateRenderer(childContext); } void HTMLSummaryElement::createShadowSubtree() { ASSERT(!shadow()); RefPtr root = ShadowRoot::create(this, ShadowRoot::UserAgentShadowRoot); root->appendChild(DetailsMarkerControl::create(document()), ASSERT_NO_EXCEPTION, true); root->appendChild(SummaryContentElement::create(document()), ASSERT_NO_EXCEPTION, true); } HTMLDetailsElement* HTMLSummaryElement::detailsElement() const { Node* mayDetails = const_cast(this)->parentNodeForRenderingAndStyle(); if (!mayDetails || !mayDetails->hasTagName(detailsTag)) return 0; return static_cast(mayDetails); } bool HTMLSummaryElement::isMainSummary() const { if (HTMLDetailsElement* details = detailsElement()) return details->findMainSummary() == this; return false; } static bool isClickableControl(Node* node) { if (!node->isElementNode()) return false; Element* element = toElement(node); if (element->isFormControlElement()) return true; Element* host = element->shadowHost(); return host && host->isFormControlElement(); } bool HTMLSummaryElement::supportsFocus() const { return isMainSummary(); } void HTMLSummaryElement::defaultEventHandler(Event* event) { if (isMainSummary() && renderer()) { if (event->type() == eventNames().DOMActivateEvent && !isClickableControl(event->target()->toNode())) { if (HTMLDetailsElement* details = detailsElement()) details->toggleOpen(); event->setDefaultHandled(); return; } if (event->isKeyboardEvent()) { if (event->type() == eventNames().keydownEvent && static_cast(event)->keyIdentifier() == "U+0020") { setActive(true, true); // No setDefaultHandled() - IE dispatches a keypress in this case. return; } if (event->type() == eventNames().keypressEvent) { switch (static_cast(event)->charCode()) { case '\r': dispatchSimulatedClick(event); event->setDefaultHandled(); return; case ' ': // Prevent scrolling down the page. event->setDefaultHandled(); return; } } if (event->type() == eventNames().keyupEvent && static_cast(event)->keyIdentifier() == "U+0020") { if (active()) dispatchSimulatedClick(event); event->setDefaultHandled(); return; } } } HTMLElement::defaultEventHandler(event); } bool HTMLSummaryElement::willRespondToMouseClickEvents() { if (isMainSummary() && renderer()) return true; return HTMLElement::willRespondToMouseClickEvents(); } } #endif