e2957b21eceefe4693bf9dac4a6c7a03c0ea0f24
[WebKit-https.git] / Source / WebCore / html / HTMLSummaryElement.cpp
1 /*
2  * Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies)
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Library General Public
6  * License as published by the Free Software Foundation; either
7  * version 2 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Library General Public License for more details.
13  *
14  * You should have received a copy of the GNU Library General Public License
15  * along with this library; see the file COPYING.LIB.  If not, write to
16  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
17  * Boston, MA 02110-1301, USA.
18  *
19  */
20
21 #include "config.h"
22 #include "HTMLSummaryElement.h"
23
24 #if ENABLE(DETAILS)
25
26 #include "DetailsMarkerControl.h"
27 #include "HTMLContentElement.h"
28 #include "HTMLDetailsElement.h"
29 #include "KeyboardEvent.h"
30 #include "HTMLNames.h"
31 #include "MouseEvent.h"
32 #include "PlatformMouseEvent.h"
33 #include "RenderSummary.h"
34 #include "ShadowRoot.h"
35 #include "ShadowRootList.h"
36
37 namespace WebCore {
38
39 using namespace HTMLNames;
40
41 class SummaryContentElement : public HTMLContentElement {
42 public:
43     static PassRefPtr<SummaryContentElement> create(Document*);
44
45 private:
46     SummaryContentElement(Document* document)
47         : HTMLContentElement(HTMLNames::divTag, document)
48     {
49     }
50 };
51
52 PassRefPtr<SummaryContentElement> SummaryContentElement::create(Document* document)
53 {
54     return adoptRef(new SummaryContentElement(document));
55 }
56
57 PassRefPtr<HTMLSummaryElement> HTMLSummaryElement::create(const QualifiedName& tagName, Document* document)
58 {
59     RefPtr<HTMLSummaryElement> result = adoptRef(new HTMLSummaryElement(tagName, document));
60     result->createShadowSubtree();
61     return result;
62 }
63
64 HTMLSummaryElement::HTMLSummaryElement(const QualifiedName& tagName, Document* document)
65     : HTMLElement(tagName, document)
66 {
67     ASSERT(hasTagName(summaryTag));
68 }
69
70 RenderObject* HTMLSummaryElement::createRenderer(RenderArena* arena, RenderStyle*)
71 {
72     return new (arena) RenderSummary(this);
73 }
74
75 void HTMLSummaryElement::createShadowSubtree()
76 {
77     ASSERT(!hasShadowRoot());
78     RefPtr<ShadowRoot> root = ShadowRoot::create(this, ShadowRoot::CreatingUserAgentShadowRoot);
79     root->appendChild(DetailsMarkerControl::create(document()), ASSERT_NO_EXCEPTION, true);
80     root->appendChild(SummaryContentElement::create(document()), ASSERT_NO_EXCEPTION, true);
81 }
82
83 HTMLDetailsElement* HTMLSummaryElement::detailsElement() const
84 {
85     Node* mayDetails = const_cast<HTMLSummaryElement*>(this)->parentNodeForRenderingAndStyle();
86     if (!mayDetails || !mayDetails->hasTagName(detailsTag))
87         return 0;
88     return static_cast<HTMLDetailsElement*>(mayDetails);
89 }
90
91 bool HTMLSummaryElement::isMainSummary() const
92 {
93     if (HTMLDetailsElement* details = detailsElement())
94         return details->findMainSummary() == this;
95
96     return false;
97 }
98
99 static bool isClickableControl(Node* node)
100 {
101     if (!node->isElementNode())
102         return false;
103     Element* element = toElement(node);
104     if (element->isFormControlElement())
105         return true;
106     Element* host = toElement(element->shadowAncestorNode());
107     return host && host->isFormControlElement();
108 }
109
110 bool HTMLSummaryElement::supportsFocus() const
111 {
112     return isMainSummary();
113 }
114
115 void HTMLSummaryElement::defaultEventHandler(Event* event)
116 {
117     if (isMainSummary() && renderer() && renderer()->isSummary()) {
118         if (event->type() == eventNames().DOMActivateEvent && !isClickableControl(event->target()->toNode())) {
119             if (HTMLDetailsElement* details = detailsElement())
120                 details->toggleOpen();
121             event->setDefaultHandled();
122             return;
123         }
124
125         if (event->isKeyboardEvent()) {
126             if (event->type() == eventNames().keydownEvent && static_cast<KeyboardEvent*>(event)->keyIdentifier() == "U+0020") {
127                 setActive(true, true);
128                 // No setDefaultHandled() - IE dispatches a keypress in this case.
129                 return;
130             }
131             if (event->type() == eventNames().keypressEvent) {
132                 switch (static_cast<KeyboardEvent*>(event)->charCode()) {
133                 case '\r':
134                     dispatchSimulatedClick(event);
135                     event->setDefaultHandled();
136                     return;
137                 case ' ':
138                     // Prevent scrolling down the page.
139                     event->setDefaultHandled();
140                     return;
141                 }
142             }
143             if (event->type() == eventNames().keyupEvent && static_cast<KeyboardEvent*>(event)->keyIdentifier() == "U+0020") {
144                 if (active())
145                     dispatchSimulatedClick(event);
146                 event->setDefaultHandled();
147                 return;
148             }
149         }
150     }
151
152     HTMLElement::defaultEventHandler(event);
153 }
154
155 }
156
157 #endif