LCOV - code coverage report
Current view: top level - svx/source/form - navigatortree.cxx (source / functions) Hit Total Coverage
Test: commit e02a6cb2c3e2b23b203b422e4e0680877f232636 Lines: 0 950 0.0 %
Date: 2014-04-14 Functions: 0 57 0.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include <svx/dialmgr.hxx>
      21             : #include <svx/fmshell.hxx>
      22             : #include <svx/fmmodel.hxx>
      23             : #include <svx/fmpage.hxx>
      24             : #include <svx/svdpagv.hxx>
      25             : #include "svx/svditer.hxx"
      26             : 
      27             : #include "fmhelp.hrc"
      28             : #include "fmexpl.hrc"
      29             : #include "fmexpl.hxx"
      30             : #include "svx/fmresids.hrc"
      31             : #include "fmshimp.hxx"
      32             : #include "fmservs.hxx"
      33             : #include "fmundo.hxx"
      34             : #include "fmpgeimp.hxx"
      35             : #include "fmitems.hxx"
      36             : #include "fmobj.hxx"
      37             : #include "fmprop.hrc"
      38             : #include <vcl/wrkwin.hxx>
      39             : #include <sfx2/viewsh.hxx>
      40             : #include <sfx2/dispatch.hxx>
      41             : #include <sfx2/viewfrm.hxx>
      42             : #include <comphelper/processfactory.hxx>
      43             : #include <comphelper/property.hxx>
      44             : #include <com/sun/star/form/FormComponentType.hpp>
      45             : #include <com/sun/star/sdb/CommandType.hpp>
      46             : #include <com/sun/star/beans/PropertyAttribute.hpp>
      47             : #include <com/sun/star/script/XEventAttacherManager.hpp>
      48             : #include <com/sun/star/datatransfer/clipboard/XClipboard.hpp>
      49             : #include <com/sun/star/datatransfer/XTransferable.hpp>
      50             : #include <svx/sdrpaintwindow.hxx>
      51             : 
      52             : #include <svx/svxdlg.hxx>
      53             : #include <svx/dialogs.hrc>
      54             : #include "svtools/treelistentry.hxx"
      55             : 
      56             : namespace svxform
      57             : {
      58             : 
      59             : 
      60             :     #define DROP_ACTION_TIMER_INITIAL_TICKS     10
      61             :         // solange dauert es, bis das Scrollen anspringt
      62             :     #define DROP_ACTION_TIMER_SCROLL_TICKS      3
      63             :         // in diesen Intervallen wird jeweils eine Zeile gescrollt
      64             :     #define DROP_ACTION_TIMER_TICK_BASE         10
      65             :         // das ist die Basis, mit der beide Angaben multipliziert werden (in ms)
      66             : 
      67             :     #define EXPLORER_SYNC_DELAY                 200
      68             :         // dieser Betrag an Millisekunden wird gewartet, ehe der Explorer nach einem Select oder Deselect die ::com::sun::star::sdbcx::View synchronisiert
      69             : 
      70             :     using namespace ::com::sun::star::uno;
      71             :     using namespace ::com::sun::star::lang;
      72             :     using namespace ::com::sun::star::beans;
      73             :     using namespace ::com::sun::star::form;
      74             :     using namespace ::com::sun::star::awt;
      75             :     using namespace ::com::sun::star::container;
      76             :     using namespace ::com::sun::star::script;
      77             :     using namespace ::com::sun::star::datatransfer;
      78             :     using namespace ::com::sun::star::datatransfer::clipboard;
      79             :     using namespace ::com::sun::star::sdb;
      80             : 
      81             : 
      82             :     // helper
      83             : 
      84             : 
      85             :     typedef ::std::map< Reference< XInterface >, SdrObject*, ::comphelper::OInterfaceCompare< XInterface > >
      86             :             MapModelToShape;
      87             :     typedef MapModelToShape::value_type ModelShapePair;
      88             : 
      89             : 
      90           0 :     void    collectShapeModelMapping( SdrPage* _pPage, MapModelToShape& _rMapping )
      91             :     {
      92             :         OSL_ENSURE( _pPage, "collectShapeModelMapping: invalid arg!" );
      93             : 
      94           0 :         _rMapping.clear();
      95             : 
      96           0 :         SdrObjListIter aIter( *_pPage );
      97           0 :         while ( aIter.IsMore() )
      98             :         {
      99           0 :             SdrObject* pSdrObject = aIter.Next();
     100           0 :             FmFormObj* pFormObject = FmFormObj::GetFormObject( pSdrObject );
     101           0 :             if ( !pFormObject )
     102           0 :                 continue;
     103             : 
     104           0 :             Reference< XInterface > xNormalizedModel( pFormObject->GetUnoControlModel(), UNO_QUERY );
     105             :                 // note that this is normalized (i.e. queried for XInterface explicitly)
     106             : 
     107             : #ifdef DBG_UTIL
     108             :             ::std::pair< MapModelToShape::iterator, bool > aPos =
     109             : #endif
     110           0 :             _rMapping.insert( ModelShapePair( xNormalizedModel, pSdrObject ) );
     111             :             DBG_ASSERT( aPos.second, "collectShapeModelMapping: model was already existent!" );
     112             :                 // if this asserts, this would mean we have 2 shapes pointing to the same model
     113           0 :         }
     114           0 :     }
     115             : 
     116             : 
     117             :     // class NavigatorTree
     118             : 
     119             : 
     120             : 
     121           0 :     NavigatorTree::NavigatorTree( Window* pParent )
     122             :         :SvTreeListBox( pParent, WB_HASBUTTONS|WB_HASLINES|WB_BORDER|WB_HSCROLL ) // #100258# OJ WB_HSCROLL added
     123             :         ,m_aControlExchange(this)
     124             :         ,m_pNavModel( NULL )
     125             :         ,m_pRootEntry(NULL)
     126             :         ,m_pEditEntry(NULL)
     127             :         ,nEditEvent(0)
     128             :         ,m_sdiState(SDI_DIRTY)
     129             :         ,m_aTimerTriggered(-1,-1)
     130             :         ,m_aDropActionType( DA_SCROLLUP )
     131             :         ,m_nSelectLock(0)
     132             :         ,m_nFormsSelected(0)
     133             :         ,m_nControlsSelected(0)
     134             :         ,m_nHiddenControls(0)
     135             :         ,m_aTimerCounter( DROP_ACTION_TIMER_INITIAL_TICKS )
     136             :         ,m_bDragDataDirty(sal_False)
     137             :         ,m_bPrevSelectionMixed(sal_False)
     138             :         ,m_bMarkingObjects(sal_False)
     139             :         ,m_bRootSelected(sal_False)
     140             :         ,m_bInitialUpdate(sal_True)
     141           0 :         ,m_bKeyboardCut( sal_False )
     142             :     {
     143           0 :         SetHelpId( HID_FORM_NAVIGATOR );
     144             : 
     145           0 :         m_aNavigatorImages = ImageList( SVX_RES( RID_SVXIMGLIST_FMEXPL ) );
     146             : 
     147             :         SetNodeBitmaps(
     148             :             m_aNavigatorImages.GetImage( RID_SVXIMG_COLLAPSEDNODE ),
     149             :             m_aNavigatorImages.GetImage( RID_SVXIMG_EXPANDEDNODE )
     150           0 :         );
     151             : 
     152           0 :         SetDragDropMode(0xFFFF);
     153           0 :         EnableInplaceEditing( true );
     154           0 :         SetSelectionMode(MULTIPLE_SELECTION);
     155             : 
     156           0 :         m_pNavModel = new NavigatorTreeModel( m_aNavigatorImages );
     157           0 :         Clear();
     158             : 
     159           0 :         StartListening( *m_pNavModel );
     160             : 
     161           0 :         m_aDropActionTimer.SetTimeoutHdl(LINK(this, NavigatorTree, OnDropActionTimer));
     162             : 
     163           0 :         m_aSynchronizeTimer.SetTimeoutHdl(LINK(this, NavigatorTree, OnSynchronizeTimer));
     164           0 :         SetSelectHdl(LINK(this, NavigatorTree, OnEntrySelDesel));
     165           0 :         SetDeselectHdl(LINK(this, NavigatorTree, OnEntrySelDesel));
     166           0 :     }
     167             : 
     168             : 
     169           0 :     NavigatorTree::~NavigatorTree()
     170             :     {
     171           0 :         if( nEditEvent )
     172           0 :             Application::RemoveUserEvent( nEditEvent );
     173             : 
     174           0 :         if (m_aSynchronizeTimer.IsActive())
     175           0 :             m_aSynchronizeTimer.Stop();
     176             : 
     177             :         DBG_ASSERT(GetNavModel() != NULL, "NavigatorTree::~NavigatorTree : unerwartet : kein ExplorerModel");
     178           0 :         EndListening( *m_pNavModel );
     179           0 :         Clear();
     180           0 :         delete m_pNavModel;
     181           0 :     }
     182             : 
     183             : 
     184           0 :     void NavigatorTree::Clear()
     185             :     {
     186           0 :         m_pNavModel->Clear();
     187           0 :     }
     188             : 
     189             : 
     190           0 :     void NavigatorTree::UpdateContent( FmFormShell* pFormShell )
     191             :     {
     192           0 :         if (m_bInitialUpdate)
     193             :         {
     194           0 :             GrabFocus();
     195           0 :             m_bInitialUpdate = sal_False;
     196             :         }
     197             : 
     198           0 :         FmFormShell* pOldShell = GetNavModel()->GetFormShell();
     199           0 :         FmFormPage* pOldPage = GetNavModel()->GetFormPage();
     200           0 :         FmFormPage* pNewPage = pFormShell ? pFormShell->GetCurPage() : NULL;
     201             : 
     202           0 :         if ((pOldShell != pFormShell) || (pOldPage != pNewPage))
     203             :         {
     204             :             // neue Shell, waehrend ich gerade editiere ?
     205           0 :             if (IsEditingActive())
     206           0 :                 CancelTextEditing();
     207             : 
     208           0 :             m_bDragDataDirty = sal_True;    // sicherheitshalber, auch wenn ich gar nicht dragge
     209             :         }
     210           0 :         GetNavModel()->UpdateContent( pFormShell );
     211             : 
     212             :         // wenn es eine Form gibt, die Root expandieren
     213           0 :         if (m_pRootEntry && !IsExpanded(m_pRootEntry))
     214           0 :             Expand(m_pRootEntry);
     215             :         // wenn es GENAU eine Form gibt, auch diese expandieren
     216           0 :         if (m_pRootEntry)
     217             :         {
     218           0 :             SvTreeListEntry* pFirst = FirstChild(m_pRootEntry);
     219           0 :             if (pFirst && !NextSibling(pFirst))
     220           0 :                 Expand(pFirst);
     221             :         }
     222           0 :     }
     223             : 
     224             : 
     225           0 :     sal_Bool NavigatorTree::implAllowExchange( sal_Int8 _nAction, sal_Bool* _pHasNonHidden )
     226             :     {
     227           0 :         SvTreeListEntry* pCurEntry = GetCurEntry();
     228           0 :         if (!pCurEntry)
     229           0 :             return sal_False;
     230             : 
     231             :         // die Informationen fuer das AcceptDrop und ExecuteDrop
     232           0 :         CollectSelectionData(SDI_ALL);
     233           0 :         if (m_arrCurrentSelection.empty())
     234             :             // nothing to do
     235           0 :             return sal_False;
     236             : 
     237             :         // testen, ob es sich vielleicht ausschliesslich um hidden controls handelt (dann koennte ich pCtrlExch noch ein
     238             :         // zusaetzliches Format geben)
     239           0 :         sal_Bool bHasNonHidden = sal_False;
     240           0 :         for ( SvLBoxEntrySortedArray::const_iterator it = m_arrCurrentSelection.begin();
     241           0 :               it != m_arrCurrentSelection.end(); ++it )
     242             :         {
     243           0 :             FmEntryData* pCurrent = static_cast< FmEntryData* >( (*it)->GetUserData() );
     244           0 :             if ( IsHiddenControl( pCurrent ) )
     245           0 :                 continue;
     246           0 :             bHasNonHidden = sal_True;
     247           0 :             break;
     248             :         }
     249             : 
     250           0 :         if ( bHasNonHidden && ( 0 == ( _nAction & DND_ACTION_MOVE ) ) )
     251             :             // non-hidden controls need to be moved
     252           0 :             return sal_False;
     253             : 
     254           0 :         if ( _pHasNonHidden )
     255           0 :             *_pHasNonHidden = bHasNonHidden;
     256             : 
     257           0 :         return sal_True;
     258             :     }
     259             : 
     260             : 
     261           0 :     sal_Bool NavigatorTree::implPrepareExchange( sal_Int8 _nAction )
     262             :     {
     263           0 :         EndSelection();
     264             : 
     265           0 :         sal_Bool bHasNonHidden = sal_False;
     266           0 :         if ( !implAllowExchange( _nAction, &bHasNonHidden ) )
     267           0 :             return sal_False;
     268             : 
     269           0 :         m_aControlExchange.prepareDrag();
     270           0 :         m_aControlExchange->setFocusEntry( GetCurEntry() );
     271             : 
     272           0 :         for ( SvLBoxEntrySortedArray::const_iterator it = m_arrCurrentSelection.begin();
     273           0 :               it != m_arrCurrentSelection.end(); ++it )
     274           0 :             m_aControlExchange->addSelectedEntry(*it);
     275             : 
     276           0 :         m_aControlExchange->setFormsRoot( GetNavModel()->GetFormPage()->GetForms() );
     277           0 :         m_aControlExchange->buildPathFormat( this, m_pRootEntry );
     278             : 
     279           0 :         if (!bHasNonHidden)
     280             :         {
     281             :             // eine entsprechende Sequenz aufbauen
     282           0 :             Sequence< Reference< XInterface > > seqIFaces(m_arrCurrentSelection.size());
     283           0 :             Reference< XInterface >* pArray = seqIFaces.getArray();
     284           0 :             for ( SvLBoxEntrySortedArray::const_iterator it = m_arrCurrentSelection.begin();
     285           0 :                   it != m_arrCurrentSelection.end(); ++it, ++pArray )
     286           0 :                 *pArray = static_cast< FmEntryData* >( (*it)->GetUserData() )->GetElement();
     287             : 
     288             :             // und das neue Format
     289           0 :             m_aControlExchange->addHiddenControlsFormat(seqIFaces);
     290             :         }
     291             : 
     292           0 :         m_bDragDataDirty = sal_False;
     293           0 :         return sal_True;
     294             :     }
     295             : 
     296             : 
     297           0 :     void NavigatorTree::StartDrag( sal_Int8 /*nAction*/, const ::Point& /*rPosPixel*/ )
     298             :     {
     299           0 :         EndSelection();
     300             : 
     301           0 :         if ( !implPrepareExchange( DND_ACTION_COPYMOVE ) )
     302             :             // nothing to do or something went wrong
     303           0 :             return;
     304             : 
     305             :         // jetzt haben wir alle in der aktuelle Situation moeglichen Formate eingesammelt, es kann also losgehen ...
     306           0 :         m_aControlExchange.startDrag( DND_ACTION_COPYMOVE );
     307             :     }
     308             : 
     309             : 
     310           0 :     void NavigatorTree::Command( const CommandEvent& rEvt )
     311             :     {
     312           0 :         sal_Bool bHandled = sal_False;
     313           0 :         switch( rEvt.GetCommand() )
     314             :         {
     315             :             case COMMAND_CONTEXTMENU:
     316             :             {
     317             :                 // die Stelle, an der geklickt wurde
     318           0 :                 ::Point ptWhere;
     319           0 :                 if (rEvt.IsMouseEvent())
     320             :                 {
     321           0 :                     ptWhere = rEvt.GetMousePosPixel();
     322           0 :                     SvTreeListEntry* ptClickedOn = GetEntry(ptWhere);
     323           0 :                     if (ptClickedOn == NULL)
     324           0 :                         break;
     325           0 :                     if ( !IsSelected(ptClickedOn) )
     326             :                     {
     327           0 :                         SelectAll(false);
     328           0 :                         Select(ptClickedOn, sal_True);
     329           0 :                         SetCurEntry(ptClickedOn);
     330             :                     }
     331             :                 }
     332             :                 else
     333             :                 {
     334           0 :                     if (m_arrCurrentSelection.empty()) // kann nur bei Kontextmenue ueber Tastatur passieren
     335           0 :                         break;
     336             : 
     337           0 :                     SvTreeListEntry* pCurrent = GetCurEntry();
     338           0 :                     if (!pCurrent)
     339           0 :                         break;
     340           0 :                     ptWhere = GetEntryPosition(pCurrent);
     341             :                 }
     342             : 
     343             :                 // meine Selektionsdaten auf den aktuellen Stand
     344           0 :                 CollectSelectionData(SDI_ALL);
     345             : 
     346             :                 // wenn mindestens ein Nicht-Root-Eintrag selektiert ist und die Root auch, dann nehme ich letztere aus der Selektion
     347             :                 // fix wieder raus
     348           0 :                 if ( (m_arrCurrentSelection.size() > 1) && m_bRootSelected )
     349             :                 {
     350           0 :                     Select( m_pRootEntry, sal_False );
     351           0 :                     SetCursor( *m_arrCurrentSelection.begin(), true);
     352             :                 }
     353           0 :                 sal_Bool bSingleSelection = (m_arrCurrentSelection.size() == 1);
     354             : 
     355             : 
     356             :                 DBG_ASSERT( (m_arrCurrentSelection.size() > 0) || m_bRootSelected, "keine Eintraege selektiert" );
     357             :                     // solte nicht passieren, da ich oben bei der IsSelected-Abfrage auf jeden Fall einen selektiert haette,
     358             :                     // wenn das vorher nicht der Fall gewesen waere
     359             : 
     360             : 
     361             :                 // das Menue zusammenbasteln
     362           0 :                 FmFormShell* pFormShell = GetNavModel()->GetFormShell();
     363           0 :                 FmFormModel* pFormModel = pFormShell ? pFormShell->GetFormModel() : NULL;
     364           0 :                 if( pFormShell && pFormModel )
     365             :                 {
     366           0 :                     PopupMenu aContextMenu(SVX_RES(RID_FMEXPLORER_POPUPMENU));
     367           0 :                     PopupMenu* pSubMenuNew = aContextMenu.GetPopupMenu( SID_FM_NEW );
     368             : 
     369             :                     // das 'Neu'-Untermenue gibt es nur, wenn genau die Root oder genau ein Formular selektiert sind
     370           0 :                     aContextMenu.EnableItem( SID_FM_NEW, bSingleSelection && (m_nFormsSelected || m_bRootSelected) );
     371             : 
     372             :                     // 'Neu'\'Formular' unter genau den selben Bedingungen
     373           0 :                     pSubMenuNew->EnableItem( SID_FM_NEW_FORM, bSingleSelection && (m_nFormsSelected || m_bRootSelected) );
     374           0 :                     pSubMenuNew->SetItemImage(SID_FM_NEW_FORM, m_aNavigatorImages.GetImage(RID_SVXIMG_FORM));
     375           0 :                     pSubMenuNew->SetItemImage(SID_FM_NEW_HIDDEN, m_aNavigatorImages.GetImage(RID_SVXIMG_HIDDEN));
     376             : 
     377             :                     // 'Neu'\'verstecktes...', wenn genau ein Formular selektiert ist
     378           0 :                     pSubMenuNew->EnableItem( SID_FM_NEW_HIDDEN, bSingleSelection && m_nFormsSelected );
     379             : 
     380             :                     // 'Delete': everything which is not root can be removed
     381           0 :                     aContextMenu.EnableItem( SID_FM_DELETE, !m_bRootSelected );
     382             : 
     383             :                     // 'Cut', 'Copy' and 'Paste'
     384           0 :                     aContextMenu.EnableItem( SID_CUT, !m_bRootSelected && implAllowExchange( DND_ACTION_MOVE ) );
     385           0 :                     aContextMenu.EnableItem( SID_COPY, !m_bRootSelected && implAllowExchange( DND_ACTION_COPY ) );
     386           0 :                     aContextMenu.EnableItem( SID_PASTE, implAcceptPaste( ) );
     387             : 
     388             :                     // der TabDialog, wenn es genau ein Formular ist ...
     389           0 :                     aContextMenu.EnableItem( SID_FM_TAB_DIALOG, bSingleSelection && m_nFormsSelected );
     390             : 
     391             :                     // in XML forms, we don't allow for the properties of a form
     392             :                     // #i36484#
     393           0 :                     if ( pFormShell->GetImpl()->isEnhancedForm() && !m_nControlsSelected )
     394           0 :                         aContextMenu.RemoveItem( aContextMenu.GetItemPos( SID_FM_SHOW_PROPERTY_BROWSER ) );
     395             : 
     396             :                     // if the property browser is already open, we don't allow for the properties, too
     397           0 :                     if( pFormShell->GetImpl()->IsPropBrwOpen() )
     398           0 :                         aContextMenu.RemoveItem( aContextMenu.GetItemPos( SID_FM_SHOW_PROPERTY_BROWSER ) );
     399             :                     // and finally, if there's a mixed selection of forms and controls, disable the entry, too
     400             :                     else
     401             :                         aContextMenu.EnableItem( SID_FM_SHOW_PROPERTY_BROWSER,
     402           0 :                             (m_nControlsSelected && !m_nFormsSelected) || (!m_nControlsSelected && m_nFormsSelected) );
     403             : 
     404             :                     // Umbenennen gdw wenn ein Element und nicht die Root
     405           0 :                     aContextMenu.EnableItem( SID_FM_RENAME_OBJECT, bSingleSelection && !m_bRootSelected );
     406             : 
     407             :                     // der Reandonly-Eintrag ist nur auf der Root erlaubt
     408           0 :                     aContextMenu.EnableItem( SID_FM_OPEN_READONLY, m_bRootSelected );
     409             :                     // the same for automatic control focus
     410           0 :                     aContextMenu.EnableItem( SID_FM_AUTOCONTROLFOCUS, m_bRootSelected );
     411             : 
     412             :                     // die ConvertTo-Slots sind enabled, wenn genau ein Control selektiert ist, der
     413             :                     // dem Control entsprechende Slot ist disabled
     414           0 :                     if (!m_bRootSelected && !m_nFormsSelected && (m_nControlsSelected == 1))
     415             :                     {
     416           0 :                         aContextMenu.SetPopupMenu( SID_FM_CHANGECONTROLTYPE, FmXFormShell::GetConversionMenu() );
     417             : #if OSL_DEBUG_LEVEL > 0
     418             :                         FmControlData* pCurrent = (FmControlData*)(*m_arrCurrentSelection.begin())->GetUserData();
     419             :                         OSL_ENSURE( pFormShell->GetImpl()->isSolelySelected( pCurrent->GetFormComponent() ),
     420             :                             "NavigatorTree::Command: inconsistency between the navigator selection, and the selection as the shell knows it!" );
     421             : #endif
     422             : 
     423           0 :                         pFormShell->GetImpl()->checkControlConversionSlotsForCurrentSelection( *aContextMenu.GetPopupMenu( SID_FM_CHANGECONTROLTYPE ) );
     424             :                     }
     425             :                     else
     426           0 :                         aContextMenu.EnableItem( SID_FM_CHANGECONTROLTYPE, false );
     427             : 
     428             :                     // jetzt alles, was disabled wurde, wech
     429           0 :                     aContextMenu.RemoveDisabledEntries(true, true);
     430             : 
     431             :                     // OpenReadOnly setzen
     432             : 
     433           0 :                     aContextMenu.CheckItem( SID_FM_OPEN_READONLY, pFormModel->GetOpenInDesignMode() );
     434           0 :                     aContextMenu.CheckItem( SID_FM_AUTOCONTROLFOCUS, pFormModel->GetAutoControlFocus() );
     435             : 
     436           0 :                     sal_uInt16 nSlotId = aContextMenu.Execute( this, ptWhere );
     437           0 :                     switch( nSlotId )
     438             :                     {
     439             :                         case SID_FM_NEW_FORM:
     440             :                         {
     441           0 :                             OUString aStr(SVX_RESSTR(RID_STR_FORM));
     442           0 :                             OUString aUndoStr = SVX_RESSTR(RID_STR_UNDO_CONTAINER_INSERT).replaceAll(OUString('#'), aStr);
     443             : 
     444           0 :                             pFormModel->BegUndo(aUndoStr);
     445             :                             // der Slot war nur verfuegbar, wenn es genau einen selektierten Eintrag gibt und dieser die Root
     446             :                             // oder ein Formular ist
     447           0 :                             NewForm( *m_arrCurrentSelection.begin() );
     448           0 :                             pFormModel->EndUndo();
     449             : 
     450           0 :                         }   break;
     451             :                         case SID_FM_NEW_HIDDEN:
     452             :                         {
     453           0 :                             OUString aStr(SVX_RESSTR(RID_STR_CONTROL));
     454           0 :                             OUString aUndoStr = SVX_RESSTR(RID_STR_UNDO_CONTAINER_INSERT).replaceAll(OUString('#'), aStr);
     455             : 
     456           0 :                             pFormModel->BegUndo(aUndoStr);
     457             :                             // dieser Slot war guletig bei (genau) einem selektierten Formular
     458           0 :                             OUString fControlName = FM_COMPONENT_HIDDEN;
     459           0 :                             NewControl( fControlName, *m_arrCurrentSelection.begin() );
     460           0 :                             pFormModel->EndUndo();
     461             : 
     462           0 :                         }   break;
     463             : 
     464             :                         case SID_CUT:
     465           0 :                             doCut();
     466           0 :                             break;
     467             : 
     468             :                         case SID_COPY:
     469           0 :                             doCopy();
     470           0 :                             break;
     471             : 
     472             :                         case SID_PASTE:
     473           0 :                             doPaste();
     474           0 :                             break;
     475             : 
     476             :                         case SID_FM_DELETE:
     477             :                         {
     478           0 :                             DeleteSelection();
     479             :                         }
     480           0 :                         break;
     481             :                         case SID_FM_TAB_DIALOG:
     482             :                         {
     483             :                             // dieser Slot galt bei genau einem selektierten Formular
     484           0 :                             SvTreeListEntry* pSelectedForm = *m_arrCurrentSelection.begin();
     485             :                             DBG_ASSERT( IsFormEntry(pSelectedForm), "NavigatorTree::Command: Dieser Eintrag muss ein FormEntry sein." );
     486             : 
     487           0 :                             FmFormData* pFormData = (FmFormData*)pSelectedForm->GetUserData();
     488           0 :                             Reference< XForm >  xForm(  pFormData->GetFormIface());
     489             : 
     490           0 :                             Reference< XTabControllerModel >  xTabController(xForm, UNO_QUERY);
     491           0 :                             if( !xTabController.is() )
     492           0 :                                 break;
     493           0 :                             GetNavModel()->GetFormShell()->GetImpl()->ExecuteTabOrderDialog( xTabController );
     494             :                         }
     495           0 :                         break;
     496             : 
     497             :                         case SID_FM_SHOW_PROPERTY_BROWSER:
     498             :                         {
     499           0 :                             ShowSelectionProperties(sal_True);
     500             :                         }
     501           0 :                         break;
     502             :                         case SID_FM_RENAME_OBJECT:
     503             :                         {
     504             :                             // das war bei genau einem Nicht-Root-Eintrag erlaubt
     505           0 :                             EditEntry( *m_arrCurrentSelection.begin() );
     506             :                         }
     507           0 :                         break;
     508             :                         case SID_FM_OPEN_READONLY:
     509             :                         {
     510           0 :                             pFormModel->SetOpenInDesignMode( !pFormModel->GetOpenInDesignMode() );
     511           0 :                             pFormShell->GetViewShell()->GetViewFrame()->GetBindings().Invalidate(SID_FM_OPEN_READONLY);
     512             :                         }
     513           0 :                         break;
     514             :                         case SID_FM_AUTOCONTROLFOCUS:
     515             :                         {
     516           0 :                             pFormModel->SetAutoControlFocus( !pFormModel->GetAutoControlFocus() );
     517           0 :                             pFormShell->GetViewShell()->GetViewFrame()->GetBindings().Invalidate(SID_FM_AUTOCONTROLFOCUS);
     518             :                         }
     519           0 :                         break;
     520             :                         default:
     521           0 :                             if (pFormShell->GetImpl()->isControlConversionSlot(nSlotId))
     522             :                             {
     523           0 :                                 FmControlData* pCurrent = (FmControlData*)(*m_arrCurrentSelection.begin())->GetUserData();
     524           0 :                                 if ( pFormShell->GetImpl()->executeControlConversionSlot( pCurrent->GetFormComponent(), nSlotId ) )
     525           0 :                                     ShowSelectionProperties();
     526             :                             }
     527           0 :                     }
     528             :                 }
     529           0 :                 bHandled = sal_True;
     530           0 :             } break;
     531             :         }
     532             : 
     533           0 :         if (!bHandled)
     534           0 :             SvTreeListBox::Command( rEvt );
     535           0 :     }
     536             : 
     537             : 
     538           0 :     SvTreeListEntry* NavigatorTree::FindEntry( FmEntryData* pEntryData )
     539             :     {
     540           0 :         if( !pEntryData ) return NULL;
     541           0 :         SvTreeListEntry* pCurEntry = First();
     542             :         FmEntryData* pCurEntryData;
     543           0 :         while( pCurEntry )
     544             :         {
     545           0 :             pCurEntryData = (FmEntryData*)pCurEntry->GetUserData();
     546           0 :             if( pCurEntryData && pCurEntryData->IsEqualWithoutChildren(pEntryData) )
     547           0 :                 return pCurEntry;
     548             : 
     549           0 :             pCurEntry = Next( pCurEntry );
     550             :         }
     551             : 
     552           0 :         return NULL;
     553             :     }
     554             : 
     555             : 
     556           0 :     void NavigatorTree::Notify( SfxBroadcaster& /*rBC*/, const SfxHint& rHint )
     557             :     {
     558           0 :         if( rHint.ISA(FmNavRemovedHint) )
     559             :         {
     560           0 :             FmNavRemovedHint* pRemovedHint = (FmNavRemovedHint*)&rHint;
     561           0 :             FmEntryData* pEntryData = pRemovedHint->GetEntryData();
     562           0 :             Remove( pEntryData );
     563             :         }
     564             : 
     565           0 :         else if( rHint.ISA(FmNavInsertedHint) )
     566             :         {
     567           0 :             FmNavInsertedHint* pInsertedHint = (FmNavInsertedHint*)&rHint;
     568           0 :             FmEntryData* pEntryData = pInsertedHint->GetEntryData();
     569           0 :             sal_uInt32 nRelPos = pInsertedHint->GetRelPos();
     570           0 :             Insert( pEntryData, nRelPos );
     571             :         }
     572             : 
     573           0 :         else if( rHint.ISA(FmNavModelReplacedHint) )
     574             :         {
     575           0 :             FmEntryData* pData = ((FmNavModelReplacedHint*)&rHint)->GetEntryData();
     576           0 :             SvTreeListEntry* pEntry = FindEntry( pData );
     577           0 :             if (pEntry)
     578             :             {   // das Image neu setzen
     579           0 :                 SetCollapsedEntryBmp( pEntry, pData->GetNormalImage() );
     580           0 :                 SetExpandedEntryBmp( pEntry, pData->GetNormalImage() );
     581             :             }
     582             :         }
     583             : 
     584           0 :         else if( rHint.ISA(FmNavNameChangedHint) )
     585             :         {
     586           0 :             FmNavNameChangedHint* pNameChangedHint = (FmNavNameChangedHint*)&rHint;
     587           0 :             SvTreeListEntry* pEntry = FindEntry( pNameChangedHint->GetEntryData() );
     588           0 :             SetEntryText( pEntry, pNameChangedHint->GetNewName() );
     589             :         }
     590             : 
     591           0 :         else if( rHint.ISA(FmNavClearedHint) )
     592             :         {
     593           0 :             SvTreeListBox::Clear();
     594             : 
     595             : 
     596             :             // Default-Eintrag "Formulare"
     597           0 :             Image aRootImage( m_aNavigatorImages.GetImage( RID_SVXIMG_FORMS ) );
     598           0 :             m_pRootEntry = InsertEntry( SVX_RESSTR(RID_STR_FORMS), aRootImage, aRootImage,
     599           0 :                 NULL, false, 0, NULL );
     600             :         }
     601           0 :         else if (!m_bMarkingObjects && rHint.ISA(FmNavRequestSelectHint))
     602             :         {   // wenn m_bMarkingObjects sal_True ist, markiere ich gerade selber Objekte, und da der ganze Mechanismus dahinter synchron ist,
     603             :             // ist das genau der Hint, der durch mein Markieren ausgeloest wird, also kann ich ihn ignorieren
     604           0 :             FmNavRequestSelectHint* pershHint = (FmNavRequestSelectHint*)&rHint;
     605           0 :             FmEntryDataArray& arredToSelect = pershHint->GetItems();
     606           0 :             SynchronizeSelection(arredToSelect);
     607             : 
     608           0 :             if (pershHint->IsMixedSelection())
     609             :                 // in diesem Fall habe ich alles deselektiert, obwohl die View u.U. eine gemischte Markierung hatte
     610             :                 // ich muss also im naechsten Select den Navigator an die View anpassen
     611           0 :                 m_bPrevSelectionMixed = sal_True;
     612             :         }
     613           0 :     }
     614             : 
     615             : 
     616           0 :     SvTreeListEntry* NavigatorTree::Insert( FmEntryData* pEntryData, sal_uIntPtr nRelPos )
     617             :     {
     618             : 
     619             :         // Aktuellen Eintrag einfuegen
     620           0 :         SvTreeListEntry* pParentEntry = FindEntry( pEntryData->GetParent() );
     621             :         SvTreeListEntry* pNewEntry;
     622             : 
     623           0 :         if( !pParentEntry )
     624             :             pNewEntry = InsertEntry( pEntryData->GetText(),
     625           0 :                 pEntryData->GetNormalImage(), pEntryData->GetNormalImage(),
     626           0 :                 m_pRootEntry, false, nRelPos, pEntryData );
     627             : 
     628             :         else
     629             :             pNewEntry = InsertEntry( pEntryData->GetText(),
     630           0 :                 pEntryData->GetNormalImage(), pEntryData->GetNormalImage(),
     631           0 :                 pParentEntry, false, nRelPos, pEntryData );
     632             : 
     633             : 
     634             :         // Wenn Root-Eintrag Root expandieren
     635           0 :         if( !pParentEntry )
     636           0 :             Expand( m_pRootEntry );
     637             : 
     638             : 
     639             :         // Children einfuegen
     640           0 :         FmEntryDataList* pChildList = pEntryData->GetChildList();
     641           0 :         size_t nChildCount = pChildList->size();
     642             :         FmEntryData* pChildData;
     643           0 :         for( size_t i = 0; i < nChildCount; i++ )
     644             :         {
     645           0 :             pChildData = pChildList->at( i );
     646           0 :             Insert( pChildData, TREELIST_APPEND );
     647             :         }
     648             : 
     649           0 :         return pNewEntry;
     650             :     }
     651             : 
     652             : 
     653           0 :     void NavigatorTree::Remove( FmEntryData* pEntryData )
     654             :     {
     655           0 :         if( !pEntryData )
     656           0 :             return;
     657             : 
     658             :         // der Entry zu den Daten
     659           0 :         SvTreeListEntry* pEntry = FindEntry( pEntryData );
     660           0 :         if (!pEntry)
     661           0 :             return;
     662             : 
     663             :         // Eintrag aus TreeListBox entfernen
     664             :         // ich darf das Select, das ich ausloese, nicht behandeln :
     665             :         // Select aendert die MarkList der View, wenn das gerade auch jemand anders macht und dabei ein Remove
     666             :         // triggert, haben wir mit ziemlicher Sicherheit ein Problem - Paradebeispiel war das Gruppieren von Controls mit
     667             :         // offenem Navigator ...)
     668           0 :         LockSelectionHandling();
     669             : 
     670             :         // ein kleines Problem : ich merke mir meine selektierten Daten, wenn mir jetzt jemand einen selektierten Eintrag
     671             :         // unter dem Hintern wegschiesst, werde ich inkonsistent ... was schlecht waere
     672           0 :         Select(pEntry, sal_False);
     673             : 
     674             :         // beim eigentlichen Entfernen kann die Selection geaendert werden, da ich aber das SelectionHandling abgeschaltet
     675             :         // habe, muss ich mich hinterher darum kuemmern
     676           0 :         sal_uIntPtr nExpectedSelectionCount = GetSelectionCount();
     677             : 
     678           0 :         if( pEntry )
     679           0 :             GetModel()->Remove( pEntry );
     680             : 
     681           0 :         if (nExpectedSelectionCount != GetSelectionCount())
     682           0 :             SynchronizeSelection();
     683             : 
     684             :         // und standardmaessig behandle ich das Select natuerlich
     685           0 :         UnlockSelectionHandling();
     686             :     }
     687             : 
     688             : 
     689           0 :     sal_Bool NavigatorTree::IsFormEntry( SvTreeListEntry* pEntry )
     690             :     {
     691           0 :         FmEntryData* pEntryData = (FmEntryData*)pEntry->GetUserData();
     692           0 :         return !pEntryData || pEntryData->ISA(FmFormData);
     693             :     }
     694             : 
     695             : 
     696           0 :     sal_Bool NavigatorTree::IsFormComponentEntry( SvTreeListEntry* pEntry )
     697             :     {
     698           0 :         FmEntryData* pEntryData = (FmEntryData*)pEntry->GetUserData();
     699           0 :         return pEntryData && pEntryData->ISA(FmControlData);
     700             :     }
     701             : 
     702             : 
     703           0 :     sal_Bool NavigatorTree::implAcceptPaste( )
     704             :     {
     705           0 :         SvTreeListEntry* pFirstSelected = FirstSelected();
     706           0 :         if ( !pFirstSelected || NextSelected( pFirstSelected ) )
     707             :             // no selected entry, or at least two selected entries
     708           0 :             return sal_False;
     709             : 
     710             :         // get the clipboard
     711           0 :         TransferableDataHelper aClipboardContent( TransferableDataHelper::CreateFromSystemClipboard( this ) );
     712             : 
     713           0 :         sal_Int8 nAction = m_aControlExchange.isClipboardOwner() && doingKeyboardCut( ) ? DND_ACTION_MOVE : DND_ACTION_COPY;
     714           0 :         return ( nAction == implAcceptDataTransfer( aClipboardContent.GetDataFlavorExVector(), nAction, pFirstSelected, sal_False ) );
     715             :     }
     716             : 
     717             : 
     718           0 :     sal_Int8 NavigatorTree::implAcceptDataTransfer( const DataFlavorExVector& _rFlavors, sal_Int8 _nAction, const ::Point& _rDropPos, sal_Bool _bDnD )
     719             :     {
     720           0 :         return implAcceptDataTransfer( _rFlavors, _nAction, GetEntry( _rDropPos ), _bDnD );
     721             :     }
     722             : 
     723             : 
     724           0 :     sal_Int8 NavigatorTree::implAcceptDataTransfer( const DataFlavorExVector& _rFlavors, sal_Int8 _nAction, SvTreeListEntry* _pTargetEntry, sal_Bool _bDnD )
     725             :     {
     726             :         // no target -> no drop
     727           0 :         if (!_pTargetEntry)
     728           0 :             return DND_ACTION_NONE;
     729             : 
     730             :         // format check
     731           0 :         sal_Bool bHasDefControlFormat = OControlExchange::hasFieldExchangeFormat( _rFlavors );
     732           0 :         sal_Bool bHasControlPathFormat = OControlExchange::hasControlPathFormat( _rFlavors );
     733           0 :         sal_Bool bHasHiddenControlsFormat = OControlExchange::hasHiddenControlModelsFormat( _rFlavors );
     734           0 :         if (!bHasDefControlFormat && !bHasControlPathFormat && !bHasHiddenControlsFormat)
     735           0 :             return DND_ACTION_NONE;
     736             : 
     737           0 :         sal_Bool bSelfSource = _bDnD ? m_aControlExchange.isDragSource() : m_aControlExchange.isClipboardOwner();
     738             : 
     739           0 :         if ( bHasHiddenControlsFormat )
     740             :         {   // bHasHiddenControlsFormat means that only hidden controls are part of the data
     741             : 
     742             :             // hidden controls can be copied to a form only
     743           0 :             if ( !_pTargetEntry || ( _pTargetEntry == m_pRootEntry ) || !IsFormEntry( _pTargetEntry ) )
     744           0 :                 return DND_ACTION_NONE;
     745             : 
     746           0 :             return bSelfSource ? ( DND_ACTION_COPYMOVE & _nAction ) : DND_ACTION_COPY;
     747             :         }
     748             : 
     749           0 :         if  ( !bSelfSource )
     750             :         {
     751             :             // DnD or CnP crossing navigator boundaries
     752             :             // The main problem here is that the current API does not allow us to sneak into the content which
     753             :             // is to be inserted. So we have to allow it for the moment, but maybe reject later on (in the real drop).
     754             : 
     755             :             // TODO: this smart behaviour later on ... at the moment, we disallow data transfer crossing navigator
     756             :             // boundaries.
     757             : 
     758           0 :             return DND_ACTION_NONE;
     759             :         }
     760             : 
     761             :         DBG_ASSERT( _bDnD ? m_aControlExchange.isDragSource() : m_aControlExchange.isClipboardOwner(),
     762             :             "NavigatorTree::implAcceptDataTransfer: here only with source=dest!" );
     763             :             // somebody changed the logic of this method ...
     764             : 
     765             :         // from here on, I can work with m_aControlExchange instead of _rData!
     766             : 
     767           0 :         sal_Bool bForeignCollection = m_aControlExchange->getFormsRoot().get() != GetNavModel()->GetFormPage()->GetForms().get();
     768           0 :         if ( bForeignCollection )
     769             :         {
     770             :             // crossing shell/page boundaries, we can exchange hidden controls only
     771             :             // But if we survived the checks above, we do not have hidden controls.
     772             :             // -> no data transfer
     773             :             DBG_ASSERT( !bHasHiddenControlsFormat, "NavigatorTree::implAcceptDataTransfer: still hidden controls format!" );
     774             :                 // somebody changed the logic of this method ...
     775             : 
     776           0 :             return DND_ACTION_COPY;
     777             :         }
     778             : 
     779           0 :         if (DND_ACTION_MOVE != _nAction) // 'normal' controls within a shell are moved only (never copied)
     780           0 :             return DND_ACTION_NONE;
     781             : 
     782           0 :         if ( m_bDragDataDirty || !bHasDefControlFormat )
     783             :         {
     784           0 :             if (!bHasControlPathFormat)
     785             :                 // ich befinde mich zwar in der Shell/Page, aus der die Controls stammen, habe aber kein Format, das den stattgefundenen
     786             :                 // Shell-Wechsel ueberlebt hat (SVX_FM_CONTROLS_AS_PATH)
     787           0 :                 return DND_ACTION_NONE;
     788             : 
     789             :             // da die Shell waehrend des Draggens umgeschaltet wude, muss ich die Liste des ExchangeObjektes wieder neu aufbauen
     790             :             // (dort stehen SvLBoxEntries drin, und die sind bei der Umschaltung floeten gegangen)
     791           0 :             m_aControlExchange->buildListFromPath(this, m_pRootEntry);
     792           0 :             m_bDragDataDirty = sal_False;
     793             :         }
     794             : 
     795             :         // die Liste der gedroppten Eintraege aus dem DragServer
     796           0 :         const ListBoxEntrySet& aDropped = m_aControlExchange->selected();
     797             :         DBG_ASSERT(aDropped.size() >= 1, "NavigatorTree::implAcceptDataTransfer: keine Eintraege !");
     798             : 
     799           0 :         sal_Bool bDropTargetIsComponent = IsFormComponentEntry( _pTargetEntry );
     800             :         //SvTreeListEntry* pDropTargetParent = GetParent( _pTargetEntry );
     801             : 
     802             :         // conditions to disallow the drop
     803             :         // 0) the root entry is part of the list (can't DnD the root!)
     804             :         // 1) one of the draged entries is to be dropped onto it's own parent
     805             :         // 2) -               "       - is to be dropped onto itself
     806             :         // 3) -               "       - is a Form and to be dropped onto one of it's descendants
     807             :         // 4) one of the entries is a control and to be dropped onto the root
     808             :         // 5) a control or form will be dropped onto a control which is _not_ a sibling (dropping onto a sibling
     809             :         //      means moving the control)
     810             : 
     811             :         // collect the ancestors of the drop targte (speeds up 3)
     812           0 :         SvLBoxEntrySortedArray arrDropAnchestors;
     813           0 :         SvTreeListEntry* pLoop = _pTargetEntry;
     814           0 :         while (pLoop)
     815             :         {
     816           0 :             arrDropAnchestors.insert(pLoop);
     817           0 :             pLoop = GetParent(pLoop);
     818             :         }
     819             : 
     820           0 :         for (   ListBoxEntrySet::const_iterator dropped = aDropped.begin();
     821           0 :                 dropped != aDropped.end();
     822             :                 ++dropped
     823             :             )
     824             :         {
     825           0 :             SvTreeListEntry* pCurrent = *dropped;
     826           0 :             SvTreeListEntry* pCurrentParent = GetParent(pCurrent);
     827             : 
     828             :             // test for 0)
     829           0 :             if (pCurrent == m_pRootEntry)
     830           0 :                 return DND_ACTION_NONE;
     831             : 
     832             :             // test for 1)
     833           0 :             if ( _pTargetEntry == pCurrentParent )
     834           0 :                 return DND_ACTION_NONE;
     835             : 
     836             :             // test for 2)
     837           0 :             if (pCurrent == _pTargetEntry)
     838           0 :                 return DND_ACTION_NONE;
     839             : 
     840             :             // test for 5)
     841             :     //      if ( bDropTargetIsComponent && (pDropTargetParent != pCurrentParent) )
     842           0 :             if ( bDropTargetIsComponent )   // TODO : die obige Zeile wieder rein, dann muss aber ExecuteDrop das Vertauschen auch beherrschen
     843           0 :                 return DND_ACTION_NONE;
     844             : 
     845             :             // test for 3)
     846           0 :             if ( IsFormEntry(pCurrent) )
     847             :             {
     848           0 :                 if ( arrDropAnchestors.find(pCurrent) != arrDropAnchestors.end() )
     849           0 :                     return DND_ACTION_NONE;
     850           0 :             } else if ( IsFormComponentEntry(pCurrent) )
     851             :             {
     852             :                 // test for 4)
     853           0 :                 if (_pTargetEntry == m_pRootEntry)
     854           0 :                     return DND_ACTION_NONE;
     855             :             }
     856             :         }
     857             : 
     858           0 :         return DND_ACTION_MOVE;
     859             :     }
     860             : 
     861             : 
     862           0 :     sal_Int8 NavigatorTree::AcceptDrop( const AcceptDropEvent& rEvt )
     863             :     {
     864           0 :         ::Point aDropPos = rEvt.maPosPixel;
     865             : 
     866             :         // kuemmern wir uns erst mal um moeglich DropActions (Scrollen und Aufklappen)
     867           0 :         if (rEvt.mbLeaving)
     868             :         {
     869           0 :             if (m_aDropActionTimer.IsActive())
     870           0 :                 m_aDropActionTimer.Stop();
     871             :         } else
     872             :         {
     873           0 :             sal_Bool bNeedTrigger = sal_False;
     874             :             // auf dem ersten Eintrag ?
     875           0 :             if ((aDropPos.Y() >= 0) && (aDropPos.Y() < GetEntryHeight()))
     876             :             {
     877           0 :                 m_aDropActionType = DA_SCROLLUP;
     878           0 :                 bNeedTrigger = sal_True;
     879             :             } else
     880             :                 // auf dem letzten (bzw. in dem Bereich, den ein Eintrag einnehmen wuerde, wenn er unten genau buendig
     881             :                 // abschliessen wuerde) ?
     882           0 :                 if ((aDropPos.Y() < GetSizePixel().Height()) && (aDropPos.Y() >= GetSizePixel().Height() - GetEntryHeight()))
     883             :                 {
     884           0 :                     m_aDropActionType = DA_SCROLLDOWN;
     885           0 :                     bNeedTrigger = sal_True;
     886             :                 } else
     887             :                 {   // auf einem Entry mit Children, der nicht aufgeklappt ist ?
     888           0 :                     SvTreeListEntry* pDropppedOn = GetEntry(aDropPos);
     889           0 :                     if (pDropppedOn && (GetChildCount(pDropppedOn) > 0) && !IsExpanded(pDropppedOn))
     890             :                     {
     891             :                         // -> aufklappen
     892           0 :                         m_aDropActionType = DA_EXPANDNODE;
     893           0 :                         bNeedTrigger = sal_True;
     894             :                     }
     895             :                 }
     896             : 
     897           0 :             if (bNeedTrigger && (m_aTimerTriggered != aDropPos))
     898             :             {
     899             :                 // neu anfangen zu zaehlen
     900           0 :                 m_aTimerCounter = DROP_ACTION_TIMER_INITIAL_TICKS;
     901             :                 // die Pos merken, da ich auch AcceptDrops bekomme, wenn sich die Maus gar nicht bewegt hat
     902           0 :                 m_aTimerTriggered = aDropPos;
     903             :                 // und den Timer los
     904           0 :                 if (!m_aDropActionTimer.IsActive()) // gibt es den Timer schon ?
     905             :                 {
     906           0 :                     m_aDropActionTimer.SetTimeout(DROP_ACTION_TIMER_TICK_BASE);
     907           0 :                     m_aDropActionTimer.Start();
     908             :                 }
     909           0 :             } else if (!bNeedTrigger)
     910           0 :                 m_aDropActionTimer.Stop();
     911             :         }
     912             : 
     913           0 :         return implAcceptDataTransfer( GetDataFlavorExVector(), rEvt.mnAction, aDropPos, sal_True );
     914             :     }
     915             : 
     916             : 
     917           0 :     sal_Int8 NavigatorTree::implExecuteDataTransfer( const OControlTransferData& _rData, sal_Int8 _nAction, const ::Point& _rDropPos, sal_Bool _bDnD )
     918             :     {
     919           0 :         return implExecuteDataTransfer( _rData, _nAction, GetEntry( _rDropPos ), _bDnD );
     920             :     }
     921             : 
     922             : 
     923           0 :     sal_Int8 NavigatorTree::implExecuteDataTransfer( const OControlTransferData& _rData, sal_Int8 _nAction, SvTreeListEntry* _pTargetEntry, sal_Bool _bDnD )
     924             :     {
     925           0 :         const DataFlavorExVector& rDataFlavors = _rData.GetDataFlavorExVector();
     926             : 
     927           0 :         if ( DND_ACTION_NONE == implAcceptDataTransfer( rDataFlavors, _nAction, _pTargetEntry, _bDnD ) )
     928             :             // under some platforms, it may happen that ExecuteDrop is called though AcceptDrop returned DND_ACTION_NONE
     929           0 :             return DND_ACTION_NONE;
     930             : 
     931             :         // ware schlecht, wenn nach dem Droppen noch gescrollt wird ...
     932           0 :         if (m_aDropActionTimer.IsActive())
     933           0 :             m_aDropActionTimer.Stop();
     934             : 
     935           0 :         if (!_pTargetEntry)
     936             :             // no target -> no drop
     937           0 :             return DND_ACTION_NONE;
     938             : 
     939             :         // format checks
     940             : #ifdef DBG_UTIL
     941             :         sal_Bool bHasHiddenControlsFormat = OControlExchange::hasHiddenControlModelsFormat( rDataFlavors );
     942             :         sal_Bool bForeignCollection = _rData.getFormsRoot().get() != GetNavModel()->GetFormPage()->GetForms().get();
     943             :         DBG_ASSERT(!bForeignCollection || bHasHiddenControlsFormat, "NavigatorTree::implExecuteDataTransfer: invalid format (AcceptDrop shouldn't have let this pass) !");
     944             :         DBG_ASSERT(bForeignCollection || !m_bDragDataDirty, "NavigatorTree::implExecuteDataTransfer: invalid state (shell changed since last exchange resync) !");
     945             :             // das sollte in AcceptDrop erledigt worden sein : dort wird in _rData die Liste der Controls aufgebaut und m_bDragDataDirty
     946             :             // zurueckgesetzt
     947             : #endif
     948             : 
     949           0 :         if ( DND_ACTION_COPY == _nAction )
     950             :         {   // bHasHiddenControlsFormat means that only hidden controls are part of the data
     951             :             DBG_ASSERT( bHasHiddenControlsFormat, "NavigatorTree::implExecuteDataTransfer: copy allowed for hidden controls only!" );
     952             :             DBG_ASSERT( _pTargetEntry && ( _pTargetEntry != m_pRootEntry ) && IsFormEntry( _pTargetEntry ),
     953             :                 "NavigatorTree::implExecuteDataTransfer: should not be here!" );
     954             :                 // implAcceptDataTransfer should have caught both cases
     955             : 
     956             :             DBG_ASSERT(bHasHiddenControlsFormat, "NavigatorTree::implExecuteDataTransfer: only copying of hidden controls is supported !");
     957             :                 // das sollte das AcceptDrop abgefangen haben
     958             : 
     959             :             // da ich gleich die Zielobjekte alle selektieren will (und nur die)
     960           0 :             SelectAll(false);
     961             : 
     962           0 :             Sequence< Reference< XInterface > > aControls = _rData.hiddenControls();
     963           0 :             sal_Int32 nCount = aControls.getLength();
     964           0 :             const Reference< XInterface >* pControls = aControls.getConstArray();
     965             : 
     966           0 :             FmFormShell* pFormShell = GetNavModel()->GetFormShell();
     967           0 :             FmFormModel* pFormModel = pFormShell ? pFormShell->GetFormModel() : NULL;
     968             : 
     969             :             // innerhalb eines Undo ...
     970           0 :             if (pFormModel)
     971             :             {
     972           0 :                 OUString aStr(SVX_RESSTR(RID_STR_CONTROL));
     973           0 :                 OUString aUndoStr = SVX_RESSTR(RID_STR_UNDO_CONTAINER_INSERT).replaceAll(OUString('#'), aStr);
     974           0 :                 pFormModel->BegUndo(aUndoStr);
     975             :             }
     976             : 
     977             :             // die Conrtols kopieren
     978           0 :             for (sal_Int32 i=0; i<nCount; ++i)
     979             :             {
     980             :                 // neues Control anlegen
     981           0 :                 OUString fControlName = FM_COMPONENT_HIDDEN;
     982           0 :                 FmControlData* pNewControlData = NewControl( fControlName, _pTargetEntry, sal_False);
     983           0 :                 Reference< XPropertySet >  xNewPropSet( pNewControlData->GetPropertySet() );
     984             : 
     985             :                 // und die Properties des alten in das neue kopieren
     986           0 :                 Reference< XPropertySet >  xCurrent(pControls[i], UNO_QUERY);
     987             : #if (OSL_DEBUG_LEVEL > 1)
     988             :                 // nur mal eben sehen, ob das Ding tatsaechlich ein hidden control ist
     989             :                 sal_Int16 nClassId = ::comphelper::getINT16(xCurrent->getPropertyValue(FM_PROP_CLASSID));
     990             :                 OSL_ENSURE(nClassId == FormComponentType::HIDDENCONTROL, "NavigatorTree::implExecuteDataTransfer: invalid control in drop list !");
     991             :                     // wenn das SVX_FM_HIDDEN_CONTROLS-Format vorhanden ist, dann sollten wirklich nur hidden controls in der Sequenz
     992             :                     // stecken
     993             : #endif // (OSL_DEBUG_LEVEL > 1)
     994           0 :                 Reference< XPropertySetInfo >  xPropInfo( xCurrent->getPropertySetInfo());
     995           0 :                 Sequence< Property> seqAllCurrentProps = xPropInfo->getProperties();
     996           0 :                 Property* pAllCurrentProps = seqAllCurrentProps.getArray();
     997           0 :                 for (sal_Int32 j=0; j<seqAllCurrentProps.getLength(); ++j)
     998             :                 {
     999           0 :                     OUString sCurrentProp = pAllCurrentProps[j].Name;
    1000           0 :                     if (((pAllCurrentProps[j].Attributes & PropertyAttribute::READONLY) == 0) && (sCurrentProp != FM_PROP_NAME))
    1001             :                     {   // (read-only attribs werden natuerlich nicht gesetzt, dito der Name, den hat das NewControl schon eindeutig
    1002             :                         // festgelegt)
    1003           0 :                         xNewPropSet->setPropertyValue(sCurrentProp, xCurrent->getPropertyValue(sCurrentProp));
    1004             :                     }
    1005           0 :                 }
    1006             : 
    1007           0 :                 SvTreeListEntry* pToSelect = FindEntry(pNewControlData);
    1008           0 :                 Select(pToSelect, sal_True);
    1009           0 :                 if (i == 0)
    1010           0 :                     SetCurEntry(pToSelect);
    1011           0 :             }
    1012             : 
    1013           0 :             if (pFormModel)
    1014           0 :                 pFormModel->EndUndo();
    1015             : 
    1016           0 :             return _nAction;
    1017             :         }
    1018             : 
    1019           0 :         if ( !OControlExchange::hasFieldExchangeFormat( _rData.GetDataFlavorExVector() ) )
    1020             :         {
    1021             :             // can't do anything without the internal format here ... usually happens when doing DnD or CnP
    1022             :             // over navigator boundaries
    1023           0 :             return DND_ACTION_NONE;
    1024             :         }
    1025             : 
    1026             :         // some data for the target
    1027           0 :         sal_Bool bDropTargetIsForm = IsFormEntry(_pTargetEntry);
    1028           0 :         FmFormData* pTargetData = bDropTargetIsForm ? (FmFormData*)_pTargetEntry->GetUserData() : NULL;
    1029             : 
    1030             :         DBG_ASSERT( DND_ACTION_COPY != _nAction, "NavigatorTree::implExecuteDataTransfer: somebody changed the logics!" );
    1031             : 
    1032             :         // die Liste der gedraggten Eintraege
    1033           0 :         ListBoxEntrySet aDropped = _rData.selected();
    1034             :         DBG_ASSERT(aDropped.size() >= 1, "NavigatorTree::implExecuteDataTransfer: no entries!");
    1035             : 
    1036             :         // die Shell und das Model
    1037           0 :         FmFormShell* pFormShell = GetNavModel()->GetFormShell();
    1038           0 :         FmFormModel* pFormModel = pFormShell ? pFormShell->GetFormModel() : NULL;
    1039           0 :         if (!pFormModel)
    1040           0 :             return DND_ACTION_NONE;
    1041             : 
    1042             :         // fuer's Undo
    1043           0 :         const bool bUndo = pFormModel->IsUndoEnabled();
    1044             : 
    1045           0 :         if( bUndo )
    1046             :         {
    1047           0 :             OUString strUndoDescription(SVX_RESSTR(RID_STR_UNDO_CONTAINER_REPLACE));
    1048           0 :             pFormModel->BegUndo(strUndoDescription);
    1049             :         }
    1050             : 
    1051             :         // ich nehme vor dem Einfuegen eines Eintrages seine Selection raus, damit die Markierung dabei nicht flackert
    1052             :         // -> das Handeln des Select locken
    1053           0 :         LockSelectionHandling();
    1054             : 
    1055             :         // jetzt durch alle gedroppten Eintraege ...
    1056           0 :         for (   ListBoxEntrySet::const_iterator dropped = aDropped.begin();
    1057           0 :                 dropped != aDropped.end();
    1058             :                 ++dropped
    1059             :             )
    1060             :         {
    1061             :             // ein paar Daten zum aktuellen Element
    1062           0 :             SvTreeListEntry* pCurrent = *dropped;
    1063             :             DBG_ASSERT(pCurrent != NULL, "NavigatorTree::implExecuteDataTransfer: ungueltiger Eintrag");
    1064             :             DBG_ASSERT(GetParent(pCurrent) != NULL, "NavigatorTree::implExecuteDataTransfer: ungueltiger Eintrag");
    1065             :                 // die Root darf nicht gedraggt werden
    1066             : 
    1067           0 :             FmEntryData* pCurrentUserData = (FmEntryData*)pCurrent->GetUserData();
    1068             : 
    1069           0 :             Reference< XChild >  xCurrentChild(pCurrentUserData->GetChildIFace(), UNO_QUERY);
    1070           0 :             Reference< XIndexContainer >  xContainer(xCurrentChild->getParent(), UNO_QUERY);
    1071             : 
    1072           0 :             FmFormData* pCurrentParentUserData = (FmFormData*)pCurrentUserData->GetParent();
    1073             :             DBG_ASSERT(pCurrentParentUserData == NULL || pCurrentParentUserData->ISA(FmFormData), "NavigatorTree::implExecuteDataTransfer: ungueltiges Parent");
    1074             : 
    1075             :             // beim Vater austragen
    1076           0 :             if (pCurrentParentUserData)
    1077           0 :                 pCurrentParentUserData->GetChildList()->remove( pCurrentUserData );
    1078             :             else
    1079           0 :                 GetNavModel()->GetRootList()->remove( pCurrentUserData );
    1080             : 
    1081             :             // aus dem Container entfernen
    1082           0 :             sal_Int32 nIndex = getElementPos(xContainer, xCurrentChild);
    1083           0 :             GetNavModel()->m_pPropChangeList->Lock();
    1084             :             // die Undo-Action fuer das Rausnehmen
    1085           0 :             if ( bUndo && GetNavModel()->m_pPropChangeList->CanUndo())
    1086             :             {
    1087             :                 pFormModel->AddUndo(new FmUndoContainerAction(*pFormModel, FmUndoContainerAction::Removed,
    1088           0 :                                                             xContainer, xCurrentChild, nIndex));
    1089             :             }
    1090           0 :             else if( !GetNavModel()->m_pPropChangeList->CanUndo() )
    1091             :             {
    1092           0 :                 FmUndoContainerAction::DisposeElement( xCurrentChild );
    1093             :             }
    1094             : 
    1095             :             // Events mitkopieren
    1096           0 :             Reference< XEventAttacherManager >  xManager(xContainer, UNO_QUERY);
    1097           0 :             Sequence< ScriptEventDescriptor > aEvts;
    1098             : 
    1099           0 :             if (xManager.is() && nIndex >= 0)
    1100           0 :                 aEvts = xManager->getScriptEvents(nIndex);
    1101           0 :             xContainer->removeByIndex(nIndex);
    1102             : 
    1103             :             // die Selection raus
    1104           0 :             Select(pCurrent, sal_False);
    1105             :             // und weg
    1106           0 :             Remove(pCurrentUserData);
    1107             : 
    1108             :             // die Stelle innerhalb des DropParents, an der ich die gedroppten Eintraege einfuegen muss
    1109           0 :             if (pTargetData)
    1110           0 :                 xContainer = Reference< XIndexContainer > (pTargetData->GetElement(), UNO_QUERY);
    1111             :             else
    1112           0 :                 xContainer = Reference< XIndexContainer > (GetNavModel()->GetForms(), UNO_QUERY);
    1113             : 
    1114             :             // immer ganz hinten einfuegen
    1115           0 :             nIndex = xContainer->getCount();
    1116             : 
    1117             :             // UndoAction fuer das Einfuegen
    1118           0 :             if ( bUndo && GetNavModel()->m_pPropChangeList->CanUndo())
    1119             :                 pFormModel->AddUndo(new FmUndoContainerAction(*pFormModel, FmUndoContainerAction::Inserted,
    1120           0 :                                                          xContainer, xCurrentChild, nIndex));
    1121             : 
    1122             :             // einfuegen im neuen Container
    1123           0 :             if (pTargetData)
    1124             :             {
    1125             :                  // es wird in eine Form eingefuegt, dann brauche ich eine FormComponent
    1126           0 :                 xContainer->insertByIndex( nIndex,
    1127           0 :                     makeAny( Reference< XFormComponent >( xCurrentChild, UNO_QUERY ) ) );
    1128             :             }
    1129             :             else
    1130             :             {
    1131           0 :                 xContainer->insertByIndex( nIndex,
    1132           0 :                     makeAny( Reference< XForm >( xCurrentChild, UNO_QUERY ) ) );
    1133             :             }
    1134             : 
    1135           0 :             if (aEvts.getLength())
    1136             :             {
    1137           0 :                 xManager = Reference< XEventAttacherManager > (xContainer, UNO_QUERY);
    1138           0 :                 if (xManager.is())
    1139           0 :                     xManager->registerScriptEvents(nIndex, aEvts);
    1140             :             }
    1141             : 
    1142           0 :             GetNavModel()->m_pPropChangeList->UnLock();
    1143             : 
    1144             :             // zuerst dem Eintrag das neue Parent
    1145           0 :             pCurrentUserData->SetParent(pTargetData);
    1146             : 
    1147             :             // dann dem Parent das neue Child
    1148           0 :             if (pTargetData)
    1149           0 :                 pTargetData->GetChildList()->insert( pCurrentUserData, nIndex );
    1150             :             else
    1151           0 :                 GetNavModel()->GetRootList()->insert( pCurrentUserData, nIndex );
    1152             : 
    1153             :             // dann bei mir selber bekanntgeben und neu selektieren
    1154           0 :             SvTreeListEntry* pNew = Insert( pCurrentUserData, nIndex );
    1155           0 :             if ( ( aDropped.begin() == dropped ) && pNew )
    1156             :             {
    1157           0 :                 SvTreeListEntry* pParent = GetParent( pNew );
    1158           0 :                 if ( pParent )
    1159           0 :                     Expand( pParent );
    1160             :             }
    1161           0 :         }
    1162             : 
    1163           0 :         UnlockSelectionHandling();
    1164             : 
    1165           0 :         if( bUndo )
    1166           0 :             pFormModel->EndUndo();
    1167             : 
    1168             :         // During the move, the markings of the underlying view did not change (because the view is not affected by the logical
    1169             :         // hierarchy of the form/control models. But my selection changed - which means I have to adjust it according to the
    1170             :         // view marks, again.
    1171           0 :         SynchronizeSelection();
    1172             : 
    1173             :         // in addition, with the move of controls such things as "the current form" may have changed - force the shell
    1174             :         // to update itself accordingly
    1175           0 :         if( pFormShell && pFormShell->GetImpl() && pFormShell->GetFormView() )
    1176           0 :             pFormShell->GetImpl()->DetermineSelection( pFormShell->GetFormView()->GetMarkedObjectList() );
    1177             : 
    1178           0 :         if ( m_aControlExchange.isClipboardOwner() && ( DND_ACTION_MOVE == _nAction ) )
    1179           0 :             m_aControlExchange->clear();
    1180             : 
    1181           0 :         return _nAction;
    1182             :     }
    1183             : 
    1184             : 
    1185           0 :     sal_Int8 NavigatorTree::ExecuteDrop( const ExecuteDropEvent& rEvt )
    1186             :     {
    1187           0 :         sal_Int8 nResult( DND_ACTION_NONE );
    1188             : 
    1189           0 :         if ( m_aControlExchange.isDragSource() )
    1190           0 :             nResult = implExecuteDataTransfer( *m_aControlExchange, rEvt.mnAction, rEvt.maPosPixel, sal_True );
    1191             :         else
    1192             :         {
    1193           0 :             OControlTransferData aDroppedData( rEvt.maDropEvent.Transferable );
    1194           0 :             nResult = implExecuteDataTransfer( aDroppedData, rEvt.mnAction, rEvt.maPosPixel, sal_True );
    1195             :         }
    1196             : 
    1197           0 :         return nResult;
    1198             :     }
    1199             : 
    1200             : 
    1201           0 :     void NavigatorTree::doPaste()
    1202             :     {
    1203             :            try
    1204             :         {
    1205           0 :             if ( m_aControlExchange.isClipboardOwner() )
    1206             :             {
    1207           0 :                 implExecuteDataTransfer( *m_aControlExchange, doingKeyboardCut( ) ? DND_ACTION_MOVE : DND_ACTION_COPY, FirstSelected(), sal_False );
    1208             :             }
    1209             :             else
    1210             :             {
    1211             :                 // the clipboard content
    1212           0 :                 Reference< XClipboard > xClipboard( GetClipboard() );
    1213           0 :                 Reference< XTransferable > xTransferable;
    1214           0 :                 if ( xClipboard.is() )
    1215           0 :                     xTransferable = xClipboard->getContents();
    1216             : 
    1217           0 :                 OControlTransferData aClipboardContent( xTransferable );
    1218           0 :                 implExecuteDataTransfer( aClipboardContent, DND_ACTION_COPY, FirstSelected(), sal_False );
    1219             :             }
    1220             :         }
    1221           0 :         catch( const Exception& )
    1222             :         {
    1223             :             OSL_FAIL( "NavigatorTree::doPaste: caught an exception!" );
    1224             :         }
    1225           0 :     }
    1226             : 
    1227             : 
    1228           0 :     void NavigatorTree::doCopy()
    1229             :     {
    1230           0 :         if ( implPrepareExchange( DND_ACTION_COPY ) )
    1231             :         {
    1232           0 :             m_aControlExchange.setClipboardListener( LINK( this, NavigatorTree, OnClipboardAction ) );
    1233           0 :             m_aControlExchange.copyToClipboard( );
    1234             :         }
    1235           0 :     }
    1236             : 
    1237             : 
    1238           0 :     void NavigatorTree::ModelHasRemoved( SvTreeListEntry* _pEntry )
    1239             :     {
    1240           0 :         SvTreeListEntry* pTypedEntry = static_cast< SvTreeListEntry* >( _pEntry );
    1241           0 :         if ( doingKeyboardCut() )
    1242           0 :             m_aCutEntries.erase( pTypedEntry );
    1243             : 
    1244           0 :         if ( m_aControlExchange.isDataExchangeActive() )
    1245             :         {
    1246           0 :             if ( 0 == m_aControlExchange->onEntryRemoved( pTypedEntry ) )
    1247             :             {
    1248             :                 // last of the entries which we put into the clipboard has been deleted from the tree.
    1249             :                 // Give up the clipboard ownership.
    1250           0 :                 m_aControlExchange.clear();
    1251             :             }
    1252             :         }
    1253           0 :     }
    1254             : 
    1255             : 
    1256           0 :     void NavigatorTree::doCut()
    1257             :     {
    1258           0 :         if ( implPrepareExchange( DND_ACTION_MOVE ) )
    1259             :         {
    1260           0 :             m_aControlExchange.setClipboardListener( LINK( this, NavigatorTree, OnClipboardAction ) );
    1261           0 :             m_aControlExchange.copyToClipboard( );
    1262           0 :             m_bKeyboardCut = sal_True;
    1263             : 
    1264             :             // mark all the entries we just "cut" into the clipboard as "nearly moved"
    1265           0 :             for ( SvLBoxEntrySortedArray::const_iterator it = m_arrCurrentSelection.begin();
    1266           0 :                   it != m_arrCurrentSelection.end(); ++it )
    1267             :             {
    1268           0 :                 SvTreeListEntry* pEntry = *it;
    1269           0 :                 if ( pEntry )
    1270             :                 {
    1271           0 :                     m_aCutEntries.insert( pEntry );
    1272           0 :                     pEntry->SetFlags( pEntry->GetFlags() | SV_ENTRYFLAG_SEMITRANSPARENT );
    1273           0 :                     InvalidateEntry( pEntry );
    1274             :                 }
    1275             :             }
    1276             :         }
    1277           0 :     }
    1278             : 
    1279             : 
    1280           0 :     void NavigatorTree::KeyInput(const ::KeyEvent& rKEvt)
    1281             :     {
    1282           0 :         const KeyCode& rCode = rKEvt.GetKeyCode();
    1283             : 
    1284             :         // delete?
    1285           0 :         if (rKEvt.GetKeyCode().GetCode() == KEY_DELETE && !rKEvt.GetKeyCode().GetModifier())
    1286             :         {
    1287           0 :             DeleteSelection();
    1288           0 :             return;
    1289             :         }
    1290             : 
    1291             :         // copy'n'paste?
    1292           0 :         switch ( rCode.GetFunction() )
    1293             :         {
    1294             :             case KEYFUNC_CUT:
    1295           0 :                 doCut();
    1296           0 :                 break;
    1297             : 
    1298             :             case KEYFUNC_PASTE:
    1299           0 :                 if ( implAcceptPaste() )
    1300           0 :                     doPaste();
    1301           0 :                 break;
    1302             : 
    1303             :             case KEYFUNC_COPY:
    1304           0 :                 doCopy();
    1305           0 :                 break;
    1306             : 
    1307             :             default:
    1308           0 :                 break;
    1309             :         }
    1310             : 
    1311           0 :         SvTreeListBox::KeyInput(rKEvt);
    1312             :     }
    1313             : 
    1314             : 
    1315           0 :     bool NavigatorTree::EditingEntry( SvTreeListEntry* pEntry, Selection& rSelection )
    1316             :     {
    1317           0 :         if (!SvTreeListBox::EditingEntry( pEntry, rSelection ))
    1318           0 :             return false;
    1319             : 
    1320           0 :         return (pEntry && (pEntry->GetUserData() != NULL));
    1321             :             // die Wurzel, die ich nicht umbenennen darf, hat als UserData NULL
    1322             :     }
    1323             : 
    1324             : 
    1325           0 :     void NavigatorTree::NewForm( SvTreeListEntry* pParentEntry )
    1326             :     {
    1327             : 
    1328             :         // ParentFormData holen
    1329           0 :         if( !IsFormEntry(pParentEntry) )
    1330           0 :             return;
    1331             : 
    1332           0 :         FmFormData* pParentFormData = (FmFormData*)pParentEntry->GetUserData();
    1333             : 
    1334             : 
    1335             :         // Neue Form erzeugen
    1336           0 :         Reference<XComponentContext> xContext = comphelper::getProcessComponentContext();
    1337           0 :         Reference< XForm >  xNewForm(xContext->getServiceManager()->createInstanceWithContext(FM_SUN_COMPONENT_FORM, xContext), UNO_QUERY);
    1338           0 :         if (!xNewForm.is())
    1339           0 :             return;
    1340             : 
    1341           0 :         FmFormData* pNewFormData = new FmFormData( xNewForm, m_aNavigatorImages, pParentFormData );
    1342             : 
    1343             : 
    1344             :         // Namen setzen
    1345           0 :         OUString aName = GenerateName(pNewFormData);
    1346           0 :         pNewFormData->SetText(aName);
    1347             : 
    1348           0 :         Reference< XPropertySet >  xPropertySet(xNewForm, UNO_QUERY);
    1349           0 :         if (!xPropertySet.is())
    1350           0 :             return;
    1351             :         try
    1352             :         {
    1353           0 :             xPropertySet->setPropertyValue( FM_PROP_NAME, makeAny(aName) );
    1354             :             // a form should always have the command type table as default
    1355           0 :             xPropertySet->setPropertyValue( FM_PROP_COMMANDTYPE, makeAny(sal_Int32(CommandType::TABLE)));
    1356             :         }
    1357           0 :         catch ( const Exception& )
    1358             :         {
    1359             :             OSL_FAIL("NavigatorTree::NewForm : could not set esssential properties !");
    1360             :         }
    1361             : 
    1362             : 
    1363             : 
    1364             :         // Form einfuegen
    1365           0 :         GetNavModel()->Insert( pNewFormData, TREELIST_APPEND, sal_True );
    1366             : 
    1367             : 
    1368             :         // Neue Form als aktive Form setzen
    1369           0 :         FmFormShell* pFormShell = GetNavModel()->GetFormShell();
    1370           0 :         if( pFormShell )
    1371             :         {
    1372           0 :             InterfaceBag aSelection;
    1373           0 :             aSelection.insert( xNewForm );
    1374           0 :             pFormShell->GetImpl()->setCurrentSelection( aSelection );
    1375             : 
    1376           0 :             pFormShell->GetViewShell()->GetViewFrame()->GetBindings().Invalidate(SID_FM_PROPERTIES, true, true);
    1377             :         }
    1378           0 :         GetNavModel()->SetModified();
    1379             : 
    1380             : 
    1381             :         // In EditMode schalten
    1382           0 :         SvTreeListEntry* pNewEntry = FindEntry( pNewFormData );
    1383           0 :         EditEntry( pNewEntry );
    1384             :     }
    1385             : 
    1386             : 
    1387           0 :     FmControlData* NavigatorTree::NewControl( const OUString& rServiceName, SvTreeListEntry* pParentEntry, sal_Bool bEditName )
    1388             :     {
    1389             : 
    1390             :         // ParentForm holen
    1391           0 :         if (!GetNavModel()->GetFormShell())
    1392           0 :             return NULL;
    1393           0 :         if (!IsFormEntry(pParentEntry))
    1394           0 :             return NULL;
    1395             : 
    1396           0 :         FmFormData* pParentFormData = (FmFormData*)pParentEntry->GetUserData();;
    1397           0 :         Reference< XForm >  xParentForm( pParentFormData->GetFormIface());
    1398             : 
    1399             : 
    1400             :         // Neue Component erzeugen
    1401           0 :         Reference<XComponentContext> xContext = comphelper::getProcessComponentContext();
    1402           0 :         Reference<XFormComponent> xNewComponent( xContext->getServiceManager()->createInstanceWithContext(rServiceName, xContext), UNO_QUERY);
    1403           0 :         if (!xNewComponent.is())
    1404           0 :             return NULL;
    1405             : 
    1406           0 :         FmControlData* pNewFormControlData = new FmControlData( xNewComponent, m_aNavigatorImages, pParentFormData );
    1407             : 
    1408             : 
    1409             :         // Namen setzen
    1410           0 :         FmFormView*     pFormView       = GetNavModel()->GetFormShell()->GetFormView();
    1411           0 :         SdrPageView*    pPageView       = pFormView->GetSdrPageView();
    1412           0 :         FmFormPage*     pPage           = (FmFormPage*)pPageView->GetPage();
    1413             : 
    1414           0 :         OUString sName = pPage->GetImpl().setUniqueName( xNewComponent, xParentForm );
    1415             : 
    1416           0 :         pNewFormControlData->SetText( sName );
    1417             : 
    1418             : 
    1419             :         // FormComponent einfuegen
    1420           0 :         GetNavModel()->Insert( pNewFormControlData, TREELIST_APPEND, sal_True );
    1421           0 :         GetNavModel()->SetModified();
    1422             : 
    1423           0 :         if (bEditName)
    1424             :         {
    1425             : 
    1426             :             // In EditMode schalten
    1427           0 :             SvTreeListEntry* pNewEntry = FindEntry( pNewFormControlData );
    1428           0 :             Select( pNewEntry, sal_True );
    1429           0 :             EditEntry( pNewEntry );
    1430             :         }
    1431             : 
    1432           0 :         return pNewFormControlData;
    1433             :     }
    1434             : 
    1435             : 
    1436           0 :     OUString NavigatorTree::GenerateName( FmEntryData* pEntryData )
    1437             :     {
    1438           0 :         const sal_uInt16 nMaxCount = 99;
    1439           0 :         OUString aNewName;
    1440             : 
    1441             : 
    1442             :         // BasisNamen erzeugen
    1443           0 :         OUString aBaseName;
    1444           0 :         if( pEntryData->ISA(FmFormData) )
    1445           0 :             aBaseName = SVX_RESSTR( RID_STR_STDFORMNAME );
    1446           0 :         else if( pEntryData->ISA(FmControlData) )
    1447           0 :             aBaseName = SVX_RESSTR( RID_STR_CONTROL );
    1448             : 
    1449             : 
    1450             :         // Neuen Namen erstellen
    1451           0 :         FmFormData* pFormParentData = (FmFormData*)pEntryData->GetParent();
    1452             : 
    1453           0 :         for( sal_Int32 i=0; i<nMaxCount; i++ )
    1454             :         {
    1455           0 :             aNewName = aBaseName;
    1456           0 :             if( i>0 )
    1457             :             {
    1458           0 :                 aNewName += " ";
    1459           0 :                 aNewName += OUString::number(i).getStr();
    1460             :             }
    1461             : 
    1462           0 :             if( GetNavModel()->FindData(aNewName, pFormParentData,sal_False) == NULL )
    1463           0 :                 break;
    1464             :         }
    1465             : 
    1466           0 :         return aNewName;
    1467             :     }
    1468             : 
    1469             : 
    1470           0 :     bool NavigatorTree::EditedEntry( SvTreeListEntry* pEntry, const OUString& rNewText )
    1471             :     {
    1472           0 :         if (EditingCanceled())
    1473           0 :             return true;
    1474             : 
    1475           0 :         GrabFocus();
    1476           0 :         FmEntryData* pEntryData = (FmEntryData*)pEntry->GetUserData();
    1477           0 :         bool bRes = GetNavModel()->Rename( pEntryData, rNewText);
    1478           0 :         if( !bRes )
    1479             :         {
    1480           0 :             m_pEditEntry = pEntry;
    1481           0 :             nEditEvent = Application::PostUserEvent( LINK(this, NavigatorTree, OnEdit) );
    1482             :         } else
    1483           0 :             SetCursor(pEntry, true);
    1484             : 
    1485           0 :         return bRes;
    1486             :     }
    1487             : 
    1488             : 
    1489           0 :     IMPL_LINK_NOARG(NavigatorTree, OnEdit)
    1490             :     {
    1491           0 :         nEditEvent = 0;
    1492           0 :         EditEntry( m_pEditEntry );
    1493           0 :         m_pEditEntry = NULL;
    1494             : 
    1495           0 :         return 0L;
    1496             :     }
    1497             : 
    1498             : 
    1499           0 :     IMPL_LINK_NOARG(NavigatorTree, OnDropActionTimer)
    1500             :     {
    1501           0 :         if (--m_aTimerCounter > 0)
    1502           0 :             return 0L;
    1503             : 
    1504           0 :         switch ( m_aDropActionType )
    1505             :         {
    1506             :         case DA_EXPANDNODE:
    1507             :         {
    1508           0 :             SvTreeListEntry* pToExpand = GetEntry(m_aTimerTriggered);
    1509           0 :             if (pToExpand && (GetChildCount(pToExpand) > 0) &&  !IsExpanded(pToExpand))
    1510             :                 // tja, eigentlich muesste ich noch testen, ob die Node nicht schon expandiert ist, aber ich
    1511             :                 // habe dazu weder in den Basisklassen noch im Model eine Methode gefunden ...
    1512             :                 // aber ich denke, die BK sollte es auch so vertragen
    1513           0 :                 Expand(pToExpand);
    1514             : 
    1515             :             // nach dem Expand habe ich im Gegensatz zum Scrollen natuerlich nix mehr zu tun
    1516           0 :             m_aDropActionTimer.Stop();
    1517             :         }
    1518           0 :         break;
    1519             : 
    1520             :         case DA_SCROLLUP :
    1521           0 :             ScrollOutputArea( 1 );
    1522           0 :             m_aTimerCounter = DROP_ACTION_TIMER_SCROLL_TICKS;
    1523           0 :             break;
    1524             : 
    1525             :         case DA_SCROLLDOWN :
    1526           0 :             ScrollOutputArea( -1 );
    1527           0 :             m_aTimerCounter = DROP_ACTION_TIMER_SCROLL_TICKS;
    1528           0 :             break;
    1529             : 
    1530             :         }
    1531             : 
    1532           0 :         return 0L;
    1533             :     }
    1534             : 
    1535             : 
    1536           0 :     IMPL_LINK(NavigatorTree, OnEntrySelDesel, NavigatorTree*, /*pThis*/)
    1537             :     {
    1538           0 :         m_sdiState = SDI_DIRTY;
    1539             : 
    1540           0 :         if (IsSelectionHandlingLocked())
    1541           0 :             return 0L;
    1542             : 
    1543           0 :         if (m_aSynchronizeTimer.IsActive())
    1544           0 :             m_aSynchronizeTimer.Stop();
    1545             : 
    1546           0 :         m_aSynchronizeTimer.SetTimeout(EXPLORER_SYNC_DELAY);
    1547           0 :         m_aSynchronizeTimer.Start();
    1548             : 
    1549           0 :         return 0L;
    1550             :     }
    1551             : 
    1552             : 
    1553           0 :     IMPL_LINK_NOARG(NavigatorTree, OnSynchronizeTimer)
    1554             :     {
    1555           0 :         SynchronizeMarkList();
    1556           0 :         return 0L;
    1557             :     }
    1558             : 
    1559             : 
    1560             : 
    1561           0 :     IMPL_LINK_NOARG(NavigatorTree, OnClipboardAction)
    1562             :     {
    1563           0 :         if ( !m_aControlExchange.isClipboardOwner() )
    1564             :         {
    1565           0 :             if ( doingKeyboardCut() )
    1566             :             {
    1567           0 :                 for (   ListBoxEntrySet::const_iterator i = m_aCutEntries.begin();
    1568           0 :                         i != m_aCutEntries.end();
    1569             :                         ++i
    1570             :                     )
    1571             :                 {
    1572           0 :                     SvTreeListEntry* pEntry = *i;
    1573           0 :                     if ( !pEntry )
    1574           0 :                         continue;
    1575             : 
    1576           0 :                     pEntry->SetFlags( pEntry->GetFlags() & ~SV_ENTRYFLAG_SEMITRANSPARENT );
    1577           0 :                     InvalidateEntry( pEntry );
    1578             :                 }
    1579           0 :                 ListBoxEntrySet aEmpty;
    1580           0 :                 m_aCutEntries.swap( aEmpty );
    1581             : 
    1582           0 :                 m_bKeyboardCut = sal_False;
    1583             :             }
    1584             :         }
    1585           0 :         return 0L;
    1586             :     }
    1587             : 
    1588             : 
    1589           0 :     void NavigatorTree::ShowSelectionProperties(sal_Bool bForce)
    1590             :     {
    1591             :         // zuerst brauche ich die FormShell
    1592           0 :         FmFormShell* pFormShell = GetNavModel()->GetFormShell();
    1593           0 :         if (!pFormShell)
    1594             :             // keine Shell -> ich koennte kein curObject setzen -> raus
    1595           0 :             return;
    1596             : 
    1597           0 :         CollectSelectionData(SDI_ALL);
    1598             :         SAL_WARN_IF(static_cast<size_t>(m_nFormsSelected + m_nControlsSelected
    1599             :                 + (m_bRootSelected ? 1 : 0)) != m_arrCurrentSelection.size(),
    1600             :             "svx.form",
    1601             :             "NavigatorTree::ShowSelectionProperties : selection meta data invalid !");
    1602             : 
    1603             : 
    1604           0 :         InterfaceBag aSelection;
    1605           0 :         sal_Bool bSetSelectionAsMarkList = sal_False;
    1606             : 
    1607           0 :         if (m_bRootSelected)
    1608             :             ;                                   // no properties for the root, neither for single nor for multi selection
    1609           0 :         else if ( m_nFormsSelected + m_nControlsSelected == 0 )   // none of the two should be less 0
    1610             :             ;                                   // no selection -> no properties
    1611           0 :         else if ( m_nFormsSelected * m_nControlsSelected != 0 )
    1612             :             ;                                   // mixed selection -> no properties
    1613             :         else
    1614             :         {   // either only forms, or only controls are selected
    1615           0 :             if (m_arrCurrentSelection.size() == 1)
    1616             :             {
    1617           0 :                 if (m_nFormsSelected > 0)
    1618             :                 {   // es ist genau eine Form selektiert
    1619           0 :                     FmFormData* pFormData = (FmFormData*)(*m_arrCurrentSelection.begin())->GetUserData();
    1620           0 :                     aSelection.insert( Reference< XInterface >( pFormData->GetFormIface(), UNO_QUERY ) );
    1621             :                 }
    1622             :                 else
    1623             :                 {   // es ist genau ein Control selektiert (egal ob hidden oder normal)
    1624           0 :                     FmEntryData* pEntryData = (FmEntryData*)(*m_arrCurrentSelection.begin())->GetUserData();
    1625             : 
    1626           0 :                     aSelection.insert( Reference< XInterface >( pEntryData->GetElement(), UNO_QUERY ) );
    1627             :                 }
    1628             :             }
    1629             :             else
    1630             :             {   // wir haben eine MultiSelection, also muessen wir ein MultiSet dafuer aufbauen
    1631           0 :                 if (m_nFormsSelected > 0)
    1632             :                 {   // ... nur Forms
    1633             :                     // erstmal die PropertySet-Interfaces der Forms einsammeln
    1634           0 :                    SvLBoxEntrySortedArray::const_iterator it = m_arrCurrentSelection.begin();
    1635           0 :                     for ( sal_Int32 i = 0; i < m_nFormsSelected; ++i )
    1636             :                     {
    1637           0 :                         FmFormData* pFormData = (FmFormData*)(*it)->GetUserData();
    1638           0 :                         aSelection.insert( pFormData->GetPropertySet().get() );
    1639           0 :                         ++it;
    1640             :                     }
    1641             :                 }
    1642             :                 else
    1643             :                 {   // ... nur Controls
    1644           0 :                     if (m_nHiddenControls == m_nControlsSelected)
    1645             :                     {   // ein MultiSet fuer die Properties der hidden controls
    1646           0 :                         SvLBoxEntrySortedArray::const_iterator it = m_arrCurrentSelection.begin();
    1647           0 :                         for ( sal_Int32 i = 0; i < m_nHiddenControls; ++i )
    1648             :                         {
    1649           0 :                             FmEntryData* pEntryData = (FmEntryData*)(*it)->GetUserData();
    1650           0 :                             aSelection.insert( pEntryData->GetPropertySet().get() );
    1651           0 :                             ++it;
    1652             :                         }
    1653             :                     }
    1654           0 :                     else if (m_nHiddenControls == 0)
    1655             :                     {   // nur normale Controls
    1656           0 :                         bSetSelectionAsMarkList = sal_True;
    1657             :                     }
    1658             :                 }
    1659             :             }
    1660             : 
    1661             :         }
    1662             : 
    1663             :         // und dann meine Form und mein SelObject
    1664           0 :         if ( bSetSelectionAsMarkList )
    1665           0 :             pFormShell->GetImpl()->setCurrentSelectionFromMark( pFormShell->GetFormView()->GetMarkedObjectList() );
    1666             :         else
    1667           0 :             pFormShell->GetImpl()->setCurrentSelection( aSelection );
    1668             : 
    1669           0 :         if ( pFormShell->GetImpl()->IsPropBrwOpen() || bForce )
    1670             :         {
    1671             :             // und jetzt kann ich das Ganze dem PropertyBrowser uebergeben
    1672           0 :             pFormShell->GetViewShell()->GetViewFrame()->GetDispatcher()->Execute( SID_FM_SHOW_PROPERTY_BROWSER, SFX_CALLMODE_ASYNCHRON );
    1673           0 :         }
    1674             :     }
    1675             : 
    1676             : 
    1677           0 :     void NavigatorTree::DeleteSelection()
    1678             :     {
    1679             :         // die Root darf ich natuerlich nicht mitloeschen
    1680           0 :         sal_Bool bRootSelected = IsSelected(m_pRootEntry);
    1681           0 :         sal_uIntPtr nSelectedEntries = GetSelectionCount();
    1682           0 :         if (bRootSelected && (nSelectedEntries > 1))     // die Root plus andere Elemente ?
    1683           0 :             Select(m_pRootEntry, sal_False);                // ja -> die Root raus
    1684             : 
    1685           0 :         if ((nSelectedEntries == 0) || bRootSelected)    // immer noch die Root ?
    1686           0 :             return;                                     // -> sie ist das einzige selektierte -> raus
    1687             : 
    1688             :         DBG_ASSERT(!m_bPrevSelectionMixed, "NavigatorTree::DeleteSelection() : loeschen nicht erlaubt wenn Markierung und Selektion nciht konsistent");
    1689             : 
    1690             :         // ich brauche unten das FormModel ...
    1691           0 :         FmFormShell* pFormShell = GetNavModel()->GetFormShell();
    1692           0 :         if (!pFormShell)
    1693           0 :             return;
    1694           0 :         FmFormModel* pFormModel = pFormShell->GetFormModel();
    1695           0 :         if (!pFormModel)
    1696           0 :             return;
    1697             : 
    1698             :         // jetzt muss ich noch die DeleteList etwas absichern : wenn man ein Formular und ein abhaengiges
    1699             :         // Element loescht - und zwar in dieser Reihenfolge - dann ist der SvLBoxEntryPtr des abhaengigen Elementes
    1700             :         // natuerlich schon ungueltig, wenn es geloescht werden soll ... diesen GPF, den es dann mit Sicherheit gibt,
    1701             :         // gilt es zu verhindern, also die 'normalisierte' Liste
    1702           0 :         CollectSelectionData( SDI_NORMALIZED );
    1703             : 
    1704             :         // see below for why we need this mapping from models to shapes
    1705           0 :         FmFormView*     pFormView       = pFormShell->GetFormView();
    1706           0 :         SdrPageView*    pPageView       = pFormView ? pFormView->GetSdrPageView() : NULL;
    1707           0 :         SdrPage*        pPage           = pPageView ? pPageView->GetPage() : NULL;
    1708             :         DBG_ASSERT( pPage, "NavigatorTree::DeleteSelection: invalid form page!" );
    1709             : 
    1710           0 :         MapModelToShape aModelShapes;
    1711           0 :         if ( pPage )
    1712           0 :             collectShapeModelMapping( pPage, aModelShapes );
    1713             : 
    1714             :         // problem: we have to use ExplorerModel::Remove, since only this one properly deletes Form objects.
    1715             :         // But, the controls themself must be deleted via DeleteMarked (else, the Writer has some problems
    1716             :         // somewhere). In case I'd first delete the structure, then the controls, the UNDO would not work
    1717             :         // (since UNDO then would mean to first restore the controls, then the structure, means their parent
    1718             :         // form). The other way round, the EntryDatas would be invalid, if I'd first delete the controls and
    1719             :         // then go on to the strucure. This means I have to delete the forms *after* the normal controls, so
    1720             :         // that during UNDO, they're restored in the proper order.
    1721           0 :         pFormShell->GetImpl()->EnableTrackProperties(sal_False);
    1722           0 :         for (SvLBoxEntrySortedArray::reverse_iterator it = m_arrCurrentSelection.rbegin();
    1723           0 :              it != m_arrCurrentSelection.rend(); )
    1724             :         {
    1725           0 :             FmEntryData* pCurrent = (FmEntryData*)((*it)->GetUserData());
    1726             : 
    1727             :             // eine Form ?
    1728           0 :             sal_Bool bIsForm = pCurrent->ISA(FmFormData);
    1729             : 
    1730             :             // da ich das Loeschen im folgenden der View ueberlasse und dabei auf deren MarkList aufbaue, im Normalfall aber bei
    1731             :             // einem makierten Formular nur die direkt, nicht die indirekt abhaengigen Controls markiert werden, muss ich das hier
    1732             :             // noch nachholen
    1733           0 :             if (bIsForm)
    1734           0 :                 MarkViewObj((FmFormData*)pCurrent, sal_True, sal_True);     // das zweite sal_True heisst "deep"
    1735             : 
    1736             :             // ein hidden control ?
    1737           0 :             sal_Bool bIsHidden = IsHiddenControl(pCurrent);
    1738             : 
    1739             :             // Forms und hidden Controls muss ich behalten, alles andere nicht
    1740           0 :             if (!bIsForm && !bIsHidden)
    1741             :             {
    1742             :                 // well, no form and no hidden control -> we can remove it from m_arrCurrentSelection, as it will
    1743             :                 // be deleted automatically. This is because for every model (except forms and hidden control models)
    1744             :                 // there exist a shape, which is marked _if_and_only_if_ the model is selected in our tree.
    1745           0 :                 if ( aModelShapes.find( pCurrent->GetElement() ) != aModelShapes.end() )
    1746             :                 {
    1747             :                     // if there's a shape for the current entry, then either it is marked or it is in a
    1748             :                     // hidden layer (#i28502#), or something like this.
    1749             :                     // In the first case, it will be deleted below, in the second case, we currently don't
    1750             :                     // delete it, as there's no real (working!) API for this, neither in UNO nor in non-UNO.
    1751           0 :                     m_arrCurrentSelection.erase( --(it.base()) );
    1752             :                 }
    1753             :                 else
    1754           0 :                    ++it;
    1755             :                 // In case there is no shape for the current entry, we keep the entry in m_arrCurrentSelection,
    1756             :                 // since then we can definitely remove it.
    1757             :                 // #103597#
    1758             :             }
    1759             :             else
    1760           0 :                 ++it;
    1761             :         }
    1762           0 :         pFormShell->GetImpl()->EnableTrackProperties(sal_True);
    1763             : 
    1764             :         // let the view delete the marked controls
    1765           0 :         pFormShell->GetFormView()->DeleteMarked();
    1766             : 
    1767             :         // start UNDO at this point. Unfortunately, this results in 2 UNDO actions, since DeleteMarked is
    1768             :         // creating an own one. However, if we'd move it before DeleteMarked, Writer does not really like
    1769             :         // this ... :(
    1770             :         // #i31038#
    1771             :         {
    1772             : 
    1773             :             // initialize UNDO
    1774           0 :             OUString aUndoStr;
    1775           0 :             if ( m_arrCurrentSelection.size() == 1 )
    1776             :             {
    1777           0 :                 aUndoStr = SVX_RESSTR(RID_STR_UNDO_CONTAINER_REMOVE);
    1778           0 :                 if (m_nFormsSelected)
    1779           0 :                     aUndoStr = aUndoStr.replaceFirst( "#", SVX_RESSTR( RID_STR_FORM ) );
    1780             :                 else
    1781             :                     // it must be a control (else the root would be selected, but it cannot be deleted)
    1782           0 :                     aUndoStr = aUndoStr.replaceFirst( "#", SVX_RESSTR( RID_STR_CONTROL ) );
    1783             :             }
    1784             :             else
    1785             :             {
    1786           0 :                 aUndoStr = SVX_RESSTR(RID_STR_UNDO_CONTAINER_REMOVE_MULTIPLE);
    1787           0 :                 aUndoStr = aUndoStr.replaceFirst( "#", OUString::number( m_arrCurrentSelection.size() ) );
    1788             :             }
    1789           0 :             pFormModel->BegUndo(aUndoStr);
    1790             :         }
    1791             : 
    1792             :         // remove remaining structure
    1793           0 :         for (SvLBoxEntrySortedArray::const_iterator it = m_arrCurrentSelection.begin();
    1794           0 :              it != m_arrCurrentSelection.end(); ++it)
    1795             :         {
    1796           0 :             FmEntryData* pCurrent = (FmEntryData*)((*it)->GetUserData());
    1797             : 
    1798             :             // if the entry still has children, we skipped deletion of one of those children.
    1799             :             // This may for instance be because the shape is in a hidden layer, where we're unable
    1800             :             // to remove it
    1801           0 :             if ( pCurrent->GetChildList()->size() )
    1802           0 :                 continue;
    1803             : 
    1804             :             // noch ein kleines Problem, bevor ich das ganz loesche : wenn es eine Form ist und die Shell diese als CurrentObject
    1805             :             // kennt, dann muss ich ihr das natuerlich ausreden
    1806           0 :             if (pCurrent->ISA(FmFormData))
    1807             :             {
    1808           0 :                 Reference< XForm >  xCurrentForm( static_cast< FmFormData* >( pCurrent )->GetFormIface() );
    1809           0 :                 if ( pFormShell->GetImpl()->getCurrentForm() == xCurrentForm )  // die Shell kennt die zu loeschende Form ?
    1810           0 :                     pFormShell->GetImpl()->forgetCurrentForm();                 // -> wegnehmen ...
    1811             :             }
    1812           0 :             GetNavModel()->Remove(pCurrent, sal_True);
    1813             :         }
    1814           0 :         pFormModel->EndUndo();
    1815             :     }
    1816             : 
    1817             : 
    1818           0 :     void NavigatorTree::CollectSelectionData(SELDATA_ITEMS sdiHow)
    1819             :     {
    1820             :         DBG_ASSERT(sdiHow != SDI_DIRTY, "NavigatorTree::CollectSelectionData : ever thought about your parameter ? DIRTY ?");
    1821           0 :         if (sdiHow == m_sdiState)
    1822           0 :             return;
    1823             : 
    1824           0 :         m_arrCurrentSelection.clear();
    1825           0 :         m_nFormsSelected = m_nControlsSelected = m_nHiddenControls = 0;
    1826           0 :         m_bRootSelected = sal_False;
    1827             : 
    1828           0 :         SvTreeListEntry* pSelectionLoop = FirstSelected();
    1829           0 :         while (pSelectionLoop)
    1830             :         {
    1831             :             // erst mal die Zaehlung der verschiedenen Elemente
    1832           0 :             if (pSelectionLoop == m_pRootEntry)
    1833           0 :                 m_bRootSelected = sal_True;
    1834             :             else
    1835             :             {
    1836           0 :                 if (IsFormEntry(pSelectionLoop))
    1837           0 :                     ++m_nFormsSelected;
    1838             :                 else
    1839             :                 {
    1840           0 :                     ++m_nControlsSelected;
    1841           0 :                     if (IsHiddenControl((FmEntryData*)(pSelectionLoop->GetUserData())))
    1842           0 :                         ++m_nHiddenControls;
    1843             :                 }
    1844             :             }
    1845             : 
    1846           0 :             if (sdiHow == SDI_NORMALIZED)
    1847             :             {
    1848             :                 // alles, was schon einen selektierten Vorfahr hat, nicht mitnehmen
    1849           0 :                 if (pSelectionLoop == m_pRootEntry)
    1850           0 :                     m_arrCurrentSelection.insert(pSelectionLoop);
    1851             :                 else
    1852             :                 {
    1853           0 :                     SvTreeListEntry* pParentLoop = GetParent(pSelectionLoop);
    1854           0 :                     while (pParentLoop)
    1855             :                     {
    1856             :                         // eigentlich muesste ich testen, ob das Parent in der m_arrCurrentSelection steht ...
    1857             :                         // Aber wenn es selektiert ist, dann steht es in m_arrCurrentSelection, oder wenigstens einer seiner Vorfahren,
    1858             :                         // wenn der auch schon selektiert war. In beiden Faellen reicht also die Abfrage IsSelected
    1859           0 :                         if (IsSelected(pParentLoop))
    1860           0 :                             break;
    1861             :                         else
    1862             :                         {
    1863           0 :                             if (m_pRootEntry == pParentLoop)
    1864             :                             {
    1865             :                                 // bis (exclusive) zur Root gab es kein selektiertes Parent -> der Eintrag gehoert in die normalisierte Liste
    1866           0 :                                 m_arrCurrentSelection.insert(pSelectionLoop);
    1867           0 :                                 break;
    1868             :                             }
    1869             :                             else
    1870           0 :                                 pParentLoop = GetParent(pParentLoop);
    1871             :                         }
    1872             :                     }
    1873             :                 }
    1874             :             }
    1875           0 :             else if (sdiHow == SDI_NORMALIZED_FORMARK)
    1876             :             {
    1877           0 :                 SvTreeListEntry* pParent = GetParent(pSelectionLoop);
    1878           0 :                 if (!pParent || !IsSelected(pParent) || IsFormEntry(pSelectionLoop))
    1879           0 :                     m_arrCurrentSelection.insert(pSelectionLoop);
    1880             :             }
    1881             :             else
    1882           0 :                 m_arrCurrentSelection.insert(pSelectionLoop);
    1883             : 
    1884             : 
    1885           0 :             pSelectionLoop = NextSelected(pSelectionLoop);
    1886             :         }
    1887             : 
    1888           0 :         m_sdiState = sdiHow;
    1889             :     }
    1890             : 
    1891             : 
    1892           0 :     void NavigatorTree::SynchronizeSelection(FmEntryDataArray& arredToSelect)
    1893             :     {
    1894           0 :         LockSelectionHandling();
    1895           0 :         if (arredToSelect.empty())
    1896             :         {
    1897           0 :             SelectAll(false);
    1898             :         }
    1899             :         else
    1900             :         {
    1901             :             // erst mal gleiche ich meine aktuelle Selektion mit der geforderten SelectList ab
    1902           0 :             SvTreeListEntry* pSelection = FirstSelected();
    1903           0 :             while (pSelection)
    1904             :             {
    1905           0 :                 FmEntryData* pCurrent = (FmEntryData*)pSelection->GetUserData();
    1906           0 :                 if (pCurrent != NULL)
    1907             :                 {
    1908           0 :                     FmEntryDataArray::iterator it = arredToSelect.find(pCurrent);
    1909           0 :                     if ( it != arredToSelect.end() )
    1910             :                     {   // der Entry ist schon selektiert, steht aber auch in der SelectList -> er kann aus letzterer
    1911             :                         // raus
    1912           0 :                         arredToSelect.erase(it);
    1913             :                     } else
    1914             :                     {   // der Entry ist selektiert, aber steht nicht in der SelectList -> Selektion rausnehmen
    1915           0 :                         Select(pSelection, sal_False);
    1916             :                         // und sichtbar machen (kann ja sein, dass das die einzige Modifikation ist, die ich hier in dem
    1917             :                         // ganzen Handler mache, dann sollte das zu sehen sein)
    1918           0 :                         MakeVisible(pSelection);
    1919             :                     }
    1920             :                 }
    1921             :                 else
    1922           0 :                     Select(pSelection, sal_False);
    1923             : 
    1924           0 :                 pSelection = NextSelected(pSelection);
    1925             :             }
    1926             : 
    1927             :             // jetzt habe ich in der SelectList genau die Eintraege, die noch selektiert werden muessen
    1928             :             // zwei Moeglichkeiten : 1) ich gehe durch die SelectList, besorge mir zu jedem Eintrag meinen SvTreeListEntry
    1929             :             // und selektiere diesen (waere irgendwie intuitiver ;)) 2) ich gehe durch alle meine SvLBoxEntries und selektiere
    1930             :             // genau die, die ich in der SelectList finde
    1931             :             // 1) braucht O(k*n) (k=Laenge der SelectList, n=Anzahl meiner Entries), plus den Fakt, dass FindEntry nicht den
    1932             :             // Pointer auf die UserDaten vergleicht, sondern ein aufwendigeres IsEqualWithoutChildren durchfuehrt
    1933             :             // 2) braucht O(n*log k), dupliziert aber etwas Code (naemlich den aus FindEntry)
    1934             :             // da das hier eine relativ oft aufgerufenen Stelle sein koennte (bei jeder Aenderung in der Markierung in der View !),
    1935             :             // nehme ich doch lieber letzteres
    1936           0 :             SvTreeListEntry* pLoop = First();
    1937           0 :             while( pLoop )
    1938             :             {
    1939           0 :                 FmEntryData* pCurEntryData = (FmEntryData*)pLoop->GetUserData();
    1940           0 :                 FmEntryDataArray::iterator it = arredToSelect.find(pCurEntryData);
    1941           0 :                 if ( it != arredToSelect.end() )
    1942             :                 {
    1943           0 :                     Select(pLoop, sal_True);
    1944           0 :                     MakeVisible(pLoop);
    1945           0 :                     SetCursor(pLoop, true);
    1946             :                 }
    1947             : 
    1948           0 :                 pLoop = Next( pLoop );
    1949             :             }
    1950             :         }
    1951           0 :         UnlockSelectionHandling();
    1952           0 :     }
    1953             : 
    1954             : 
    1955           0 :     void NavigatorTree::SynchronizeSelection()
    1956             :     {
    1957             :         // Shell und View
    1958           0 :         FmFormShell* pFormShell = GetNavModel()->GetFormShell();
    1959           0 :         if(!pFormShell) return;
    1960             : 
    1961           0 :         FmFormView* pFormView = pFormShell->GetFormView();
    1962           0 :         if (!pFormView) return;
    1963             : 
    1964           0 :         GetNavModel()->BroadcastMarkedObjects(pFormView->GetMarkedObjectList());
    1965             :     }
    1966             : 
    1967             : 
    1968           0 :     void NavigatorTree::SynchronizeMarkList()
    1969             :     {
    1970             :         // die Shell werde ich brauchen ...
    1971           0 :         FmFormShell* pFormShell = GetNavModel()->GetFormShell();
    1972           0 :         if (!pFormShell) return;
    1973             : 
    1974           0 :         CollectSelectionData(SDI_NORMALIZED_FORMARK);
    1975             : 
    1976             :         // Die View soll jetzt kein Notify bei einer Aenderung der MarkList rauslassen
    1977           0 :         pFormShell->GetImpl()->EnableTrackProperties(sal_False);
    1978             : 
    1979           0 :         UnmarkAllViewObj();
    1980             : 
    1981           0 :         for (SvLBoxEntrySortedArray::const_iterator it = m_arrCurrentSelection.begin();
    1982           0 :              it != m_arrCurrentSelection.end(); ++it)
    1983             :         {
    1984           0 :             SvTreeListEntry* pSelectionLoop = *it;
    1985             :             // Bei Formselektion alle Controls dieser Form markieren
    1986           0 :             if (IsFormEntry(pSelectionLoop) && (pSelectionLoop != m_pRootEntry))
    1987           0 :                 MarkViewObj((FmFormData*)pSelectionLoop->GetUserData(), sal_True, sal_False);
    1988             : 
    1989             :             // Bei Controlselektion Control-SdrObjects markieren
    1990           0 :             else if (IsFormComponentEntry(pSelectionLoop))
    1991             :             {
    1992           0 :                 FmControlData* pControlData = (FmControlData*)pSelectionLoop->GetUserData();
    1993           0 :                 if (pControlData)
    1994             :                 {
    1995             : 
    1996             :                     // Beim HiddenControl kann kein Object selektiert werden
    1997           0 :                     Reference< XFormComponent >  xFormComponent( pControlData->GetFormComponent());
    1998           0 :                     if (!xFormComponent.is())
    1999           0 :                         continue;
    2000           0 :                     Reference< XPropertySet >  xSet(xFormComponent, UNO_QUERY);
    2001           0 :                     if (!xSet.is())
    2002           0 :                         continue;
    2003             : 
    2004           0 :                     sal_uInt16 nClassId = ::comphelper::getINT16(xSet->getPropertyValue(FM_PROP_CLASSID));
    2005           0 :                     if (nClassId != FormComponentType::HIDDENCONTROL)
    2006           0 :                         MarkViewObj(pControlData, sal_True, sal_True);
    2007             :                 }
    2008             :             }
    2009             :         }
    2010             : 
    2011             :         // wenn der PropertyBrowser offen ist, muss ich den entsprechend meiner Selektion anpassen
    2012             :         // (NICHT entsprechend der MarkList der View : wenn ich ein Formular selektiert habe, sind in der
    2013             :         // View alle zugehoerigen Controls markiert, trotzdem moechte ich natuerlich die Formular-Eigenschaften
    2014             :         // sehen)
    2015           0 :         ShowSelectionProperties(sal_False);
    2016             : 
    2017             :         // Flag an View wieder zuruecksetzen
    2018           0 :         pFormShell->GetImpl()->EnableTrackProperties(sal_True);
    2019             : 
    2020             :         // wenn jetzt genau eine Form selektiert ist, sollte die Shell das als CurrentForm mitbekommen
    2021             :         // (wenn SelectionHandling nicht locked ist, kuemmert sich die View eigentlich in MarkListHasChanged drum,
    2022             :         // aber der Mechanismus greift zum Beispiel nicht, wenn die Form leer ist)
    2023           0 :         if ((m_arrCurrentSelection.size() == 1) && (m_nFormsSelected == 1))
    2024             :         {
    2025           0 :             FmFormData* pSingleSelectionData = PTR_CAST( FmFormData, static_cast< FmEntryData* >( FirstSelected()->GetUserData() ) );
    2026             :             DBG_ASSERT( pSingleSelectionData, "NavigatorTree::SynchronizeMarkList: invalid selected form!" );
    2027           0 :             if ( pSingleSelectionData )
    2028             :             {
    2029           0 :                 InterfaceBag aSelection;
    2030           0 :                 aSelection.insert( Reference< XInterface >( pSingleSelectionData->GetFormIface(), UNO_QUERY ) );
    2031           0 :                 pFormShell->GetImpl()->setCurrentSelection( aSelection );
    2032             :             }
    2033             :         }
    2034             :     }
    2035             : 
    2036             : 
    2037           0 :     sal_Bool NavigatorTree::IsHiddenControl(FmEntryData* pEntryData)
    2038             :     {
    2039           0 :         if (pEntryData == NULL) return sal_False;
    2040             : 
    2041           0 :         Reference< XPropertySet > xProperties( pEntryData->GetPropertySet() );
    2042           0 :         if (::comphelper::hasProperty(FM_PROP_CLASSID, xProperties))
    2043             :         {
    2044           0 :             Any aClassID = xProperties->getPropertyValue( FM_PROP_CLASSID );
    2045           0 :             return (::comphelper::getINT16(aClassID) == FormComponentType::HIDDENCONTROL);
    2046             :         }
    2047           0 :         return sal_False;
    2048             :     }
    2049             : 
    2050             : 
    2051           0 :     sal_Bool NavigatorTree::Select( SvTreeListEntry* pEntry, sal_Bool bSelect )
    2052             :     {
    2053           0 :         if (bSelect == (IsSelected(pEntry) ? 1 : 0))  // das passiert manchmal, ich glaube, die Basisklasse geht zu sehr auf Nummer sicher ;)
    2054           0 :             return sal_True;
    2055             : 
    2056           0 :         return SvTreeListBox::Select(pEntry, bSelect );
    2057             :     }
    2058             : 
    2059             : 
    2060           0 :     void NavigatorTree::UnmarkAllViewObj()
    2061             :     {
    2062           0 :         FmFormShell* pFormShell = GetNavModel()->GetFormShell();
    2063           0 :         if( !pFormShell )
    2064           0 :             return;
    2065           0 :         FmFormView* pFormView = pFormShell->GetFormView();
    2066           0 :         pFormView->UnMarkAll();
    2067             :     }
    2068             : 
    2069           0 :     void NavigatorTree::MarkViewObj(FmFormData* pFormData, sal_Bool bMark, sal_Bool bDeep )
    2070             :     {
    2071           0 :         FmFormShell* pFormShell = GetNavModel()->GetFormShell();
    2072           0 :         if( !pFormShell )
    2073           0 :             return;
    2074             : 
    2075             :         // first collect all sdrobjects
    2076           0 :         ::std::set< Reference< XFormComponent > > aObjects;
    2077           0 :         CollectObjects(pFormData,bDeep,aObjects);
    2078             : 
    2079             : 
    2080             :         // In der Page das entsprechende SdrObj finden und selektieren
    2081           0 :         FmFormView*     pFormView       = pFormShell->GetFormView();
    2082           0 :         SdrPageView*    pPageView       = pFormView->GetSdrPageView();
    2083           0 :         SdrPage*        pPage           = pPageView->GetPage();
    2084             :         //FmFormPage*     pFormPage       = dynamic_cast< FmFormPage* >( pPage );
    2085             : 
    2086           0 :         SdrObjListIter aIter( *pPage );
    2087           0 :         while ( aIter.IsMore() )
    2088             :         {
    2089           0 :             SdrObject* pSdrObject = aIter.Next();
    2090           0 :             FmFormObj* pFormObject = FmFormObj::GetFormObject( pSdrObject );
    2091           0 :             if ( !pFormObject )
    2092           0 :                 continue;
    2093             : 
    2094           0 :             Reference< XFormComponent > xControlModel( pFormObject->GetUnoControlModel(),UNO_QUERY );
    2095           0 :             if ( xControlModel.is() && aObjects.find(xControlModel) != aObjects.end() && bMark != (pFormView->IsObjMarked( pSdrObject ) ? 1 : 0) )
    2096             :             {
    2097             :                 // unfortunately, the writer doesn't like marking an already-marked object, again, so reset the mark first
    2098           0 :                 pFormView->MarkObj( pSdrObject, pPageView, !bMark, false );
    2099             :             }
    2100           0 :         } // while ( aIter.IsMore() )
    2101           0 :         if ( bMark )
    2102             :         {
    2103             :             // make the mark visible
    2104           0 :             ::Rectangle aMarkRect( pFormView->GetAllMarkedRect());
    2105           0 :             for ( sal_uInt32 i = 0; i < pFormView->PaintWindowCount(); ++i )
    2106             :             {
    2107           0 :                 SdrPaintWindow* pPaintWindow = pFormView->GetPaintWindow( i );
    2108           0 :                 OutputDevice& rOutDev = pPaintWindow->GetOutputDevice();
    2109           0 :                 if ( ( OUTDEV_WINDOW == rOutDev.GetOutDevType() ) && !aMarkRect.IsEmpty() )
    2110             :                 {
    2111           0 :                     pFormView->MakeVisible( aMarkRect, (Window&)rOutDev );
    2112             :                 }
    2113             :             } // for ( sal_uInt32 i = 0; i < pFormView->PaintWindowCount(); ++i )
    2114           0 :         }
    2115             :     }
    2116             : 
    2117           0 :     void NavigatorTree::CollectObjects(FmFormData* pFormData, sal_Bool bDeep, ::std::set< Reference< XFormComponent > >& _rObjects)
    2118             :     {
    2119           0 :         FmEntryDataList* pChildList = pFormData->GetChildList();
    2120             :         FmEntryData* pEntryData;
    2121             :         FmControlData* pControlData;
    2122           0 :         for( size_t i = 0; i < pChildList->size(); ++i )
    2123             :         {
    2124           0 :             pEntryData = pChildList->at( i );
    2125           0 :             if( pEntryData->ISA(FmControlData) )
    2126             :             {
    2127           0 :                 pControlData = (FmControlData*)pEntryData;
    2128           0 :                 _rObjects.insert(pControlData->GetFormComponent());
    2129             :             } // if( pEntryData->ISA(FmControlData) )
    2130           0 :             else if (bDeep && (pEntryData->ISA(FmFormData)))
    2131           0 :                 CollectObjects((FmFormData*)pEntryData,bDeep,_rObjects);
    2132             :         } // for( sal_uInt32 i=0; i<pChildList->Count(); i++ )
    2133           0 :     }
    2134             : 
    2135           0 :     void NavigatorTree::MarkViewObj( FmControlData* pControlData, sal_Bool bMarkHandles, sal_Bool bMark)
    2136             :     {
    2137           0 :         if( !pControlData )
    2138           0 :             return;
    2139           0 :         FmFormShell* pFormShell = GetNavModel()->GetFormShell();
    2140           0 :         if( !pFormShell )
    2141           0 :             return;
    2142             : 
    2143             : 
    2144             :         // In der Page das entsprechende SdrObj finden und selektieren
    2145           0 :         FmFormView*     pFormView       = pFormShell->GetFormView();
    2146           0 :         Reference< XFormComponent >  xFormComponent( pControlData->GetFormComponent());
    2147           0 :         SdrPageView*    pPageView       = pFormView->GetSdrPageView();
    2148           0 :         SdrPage*        pPage           = pPageView->GetPage();
    2149             : 
    2150           0 :         bool bPaint = false;
    2151           0 :         SdrObjListIter aIter( *pPage );
    2152           0 :         while ( aIter.IsMore() )
    2153             :         {
    2154           0 :             SdrObject* pSdrObject = aIter.Next();
    2155           0 :             FmFormObj* pFormObject = FmFormObj::GetFormObject( pSdrObject );
    2156           0 :             if ( !pFormObject )
    2157           0 :                 continue;
    2158             : 
    2159           0 :             Reference< XInterface > xControlModel( pFormObject->GetUnoControlModel() );
    2160           0 :             if ( xControlModel != xFormComponent )
    2161           0 :                 continue;
    2162             : 
    2163             :             // mark the object
    2164           0 :             if ( bMark != (pFormView->IsObjMarked( pSdrObject ) ? 1 : 0) )
    2165             :                 // unfortunately, the writer doesn't like marking an already-marked object, again, so reset the mark first
    2166           0 :                 pFormView->MarkObj( pSdrObject, pPageView, !bMark, false );
    2167             : 
    2168           0 :             if ( !bMarkHandles || !bMark )
    2169           0 :                 continue;
    2170             : 
    2171           0 :             bPaint = true;
    2172             : 
    2173           0 :         } // while ( aIter.IsMore() )
    2174           0 :         if ( bPaint )
    2175             :         {
    2176             :             // make the mark visible
    2177           0 :             ::Rectangle aMarkRect( pFormView->GetAllMarkedRect());
    2178           0 :             for ( sal_uInt32 i = 0; i < pFormView->PaintWindowCount(); ++i )
    2179             :             {
    2180           0 :                 SdrPaintWindow* pPaintWindow = pFormView->GetPaintWindow( i );
    2181           0 :                 OutputDevice& rOutDev = pPaintWindow->GetOutputDevice();
    2182           0 :                 if ( OUTDEV_WINDOW == rOutDev.GetOutDevType() )
    2183             :                 {
    2184           0 :                     pFormView->MakeVisible( aMarkRect, (Window&)rOutDev );
    2185             :                 }
    2186             :             } // for ( sal_uInt32 i = 0; i < pFormView->PaintWindowCount(); ++i )
    2187           0 :         }
    2188             :     }
    2189             : 
    2190             : 
    2191           0 : }   // namespace svxform
    2192             : 
    2193             : 
    2194             : 
    2195             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.10