forked from mirrors/gecko-dev
		
	 2d0a072174
			
		
	
	
		2d0a072174
		
	
	
	
	
		
			
			This will make implementing the new behavior behind a pref really straight-forward, and is generally nicer. Depends on D121858 Differential Revision: https://phabricator.services.mozilla.com/D121705
		
			
				
	
	
		
			222 lines
		
	
	
	
		
			7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			222 lines
		
	
	
	
		
			7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- 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 "AnonymousContent.h"
 | |
| #include "mozilla/PresShell.h"
 | |
| #include "mozilla/dom/Document.h"
 | |
| #include "mozilla/dom/Element.h"
 | |
| #include "mozilla/dom/Event.h"
 | |
| #include "mozilla/dom/AnonymousContentBinding.h"
 | |
| #include "nsComputedDOMStyle.h"
 | |
| #include "nsCycleCollectionParticipant.h"
 | |
| #include "nsIFrame.h"
 | |
| #include "nsStyledElement.h"
 | |
| #include "HTMLCanvasElement.h"
 | |
| 
 | |
| namespace mozilla::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)
 | |
| 
 | |
| AnonymousContent::AnonymousContent(already_AddRefed<Element> aContentNode)
 | |
|     : mContentNode(aContentNode) {
 | |
|   MOZ_ASSERT(mContentNode);
 | |
| }
 | |
| 
 | |
| AnonymousContent::~AnonymousContent() = default;
 | |
| 
 | |
| 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,
 | |
|                                               nsIPrincipal* aSubjectPrincipal,
 | |
|                                               ErrorResult& aRv) {
 | |
|   Element* element = GetElementById(aElementId);
 | |
|   if (!element) {
 | |
|     aRv.Throw(NS_ERROR_NOT_AVAILABLE);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   element->SetAttribute(aName, aValue, aSubjectPrincipal, aRv);
 | |
| }
 | |
| 
 | |
| 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);
 | |
| }
 | |
| 
 | |
| 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();
 | |
| }
 | |
| 
 | |
| 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);
 | |
| }
 | |
| 
 | |
| 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();
 | |
|   }
 | |
| }
 | |
| 
 | |
| Element* AnonymousContent::GetElementById(const nsAString& aElementId) {
 | |
|   // This can be made faster in the future if needed.
 | |
|   RefPtr<nsAtom> elementId = NS_Atomize(aElementId);
 | |
|   for (nsIContent* node = mContentNode; node;
 | |
|        node = node->GetNextNode(mContentNode)) {
 | |
|     if (!node->IsElement()) {
 | |
|       continue;
 | |
|     }
 | |
|     nsAtom* id = node->AsElement()->GetID();
 | |
|     if (id && id == elementId) {
 | |
|       return node->AsElement();
 | |
|     }
 | |
|   }
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| bool AnonymousContent::WrapObject(JSContext* aCx,
 | |
|                                   JS::Handle<JSObject*> aGivenProto,
 | |
|                                   JS::MutableHandle<JSObject*> aReflector) {
 | |
|   return AnonymousContent_Binding::Wrap(aCx, this, aGivenProto, aReflector);
 | |
| }
 | |
| 
 | |
| void AnonymousContent::GetComputedStylePropertyValue(
 | |
|     const nsAString& aElementId, const nsACString& aPropertyName,
 | |
|     nsACString& aResult, ErrorResult& aRv) {
 | |
|   Element* element = GetElementById(aElementId);
 | |
|   if (!element) {
 | |
|     aRv.Throw(NS_ERROR_NOT_AVAILABLE);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!element->OwnerDoc()->GetPresShell()) {
 | |
|     aRv.Throw(NS_ERROR_NOT_AVAILABLE);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   RefPtr<nsComputedDOMStyle> cs = new nsComputedDOMStyle(
 | |
|       element, PseudoStyleType::NotPseudo, element->OwnerDoc(),
 | |
|       nsComputedDOMStyle::StyleType::All);
 | |
|   aRv = cs->GetPropertyValue(aPropertyName, aResult);
 | |
| }
 | |
| 
 | |
| 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();
 | |
| }
 | |
| 
 | |
| void AnonymousContent::SetStyle(const nsACString& aProperty,
 | |
|                                 const nsACString& 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, ""_ns, IgnoreErrors());
 | |
| }
 | |
| 
 | |
| }  // namespace mozilla::dom
 |