/************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_sd.hxx" #include "taskpane/ControlContainer.hxx" #include "taskpane/TaskPaneTreeNode.hxx" #include #include namespace sd { namespace toolpanel { ControlContainer::ControlContainer (TreeNode* pNode) : mpNode(pNode), mnActiveControlIndex((sal_uInt32)-1), mbMultiSelection(false) { } ControlContainer::~ControlContainer (void) { // Set mpNode to NULL so that no one calls it from now on. mpNode = NULL; DeleteChildren(); } void ControlContainer::DeleteChildren (void) { // Deleting the children may lead to calls back to the container. To // prevent the container from accessing the just deleted children, the // maControlList member is first cleared (by transferring its content to // a local list) before the children are destroyed. ControlList maList; maList.swap(maControlList); ControlList::iterator I; ControlList::iterator Iend (maList.end()); for (I=maList.begin(); I!=Iend; ++I) delete *I; if (mpNode != NULL) mpNode->FireStateChangeEvent(EID_ALL_CHILDREN_REMOVED); } sal_uInt32 ControlContainer::AddControl (::std::auto_ptr pControl) { ::osl::MutexGuard aGuard (maMutex); pControl->GetWindow()->Show(); sal_uInt32 nIndex = maControlList.size(); maControlList.push_back (pControl.get()); pControl.release(); ListHasChanged (); if (mpNode != NULL) mpNode->FireStateChangeEvent(EID_CHILD_ADDED, pControl.get()); return nIndex; } void ControlContainer::SetExpansionState ( UINT32 nIndex, ExpansionState aState) { ::osl::MutexGuard aGuard (maMutex); bool bResizeNecessary (false); if (mbMultiSelection) { TreeNode* pControl = GetControl(nIndex); switch (aState) { case ES_TOGGLE: bResizeNecessary = pControl->Expand( ! pControl->IsExpanded()); break; case ES_EXPAND: bResizeNecessary = pControl->Expand(true); break; case ES_COLLAPSE: bResizeNecessary = pControl->Expand(false); break; } } else { // When bExpansionState is true then the control to expand is the // one with the given index. If bExpansionState is false and the // given index points to the active control then then following // control (in cyclic order) it is expanded. When there is only one // control then that is always expanded. do { // Ignore a call with an invalid index. (The seperate comparison // with -1 is not strictly necessary but it is here just in // case.) if (nIndex>=GetControlCount() || nIndex==(sal_uInt32)-1) break; bool bExpand; switch (aState) { default: case ES_TOGGLE: bExpand = ! GetControl(nIndex)->IsExpanded(); break; case ES_EXPAND: bExpand = true; break; case ES_COLLAPSE: bExpand = false; break; } if (bExpand) { // Make the specified control the active one and expand it. mnActiveControlIndex = nIndex; } else { if (nIndex == mnActiveControlIndex) { // We have to determine a new active control since the // current one is about to be collapsed. Choose the // previous one for the last and the next one for all // other. if (mnActiveControlIndex+1 == GetControlCount()) mnActiveControlIndex = GetPreviousIndex(mnActiveControlIndex); else mnActiveControlIndex = GetNextIndex (mnActiveControlIndex); } } // Update the expansion state of all controls. for (UINT32 i=0; iExpand(i == mnActiveControlIndex); } } while (false); } if (bResizeNecessary && mpNode != NULL) mpNode->RequestResize(); } void ControlContainer::SetExpansionState ( TreeNode* pControl, ExpansionState aState) { SetExpansionState (GetControlIndex(pControl), aState); } sal_uInt32 ControlContainer::GetControlIndex (TreeNode* pControlToExpand) const { sal_uInt32 nIndex; for (nIndex=0; nIndexGetWindow()->IsVisible()) nCount += 1; } return nCount; } TreeNode* ControlContainer::GetControl (sal_uInt32 nIndex) const { if (nIndex