Source code

Revision control

Copy as Markdown

Other Tools

/* -*- 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/dom/CSSKeyframesRule.h"
#include "mozilla/dom/CSSKeyframesRuleBinding.h"
#include "mozilla/dom/CSSRuleList.h"
#include "mozilla/ServoBindings.h"
#include "nsCOMArray.h"
#include <limits>
namespace mozilla::dom {
// -------------------------------------------
// CSSKeyframeList
//
class CSSKeyframeList : public dom::CSSRuleList {
public:
CSSKeyframeList(already_AddRefed<StyleLockedKeyframesRule> aRawRule,
StyleSheet* aSheet, CSSKeyframesRule* aParentRule)
: mStyleSheet(aSheet), mParentRule(aParentRule), mRawRule(aRawRule) {
mRules.SetCount(Servo_KeyframesRule_GetCount(mRawRule));
}
NS_DECL_ISUPPORTS_INHERITED
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(CSSKeyframeList, dom::CSSRuleList)
void SetRawAfterClone(RefPtr<StyleLockedKeyframesRule> aRaw) {
mRawRule = std::move(aRaw);
uint32_t index = 0;
for (css::Rule* rule : mRules) {
if (rule) {
uint32_t line = 0, column = 0;
RefPtr<StyleLockedKeyframe> keyframe =
Servo_KeyframesRule_GetKeyframeAt(mRawRule, index, &line, &column)
.Consume();
static_cast<CSSKeyframeRule*>(rule)->SetRawAfterClone(
std::move(keyframe));
}
index++;
}
}
void DropSheetReference() {
if (!mStyleSheet) {
return;
}
mStyleSheet = nullptr;
for (css::Rule* rule : mRules) {
if (rule) {
rule->DropSheetReference();
}
}
}
StyleSheet* GetParentObject() final { return mStyleSheet; }
CSSKeyframeRule* GetRule(uint32_t aIndex) {
if (!mRules[aIndex]) {
uint32_t line = 0, column = 0;
RefPtr<StyleLockedKeyframe> rule =
Servo_KeyframesRule_GetKeyframeAt(mRawRule, aIndex, &line, &column)
.Consume();
CSSKeyframeRule* ruleObj = new CSSKeyframeRule(rule.forget(), mStyleSheet,
mParentRule, line, column);
mRules.ReplaceObjectAt(ruleObj, aIndex);
}
return static_cast<CSSKeyframeRule*>(mRules[aIndex]);
}
CSSKeyframeRule* IndexedGetter(uint32_t aIndex, bool& aFound) final {
if (aIndex >= mRules.Length()) {
aFound = false;
return nullptr;
}
aFound = true;
return GetRule(aIndex);
}
void AppendRule() {
MOZ_ASSERT(!mParentRule->IsReadOnly());
mRules.AppendObject(nullptr);
}
void RemoveRule(uint32_t aIndex) {
MOZ_ASSERT(!mParentRule->IsReadOnly());
if (aIndex >= mRules.Length()) {
return;
}
if (css::Rule* child = mRules[aIndex]) {
child->DropReferences();
}
mRules.RemoveObjectAt(aIndex);
}
uint32_t Length() final { return mRules.Length(); }
void DropReferences() {
if (!mStyleSheet && !mParentRule) {
return;
}
mStyleSheet = nullptr;
mParentRule = nullptr;
for (css::Rule* rule : mRules) {
if (rule) {
rule->DropParentRuleReference();
rule->DropSheetReference();
}
}
}
size_t SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const {
size_t n = aMallocSizeOf(this);
for (const css::Rule* rule : mRules) {
n += rule ? rule->SizeOfIncludingThis(aMallocSizeOf) : 0;
}
return n;
}
private:
virtual ~CSSKeyframeList() {
MOZ_ASSERT(!mParentRule, "Backpointer should have been cleared");
MOZ_ASSERT(!mStyleSheet, "Backpointer should have been cleared");
DropAllRules();
}
void DropAllRules() {
DropReferences();
mRules.Clear();
mRawRule = nullptr;
}
// may be nullptr when the style sheet drops the reference to us.
StyleSheet* mStyleSheet = nullptr;
CSSKeyframesRule* mParentRule = nullptr;
RefPtr<StyleLockedKeyframesRule> mRawRule;
nsCOMArray<css::Rule> mRules;
};
// QueryInterface implementation for CSSKeyframeList
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(CSSKeyframeList)
NS_INTERFACE_MAP_END_INHERITING(dom::CSSRuleList)
NS_IMPL_ADDREF_INHERITED(CSSKeyframeList, dom::CSSRuleList)
NS_IMPL_RELEASE_INHERITED(CSSKeyframeList, dom::CSSRuleList)
NS_IMPL_CYCLE_COLLECTION_CLASS(CSSKeyframeList)
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(CSSKeyframeList)
tmp->DropAllRules();
NS_IMPL_CYCLE_COLLECTION_UNLINK_END_INHERITED(dom::CSSRuleList)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(CSSKeyframeList,
dom::CSSRuleList)
for (css::Rule* rule : tmp->mRules) {
if (rule) {
NS_CYCLE_COLLECTION_NOTE_EDGE_NAME(cb, "mRules[i]");
cb.NoteXPCOMChild(rule);
}
}
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
// -------------------------------------------
// CSSKeyframesRule
//
CSSKeyframesRule::CSSKeyframesRule(RefPtr<StyleLockedKeyframesRule> aRawRule,
StyleSheet* aSheet, css::Rule* aParentRule,
uint32_t aLine, uint32_t aColumn)
: css::Rule(aSheet, aParentRule, aLine, aColumn),
mRawRule(std::move(aRawRule)) {}
CSSKeyframesRule::~CSSKeyframesRule() {
if (mKeyframeList) {
mKeyframeList->DropReferences();
}
}
NS_IMPL_ADDREF_INHERITED(CSSKeyframesRule, css::Rule)
NS_IMPL_RELEASE_INHERITED(CSSKeyframesRule, css::Rule)
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(CSSKeyframesRule)
NS_INTERFACE_MAP_END_INHERITING(css::Rule)
NS_IMPL_CYCLE_COLLECTION_CLASS(CSSKeyframesRule)
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(CSSKeyframesRule, css::Rule)
if (tmp->mKeyframeList) {
tmp->mKeyframeList->DropReferences();
tmp->mKeyframeList = nullptr;
}
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(CSSKeyframesRule, Rule)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mKeyframeList)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
/* virtual */
bool CSSKeyframesRule::IsCCLeaf() const {
// If we don't have rule list constructed, we are a leaf.
return Rule::IsCCLeaf() && !mKeyframeList;
}
StyleCssRuleType CSSKeyframesRule::Type() const {
return StyleCssRuleType::Keyframes;
}
void CSSKeyframesRule::SetRawAfterClone(RefPtr<StyleLockedKeyframesRule> aRaw) {
mRawRule = std::move(aRaw);
if (mKeyframeList) {
mKeyframeList->SetRawAfterClone(mRawRule);
}
}
#ifdef DEBUG
/* virtual */
void CSSKeyframesRule::List(FILE* out, int32_t aIndent) const {
nsAutoCString str;
for (int32_t i = 0; i < aIndent; i++) {
str.AppendLiteral(" ");
}
Servo_KeyframesRule_Debug(mRawRule, &str);
fprintf_stderr(out, "%s\n", str.get());
}
#endif
/* virtual */
void CSSKeyframesRule::DropSheetReference() {
if (mKeyframeList) {
mKeyframeList->DropSheetReference();
}
css::Rule::DropSheetReference();
}
static const uint32_t kRuleNotFound = std::numeric_limits<uint32_t>::max();
uint32_t CSSKeyframesRule::FindRuleIndexForKey(const nsAString& aKey) {
NS_ConvertUTF16toUTF8 key(aKey);
return Servo_KeyframesRule_FindRule(mRawRule, &key);
}
template <typename Func>
nsresult CSSKeyframesRule::UpdateRule(Func aCallback) {
if (IsReadOnly()) {
return NS_OK;
}
StyleSheet* sheet = GetStyleSheet();
if (sheet) {
sheet->WillDirty();
}
aCallback();
if (sheet) {
sheet->RuleChanged(this, StyleRuleChangeKind::Generic);
}
return NS_OK;
}
void CSSKeyframesRule::GetName(nsAString& aName) const {
nsAtom* name = Servo_KeyframesRule_GetName(mRawRule);
aName = nsDependentAtomString(name);
}
void CSSKeyframesRule::SetName(const nsAString& aName) {
RefPtr<nsAtom> name = NS_Atomize(aName);
nsAtom* oldName = Servo_KeyframesRule_GetName(mRawRule);
if (name == oldName) {
return;
}
UpdateRule([this, &name]() {
Servo_KeyframesRule_SetName(mRawRule, name.forget().take());
});
}
void CSSKeyframesRule::AppendRule(const nsAString& aRule) {
StyleSheet* sheet = GetStyleSheet();
if (!sheet) {
// We cannot parse the rule if we don't have a stylesheet.
return;
}
NS_ConvertUTF16toUTF8 rule(aRule);
UpdateRule([this, sheet, &rule]() {
bool parsedOk =
Servo_KeyframesRule_AppendRule(mRawRule, sheet->RawContents(), &rule);
if (parsedOk && mKeyframeList) {
mKeyframeList->AppendRule();
}
});
}
void CSSKeyframesRule::DeleteRule(const nsAString& aKey) {
auto index = FindRuleIndexForKey(aKey);
if (index == kRuleNotFound) {
return;
}
UpdateRule([this, index]() {
Servo_KeyframesRule_DeleteRule(mRawRule, index);
if (mKeyframeList) {
mKeyframeList->RemoveRule(index);
}
});
}
/* virtual */
void CSSKeyframesRule::GetCssText(nsACString& aCssText) const {
Servo_KeyframesRule_GetCssText(mRawRule, &aCssText);
}
/* virtual */ dom::CSSRuleList* CSSKeyframesRule::CssRules() {
return EnsureRules();
}
/* virtual */ dom::CSSKeyframeRule* CSSKeyframesRule::IndexedGetter(
uint32_t aIndex, bool& aFound) {
return EnsureRules()->IndexedGetter(aIndex, aFound);
}
/* virtual */ uint32_t CSSKeyframesRule::Length() {
return EnsureRules()->Length();
}
/* virtual */ dom::CSSKeyframeRule* CSSKeyframesRule::FindRule(
const nsAString& aKey) {
auto index = FindRuleIndexForKey(aKey);
if (index != kRuleNotFound) {
return EnsureRules()->GetRule(index);
}
return nullptr;
}
/* virtual */
size_t CSSKeyframesRule::SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const {
size_t n = aMallocSizeOf(this);
if (mKeyframeList) {
n += mKeyframeList->SizeOfIncludingThis(aMallocSizeOf);
}
return n;
}
/* virtual */
JSObject* CSSKeyframesRule::WrapObject(JSContext* aCx,
JS::Handle<JSObject*> aGivenProto) {
return CSSKeyframesRule_Binding::Wrap(aCx, this, aGivenProto);
}
dom::CSSKeyframeList* CSSKeyframesRule::EnsureRules() {
if (!mKeyframeList) {
mKeyframeList = new CSSKeyframeList(do_AddRef(mRawRule), mSheet, this);
}
return mKeyframeList;
}
} // namespace mozilla::dom