/* -*- Mode: C++; tab-width: 2; 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 "nsCOMPtr.h" #include "nsPIListBoxObject.h" #include "nsBoxObject.h" #include "nsIFrame.h" #include "nsBindingManager.h" #include "nsIDOMElement.h" #include "nsIDOMNodeList.h" #include "nsGkAtoms.h" #include "nsIScrollableFrame.h" #include "nsListBoxBodyFrame.h" class nsListBoxObject : public nsPIListBoxObject, public nsBoxObject { public: NS_DECL_ISUPPORTS_INHERITED NS_DECL_NSILISTBOXOBJECT // nsPIListBoxObject virtual nsListBoxBodyFrame* GetListBoxBody(bool aFlush); nsListBoxObject(); // nsPIBoxObject virtual void Clear(); virtual void ClearCachedValues(); protected: nsListBoxBodyFrame *mListBoxBody; }; NS_IMPL_ISUPPORTS_INHERITED2(nsListBoxObject, nsBoxObject, nsIListBoxObject, nsPIListBoxObject) nsListBoxObject::nsListBoxObject() : mListBoxBody(nullptr) { } ////////////////////////////////////////////////////////////////////////// //// nsIListBoxObject NS_IMETHODIMP nsListBoxObject::GetRowCount(PRInt32 *aResult) { nsListBoxBodyFrame* body = GetListBoxBody(true); if (body) return body->GetRowCount(aResult); return NS_OK; } NS_IMETHODIMP nsListBoxObject::GetNumberOfVisibleRows(PRInt32 *aResult) { nsListBoxBodyFrame* body = GetListBoxBody(true); if (body) return body->GetNumberOfVisibleRows(aResult); return NS_OK; } NS_IMETHODIMP nsListBoxObject::GetIndexOfFirstVisibleRow(PRInt32 *aResult) { nsListBoxBodyFrame* body = GetListBoxBody(true); if (body) return body->GetIndexOfFirstVisibleRow(aResult); return NS_OK; } NS_IMETHODIMP nsListBoxObject::EnsureIndexIsVisible(PRInt32 aRowIndex) { nsListBoxBodyFrame* body = GetListBoxBody(true); if (body) return body->EnsureIndexIsVisible(aRowIndex); return NS_OK; } NS_IMETHODIMP nsListBoxObject::ScrollToIndex(PRInt32 aRowIndex) { nsListBoxBodyFrame* body = GetListBoxBody(true); if (body) return body->ScrollToIndex(aRowIndex); return NS_OK; } NS_IMETHODIMP nsListBoxObject::ScrollByLines(PRInt32 aNumLines) { nsListBoxBodyFrame* body = GetListBoxBody(true); if (body) return body->ScrollByLines(aNumLines); return NS_OK; } NS_IMETHODIMP nsListBoxObject::GetItemAtIndex(PRInt32 index, nsIDOMElement **_retval) { nsListBoxBodyFrame* body = GetListBoxBody(true); if (body) return body->GetItemAtIndex(index, _retval); return NS_OK; } NS_IMETHODIMP nsListBoxObject::GetIndexOfItem(nsIDOMElement* aElement, PRInt32 *aResult) { *aResult = 0; nsListBoxBodyFrame* body = GetListBoxBody(true); if (body) return body->GetIndexOfItem(aElement, aResult); return NS_OK; } ////////////////////// static void FindBodyContent(nsIContent* aParent, nsIContent** aResult) { if (aParent->Tag() == nsGkAtoms::listboxbody) { *aResult = aParent; NS_IF_ADDREF(*aResult); } else { nsCOMPtr kids; aParent->OwnerDoc()->BindingManager()->GetXBLChildNodesFor(aParent, getter_AddRefs(kids)); if (!kids) return; PRUint32 i; kids->GetLength(&i); // start from the end, cuz we're smart and we know the listboxbody is probably at the end while (i > 0) { nsCOMPtr childNode; kids->Item(--i, getter_AddRefs(childNode)); nsCOMPtr childContent(do_QueryInterface(childNode)); FindBodyContent(childContent, aResult); if (*aResult) break; } } } nsListBoxBodyFrame* nsListBoxObject::GetListBoxBody(bool aFlush) { if (mListBoxBody) { return mListBoxBody; } nsIPresShell* shell = GetPresShell(false); if (!shell) { return nullptr; } nsIFrame* frame = aFlush ? GetFrame(false) /* does Flush_Frames */ : mContent->GetPrimaryFrame(); if (!frame) return nullptr; // Iterate over our content model children looking for the body. nsCOMPtr content; FindBodyContent(frame->GetContent(), getter_AddRefs(content)); if (!content) return nullptr; // this frame will be a nsGFXScrollFrame frame = content->GetPrimaryFrame(); if (!frame) return nullptr; nsIScrollableFrame* scrollFrame = do_QueryFrame(frame); if (!scrollFrame) return nullptr; // this frame will be the one we want nsIFrame* yeahBaby = scrollFrame->GetScrolledFrame(); if (!yeahBaby) return nullptr; // It's a frame. Refcounts are irrelevant. nsListBoxBodyFrame* listBoxBody = do_QueryFrame(yeahBaby); NS_ENSURE_TRUE(listBoxBody && listBoxBody->SetBoxObject(this), nullptr); mListBoxBody = listBoxBody; return mListBoxBody; } void nsListBoxObject::Clear() { ClearCachedValues(); nsBoxObject::Clear(); } void nsListBoxObject::ClearCachedValues() { mListBoxBody = nullptr; } // Creation Routine /////////////////////////////////////////////////////////////////////// nsresult NS_NewListBoxObject(nsIBoxObject** aResult) { *aResult = new nsListBoxObject; if (!*aResult) return NS_ERROR_OUT_OF_MEMORY; NS_ADDREF(*aResult); return NS_OK; }