68660963bbacbfd392a870427328280e57a80910
[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 #include "DetailsMarkerControl.h"
25 #include "EventNames.h"
26 #include "HTMLDetailsElement.h"
27 #include "HTMLFormControlElement.h"
28 #include "HTMLSlotElement.h"
29 #include "KeyboardEvent.h"
30 #include "MouseEvent.h"
31 #include "PlatformMouseEvent.h"
32 #include "RenderBlockFlow.h"
33 #include "ShadowRoot.h"
34 #include "SlotAssignment.h"
35
36 namespace WebCore {
37
38 using namespace HTMLNames;
39
40 class SummarySlotElement final : public SlotAssignment {
41 private:
42     void hostChildElementDidChange(const Element&, ShadowRoot& shadowRoot) override
43     {
44         didChangeSlot(SlotAssignment::defaultSlotName(), shadowRoot);
45     }
46
47     const AtomicString& slotNameForHostChild(const Node&) const override { return SlotAssignment::defaultSlotName(); }
48 };
49
50 Ref<HTMLSummaryElement> HTMLSummaryElement::create(const QualifiedName& tagName, Document& document)
51 {
52     Ref<HTMLSummaryElement> summary = adoptRef(*new HTMLSummaryElement(tagName, document));
53     summary->addShadowRoot(ShadowRoot::create(document, std::make_unique<SummarySlotElement>()));
54     return summary;
55 }
56
57 HTMLSummaryElement::HTMLSummaryElement(const QualifiedName& tagName, Document& document)
58     : HTMLElement(tagName, document)
59 {
60     ASSERT(hasTagName(summaryTag));
61 }
62
63 RenderPtr<RenderElement> HTMLSummaryElement::createElementRenderer(RenderStyle&& style, const RenderTreePosition&)
64 {
65     return createRenderer<RenderBlockFlow>(*this, WTFMove(style));
66 }
67
68 void HTMLSummaryElement::didAddUserAgentShadowRoot(ShadowRoot& root)
69 {
70     root.appendChild(DetailsMarkerControl::create(document()));
71     root.appendChild(HTMLSlotElement::create(slotTag, document()));
72 }
73
74 HTMLDetailsElement* HTMLSummaryElement::detailsElement() const
75 {
76     auto* parent = parentElement();
77     if (parent && is<HTMLDetailsElement>(*parent))
78         return downcast<HTMLDetailsElement>(parent);
79     // Fallback summary element is in the shadow tree.
80     auto* host = shadowHost();
81     if (host && is<HTMLDetailsElement>(*host))
82         return downcast<HTMLDetailsElement>(host);
83     return nullptr;
84 }
85
86 bool HTMLSummaryElement::isActiveSummary() const
87 {
88     RefPtr<HTMLDetailsElement> details = detailsElement();
89     if (!details)
90         return false;
91     return details->isActiveSummary(*this);
92 }
93
94 static bool isClickableControl(Node* node)
95 {
96     ASSERT(node);
97     if (!is<Element>(*node))
98         return false;
99     Element& element = downcast<Element>(*node);
100     if (is<HTMLFormControlElement>(element))
101         return true;
102     RefPtr<Element> host = element.shadowHost();
103     return host && is<HTMLFormControlElement>(host);
104 }
105
106 bool HTMLSummaryElement::supportsFocus() const
107 {
108     return isActiveSummary();
109 }
110
111 void HTMLSummaryElement::defaultEventHandler(Event& event)
112 {
113     if (isActiveSummary() && renderer()) {
114         if (event.type() == eventNames().DOMActivateEvent && !isClickableControl(event.target()->toNode().get())) {
115             if (RefPtr<HTMLDetailsElement> details = detailsElement())
116                 details->toggleOpen();
117             event.setDefaultHandled();
118             return;
119         }
120
121         if (is<KeyboardEvent>(event)) {
122             KeyboardEvent& keyboardEvent = downcast<KeyboardEvent>(event);
123             if (keyboardEvent.type() == eventNames().keydownEvent && keyboardEvent.keyIdentifier() == "U+0020") {
124                 setActive(true, true);
125                 // No setDefaultHandled() - IE dispatches a keypress in this case.
126                 return;
127             }
128             if (keyboardEvent.type() == eventNames().keypressEvent) {
129                 switch (keyboardEvent.charCode()) {
130                 case '\r':
131                     dispatchSimulatedClick(&event);
132                     keyboardEvent.setDefaultHandled();
133                     return;
134                 case ' ':
135                     // Prevent scrolling down the page.
136                     keyboardEvent.setDefaultHandled();
137                     return;
138                 }
139             }
140             if (keyboardEvent.type() == eventNames().keyupEvent && keyboardEvent.keyIdentifier() == "U+0020") {
141                 if (active())
142                     dispatchSimulatedClick(&event);
143                 keyboardEvent.setDefaultHandled();
144                 return;
145             }
146         }
147     }
148
149     HTMLElement::defaultEventHandler(event);
150 }
151
152 bool HTMLSummaryElement::willRespondToMouseClickEvents()
153 {
154     if (isActiveSummary() && renderer())
155         return true;
156
157     return HTMLElement::willRespondToMouseClickEvents();
158 }
159
160 }