blob: f25bc95720800ea7f3b294502d610d094bd9a194 [file] [log] [blame]
/*
* Copyright (C) 2019 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "HTMLDialogElement.h"
#include "CSSSelector.h"
#include "DocumentInlines.h"
#include "EventLoop.h"
#include "EventNames.h"
#include "FocusOptions.h"
#include "HTMLNames.h"
#include "PseudoClassChangeInvalidation.h"
#include "RenderElement.h"
#include "ScopedEventQueue.h"
#include "TypedElementDescendantIterator.h"
#include <wtf/IsoMallocInlines.h>
namespace WebCore {
WTF_MAKE_ISO_ALLOCATED_IMPL(HTMLDialogElement);
using namespace HTMLNames;
HTMLDialogElement::HTMLDialogElement(const QualifiedName& tagName, Document& document)
: HTMLElement(tagName, document)
{
}
void HTMLDialogElement::show()
{
// If the element already has an open attribute, then return.
if (isOpen())
return;
setBooleanAttribute(openAttr, true);
m_previouslyFocusedElement = document().focusedElement();
runFocusingSteps();
}
ExceptionOr<void> HTMLDialogElement::showModal()
{
// If subject already has an open attribute, then throw an "InvalidStateError" DOMException.
if (isOpen())
return Exception { InvalidStateError };
// If subject is not connected, then throw an "InvalidStateError" DOMException.
if (!isConnected())
return Exception { InvalidStateError };
// setBooleanAttribute will dispatch a DOMSubtreeModified event.
// Postpone callback execution that can potentially make the dialog disconnected.
EventQueueScope scope;
setBooleanAttribute(openAttr, true);
setIsModal(true);
if (!isInTopLayer())
addToTopLayer();
m_previouslyFocusedElement = document().focusedElement();
runFocusingSteps();
return { };
}
void HTMLDialogElement::close(const String& result)
{
if (!isOpen())
return;
setBooleanAttribute(openAttr, false);
setIsModal(false);
if (!result.isNull())
m_returnValue = result;
if (isInTopLayer())
removeFromTopLayer();
if (RefPtr element = std::exchange(m_previouslyFocusedElement, nullptr).get()) {
FocusOptions options;
options.preventScroll = true;
element->focus(options);
}
queueTaskToDispatchEvent(TaskSource::UserInteraction, Event::create(eventNames().closeEvent, Event::CanBubble::No, Event::IsCancelable::No));
}
void HTMLDialogElement::queueCancelTask()
{
queueTaskKeepingThisNodeAlive(TaskSource::UserInteraction, [this] {
auto cancelEvent = Event::create(eventNames().cancelEvent, Event::CanBubble::No, Event::IsCancelable::Yes);
dispatchEvent(cancelEvent);
if (!cancelEvent->defaultPrevented())
close(nullString());
});
}
// https://html.spec.whatwg.org/multipage/interactive-elements.html#dialog-focusing-steps
void HTMLDialogElement::runFocusingSteps()
{
RefPtr<Element> control;
for (auto& element : descendantsOfType<Element>(*this)) {
if (!element.isFocusable())
continue;
if (element.hasAttribute(autofocusAttr)) {
control = &element;
break;
}
// FIXME: Potentially remove this and adjust related WPTs after https://github.com/whatwg/html/pull/4184.
if (!control)
control = &element;
}
if (!control)
control = this;
if (control->isFocusable())
control->runFocusingStepsForAutofocus();
else if (m_isModal)
document().setFocusedElement(nullptr); // Focus fixup rule
if (!control->document().isSameOriginAsTopDocument())
return;
Ref topDocument = control->document().topDocument();
topDocument->clearAutofocusCandidates();
topDocument->setAutofocusProcessed();
}
void HTMLDialogElement::removedFromAncestor(RemovalType removalType, ContainerNode& oldParentOfRemovedTree)
{
HTMLElement::removedFromAncestor(removalType, oldParentOfRemovedTree);
setIsModal(false);
}
void HTMLDialogElement::setIsModal(bool newValue)
{
if (m_isModal == newValue)
return;
Style::PseudoClassChangeInvalidation styleInvalidation(*this, CSSSelector::PseudoClassModal, newValue);
m_isModal = newValue;
}
}