blob: c8d3e78becd688479bf5b9be7d00830a56506b40 [file] [log] [blame]
/*
* 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"
#include "DetailsMarkerControl.h"
#include "ElementInlines.h"
#include "EventNames.h"
#include "HTMLDetailsElement.h"
#include "HTMLFormControlElement.h"
#include "HTMLSlotElement.h"
#include "KeyboardEvent.h"
#include "MouseEvent.h"
#include "PlatformMouseEvent.h"
#include "RenderBlockFlow.h"
#include "ShadowRoot.h"
#include "SlotAssignment.h"
#include <wtf/IsoMallocInlines.h>
namespace WebCore {
WTF_MAKE_ISO_ALLOCATED_IMPL(HTMLSummaryElement);
using namespace HTMLNames;
class SummarySlotElement final : public SlotAssignment {
private:
void hostChildElementDidChange(const Element&, ShadowRoot& shadowRoot) override
{
didChangeSlot(SlotAssignment::defaultSlotName(), shadowRoot);
}
const AtomString& slotNameForHostChild(const Node&) const override { return SlotAssignment::defaultSlotName(); }
};
Ref<HTMLSummaryElement> HTMLSummaryElement::create(const QualifiedName& tagName, Document& document)
{
Ref<HTMLSummaryElement> summary = adoptRef(*new HTMLSummaryElement(tagName, document));
summary->addShadowRoot(ShadowRoot::create(document, makeUnique<SummarySlotElement>()));
return summary;
}
HTMLSummaryElement::HTMLSummaryElement(const QualifiedName& tagName, Document& document)
: HTMLElement(tagName, document)
{
ASSERT(hasTagName(summaryTag));
}
RenderPtr<RenderElement> HTMLSummaryElement::createElementRenderer(RenderStyle&& style, const RenderTreePosition&)
{
// <summary> elements with display:list-item should not be rendered as list items because they'd end up with two markers before the text (one from summary element and the other as a list item).
return RenderElement::createFor(*this, WTFMove(style), { RenderElement::ConstructBlockLevelRendererFor::ListItem, RenderElement::ConstructBlockLevelRendererFor::Inline, RenderElement::ConstructBlockLevelRendererFor::TableOrTablePart });
}
void HTMLSummaryElement::didAddUserAgentShadowRoot(ShadowRoot& root)
{
root.appendChild(DetailsMarkerControl::create(document()));
root.appendChild(HTMLSlotElement::create(slotTag, document()));
}
RefPtr<HTMLDetailsElement> HTMLSummaryElement::detailsElement() const
{
auto* parent = parentElement();
if (parent && is<HTMLDetailsElement>(*parent))
return downcast<HTMLDetailsElement>(parent);
// Fallback summary element is in the shadow tree.
auto* host = shadowHost();
if (host && is<HTMLDetailsElement>(*host))
return downcast<HTMLDetailsElement>(host);
return nullptr;
}
bool HTMLSummaryElement::isActiveSummary() const
{
RefPtr<HTMLDetailsElement> details = detailsElement();
if (!details)
return false;
return details->isActiveSummary(*this);
}
static bool isClickableControl(EventTarget* target)
{
if (!is<Element>(target))
return false;
auto& element = downcast<Element>(*target);
return is<HTMLFormControlElement>(element) || is<HTMLFormControlElement>(element.shadowHost());
}
int HTMLSummaryElement::defaultTabIndex() const
{
return isActiveSummary() ? 0 : -1;
}
bool HTMLSummaryElement::supportsFocus() const
{
return isActiveSummary();
}
void HTMLSummaryElement::defaultEventHandler(Event& event)
{
if (isActiveSummary() && renderer()) {
auto& eventNames = WebCore::eventNames();
if (event.type() == eventNames.DOMActivateEvent && !isClickableControl(event.target())) {
if (RefPtr<HTMLDetailsElement> details = detailsElement())
details->toggleOpen();
event.setDefaultHandled();
return;
}
if (is<KeyboardEvent>(event)) {
KeyboardEvent& keyboardEvent = downcast<KeyboardEvent>(event);
if (keyboardEvent.type() == eventNames.keydownEvent && keyboardEvent.keyIdentifier() == "U+0020"_s) {
setActive(true, true);
// No setDefaultHandled() - IE dispatches a keypress in this case.
return;
}
if (keyboardEvent.type() == eventNames.keypressEvent) {
switch (keyboardEvent.charCode()) {
case '\r':
dispatchSimulatedClick(&event);
keyboardEvent.setDefaultHandled();
return;
case ' ':
// Prevent scrolling down the page.
keyboardEvent.setDefaultHandled();
return;
}
}
if (keyboardEvent.type() == eventNames.keyupEvent && keyboardEvent.keyIdentifier() == "U+0020"_s) {
if (active())
dispatchSimulatedClick(&event);
keyboardEvent.setDefaultHandled();
return;
}
}
}
HTMLElement::defaultEventHandler(event);
}
bool HTMLSummaryElement::willRespondToMouseClickEventsWithEditability(Editability editability) const
{
if (isActiveSummary() && renderer())
return true;
return HTMLElement::willRespondToMouseClickEventsWithEditability(editability);
}
}