123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205 |
- /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
- /* 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 "nsDocShellEnumerator.h"
- #include "nsIDocShellTreeItem.h"
- nsDocShellEnumerator::nsDocShellEnumerator(int32_t aEnumerationDirection)
- : mRootItem(nullptr)
- , mCurIndex(0)
- , mDocShellType(nsIDocShellTreeItem::typeAll)
- , mArrayValid(false)
- , mEnumerationDirection(aEnumerationDirection)
- {
- }
- nsDocShellEnumerator::~nsDocShellEnumerator()
- {
- }
- NS_IMPL_ISUPPORTS(nsDocShellEnumerator, nsISimpleEnumerator)
- NS_IMETHODIMP
- nsDocShellEnumerator::GetNext(nsISupports** aResult)
- {
- NS_ENSURE_ARG_POINTER(aResult);
- *aResult = nullptr;
- nsresult rv = EnsureDocShellArray();
- if (NS_FAILED(rv)) {
- return rv;
- }
- if (mCurIndex >= mItemArray.Length()) {
- return NS_ERROR_FAILURE;
- }
- // post-increment is important here
- nsCOMPtr<nsISupports> item = do_QueryReferent(mItemArray[mCurIndex++], &rv);
- item.forget(aResult);
- return rv;
- }
- NS_IMETHODIMP
- nsDocShellEnumerator::HasMoreElements(bool* aResult)
- {
- NS_ENSURE_ARG_POINTER(aResult);
- *aResult = false;
- nsresult rv = EnsureDocShellArray();
- if (NS_FAILED(rv)) {
- return rv;
- }
- *aResult = (mCurIndex < mItemArray.Length());
- return NS_OK;
- }
- nsresult
- nsDocShellEnumerator::GetEnumerationRootItem(
- nsIDocShellTreeItem** aEnumerationRootItem)
- {
- NS_ENSURE_ARG_POINTER(aEnumerationRootItem);
- nsCOMPtr<nsIDocShellTreeItem> item = do_QueryReferent(mRootItem);
- item.forget(aEnumerationRootItem);
- return NS_OK;
- }
- nsresult
- nsDocShellEnumerator::SetEnumerationRootItem(
- nsIDocShellTreeItem* aEnumerationRootItem)
- {
- mRootItem = do_GetWeakReference(aEnumerationRootItem);
- ClearState();
- return NS_OK;
- }
- nsresult
- nsDocShellEnumerator::GetEnumDocShellType(int32_t* aEnumerationItemType)
- {
- NS_ENSURE_ARG_POINTER(aEnumerationItemType);
- *aEnumerationItemType = mDocShellType;
- return NS_OK;
- }
- nsresult
- nsDocShellEnumerator::SetEnumDocShellType(int32_t aEnumerationItemType)
- {
- mDocShellType = aEnumerationItemType;
- ClearState();
- return NS_OK;
- }
- nsresult
- nsDocShellEnumerator::First()
- {
- mCurIndex = 0;
- return EnsureDocShellArray();
- }
- nsresult
- nsDocShellEnumerator::EnsureDocShellArray()
- {
- if (!mArrayValid) {
- mArrayValid = true;
- return BuildDocShellArray(mItemArray);
- }
- return NS_OK;
- }
- nsresult
- nsDocShellEnumerator::ClearState()
- {
- mItemArray.Clear();
- mArrayValid = false;
- mCurIndex = 0;
- return NS_OK;
- }
- nsresult
- nsDocShellEnumerator::BuildDocShellArray(nsTArray<nsWeakPtr>& aItemArray)
- {
- NS_ENSURE_TRUE(mRootItem, NS_ERROR_NOT_INITIALIZED);
- aItemArray.Clear();
- nsCOMPtr<nsIDocShellTreeItem> item = do_QueryReferent(mRootItem);
- return BuildArrayRecursive(item, aItemArray);
- }
- nsresult
- nsDocShellForwardsEnumerator::BuildArrayRecursive(
- nsIDocShellTreeItem* aItem,
- nsTArray<nsWeakPtr>& aItemArray)
- {
- nsresult rv;
- // add this item to the array
- if (mDocShellType == nsIDocShellTreeItem::typeAll ||
- aItem->ItemType() == mDocShellType) {
- if (!aItemArray.AppendElement(do_GetWeakReference(aItem))) {
- return NS_ERROR_OUT_OF_MEMORY;
- }
- }
- int32_t numChildren;
- rv = aItem->GetChildCount(&numChildren);
- if (NS_FAILED(rv)) {
- return rv;
- }
- for (int32_t i = 0; i < numChildren; ++i) {
- nsCOMPtr<nsIDocShellTreeItem> curChild;
- rv = aItem->GetChildAt(i, getter_AddRefs(curChild));
- if (NS_FAILED(rv)) {
- return rv;
- }
- rv = BuildArrayRecursive(curChild, aItemArray);
- if (NS_FAILED(rv)) {
- return rv;
- }
- }
- return NS_OK;
- }
- nsresult
- nsDocShellBackwardsEnumerator::BuildArrayRecursive(
- nsIDocShellTreeItem* aItem,
- nsTArray<nsWeakPtr>& aItemArray)
- {
- nsresult rv;
- int32_t numChildren;
- rv = aItem->GetChildCount(&numChildren);
- if (NS_FAILED(rv)) {
- return rv;
- }
- for (int32_t i = numChildren - 1; i >= 0; --i) {
- nsCOMPtr<nsIDocShellTreeItem> curChild;
- rv = aItem->GetChildAt(i, getter_AddRefs(curChild));
- if (NS_FAILED(rv)) {
- return rv;
- }
- rv = BuildArrayRecursive(curChild, aItemArray);
- if (NS_FAILED(rv)) {
- return rv;
- }
- }
- // add this item to the array
- if (mDocShellType == nsIDocShellTreeItem::typeAll ||
- aItem->ItemType() == mDocShellType) {
- if (!aItemArray.AppendElement(do_GetWeakReference(aItem))) {
- return NS_ERROR_OUT_OF_MEMORY;
- }
- }
- return NS_OK;
- }
|