mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:30:27 +00:00
5499b4356b
- Bug 1155371 - Include DOMMediaStream and MediaSource object URLs in memory reports. r=khuey (b62513f5c6)
- Bug 1187782 (part 1) - Replace nsBaseHashtable::EnumerateRead() calls in dom/base/ with iterators. r=khuey. (aa14b536c4)
- Bug 1187782 (part 2) - Replace nsBaseHashtable::EnumerateRead() calls in dom/base/ with iterators. r=khuey. (1645a9b910)
- Bug 1187782 (part 3) - Replace nsBaseHashtable::EnumerateRead() calls in dom/base/ with iterators. r=khuey. (67a83f37d5)
- Bug 1187782 (part 4) - Replace nsBaseHashtable::EnumerateRead() calls in dom/base/ with iterators. r=khuey. (bf1f7a8dfe)
- Bug 1187782 (part 5) - Replace nsBaseHashtable::EnumerateRead() calls in dom/base/ with iterators. r=khuey. (b76f1f34c8)
- Bug 1187782 (part 6) - Replace nsBaseHashtable::EnumerateRead() calls in dom/base/ with iterators. r=khuey. (f44312f3a9)
- Bug 1187782 (part 7) - Replace nsBaseHashtable::EnumerateRead() calls in dom/base/ with iterators. r=khuey. (33bb281955)
- Bug 1187782 (part 8) - Replace nsBaseHashtable::EnumerateRead() calls in dom/base/ with iterators. r=khuey. (52f7bc040e)
- Bug 1187782 (part 9) - Replace nsBaseHashtable::EnumerateRead() calls in dom/base/ with iterators. r=khuey. (7e797025a3)
- Bug 1207355 (Part 1) - Stop requesting decodes in nsDocument. r=tn (b2e6eb49c0)
- Bug 1202085 - Part 1: Remove imgICache::RemoveEntry(); r=seth (b5e4e2242f)
- Bug 1202085 - Part 2: Add an optional document argument to imgICache::FindEntryProperties(); r=seth (3f3138e5ba)
- Bug 1202085 - Part 3: Relax the assertion in ServiceWorkerManager::IsControlled to only happen when we think the document is controlled; r=jdm (6a829a6985)
- Bug 1202085 - Part 4: Add an ID for controlled document to the image cache key; r=seth (525ca69231)
- Bug 1202085 - Part 5: Add an automated test for the interaction of image cache with controlled documents; r=jdm (627e28d58d)
- Bug 1202085 - Part 6: Clear the entries in the image cache belonging to a controlled document when it gets destroyed; r=seth (935b858eac)
- Bug 1221279 Don't crash while clearing imgLoader cache when disconnected document is destroyed. r=smaug (040d3fa2be)
- Bug 1207084 - keep sheet level on tokenstream for use in ruledata when resolving variables, to fix hcm issues with CSS variables, r=heycam (2d8f5af005)
- Bug 1207084 - addendum: fix operator== for nsCSSValueTokenStream, implied-r=heycam (ee66d21638)
- Bug 1209124 - Bail out of nsStyleSet::GatherRuleProcessors and PresShell::Observe if we're called during/after shutdown. r=dbaron (7043eb3ee7)
- Bug 1216043 - Rename nsStyleSheet::sheetType and make it an enum class. r=dbaron (f03c5328e6)
- Bug 1168158 - Make URLValue::mString a smart pointer. r=heycam (06e37619be)
- Bug 1216038 - Deduce underlying integer type for MakeEnumeratedRange starting at 0. r=froydnj (883c74fb6b)
- Bug 1216040 - Make ArrayLength support mozilla::EnumeratedArray. r=froydnj (d1fc4aa43c)
- Bug 1216041 - Add ranged iterator support to mozilla::{Array,RangedArray,EnumerationArray}. r=froydnj (bf71f0ea1f)
- Bug 1128787 part.1 Add test r=ehsan (97016bbd85)
- Bug 1128787 part.2 nsHTMLDocument should clear focus before making itself editable when designMode is enabled and it makes the focused content non-focusable r=smaug+enndeakin (eccab99994)
- Bug 1171342 - Store contenteditable.css and designmode.css in the style sheet cache. r=jwatt (7e6aa290d2)
- Bug 1164279 - Disable font-variant-east-asian: ruby for <rt> on Windows. r=jfkthame (04117062d7)
- Bug 1126230 part 9 - Remove fullscreen override and related test. r=dbaron (18818d67fa)
- Bug 1126230 part 1 - Use delegated constructor to simplify constructor of nsFrameConstructorState. r=bz (5c1193f9f5)
- Bug 1126230 part 2 - Refactor part of nsFrameConstructorState::AddChild. r=bz (3c02f9e8c2)
- Bug 1126230 part 3 - Add :-moz-browser-frame pseudo class for HTML browser frame elements. r=dbaron (6a7b01f092)
- Bug 1211040 - Restrict -moz-window-{dragging,shadow} to be chrome-only. r=dbaron (005f02f673)
- Bug 1126230 part 4 - Add -moz-top-layer internal CSS property and set it for fullscreen elements. r=dbaron (02d841cab7)
- Bug 1203405 - Ignore order of abs/fixed-pos frames in the frame list. r=roc (d67696d0ea)
- Bug 1126230 part 5 - Give proper geometric parent for top layer frames. r=bz,dbaron (19504f223e)
- Bug 1126230 part 6 - Add nsIDocument::GetFullscreenStack() method. r=smaug (d401d534d1)
- Bug 1126230 part 7 - Add static method nsDisplayListBuilder::GetOutOfFlowData(). r=roc (3bdafef9b2)
- Bug 1126230 part 8 - Implement painting part for the top layer. r=roc (8875115dc1)
- Bug 1126230 part 10 - Add test for fullscreen top layer. r=dbaron (d095d85cd0)
- Bug 1069192 part 1 - Force users of nsCSSProps::IsEnabled() to pass in the enabled state. r=dbaron (38e40ebf1d)
- Bug 1195978 - set line number when re-parsing sheet. r=heycam (41cdf15319)
- Bug 1202095 - re-use @imported style sheets from inIDOMUtils.parseStyleSheet. r=heycam (2f54fdacf0)
- Bug 1099557 followup - Put the default setting for control-character visibility behind a runtime pref, and keep it off-by-default on release builds for now. r=dbaron (3ada076d9b)
- Bug 1069192 part 2 - Add a flag for chrome-only properties and change semantics of enabling flags. r=dbaron (2e887397ac)
- Bug 1220496 - Create PresHint/StyleAttr rule processors in nsStyleSet::Init and make DirtyRuleProcessors private. r=bzbarsky (0b538fe7cc)
- No bug - Cleanup the layout/style mochitest manifest. (e26676496f)
- Bug 1183461 part 1 - Move InitialAdvance to CSSAnimation; r=heycam (dac91a759a)
- Bug 1183461 part 2 - Reorder the parameters to EventInfo constructors; r=heycam (29fc1b2096)
- Bug 1183461 part 3 - Add Animation::AnimationTimeToTimeStamp; r=heycam (d3410ccc01)
- Bug 1183461 part 4 - Add CSSAnimation::ElapsedTimeToTimeStamp; r=heycam (10bac42e8e)
- Bug 1183461 part 5 - Store event TimeStamp along with event info; r=heycam (0250a63ee8)
- Bug 1220599 - Fix erroneous nsHTMLDocument::Open comment to note that we reuse the document rather than creating a new one. r=baku (5d492865d3)
- Bug 1220827 - Replace question comment in nsHTMLDocument::Open with an explanatory comment to explain why we create a new Window. r=smaug (69f16ef5dc)
- missing bits of Bug 1170911 - Add pref for switching off execCommand("cut"/"copy"). r=ehsan (613644fa39)
- Bug 1183461 part 6 - Store the generating animation on the event; r=heycam (c490af70cc)
- Bug 1194037 part 4 - Remove ticking from FlushAnimations; r=dholbert (8df1f3eec4)
- Bug 1200568 - Don't create animations for elements that are not part of the document tree; r=dbaron (3027949d8b)
- Bug 1183461 part 7 - Add EventInfoComparator and sort events; r=heycam (550aad09bc)
- Bug 1183461 part 8 - Add tests for event order dispatch; r=heycam (964ee5fe23)
- Bug 978833 patch 1 - Add mochitest for bug 978833. r=heycam (4b4579d3f7)
- Bug 1208951 - Part 1: Split half of KeyframeValueEntry into a base class. r=birtles (32d7cbd477)
- Bug 1208951 - Part 2: Use a comparator object instead of operator< onKeyframeValueEntry. r=birtles (3b83723ef0)
- Bug 1209603 patch 1 - Add comments to RuleNodeCacheConditions.h . r=heycam (444eabdddc)
- Bug 1209603 patch 2 - Reduce calls to StyleVisibility() in WritingMode constructor. r=heycam (f079fb5765)
- Bug 1209603 patch 3 - Don't call SetFontSizeDependency for 0em. r=heycam (7a87fe5229)
- Bug 1209603 patch 4 - Add reftest for bug 1209603. r=heycam (1680144fc2)
- Bug 1209603 patch 5 - Move inline method nsStyleContext::GetCachedStyleData into header file, and make it public. r=heycam (32c38f790e)
- Bug 1209603 patch 6 - Prepare to use a different meaning of mBits when cached style data pointer is null. r=heycam (15e6471e58)
- Bug 1209603 patch 7 - Add assertions that we don't ask the rule node for data when we have cached data on the style context. r=heycam (0f27687446)
- Bug 1209603 patch 8 - Record in mBits when we have gotten a reset style struct that is cached on the rule node. r=heycam (b5e693a986)
- Bug 1209603 patch 9 - Cache inherited style structs on the style context when we found already-cached data in the rule tree. r=heycam (bfa6a169cb)
- Bug 1209603 patch 10 - Make PeekStyle* exact, i.e., guaranteed to return null if we haven't computed the data for this context. r=heycam (6ffdf7f0bf)
- Bug 1209603 patch 11 - Assert that PeekStyle* results don't change during difference computation. r=heycam (a3d7336dce)
- Bug 1193518 - Ensure that when we update a WritingMode after resolving dir=auto, we set the inline-dir bit appropriately as well as the inline-bidi bit. r=dholbert (072c062ddc)
- Bug 1216431 patch 1 - Give WritingMode an nsStyleVisibility* constructor. r=heycam (b7efb1204c)
- Bug 1216431 patch 2 - Pass nsStyleVisibility to nsStylePosition::CalcDifference to avoid computing new nsStyleVisibility during nsStyleContext::CalcStyleDifference. r=heycam (b842123d9b)
- Bug 1216431 patch 3 - Cache structs that are stored with conditions on the rule node all the time, rather than only when freshly computed. r=heycam (d4ea9adc19)
- Bug 1216431 patch 4 - Don't trigger computation of new structs via testing of conditions for conditionally-stored structs on the rule node. r=heycam (95cc7e8a52)
- Bug 1216431 patch 5 - Back out half of bug 1209603 patch 10. r=heycam (5949a2e9d0)
- Bug 1216431 patch 6 - Back out bug 1209603 patch 8. r=heycam (45a05ad60e)
- Bug 1209603 patch 6b - Rename nsStyleContext::HasCachedInheritedStyleData to HasCachedDependentStyleData. r=heycam (c4f52f00d0)
- Bug 1216431 patch 7 - Back out bug 1209603 patch 6. r=heycam (6eb8677af2)
- Bug 1216431 patch 8 - Fix whitespace error I made when addressing review comments. (19abf29215)
- Bug 1214958 - Add operators &=, |=, and ^= to mozilla::DebugOnly. r=froydnj (c76229e817)
- Bug 1208951 - Part 3: Expose nsCSSValue -> nsTimingFunction computation function. r=birtles (f0124bbbf2)
- Bug 1208951 - Part 4: Add an nsCSSParser entry point to parse a single longhand property and return an nsCSSValue. r=birtles (a317aa3255)
- Bug 1208951 - Part 5: Add nsCSSProps method to look a property by its IDL name. r=bzbarsky (1d0b78374d)
- Bug 1208951 - Part 6: Add method to compare nsCSSPropertySets for equality. r=birtles (9039197290)
- Bug 1208951 - Part 7: Add a StyleAnimationValue::ComputeValues method to compute components of a shorthand. r=birtles (36149d8c82)
- Bug 1208951 - Part 8: Don't warn if StyleAnimationValue::ComputeValue fails to parse a CSS property. r=birtles (12e06d0356)
- Bug 1208385 part 1 - Store a pointer to the owning animation on each KeyframeEffect; r=heycam (8f48533047)
- Bug 1208385 part 2 - Remove stored parent time from KeyframeEffectReadOnly and get the time directly from the owning animation; r=heycam (7bd25845be)
- Bug 1208951 - Part 9: Implement KeyframeEffectReadOnly constructor. r=bzbarsky r=birtles (49049edfac)
- Bug 1208951 - Part 10: Make GetFrames aware of initial/final zero-length segments and discontinuities between segments. r=birtles (71066c4941)
- Bug 1208951 - Part 11: Tests. r=birtles (308ad9ba9f)
- Bug 1210575 part 1: Rename CSS parser function "ParseWebkitPrefixedGradient()" to be more specific, now that we'll have several ways of parsing these expressions. r=heycam (060a01b43a)
- Bug 1210575 part 2: Refactor out the body of a long compound "if" expression into a helper function, IsFunctionTokenValidForBackgroundImage. r=heycam (ed784fb98d)
- Bug 1210575 part 3: Refactor CSS gradient-parsing code to use a flags bitfield instead of multiple bool args for customizing behavior. r=heycam (5c9df6466c)
- Bug 1210575 part 4: Prevent CSS parser from invoking CSSUnprefixingService if native webkit prefix support is enabled. r=heycam (d4c7997c94)
- Bug 1210575 part 5: Allow CSS -webkit-{linear|radial}-gradient expressions (& 'repeating' variants) to make it to our gradient-parsing code, if webkit prefix support is preffed on. r=heycam (4113bb47c4)
- Bug 1207002 - Use CSS property flag to restrict MathML-related internal properties. r=dbaron (9b021c7819)
- Bug 1199610 - Correctly handle parsing failures inside CSS functions. r=dbaron (5255fe445d)
- Bug 1210575 part 6: Implement -webkit-radial-gradient parse quirks with contain/cover keywords, sized shapes, "at" keyword, & angles. r=heycam (ba1002fbae)
- Bug 1210575 part 7: Refactor linear-gradient parsing code slightly & add/extend comments. r=heycam (1bd144ff04)
- Bug 1210575 part 8: Implement -webkit-linear-gradient parse quirks with "to", angles, and "center". r=heycam (97587d49ea)
- Bug 1210575 part 9: Remove some obsolete XXXdholbert comments from property_database.js. (no review, comment-only so DONTBUILD) (961a2c43e9)
- Bug 978833 patch 2 - Make css::Declaration reference-counted. r=heycam (4aab911856)
1283 lines
37 KiB
C++
1283 lines
37 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* 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/ArrayUtils.h"
|
|
#include "mozilla/EventStates.h"
|
|
|
|
#include "inDOMUtils.h"
|
|
#include "inLayoutUtils.h"
|
|
|
|
#include "nsIServiceManager.h"
|
|
#include "nsISupportsArray.h"
|
|
#include "nsString.h"
|
|
#include "nsIStyleSheetLinkingElement.h"
|
|
#include "nsIDOMElement.h"
|
|
#include "nsIDocument.h"
|
|
#include "nsIPresShell.h"
|
|
#include "nsIDOMDocument.h"
|
|
#include "nsIDOMCharacterData.h"
|
|
#include "nsRuleNode.h"
|
|
#include "nsIStyleRule.h"
|
|
#include "mozilla/css/StyleRule.h"
|
|
#include "nsICSSStyleRuleDOMWrapper.h"
|
|
#include "nsIDOMWindow.h"
|
|
#include "nsXBLBinding.h"
|
|
#include "nsXBLPrototypeBinding.h"
|
|
#include "nsIMutableArray.h"
|
|
#include "nsBindingManager.h"
|
|
#include "ChildIterator.h"
|
|
#include "nsComputedDOMStyle.h"
|
|
#include "mozilla/EventStateManager.h"
|
|
#include "nsIAtom.h"
|
|
#include "nsRange.h"
|
|
#include "nsContentList.h"
|
|
#include "mozilla/CSSStyleSheet.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "nsRuleWalker.h"
|
|
#include "nsRuleProcessorData.h"
|
|
#include "nsCSSRuleProcessor.h"
|
|
#include "mozilla/dom/CSSLexer.h"
|
|
#include "mozilla/dom/InspectorUtilsBinding.h"
|
|
#include "mozilla/dom/ToJSValue.h"
|
|
#include "nsCSSParser.h"
|
|
#include "nsCSSProps.h"
|
|
#include "nsCSSValue.h"
|
|
#include "nsColor.h"
|
|
#include "nsStyleSet.h"
|
|
#include "nsStyleUtil.h"
|
|
#include "nsQueryObject.h"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::css;
|
|
using namespace mozilla::dom;
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
inDOMUtils::inDOMUtils()
|
|
{
|
|
}
|
|
|
|
inDOMUtils::~inDOMUtils()
|
|
{
|
|
}
|
|
|
|
NS_IMPL_ISUPPORTS(inDOMUtils, inIDOMUtils)
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// inIDOMUtils
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetAllStyleSheets(nsIDOMDocument *aDocument, uint32_t *aLength,
|
|
nsISupports ***aSheets)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aDocument);
|
|
|
|
nsCOMArray<nsIStyleSheet> sheets;
|
|
|
|
nsCOMPtr<nsIDocument> document = do_QueryInterface(aDocument);
|
|
MOZ_ASSERT(document);
|
|
|
|
// Get the agent, then user and finally xbl sheets in the style set.
|
|
nsIPresShell* presShell = document->GetShell();
|
|
if (presShell) {
|
|
nsStyleSet* styleSet = presShell->StyleSet();
|
|
SheetType sheetType = SheetType::Agent;
|
|
for (int32_t i = 0; i < styleSet->SheetCount(sheetType); i++) {
|
|
sheets.AppendElement(styleSet->StyleSheetAt(sheetType, i));
|
|
}
|
|
sheetType = SheetType::User;
|
|
for (int32_t i = 0; i < styleSet->SheetCount(sheetType); i++) {
|
|
sheets.AppendElement(styleSet->StyleSheetAt(sheetType, i));
|
|
}
|
|
nsAutoTArray<CSSStyleSheet*, 32> xblSheetArray;
|
|
styleSet->AppendAllXBLStyleSheets(xblSheetArray);
|
|
|
|
// The XBL stylesheet array will quite often be full of duplicates. Cope:
|
|
nsTHashtable<nsPtrHashKey<CSSStyleSheet>> sheetSet;
|
|
for (CSSStyleSheet* sheet : xblSheetArray) {
|
|
if (!sheetSet.Contains(sheet)) {
|
|
sheetSet.PutEntry(sheet);
|
|
sheets.AppendElement(sheet);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Get the document sheets.
|
|
for (int32_t i = 0; i < document->GetNumberOfStyleSheets(); i++) {
|
|
sheets.AppendElement(document->GetStyleSheetAt(i));
|
|
}
|
|
|
|
nsISupports** ret = static_cast<nsISupports**>(NS_Alloc(sheets.Count() *
|
|
sizeof(nsISupports*)));
|
|
|
|
for (int32_t i = 0; i < sheets.Count(); i++) {
|
|
NS_ADDREF(ret[i] = sheets[i]);
|
|
}
|
|
|
|
*aLength = sheets.Count();
|
|
*aSheets = ret;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::IsIgnorableWhitespace(nsIDOMCharacterData *aDataNode,
|
|
bool *aReturn)
|
|
{
|
|
NS_PRECONDITION(aReturn, "Must have an out parameter");
|
|
|
|
NS_ENSURE_ARG_POINTER(aDataNode);
|
|
|
|
*aReturn = false;
|
|
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aDataNode);
|
|
NS_ASSERTION(content, "Does not implement nsIContent!");
|
|
|
|
if (!content->TextIsOnlyWhitespace()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
// Okay. We have only white space. Let's check the white-space
|
|
// property now and make sure that this isn't preformatted text...
|
|
nsIFrame* frame = content->GetPrimaryFrame();
|
|
if (frame) {
|
|
const nsStyleText* text = frame->StyleText();
|
|
*aReturn = !text->WhiteSpaceIsSignificant();
|
|
}
|
|
else {
|
|
// empty inter-tag text node without frame, e.g., in between <table>\n<tr>
|
|
*aReturn = true;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetParentForNode(nsIDOMNode* aNode,
|
|
bool aShowingAnonymousContent,
|
|
nsIDOMNode** aParent)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aNode);
|
|
|
|
// First do the special cases -- document nodes and anonymous content
|
|
nsCOMPtr<nsIDocument> doc(do_QueryInterface(aNode));
|
|
nsCOMPtr<nsIDOMNode> parent;
|
|
|
|
if (doc) {
|
|
parent = inLayoutUtils::GetContainerFor(*doc);
|
|
} else if (aShowingAnonymousContent) {
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aNode);
|
|
if (content) {
|
|
nsIContent* bparent = content->GetFlattenedTreeParent();
|
|
parent = do_QueryInterface(bparent);
|
|
}
|
|
}
|
|
|
|
if (!parent) {
|
|
// Ok, just get the normal DOM parent node
|
|
aNode->GetParentNode(getter_AddRefs(parent));
|
|
}
|
|
|
|
NS_IF_ADDREF(*aParent = parent);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetChildrenForNode(nsIDOMNode* aNode,
|
|
bool aShowingAnonymousContent,
|
|
nsIDOMNodeList** aChildren)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aNode);
|
|
NS_PRECONDITION(aChildren, "Must have an out parameter");
|
|
|
|
nsCOMPtr<nsIDOMNodeList> kids;
|
|
|
|
if (aShowingAnonymousContent) {
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aNode);
|
|
if (content) {
|
|
kids = content->GetChildren(nsIContent::eAllChildren);
|
|
}
|
|
}
|
|
|
|
if (!kids) {
|
|
aNode->GetChildNodes(getter_AddRefs(kids));
|
|
}
|
|
|
|
kids.forget(aChildren);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetCSSStyleRules(nsIDOMElement *aElement,
|
|
const nsAString& aPseudo,
|
|
nsISupportsArray **_retval)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aElement);
|
|
|
|
*_retval = nullptr;
|
|
|
|
nsCOMPtr<nsIAtom> pseudoElt;
|
|
if (!aPseudo.IsEmpty()) {
|
|
pseudoElt = do_GetAtom(aPseudo);
|
|
}
|
|
|
|
nsRuleNode* ruleNode = nullptr;
|
|
nsCOMPtr<Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_STATE(element);
|
|
RefPtr<nsStyleContext> styleContext;
|
|
GetRuleNodeForElement(element, pseudoElt, getter_AddRefs(styleContext), &ruleNode);
|
|
if (!ruleNode) {
|
|
// This can fail for elements that are not in the document or
|
|
// if the document they're in doesn't have a presshell. Bail out.
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsISupportsArray> rules;
|
|
NS_NewISupportsArray(getter_AddRefs(rules));
|
|
if (!rules) return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
RefPtr<mozilla::css::StyleRule> cssRule;
|
|
for ( ; !ruleNode->IsRoot(); ruleNode = ruleNode->GetParent()) {
|
|
cssRule = do_QueryObject(ruleNode->GetRule());
|
|
if (cssRule) {
|
|
nsCOMPtr<nsIDOMCSSRule> domRule = cssRule->GetDOMRule();
|
|
if (domRule)
|
|
rules->InsertElementAt(domRule, 0);
|
|
}
|
|
}
|
|
|
|
rules.forget(_retval);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static already_AddRefed<StyleRule>
|
|
GetRuleFromDOMRule(nsIDOMCSSStyleRule *aRule, ErrorResult& rv)
|
|
{
|
|
nsCOMPtr<nsICSSStyleRuleDOMWrapper> rule = do_QueryInterface(aRule);
|
|
if (!rule) {
|
|
rv.Throw(NS_ERROR_INVALID_POINTER);
|
|
return nullptr;
|
|
}
|
|
|
|
RefPtr<StyleRule> cssrule;
|
|
rv = rule->GetCSSStyleRule(getter_AddRefs(cssrule));
|
|
if (rv.Failed()) {
|
|
return nullptr;
|
|
}
|
|
|
|
if (!cssrule) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
}
|
|
return cssrule.forget();
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetRuleLine(nsIDOMCSSRule* aRule, uint32_t* _retval)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aRule);
|
|
|
|
Rule* rule = aRule->GetCSSRule();
|
|
if (!rule) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
*_retval = rule->GetLineNumber();
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetRuleColumn(nsIDOMCSSRule* aRule, uint32_t* _retval)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aRule);
|
|
|
|
Rule* rule = aRule->GetCSSRule();
|
|
if (!rule) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
*_retval = rule->GetColumnNumber();
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetRelativeRuleLine(nsIDOMCSSRule* aRule, uint32_t* _retval)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aRule);
|
|
|
|
Rule* rule = aRule->GetCSSRule();
|
|
if (!rule) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
uint32_t lineNumber = rule->GetLineNumber();
|
|
CSSStyleSheet* sheet = rule->GetStyleSheet();
|
|
if (sheet) {
|
|
nsINode* owningNode = sheet->GetOwnerNode();
|
|
if (owningNode) {
|
|
nsCOMPtr<nsIStyleSheetLinkingElement> link =
|
|
do_QueryInterface(owningNode);
|
|
if (link) {
|
|
lineNumber -= link->GetLineNumber() - 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
*_retval = lineNumber;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetCSSLexer(const nsAString& aText, JSContext* aCx,
|
|
JS::MutableHandleValue aResult)
|
|
{
|
|
MOZ_ASSERT(JS::CurrentGlobalOrNull(aCx));
|
|
JS::Rooted<JSObject*> scope(aCx, JS::CurrentGlobalOrNull(aCx));
|
|
nsAutoPtr<CSSLexer> lexer(new CSSLexer(aText));
|
|
if (!WrapNewBindingNonWrapperCachedObject(aCx, scope, lexer, aResult)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetSelectorCount(nsIDOMCSSStyleRule* aRule, uint32_t *aCount)
|
|
{
|
|
ErrorResult rv;
|
|
RefPtr<StyleRule> rule = GetRuleFromDOMRule(aRule, rv);
|
|
if (rv.Failed()) {
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
uint32_t count = 0;
|
|
for (nsCSSSelectorList* sel = rule->Selector(); sel; sel = sel->mNext) {
|
|
++count;
|
|
}
|
|
*aCount = count;
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsCSSSelectorList*
|
|
GetSelectorAtIndex(nsIDOMCSSStyleRule* aRule, uint32_t aIndex, ErrorResult& rv)
|
|
{
|
|
RefPtr<StyleRule> rule = GetRuleFromDOMRule(aRule, rv);
|
|
if (rv.Failed()) {
|
|
return nullptr;
|
|
}
|
|
|
|
for (nsCSSSelectorList* sel = rule->Selector(); sel;
|
|
sel = sel->mNext, --aIndex) {
|
|
if (aIndex == 0) {
|
|
return sel;
|
|
}
|
|
}
|
|
|
|
// Ran out of selectors
|
|
rv.Throw(NS_ERROR_INVALID_ARG);
|
|
return nullptr;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetSelectorText(nsIDOMCSSStyleRule* aRule,
|
|
uint32_t aSelectorIndex,
|
|
nsAString& aText)
|
|
{
|
|
ErrorResult rv;
|
|
nsCSSSelectorList* sel = GetSelectorAtIndex(aRule, aSelectorIndex, rv);
|
|
if (rv.Failed()) {
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
RefPtr<StyleRule> rule = GetRuleFromDOMRule(aRule, rv);
|
|
MOZ_ASSERT(!rv.Failed(), "How could we get a selector but not a rule?");
|
|
|
|
sel->mSelectors->ToString(aText, rule->GetStyleSheet(), false);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetSpecificity(nsIDOMCSSStyleRule* aRule,
|
|
uint32_t aSelectorIndex,
|
|
uint64_t* aSpecificity)
|
|
{
|
|
ErrorResult rv;
|
|
nsCSSSelectorList* sel = GetSelectorAtIndex(aRule, aSelectorIndex, rv);
|
|
if (rv.Failed()) {
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
*aSpecificity = sel->mWeight;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::SelectorMatchesElement(nsIDOMElement* aElement,
|
|
nsIDOMCSSStyleRule* aRule,
|
|
uint32_t aSelectorIndex,
|
|
const nsAString& aPseudo,
|
|
bool* aMatches)
|
|
{
|
|
nsCOMPtr<Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(element);
|
|
|
|
ErrorResult rv;
|
|
nsCSSSelectorList* tail = GetSelectorAtIndex(aRule, aSelectorIndex, rv);
|
|
if (rv.Failed()) {
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
// We want just the one list item, not the whole list tail
|
|
nsAutoPtr<nsCSSSelectorList> sel(tail->Clone(false));
|
|
|
|
// Do not attempt to match if a pseudo element is requested and this is not
|
|
// a pseudo element selector, or vice versa.
|
|
if (aPseudo.IsEmpty() == sel->mSelectors->IsPseudoElement()) {
|
|
*aMatches = false;
|
|
return NS_OK;
|
|
}
|
|
|
|
if (!aPseudo.IsEmpty()) {
|
|
// We need to make sure that the requested pseudo element type
|
|
// matches the selector pseudo element type before proceeding.
|
|
nsCOMPtr<nsIAtom> pseudoElt = do_GetAtom(aPseudo);
|
|
if (sel->mSelectors->PseudoType() !=
|
|
nsCSSPseudoElements::GetPseudoType(pseudoElt)) {
|
|
*aMatches = false;
|
|
return NS_OK;
|
|
}
|
|
|
|
// We have a matching pseudo element, now remove it so we can compare
|
|
// directly against |element| when proceeding into SelectorListMatches.
|
|
// It's OK to do this - we just cloned sel and nothing else is using it.
|
|
sel->RemoveRightmostSelector();
|
|
}
|
|
|
|
element->OwnerDoc()->FlushPendingLinkUpdates();
|
|
// XXXbz what exactly should we do with visited state here?
|
|
TreeMatchContext matchingContext(false,
|
|
nsRuleWalker::eRelevantLinkUnvisited,
|
|
element->OwnerDoc(),
|
|
TreeMatchContext::eNeverMatchVisited);
|
|
*aMatches = nsCSSRuleProcessor::SelectorListMatches(element, matchingContext,
|
|
sel);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::IsInheritedProperty(const nsAString &aPropertyName, bool *_retval)
|
|
{
|
|
nsCSSProperty prop =
|
|
nsCSSProps::LookupProperty(aPropertyName, nsCSSProps::eIgnoreEnabledState);
|
|
if (prop == eCSSProperty_UNKNOWN) {
|
|
*_retval = false;
|
|
return NS_OK;
|
|
}
|
|
|
|
if (prop == eCSSPropertyExtra_variable) {
|
|
*_retval = true;
|
|
return NS_OK;
|
|
}
|
|
|
|
if (nsCSSProps::IsShorthand(prop)) {
|
|
prop = nsCSSProps::SubpropertyEntryFor(prop)[0];
|
|
}
|
|
|
|
nsStyleStructID sid = nsCSSProps::kSIDTable[prop];
|
|
*_retval = !nsCachedStyleData::IsReset(sid);
|
|
return NS_OK;
|
|
}
|
|
|
|
extern const char* const kCSSRawProperties[];
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetCSSPropertyNames(uint32_t aFlags, uint32_t* aCount,
|
|
char16_t*** aProps)
|
|
{
|
|
// maxCount is the largest number of properties we could have; our actual
|
|
// number might be smaller because properties might be disabled.
|
|
uint32_t maxCount;
|
|
if (aFlags & EXCLUDE_SHORTHANDS) {
|
|
maxCount = eCSSProperty_COUNT_no_shorthands;
|
|
} else {
|
|
maxCount = eCSSProperty_COUNT;
|
|
}
|
|
|
|
if (aFlags & INCLUDE_ALIASES) {
|
|
maxCount += (eCSSProperty_COUNT_with_aliases - eCSSProperty_COUNT);
|
|
}
|
|
|
|
char16_t** props =
|
|
static_cast<char16_t**>(moz_xmalloc(maxCount * sizeof(char16_t*)));
|
|
|
|
#define DO_PROP(_prop) \
|
|
PR_BEGIN_MACRO \
|
|
nsCSSProperty cssProp = nsCSSProperty(_prop); \
|
|
if (nsCSSProps::IsEnabled(cssProp, nsCSSProps::eEnabledForAllContent)) { \
|
|
props[propCount] = \
|
|
ToNewUnicode(nsDependentCString(kCSSRawProperties[_prop])); \
|
|
++propCount; \
|
|
} \
|
|
PR_END_MACRO
|
|
|
|
// prop is the property id we're considering; propCount is how many properties
|
|
// we've put into props so far.
|
|
uint32_t prop = 0, propCount = 0;
|
|
for ( ; prop < eCSSProperty_COUNT_no_shorthands; ++prop) {
|
|
if (nsCSSProps::PropertyParseType(nsCSSProperty(prop)) !=
|
|
CSS_PROPERTY_PARSE_INACCESSIBLE) {
|
|
DO_PROP(prop);
|
|
}
|
|
}
|
|
|
|
if (!(aFlags & EXCLUDE_SHORTHANDS)) {
|
|
for ( ; prop < eCSSProperty_COUNT; ++prop) {
|
|
// Some shorthands are also aliases
|
|
if ((aFlags & INCLUDE_ALIASES) ||
|
|
!nsCSSProps::PropHasFlags(nsCSSProperty(prop),
|
|
CSS_PROPERTY_IS_ALIAS)) {
|
|
DO_PROP(prop);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (aFlags & INCLUDE_ALIASES) {
|
|
for (prop = eCSSProperty_COUNT; prop < eCSSProperty_COUNT_with_aliases; ++prop) {
|
|
DO_PROP(prop);
|
|
}
|
|
}
|
|
|
|
#undef DO_PROP
|
|
|
|
*aCount = propCount;
|
|
*aProps = props;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static void InsertNoDuplicates(nsTArray<nsString>& aArray,
|
|
const nsAString& aString)
|
|
{
|
|
size_t i = aArray.IndexOfFirstElementGt(aString);
|
|
if (i > 0 && aArray[i-1].Equals(aString)) {
|
|
return;
|
|
}
|
|
aArray.InsertElementAt(i, aString);
|
|
}
|
|
|
|
static void GetKeywordsForProperty(const nsCSSProperty aProperty,
|
|
nsTArray<nsString>& aArray)
|
|
{
|
|
if (nsCSSProps::IsShorthand(aProperty)) {
|
|
// Shorthand props have no keywords.
|
|
return;
|
|
}
|
|
const nsCSSProps::KTableValue *keywordTable =
|
|
nsCSSProps::kKeywordTableTable[aProperty];
|
|
if (keywordTable) {
|
|
size_t i = 0;
|
|
while (nsCSSKeyword(keywordTable[i]) != eCSSKeyword_UNKNOWN) {
|
|
nsCSSKeyword word = nsCSSKeyword(keywordTable[i]);
|
|
InsertNoDuplicates(aArray,
|
|
NS_ConvertASCIItoUTF16(nsCSSKeywords::GetStringValue(word)));
|
|
// Increment counter by 2, because in this table every second
|
|
// element is a nsCSSKeyword.
|
|
i += 2;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void GetColorsForProperty(const uint32_t aParserVariant,
|
|
nsTArray<nsString>& aArray)
|
|
{
|
|
if (aParserVariant & VARIANT_COLOR) {
|
|
// GetKeywordsForProperty and GetOtherValuesForProperty assume aArray is sorted,
|
|
// and if aArray is not empty here, then it's not going to be sorted coming out.
|
|
MOZ_ASSERT(aArray.Length() == 0);
|
|
size_t size;
|
|
const char * const *allColorNames = NS_AllColorNames(&size);
|
|
for (size_t i = 0; i < size; i++) {
|
|
CopyASCIItoUTF16(allColorNames[i], *aArray.AppendElement());
|
|
}
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("currentColor"));
|
|
}
|
|
return;
|
|
}
|
|
|
|
static void GetOtherValuesForProperty(const uint32_t aParserVariant,
|
|
nsTArray<nsString>& aArray)
|
|
{
|
|
if (aParserVariant & VARIANT_AUTO) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("auto"));
|
|
}
|
|
if (aParserVariant & VARIANT_NORMAL) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("normal"));
|
|
}
|
|
if(aParserVariant & VARIANT_ALL) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("all"));
|
|
}
|
|
if (aParserVariant & VARIANT_NONE) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("none"));
|
|
}
|
|
if (aParserVariant & VARIANT_ELEMENT) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("-moz-element"));
|
|
}
|
|
if (aParserVariant & VARIANT_IMAGE_RECT) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("-moz-image-rect"));
|
|
}
|
|
if (aParserVariant & VARIANT_COLOR) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("rgb"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("hsl"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("rgba"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("hsla"));
|
|
}
|
|
if (aParserVariant & VARIANT_TIMING_FUNCTION) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("cubic-bezier"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("steps"));
|
|
}
|
|
if (aParserVariant & VARIANT_CALC) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("calc"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("-moz-calc"));
|
|
}
|
|
if (aParserVariant & VARIANT_URL) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("url"));
|
|
}
|
|
if (aParserVariant & VARIANT_GRADIENT) {
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("linear-gradient"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("radial-gradient"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("repeating-linear-gradient"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("repeating-radial-gradient"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("-moz-linear-gradient"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("-moz-radial-gradient"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("-moz-repeating-linear-gradient"));
|
|
InsertNoDuplicates(aArray, NS_LITERAL_STRING("-moz-repeating-radial-gradient"));
|
|
}
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetSubpropertiesForCSSProperty(const nsAString& aProperty,
|
|
uint32_t* aLength,
|
|
char16_t*** aValues)
|
|
{
|
|
nsCSSProperty propertyID =
|
|
nsCSSProps::LookupProperty(aProperty, nsCSSProps::eEnabledForAllContent);
|
|
|
|
if (propertyID == eCSSProperty_UNKNOWN) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
if (propertyID == eCSSPropertyExtra_variable) {
|
|
*aValues = static_cast<char16_t**>(moz_xmalloc(sizeof(char16_t*)));
|
|
(*aValues)[0] = ToNewUnicode(aProperty);
|
|
*aLength = 1;
|
|
return NS_OK;
|
|
}
|
|
|
|
if (!nsCSSProps::IsShorthand(propertyID)) {
|
|
*aValues = static_cast<char16_t**>(moz_xmalloc(sizeof(char16_t*)));
|
|
(*aValues)[0] = ToNewUnicode(nsCSSProps::GetStringValue(propertyID));
|
|
*aLength = 1;
|
|
return NS_OK;
|
|
}
|
|
|
|
// Count up how many subproperties we have.
|
|
size_t subpropCount = 0;
|
|
for (const nsCSSProperty *props = nsCSSProps::SubpropertyEntryFor(propertyID);
|
|
*props != eCSSProperty_UNKNOWN; ++props) {
|
|
++subpropCount;
|
|
}
|
|
|
|
*aValues =
|
|
static_cast<char16_t**>(moz_xmalloc(subpropCount * sizeof(char16_t*)));
|
|
*aLength = subpropCount;
|
|
for (const nsCSSProperty *props = nsCSSProps::SubpropertyEntryFor(propertyID),
|
|
*props_start = props;
|
|
*props != eCSSProperty_UNKNOWN; ++props) {
|
|
(*aValues)[props-props_start] = ToNewUnicode(nsCSSProps::GetStringValue(*props));
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::CssPropertyIsShorthand(const nsAString& aProperty, bool *_retval)
|
|
{
|
|
nsCSSProperty propertyID =
|
|
nsCSSProps::LookupProperty(aProperty, nsCSSProps::eEnabledForAllContent);
|
|
if (propertyID == eCSSProperty_UNKNOWN) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
if (propertyID == eCSSPropertyExtra_variable) {
|
|
*_retval = false;
|
|
} else {
|
|
*_retval = nsCSSProps::IsShorthand(propertyID);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
// A helper function that determines whether the given property
|
|
// supports the given type.
|
|
static bool
|
|
PropertySupportsVariant(nsCSSProperty aPropertyID, uint32_t aVariant)
|
|
{
|
|
if (nsCSSProps::IsShorthand(aPropertyID)) {
|
|
// We need a special case for border here, because while it resets
|
|
// border-image, it can't actually parse an image.
|
|
if (aPropertyID == eCSSProperty_border) {
|
|
return (aVariant & (VARIANT_COLOR | VARIANT_LENGTH)) != 0;
|
|
}
|
|
|
|
for (const nsCSSProperty* props = nsCSSProps::SubpropertyEntryFor(aPropertyID);
|
|
*props != eCSSProperty_UNKNOWN; ++props) {
|
|
if (PropertySupportsVariant(*props, aVariant)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Properties that are parsed by functions must have their
|
|
// attributes hand-maintained here.
|
|
if (nsCSSProps::PropHasFlags(aPropertyID, CSS_PROPERTY_VALUE_PARSER_FUNCTION) ||
|
|
nsCSSProps::PropertyParseType(aPropertyID) == CSS_PROPERTY_PARSE_FUNCTION) {
|
|
// These must all be special-cased.
|
|
uint32_t supported;
|
|
switch (aPropertyID) {
|
|
case eCSSProperty_border_image_slice:
|
|
case eCSSProperty_grid_template:
|
|
case eCSSProperty_grid:
|
|
supported = VARIANT_PN;
|
|
break;
|
|
|
|
case eCSSProperty_border_image_outset:
|
|
supported = VARIANT_LN;
|
|
break;
|
|
|
|
case eCSSProperty_border_image_width:
|
|
case eCSSProperty_stroke_dasharray:
|
|
supported = VARIANT_LPN;
|
|
break;
|
|
|
|
case eCSSProperty_border_top_left_radius:
|
|
case eCSSProperty_border_top_right_radius:
|
|
case eCSSProperty_border_bottom_left_radius:
|
|
case eCSSProperty_border_bottom_right_radius:
|
|
case eCSSProperty_background_position:
|
|
case eCSSProperty_background_size:
|
|
case eCSSProperty_grid_auto_columns:
|
|
case eCSSProperty_grid_auto_rows:
|
|
case eCSSProperty_grid_template_columns:
|
|
case eCSSProperty_grid_template_rows:
|
|
case eCSSProperty_object_position:
|
|
case eCSSProperty_scroll_snap_coordinate:
|
|
case eCSSProperty_scroll_snap_destination:
|
|
case eCSSProperty_transform_origin:
|
|
case eCSSProperty_perspective_origin:
|
|
case eCSSProperty__moz_outline_radius_topLeft:
|
|
case eCSSProperty__moz_outline_radius_topRight:
|
|
case eCSSProperty__moz_outline_radius_bottomLeft:
|
|
case eCSSProperty__moz_outline_radius_bottomRight:
|
|
supported = VARIANT_LP;
|
|
break;
|
|
|
|
case eCSSProperty_border_bottom_colors:
|
|
case eCSSProperty_border_left_colors:
|
|
case eCSSProperty_border_right_colors:
|
|
case eCSSProperty_border_top_colors:
|
|
supported = VARIANT_COLOR;
|
|
break;
|
|
|
|
case eCSSProperty_text_shadow:
|
|
case eCSSProperty_box_shadow:
|
|
supported = VARIANT_LENGTH | VARIANT_COLOR;
|
|
break;
|
|
|
|
case eCSSProperty_border_spacing:
|
|
supported = VARIANT_LENGTH;
|
|
break;
|
|
|
|
case eCSSProperty_content:
|
|
case eCSSProperty_cursor:
|
|
case eCSSProperty_clip_path:
|
|
supported = VARIANT_URL;
|
|
break;
|
|
|
|
case eCSSProperty_fill:
|
|
case eCSSProperty_stroke:
|
|
supported = VARIANT_COLOR | VARIANT_URL;
|
|
break;
|
|
|
|
case eCSSProperty_image_orientation:
|
|
supported = VARIANT_ANGLE;
|
|
break;
|
|
|
|
case eCSSProperty_filter:
|
|
supported = VARIANT_URL;
|
|
break;
|
|
|
|
case eCSSProperty_grid_column_start:
|
|
case eCSSProperty_grid_column_end:
|
|
case eCSSProperty_grid_row_start:
|
|
case eCSSProperty_grid_row_end:
|
|
case eCSSProperty_font_weight:
|
|
supported = VARIANT_NUMBER;
|
|
break;
|
|
|
|
default:
|
|
supported = 0;
|
|
break;
|
|
}
|
|
|
|
return (supported & aVariant) != 0;
|
|
}
|
|
|
|
return (nsCSSProps::ParserVariant(aPropertyID) & aVariant) != 0;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::CssPropertySupportsType(const nsAString& aProperty, uint32_t aType,
|
|
bool *_retval)
|
|
{
|
|
nsCSSProperty propertyID =
|
|
nsCSSProps::LookupProperty(aProperty, nsCSSProps::eEnabledForAllContent);
|
|
if (propertyID == eCSSProperty_UNKNOWN) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
if (propertyID >= eCSSProperty_COUNT) {
|
|
*_retval = false;
|
|
return NS_OK;
|
|
}
|
|
|
|
uint32_t variant;
|
|
switch (aType) {
|
|
case TYPE_LENGTH:
|
|
variant = VARIANT_LENGTH;
|
|
break;
|
|
case TYPE_PERCENTAGE:
|
|
variant = VARIANT_PERCENT;
|
|
break;
|
|
case TYPE_COLOR:
|
|
variant = VARIANT_COLOR;
|
|
break;
|
|
case TYPE_URL:
|
|
variant = VARIANT_URL;
|
|
break;
|
|
case TYPE_ANGLE:
|
|
variant = VARIANT_ANGLE;
|
|
break;
|
|
case TYPE_FREQUENCY:
|
|
variant = VARIANT_FREQUENCY;
|
|
break;
|
|
case TYPE_TIME:
|
|
variant = VARIANT_TIME;
|
|
break;
|
|
case TYPE_GRADIENT:
|
|
variant = VARIANT_GRADIENT;
|
|
break;
|
|
case TYPE_TIMING_FUNCTION:
|
|
variant = VARIANT_TIMING_FUNCTION;
|
|
break;
|
|
case TYPE_IMAGE_RECT:
|
|
variant = VARIANT_IMAGE_RECT;
|
|
break;
|
|
case TYPE_NUMBER:
|
|
// Include integers under "number"?
|
|
variant = VARIANT_NUMBER | VARIANT_INTEGER;
|
|
break;
|
|
default:
|
|
// Unknown type
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
}
|
|
|
|
*_retval = PropertySupportsVariant(propertyID, variant);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetCSSValuesForProperty(const nsAString& aProperty,
|
|
uint32_t* aLength,
|
|
char16_t*** aValues)
|
|
{
|
|
nsCSSProperty propertyID = nsCSSProps::LookupProperty(aProperty,
|
|
nsCSSProps::eEnabledForAllContent);
|
|
if (propertyID == eCSSProperty_UNKNOWN) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsTArray<nsString> array;
|
|
// We start collecting the values, BUT colors need to go in first, because array
|
|
// needs to stay sorted, and the colors are sorted, so we just append them.
|
|
if (propertyID == eCSSPropertyExtra_variable) {
|
|
// No other values we can report.
|
|
} else if (!nsCSSProps::IsShorthand(propertyID)) {
|
|
// Property is longhand.
|
|
uint32_t propertyParserVariant = nsCSSProps::ParserVariant(propertyID);
|
|
// Get colors first.
|
|
GetColorsForProperty(propertyParserVariant, array);
|
|
if (propertyParserVariant & VARIANT_KEYWORD) {
|
|
GetKeywordsForProperty(propertyID, array);
|
|
}
|
|
GetOtherValuesForProperty(propertyParserVariant, array);
|
|
} else {
|
|
// Property is shorthand.
|
|
CSSPROPS_FOR_SHORTHAND_SUBPROPERTIES(subproperty, propertyID,
|
|
nsCSSProps::eEnabledForAllContent) {
|
|
// Get colors (once) first.
|
|
uint32_t propertyParserVariant = nsCSSProps::ParserVariant(*subproperty);
|
|
if (propertyParserVariant & VARIANT_COLOR) {
|
|
GetColorsForProperty(propertyParserVariant, array);
|
|
break;
|
|
}
|
|
}
|
|
CSSPROPS_FOR_SHORTHAND_SUBPROPERTIES(subproperty, propertyID,
|
|
nsCSSProps::eEnabledForAllContent) {
|
|
uint32_t propertyParserVariant = nsCSSProps::ParserVariant(*subproperty);
|
|
if (propertyParserVariant & VARIANT_KEYWORD) {
|
|
GetKeywordsForProperty(*subproperty, array);
|
|
}
|
|
GetOtherValuesForProperty(propertyParserVariant, array);
|
|
}
|
|
}
|
|
// All CSS properties take initial, inherit and unset.
|
|
InsertNoDuplicates(array, NS_LITERAL_STRING("initial"));
|
|
InsertNoDuplicates(array, NS_LITERAL_STRING("inherit"));
|
|
InsertNoDuplicates(array, NS_LITERAL_STRING("unset"));
|
|
|
|
*aLength = array.Length();
|
|
char16_t** ret =
|
|
static_cast<char16_t**>(NS_Alloc(*aLength * sizeof(char16_t*)));
|
|
for (uint32_t i = 0; i < *aLength; ++i) {
|
|
ret[i] = ToNewUnicode(array[i]);
|
|
}
|
|
*aValues = ret;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::ColorNameToRGB(const nsAString& aColorName, JSContext* aCx,
|
|
JS::MutableHandle<JS::Value> aValue)
|
|
{
|
|
nscolor color;
|
|
if (!NS_ColorNameToRGB(aColorName, &color)) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
InspectorRGBTriple triple;
|
|
triple.mR = NS_GET_R(color);
|
|
triple.mG = NS_GET_G(color);
|
|
triple.mB = NS_GET_B(color);
|
|
|
|
if (!ToJSValue(aCx, triple, aValue)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::RgbToColorName(uint8_t aR, uint8_t aG, uint8_t aB,
|
|
nsAString& aColorName)
|
|
{
|
|
const char* color = NS_RGBToColorName(NS_RGB(aR, aG, aB));
|
|
if (!color) {
|
|
aColorName.Truncate();
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
aColorName.AssignASCII(color);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::ColorToRGBA(const nsAString& aColorString, JSContext* aCx,
|
|
JS::MutableHandle<JS::Value> aValue)
|
|
{
|
|
nscolor color = 0;
|
|
nsCSSParser cssParser;
|
|
nsCSSValue cssValue;
|
|
|
|
bool isColor = cssParser.ParseColorString(aColorString, nullptr, 0,
|
|
cssValue, true);
|
|
|
|
if (!isColor) {
|
|
aValue.setNull();
|
|
return NS_OK;
|
|
}
|
|
|
|
nsRuleNode::ComputeColor(cssValue, nullptr, nullptr, color);
|
|
|
|
InspectorRGBATuple tuple;
|
|
tuple.mR = NS_GET_R(color);
|
|
tuple.mG = NS_GET_G(color);
|
|
tuple.mB = NS_GET_B(color);
|
|
tuple.mA = nsStyleUtil::ColorComponentToFloat(NS_GET_A(color));
|
|
|
|
if (!ToJSValue(aCx, tuple, aValue)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::IsValidCSSColor(const nsAString& aColorString, bool *_retval)
|
|
{
|
|
nsCSSParser cssParser;
|
|
nsCSSValue cssValue;
|
|
*_retval = cssParser.ParseColorString(aColorString, nullptr, 0, cssValue, true);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::CssPropertyIsValid(const nsAString& aPropertyName,
|
|
const nsAString& aPropertyValue,
|
|
bool *_retval)
|
|
{
|
|
nsCSSProperty propertyID =
|
|
nsCSSProps::LookupProperty(aPropertyName, nsCSSProps::eIgnoreEnabledState);
|
|
|
|
if (propertyID == eCSSProperty_UNKNOWN) {
|
|
*_retval = false;
|
|
return NS_OK;
|
|
}
|
|
|
|
if (propertyID == eCSSPropertyExtra_variable) {
|
|
*_retval = true;
|
|
return NS_OK;
|
|
}
|
|
|
|
// Get a parser, parse the property.
|
|
nsCSSParser parser;
|
|
*_retval = parser.IsValueValidForProperty(propertyID, aPropertyValue);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetBindingURLs(nsIDOMElement *aElement, nsIArray **_retval)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aElement);
|
|
|
|
*_retval = nullptr;
|
|
|
|
nsCOMPtr<nsIMutableArray> urls = do_CreateInstance(NS_ARRAY_CONTRACTID);
|
|
if (!urls)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(content);
|
|
|
|
nsXBLBinding *binding = content->GetXBLBinding();
|
|
|
|
while (binding) {
|
|
urls->AppendElement(binding->PrototypeBinding()->BindingURI(), false);
|
|
binding = binding->GetBaseBinding();
|
|
}
|
|
|
|
urls.forget(_retval);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::SetContentState(nsIDOMElement* aElement,
|
|
EventStates::InternalType aState)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aElement);
|
|
|
|
RefPtr<EventStateManager> esm =
|
|
inLayoutUtils::GetEventStateManagerFor(aElement);
|
|
if (esm) {
|
|
nsCOMPtr<nsIContent> content;
|
|
content = do_QueryInterface(aElement);
|
|
|
|
// XXX Invalid cast of bool to nsresult (bug 778108)
|
|
return (nsresult)esm->SetContentState(content, EventStates(aState));
|
|
}
|
|
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetContentState(nsIDOMElement* aElement,
|
|
EventStates::InternalType* aState)
|
|
{
|
|
*aState = 0;
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(content);
|
|
|
|
// NOTE: if this method is removed,
|
|
// please remove GetInternalValue from EventStates
|
|
*aState = content->AsElement()->State().GetInternalValue();
|
|
return NS_OK;
|
|
}
|
|
|
|
/* static */ nsresult
|
|
inDOMUtils::GetRuleNodeForElement(dom::Element* aElement,
|
|
nsIAtom* aPseudo,
|
|
nsStyleContext** aStyleContext,
|
|
nsRuleNode** aRuleNode)
|
|
{
|
|
MOZ_ASSERT(aElement);
|
|
|
|
*aRuleNode = nullptr;
|
|
*aStyleContext = nullptr;
|
|
|
|
nsIDocument* doc = aElement->GetComposedDoc();
|
|
NS_ENSURE_TRUE(doc, NS_ERROR_UNEXPECTED);
|
|
|
|
nsIPresShell *presShell = doc->GetShell();
|
|
NS_ENSURE_TRUE(presShell, NS_ERROR_UNEXPECTED);
|
|
|
|
nsPresContext *presContext = presShell->GetPresContext();
|
|
NS_ENSURE_TRUE(presContext, NS_ERROR_UNEXPECTED);
|
|
|
|
presContext->EnsureSafeToHandOutCSSRules();
|
|
|
|
RefPtr<nsStyleContext> sContext =
|
|
nsComputedDOMStyle::GetStyleContextForElement(aElement, aPseudo, presShell);
|
|
if (sContext) {
|
|
*aRuleNode = sContext->RuleNode();
|
|
sContext.forget(aStyleContext);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::GetUsedFontFaces(nsIDOMRange* aRange,
|
|
nsIDOMFontFaceList** aFontFaceList)
|
|
{
|
|
return static_cast<nsRange*>(aRange)->GetUsedFontFaces(aFontFaceList);
|
|
}
|
|
|
|
static EventStates
|
|
GetStatesForPseudoClass(const nsAString& aStatePseudo)
|
|
{
|
|
// An array of the states that are relevant for various pseudoclasses.
|
|
// XXXbz this duplicates code in nsCSSRuleProcessor
|
|
static const EventStates sPseudoClassStates[] = {
|
|
#define CSS_PSEUDO_CLASS(_name, _value, _flags, _pref) \
|
|
EventStates(),
|
|
#define CSS_STATE_PSEUDO_CLASS(_name, _value, _flags, _pref, _states) \
|
|
_states,
|
|
#include "nsCSSPseudoClassList.h"
|
|
#undef CSS_STATE_PSEUDO_CLASS
|
|
#undef CSS_PSEUDO_CLASS
|
|
|
|
// Add more entries for our fake values to make sure we can't
|
|
// index out of bounds into this array no matter what.
|
|
EventStates(),
|
|
EventStates()
|
|
};
|
|
static_assert(MOZ_ARRAY_LENGTH(sPseudoClassStates) ==
|
|
nsCSSPseudoClasses::ePseudoClass_NotPseudoClass + 1,
|
|
"Length of PseudoClassStates array is incorrect");
|
|
|
|
nsCOMPtr<nsIAtom> atom = do_GetAtom(aStatePseudo);
|
|
|
|
// Ignore :moz-any-link so we don't give the element simultaneous
|
|
// visited and unvisited style state
|
|
if (nsCSSPseudoClasses::GetPseudoType(atom) ==
|
|
nsCSSPseudoClasses::ePseudoClass_mozAnyLink) {
|
|
return EventStates();
|
|
}
|
|
// Our array above is long enough that indexing into it with
|
|
// NotPseudoClass is ok.
|
|
return sPseudoClassStates[nsCSSPseudoClasses::GetPseudoType(atom)];
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::AddPseudoClassLock(nsIDOMElement *aElement,
|
|
const nsAString &aPseudoClass)
|
|
{
|
|
EventStates state = GetStatesForPseudoClass(aPseudoClass);
|
|
if (state.IsEmpty()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<mozilla::dom::Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(element);
|
|
|
|
element->LockStyleStates(state);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::RemovePseudoClassLock(nsIDOMElement *aElement,
|
|
const nsAString &aPseudoClass)
|
|
{
|
|
EventStates state = GetStatesForPseudoClass(aPseudoClass);
|
|
if (state.IsEmpty()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<mozilla::dom::Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(element);
|
|
|
|
element->UnlockStyleStates(state);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::HasPseudoClassLock(nsIDOMElement *aElement,
|
|
const nsAString &aPseudoClass,
|
|
bool *_retval)
|
|
{
|
|
EventStates state = GetStatesForPseudoClass(aPseudoClass);
|
|
if (state.IsEmpty()) {
|
|
*_retval = false;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<mozilla::dom::Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(element);
|
|
|
|
EventStates locks = element->LockedStyleStates();
|
|
|
|
*_retval = locks.HasAllStates(state);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::ClearPseudoClassLocks(nsIDOMElement *aElement)
|
|
{
|
|
nsCOMPtr<mozilla::dom::Element> element = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(element);
|
|
|
|
element->ClearStyleStateLocks();
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::ParseStyleSheet(nsIDOMCSSStyleSheet *aSheet,
|
|
const nsAString& aInput)
|
|
{
|
|
RefPtr<CSSStyleSheet> sheet = do_QueryObject(aSheet);
|
|
NS_ENSURE_ARG_POINTER(sheet);
|
|
|
|
return sheet->ReparseSheet(aInput);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
inDOMUtils::ScrollElementIntoView(nsIDOMElement *aElement)
|
|
{
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(aElement);
|
|
NS_ENSURE_ARG_POINTER(content);
|
|
|
|
nsIPresShell* presShell = content->OwnerDoc()->GetShell();
|
|
if (!presShell) {
|
|
return NS_OK;
|
|
}
|
|
|
|
presShell->ScrollContentIntoView(content,
|
|
nsIPresShell::ScrollAxis(),
|
|
nsIPresShell::ScrollAxis(),
|
|
nsIPresShell::SCROLL_OVERFLOW_HIDDEN);
|
|
|
|
return NS_OK;
|
|
}
|