2fdee9fd60
Following Change-Id: Id48f81deb05aee2026509037f7d14575735e5be0 Author: Michael Weghorn <m.weghorn@posteo.de> Date: Wed Jul 10 14:49:03 2024 +0200 VCLUnoHelper: Align AWT <-> VCL helpers with convert.hxx impl , port all uses of the helper functions defined in `include/toolkit/helper/convert.hxx` to use the `VCLUnoHelper` equivalents instead, to unify usage and avoid duplication. Drop `include/toolkit/helper/convert.hxx` now that it's unused. Change-Id: I22695a93e40e47bb2b14d191a2e0a4eb7c856895 Reviewed-on: https://gerrit.libreoffice.org/c/core/+/170317 Tested-by: Jenkins Reviewed-by: Michael Weghorn <m.weghorn@posteo.de>
1579 lines
43 KiB
C++
1579 lines
43 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*
|
|
* This file is part of the LibreOffice project.
|
|
*
|
|
* 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/.
|
|
*
|
|
* This file incorporates work covered by the following license notice:
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed
|
|
* with this work for additional information regarding copyright
|
|
* ownership. The ASF licenses this file to you under the Apache
|
|
* License, Version 2.0 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
|
|
*/
|
|
|
|
|
|
#include <com/sun/star/graphic/GraphicProvider.hpp>
|
|
#include <com/sun/star/lang/DisposedException.hpp>
|
|
#include <com/sun/star/view/SelectionType.hpp>
|
|
#include <com/sun/star/util/VetoException.hpp>
|
|
#include <o3tl/any.hxx>
|
|
#include <helper/property.hxx>
|
|
#include <toolkit/helper/vclunohelper.hxx>
|
|
|
|
#include <com/sun/star/awt/tree/XMutableTreeNode.hpp>
|
|
#include <controls/treecontrolpeer.hxx>
|
|
#include <comphelper/processfactory.hxx>
|
|
#include <comphelper/propertyvalue.hxx>
|
|
|
|
#include <cppuhelper/implbase.hxx>
|
|
#include <rtl/ref.hxx>
|
|
#include <vcl/graph.hxx>
|
|
#include <vcl/svapp.hxx>
|
|
#include <vcl/toolkit/treelistbox.hxx>
|
|
#include <vcl/toolkit/treelistentry.hxx>
|
|
#include <vcl/toolkit/viewdataentry.hxx>
|
|
#include <vcl/toolkit/svlbitm.hxx>
|
|
|
|
#include <map>
|
|
#include <memory>
|
|
#include <list>
|
|
|
|
using namespace ::com::sun::star;
|
|
using namespace css::uno;
|
|
using namespace css::lang;
|
|
using namespace css::awt::tree;
|
|
using namespace css::beans;
|
|
using namespace css::view;
|
|
using namespace css::container;
|
|
using namespace css::util;
|
|
using namespace css::graphic;
|
|
|
|
namespace {
|
|
|
|
struct LockGuard
|
|
{
|
|
public:
|
|
explicit LockGuard( sal_Int32& rLock )
|
|
: mrLock( rLock )
|
|
{
|
|
rLock++;
|
|
}
|
|
|
|
~LockGuard()
|
|
{
|
|
mrLock--;
|
|
}
|
|
|
|
sal_Int32& mrLock;
|
|
};
|
|
|
|
|
|
class ImplContextGraphicItem : public SvLBoxContextBmp
|
|
{
|
|
public:
|
|
ImplContextGraphicItem( Image const & rI1, Image const & rI2, bool bExpanded)
|
|
: SvLBoxContextBmp(rI1, rI2, bExpanded) {}
|
|
|
|
OUString msExpandedGraphicURL;
|
|
OUString msCollapsedGraphicURL;
|
|
};
|
|
|
|
|
|
}
|
|
|
|
class UnoTreeListBoxImpl : public SvTreeListBox
|
|
{
|
|
public:
|
|
UnoTreeListBoxImpl( TreeControlPeer* pPeer, vcl::Window* pParent, WinBits nWinStyle );
|
|
virtual ~UnoTreeListBoxImpl() override;
|
|
virtual void dispose() override;
|
|
|
|
void insert( SvTreeListEntry* pEntry, SvTreeListEntry* pParent, sal_uInt32 nPos );
|
|
|
|
virtual void RequestingChildren( SvTreeListEntry* pParent ) override;
|
|
|
|
virtual bool EditingEntry( SvTreeListEntry* pEntry ) override;
|
|
virtual bool EditedEntry( SvTreeListEntry* pEntry, const OUString& rNewText ) override;
|
|
|
|
DECL_LINK(OnSelectionChangeHdl, SvTreeListBox*, void);
|
|
DECL_LINK(OnExpandingHdl, SvTreeListBox*, bool);
|
|
DECL_LINK(OnExpandedHdl, SvTreeListBox*, void);
|
|
|
|
private:
|
|
rtl::Reference< TreeControlPeer > mxPeer;
|
|
};
|
|
|
|
|
|
namespace {
|
|
|
|
class UnoTreeListItem : public SvLBoxString
|
|
{
|
|
public:
|
|
UnoTreeListItem();
|
|
|
|
void InitViewData( SvTreeListBox*,SvTreeListEntry*,SvViewDataItem * = nullptr ) override;
|
|
void SetImage( const Image& rImage );
|
|
const OUString& GetGraphicURL() const { return maGraphicURL;}
|
|
void SetGraphicURL( const OUString& rGraphicURL );
|
|
virtual void Paint(const Point& rPos, SvTreeListBox& rOutDev, vcl::RenderContext& rRenderContext,
|
|
const SvViewDataEntry* pView, const SvTreeListEntry& rEntry) override;
|
|
std::unique_ptr<SvLBoxItem> Clone( SvLBoxItem const * pSource ) const override;
|
|
|
|
private:
|
|
OUString maGraphicURL;
|
|
Image maImage;
|
|
};
|
|
|
|
}
|
|
|
|
class UnoTreeListEntry : public SvTreeListEntry
|
|
{
|
|
public:
|
|
UnoTreeListEntry( const Reference< XTreeNode >& xNode, TreeControlPeer* pPeer );
|
|
virtual ~UnoTreeListEntry() override;
|
|
|
|
Reference< XTreeNode > mxNode;
|
|
TreeControlPeer* mpPeer;
|
|
};
|
|
|
|
TreeControlPeer::TreeControlPeer()
|
|
: maSelectionListeners( *this )
|
|
, maTreeExpansionListeners( *this )
|
|
, maTreeEditListeners( *this )
|
|
, mbIsRootDisplayed(false)
|
|
, mpTreeImpl( nullptr )
|
|
, mnEditLock( 0 )
|
|
{
|
|
}
|
|
|
|
|
|
TreeControlPeer::~TreeControlPeer()
|
|
{
|
|
if( mpTreeImpl )
|
|
mpTreeImpl->Clear();
|
|
}
|
|
|
|
|
|
void TreeControlPeer::addEntry( UnoTreeListEntry* pEntry )
|
|
{
|
|
if( pEntry && pEntry->mxNode.is() )
|
|
{
|
|
if( !mpTreeNodeMap )
|
|
{
|
|
mpTreeNodeMap.reset( new TreeNodeMap );
|
|
}
|
|
|
|
(*mpTreeNodeMap)[ pEntry->mxNode ] = pEntry;
|
|
}
|
|
}
|
|
|
|
|
|
void TreeControlPeer::removeEntry( UnoTreeListEntry const * pEntry )
|
|
{
|
|
if( mpTreeNodeMap && pEntry && pEntry->mxNode.is() )
|
|
{
|
|
TreeNodeMap::iterator aIter( mpTreeNodeMap->find( pEntry->mxNode ) );
|
|
if( aIter != mpTreeNodeMap->end() )
|
|
{
|
|
mpTreeNodeMap->erase( aIter );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
UnoTreeListEntry* TreeControlPeer::getEntry( const Reference< XTreeNode >& xNode, bool bThrow /* = true */ )
|
|
{
|
|
if( mpTreeNodeMap )
|
|
{
|
|
TreeNodeMap::iterator aIter( mpTreeNodeMap->find( xNode ) );
|
|
if( aIter != mpTreeNodeMap->end() )
|
|
return (*aIter).second;
|
|
}
|
|
|
|
if( bThrow )
|
|
throw IllegalArgumentException();
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
|
|
vcl::Window* TreeControlPeer::createVclControl( vcl::Window* pParent, sal_Int64 nWinStyle )
|
|
{
|
|
mpTreeImpl = VclPtr<UnoTreeListBoxImpl>::Create( this, pParent, nWinStyle );
|
|
return mpTreeImpl;
|
|
}
|
|
|
|
|
|
/** called from the UnoTreeListBoxImpl when it gets deleted */
|
|
void TreeControlPeer::disposeControl()
|
|
{
|
|
mpTreeNodeMap.reset();
|
|
mpTreeImpl = nullptr;
|
|
}
|
|
|
|
|
|
UnoTreeListEntry* TreeControlPeer::createEntry( const Reference< XTreeNode >& xNode, UnoTreeListEntry* pParent, sal_uInt32 nPos /* = TREELIST_APPEND */ )
|
|
{
|
|
UnoTreeListEntry* pEntry = nullptr;
|
|
if( mpTreeImpl )
|
|
{
|
|
Image aImage;
|
|
pEntry = new UnoTreeListEntry( xNode, this );
|
|
pEntry->AddItem(std::make_unique<ImplContextGraphicItem>(aImage, aImage, true));
|
|
|
|
std::unique_ptr<UnoTreeListItem> pUnoItem(new UnoTreeListItem);
|
|
|
|
if( !xNode->getNodeGraphicURL().isEmpty() )
|
|
{
|
|
pUnoItem->SetGraphicURL( xNode->getNodeGraphicURL() );
|
|
Image aNodeImage;
|
|
loadImage( xNode->getNodeGraphicURL(), aNodeImage );
|
|
pUnoItem->SetImage( aNodeImage );
|
|
mpTreeImpl->AdjustEntryHeight( aNodeImage );
|
|
}
|
|
|
|
pEntry->AddItem(std::move(pUnoItem));
|
|
|
|
mpTreeImpl->insert( pEntry, pParent, nPos );
|
|
|
|
if( !msDefaultExpandedGraphicURL.isEmpty() )
|
|
mpTreeImpl->SetExpandedEntryBmp( pEntry, maDefaultExpandedImage );
|
|
|
|
if( !msDefaultCollapsedGraphicURL.isEmpty() )
|
|
mpTreeImpl->SetCollapsedEntryBmp( pEntry, maDefaultCollapsedImage );
|
|
|
|
updateEntry( pEntry );
|
|
}
|
|
return pEntry;
|
|
}
|
|
|
|
|
|
void TreeControlPeer::updateEntry( UnoTreeListEntry* pEntry )
|
|
{
|
|
bool bChanged = false;
|
|
if( !(pEntry && pEntry->mxNode.is() && mpTreeImpl) )
|
|
return;
|
|
|
|
const OUString aValue( getEntryString( pEntry->mxNode->getDisplayValue() ) );
|
|
UnoTreeListItem* pUnoItem = dynamic_cast< UnoTreeListItem* >( &pEntry->GetItem( 1 ) );
|
|
if( pUnoItem )
|
|
{
|
|
if( aValue != pUnoItem->GetText() )
|
|
{
|
|
pUnoItem->SetText( aValue );
|
|
bChanged = true;
|
|
}
|
|
|
|
if( pUnoItem->GetGraphicURL() != pEntry->mxNode->getNodeGraphicURL() )
|
|
{
|
|
Image aImage;
|
|
if( loadImage( pEntry->mxNode->getNodeGraphicURL(), aImage ) )
|
|
{
|
|
pUnoItem->SetGraphicURL( pEntry->mxNode->getNodeGraphicURL() );
|
|
pUnoItem->SetImage( aImage );
|
|
mpTreeImpl->AdjustEntryHeight( aImage );
|
|
bChanged = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if( bool(pEntry->mxNode->hasChildrenOnDemand()) != pEntry->HasChildrenOnDemand() )
|
|
{
|
|
pEntry->EnableChildrenOnDemand( pEntry->mxNode->hasChildrenOnDemand() );
|
|
bChanged = true;
|
|
}
|
|
|
|
ImplContextGraphicItem* pContextGraphicItem = dynamic_cast< ImplContextGraphicItem* >( &pEntry->GetItem( 0 ) );
|
|
if( pContextGraphicItem )
|
|
{
|
|
if( pContextGraphicItem->msExpandedGraphicURL != pEntry->mxNode->getExpandedGraphicURL() )
|
|
{
|
|
Image aImage;
|
|
if( loadImage( pEntry->mxNode->getExpandedGraphicURL(), aImage ) )
|
|
{
|
|
pContextGraphicItem->msExpandedGraphicURL = pEntry->mxNode->getExpandedGraphicURL();
|
|
mpTreeImpl->SetExpandedEntryBmp( pEntry, aImage );
|
|
bChanged = true;
|
|
}
|
|
}
|
|
if( pContextGraphicItem->msCollapsedGraphicURL != pEntry->mxNode->getCollapsedGraphicURL() )
|
|
{
|
|
Image aImage;
|
|
if( loadImage( pEntry->mxNode->getCollapsedGraphicURL(), aImage ) )
|
|
{
|
|
pContextGraphicItem->msCollapsedGraphicURL = pEntry->mxNode->getCollapsedGraphicURL();
|
|
mpTreeImpl->SetCollapsedEntryBmp( pEntry, aImage );
|
|
bChanged = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if( bChanged )
|
|
mpTreeImpl->GetModel()->InvalidateEntry( pEntry );
|
|
}
|
|
|
|
|
|
void TreeControlPeer::onSelectionChanged()
|
|
{
|
|
Reference< XInterface > xSource( getXWeak() );
|
|
EventObject aEvent( xSource );
|
|
maSelectionListeners.selectionChanged( aEvent );
|
|
}
|
|
|
|
|
|
void TreeControlPeer::onRequestChildNodes( const Reference< XTreeNode >& xNode )
|
|
{
|
|
try
|
|
{
|
|
Reference< XInterface > xSource( getXWeak() );
|
|
TreeExpansionEvent aEvent( xSource, xNode );
|
|
maTreeExpansionListeners.requestChildNodes( aEvent );
|
|
}
|
|
catch( Exception& )
|
|
{
|
|
}
|
|
}
|
|
|
|
|
|
bool TreeControlPeer::onExpanding( const Reference< XTreeNode >& xNode, bool bExpanding )
|
|
{
|
|
try
|
|
{
|
|
Reference< XInterface > xSource( getXWeak() );
|
|
TreeExpansionEvent aEvent( xSource, xNode );
|
|
if( bExpanding )
|
|
{
|
|
maTreeExpansionListeners.treeExpanding( aEvent );
|
|
}
|
|
else
|
|
{
|
|
maTreeExpansionListeners.treeCollapsing( aEvent );
|
|
}
|
|
}
|
|
catch( Exception& )
|
|
{
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
void TreeControlPeer::onExpanded( const Reference< XTreeNode >& xNode, bool bExpanding )
|
|
{
|
|
try
|
|
{
|
|
Reference< XInterface > xSource( getXWeak() );
|
|
TreeExpansionEvent aEvent( xSource, xNode );
|
|
|
|
if( bExpanding )
|
|
{
|
|
maTreeExpansionListeners.treeExpanded( aEvent );
|
|
}
|
|
else
|
|
{
|
|
maTreeExpansionListeners.treeCollapsed( aEvent );
|
|
}
|
|
}
|
|
catch( Exception& )
|
|
{
|
|
}
|
|
}
|
|
|
|
|
|
void TreeControlPeer::fillTree( UnoTreeListBoxImpl& rTree, const Reference< XTreeDataModel >& xDataModel )
|
|
{
|
|
rTree.Clear();
|
|
|
|
if( !xDataModel.is() )
|
|
return;
|
|
|
|
Reference< XTreeNode > xRootNode( xDataModel->getRoot() );
|
|
if( !xRootNode.is() )
|
|
return;
|
|
|
|
if( mbIsRootDisplayed )
|
|
{
|
|
addNode( rTree, xRootNode, nullptr );
|
|
}
|
|
else
|
|
{
|
|
const sal_Int32 nChildCount = xRootNode->getChildCount();
|
|
for( sal_Int32 nChild = 0; nChild < nChildCount; nChild++ )
|
|
addNode( rTree, xRootNode->getChildAt( nChild ), nullptr );
|
|
}
|
|
}
|
|
|
|
|
|
void TreeControlPeer::addNode( UnoTreeListBoxImpl& rTree, const Reference< XTreeNode >& xNode, UnoTreeListEntry* pParentEntry )
|
|
{
|
|
if( xNode.is() )
|
|
{
|
|
UnoTreeListEntry* pEntry = createEntry( xNode, pParentEntry, TREELIST_APPEND );
|
|
const sal_Int32 nChildCount = xNode->getChildCount();
|
|
for( sal_Int32 nChild = 0; nChild < nChildCount; nChild++ )
|
|
addNode( rTree, xNode->getChildAt( nChild ), pEntry );
|
|
}
|
|
}
|
|
|
|
|
|
UnoTreeListBoxImpl& TreeControlPeer::getTreeListBoxOrThrow() const
|
|
{
|
|
if( !mpTreeImpl )
|
|
throw DisposedException();
|
|
return *mpTreeImpl;
|
|
}
|
|
|
|
|
|
void TreeControlPeer::ChangeNodesSelection( const Any& rSelection, bool bSelect, bool bSetSelection )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
Reference< XTreeNode > xTempNode;
|
|
|
|
Sequence<Reference<XTreeNode>> pNodes;
|
|
sal_Int32 nCount = 0;
|
|
|
|
if( rSelection.hasValue() )
|
|
{
|
|
switch( rSelection.getValueTypeClass() )
|
|
{
|
|
case TypeClass_INTERFACE:
|
|
{
|
|
rSelection >>= xTempNode;
|
|
if( xTempNode.is() )
|
|
{
|
|
nCount = 1;
|
|
pNodes = {xTempNode};
|
|
}
|
|
break;
|
|
}
|
|
case TypeClass_SEQUENCE:
|
|
{
|
|
if( auto rSeq = o3tl::tryAccess<Sequence<Reference<XTreeNode>>>(
|
|
rSelection) )
|
|
{
|
|
nCount = rSeq->getLength();
|
|
pNodes = *rSeq;
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if( nCount == 0 )
|
|
throw IllegalArgumentException();
|
|
}
|
|
|
|
if( bSetSelection )
|
|
rTree.SelectAll( false );
|
|
|
|
for( sal_Int32 i = 0; i != nCount; ++i )
|
|
{
|
|
UnoTreeListEntry* pEntry = getEntry( pNodes[i] );
|
|
rTree.Select( pEntry, bSelect );
|
|
}
|
|
}
|
|
|
|
|
|
// css::view::XSelectionSupplier
|
|
|
|
|
|
sal_Bool SAL_CALL TreeControlPeer::select( const Any& rSelection )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
ChangeNodesSelection( rSelection, true, true );
|
|
return true;
|
|
}
|
|
|
|
|
|
Any SAL_CALL TreeControlPeer::getSelection()
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
Any aRet;
|
|
|
|
sal_uInt32 nSelectionCount = rTree.GetSelectionCount();
|
|
if( nSelectionCount == 1 )
|
|
{
|
|
UnoTreeListEntry* pEntry = dynamic_cast< UnoTreeListEntry* >( rTree.FirstSelected() );
|
|
if( pEntry && pEntry->mxNode.is() )
|
|
aRet <<= pEntry->mxNode;
|
|
}
|
|
else if( nSelectionCount > 1 )
|
|
{
|
|
Sequence< Reference< XTreeNode > > aSelection( nSelectionCount );
|
|
Reference< XTreeNode >* pNodes = aSelection.getArray();
|
|
UnoTreeListEntry* pEntry = dynamic_cast< UnoTreeListEntry* >( rTree.FirstSelected() );
|
|
while( pEntry && nSelectionCount )
|
|
{
|
|
*pNodes++ = pEntry->mxNode;
|
|
pEntry = dynamic_cast< UnoTreeListEntry* >( rTree.NextSelected( pEntry ) );
|
|
--nSelectionCount;
|
|
}
|
|
|
|
OSL_ASSERT( (pEntry == nullptr) && (nSelectionCount == 0) );
|
|
aRet <<= aSelection;
|
|
}
|
|
|
|
return aRet;
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::addSelectionChangeListener( const Reference< XSelectionChangeListener >& xListener )
|
|
{
|
|
maSelectionListeners.addInterface( xListener );
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::removeSelectionChangeListener( const Reference< XSelectionChangeListener >& xListener )
|
|
{
|
|
maSelectionListeners.addInterface( xListener );
|
|
}
|
|
|
|
|
|
// css::view::XMultiSelectionSupplier
|
|
|
|
|
|
sal_Bool SAL_CALL TreeControlPeer::addSelection( const Any& rSelection )
|
|
{
|
|
ChangeNodesSelection( rSelection, true, false );
|
|
return true;
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::removeSelection( const Any& rSelection )
|
|
{
|
|
ChangeNodesSelection( rSelection, false, false );
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::clearSelection()
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
getTreeListBoxOrThrow().SelectAll( false );
|
|
}
|
|
|
|
|
|
sal_Int32 SAL_CALL TreeControlPeer::getSelectionCount()
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
return getTreeListBoxOrThrow().GetSelectionCount();
|
|
}
|
|
|
|
namespace {
|
|
|
|
class TreeSelectionEnumeration : public ::cppu::WeakImplHelper< XEnumeration >
|
|
{
|
|
public:
|
|
explicit TreeSelectionEnumeration( std::list< Any >& rSelection );
|
|
virtual sal_Bool SAL_CALL hasMoreElements() override;
|
|
virtual Any SAL_CALL nextElement() override;
|
|
|
|
std::list< Any > maSelection;
|
|
std::list< Any >::iterator maIter;
|
|
};
|
|
|
|
}
|
|
|
|
TreeSelectionEnumeration::TreeSelectionEnumeration( std::list< Any >& rSelection )
|
|
{
|
|
maSelection.swap( rSelection );
|
|
maIter = maSelection.begin();
|
|
}
|
|
|
|
|
|
sal_Bool SAL_CALL TreeSelectionEnumeration::hasMoreElements()
|
|
{
|
|
return maIter != maSelection.end();
|
|
}
|
|
|
|
|
|
Any SAL_CALL TreeSelectionEnumeration::nextElement()
|
|
{
|
|
if( maIter == maSelection.end() )
|
|
throw NoSuchElementException();
|
|
|
|
return (*maIter++);
|
|
}
|
|
|
|
|
|
Reference< XEnumeration > SAL_CALL TreeControlPeer::createSelectionEnumeration()
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
sal_uInt32 nSelectionCount = rTree.GetSelectionCount();
|
|
std::list< Any > aSelection( nSelectionCount );
|
|
|
|
UnoTreeListEntry* pEntry = dynamic_cast< UnoTreeListEntry* >( rTree.FirstSelected() );
|
|
while( pEntry && nSelectionCount )
|
|
{
|
|
aSelection.emplace_back( pEntry->mxNode );
|
|
pEntry = dynamic_cast< UnoTreeListEntry* >( rTree.NextSelected( pEntry ) );
|
|
--nSelectionCount;
|
|
}
|
|
|
|
OSL_ASSERT( (pEntry == nullptr) && (nSelectionCount == 0) );
|
|
|
|
return Reference< XEnumeration >( new TreeSelectionEnumeration( aSelection ) );
|
|
}
|
|
|
|
|
|
Reference< XEnumeration > SAL_CALL TreeControlPeer::createReverseSelectionEnumeration()
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
sal_uInt32 nSelectionCount = rTree.GetSelectionCount();
|
|
std::list< Any > aSelection;
|
|
|
|
UnoTreeListEntry* pEntry = dynamic_cast< UnoTreeListEntry* >( rTree.FirstSelected() );
|
|
while( pEntry && nSelectionCount )
|
|
{
|
|
aSelection.push_front( Any( pEntry->mxNode ) );
|
|
pEntry = dynamic_cast< UnoTreeListEntry* >( rTree.NextSelected( pEntry ) );
|
|
--nSelectionCount;
|
|
}
|
|
|
|
OSL_ASSERT( (pEntry == nullptr) && (nSelectionCount == 0) );
|
|
|
|
return Reference< XEnumeration >( new TreeSelectionEnumeration( aSelection ) );
|
|
}
|
|
|
|
|
|
// css::awt::XTreeControl
|
|
|
|
|
|
OUString SAL_CALL TreeControlPeer::getDefaultExpandedGraphicURL()
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
return msDefaultExpandedGraphicURL;
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::setDefaultExpandedGraphicURL( const OUString& sDefaultExpandedGraphicURL )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
if( msDefaultExpandedGraphicURL == sDefaultExpandedGraphicURL )
|
|
return;
|
|
|
|
if( !sDefaultExpandedGraphicURL.isEmpty() )
|
|
loadImage( sDefaultExpandedGraphicURL, maDefaultExpandedImage );
|
|
else
|
|
maDefaultExpandedImage = Image();
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
SvTreeListEntry* pEntry = rTree.First();
|
|
while( pEntry )
|
|
{
|
|
ImplContextGraphicItem* pContextGraphicItem = dynamic_cast< ImplContextGraphicItem* >( &pEntry->GetItem( 0 ) );
|
|
if( pContextGraphicItem )
|
|
{
|
|
if( pContextGraphicItem->msExpandedGraphicURL.isEmpty() )
|
|
rTree.SetExpandedEntryBmp( pEntry, maDefaultExpandedImage );
|
|
}
|
|
pEntry = rTree.Next( pEntry );
|
|
}
|
|
|
|
msDefaultExpandedGraphicURL = sDefaultExpandedGraphicURL;
|
|
}
|
|
|
|
|
|
OUString SAL_CALL TreeControlPeer::getDefaultCollapsedGraphicURL()
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
return msDefaultCollapsedGraphicURL;
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::setDefaultCollapsedGraphicURL( const OUString& sDefaultCollapsedGraphicURL )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
if( msDefaultCollapsedGraphicURL == sDefaultCollapsedGraphicURL )
|
|
return;
|
|
|
|
if( !sDefaultCollapsedGraphicURL.isEmpty() )
|
|
loadImage( sDefaultCollapsedGraphicURL, maDefaultCollapsedImage );
|
|
else
|
|
maDefaultCollapsedImage = Image();
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
SvTreeListEntry* pEntry = rTree.First();
|
|
while( pEntry )
|
|
{
|
|
ImplContextGraphicItem* pContextGraphicItem = dynamic_cast< ImplContextGraphicItem* >( &pEntry->GetItem( 0 ) );
|
|
if( pContextGraphicItem )
|
|
{
|
|
if( pContextGraphicItem->msCollapsedGraphicURL.isEmpty() )
|
|
rTree.SetCollapsedEntryBmp( pEntry, maDefaultCollapsedImage );
|
|
}
|
|
pEntry = rTree.Next( pEntry );
|
|
}
|
|
|
|
msDefaultCollapsedGraphicURL = sDefaultCollapsedGraphicURL;
|
|
}
|
|
|
|
|
|
sal_Bool SAL_CALL TreeControlPeer::isNodeExpanded( const Reference< XTreeNode >& xNode )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
UnoTreeListEntry* pEntry = getEntry( xNode );
|
|
return pEntry && rTree.IsExpanded( pEntry );
|
|
}
|
|
|
|
|
|
sal_Bool SAL_CALL TreeControlPeer::isNodeCollapsed( const Reference< XTreeNode >& xNode )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
return !isNodeExpanded( xNode );
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::makeNodeVisible( const Reference< XTreeNode >& xNode )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
UnoTreeListEntry* pEntry = getEntry( xNode );
|
|
if( pEntry )
|
|
rTree.MakeVisible( pEntry );
|
|
}
|
|
|
|
|
|
sal_Bool SAL_CALL TreeControlPeer::isNodeVisible( const Reference< XTreeNode >& xNode )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
UnoTreeListEntry* pEntry = getEntry( xNode );
|
|
return pEntry && rTree.IsEntryVisible( pEntry );
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::expandNode( const Reference< XTreeNode >& xNode )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
UnoTreeListEntry* pEntry = getEntry( xNode );
|
|
if( pEntry )
|
|
rTree.Expand( pEntry );
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::collapseNode( const Reference< XTreeNode >& xNode )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
UnoTreeListEntry* pEntry = getEntry( xNode );
|
|
if( pEntry )
|
|
rTree.Collapse( pEntry );
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::addTreeExpansionListener( const Reference< XTreeExpansionListener >& xListener )
|
|
{
|
|
maTreeExpansionListeners.addInterface( xListener );
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::removeTreeExpansionListener( const Reference< XTreeExpansionListener >& xListener )
|
|
{
|
|
maTreeExpansionListeners.removeInterface( xListener );
|
|
}
|
|
|
|
|
|
Reference< XTreeNode > SAL_CALL TreeControlPeer::getNodeForLocation( sal_Int32 x, sal_Int32 y )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
Reference< XTreeNode > xNode;
|
|
|
|
const Point aPos( x, y );
|
|
UnoTreeListEntry* pEntry = dynamic_cast< UnoTreeListEntry* >( rTree.GetEntry( aPos, true ) );
|
|
if( pEntry )
|
|
xNode = pEntry->mxNode;
|
|
|
|
return xNode;
|
|
}
|
|
|
|
|
|
Reference< XTreeNode > SAL_CALL TreeControlPeer::getClosestNodeForLocation( sal_Int32 x, sal_Int32 y )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
Reference< XTreeNode > xNode;
|
|
|
|
const Point aPos( x, y );
|
|
UnoTreeListEntry* pEntry = dynamic_cast< UnoTreeListEntry* >( rTree.GetEntry( aPos, true ) );
|
|
if( pEntry )
|
|
xNode = pEntry->mxNode;
|
|
|
|
return xNode;
|
|
}
|
|
|
|
|
|
awt::Rectangle SAL_CALL TreeControlPeer::getNodeRect( const Reference< XTreeNode >& i_Node )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
UnoTreeListEntry* pEntry = getEntry( i_Node );
|
|
|
|
::tools::Rectangle aEntryRect( rTree.GetFocusRect( pEntry, rTree.GetEntryPosition( pEntry ).Y() ) );
|
|
return VCLUnoHelper::ConvertToAWTRect( aEntryRect );
|
|
}
|
|
|
|
|
|
sal_Bool SAL_CALL TreeControlPeer::isEditing( )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
return rTree.IsEditingActive();
|
|
}
|
|
|
|
|
|
sal_Bool SAL_CALL TreeControlPeer::stopEditing()
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
if( rTree.IsEditingActive() )
|
|
{
|
|
rTree.EndEditing();
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::cancelEditing( )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
rTree.EndEditing();
|
|
}
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::startEditingAtNode( const Reference< XTreeNode >& xNode )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
UnoTreeListEntry* pEntry = getEntry( xNode );
|
|
rTree.EditEntry( pEntry );
|
|
}
|
|
|
|
void SAL_CALL TreeControlPeer::addTreeEditListener( const Reference< XTreeEditListener >& xListener )
|
|
{
|
|
maTreeEditListeners.addInterface( xListener );
|
|
}
|
|
|
|
void SAL_CALL TreeControlPeer::removeTreeEditListener( const Reference< XTreeEditListener >& xListener )
|
|
{
|
|
maTreeEditListeners.removeInterface( xListener );
|
|
}
|
|
|
|
bool TreeControlPeer::onEditingEntry( UnoTreeListEntry const * pEntry )
|
|
{
|
|
if( mpTreeImpl && pEntry && pEntry->mxNode.is() && (maTreeEditListeners.getLength() > 0) )
|
|
{
|
|
try
|
|
{
|
|
maTreeEditListeners.nodeEditing( pEntry->mxNode );
|
|
}
|
|
catch( VetoException& )
|
|
{
|
|
return false;
|
|
}
|
|
catch( Exception& )
|
|
{
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool TreeControlPeer::onEditedEntry( UnoTreeListEntry const * pEntry, const OUString& rNewText )
|
|
{
|
|
if( mpTreeImpl && pEntry && pEntry->mxNode.is() ) try
|
|
{
|
|
LockGuard aLockGuard( mnEditLock );
|
|
if( maTreeEditListeners.getLength() > 0 )
|
|
{
|
|
maTreeEditListeners.nodeEdited( pEntry->mxNode, rNewText );
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
Reference< XMutableTreeNode > xMutableNode( pEntry->mxNode, UNO_QUERY );
|
|
if( xMutableNode.is() )
|
|
xMutableNode->setDisplayValue( Any( rNewText ) );
|
|
else
|
|
return false;
|
|
}
|
|
|
|
}
|
|
catch( Exception& )
|
|
{
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
// css::awt::tree::TreeDataModelListener
|
|
|
|
|
|
void SAL_CALL TreeControlPeer::treeNodesChanged( const css::awt::tree::TreeDataModelEvent& rEvent )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
if( mnEditLock != 0 )
|
|
return;
|
|
|
|
updateTree( rEvent );
|
|
}
|
|
|
|
void SAL_CALL TreeControlPeer::treeNodesInserted( const css::awt::tree::TreeDataModelEvent& rEvent )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
if( mnEditLock != 0 )
|
|
return;
|
|
|
|
updateTree( rEvent );
|
|
}
|
|
|
|
void SAL_CALL TreeControlPeer::treeNodesRemoved( const css::awt::tree::TreeDataModelEvent& rEvent )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
if( mnEditLock != 0 )
|
|
return;
|
|
|
|
updateTree( rEvent );
|
|
}
|
|
|
|
void SAL_CALL TreeControlPeer::treeStructureChanged( const css::awt::tree::TreeDataModelEvent& rEvent )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
if( mnEditLock != 0 )
|
|
return;
|
|
|
|
updateTree( rEvent );
|
|
}
|
|
|
|
void TreeControlPeer::updateTree( const css::awt::tree::TreeDataModelEvent& rEvent )
|
|
{
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
Sequence< Reference< XTreeNode > > Nodes;
|
|
Reference< XTreeNode > xNode( rEvent.ParentNode );
|
|
if( !xNode.is() && Nodes.hasElements() )
|
|
{
|
|
xNode = Nodes[0];
|
|
}
|
|
|
|
if( xNode.is() )
|
|
updateNode( rTree, xNode );
|
|
}
|
|
|
|
void TreeControlPeer::updateNode( UnoTreeListBoxImpl const & rTree, const Reference< XTreeNode >& xNode )
|
|
{
|
|
if( !xNode.is() )
|
|
return;
|
|
|
|
UnoTreeListEntry* pNodeEntry = getEntry( xNode, false );
|
|
|
|
if( !pNodeEntry )
|
|
{
|
|
Reference< XTreeNode > xParentNode( xNode->getParent() );
|
|
UnoTreeListEntry* pParentEntry = nullptr;
|
|
sal_uInt32 nChild = TREELIST_APPEND;
|
|
|
|
if( xParentNode.is() )
|
|
{
|
|
pParentEntry = getEntry( xParentNode );
|
|
nChild = xParentNode->getIndex( xNode );
|
|
}
|
|
|
|
pNodeEntry = createEntry( xNode, pParentEntry, nChild );
|
|
}
|
|
|
|
updateChildNodes( rTree, xNode, pNodeEntry );
|
|
}
|
|
|
|
void TreeControlPeer::updateChildNodes( UnoTreeListBoxImpl const & rTree, const Reference< XTreeNode >& xParentNode, UnoTreeListEntry* pParentEntry )
|
|
{
|
|
if( !(xParentNode.is() && pParentEntry) )
|
|
return;
|
|
|
|
UnoTreeListEntry* pCurrentChild = dynamic_cast< UnoTreeListEntry* >( rTree.FirstChild( pParentEntry ) );
|
|
|
|
const sal_Int32 nChildCount = xParentNode->getChildCount();
|
|
for( sal_Int32 nChild = 0; nChild < nChildCount; nChild++ )
|
|
{
|
|
Reference< XTreeNode > xNode( xParentNode->getChildAt( nChild ) );
|
|
if( !pCurrentChild || ( pCurrentChild->mxNode != xNode ) )
|
|
{
|
|
UnoTreeListEntry* pNodeEntry = getEntry( xNode, false );
|
|
if( pNodeEntry == nullptr )
|
|
{
|
|
// child node is not yet part of the tree, add it
|
|
pCurrentChild = createEntry( xNode, pParentEntry, nChild );
|
|
}
|
|
else if( pNodeEntry != pCurrentChild )
|
|
{
|
|
// node is already part of the tree, but not on the correct position
|
|
rTree.GetModel()->Move( pNodeEntry, pParentEntry, nChild );
|
|
pCurrentChild = pNodeEntry;
|
|
updateEntry( pCurrentChild );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// child node has entry and entry is equal to current entry,
|
|
// so no structural changes happened
|
|
updateEntry( pCurrentChild );
|
|
}
|
|
|
|
pCurrentChild = dynamic_cast< UnoTreeListEntry* >( pCurrentChild->NextSibling() );
|
|
}
|
|
|
|
// check if we have entries without nodes left, we need to remove them
|
|
while( pCurrentChild )
|
|
{
|
|
UnoTreeListEntry* pNextChild = dynamic_cast< UnoTreeListEntry* >( pCurrentChild->NextSibling() );
|
|
rTree.GetModel()->Remove( pCurrentChild );
|
|
pCurrentChild = pNextChild;
|
|
}
|
|
}
|
|
|
|
OUString TreeControlPeer::getEntryString( const Any& rValue )
|
|
{
|
|
OUString sValue;
|
|
if( rValue.hasValue() )
|
|
{
|
|
switch( rValue.getValueTypeClass() )
|
|
{
|
|
case TypeClass_SHORT:
|
|
case TypeClass_LONG:
|
|
{
|
|
sal_Int32 nValue = 0;
|
|
if( rValue >>= nValue )
|
|
sValue = OUString::number( nValue );
|
|
break;
|
|
}
|
|
case TypeClass_BYTE:
|
|
case TypeClass_UNSIGNED_SHORT:
|
|
case TypeClass_UNSIGNED_LONG:
|
|
{
|
|
sal_uInt32 nValue = 0;
|
|
if( rValue >>= nValue )
|
|
sValue = OUString::number( nValue );
|
|
break;
|
|
}
|
|
case TypeClass_HYPER:
|
|
{
|
|
sal_Int64 nValue = 0;
|
|
if( rValue >>= nValue )
|
|
sValue = OUString::number( nValue );
|
|
break;
|
|
}
|
|
case TypeClass_UNSIGNED_HYPER:
|
|
{
|
|
sal_uInt64 nValue = 0;
|
|
if( rValue >>= nValue )
|
|
sValue = OUString::number( nValue );
|
|
break;
|
|
}
|
|
case TypeClass_FLOAT:
|
|
case TypeClass_DOUBLE:
|
|
{
|
|
double fValue = 0.0;
|
|
if( rValue >>= fValue )
|
|
sValue = OUString::number( fValue );
|
|
break;
|
|
}
|
|
case TypeClass_STRING:
|
|
rValue >>= sValue;
|
|
break;
|
|
/*
|
|
case TypeClass_INTERFACE:
|
|
// @todo
|
|
break;
|
|
case TypeClass_SEQUENCE:
|
|
{
|
|
Sequence< Any > aValues;
|
|
if( aValue >>= aValues )
|
|
{
|
|
updateEntry( SvTreeListEntry& rEntry, aValues );
|
|
return;
|
|
}
|
|
}
|
|
break;
|
|
*/
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
return sValue;
|
|
}
|
|
|
|
// XEventListener
|
|
void SAL_CALL TreeControlPeer::disposing( const css::lang::EventObject& )
|
|
{
|
|
// model is disposed, so we clear our tree
|
|
SolarMutexGuard aGuard;
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
rTree.Clear();
|
|
mxDataModel.clear();
|
|
}
|
|
|
|
void TreeControlPeer::onChangeDataModel( UnoTreeListBoxImpl& rTree, const Reference< XTreeDataModel >& xDataModel )
|
|
{
|
|
if( xDataModel.is() && (mxDataModel == xDataModel) )
|
|
return; // do nothing
|
|
|
|
Reference< XTreeDataModelListener > xListener( this );
|
|
|
|
if( mxDataModel.is() )
|
|
mxDataModel->removeTreeDataModelListener( xListener );
|
|
|
|
mxDataModel = xDataModel;
|
|
|
|
fillTree( rTree, mxDataModel );
|
|
|
|
if( mxDataModel.is() )
|
|
mxDataModel->addTreeDataModelListener( xListener );
|
|
}
|
|
|
|
|
|
// css::awt::XLayoutConstrains
|
|
|
|
|
|
css::awt::Size TreeControlPeer::getMinimumSize()
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
css::awt::Size aSz;
|
|
/* todo
|
|
MultiLineEdit* pEdit = (MultiLineEdit*) GetWindow();
|
|
if ( pEdit )
|
|
aSz = VCLUnoHelper::ConvertToAWTSize(pEdit->CalcMinimumSize());
|
|
*/
|
|
return aSz;
|
|
}
|
|
|
|
css::awt::Size TreeControlPeer::getPreferredSize()
|
|
{
|
|
return getMinimumSize();
|
|
}
|
|
|
|
css::awt::Size TreeControlPeer::calcAdjustedSize( const css::awt::Size& rNewSize )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
css::awt::Size aSz = rNewSize;
|
|
/* todo
|
|
MultiLineEdit* pEdit = (MultiLineEdit*) GetWindow();
|
|
if ( pEdit )
|
|
aSz = VCLUnoHelper::ConvertToAWTSize(pEdit->CalcAdjustedSize(VCLUnoHelper::ConvertToVCLSize(rNewSize)));
|
|
*/
|
|
return aSz;
|
|
}
|
|
|
|
|
|
// css::awt::XVclWindowPeer
|
|
|
|
|
|
void TreeControlPeer::setProperty( const OUString& PropertyName, const Any& aValue)
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
switch( GetPropertyId( PropertyName ) )
|
|
{
|
|
case BASEPROPERTY_HIDEINACTIVESELECTION:
|
|
{
|
|
bool bEnabled = false;
|
|
if ( aValue >>= bEnabled )
|
|
{
|
|
WinBits nStyle = rTree.GetStyle();
|
|
if ( bEnabled )
|
|
nStyle |= WB_HIDESELECTION;
|
|
else
|
|
nStyle &= ~WB_HIDESELECTION;
|
|
rTree.SetStyle( nStyle );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case BASEPROPERTY_TREE_SELECTIONTYPE:
|
|
{
|
|
SelectionType eSelectionType;
|
|
if( aValue >>= eSelectionType )
|
|
{
|
|
SelectionMode eSelMode;
|
|
switch( eSelectionType )
|
|
{
|
|
case SelectionType_SINGLE: eSelMode = SelectionMode::Single; break;
|
|
case SelectionType_RANGE: eSelMode = SelectionMode::Range; break;
|
|
case SelectionType_MULTI: eSelMode = SelectionMode::Multiple; break;
|
|
// case SelectionType_NONE:
|
|
default: eSelMode = SelectionMode::NONE; break;
|
|
}
|
|
if( rTree.GetSelectionMode() != eSelMode )
|
|
rTree.SetSelectionMode( eSelMode );
|
|
}
|
|
break;
|
|
}
|
|
|
|
case BASEPROPERTY_TREE_DATAMODEL:
|
|
onChangeDataModel( rTree, Reference< XTreeDataModel >( aValue, UNO_QUERY ) );
|
|
break;
|
|
case BASEPROPERTY_ROW_HEIGHT:
|
|
{
|
|
sal_Int32 nHeight = 0;
|
|
if( aValue >>= nHeight )
|
|
rTree.SetEntryHeight( static_cast<short>(nHeight) );
|
|
break;
|
|
}
|
|
case BASEPROPERTY_TREE_EDITABLE:
|
|
{
|
|
bool bEnabled = false;
|
|
if( aValue >>= bEnabled )
|
|
rTree.EnableInplaceEditing( bEnabled );
|
|
break;
|
|
}
|
|
case BASEPROPERTY_TREE_INVOKESSTOPNODEEDITING:
|
|
break; // @todo
|
|
case BASEPROPERTY_TREE_ROOTDISPLAYED:
|
|
{
|
|
bool bDisplayed = false;
|
|
if( (aValue >>= bDisplayed) && ( bDisplayed != mbIsRootDisplayed) )
|
|
{
|
|
onChangeRootDisplayed(bDisplayed);
|
|
}
|
|
break;
|
|
}
|
|
case BASEPROPERTY_TREE_SHOWSHANDLES:
|
|
{
|
|
bool bEnabled = false;
|
|
if( aValue >>= bEnabled )
|
|
{
|
|
WinBits nBits = rTree.GetStyle() & (~WB_HASLINES);
|
|
if( bEnabled )
|
|
nBits |= WB_HASLINES;
|
|
if( nBits != rTree.GetStyle() )
|
|
rTree.SetStyle( nBits );
|
|
}
|
|
break;
|
|
}
|
|
case BASEPROPERTY_TREE_SHOWSROOTHANDLES:
|
|
{
|
|
bool bEnabled = false;
|
|
if( aValue >>= bEnabled )
|
|
{
|
|
WinBits nBits = rTree.GetStyle() & (~WB_HASLINESATROOT);
|
|
if( bEnabled )
|
|
nBits |= WB_HASLINESATROOT;
|
|
if( nBits != rTree.GetStyle() )
|
|
rTree.SetStyle( nBits );
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
VCLXWindow::setProperty( PropertyName, aValue );
|
|
break;
|
|
}
|
|
}
|
|
|
|
Any TreeControlPeer::getProperty( const OUString& PropertyName )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
const sal_uInt16 nPropId = GetPropertyId( PropertyName );
|
|
if( (nPropId >= BASEPROPERTY_TREE_START) && (nPropId <= BASEPROPERTY_TREE_END) )
|
|
{
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
switch(nPropId)
|
|
{
|
|
case BASEPROPERTY_TREE_SELECTIONTYPE:
|
|
{
|
|
SelectionType eSelectionType;
|
|
|
|
SelectionMode eSelMode = rTree.GetSelectionMode();
|
|
switch( eSelMode )
|
|
{
|
|
case SelectionMode::Single: eSelectionType = SelectionType_SINGLE; break;
|
|
case SelectionMode::Range: eSelectionType = SelectionType_RANGE; break;
|
|
case SelectionMode::Multiple:eSelectionType = SelectionType_MULTI; break;
|
|
// case SelectionMode::NONE:
|
|
default: eSelectionType = SelectionType_NONE; break;
|
|
}
|
|
return Any( eSelectionType );
|
|
}
|
|
case BASEPROPERTY_ROW_HEIGHT:
|
|
return Any( static_cast<sal_Int32>(rTree.GetEntryHeight()) );
|
|
case BASEPROPERTY_TREE_DATAMODEL:
|
|
return Any( mxDataModel );
|
|
case BASEPROPERTY_TREE_EDITABLE:
|
|
return Any( rTree.IsInplaceEditingEnabled() );
|
|
case BASEPROPERTY_TREE_INVOKESSTOPNODEEDITING:
|
|
return Any( true ); // @todo
|
|
case BASEPROPERTY_TREE_ROOTDISPLAYED:
|
|
return Any( mbIsRootDisplayed );
|
|
case BASEPROPERTY_TREE_SHOWSHANDLES:
|
|
return Any( (rTree.GetStyle() & WB_HASLINES) != 0 );
|
|
case BASEPROPERTY_TREE_SHOWSROOTHANDLES:
|
|
return Any( (rTree.GetStyle() & WB_HASLINESATROOT) != 0 );
|
|
}
|
|
}
|
|
return VCLXWindow::getProperty( PropertyName );
|
|
}
|
|
|
|
void TreeControlPeer::onChangeRootDisplayed( bool bIsRootDisplayed )
|
|
{
|
|
if( mbIsRootDisplayed == bIsRootDisplayed )
|
|
return;
|
|
|
|
mbIsRootDisplayed = bIsRootDisplayed;
|
|
|
|
UnoTreeListBoxImpl& rTree = getTreeListBoxOrThrow();
|
|
|
|
if( rTree.GetEntryCount() == 0 )
|
|
return;
|
|
|
|
// todo
|
|
fillTree( rTree, mxDataModel );
|
|
}
|
|
|
|
bool TreeControlPeer::loadImage( const OUString& rURL, Image& rImage )
|
|
{
|
|
if( !mxGraphicProvider.is() )
|
|
{
|
|
mxGraphicProvider = graphic::GraphicProvider::create(
|
|
comphelper::getProcessComponentContext());
|
|
}
|
|
|
|
try
|
|
{
|
|
css::beans::PropertyValues aProps{ comphelper::makePropertyValue(u"URL"_ustr, rURL) };
|
|
Reference< XGraphic > xGraphic( mxGraphicProvider->queryGraphic( aProps ) );
|
|
|
|
Graphic aGraphic( xGraphic );
|
|
rImage = Image(aGraphic.GetBitmapEx());
|
|
return true;
|
|
}
|
|
catch( Exception& )
|
|
{
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
|
|
|
|
UnoTreeListBoxImpl::UnoTreeListBoxImpl( TreeControlPeer* pPeer, vcl::Window* pParent, WinBits nWinStyle )
|
|
: SvTreeListBox( pParent, nWinStyle )
|
|
, mxPeer( pPeer )
|
|
{
|
|
SetStyle( WB_BORDER | WB_HASLINES |WB_HASBUTTONS | WB_HASLINESATROOT | WB_HASBUTTONSATROOT | WB_HSCROLL );
|
|
SetNodeDefaultImages();
|
|
SetSelectHdl( LINK(this, UnoTreeListBoxImpl, OnSelectionChangeHdl) );
|
|
SetDeselectHdl( LINK(this, UnoTreeListBoxImpl, OnSelectionChangeHdl) );
|
|
|
|
SetExpandingHdl( LINK(this, UnoTreeListBoxImpl, OnExpandingHdl) );
|
|
SetExpandedHdl( LINK(this, UnoTreeListBoxImpl, OnExpandedHdl) );
|
|
|
|
}
|
|
|
|
|
|
UnoTreeListBoxImpl::~UnoTreeListBoxImpl()
|
|
{
|
|
disposeOnce();
|
|
}
|
|
|
|
void UnoTreeListBoxImpl::dispose()
|
|
{
|
|
if( mxPeer.is() )
|
|
mxPeer->disposeControl();
|
|
mxPeer.clear();
|
|
SvTreeListBox::dispose();
|
|
}
|
|
|
|
|
|
IMPL_LINK_NOARG(UnoTreeListBoxImpl, OnSelectionChangeHdl, SvTreeListBox*, void)
|
|
{
|
|
if( mxPeer.is() )
|
|
mxPeer->onSelectionChanged();
|
|
}
|
|
|
|
|
|
IMPL_LINK_NOARG(UnoTreeListBoxImpl, OnExpandingHdl, SvTreeListBox*, bool)
|
|
{
|
|
UnoTreeListEntry* pEntry = dynamic_cast< UnoTreeListEntry* >( GetHdlEntry() );
|
|
|
|
if( pEntry && mxPeer.is() )
|
|
{
|
|
return mxPeer->onExpanding( pEntry->mxNode, !IsExpanded( pEntry ) );
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
IMPL_LINK_NOARG(UnoTreeListBoxImpl, OnExpandedHdl, SvTreeListBox*, void)
|
|
{
|
|
UnoTreeListEntry* pEntry = dynamic_cast< UnoTreeListEntry* >( GetHdlEntry() );
|
|
if( pEntry && mxPeer.is() )
|
|
{
|
|
mxPeer->onExpanded( pEntry->mxNode, IsExpanded( pEntry ) );
|
|
}
|
|
}
|
|
|
|
|
|
void UnoTreeListBoxImpl::insert( SvTreeListEntry* pEntry,SvTreeListEntry* pParent,sal_uInt32 nPos )
|
|
{
|
|
if( pParent )
|
|
SvTreeListBox::Insert( pEntry, pParent, nPos );
|
|
else
|
|
SvTreeListBox::Insert( pEntry, nPos );
|
|
}
|
|
|
|
|
|
void UnoTreeListBoxImpl::RequestingChildren( SvTreeListEntry* pParent )
|
|
{
|
|
UnoTreeListEntry* pEntry = dynamic_cast< UnoTreeListEntry* >( pParent );
|
|
if( pEntry && pEntry->mxNode.is() && mxPeer.is() )
|
|
mxPeer->onRequestChildNodes( pEntry->mxNode );
|
|
}
|
|
|
|
|
|
bool UnoTreeListBoxImpl::EditingEntry( SvTreeListEntry* pEntry )
|
|
{
|
|
return mxPeer.is() && mxPeer->onEditingEntry( dynamic_cast< UnoTreeListEntry* >( pEntry ) );
|
|
}
|
|
|
|
|
|
bool UnoTreeListBoxImpl::EditedEntry( SvTreeListEntry* pEntry, const OUString& rNewText )
|
|
{
|
|
return mxPeer.is() && mxPeer->onEditedEntry( dynamic_cast< UnoTreeListEntry* >( pEntry ), rNewText );
|
|
}
|
|
|
|
|
|
|
|
|
|
UnoTreeListItem::UnoTreeListItem()
|
|
: SvLBoxString(OUString())
|
|
{
|
|
}
|
|
|
|
void UnoTreeListItem::Paint(
|
|
const Point& rPos, SvTreeListBox& rDev, vcl::RenderContext& rRenderContext, const SvViewDataEntry* /*pView*/, const SvTreeListEntry& rEntry)
|
|
{
|
|
Point aPos(rPos);
|
|
Size aSize(GetWidth(&rDev, &rEntry), GetHeight(&rDev, &rEntry));
|
|
if (!!maImage)
|
|
{
|
|
rRenderContext.DrawImage(aPos, maImage, rDev.IsEnabled() ? DrawImageFlags::NONE : DrawImageFlags::Disable);
|
|
int nWidth = maImage.GetSizePixel().Width() + 6;
|
|
aPos.AdjustX(nWidth );
|
|
aSize.AdjustWidth( -nWidth );
|
|
}
|
|
rRenderContext.DrawText(tools::Rectangle(aPos,aSize),maText, rDev.IsEnabled() ? DrawTextFlags::NONE : DrawTextFlags::Disable);
|
|
}
|
|
|
|
|
|
std::unique_ptr<SvLBoxItem> UnoTreeListItem::Clone(SvLBoxItem const * pSource) const
|
|
{
|
|
std::unique_ptr<UnoTreeListItem> pNew(new UnoTreeListItem);
|
|
UnoTreeListItem const * pSourceItem = static_cast< UnoTreeListItem const * >( pSource );
|
|
pNew->maText = pSourceItem->maText;
|
|
pNew->maImage = pSourceItem->maImage;
|
|
return std::unique_ptr<SvLBoxItem>(pNew.release());
|
|
}
|
|
|
|
|
|
void UnoTreeListItem::SetImage( const Image& rImage )
|
|
{
|
|
maImage = rImage;
|
|
}
|
|
|
|
|
|
void UnoTreeListItem::SetGraphicURL( const OUString& rGraphicURL )
|
|
{
|
|
maGraphicURL = rGraphicURL;
|
|
}
|
|
|
|
|
|
void UnoTreeListItem::InitViewData( SvTreeListBox* pView,SvTreeListEntry* pEntry, SvViewDataItem* pViewData)
|
|
{
|
|
if( !pViewData )
|
|
pViewData = pView->GetViewDataItem( pEntry, this );
|
|
|
|
Size aSize(maImage.GetSizePixel());
|
|
pViewData->mnWidth = aSize.Width();
|
|
pViewData->mnHeight = aSize.Height();
|
|
|
|
const Size aTextSize(pView->GetTextWidth( maText ), pView->GetTextHeight());
|
|
if( pViewData->mnWidth )
|
|
{
|
|
pViewData->mnWidth += (6 + aTextSize.Width());
|
|
if( pViewData->mnHeight < aTextSize.Height() )
|
|
pViewData->mnHeight = aTextSize.Height();
|
|
}
|
|
else
|
|
{
|
|
pViewData->mnWidth = aTextSize.Width();
|
|
pViewData->mnHeight = aTextSize.Height();
|
|
}
|
|
}
|
|
|
|
|
|
UnoTreeListEntry::UnoTreeListEntry( const Reference< XTreeNode >& xNode, TreeControlPeer* pPeer )
|
|
: mxNode( xNode )
|
|
, mpPeer( pPeer )
|
|
{
|
|
if( mpPeer )
|
|
mpPeer->addEntry( this );
|
|
}
|
|
|
|
|
|
UnoTreeListEntry::~UnoTreeListEntry()
|
|
{
|
|
if( mpPeer )
|
|
mpPeer->removeEntry( this );
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|