2015-05-03 22:32:37 +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: */
|
2014-10-28 13:15:21 +03:00
|
|
|
/* 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 "AnonymousContent.h"
|
2019-03-29 18:12:47 +03:00
|
|
|
#include "mozilla/PresShell.h"
|
|
|
|
#include "mozilla/dom/Document.h"
|
2014-10-28 13:15:21 +03:00
|
|
|
#include "mozilla/dom/Element.h"
|
|
|
|
#include "mozilla/dom/AnonymousContentBinding.h"
|
2017-03-16 17:46:43 +03:00
|
|
|
#include "nsComputedDOMStyle.h"
|
2014-10-28 13:15:21 +03:00
|
|
|
#include "nsCycleCollectionParticipant.h"
|
2016-09-15 21:58:15 +03:00
|
|
|
#include "nsIFrame.h"
|
2014-10-28 13:15:21 +03:00
|
|
|
#include "nsStyledElement.h"
|
2015-11-15 16:48:32 +03:00
|
|
|
#include "HTMLCanvasElement.h"
|
2014-10-28 13:15:21 +03:00
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace dom {
|
|
|
|
|
|
|
|
// Ref counting and cycle collection
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_ROOT_NATIVE(AnonymousContent, AddRef)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNROOT_NATIVE(AnonymousContent, Release)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION(AnonymousContent, mContentNode)
|
|
|
|
|
2018-08-13 12:56:48 +03:00
|
|
|
AnonymousContent::AnonymousContent(already_AddRefed<Element> aContentNode)
|
|
|
|
: mContentNode(aContentNode) {
|
|
|
|
MOZ_ASSERT(mContentNode);
|
2014-10-28 13:15:21 +03:00
|
|
|
}
|
|
|
|
|
2018-08-13 12:56:48 +03:00
|
|
|
AnonymousContent::~AnonymousContent() = default;
|
2014-10-28 13:15:25 +03:00
|
|
|
|
2014-10-28 13:15:21 +03:00
|
|
|
void AnonymousContent::SetTextContentForElement(const nsAString& aElementId,
|
|
|
|
const nsAString& aText,
|
|
|
|
ErrorResult& aRv) {
|
|
|
|
Element* element = GetElementById(aElementId);
|
|
|
|
if (!element) {
|
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
element->SetTextContent(aText, aRv);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnonymousContent::GetTextContentForElement(const nsAString& aElementId,
|
|
|
|
DOMString& aText,
|
|
|
|
ErrorResult& aRv) {
|
|
|
|
Element* element = GetElementById(aElementId);
|
|
|
|
if (!element) {
|
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
element->GetTextContent(aText, aRv);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnonymousContent::SetAttributeForElement(const nsAString& aElementId,
|
|
|
|
const nsAString& aName,
|
|
|
|
const nsAString& aValue,
|
2017-10-10 00:33:38 +03:00
|
|
|
nsIPrincipal* aSubjectPrincipal,
|
2014-10-28 13:15:21 +03:00
|
|
|
ErrorResult& aRv) {
|
|
|
|
Element* element = GetElementById(aElementId);
|
|
|
|
if (!element) {
|
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-10-10 00:33:38 +03:00
|
|
|
element->SetAttribute(aName, aValue, aSubjectPrincipal, aRv);
|
2014-10-28 13:15:21 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void AnonymousContent::GetAttributeForElement(const nsAString& aElementId,
|
|
|
|
const nsAString& aName,
|
|
|
|
DOMString& aValue,
|
|
|
|
ErrorResult& aRv) {
|
|
|
|
Element* element = GetElementById(aElementId);
|
|
|
|
if (!element) {
|
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
element->GetAttribute(aName, aValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnonymousContent::RemoveAttributeForElement(const nsAString& aElementId,
|
|
|
|
const nsAString& aName,
|
|
|
|
ErrorResult& aRv) {
|
|
|
|
Element* element = GetElementById(aElementId);
|
|
|
|
if (!element) {
|
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
element->RemoveAttribute(aName, aRv);
|
|
|
|
}
|
|
|
|
|
2015-11-15 16:48:32 +03:00
|
|
|
already_AddRefed<nsISupports> AnonymousContent::GetCanvasContext(
|
|
|
|
const nsAString& aElementId, const nsAString& aContextId,
|
|
|
|
ErrorResult& aRv) {
|
|
|
|
Element* element = GetElementById(aElementId);
|
|
|
|
|
|
|
|
if (!element) {
|
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!element->IsHTMLElement(nsGkAtoms::canvas)) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsISupports> context;
|
|
|
|
|
|
|
|
HTMLCanvasElement* canvas = static_cast<HTMLCanvasElement*>(element);
|
|
|
|
canvas->GetContext(aContextId, getter_AddRefs(context));
|
|
|
|
|
|
|
|
return context.forget();
|
|
|
|
}
|
|
|
|
|
2016-09-08 23:38:53 +03:00
|
|
|
already_AddRefed<Animation> AnonymousContent::SetAnimationForElement(
|
|
|
|
JSContext* aContext, const nsAString& aElementId,
|
|
|
|
JS::Handle<JSObject*> aKeyframes,
|
|
|
|
const UnrestrictedDoubleOrKeyframeAnimationOptions& aOptions,
|
|
|
|
ErrorResult& aRv) {
|
|
|
|
Element* element = GetElementById(aElementId);
|
|
|
|
|
|
|
|
if (!element) {
|
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return element->Animate(aContext, aKeyframes, aOptions, aRv);
|
|
|
|
}
|
|
|
|
|
2016-09-15 21:58:15 +03:00
|
|
|
void AnonymousContent::SetCutoutRectsForElement(
|
|
|
|
const nsAString& aElementId, const Sequence<OwningNonNull<DOMRect>>& aRects,
|
|
|
|
ErrorResult& aRv) {
|
|
|
|
Element* element = GetElementById(aElementId);
|
|
|
|
|
|
|
|
if (!element) {
|
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsRegion cutOutRegion;
|
|
|
|
for (const auto& r : aRects) {
|
|
|
|
CSSRect rect(r->X(), r->Y(), r->Width(), r->Height());
|
|
|
|
cutOutRegion.OrWith(CSSRect::ToAppUnits(rect));
|
|
|
|
}
|
|
|
|
|
|
|
|
element->SetProperty(nsGkAtoms::cutoutregion, new nsRegion(cutOutRegion),
|
|
|
|
nsINode::DeleteProperty<nsRegion>);
|
|
|
|
|
|
|
|
nsIFrame* frame = element->GetPrimaryFrame();
|
|
|
|
if (frame) {
|
|
|
|
frame->SchedulePaint();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-28 13:15:21 +03:00
|
|
|
Element* AnonymousContent::GetElementById(const nsAString& aElementId) {
|
|
|
|
// This can be made faster in the future if needed.
|
2017-10-03 01:05:19 +03:00
|
|
|
RefPtr<nsAtom> elementId = NS_Atomize(aElementId);
|
2016-09-14 01:19:07 +03:00
|
|
|
for (nsIContent* node = mContentNode; node;
|
|
|
|
node = node->GetNextNode(mContentNode)) {
|
|
|
|
if (!node->IsElement()) {
|
2014-10-28 13:15:21 +03:00
|
|
|
continue;
|
|
|
|
}
|
2017-10-03 01:05:19 +03:00
|
|
|
nsAtom* id = node->AsElement()->GetID();
|
2014-10-28 13:15:21 +03:00
|
|
|
if (id && id == elementId) {
|
2016-09-14 01:19:07 +03:00
|
|
|
return node->AsElement();
|
2014-10-28 13:15:21 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2015-01-09 00:56:42 +03:00
|
|
|
bool AnonymousContent::WrapObject(JSContext* aCx,
|
Bug 1117172 part 2. Change the non-wrappercached WrapObject methods to allow passing in aGivenProto. r=peterv
The only manual changes here are to BindingUtils.h, Codegen.py, and
StructuredClone.cpp. The rest of this diff was generated by running the following commands:
find . -name "*.h" -o -name "*.cpp" | xargs perl -pi -e 'BEGIN { $/ = undef } s/WrapObject\((JSContext *\* *(?:aCx|cx)),(\s*)(JS::MutableHandle<JSObject\*> aReflector)/WrapObject(\1,\2JS::Handle<JSObject*> aGivenProto,\2\3/g'
find . -name "*.h" -o -name "*.cpp" | xargs perl -pi -e 'BEGIN { $/ = undef } s/(Binding(?:_workers)?::Wrap\((?:aCx|cx)), this, aReflector/\1, this, aGivenProto, aReflector/'
2015-03-19 17:13:32 +03:00
|
|
|
JS::Handle<JSObject*> aGivenProto,
|
2015-01-09 00:56:42 +03:00
|
|
|
JS::MutableHandle<JSObject*> aReflector) {
|
2018-06-26 00:20:54 +03:00
|
|
|
return AnonymousContent_Binding::Wrap(aCx, this, aGivenProto, aReflector);
|
2014-10-28 13:15:21 +03:00
|
|
|
}
|
|
|
|
|
2017-03-16 17:46:43 +03:00
|
|
|
void AnonymousContent::GetComputedStylePropertyValue(
|
|
|
|
const nsAString& aElementId, const nsAString& aPropertyName,
|
|
|
|
DOMString& aResult, ErrorResult& aRv) {
|
|
|
|
Element* element = GetElementById(aElementId);
|
|
|
|
if (!element) {
|
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-03-29 18:12:47 +03:00
|
|
|
if (!element->OwnerDoc()->GetPresShell()) {
|
2017-03-16 17:46:43 +03:00
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
RefPtr<nsComputedDOMStyle> cs =
|
2018-06-08 12:15:34 +03:00
|
|
|
new nsComputedDOMStyle(element, NS_LITERAL_STRING(""),
|
2017-05-19 07:57:35 +03:00
|
|
|
element->OwnerDoc(), nsComputedDOMStyle::eAll);
|
2017-03-16 17:46:43 +03:00
|
|
|
aRv = cs->GetPropertyValue(aPropertyName, aResult);
|
|
|
|
}
|
|
|
|
|
2019-03-26 22:40:29 +03:00
|
|
|
void AnonymousContent::GetTargetIdForEvent(Event& aEvent, DOMString& aResult) {
|
|
|
|
nsCOMPtr<Element> el = do_QueryInterface(aEvent.GetOriginalTarget());
|
|
|
|
if (el && el->IsInNativeAnonymousSubtree() && mContentNode->Contains(el)) {
|
|
|
|
aResult.SetKnownLiveAtom(el->GetID(), DOMString::eTreatNullAsNull);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
aResult.SetNull();
|
|
|
|
}
|
|
|
|
|
2019-08-16 10:38:39 +03:00
|
|
|
void AnonymousContent::SetStyle(const nsAString& aProperty,
|
|
|
|
const nsAString& aValue, ErrorResult& aRv) {
|
|
|
|
if (!mContentNode->IsHTMLElement()) {
|
|
|
|
aRv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsGenericHTMLElement* element = nsGenericHTMLElement::FromNode(mContentNode);
|
|
|
|
nsCOMPtr<nsICSSDeclaration> declaration = element->Style();
|
|
|
|
declaration->SetProperty(aProperty, aValue, EmptyString());
|
|
|
|
}
|
|
|
|
|
2015-07-13 18:25:42 +03:00
|
|
|
} // namespace dom
|
|
|
|
} // namespace mozilla
|