blob: 97b2812dabd1d8cfd2301b2347c648045eae851d [file] [log] [blame]
/*
* Copyright (C) 2012 Company 100, 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 GOOGLE 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 GOOGLE 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.
*/
#ifndef HTMLStackItem_h
#define HTMLStackItem_h
#include "core/HTMLNames.h"
#include "core/MathMLNames.h"
#include "core/SVGNames.h"
#include "core/dom/Element.h"
#include "core/html/parser/AtomicHTMLToken.h"
#include "platform/RuntimeEnabledFeatures.h"
#include "platform/wtf/text/AtomicString.h"
namespace blink {
class ContainerNode;
class HTMLStackItem : public GarbageCollectedFinalized<HTMLStackItem> {
public:
enum ItemType { kItemForContextElement, kItemForDocumentFragmentNode };
// Used by document fragment node and context element.
static HTMLStackItem* Create(ContainerNode* node, ItemType type) {
return new HTMLStackItem(node, type);
}
// Used by HTMLElementStack and HTMLFormattingElementList.
static HTMLStackItem* Create(
ContainerNode* node,
AtomicHTMLToken* token,
const AtomicString& namespace_uri = HTMLNames::xhtmlNamespaceURI) {
return new HTMLStackItem(node, token, namespace_uri);
}
Element* GetElement() const { return ToElement(node_.Get()); }
ContainerNode* GetNode() const { return node_.Get(); }
bool IsDocumentFragmentNode() const { return is_document_fragment_node_; }
bool IsElementNode() const { return !is_document_fragment_node_; }
const AtomicString& NamespaceURI() const { return namespace_uri_; }
const AtomicString& LocalName() const { return token_local_name_; }
const Vector<Attribute>& Attributes() const {
DCHECK(token_local_name_);
return token_attributes_;
}
Attribute* GetAttributeItem(const QualifiedName& attribute_name) {
DCHECK(token_local_name_);
return FindAttributeInVector(token_attributes_, attribute_name);
}
bool HasLocalName(const AtomicString& name) const {
return token_local_name_ == name;
}
bool HasTagName(const QualifiedName& name) const {
return token_local_name_ == name.LocalName() &&
namespace_uri_ == name.NamespaceURI();
}
bool MatchesHTMLTag(const AtomicString& name) const {
return token_local_name_ == name &&
namespace_uri_ == HTMLNames::xhtmlNamespaceURI;
}
bool MatchesHTMLTag(const QualifiedName& name) const {
return token_local_name_ == name &&
namespace_uri_ == HTMLNames::xhtmlNamespaceURI;
}
bool CausesFosterParenting() {
return HasTagName(HTMLNames::tableTag) || HasTagName(HTMLNames::tbodyTag) ||
HasTagName(HTMLNames::tfootTag) || HasTagName(HTMLNames::theadTag) ||
HasTagName(HTMLNames::trTag);
}
bool IsInHTMLNamespace() const {
// A DocumentFragment takes the place of the document element when parsing
// fragments and should be considered in the HTML namespace.
return NamespaceURI() == HTMLNames::xhtmlNamespaceURI ||
IsDocumentFragmentNode(); // FIXME: Does this also apply to
// ShadowRoot?
}
bool IsNumberedHeaderElement() const {
return HasTagName(HTMLNames::h1Tag) || HasTagName(HTMLNames::h2Tag) ||
HasTagName(HTMLNames::h3Tag) || HasTagName(HTMLNames::h4Tag) ||
HasTagName(HTMLNames::h5Tag) || HasTagName(HTMLNames::h6Tag);
}
bool IsTableBodyContextElement() const {
return HasTagName(HTMLNames::tbodyTag) || HasTagName(HTMLNames::tfootTag) ||
HasTagName(HTMLNames::theadTag);
}
// http://www.whatwg.org/specs/web-apps/current-work/multipage/parsing.html#special
bool IsSpecialNode() const {
if (HasTagName(MathMLNames::miTag) || HasTagName(MathMLNames::moTag) ||
HasTagName(MathMLNames::mnTag) || HasTagName(MathMLNames::msTag) ||
HasTagName(MathMLNames::mtextTag) ||
HasTagName(MathMLNames::annotation_xmlTag) ||
HasTagName(SVGNames::foreignObjectTag) ||
HasTagName(SVGNames::descTag) || HasTagName(SVGNames::titleTag))
return true;
if (IsDocumentFragmentNode())
return true;
if (!IsInHTMLNamespace())
return false;
const AtomicString& tag_name = LocalName();
return tag_name == HTMLNames::addressTag ||
tag_name == HTMLNames::areaTag || tag_name == HTMLNames::appletTag ||
tag_name == HTMLNames::articleTag ||
tag_name == HTMLNames::asideTag || tag_name == HTMLNames::baseTag ||
tag_name == HTMLNames::basefontTag ||
tag_name == HTMLNames::bgsoundTag ||
tag_name == HTMLNames::blockquoteTag ||
tag_name == HTMLNames::bodyTag || tag_name == HTMLNames::brTag ||
tag_name == HTMLNames::buttonTag ||
tag_name == HTMLNames::captionTag ||
tag_name == HTMLNames::centerTag || tag_name == HTMLNames::colTag ||
tag_name == HTMLNames::colgroupTag ||
tag_name == HTMLNames::commandTag || tag_name == HTMLNames::ddTag ||
tag_name == HTMLNames::detailsTag || tag_name == HTMLNames::dirTag ||
tag_name == HTMLNames::divTag || tag_name == HTMLNames::dlTag ||
tag_name == HTMLNames::dtTag || tag_name == HTMLNames::embedTag ||
tag_name == HTMLNames::fieldsetTag ||
tag_name == HTMLNames::figcaptionTag ||
tag_name == HTMLNames::figureTag ||
tag_name == HTMLNames::footerTag || tag_name == HTMLNames::formTag ||
tag_name == HTMLNames::frameTag ||
tag_name == HTMLNames::framesetTag || IsNumberedHeaderElement() ||
tag_name == HTMLNames::headTag || tag_name == HTMLNames::headerTag ||
tag_name == HTMLNames::hgroupTag || tag_name == HTMLNames::hrTag ||
tag_name == HTMLNames::htmlTag || tag_name == HTMLNames::iframeTag ||
tag_name == HTMLNames::imgTag || tag_name == HTMLNames::inputTag ||
tag_name == HTMLNames::liTag || tag_name == HTMLNames::linkTag ||
tag_name == HTMLNames::listingTag ||
tag_name == HTMLNames::mainTag ||
tag_name == HTMLNames::marqueeTag ||
tag_name == HTMLNames::menuTag ||
tag_name == HTMLNames::metaTag || tag_name == HTMLNames::navTag ||
tag_name == HTMLNames::noembedTag ||
tag_name == HTMLNames::noframesTag ||
tag_name == HTMLNames::noscriptTag ||
tag_name == HTMLNames::objectTag || tag_name == HTMLNames::olTag ||
tag_name == HTMLNames::pTag || tag_name == HTMLNames::paramTag ||
tag_name == HTMLNames::plaintextTag ||
tag_name == HTMLNames::preTag || tag_name == HTMLNames::scriptTag ||
tag_name == HTMLNames::sectionTag ||
tag_name == HTMLNames::selectTag ||
tag_name == HTMLNames::styleTag ||
tag_name == HTMLNames::summaryTag ||
tag_name == HTMLNames::tableTag || IsTableBodyContextElement() ||
tag_name == HTMLNames::tdTag || tag_name == HTMLNames::templateTag ||
tag_name == HTMLNames::textareaTag || tag_name == HTMLNames::thTag ||
tag_name == HTMLNames::titleTag || tag_name == HTMLNames::trTag ||
tag_name == HTMLNames::ulTag || tag_name == HTMLNames::wbrTag ||
tag_name == HTMLNames::xmpTag;
}
DEFINE_INLINE_TRACE() { visitor->Trace(node_); }
private:
HTMLStackItem(ContainerNode* node, ItemType type) : node_(node) {
switch (type) {
case kItemForDocumentFragmentNode:
is_document_fragment_node_ = true;
break;
case kItemForContextElement:
token_local_name_ = GetElement()->localName();
namespace_uri_ = GetElement()->namespaceURI();
is_document_fragment_node_ = false;
break;
}
}
HTMLStackItem(
ContainerNode* node,
AtomicHTMLToken* token,
const AtomicString& namespace_uri = HTMLNames::xhtmlNamespaceURI)
: node_(node),
token_local_name_(token->GetName()),
token_attributes_(token->Attributes()),
namespace_uri_(namespace_uri),
is_document_fragment_node_(false) {}
Member<ContainerNode> node_;
AtomicString token_local_name_;
Vector<Attribute> token_attributes_;
AtomicString namespace_uri_;
bool is_document_fragment_node_;
};
} // namespace blink
#endif // HTMLStackItem_h