summaryrefslogtreecommitdiffstats
path: root/layout/base/nsGenConList.h
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--layout/base/nsGenConList.h143
1 files changed, 143 insertions, 0 deletions
diff --git a/layout/base/nsGenConList.h b/layout/base/nsGenConList.h
new file mode 100644
index 0000000000..37becdf287
--- /dev/null
+++ b/layout/base/nsGenConList.h
@@ -0,0 +1,143 @@
+/* -*- 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/. */
+
+/* base class for nsCounterList and nsQuoteList */
+
+#ifndef nsGenConList_h___
+#define nsGenConList_h___
+
+#include "mozilla/LinkedList.h"
+#include "nsStyleStruct.h"
+#include "nsCSSPseudoElements.h"
+#include "nsTextNode.h"
+
+class nsGenConList;
+class nsIFrame;
+
+struct nsGenConNode : public mozilla::LinkedListElement<nsGenConNode> {
+ using StyleContentType = mozilla::StyleContentItem::Tag;
+
+ // The wrapper frame for all of the pseudo-element's content. This
+ // frame generally has useful style data and has the
+ // NS_FRAME_GENERATED_CONTENT bit set (so we use it to track removal),
+ // but does not necessarily for |nsCounterChangeNode|s.
+ nsIFrame* mPseudoFrame;
+
+ // Index within the list of things specified by the 'content' property,
+ // which is needed to do 'content: open-quote open-quote' correctly,
+ // and needed for similar cases for counters.
+ const int32_t mContentIndex;
+
+ // null for:
+ // * content: no-open-quote / content: no-close-quote
+ // * counter nodes for increments and resets
+ // * counter nodes for bullets (mPseudoFrame->IsBulletFrame()).
+ RefPtr<nsTextNode> mText;
+
+ explicit nsGenConNode(int32_t aContentIndex)
+ : mPseudoFrame(nullptr), mContentIndex(aContentIndex) {}
+
+ /**
+ * Finish initializing the generated content node once we know the
+ * relevant text frame. This must be called just after
+ * the textframe has been initialized. This need not be called at all
+ * for nodes that don't generate text. This will generally set the
+ * mPseudoFrame, insert the node into aList, and set aTextFrame up
+ * with the correct text.
+ * @param aList the list the node belongs to
+ * @param aPseudoFrame the :before or :after frame
+ * @param aTextFrame the textframe where the node contents will render
+ * @return true iff this marked the list dirty
+ */
+ virtual bool InitTextFrame(nsGenConList* aList, nsIFrame* aPseudoFrame,
+ nsIFrame* aTextFrame) {
+ mPseudoFrame = aPseudoFrame;
+ CheckFrameAssertions();
+ return false;
+ }
+
+ virtual ~nsGenConNode() = default; // XXX Avoid, perhaps?
+
+ protected:
+ void CheckFrameAssertions() {
+ NS_ASSERTION(
+ mContentIndex < int32_t(mPseudoFrame->StyleContent()->ContentCount()) ||
+ // Special-case for the use node created for the legacy markers,
+ // which don't use the content property.
+ (mPseudoFrame->IsBulletFrame() && mContentIndex == 0 &&
+ mPseudoFrame->Style()->GetPseudoType() ==
+ mozilla::PseudoStyleType::marker &&
+ !mPseudoFrame->StyleContent()->ContentCount()),
+ "index out of range");
+ // We allow negative values of mContentIndex for 'counter-reset' and
+ // 'counter-increment'.
+
+ NS_ASSERTION(mContentIndex < 0 ||
+ mPseudoFrame->Style()->GetPseudoType() ==
+ mozilla::PseudoStyleType::before ||
+ mPseudoFrame->Style()->GetPseudoType() ==
+ mozilla::PseudoStyleType::after ||
+ mPseudoFrame->Style()->GetPseudoType() ==
+ mozilla::PseudoStyleType::marker,
+ "not CSS generated content and not counter change");
+ NS_ASSERTION(mContentIndex < 0 ||
+ mPseudoFrame->HasAnyStateBits(NS_FRAME_GENERATED_CONTENT),
+ "not generated content and not counter change");
+ }
+};
+
+class nsGenConList {
+ protected:
+ mozilla::LinkedList<nsGenConNode> mList;
+ uint32_t mSize;
+
+ public:
+ nsGenConList() : mSize(0), mLastInserted(nullptr) {}
+ ~nsGenConList() { Clear(); }
+ void Clear();
+ static nsGenConNode* Next(nsGenConNode* aNode) {
+ MOZ_ASSERT(aNode, "aNode cannot be nullptr!");
+ return aNode->getNext();
+ }
+ static nsGenConNode* Prev(nsGenConNode* aNode) {
+ MOZ_ASSERT(aNode, "aNode cannot be nullptr!");
+ return aNode->getPrevious();
+ }
+ void Insert(nsGenConNode* aNode);
+
+ // Destroy all nodes with aFrame as parent. Returns true if some nodes
+ // have been destroyed; otherwise false.
+ bool DestroyNodesFor(nsIFrame* aFrame);
+
+ // Return true if |aNode1| is after |aNode2|.
+ static bool NodeAfter(const nsGenConNode* aNode1, const nsGenConNode* aNode2);
+
+ bool IsFirst(nsGenConNode* aNode) {
+ MOZ_ASSERT(aNode, "aNode cannot be nullptr!");
+ return aNode == mList.getFirst();
+ }
+
+ bool IsLast(nsGenConNode* aNode) {
+ MOZ_ASSERT(aNode, "aNode cannot be nullptr!");
+ return aNode == mList.getLast();
+ }
+
+ private:
+ void Destroy(nsGenConNode* aNode) {
+ MOZ_ASSERT(aNode, "aNode cannot be nullptr!");
+ delete aNode;
+ mSize--;
+ }
+
+ // Map from frame to the first nsGenConNode of it in the list.
+ nsDataHashtable<nsPtrHashKey<nsIFrame>, nsGenConNode*> mNodes;
+
+ // A weak pointer to the node most recently inserted, used to avoid repeated
+ // list traversals in Insert().
+ nsGenConNode* mLastInserted;
+};
+
+#endif /* nsGenConList_h___ */