diff options
Diffstat (limited to '')
-rw-r--r-- | dom/base/ResizeObserver.h | 321 |
1 files changed, 321 insertions, 0 deletions
diff --git a/dom/base/ResizeObserver.h b/dom/base/ResizeObserver.h new file mode 100644 index 0000000000..6f1fc5b6cd --- /dev/null +++ b/dom/base/ResizeObserver.h @@ -0,0 +1,321 @@ +/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim:set ts=2 sw=2 sts=2 et cindent: */ +/* 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/. */ + +#ifndef mozilla_dom_ResizeObserver_h +#define mozilla_dom_ResizeObserver_h + +#include "gfxPoint.h" +#include "js/TypeDecls.h" +#include "mozilla/AppUnits.h" +#include "mozilla/Attributes.h" +#include "mozilla/LinkedList.h" +#include "mozilla/WritingModes.h" +#include "mozilla/dom/DOMRect.h" +#include "mozilla/dom/BindingDeclarations.h" +#include "mozilla/dom/ResizeObserverBinding.h" +#include "nsCoord.h" +#include "nsCycleCollectionParticipant.h" +#include "nsRefPtrHashtable.h" +#include "nsTArray.h" +#include "nsWrapperCache.h" + +// XXX Avoid including this here by moving function bodies to the cpp file +#include "nsPIDOMWindow.h" + +namespace mozilla { +class ErrorResult; + +namespace dom { + +class Element; + +// The logical size in pixels. +// Note: if LogicalPixelSize have usages other than ResizeObserver in the +// future, it might be better to change LogicalSize into a template class, and +// use it to implement LogicalPixelSize. +class LogicalPixelSize { + public: + LogicalPixelSize() = default; + LogicalPixelSize(WritingMode aWM, const gfx::Size& aSize) { + mSize = aSize; + if (aWM.IsVertical()) { + std::swap(mSize.width, mSize.height); + } + } + + gfx::Size PhysicalSize(WritingMode aWM) const { + if (!aWM.IsVertical()) { + return mSize; + } + gfx::Size result(mSize); + std::swap(result.width, result.height); + return result; + } + + bool operator==(const LogicalPixelSize& aOther) const { + return mSize == aOther.mSize; + } + bool operator!=(const LogicalPixelSize& aOther) const { + return !(*this == aOther); + } + + float ISize() const { return mSize.width; } + float BSize() const { return mSize.height; } + float& ISize() { return mSize.width; } + float& BSize() { return mSize.height; } + + private: + // |mSize.width| represents inline-size and |mSize.height| represents + // block-size. + gfx::Size mSize; +}; + +// For the internal implementation in ResizeObserver. Normally, this is owned by +// ResizeObserver. +class ResizeObservation final : public LinkedListElement<ResizeObservation> { + public: + NS_INLINE_DECL_CYCLE_COLLECTING_NATIVE_REFCOUNTING(ResizeObservation) + NS_DECL_CYCLE_COLLECTION_NATIVE_CLASS(ResizeObservation) + + ResizeObservation(Element&, ResizeObserver&, ResizeObserverBoxOptions); + + Element* Target() const { return mTarget; } + + ResizeObserverBoxOptions BoxOptions() const { return mObservedBox; } + + /** + * Returns whether the observed target element size differs from the saved + * mLastReportedSize. + */ + bool IsActive() const; + + /** + * Update current mLastReportedSize to aSize. + */ + void UpdateLastReportedSize(const nsTArray<LogicalPixelSize>& aSize); + + enum class RemoveFromObserver : bool { No, Yes }; + void Unlink(RemoveFromObserver); + + protected: + ~ResizeObservation() { Unlink(RemoveFromObserver::No); }; + + nsCOMPtr<Element> mTarget; + + // Weak, observer always outlives us. + ResizeObserver* mObserver; + + const ResizeObserverBoxOptions mObservedBox; + + // The latest recorded of observed target. + // This will be CSS pixels for border-box/content-box, or device pixels for + // device-pixel-content-box. + AutoTArray<LogicalPixelSize, 1> mLastReportedSize; +}; + +/** + * ResizeObserver interfaces and algorithms are based on + * https://drafts.csswg.org/resize-observer/#api + */ +class ResizeObserver final : public nsISupports, public nsWrapperCache { + using NativeCallback = void (*)( + const Sequence<OwningNonNull<ResizeObserverEntry>>&, ResizeObserver&); + ResizeObserver(Document& aDocument, NativeCallback aCallback); + + public: + NS_DECL_CYCLE_COLLECTING_ISUPPORTS + NS_DECL_CYCLE_COLLECTION_WRAPPERCACHE_CLASS(ResizeObserver) + + ResizeObserver(nsCOMPtr<nsPIDOMWindowInner>&& aOwner, Document* aDocument, + ResizeObserverCallback& aCb) + : mOwner(std::move(aOwner)), + mDocument(aDocument), + mCallback(RefPtr<ResizeObserverCallback>(&aCb)) { + MOZ_ASSERT(mOwner, "Need a non-null owner window"); + MOZ_ASSERT(mDocument, "Need a non-null doc"); + MOZ_ASSERT(mDocument == mOwner->GetExtantDoc()); + } + + nsISupports* GetParentObject() const { return mOwner; } + + JSObject* WrapObject(JSContext* aCx, + JS::Handle<JSObject*> aGivenProto) override { + return ResizeObserver_Binding::Wrap(aCx, this, aGivenProto); + } + + static already_AddRefed<ResizeObserver> Constructor( + const GlobalObject& aGlobal, ResizeObserverCallback& aCb, + ErrorResult& aRv); + + void Observe(Element&, const ResizeObserverOptions&); + void Unobserve(Element&); + + void Disconnect(); + + /** + * Gather all observations which have an observed target with size changed + * since last BroadcastActiveObservations() in this ResizeObserver. + * An observation will be skipped if the depth of its observed target is less + * or equal than aDepth. All gathered observations will be added to + * mActiveTargets. + */ + void GatherActiveObservations(uint32_t aDepth); + + /** + * Returns whether this ResizeObserver has any active observations + * since last GatherActiveObservations(). + */ + bool HasActiveObservations() const { return !mActiveTargets.IsEmpty(); } + + /** + * Returns whether this ResizeObserver has any skipped observations + * since last GatherActiveObservations(). + */ + bool HasSkippedObservations() const { return mHasSkippedTargets; } + + /** + * Returns whether this is an internal ResizeObserver with a native callback. + */ + bool HasNativeCallback() const { return mCallback.is<NativeCallback>(); } + + /** + * Invoke the callback function in JavaScript for all active observations + * and pass the sequence of ResizeObserverEntry so JavaScript can access them. + * The active observations' mLastReportedSize fields will be updated, and + * mActiveTargets will be cleared. It also returns the shallowest depth of + * elements from active observations or numeric_limits<uint32_t>::max() if + * there are not any active observations. + */ + MOZ_CAN_RUN_SCRIPT uint32_t BroadcastActiveObservations(); + + static already_AddRefed<ResizeObserver> CreateLastRememberedSizeObserver( + Document&); + + protected: + ~ResizeObserver() { Disconnect(); } + + nsCOMPtr<nsPIDOMWindowInner> mOwner; + // The window's document at the time of ResizeObserver creation. + RefPtr<Document> mDocument; + Variant<RefPtr<ResizeObserverCallback>, NativeCallback> mCallback; + nsTArray<RefPtr<ResizeObservation>> mActiveTargets; + // The spec uses a list to store the skipped targets. However, it seems what + // we want is to check if there are any skipped targets (i.e. existence). + // Therefore, we use a boolean value to represent the existence of skipped + // targets. + bool mHasSkippedTargets = false; + + // Combination of HashTable and LinkedList so we can iterate through + // the elements of HashTable in order of insertion time, so we can deliver + // observations in the correct order + // FIXME: it will be nice if we have our own data structure for this in the + // future, and mObservationMap should be considered the "owning" storage for + // the observations, so it'd be better to drop mObservationList later. + nsRefPtrHashtable<nsPtrHashKey<Element>, ResizeObservation> mObservationMap; + LinkedList<ResizeObservation> mObservationList; +}; + +/** + * ResizeObserverEntry is the entry that contains the information for observed + * elements. This object is the one that's visible to JavaScript in callback + * function that is fired by ResizeObserver. + */ +class ResizeObserverEntry final : public nsISupports, public nsWrapperCache { + public: + NS_DECL_CYCLE_COLLECTING_ISUPPORTS + NS_DECL_CYCLE_COLLECTION_WRAPPERCACHE_CLASS(ResizeObserverEntry) + + ResizeObserverEntry( + nsISupports* aOwner, Element& aTarget, + const nsTArray<LogicalPixelSize>& aBorderBoxSize, + const nsTArray<LogicalPixelSize>& aContentBoxSize, + const nsTArray<LogicalPixelSize>& aDevicePixelContentBoxSize) + : mOwner(aOwner), mTarget(&aTarget) { + MOZ_ASSERT(mOwner, "Need a non-null owner"); + MOZ_ASSERT(mTarget, "Need a non-null target element"); + + SetBorderBoxSize(aBorderBoxSize); + SetContentRectAndSize(aContentBoxSize); + SetDevicePixelContentSize(aDevicePixelContentBoxSize); + } + + nsISupports* GetParentObject() const { return mOwner; } + + JSObject* WrapObject(JSContext* aCx, + JS::Handle<JSObject*> aGivenProto) override { + return ResizeObserverEntry_Binding::Wrap(aCx, this, aGivenProto); + } + + Element* Target() const { return mTarget; } + + /** + * Returns the DOMRectReadOnly of target's content rect so it can be + * accessed from JavaScript in callback function of ResizeObserver. + */ + DOMRectReadOnly* ContentRect() const { return mContentRect; } + + /** + * Returns target's logical border-box size, content-box size, and + * device-pixel-content-box as an array of ResizeObserverSize. + */ + void GetBorderBoxSize(nsTArray<RefPtr<ResizeObserverSize>>& aRetVal) const; + void GetContentBoxSize(nsTArray<RefPtr<ResizeObserverSize>>& aRetVal) const; + void GetDevicePixelContentBoxSize( + nsTArray<RefPtr<ResizeObserverSize>>& aRetVal) const; + + private: + ~ResizeObserverEntry() = default; + + // Set borderBoxSize. + void SetBorderBoxSize(const nsTArray<LogicalPixelSize>& aSize); + // Set contentRect and contentBoxSize. + void SetContentRectAndSize(const nsTArray<LogicalPixelSize>& aSize); + // Set devicePixelContentBoxSize. + void SetDevicePixelContentSize(const nsTArray<LogicalPixelSize>& aSize); + + nsCOMPtr<nsISupports> mOwner; + nsCOMPtr<Element> mTarget; + + RefPtr<DOMRectReadOnly> mContentRect; + AutoTArray<RefPtr<ResizeObserverSize>, 1> mBorderBoxSize; + AutoTArray<RefPtr<ResizeObserverSize>, 1> mContentBoxSize; + AutoTArray<RefPtr<ResizeObserverSize>, 1> mDevicePixelContentBoxSize; +}; + +class ResizeObserverSize final : public nsISupports, public nsWrapperCache { + public: + NS_DECL_CYCLE_COLLECTING_ISUPPORTS + NS_DECL_CYCLE_COLLECTION_WRAPPERCACHE_CLASS(ResizeObserverSize) + + ResizeObserverSize(nsISupports* aOwner, const LogicalPixelSize& aSize) + : mOwner(aOwner), mSize(aSize) { + MOZ_ASSERT(mOwner, "Need a non-null owner"); + } + + nsISupports* GetParentObject() const { return mOwner; } + + JSObject* WrapObject(JSContext* aCx, + JS::Handle<JSObject*> aGivenProto) override { + return ResizeObserverSize_Binding::Wrap(aCx, this, aGivenProto); + } + + float InlineSize() const { return mSize.ISize(); } + float BlockSize() const { return mSize.BSize(); } + + protected: + ~ResizeObserverSize() = default; + + nsCOMPtr<nsISupports> mOwner; + // The logical size value: + // 1. content-box/border-box: in CSS pixels. + // 2. device-pixel-content-box: in device pixels. + const LogicalPixelSize mSize; +}; + +} // namespace dom +} // namespace mozilla + +#endif // mozilla_dom_ResizeObserver_h |