2015-11-25 04:14:39 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
#include "mozilla/ServoBindings.h"
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
#include "StyleStructContext.h"
|
2016-06-07 22:13:24 +03:00
|
|
|
#include "gfxFontFamilyList.h"
|
2016-05-25 21:16:26 +03:00
|
|
|
#include "nsAttrValueInlines.h"
|
2015-11-25 04:14:39 +03:00
|
|
|
#include "nsCSSRuleProcessor.h"
|
|
|
|
#include "nsContentUtils.h"
|
2016-05-25 21:16:26 +03:00
|
|
|
#include "nsDOMTokenList.h"
|
2015-11-25 04:14:39 +03:00
|
|
|
#include "nsIDOMNode.h"
|
|
|
|
#include "nsIDocument.h"
|
|
|
|
#include "nsINode.h"
|
2016-05-21 03:02:54 +03:00
|
|
|
#include "nsIPrincipal.h"
|
2015-11-25 04:14:39 +03:00
|
|
|
#include "nsNameSpaceManager.h"
|
|
|
|
#include "nsString.h"
|
2016-04-02 03:04:59 +03:00
|
|
|
#include "nsStyleStruct.h"
|
2016-06-30 23:37:52 +03:00
|
|
|
#include "nsStyleUtil.h"
|
2016-07-19 04:02:55 +03:00
|
|
|
#include "nsTArray.h"
|
2015-11-25 04:14:39 +03:00
|
|
|
|
|
|
|
#include "mozilla/EventStates.h"
|
2016-07-19 04:02:55 +03:00
|
|
|
#include "mozilla/ServoElementSnapshot.h"
|
2015-11-25 04:14:39 +03:00
|
|
|
#include "mozilla/dom/Element.h"
|
|
|
|
|
2016-08-16 08:08:46 +03:00
|
|
|
#define IMPL_STRONG_REF_TYPE(name, T) \
|
|
|
|
already_AddRefed<T> name::Consume() { \
|
|
|
|
RefPtr<T> result = dont_AddRef(mPtr); \
|
|
|
|
mPtr = nullptr; \
|
|
|
|
return result.forget(); \
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
IMPL_STRONG_REF_TYPE(ServoComputedValuesStrong, ServoComputedValues);
|
|
|
|
IMPL_STRONG_REF_TYPE(RawServoStyleSheetStrong, RawServoStyleSheet);
|
|
|
|
|
2015-11-25 04:14:39 +03:00
|
|
|
uint32_t
|
|
|
|
Gecko_ChildrenCount(RawGeckoNode* aNode)
|
|
|
|
{
|
|
|
|
return aNode->GetChildCount();
|
|
|
|
}
|
|
|
|
|
2016-03-03 07:27:07 +03:00
|
|
|
bool
|
2015-11-25 04:14:39 +03:00
|
|
|
Gecko_NodeIsElement(RawGeckoNode* aNode)
|
|
|
|
{
|
|
|
|
return aNode->IsElement();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoNode*
|
|
|
|
Gecko_GetParentNode(RawGeckoNode* aNode)
|
|
|
|
{
|
|
|
|
return aNode->GetParentNode();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoNode*
|
|
|
|
Gecko_GetFirstChild(RawGeckoNode* aNode)
|
|
|
|
{
|
|
|
|
return aNode->GetFirstChild();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoNode*
|
|
|
|
Gecko_GetLastChild(RawGeckoNode* aNode)
|
|
|
|
{
|
|
|
|
return aNode->GetLastChild();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoNode*
|
|
|
|
Gecko_GetPrevSibling(RawGeckoNode* aNode)
|
|
|
|
{
|
|
|
|
return aNode->GetPreviousSibling();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoNode*
|
|
|
|
Gecko_GetNextSibling(RawGeckoNode* aNode)
|
|
|
|
{
|
|
|
|
return aNode->GetNextSibling();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoElement*
|
|
|
|
Gecko_GetParentElement(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return aElement->GetParentElement();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoElement*
|
|
|
|
Gecko_GetFirstChildElement(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return aElement->GetFirstElementChild();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoElement* Gecko_GetLastChildElement(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return aElement->GetLastElementChild();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoElement*
|
|
|
|
Gecko_GetPrevSiblingElement(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return aElement->GetPreviousElementSibling();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoElement*
|
|
|
|
Gecko_GetNextSiblingElement(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return aElement->GetNextElementSibling();
|
|
|
|
}
|
|
|
|
|
|
|
|
RawGeckoElement*
|
|
|
|
Gecko_GetDocumentElement(RawGeckoDocument* aDoc)
|
|
|
|
{
|
|
|
|
return aDoc->GetDocumentElement();
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
EventStates::ServoType
|
2016-02-24 04:51:47 +03:00
|
|
|
Gecko_ElementState(RawGeckoElement* aElement)
|
|
|
|
{
|
2016-07-19 04:02:55 +03:00
|
|
|
return aElement->StyleState().ServoValue();
|
2016-02-24 04:51:47 +03:00
|
|
|
}
|
|
|
|
|
2016-03-03 07:27:07 +03:00
|
|
|
bool
|
2015-11-25 04:14:39 +03:00
|
|
|
Gecko_IsHTMLElementInHTMLDocument(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return aElement->IsHTMLElement() && aElement->OwnerDoc()->IsHTMLDocument();
|
|
|
|
}
|
|
|
|
|
2016-03-03 07:27:07 +03:00
|
|
|
bool
|
2015-11-25 04:14:39 +03:00
|
|
|
Gecko_IsLink(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return nsCSSRuleProcessor::IsLink(aElement);
|
|
|
|
}
|
|
|
|
|
2016-03-03 07:27:07 +03:00
|
|
|
bool
|
|
|
|
Gecko_IsTextNode(RawGeckoNode* aNode)
|
2015-11-25 04:14:39 +03:00
|
|
|
{
|
|
|
|
return aNode->NodeInfo()->NodeType() == nsIDOMNode::TEXT_NODE;
|
|
|
|
}
|
|
|
|
|
2016-03-03 07:27:07 +03:00
|
|
|
bool
|
2015-11-25 04:14:39 +03:00
|
|
|
Gecko_IsVisitedLink(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return aElement->StyleState().HasState(NS_EVENT_STATE_VISITED);
|
|
|
|
}
|
|
|
|
|
2016-03-03 07:27:07 +03:00
|
|
|
bool
|
2015-11-25 04:14:39 +03:00
|
|
|
Gecko_IsUnvisitedLink(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return aElement->StyleState().HasState(NS_EVENT_STATE_UNVISITED);
|
|
|
|
}
|
|
|
|
|
2016-03-03 07:27:07 +03:00
|
|
|
bool
|
2015-11-25 04:14:39 +03:00
|
|
|
Gecko_IsRootElement(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return aElement->OwnerDoc()->GetRootElement() == aElement;
|
|
|
|
}
|
|
|
|
|
2016-05-18 22:05:36 +03:00
|
|
|
nsIAtom*
|
|
|
|
Gecko_LocalName(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
return aElement->NodeInfo()->NameAtom();
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIAtom*
|
|
|
|
Gecko_Namespace(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
int32_t id = aElement->NodeInfo()->NamespaceID();
|
2016-08-06 00:21:54 +03:00
|
|
|
return nsContentUtils::NameSpaceManager()->NameSpaceURIAtomForServo(id);
|
2016-05-18 22:05:36 +03:00
|
|
|
}
|
|
|
|
|
2016-05-25 21:16:26 +03:00
|
|
|
nsIAtom*
|
|
|
|
Gecko_GetElementId(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
const nsAttrValue* attr = aElement->GetParsedAttr(nsGkAtoms::id);
|
|
|
|
return attr ? attr->GetAtomValue() : nullptr;
|
|
|
|
}
|
|
|
|
|
2016-07-08 10:07:06 +03:00
|
|
|
// Dirtiness tracking.
|
|
|
|
uint32_t
|
|
|
|
Gecko_GetNodeFlags(RawGeckoNode* aNode)
|
|
|
|
{
|
|
|
|
return aNode->GetFlags();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_SetNodeFlags(RawGeckoNode* aNode, uint32_t aFlags)
|
|
|
|
{
|
|
|
|
aNode->SetFlags(aFlags);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_UnsetNodeFlags(RawGeckoNode* aNode, uint32_t aFlags)
|
|
|
|
{
|
|
|
|
aNode->UnsetFlags(aFlags);
|
|
|
|
}
|
|
|
|
|
2016-07-29 05:20:45 +03:00
|
|
|
nsStyleContext*
|
|
|
|
Gecko_GetStyleContext(RawGeckoNode* aNode)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(aNode->IsContent());
|
|
|
|
nsIFrame* primaryFrame = aNode->AsContent()->GetPrimaryFrame();
|
|
|
|
if (!primaryFrame) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return primaryFrame->StyleContext();
|
|
|
|
}
|
|
|
|
|
2016-07-27 00:22:57 +03:00
|
|
|
nsChangeHint
|
2016-07-29 05:20:45 +03:00
|
|
|
Gecko_CalcStyleDifference(nsStyleContext* aOldStyleContext,
|
2016-08-16 08:10:05 +03:00
|
|
|
ServoComputedValuesBorrowed aComputedValues)
|
2016-07-27 00:22:57 +03:00
|
|
|
{
|
2016-07-29 05:20:45 +03:00
|
|
|
MOZ_ASSERT(aOldStyleContext);
|
|
|
|
MOZ_ASSERT(aComputedValues);
|
2016-07-27 00:22:57 +03:00
|
|
|
|
|
|
|
// Pass the safe thing, which causes us to miss a potential optimization. See
|
|
|
|
// bug 1289863.
|
|
|
|
nsChangeHint forDescendants = nsChangeHint_Hints_NotHandledForDescendants;
|
|
|
|
|
|
|
|
// Eventually, we should compute things out of these flags like
|
|
|
|
// ElementRestyler::RestyleSelf does and pass the result to the caller to
|
|
|
|
// potentially halt traversal. See bug 1289868.
|
|
|
|
uint32_t equalStructs, samePointerStructs;
|
|
|
|
nsChangeHint result =
|
2016-07-29 05:20:45 +03:00
|
|
|
aOldStyleContext->CalcStyleDifference(aComputedValues,
|
|
|
|
forDescendants,
|
|
|
|
&equalStructs,
|
|
|
|
&samePointerStructs);
|
|
|
|
|
2016-07-27 00:22:57 +03:00
|
|
|
return result;
|
2016-07-29 05:20:45 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_StoreStyleDifference(RawGeckoNode* aNode, nsChangeHint aChangeHintToStore)
|
|
|
|
{
|
|
|
|
#ifdef MOZ_STYLO
|
|
|
|
MOZ_ASSERT(aNode->IsContent());
|
2016-08-06 23:34:53 +03:00
|
|
|
MOZ_ASSERT(aNode->IsDirtyForServo(),
|
|
|
|
"Change hint stored in a not-dirty node");
|
|
|
|
|
|
|
|
// For elements, we need to store the change hint in the proper style context.
|
|
|
|
// For text nodes, we want to store the change hint in the parent element,
|
|
|
|
// since Gecko's change list only operates on Elements, and we'll fail to
|
|
|
|
// compute the change hint for the element properly because the property won't
|
|
|
|
// always be in the cache of the parent's nsStyleContext.
|
|
|
|
//
|
|
|
|
// For Gecko this is not a problem, because they access the inherited structs
|
|
|
|
// from the parent style context in order to inherit them, so they're found in
|
|
|
|
// the cache and get compared.
|
|
|
|
Element* aElement =
|
|
|
|
aNode->IsElement() ? aNode->AsElement() : aNode->GetParentElement();
|
|
|
|
|
|
|
|
nsIFrame* primaryFrame = aElement->GetPrimaryFrame();
|
2016-07-29 05:20:45 +03:00
|
|
|
if (!primaryFrame) {
|
|
|
|
// TODO: Pick the undisplayed content map from the frame-constructor, and
|
|
|
|
// stick it there. For now we're generating ReconstructFrame
|
|
|
|
// unconditionally, which is suboptimal.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
primaryFrame->StyleContext()->StoreChangeHint(aChangeHintToStore);
|
2016-07-27 00:22:57 +03:00
|
|
|
#else
|
2016-07-29 05:20:45 +03:00
|
|
|
MOZ_CRASH("stylo: Shouldn't call Gecko_StoreStyleDifference in "
|
2016-07-27 00:22:57 +03:00
|
|
|
"non-stylo build");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
ServoDeclarationBlock*
|
|
|
|
Gecko_GetServoDeclarationBlock(RawGeckoElement* aElement)
|
|
|
|
{
|
|
|
|
const nsAttrValue* attr = aElement->GetParsedAttr(nsGkAtoms::style);
|
|
|
|
if (!attr || attr->Type() != nsAttrValue::eServoCSSDeclaration) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return attr->GetServoCSSDeclarationValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Implementor>
|
|
|
|
static nsIAtom*
|
|
|
|
AtomAttrValue(Implementor* aElement, nsIAtom* aName)
|
|
|
|
{
|
|
|
|
const nsAttrValue* attr = aElement->GetParsedAttr(aName);
|
|
|
|
return attr ? attr->GetAtomValue() : nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Implementor, typename MatchFn>
|
|
|
|
static bool
|
|
|
|
DoMatch(Implementor* aElement, nsIAtom* aNS, nsIAtom* aName, MatchFn aMatch)
|
2016-06-30 23:37:52 +03:00
|
|
|
{
|
|
|
|
if (aNS) {
|
2016-06-28 17:24:48 +03:00
|
|
|
int32_t ns = nsContentUtils::NameSpaceManager()->GetNameSpaceID(aNS,
|
|
|
|
aElement->IsInChromeDocument());
|
2016-06-30 23:37:52 +03:00
|
|
|
NS_ENSURE_TRUE(ns != kNameSpaceID_Unknown, false);
|
|
|
|
const nsAttrValue* value = aElement->GetParsedAttr(aName, ns);
|
|
|
|
return value && aMatch(value);
|
|
|
|
}
|
|
|
|
// No namespace means any namespace - we have to check them all. :-(
|
2016-07-23 00:11:41 +03:00
|
|
|
BorrowedAttrInfo attrInfo;
|
2016-07-22 05:43:38 +03:00
|
|
|
for (uint32_t i = 0; (attrInfo = aElement->GetAttrInfoAt(i)); ++i) {
|
|
|
|
if (attrInfo.mName->LocalName() != aName) {
|
2016-06-30 23:37:52 +03:00
|
|
|
continue;
|
|
|
|
}
|
2016-07-22 05:43:38 +03:00
|
|
|
if (aMatch(attrInfo.mValue)) {
|
2016-06-30 23:37:52 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
template <typename Implementor>
|
|
|
|
static bool
|
|
|
|
HasAttr(Implementor* aElement, nsIAtom* aNS, nsIAtom* aName)
|
2016-06-30 23:37:52 +03:00
|
|
|
{
|
|
|
|
auto match = [](const nsAttrValue* aValue) { return true; };
|
|
|
|
return DoMatch(aElement, aNS, aName, match);
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
template <typename Implementor>
|
|
|
|
static bool
|
2016-07-19 00:53:45 +03:00
|
|
|
AttrEquals(Implementor* aElement, nsIAtom* aNS, nsIAtom* aName, nsIAtom* aStr,
|
2016-07-19 04:02:55 +03:00
|
|
|
bool aIgnoreCase)
|
2016-06-30 23:37:52 +03:00
|
|
|
{
|
|
|
|
auto match = [aStr, aIgnoreCase](const nsAttrValue* aValue) {
|
|
|
|
return aValue->Equals(aStr, aIgnoreCase ? eIgnoreCase : eCaseMatters);
|
|
|
|
};
|
|
|
|
return DoMatch(aElement, aNS, aName, match);
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
template <typename Implementor>
|
|
|
|
static bool
|
|
|
|
AttrDashEquals(Implementor* aElement, nsIAtom* aNS, nsIAtom* aName,
|
2016-07-19 00:53:45 +03:00
|
|
|
nsIAtom* aStr)
|
2016-06-30 23:37:52 +03:00
|
|
|
{
|
|
|
|
auto match = [aStr](const nsAttrValue* aValue) {
|
|
|
|
nsAutoString str;
|
|
|
|
aValue->ToString(str);
|
|
|
|
const nsDefaultStringComparator c;
|
|
|
|
return nsStyleUtil::DashMatchCompare(str, nsDependentAtomString(aStr), c);
|
|
|
|
};
|
|
|
|
return DoMatch(aElement, aNS, aName, match);
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
template <typename Implementor>
|
|
|
|
static bool
|
|
|
|
AttrIncludes(Implementor* aElement, nsIAtom* aNS, nsIAtom* aName,
|
2016-07-19 00:53:45 +03:00
|
|
|
nsIAtom* aStr)
|
2016-06-30 23:37:52 +03:00
|
|
|
{
|
|
|
|
auto match = [aStr](const nsAttrValue* aValue) {
|
|
|
|
nsAutoString str;
|
|
|
|
aValue->ToString(str);
|
|
|
|
const nsDefaultStringComparator c;
|
|
|
|
return nsStyleUtil::ValueIncludes(str, nsDependentAtomString(aStr), c);
|
|
|
|
};
|
|
|
|
return DoMatch(aElement, aNS, aName, match);
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
template <typename Implementor>
|
|
|
|
static bool
|
|
|
|
AttrHasSubstring(Implementor* aElement, nsIAtom* aNS, nsIAtom* aName,
|
2016-07-19 00:53:45 +03:00
|
|
|
nsIAtom* aStr)
|
2016-06-30 23:37:52 +03:00
|
|
|
{
|
|
|
|
auto match = [aStr](const nsAttrValue* aValue) {
|
|
|
|
nsAutoString str;
|
|
|
|
aValue->ToString(str);
|
|
|
|
return FindInReadable(str, nsDependentAtomString(aStr));
|
|
|
|
};
|
|
|
|
return DoMatch(aElement, aNS, aName, match);
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
template <typename Implementor>
|
|
|
|
static bool
|
|
|
|
AttrHasPrefix(Implementor* aElement, nsIAtom* aNS, nsIAtom* aName,
|
2016-07-19 00:53:45 +03:00
|
|
|
nsIAtom* aStr)
|
2016-06-30 23:37:52 +03:00
|
|
|
{
|
|
|
|
auto match = [aStr](const nsAttrValue* aValue) {
|
|
|
|
nsAutoString str;
|
|
|
|
aValue->ToString(str);
|
|
|
|
return StringBeginsWith(str, nsDependentAtomString(aStr));
|
|
|
|
};
|
|
|
|
return DoMatch(aElement, aNS, aName, match);
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
template <typename Implementor>
|
|
|
|
static bool
|
|
|
|
AttrHasSuffix(Implementor* aElement, nsIAtom* aNS, nsIAtom* aName,
|
2016-07-19 00:53:45 +03:00
|
|
|
nsIAtom* aStr)
|
2016-06-30 23:37:52 +03:00
|
|
|
{
|
|
|
|
auto match = [aStr](const nsAttrValue* aValue) {
|
|
|
|
nsAutoString str;
|
|
|
|
aValue->ToString(str);
|
|
|
|
return StringEndsWith(str, nsDependentAtomString(aStr));
|
|
|
|
};
|
|
|
|
return DoMatch(aElement, aNS, aName, match);
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
/**
|
|
|
|
* Gets the class or class list (if any) of the implementor. The calling
|
|
|
|
* convention here is rather hairy, and is optimized for getting Servo the
|
|
|
|
* information it needs for hot calls.
|
|
|
|
*
|
|
|
|
* The return value indicates the number of classes. If zero, neither outparam
|
|
|
|
* is valid. If one, the class_ outparam is filled with the atom of the class.
|
|
|
|
* If two or more, the classList outparam is set to point to an array of atoms
|
|
|
|
* representing the class list.
|
|
|
|
*
|
|
|
|
* The array is borrowed and the atoms are not addrefed. These values can be
|
|
|
|
* invalidated by any DOM mutation. Use them in a tight scope.
|
|
|
|
*/
|
|
|
|
template <typename Implementor>
|
|
|
|
static uint32_t
|
|
|
|
ClassOrClassList(Implementor* aElement, nsIAtom** aClass, nsIAtom*** aClassList)
|
2016-05-25 21:16:26 +03:00
|
|
|
{
|
|
|
|
const nsAttrValue* attr = aElement->GetParsedAttr(nsGkAtoms::_class);
|
|
|
|
if (!attr) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// For class values with only whitespace, Gecko just stores a string. For the
|
|
|
|
// purposes of the style system, there is no class in this case.
|
|
|
|
if (attr->Type() == nsAttrValue::eString) {
|
|
|
|
MOZ_ASSERT(nsContentUtils::TrimWhitespace<nsContentUtils::IsHTMLWhitespace>(
|
|
|
|
attr->GetStringValue()).IsEmpty());
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Single tokens are generally stored as an atom. Check that case.
|
|
|
|
if (attr->Type() == nsAttrValue::eAtom) {
|
|
|
|
*aClass = attr->GetAtomValue();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// At this point we should have an atom array. It is likely, but not
|
|
|
|
// guaranteed, that we have two or more elements in the array.
|
|
|
|
MOZ_ASSERT(attr->Type() == nsAttrValue::eAtomArray);
|
|
|
|
nsTArray<nsCOMPtr<nsIAtom>>* atomArray = attr->GetAtomArrayValue();
|
|
|
|
uint32_t length = atomArray->Length();
|
|
|
|
|
|
|
|
// Special case: zero elements.
|
|
|
|
if (length == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Special case: one element.
|
|
|
|
if (length == 1) {
|
|
|
|
*aClass = atomArray->ElementAt(0);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// General case: Two or more elements.
|
|
|
|
//
|
|
|
|
// Note: We could also expose this array as an array of nsCOMPtrs, since
|
|
|
|
// bindgen knows what those look like, and eliminate the reinterpret_cast.
|
|
|
|
// But it's not obvious that that would be preferable.
|
|
|
|
static_assert(sizeof(nsCOMPtr<nsIAtom>) == sizeof(nsIAtom*), "Bad simplification");
|
|
|
|
static_assert(alignof(nsCOMPtr<nsIAtom>) == alignof(nsIAtom*), "Bad simplification");
|
|
|
|
|
|
|
|
nsCOMPtr<nsIAtom>* elements = atomArray->Elements();
|
|
|
|
nsIAtom** rawElements = reinterpret_cast<nsIAtom**>(elements);
|
|
|
|
*aClassList = rawElements;
|
|
|
|
return atomArray->Length();
|
|
|
|
}
|
|
|
|
|
2016-07-19 04:02:55 +03:00
|
|
|
#define SERVO_IMPL_ELEMENT_ATTR_MATCHING_FUNCTIONS(prefix_, implementor_) \
|
|
|
|
nsIAtom* prefix_##AtomAttrValue(implementor_* aElement, nsIAtom* aName) \
|
|
|
|
{ \
|
|
|
|
return AtomAttrValue(aElement, aName); \
|
|
|
|
} \
|
|
|
|
bool prefix_##HasAttr(implementor_* aElement, nsIAtom* aNS, nsIAtom* aName) \
|
|
|
|
{ \
|
|
|
|
return HasAttr(aElement, aNS, aName); \
|
|
|
|
} \
|
|
|
|
bool prefix_##AttrEquals(implementor_* aElement, nsIAtom* aNS, \
|
|
|
|
nsIAtom* aName, nsIAtom* aStr, bool aIgnoreCase) \
|
|
|
|
{ \
|
|
|
|
return AttrEquals(aElement, aNS, aName, aStr, aIgnoreCase); \
|
|
|
|
} \
|
|
|
|
bool prefix_##AttrDashEquals(implementor_* aElement, nsIAtom* aNS, \
|
|
|
|
nsIAtom* aName, nsIAtom* aStr) \
|
|
|
|
{ \
|
|
|
|
return AttrDashEquals(aElement, aNS, aName, aStr); \
|
|
|
|
} \
|
|
|
|
bool prefix_##AttrIncludes(implementor_* aElement, nsIAtom* aNS, \
|
|
|
|
nsIAtom* aName, nsIAtom* aStr) \
|
|
|
|
{ \
|
|
|
|
return AttrIncludes(aElement, aNS, aName, aStr); \
|
|
|
|
} \
|
|
|
|
bool prefix_##AttrHasSubstring(implementor_* aElement, nsIAtom* aNS, \
|
|
|
|
nsIAtom* aName, nsIAtom* aStr) \
|
|
|
|
{ \
|
|
|
|
return AttrHasSubstring(aElement, aNS, aName, aStr); \
|
|
|
|
} \
|
|
|
|
bool prefix_##AttrHasPrefix(implementor_* aElement, nsIAtom* aNS, \
|
|
|
|
nsIAtom* aName, nsIAtom* aStr) \
|
|
|
|
{ \
|
|
|
|
return AttrHasPrefix(aElement, aNS, aName, aStr); \
|
|
|
|
} \
|
|
|
|
bool prefix_##AttrHasSuffix(implementor_* aElement, nsIAtom* aNS, \
|
|
|
|
nsIAtom* aName, nsIAtom* aStr) \
|
|
|
|
{ \
|
|
|
|
return AttrHasSuffix(aElement, aNS, aName, aStr); \
|
|
|
|
} \
|
|
|
|
uint32_t prefix_##ClassOrClassList(implementor_* aElement, nsIAtom** aClass, \
|
|
|
|
nsIAtom*** aClassList) \
|
|
|
|
{ \
|
|
|
|
return ClassOrClassList(aElement, aClass, aClassList); \
|
2016-06-24 06:35:12 +03:00
|
|
|
}
|
2016-07-19 04:02:55 +03:00
|
|
|
|
|
|
|
SERVO_IMPL_ELEMENT_ATTR_MATCHING_FUNCTIONS(Gecko_, RawGeckoElement)
|
|
|
|
SERVO_IMPL_ELEMENT_ATTR_MATCHING_FUNCTIONS(Gecko_Snapshot, ServoElementSnapshot)
|
|
|
|
|
|
|
|
#undef SERVO_IMPL_ELEMENT_ATTR_MATCHING_FUNCTIONS
|
2016-06-24 06:35:12 +03:00
|
|
|
|
2016-02-24 04:28:50 +03:00
|
|
|
ServoNodeData*
|
|
|
|
Gecko_GetNodeData(RawGeckoNode* aNode)
|
|
|
|
{
|
2016-07-21 22:47:29 +03:00
|
|
|
return aNode->ServoData().get();
|
2016-02-24 04:28:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_SetNodeData(RawGeckoNode* aNode, ServoNodeData* aData)
|
|
|
|
{
|
2016-07-21 22:47:29 +03:00
|
|
|
MOZ_ASSERT(!aNode->ServoData());
|
|
|
|
aNode->ServoData().reset(aData);
|
2016-02-24 04:28:50 +03:00
|
|
|
}
|
2016-02-26 04:51:01 +03:00
|
|
|
|
2016-05-18 22:05:36 +03:00
|
|
|
nsIAtom*
|
|
|
|
Gecko_Atomize(const char* aString, uint32_t aLength)
|
|
|
|
{
|
|
|
|
return NS_Atomize(nsDependentCSubstring(aString, aLength)).take();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_AddRefAtom(nsIAtom* aAtom)
|
|
|
|
{
|
|
|
|
NS_ADDREF(aAtom);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_ReleaseAtom(nsIAtom* aAtom)
|
|
|
|
{
|
|
|
|
NS_RELEASE(aAtom);
|
|
|
|
}
|
|
|
|
|
|
|
|
const uint16_t*
|
|
|
|
Gecko_GetAtomAsUTF16(nsIAtom* aAtom, uint32_t* aLength)
|
|
|
|
{
|
|
|
|
static_assert(sizeof(char16_t) == sizeof(uint16_t), "Servo doesn't know what a char16_t is");
|
|
|
|
MOZ_ASSERT(aAtom);
|
|
|
|
*aLength = aAtom->GetLength();
|
|
|
|
|
|
|
|
// We need to manually cast from char16ptr_t to const char16_t* to handle the
|
|
|
|
// MOZ_USE_CHAR16_WRAPPER we use on WIndows.
|
|
|
|
return reinterpret_cast<const uint16_t*>(static_cast<const char16_t*>(aAtom->GetUTF16String()));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Gecko_AtomEqualsUTF8(nsIAtom* aAtom, const char* aString, uint32_t aLength)
|
|
|
|
{
|
|
|
|
// XXXbholley: We should be able to do this without converting, I just can't
|
|
|
|
// find the right thing to call.
|
|
|
|
nsAutoString atomStr;
|
|
|
|
aAtom->ToString(atomStr);
|
2016-05-19 09:54:58 +03:00
|
|
|
NS_ConvertUTF8toUTF16 inStr(nsDependentCSubstring(aString, aLength));
|
2016-05-18 22:05:36 +03:00
|
|
|
return atomStr.Equals(inStr);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Gecko_AtomEqualsUTF8IgnoreCase(nsIAtom* aAtom, const char* aString, uint32_t aLength)
|
|
|
|
{
|
|
|
|
// XXXbholley: We should be able to do this without converting, I just can't
|
|
|
|
// find the right thing to call.
|
|
|
|
nsAutoString atomStr;
|
|
|
|
aAtom->ToString(atomStr);
|
2016-05-19 09:54:58 +03:00
|
|
|
NS_ConvertUTF8toUTF16 inStr(nsDependentCSubstring(aString, aLength));
|
2016-05-18 22:05:36 +03:00
|
|
|
return nsContentUtils::EqualsIgnoreASCIICase(atomStr, inStr);
|
|
|
|
}
|
|
|
|
|
2016-06-07 22:13:24 +03:00
|
|
|
void
|
|
|
|
Gecko_FontFamilyList_Clear(FontFamilyList* aList) {
|
|
|
|
aList->Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_FontFamilyList_AppendNamed(FontFamilyList* aList, nsIAtom* aName)
|
|
|
|
{
|
|
|
|
// Servo doesn't record whether the name was quoted or unquoted, so just
|
|
|
|
// assume unquoted for now.
|
|
|
|
FontFamilyName family;
|
|
|
|
aName->ToString(family.mName);
|
|
|
|
aList->Append(family);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_FontFamilyList_AppendGeneric(FontFamilyList* aList, FontFamilyType aType)
|
|
|
|
{
|
|
|
|
aList->Append(FontFamilyName(aType));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_CopyFontFamilyFrom(nsFont* dst, const nsFont* src)
|
|
|
|
{
|
|
|
|
dst->fontlist = src->fontlist;
|
|
|
|
}
|
|
|
|
|
2016-05-04 03:42:40 +03:00
|
|
|
void
|
|
|
|
Gecko_SetListStyleType(nsStyleList* style_struct, uint32_t type)
|
|
|
|
{
|
|
|
|
// Builtin counter styles are static and use no-op refcounting, and thus are
|
|
|
|
// safe to use off-main-thread.
|
|
|
|
style_struct->SetCounterStyle(CounterStyleManager::GetBuiltinStyle(type));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_CopyListStyleTypeFrom(nsStyleList* dst, const nsStyleList* src)
|
|
|
|
{
|
|
|
|
dst->SetCounterStyle(src->GetCounterStyle());
|
|
|
|
}
|
|
|
|
|
2016-05-21 03:02:54 +03:00
|
|
|
NS_IMPL_HOLDER_FFI_REFCOUNTING(nsIPrincipal, Principal)
|
|
|
|
NS_IMPL_HOLDER_FFI_REFCOUNTING(nsIURI, URI)
|
|
|
|
|
2016-05-21 03:02:54 +03:00
|
|
|
void
|
|
|
|
Gecko_SetMozBinding(nsStyleDisplay* aDisplay,
|
|
|
|
const uint8_t* aURLString, uint32_t aURLStringLength,
|
|
|
|
ThreadSafeURIHolder* aBaseURI,
|
|
|
|
ThreadSafeURIHolder* aReferrer,
|
|
|
|
ThreadSafePrincipalHolder* aPrincipal)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(aDisplay);
|
|
|
|
MOZ_ASSERT(aURLString);
|
|
|
|
MOZ_ASSERT(aBaseURI);
|
|
|
|
MOZ_ASSERT(aReferrer);
|
|
|
|
MOZ_ASSERT(aPrincipal);
|
|
|
|
|
|
|
|
nsString url;
|
|
|
|
nsDependentCSubstring urlString(reinterpret_cast<const char*>(aURLString),
|
|
|
|
aURLStringLength);
|
|
|
|
AppendUTF8toUTF16(urlString, url);
|
|
|
|
RefPtr<nsStringBuffer> urlBuffer = nsCSSValue::BufferFromString(url);
|
|
|
|
|
|
|
|
aDisplay->mBinding =
|
2016-05-25 08:04:50 +03:00
|
|
|
new css::URLValue(urlBuffer, do_AddRef(aBaseURI),
|
|
|
|
do_AddRef(aReferrer), do_AddRef(aPrincipal));
|
2016-05-21 03:02:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_CopyMozBindingFrom(nsStyleDisplay* aDest, const nsStyleDisplay* aSrc)
|
|
|
|
{
|
|
|
|
aDest->mBinding = aSrc->mBinding;
|
|
|
|
}
|
|
|
|
|
2016-05-27 01:18:09 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_SetNullImageValue(nsStyleImage* aImage)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(aImage);
|
|
|
|
aImage->SetNull();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_SetGradientImageValue(nsStyleImage* aImage, nsStyleGradient* aGradient)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(aImage);
|
|
|
|
aImage->SetGradientData(aGradient);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_CopyImageValueFrom(nsStyleImage* aImage, const nsStyleImage* aOther)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(aImage);
|
|
|
|
MOZ_ASSERT(aOther);
|
|
|
|
|
|
|
|
*aImage = *aOther;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsStyleGradient*
|
|
|
|
Gecko_CreateGradient(uint8_t aShape,
|
|
|
|
uint8_t aSize,
|
|
|
|
bool aRepeating,
|
|
|
|
bool aLegacySyntax,
|
|
|
|
uint32_t aStopCount)
|
|
|
|
{
|
|
|
|
nsStyleGradient* result = new nsStyleGradient();
|
|
|
|
|
|
|
|
result->mShape = aShape;
|
|
|
|
result->mSize = aSize;
|
|
|
|
result->mRepeating = aRepeating;
|
|
|
|
result->mLegacySyntax = aLegacySyntax;
|
|
|
|
|
|
|
|
result->mAngle.SetNoneValue();
|
|
|
|
result->mBgPosX.SetNoneValue();
|
|
|
|
result->mBgPosY.SetNoneValue();
|
|
|
|
result->mRadiusX.SetNoneValue();
|
|
|
|
result->mRadiusY.SetNoneValue();
|
|
|
|
|
|
|
|
nsStyleGradientStop dummyStop;
|
|
|
|
dummyStop.mLocation.SetNoneValue();
|
|
|
|
dummyStop.mColor = NS_RGB(0, 0, 0);
|
|
|
|
dummyStop.mIsInterpolationHint = 0;
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < aStopCount; i++) {
|
|
|
|
result->mStops.AppendElement(dummyStop);
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-07-14 11:59:37 +03:00
|
|
|
void
|
2016-07-15 18:26:48 +03:00
|
|
|
Gecko_EnsureTArrayCapacity(void* aArray, size_t aCapacity, size_t aElemSize)
|
|
|
|
{
|
2016-08-12 07:54:14 +03:00
|
|
|
auto base =
|
|
|
|
reinterpret_cast<nsTArray_base<nsTArrayInfallibleAllocator,
|
|
|
|
nsTArray_CopyWithMemutils>*>(aArray);
|
|
|
|
|
2016-07-14 11:59:37 +03:00
|
|
|
base->EnsureCapacity<nsTArrayInfallibleAllocator>(aCapacity, aElemSize);
|
|
|
|
}
|
|
|
|
|
2016-08-12 07:54:14 +03:00
|
|
|
void
|
|
|
|
Gecko_ClearPODTArray(void* aArray, size_t aElementSize, size_t aElementAlign)
|
|
|
|
{
|
|
|
|
auto base =
|
|
|
|
reinterpret_cast<nsTArray_base<nsTArrayInfallibleAllocator,
|
|
|
|
nsTArray_CopyWithMemutils>*>(aArray);
|
|
|
|
|
|
|
|
base->template ShiftData<nsTArrayInfallibleAllocator>(0, base->Length(), 0,
|
|
|
|
aElementSize, aElementAlign);
|
|
|
|
}
|
|
|
|
|
2016-07-15 18:26:48 +03:00
|
|
|
void
|
|
|
|
Gecko_EnsureImageLayersLength(nsStyleImageLayers* aLayers, size_t aLen)
|
|
|
|
{
|
2016-07-14 11:59:48 +03:00
|
|
|
aLayers->mLayers.EnsureLengthAtLeast(aLen);
|
|
|
|
}
|
|
|
|
|
2016-07-15 18:26:48 +03:00
|
|
|
void
|
|
|
|
Gecko_InitializeImageLayer(nsStyleImageLayers::Layer* aLayer,
|
|
|
|
nsStyleImageLayers::LayerType aLayerType)
|
|
|
|
{
|
2016-07-12 11:10:39 +03:00
|
|
|
aLayer->Initialize(aLayerType);
|
2016-07-14 11:59:48 +03:00
|
|
|
}
|
|
|
|
|
2016-07-15 18:26:48 +03:00
|
|
|
void
|
|
|
|
Gecko_ResetStyleCoord(nsStyleUnit* aUnit, nsStyleUnion* aValue)
|
|
|
|
{
|
|
|
|
nsStyleCoord::Reset(*aUnit, *aValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_SetStyleCoordCalcValue(nsStyleUnit* aUnit, nsStyleUnion* aValue, nsStyleCoord::CalcValue aCalc)
|
|
|
|
{
|
|
|
|
// Calc units should be cleaned up first
|
|
|
|
MOZ_ASSERT(*aUnit != nsStyleUnit::eStyleUnit_Calc);
|
|
|
|
nsStyleCoord::Calc* calcRef = new nsStyleCoord::Calc();
|
|
|
|
calcRef->mLength = aCalc.mLength;
|
|
|
|
calcRef->mPercent = aCalc.mPercent;
|
|
|
|
calcRef->mHasPercent = aCalc.mHasPercent;
|
|
|
|
*aUnit = nsStyleUnit::eStyleUnit_Calc;
|
|
|
|
aValue->mPointer = calcRef;
|
|
|
|
calcRef->AddRef();
|
|
|
|
}
|
|
|
|
|
2016-08-15 18:48:11 +03:00
|
|
|
void
|
|
|
|
Gecko_CopyClipPathValueFrom(mozilla::StyleClipPath* aDst, const mozilla::StyleClipPath* aSrc)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(aDst);
|
|
|
|
MOZ_ASSERT(aSrc);
|
|
|
|
|
|
|
|
*aDst = *aSrc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Gecko_DestroyClipPath(mozilla::StyleClipPath* aClip)
|
|
|
|
{
|
|
|
|
aClip->~StyleClipPath();
|
|
|
|
}
|
|
|
|
|
|
|
|
mozilla::StyleBasicShape*
|
|
|
|
Gecko_NewBasicShape(mozilla::StyleBasicShapeType aType)
|
|
|
|
{
|
|
|
|
RefPtr<StyleBasicShape> ptr = new mozilla::StyleBasicShape(aType);
|
|
|
|
return ptr.forget().take();
|
|
|
|
}
|
|
|
|
|
2016-07-15 18:26:48 +03:00
|
|
|
NS_IMPL_THREADSAFE_FFI_REFCOUNTING(nsStyleCoord::Calc, Calc);
|
|
|
|
|
2016-04-02 03:04:59 +03:00
|
|
|
#define STYLE_STRUCT(name, checkdata_cb) \
|
|
|
|
\
|
|
|
|
void \
|
|
|
|
Gecko_Construct_nsStyle##name(nsStyle##name* ptr) \
|
|
|
|
{ \
|
|
|
|
new (ptr) nsStyle##name(StyleStructContext::ServoContext()); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
void \
|
|
|
|
Gecko_CopyConstruct_nsStyle##name(nsStyle##name* ptr, \
|
|
|
|
const nsStyle##name* other) \
|
|
|
|
{ \
|
|
|
|
new (ptr) nsStyle##name(*other); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
void \
|
|
|
|
Gecko_Destroy_nsStyle##name(nsStyle##name* ptr) \
|
|
|
|
{ \
|
|
|
|
ptr->~nsStyle##name(); \
|
|
|
|
}
|
|
|
|
|
|
|
|
#include "nsStyleStructList.h"
|
|
|
|
|
|
|
|
#undef STYLE_STRUCT
|
|
|
|
|
2016-02-26 04:51:01 +03:00
|
|
|
#ifndef MOZ_STYLO
|
2016-08-19 14:44:43 +03:00
|
|
|
#define SERVO_BINDING_FUNC(name_, return_, ...) \
|
|
|
|
return_ name_(__VA_ARGS__) { \
|
|
|
|
MOZ_CRASH("stylo: shouldn't be calling " #name_ "in a non-stylo build"); \
|
|
|
|
}
|
|
|
|
#include "ServoBindingList.h"
|
|
|
|
#undef SERVO_BINDING_FUNC
|
2016-04-27 05:34:49 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef MOZ_STYLO
|
|
|
|
const nsStyleVariables*
|
2016-08-16 08:10:05 +03:00
|
|
|
Servo_GetStyleVariables(ServoComputedValuesBorrowed aComputedValues)
|
2016-04-27 05:34:49 +03:00
|
|
|
{
|
|
|
|
// Servo can't provide us with Variables structs yet, so instead of linking
|
|
|
|
// to a Servo_GetStyleVariables defined in Servo we define one here that
|
|
|
|
// always returns the same, empty struct.
|
|
|
|
static nsStyleVariables variables(StyleStructContext::ServoContext());
|
|
|
|
return &variables;
|
|
|
|
}
|
2016-02-26 04:51:01 +03:00
|
|
|
#endif
|