/************************************************************** * * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. * *************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_vcl.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace vcl; using namespace rtl; // ======================================================================= #define TB_SEP_SIZE 8 // ----------------------------------------------------------------------- ImplToolBoxPrivateData::ImplToolBoxPrivateData() : m_pLayoutData( NULL ), mpImageListProvider( NULL ), meImageListType( vcl::IMAGELISTTYPE_UNKNOWN ) { meButtonSize = TOOLBOX_BUTTONSIZE_DONTCARE; mpMenu = new PopupMenu(); mnEventId = 0; maMenuType = TOOLBOX_MENUTYPE_NONE; maMenubuttonItem.maItemSize = Size( TB_MENUBUTTON_SIZE+TB_MENUBUTTON_OFFSET, TB_MENUBUTTON_SIZE+TB_MENUBUTTON_OFFSET ); maMenubuttonItem.meState = STATE_NOCHECK; mnMenuButtonWidth = TB_MENUBUTTON_SIZE; mbIsLocked = sal_False; mbNativeButtons = sal_False; mbIsPaintLocked = sal_False; mbAssumeDocked = sal_False; mbAssumePopupMode = sal_False; mbAssumeFloating = sal_False; mbKeyInputDisabled = sal_False; mbMenubuttonSelected = sal_False; mbPageScroll = sal_False; mbWillUsePopupMode = sal_False; mbDropDownByKeyboard = sal_False; } ImplToolBoxPrivateData::~ImplToolBoxPrivateData() { if( m_pLayoutData ) delete m_pLayoutData; delete mpMenu; } // ----------------------------------------------------------------------- ImplToolItem::ImplToolItem() { mnId = 0; mpWindow = NULL; mpUserData = NULL; meType = TOOLBOXITEM_BUTTON; mnBits = 0; meState = STATE_NOCHECK; mbEnabled = sal_True; mbVisible = sal_True; mbEmptyBtn = sal_True; mbShowWindow = sal_False; mbBreak = sal_False; mnSepSize = TB_SEP_SIZE; mnDropDownArrowWidth = TB_DROPDOWNARROWWIDTH; mnImageAngle = 0; mbMirrorMode = sal_False; mbVisibleText = sal_False; } // ----------------------------------------------------------------------- ImplToolItem::ImplToolItem( sal_uInt16 nItemId, const Image& rImage, ToolBoxItemBits nItemBits ) : maImage( rImage ) { mnId = nItemId; mpWindow = NULL; mpUserData = NULL; meType = TOOLBOXITEM_BUTTON; mnBits = nItemBits; meState = STATE_NOCHECK; mbEnabled = sal_True; mbVisible = sal_True; mbEmptyBtn = sal_False; mbShowWindow = sal_False; mbBreak = sal_False; mnSepSize = TB_SEP_SIZE; mnDropDownArrowWidth = TB_DROPDOWNARROWWIDTH; mnImageAngle = 0; mbMirrorMode = false; mbVisibleText = false; } // ----------------------------------------------------------------------- ImplToolItem::ImplToolItem( sal_uInt16 nItemId, const XubString& rText, ToolBoxItemBits nItemBits ) : maText( rText ) { mnId = nItemId; mpWindow = NULL; mpUserData = NULL; meType = TOOLBOXITEM_BUTTON; mnBits = nItemBits; meState = STATE_NOCHECK; mbEnabled = sal_True; mbVisible = sal_True; mbEmptyBtn = sal_False; mbShowWindow = sal_False; mbBreak = sal_False; mnSepSize = TB_SEP_SIZE; mnDropDownArrowWidth = TB_DROPDOWNARROWWIDTH; mnImageAngle = 0; mbMirrorMode = false; mbVisibleText = false; } // ----------------------------------------------------------------------- ImplToolItem::ImplToolItem( sal_uInt16 nItemId, const Image& rImage, const XubString& rText, ToolBoxItemBits nItemBits ) : maImage( rImage ), maText( rText ) { mnId = nItemId; mpWindow = NULL; mpUserData = NULL; meType = TOOLBOXITEM_BUTTON; mnBits = nItemBits; meState = STATE_NOCHECK; mbEnabled = sal_True; mbVisible = sal_True; mbEmptyBtn = sal_False; mbShowWindow = sal_False; mbBreak = sal_False; mnSepSize = TB_SEP_SIZE; mnDropDownArrowWidth = TB_DROPDOWNARROWWIDTH; mnImageAngle = 0; mbMirrorMode = false; mbVisibleText = false; } // ----------------------------------------------------------------------- ImplToolItem::ImplToolItem( const ImplToolItem& rItem ) : mpWindow ( rItem.mpWindow ), mpUserData ( rItem.mpUserData ), maImage ( rItem.maImage ), maHighImage ( rItem.maHighImage ), mnImageAngle ( rItem.mnImageAngle ), mbMirrorMode ( rItem.mbMirrorMode ), maText ( rItem.maText ), maQuickHelpText ( rItem.maQuickHelpText ), maHelpText ( rItem.maHelpText ), maCommandStr ( rItem.maCommandStr ), maHelpId ( rItem.maHelpId ), maRect ( rItem.maRect ), maCalcRect ( rItem.maCalcRect ), maItemSize ( rItem.maItemSize ), mnSepSize ( rItem.mnSepSize ), mnDropDownArrowWidth ( rItem.mnDropDownArrowWidth ), meType ( rItem.meType ), mnBits ( rItem.mnBits ), meState ( rItem.meState ), mnId ( rItem.mnId ), mbEnabled ( rItem.mbEnabled ), mbVisible ( rItem.mbVisible ), mbEmptyBtn ( rItem.mbEmptyBtn ), mbShowWindow ( rItem.mbShowWindow ), mbBreak ( rItem.mbBreak ), mbVisibleText ( rItem.mbVisibleText ) { } // ----------------------------------------------------------------------- ImplToolItem::~ImplToolItem() { } // ----------------------------------------------------------------------- ImplToolItem& ImplToolItem::operator=( const ImplToolItem& rItem ) { mpWindow = rItem.mpWindow; mpUserData = rItem.mpUserData; maImage = rItem.maImage; maHighImage = rItem.maHighImage; mnImageAngle = rItem.mnImageAngle; mbMirrorMode = rItem.mbMirrorMode; maText = rItem.maText; maQuickHelpText = rItem.maQuickHelpText; maHelpText = rItem.maHelpText; maCommandStr = rItem.maCommandStr; maHelpId = rItem.maHelpId; maRect = rItem.maRect; maCalcRect = rItem.maCalcRect; mnSepSize = rItem.mnSepSize; mnDropDownArrowWidth = rItem.mnDropDownArrowWidth; maItemSize = rItem.maItemSize; mbVisibleText = rItem.mbVisibleText; meType = rItem.meType; mnBits = rItem.mnBits; meState = rItem.meState; mnId = rItem.mnId; mbEnabled = rItem.mbEnabled; mbVisible = rItem.mbVisible; mbEmptyBtn = rItem.mbEmptyBtn; mbShowWindow = rItem.mbShowWindow; mbBreak = rItem.mbBreak; return *this; } // ----------------------------------------------------------------------- Size ImplToolItem::GetSize( sal_Bool bHorz, sal_Bool bCheckMaxWidth, long maxWidth, const Size& rDefaultSize ) { Size aSize( rDefaultSize ); // the size of 'standard' toolbox items // non-standard items are eg windows or buttons with text if ( (meType == TOOLBOXITEM_BUTTON) || (meType == TOOLBOXITEM_SPACE) ) { aSize = maItemSize; if ( mpWindow && bHorz ) { // get size of item window and check if it fits // no windows in vertical toolbars (the default is mbShowWindow=sal_False) Size aWinSize = mpWindow->GetSizePixel(); if ( !bCheckMaxWidth || (aWinSize.Width() <= maxWidth) ) { aSize.Width() = aWinSize.Width(); aSize.Height() = aWinSize.Height(); mbShowWindow = sal_True; } else { if ( mbEmptyBtn ) { aSize.Width() = 0; aSize.Height() = 0; } } } } else if ( meType == TOOLBOXITEM_SEPARATOR ) { if ( bHorz ) { aSize.Width() = mnSepSize; aSize.Height() = rDefaultSize.Height(); } else { aSize.Width() = rDefaultSize.Width(); aSize.Height() = mnSepSize; } } else if ( meType == TOOLBOXITEM_BREAK ) { aSize.Width() = 0; aSize.Height() = 0; } return aSize; } // ----------------------------------------------------------------------- void ImplToolItem::DetermineButtonDrawStyle( ButtonType eButtonType, sal_Bool& rbImage, sal_Bool& rbText ) const { if ( meType != TOOLBOXITEM_BUTTON ) { // no button -> draw nothing rbImage = rbText = sal_False; return; } sal_Bool bHasImage; sal_Bool bHasText; // check for image and/or text if ( !(maImage) ) bHasImage = sal_False; else bHasImage = sal_True; if ( !maText.Len() ) bHasText = sal_False; else bHasText = sal_True; // prefer images if symbolonly buttons are drawn // prefer texts if textonly buttons are dreawn if ( eButtonType == BUTTON_SYMBOL ) // drawing icons only { if( bHasImage || !bHasText ) { rbImage = sal_True; rbText = sal_False; } else { rbImage = sal_False; rbText = sal_True; } } else if ( eButtonType == BUTTON_TEXT ) // drawing text only { if( bHasText || !bHasImage ) { rbImage = sal_False; rbText = sal_True; } else { rbImage = sal_True; rbText = sal_False; } } else // drawing icons and text both { rbImage = sal_True; rbText = sal_True; } } // ----------------------------------------------------------------------- Rectangle ImplToolItem::GetDropDownRect( sal_Bool bHorz ) const { Rectangle aRect; if( (mnBits & TIB_DROPDOWN) && !maRect.IsEmpty() ) { aRect = maRect; if( mbVisibleText && !bHorz ) // item will be rotated -> place dropdown to the bottom aRect.Top() = aRect.Bottom() - mnDropDownArrowWidth; else // place dropdown to the right aRect.Left() = aRect.Right() - mnDropDownArrowWidth; } return aRect; } // ----------------------------------------------------------------------- sal_Bool ImplToolItem::IsClipped() const { return ( meType == TOOLBOXITEM_BUTTON && mbVisible && maRect.IsEmpty() ); } // ----------------------------------------------------------------------- // ----------------------------------------------------------------------- const XubString& ToolBox::ImplConvertMenuString( const XubString& rStr ) { maCvtStr = rStr; if ( mbMenuStrings ) maCvtStr.EraseTrailingChars( '.' ); maCvtStr = MnemonicGenerator::EraseAllMnemonicChars( maCvtStr ); return maCvtStr; } // ----------------------------------------------------------------------- void ToolBox::ImplInvalidate( sal_Bool bNewCalc, sal_Bool bFullPaint ) { ImplUpdateInputEnable(); if ( bNewCalc ) mbCalc = sal_True; if ( bFullPaint ) { mbFormat = sal_True; // Muss ueberhaupt eine neue Ausgabe erfolgen if ( IsReallyVisible() && IsUpdateMode() ) { Invalidate( Rectangle( mnLeftBorder, mnTopBorder, mnDX-mnRightBorder-1, mnDY-mnBottomBorder-1 ) ); maTimer.Stop(); } } else { if ( !mbFormat ) { mbFormat = sal_True; // Muss ueberhaupt eine neue Ausgabe erfolgen if ( IsReallyVisible() && IsUpdateMode() ) maTimer.Start(); } } // request new layout by layoutmanager ImplCallEventListeners( VCLEVENT_TOOLBOX_FORMATCHANGED ); } // ----------------------------------------------------------------------- void ToolBox::ImplUpdateItem( sal_uInt16 nIndex ) { // Muss ueberhaupt eine neue Ausgabe erfolgen if ( IsReallyVisible() && IsUpdateMode() ) { if ( nIndex == 0xFFFF ) { // #i52217# no immediate draw as this might lead to paint problems Invalidate( Rectangle( mnLeftBorder, mnTopBorder, mnDX-mnRightBorder-1, mnDY-mnBottomBorder-1 ) ); } else { if ( !mbFormat ) { // #i52217# no immediate draw as this might lead to paint problems Invalidate( mpData->m_aItems[nIndex].maRect ); } else maPaintRect.Union( mpData->m_aItems[nIndex].maRect ); } } } // ----------------------------------------------------------------------- void ToolBox::Click() { ImplCallEventListeners( VCLEVENT_TOOLBOX_CLICK ); maClickHdl.Call( this ); } // ----------------------------------------------------------------------- void ToolBox::DoubleClick() { ImplCallEventListeners( VCLEVENT_TOOLBOX_DOUBLECLICK ); maDoubleClickHdl.Call( this ); } // ----------------------------------------------------------------------- void ToolBox::Activate() { mnActivateCount++; ImplCallEventListeners( VCLEVENT_TOOLBOX_ACTIVATE ); maActivateHdl.Call( this ); } // ----------------------------------------------------------------------- void ToolBox::Deactivate() { mnActivateCount--; ImplCallEventListeners( VCLEVENT_TOOLBOX_DEACTIVATE ); maDeactivateHdl.Call( this ); if ( mbHideStatusText ) { GetpApp()->HideHelpStatusText(); mbHideStatusText = sal_False; } } // ----------------------------------------------------------------------- void ToolBox::Highlight() { ImplCallEventListeners( VCLEVENT_TOOLBOX_HIGHLIGHT ); maHighlightHdl.Call( this ); XubString aStr = GetHelpText( mnCurItemId ); if ( aStr.Len() || mbHideStatusText ) { GetpApp()->ShowHelpStatusText( aStr ); mbHideStatusText = sal_True; } } // ----------------------------------------------------------------------- void ToolBox::Select() { ImplDelData aDelData; ImplAddDel( &aDelData ); ImplCallEventListeners( VCLEVENT_TOOLBOX_SELECT ); maSelectHdl.Call( this ); if ( aDelData.IsDelete() ) return; ImplRemoveDel( &aDelData ); // TODO: GetFloatingWindow in DockingWindow is currently inline, change it to check dockingwrapper ImplDockingWindowWrapper *pWrapper = ImplGetDockingManager()->GetDockingWindowWrapper( this ); if( pWrapper && pWrapper->GetFloatingWindow() && pWrapper->GetFloatingWindow()->IsInPopupMode() ) pWrapper->GetFloatingWindow()->EndPopupMode(); } // ----------------------------------------------------------------------- void ToolBox::NextToolBox() { maNextToolBoxHdl.Call( this ); } // ----------------------------------------------------------------------- void ToolBox::Customize( const ToolBoxCustomizeEvent& ) { } // ----------------------------------------------------------------------- void ToolBox::UserDraw( const UserDrawEvent& ) { } // ----------------------------------------------------------------------- void ToolBox::InsertItem( const ResId& rResId, sal_uInt16 nPos ) { sal_uLong nObjMask; sal_Bool bImage = sal_False; // Wurde Image gesetzt // Item anlegen ImplToolItem aItem; GetRes( rResId.SetRT( RSC_TOOLBOXITEM ) ); nObjMask = ReadLongRes(); if ( nObjMask & RSC_TOOLBOXITEM_ID ) aItem.mnId = sal::static_int_cast(ReadLongRes()); else aItem.mnId = 1; if ( nObjMask & RSC_TOOLBOXITEM_TYPE ) aItem.meType = (ToolBoxItemType)ReadLongRes(); if ( nObjMask & RSC_TOOLBOXITEM_STATUS ) aItem.mnBits = (ToolBoxItemBits)ReadLongRes(); if( nObjMask & RSC_TOOLBOXITEM_HELPID ) aItem.maHelpId = ReadByteStringRes(); if ( nObjMask & RSC_TOOLBOXITEM_TEXT ) { aItem.maText = ReadStringRes(); aItem.maText = ImplConvertMenuString( aItem.maText ); } if ( nObjMask & RSC_TOOLBOXITEM_HELPTEXT ) aItem.maHelpText = ReadStringRes(); if ( nObjMask & RSC_TOOLBOXITEM_BITMAP ) { Bitmap aBmp = Bitmap( ResId( (RSHEADER_TYPE*)GetClassRes(), *rResId.GetResMgr() ) ); IncrementRes( GetObjSizeRes( (RSHEADER_TYPE*)GetClassRes() ) ); aItem.maImage = Image( aBmp, IMAGE_STDBTN_COLOR ); bImage = sal_True; } if ( nObjMask & RSC_TOOLBOXITEM_IMAGE ) { aItem.maImage = Image( ResId( (RSHEADER_TYPE*)GetClassRes(), *rResId.GetResMgr() ) ); IncrementRes( GetObjSizeRes( (RSHEADER_TYPE*)GetClassRes() ) ); bImage = sal_True; } if ( nObjMask & RSC_TOOLBOXITEM_DISABLE ) aItem.mbEnabled = !(sal_Bool)ReadShortRes(); if ( nObjMask & RSC_TOOLBOXITEM_STATE ) aItem.meState = (TriState)ReadLongRes(); if ( nObjMask & RSC_TOOLBOXITEM_HIDE ) aItem.mbVisible = !((sal_Bool)ReadShortRes()); if ( nObjMask & RSC_TOOLBOXITEM_COMMAND ) aItem.maCommandStr = ReadStringRes(); // Wenn kein Image geladen wurde, versuchen wir das Image aus der // Image-Liste zu holen if ( !bImage && aItem.mnId ) aItem.maImage = maImageList.GetImage( aItem.mnId ); // Wenn es sich um ein ButtonItem handelt, die ID ueberpruefen sal_Bool bNewCalc; if ( aItem.meType != TOOLBOXITEM_BUTTON ) { bNewCalc = sal_False; aItem.mnId = 0; } else { bNewCalc = sal_True; DBG_ASSERT( aItem.mnId, "ToolBox::InsertItem(): ItemId == 0" ); DBG_ASSERT( GetItemPos( aItem.mnId ) == TOOLBOX_ITEM_NOTFOUND, "ToolBox::InsertItem(): ItemId already exists" ); } // Item anlegen und in die Liste einfuegen mpData->m_aItems.insert( (nPos < mpData->m_aItems.size()) ? mpData->m_aItems.begin()+nPos : mpData->m_aItems.end(), aItem ); mpData->ImplClearLayoutData(); // ToolBox neu brechnen und neu ausgeben ImplInvalidate( bNewCalc ); // Notify sal_uInt16 nNewPos = sal::static_int_cast(( nPos == TOOLBOX_APPEND ) ? ( mpData->m_aItems.size() - 1 ) : nPos); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMADDED, reinterpret_cast< void* >( nNewPos ) ); } // ----------------------------------------------------------------------- void ToolBox::InsertItem( sal_uInt16 nItemId, const Image& rImage, ToolBoxItemBits nBits, sal_uInt16 nPos ) { DBG_ASSERT( nItemId, "ToolBox::InsertItem(): ItemId == 0" ); DBG_ASSERT( GetItemPos( nItemId ) == TOOLBOX_ITEM_NOTFOUND, "ToolBox::InsertItem(): ItemId already exists" ); // Item anlegen und in die Liste einfuegen mpData->m_aItems.insert( (nPos < mpData->m_aItems.size()) ? mpData->m_aItems.begin()+nPos : mpData->m_aItems.end(), ImplToolItem( nItemId, rImage, nBits ) ); mpData->ImplClearLayoutData(); ImplInvalidate( sal_True ); // Notify sal_uInt16 nNewPos = sal::static_int_cast(( nPos == TOOLBOX_APPEND ) ? ( mpData->m_aItems.size() - 1 ) : nPos); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMADDED, reinterpret_cast< void* >(nNewPos ) ); } // ----------------------------------------------------------------------- void ToolBox::InsertItem( sal_uInt16 nItemId, const Image& rImage, const XubString& rText, ToolBoxItemBits nBits, sal_uInt16 nPos ) { DBG_ASSERT( nItemId, "ToolBox::InsertItem(): ItemId == 0" ); DBG_ASSERT( GetItemPos( nItemId ) == TOOLBOX_ITEM_NOTFOUND, "ToolBox::InsertItem(): ItemId already exists" ); // Item anlegen und in die Liste einfuegen mpData->m_aItems.insert( (nPos < mpData->m_aItems.size()) ? mpData->m_aItems.begin()+nPos : mpData->m_aItems.end(), ImplToolItem( nItemId, rImage, ImplConvertMenuString( rText ), nBits ) ); mpData->ImplClearLayoutData(); ImplInvalidate( sal_True ); // Notify sal_uInt16 nNewPos = sal::static_int_cast(( nPos == TOOLBOX_APPEND ) ? ( mpData->m_aItems.size() - 1 ) : nPos); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMADDED, reinterpret_cast< void* >( nNewPos ) ); } // ----------------------------------------------------------------------- void ToolBox::InsertItem( sal_uInt16 nItemId, const XubString& rText, ToolBoxItemBits nBits, sal_uInt16 nPos ) { DBG_ASSERT( nItemId, "ToolBox::InsertItem(): ItemId == 0" ); DBG_ASSERT( GetItemPos( nItemId ) == TOOLBOX_ITEM_NOTFOUND, "ToolBox::InsertItem(): ItemId already exists" ); // Item anlegen und in die Liste einfuegen mpData->m_aItems.insert( (nPos < mpData->m_aItems.size()) ? mpData->m_aItems.begin()+nPos : mpData->m_aItems.end(), ImplToolItem( nItemId, ImplConvertMenuString( rText ), nBits ) ); mpData->ImplClearLayoutData(); ImplInvalidate( sal_True ); // Notify sal_uInt16 nNewPos = sal::static_int_cast(( nPos == TOOLBOX_APPEND ) ? ( mpData->m_aItems.size() - 1 ) : nPos); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMADDED, reinterpret_cast< void* >( nNewPos ) ); } // ----------------------------------------------------------------------- void ToolBox::InsertWindow( sal_uInt16 nItemId, Window* pWindow, ToolBoxItemBits nBits, sal_uInt16 nPos ) { DBG_ASSERT( nItemId, "ToolBox::InsertWindow(): ItemId == 0" ); DBG_ASSERT( GetItemPos( nItemId ) == TOOLBOX_ITEM_NOTFOUND, "ToolBox::InsertWindow(): ItemId already exists" ); // Item anlegen und in die Liste einfuegen ImplToolItem aItem; aItem.mnId = nItemId; aItem.meType = TOOLBOXITEM_BUTTON; aItem.mnBits = nBits; aItem.mpWindow = pWindow; mpData->m_aItems.insert( (nPos < mpData->m_aItems.size()) ? mpData->m_aItems.begin()+nPos : mpData->m_aItems.end(), aItem ); mpData->ImplClearLayoutData(); if ( pWindow ) pWindow->Hide(); ImplInvalidate( sal_True ); // Notify sal_uInt16 nNewPos = sal::static_int_cast(( nPos == TOOLBOX_APPEND ) ? ( mpData->m_aItems.size() - 1 ) : nPos); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMADDED, reinterpret_cast< void* >( nNewPos ) ); } // ----------------------------------------------------------------------- void ToolBox::InsertSpace( sal_uInt16 nPos ) { // Item anlegen und in die Liste einfuegen ImplToolItem aItem; aItem.meType = TOOLBOXITEM_SPACE; aItem.mbEnabled = sal_False; mpData->m_aItems.insert( (nPos < mpData->m_aItems.size()) ? mpData->m_aItems.begin()+nPos : mpData->m_aItems.end(), aItem ); mpData->ImplClearLayoutData(); ImplInvalidate( sal_False ); // Notify sal_uInt16 nNewPos = sal::static_int_cast(( nPos == TOOLBOX_APPEND ) ? ( mpData->m_aItems.size() - 1 ) : nPos); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMADDED, reinterpret_cast< void* >( nNewPos ) ); } // ----------------------------------------------------------------------- void ToolBox::InsertSeparator( sal_uInt16 nPos, sal_uInt16 nPixSize ) { // Item anlegen und in die Liste einfuegen ImplToolItem aItem; aItem.meType = TOOLBOXITEM_SEPARATOR; aItem.mbEnabled = sal_False; if ( nPixSize ) aItem.mnSepSize = nPixSize; mpData->m_aItems.insert( (nPos < mpData->m_aItems.size()) ? mpData->m_aItems.begin()+nPos : mpData->m_aItems.end(), aItem ); mpData->ImplClearLayoutData(); ImplInvalidate( sal_False ); // Notify sal_uInt16 nNewPos = sal::static_int_cast(( nPos == TOOLBOX_APPEND ) ? ( mpData->m_aItems.size() - 1 ) : nPos); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMADDED, reinterpret_cast< void* >( nNewPos ) ); } // ----------------------------------------------------------------------- void ToolBox::InsertBreak( sal_uInt16 nPos ) { // Item anlegen und in die Liste einfuegen ImplToolItem aItem; aItem.meType = TOOLBOXITEM_BREAK; aItem.mbEnabled = sal_False; mpData->m_aItems.insert( (nPos < mpData->m_aItems.size()) ? mpData->m_aItems.begin()+nPos : mpData->m_aItems.end(), aItem ); mpData->ImplClearLayoutData(); ImplInvalidate( sal_False ); // Notify sal_uInt16 nNewPos = sal::static_int_cast(( nPos == TOOLBOX_APPEND ) ? ( mpData->m_aItems.size() - 1 ) : nPos); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMADDED, reinterpret_cast< void* >( nNewPos ) ); } // ----------------------------------------------------------------------- void ToolBox::RemoveItem( sal_uInt16 nPos ) { if( nPos < mpData->m_aItems.size() ) { sal_Bool bMustCalc; if ( mpData->m_aItems[nPos].meType == TOOLBOXITEM_BUTTON ) bMustCalc = sal_True; else bMustCalc = sal_False; if ( mpData->m_aItems[nPos].mpWindow ) mpData->m_aItems[nPos].mpWindow->Hide(); // PaintRect um das removete Item erweitern maPaintRect.Union( mpData->m_aItems[nPos].maRect ); // Absichern gegen das Loeschen im Select-Handler if ( mpData->m_aItems[nPos].mnId == mnCurItemId ) mnCurItemId = 0; if ( mpData->m_aItems[nPos].mnId == mnHighItemId ) mnHighItemId = 0; ImplInvalidate( bMustCalc ); mpData->m_aItems.erase( mpData->m_aItems.begin()+nPos ); mpData->ImplClearLayoutData(); // Notify ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMREMOVED, reinterpret_cast< void* >( nPos ) ); } } // ----------------------------------------------------------------------- void ToolBox::MoveItem( sal_uInt16 nItemId, sal_uInt16 nNewPos ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos == nNewPos ) return; if ( nPos < nNewPos ) nNewPos--; // Existiert Item if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { // ToolBox-Item in der Liste verschieben ImplToolItem aItem = mpData->m_aItems[nPos]; mpData->m_aItems.erase( mpData->m_aItems.begin()+nPos ); mpData->m_aItems.insert( (nNewPos < mpData->m_aItems.size()) ? mpData->m_aItems.begin()+nNewPos : mpData->m_aItems.end(), aItem ); mpData->ImplClearLayoutData(); // ToolBox neu ausgeben ImplInvalidate( sal_False ); // Notify if( nPos < nNewPos ) // only send one event, all indices above this item are invalid anyway ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMREMOVED, reinterpret_cast< void* >( nPos ) ); else { sal_uInt16 nNewPos2 = sal::static_int_cast(( nNewPos == TOOLBOX_APPEND ) ? ( mpData->m_aItems.size() - 1 ) : nNewPos); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMADDED, reinterpret_cast< void* >( nNewPos2 ) ); } } } // ----------------------------------------------------------------------- void ToolBox::CopyItem( const ToolBox& rToolBox, sal_uInt16 nItemId, sal_uInt16 nNewPos ) { DBG_ASSERT( GetItemPos( nItemId ) == TOOLBOX_ITEM_NOTFOUND, "ToolBox::CopyItem(): ItemId already exists" ); sal_uInt16 nPos = rToolBox.GetItemPos( nItemId ); // Existiert Item if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { // ToolBox-Item in der Liste verschieben ImplToolItem aNewItem = rToolBox.mpData->m_aItems[nPos]; // Bestimme Daten zuruecksetzen aNewItem.mpWindow = NULL; aNewItem.mbShowWindow = sal_False; mpData->m_aItems.insert( (nNewPos < mpData->m_aItems.size()) ? mpData->m_aItems.begin()+nNewPos : mpData->m_aItems.end(), aNewItem ); mpData->ImplClearLayoutData(); // ToolBox neu ausgeben ImplInvalidate( sal_False ); // Notify sal_uInt16 nNewPos2 = sal::static_int_cast(( nNewPos == TOOLBOX_APPEND ) ? ( mpData->m_aItems.size() - 1 ) : nNewPos); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMADDED, reinterpret_cast< void* >( nNewPos2 ) ); } } // ----------------------------------------------------------------------- void ToolBox::CopyItems( const ToolBox& rToolBox ) { mpData->ImplClearLayoutData(); mpData->m_aItems = rToolBox.mpData->m_aItems; // Absichern gegen das Loeschen im Select-Handler mnCurItemId = 0; mnHighItemId = 0; for( std::vector< ImplToolItem >::iterator it = mpData->m_aItems.begin(); it != mpData->m_aItems.end(); ++it ) { it->mpWindow = NULL; it->mbShowWindow = sal_False; } ImplInvalidate( sal_True, sal_True ); // Notify ImplCallEventListeners( VCLEVENT_TOOLBOX_ALLITEMSCHANGED ); } // ----------------------------------------------------------------------- void ToolBox::Clear() { mpData->m_aItems.clear(); mpData->ImplClearLayoutData(); // Absichern gegen das Loeschen im Select-Handler mnCurItemId = 0; mnHighItemId = 0; ImplInvalidate( sal_True, sal_True ); // Notify ImplCallEventListeners( VCLEVENT_TOOLBOX_ALLITEMSCHANGED ); } // ----------------------------------------------------------------------- void ToolBox::SetButtonType( ButtonType eNewType ) { if ( meButtonType != eNewType ) { meButtonType = eNewType; // Hier besser alles neu ausgeben, da es ansonsten zu Problemen // mit den per CopyBits kopierten Bereichen geben kann ImplInvalidate( sal_True ); } } // ----------------------------------------------------------------------- void ToolBox::SetToolboxButtonSize( ToolBoxButtonSize eSize ) { if( mpData->meButtonSize != eSize ) { mpData->meButtonSize = eSize; mbCalc = sal_True; mbFormat = sal_True; } } ToolBoxButtonSize ToolBox::GetToolboxButtonSize() const { return mpData->meButtonSize; } // ----------------------------------------------------------------------- const Size& ToolBox::GetDefaultImageSize() const { static Size aSmallButtonSize( TB_SMALLIMAGESIZE, TB_SMALLIMAGESIZE ); static sal_uLong s_nSymbolsStyle = STYLE_SYMBOLS_DEFAULT; static Size aLargeButtonSize( TB_LARGEIMAGESIZE, TB_LARGEIMAGESIZE ); sal_uLong nSymbolsStyle = Application::GetSettings().GetStyleSettings().GetCurrentSymbolsStyle(); if ( s_nSymbolsStyle != nSymbolsStyle ) { s_nSymbolsStyle = nSymbolsStyle; switch ( nSymbolsStyle ) { case STYLE_SYMBOLS_INDUSTRIAL: aLargeButtonSize = Size( TB_LARGEIMAGESIZE_INDUSTRIAL, TB_LARGEIMAGESIZE_INDUSTRIAL ); break; case STYLE_SYMBOLS_CRYSTAL: aLargeButtonSize = Size( TB_LARGEIMAGESIZE_CRYSTAL, TB_LARGEIMAGESIZE_CRYSTAL ); break; default: aLargeButtonSize = Size( TB_LARGEIMAGESIZE, TB_LARGEIMAGESIZE ); } } return GetToolboxButtonSize() == TOOLBOX_BUTTONSIZE_LARGE ? aLargeButtonSize : aSmallButtonSize; } // ----------------------------------------------------------------------- void ToolBox::SetAlign( WindowAlign eNewAlign ) { if ( meAlign != eNewAlign ) { meAlign = eNewAlign; if ( !ImplIsFloatingMode() ) { // Setzen, ob Items horizontal oder vertikal angeordnet werden sollen if ( (eNewAlign == WINDOWALIGN_LEFT) || (eNewAlign == WINDOWALIGN_RIGHT) ) mbHorz = sal_False; else mbHorz = sal_True; // Hier alles neu ausgeben, da sich Border auch aendert mbCalc = sal_True; mbFormat = sal_True; if ( IsReallyVisible() && IsUpdateMode() ) Invalidate(); } } } // ----------------------------------------------------------------------- void ToolBox::SetLineCount( sal_uInt16 nNewLines ) { if ( !nNewLines ) nNewLines = 1; if ( mnLines != nNewLines ) { mnLines = nNewLines; // Hier besser alles neu ausgeben, da es ansonsten zu Problemen // mit den per CopyBits kopierten Bereichen geben kann ImplInvalidate( sal_False ); } } // ----------------------------------------------------------------------- void ToolBox::SetPageScroll( sal_Bool b ) { mpData->mbPageScroll = b; } sal_Bool ToolBox::GetPageScroll() { return mpData->mbPageScroll; } // ----------------------------------------------------------------------- void ToolBox::SetNextToolBox( const XubString& rStr ) { sal_Bool bCalcNew = (!maNextToolBoxStr.Len() != !rStr.Len()); maNextToolBoxStr = rStr; if ( bCalcNew ) ImplInvalidate( sal_True, sal_False ); } // ----------------------------------------------------------------------- sal_uInt16 ToolBox::GetItemCount() const { return (sal_uInt16)mpData->m_aItems.size(); } // ----------------------------------------------------------------------- ToolBoxItemType ToolBox::GetItemType( sal_uInt16 nPos ) const { return (nPos < mpData->m_aItems.size()) ? mpData->m_aItems[nPos].meType : TOOLBOXITEM_DONTKNOW; } // ----------------------------------------------------------------------- sal_uInt16 ToolBox::GetItemPos( sal_uInt16 nItemId ) const { int nCount = mpData->m_aItems.size(); for( int nPos = 0; nPos < nCount; nPos++ ) if( mpData->m_aItems[nPos].mnId == nItemId ) return (sal_uInt16)nPos; return TOOLBOX_ITEM_NOTFOUND; } // ----------------------------------------------------------------------- sal_uInt16 ToolBox::GetItemPos( const Point& rPos ) const { // search the item position on the given point sal_uInt16 nRet = TOOLBOX_ITEM_NOTFOUND; sal_uInt16 nPos = 0; std::vector< ImplToolItem >::const_iterator it = mpData->m_aItems.begin(); while( it != mpData->m_aItems.end() ) { if ( it->maRect.IsInside( rPos ) ) { // item found -> save position and break nRet = nPos; break; } ++it; ++nPos; } return nRet; } // ----------------------------------------------------------------------- sal_uInt16 ToolBox::GetItemId( sal_uInt16 nPos ) const { return (nPos < mpData->m_aItems.size()) ? mpData->m_aItems[nPos].mnId : 0; } // ----------------------------------------------------------------------- sal_uInt16 ToolBox::GetItemId( const Point& rPos ) const { // Item suchen, das geklickt wurde std::vector< ImplToolItem >::const_iterator it = mpData->m_aItems.begin(); while( it != mpData->m_aItems.end() ) { // Ist es dieses Item if ( it->maRect.IsInside( rPos ) ) { if ( it->meType == TOOLBOXITEM_BUTTON ) return it->mnId; else return 0; } ++it; } return 0; } // ----------------------------------------------------------------------- Point ToolBox::ImplGetPopupPosition( const Rectangle& rRect, const Size& rSize ) const { Point aPos; if( !rRect.IsEmpty() ) { Rectangle aScreen = GetDesktopRectPixel(); // the popup should be positioned so that it will not cover // the item rect and that it fits the desktop // the preferred direction is always towards the center of // the application window Point devPos; // the position in device coordinates for screen comparison switch( meAlign ) { case WINDOWALIGN_TOP: aPos = rRect.BottomLeft(); aPos.Y()++; devPos = OutputToAbsoluteScreenPixel( aPos ); if( devPos.Y() + rSize.Height() >= aScreen.Bottom() ) aPos.Y() = rRect.Top() - rSize.Height(); break; case WINDOWALIGN_BOTTOM: aPos = rRect.TopLeft(); aPos.Y()--; devPos = OutputToAbsoluteScreenPixel( aPos ); if( devPos.Y() - rSize.Height() > aScreen.Top() ) aPos.Y() -= rSize.Height(); else aPos.Y() = rRect.Bottom(); break; case WINDOWALIGN_LEFT: aPos = rRect.TopRight(); aPos.X()++; devPos = OutputToAbsoluteScreenPixel( aPos ); if( devPos.X() + rSize.Width() >= aScreen.Right() ) aPos.X() = rRect.Left() - rSize.Width(); break; case WINDOWALIGN_RIGHT: aPos = rRect.TopLeft(); aPos.X()--; devPos = OutputToAbsoluteScreenPixel( aPos ); if( devPos.X() - rSize.Width() > aScreen.Left() ) aPos.X() -= rSize.Width(); else aPos.X() = rRect.Right(); break; default: break; }; } return aPos; } Point ToolBox::GetItemPopupPosition( sal_uInt16 nItemId, const Size& rSize ) const { return ImplGetPopupPosition( GetItemRect( nItemId ), rSize ); } // ----------------------------------------------------------------------- Rectangle ToolBox::GetItemRect( sal_uInt16 nItemId ) const { if ( mbCalc || mbFormat ) ((ToolBox*)this)->ImplFormat(); sal_uInt16 nPos = GetItemPos( nItemId ); return GetItemPosRect( nPos ); } // ----------------------------------------------------------------------- Rectangle ToolBox::GetItemPosRect( sal_uInt16 nPos ) const { if ( mbCalc || mbFormat ) ((ToolBox*)this)->ImplFormat(); if ( nPos < mpData->m_aItems.size() ) return mpData->m_aItems[nPos].maRect; else return Rectangle(); } // ----------------------------------------------------------------------- Rectangle ToolBox::GetItemDropDownRect( sal_uInt16 nItemId ) const { if ( mbCalc || mbFormat ) ((ToolBox*)this)->ImplFormat(); sal_uInt16 nPos = GetItemPos( nItemId ); return GetItemPosDropDownRect( nPos ); } // ----------------------------------------------------------------------- Rectangle ToolBox::GetItemPosDropDownRect( sal_uInt16 nPos ) const { if ( mbCalc || mbFormat ) ((ToolBox*)this)->ImplFormat(); if ( nPos < mpData->m_aItems.size() ) return mpData->m_aItems[nPos].GetDropDownRect( mbHorz ); else return Rectangle(); } // ----------------------------------------------------------------------- Rectangle ToolBox::GetMenubuttonRect() const { return mpData->maMenubuttonItem.maRect; } sal_Bool ToolBox::ImplHasExternalMenubutton() { // check if the borderwindow (i.e. the decoration) provides the menu button sal_Bool bRet = sal_False; if( ImplIsFloatingMode() ) { // custom menu is placed in the decoration ImplBorderWindow *pBorderWin = dynamic_cast( GetWindow( WINDOW_BORDER ) ); if( pBorderWin && !pBorderWin->GetMenuRect().IsEmpty() ) bRet = sal_True; } return bRet; } // ----------------------------------------------------------------------- void ToolBox::SetItemBits( sal_uInt16 nItemId, ToolBoxItemBits nBits ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos < mpData->m_aItems.size() ) { ToolBoxItemBits nOldBits = mpData->m_aItems[nPos].mnBits; mpData->m_aItems[nPos].mnBits = nBits; nBits &= TIB_LEFT | TIB_AUTOSIZE | TIB_DROPDOWN; nOldBits &= TIB_LEFT | TIB_AUTOSIZE | TIB_DROPDOWN; // trigger reformat when the item width has changed (dropdown arrow) sal_Bool bFormat = (nBits & TIB_DROPDOWN) != (nOldBits & TIB_DROPDOWN); if ( nBits != nOldBits ) ImplInvalidate( sal_True, bFormat ); } } // ----------------------------------------------------------------------- ToolBoxItemBits ToolBox::GetItemBits( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->mnBits; else return 0; } // ----------------------------------------------------------------------- void ToolBox::SetItemData( sal_uInt16 nItemId, void* pNewData ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos < mpData->m_aItems.size() ) { mpData->m_aItems[nPos].mpUserData = pNewData; ImplUpdateItem( nPos ); } } // ----------------------------------------------------------------------- void* ToolBox::GetItemData( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->mpUserData; else return NULL; } // ----------------------------------------------------------------------- void ToolBox::SetItemImage( sal_uInt16 nItemId, const Image& rImage ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { ImplToolItem* pItem = &mpData->m_aItems[nPos]; // Nur wenn alles berechnet ist, mehr Aufwand treiben if ( !mbCalc ) { Size aOldSize = pItem->maImage.GetSizePixel(); pItem->maImage = rImage; if ( aOldSize != pItem->maImage.GetSizePixel() ) ImplInvalidate( sal_True ); else ImplUpdateItem( nPos ); } else pItem->maImage = rImage; } } // ----------------------------------------------------------------------- void ToolBox::SetImageList( const ImageList& rImageList ) { maImageList = rImageList; sal_uInt16 nCount = (sal_uInt16)mpData->m_aItems.size(); for( sal_uInt16 i = 0; i < nCount; i++ ) { Image aImage; if ( mpData->m_aItems[i].mnId ) aImage = maImageList.GetImage( mpData->m_aItems[i].mnId ); if( !!aImage ) SetItemImage( mpData->m_aItems[i].mnId, aImage ); } } // ----------------------------------------------------------------------- static Image ImplRotImage( const Image& rImage, long nAngle10 ) { Image aRet; BitmapEx aRotBitmapEx( rImage.GetBitmapEx() ); aRotBitmapEx.Rotate( nAngle10, Color( COL_WHITE ) ); return Image( aRotBitmapEx ); } void ToolBox::SetItemImageAngle( sal_uInt16 nItemId, long nAngle10 ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { ImplToolItem* pItem = &mpData->m_aItems[nPos]; Size aOldSize = pItem->maImage.GetSizePixel(); long nDeltaAngle = (nAngle10 - pItem->mnImageAngle) % 3600; while( nDeltaAngle < 0 ) nDeltaAngle += 3600; pItem->mnImageAngle = nAngle10; if( nDeltaAngle && !!pItem->maImage ) { pItem->maImage = ImplRotImage( pItem->maImage, nDeltaAngle ); if( !!pItem->maHighImage ) pItem->maHighImage = ImplRotImage( pItem->maHighImage, nDeltaAngle ); } if ( !mbCalc ) { if ( aOldSize != pItem->maImage.GetSizePixel() ) ImplInvalidate( sal_True ); else ImplUpdateItem( nPos ); } } } // ----------------------------------------------------------------------- static Image ImplMirrorImage( const Image& rImage ) { Image aRet; BitmapEx aMirrBitmapEx( rImage.GetBitmapEx() ); aMirrBitmapEx.Mirror( BMP_MIRROR_HORZ ); return Image( aMirrBitmapEx ); } void ToolBox::SetItemImageMirrorMode( sal_uInt16 nItemId, sal_Bool bMirror ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { ImplToolItem* pItem = &mpData->m_aItems[nPos]; if( ( pItem->mbMirrorMode && ! bMirror ) || ( ! pItem->mbMirrorMode && bMirror ) ) { pItem->mbMirrorMode = bMirror ? true : false; if( !!pItem->maImage ) { pItem->maImage = ImplMirrorImage( pItem->maImage ); if( !!pItem->maHighImage ) pItem->maHighImage = ImplMirrorImage( pItem->maHighImage ); } if ( !mbCalc ) ImplUpdateItem( nPos ); } } } // ----------------------------------------------------------------------- Image ToolBox::GetItemImage( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->maImage; else return Image(); } // ----------------------------------------------------------------------- long ToolBox::GetItemImageAngle( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->mnImageAngle; else return 0; } // ----------------------------------------------------------------------- sal_Bool ToolBox::GetItemImageMirrorMode( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->mbMirrorMode; else return sal_False; } // ----------------------------------------------------------------------- void ToolBox::SetItemHighImage( sal_uInt16 nItemId, const Image& rImage ) { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) { DBG_ASSERT( (pItem->maImage.GetSizePixel() == rImage.GetSizePixel()) || ((!rImage) == sal_True), "ToolBox::SetItemHighImage() - ImageSize != HighImageSize" ); pItem->maHighImage = rImage; } } // ----------------------------------------------------------------------- Image ToolBox::GetItemHighImage( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->maHighImage; else return Image(); } // ----------------------------------------------------------------------- void ToolBox::SetItemText( sal_uInt16 nItemId, const XubString& rText ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { ImplToolItem* pItem = &mpData->m_aItems[nPos]; // Nur wenn alles berechnet ist, mehr Aufwand treiben if ( !mbCalc && ((meButtonType != BUTTON_SYMBOL) || !pItem->maImage) ) { long nOldWidth = GetCtrlTextWidth( pItem->maText ); pItem->maText = ImplConvertMenuString( rText ); mpData->ImplClearLayoutData(); if ( nOldWidth != GetCtrlTextWidth( pItem->maText ) ) ImplInvalidate( sal_True ); else ImplUpdateItem( nPos ); } else pItem->maText = ImplConvertMenuString( rText ); // Notify button changed event to prepare accessibility bridge ImplCallEventListeners( VCLEVENT_TOOLBOX_BUTTONSTATECHANGED, reinterpret_cast< void* >( nPos ) ); // Notify ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMTEXTCHANGED, reinterpret_cast< void* >( nPos ) ); } } // ----------------------------------------------------------------------- const XubString& ToolBox::GetItemText( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->maText; else return ImplGetSVEmptyStr(); } // ----------------------------------------------------------------------- void ToolBox::SetItemWindow( sal_uInt16 nItemId, Window* pNewWindow ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { ImplToolItem* pItem = &mpData->m_aItems[nPos]; pItem->mpWindow = pNewWindow; if ( pNewWindow ) pNewWindow->Hide(); ImplInvalidate( sal_True ); ImplCallEventListeners( VCLEVENT_TOOLBOX_ITEMWINDOWCHANGED, reinterpret_cast< void* >( nPos ) ); } } // ----------------------------------------------------------------------- Window* ToolBox::GetItemWindow( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->mpWindow; else return NULL; } // ----------------------------------------------------------------------- void ToolBox::StartSelection() { if ( mbDrag ) EndSelection(); if ( !mbSelection ) { mbSelection = sal_True; mnCurPos = TOOLBOX_ITEM_NOTFOUND; mnCurItemId = 0; Activate(); } } // ----------------------------------------------------------------------- void ToolBox::EndSelection() { mbCommandDrag = sal_False; if ( mbDrag || mbSelection ) { // Daten zuruecksetzen mbDrag = sal_False; mbSelection = sal_False; if ( mnCurPos != TOOLBOX_ITEM_NOTFOUND ) ImplDrawItem( mnCurPos ); EndTracking(); ReleaseMouse(); Deactivate(); } mnCurPos = TOOLBOX_ITEM_NOTFOUND; mnCurItemId = 0; mnDownItemId = 0; mnMouseClicks = 0; mnMouseModifier = 0; } // ----------------------------------------------------------------------- void ToolBox::SetItemDown( sal_uInt16 nItemId, sal_Bool bDown, sal_Bool bRelease ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { if ( bDown ) { if ( nPos != mnCurPos ) { mnCurPos = nPos; ImplDrawItem( mnCurPos, sal_True ); Flush(); } } else { if ( nPos == mnCurPos ) { ImplDrawItem( mnCurPos, sal_False ); Flush(); mnCurPos = TOOLBOX_ITEM_NOTFOUND; } } if ( bRelease ) { if ( mbDrag || mbSelection ) { mbDrag = sal_False; mbSelection = sal_False; EndTracking(); ReleaseMouse(); Deactivate(); } mnCurItemId = 0; mnDownItemId = 0; mnMouseClicks = 0; mnMouseModifier = 0; } } } // ----------------------------------------------------------------------- sal_Bool ToolBox::IsItemDown( sal_uInt16 nItemId ) const { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos != TOOLBOX_ITEM_NOTFOUND ) return (nPos == mnCurPos); else return sal_False; } // ----------------------------------------------------------------------- void ToolBox::SetItemState( sal_uInt16 nItemId, TriState eState ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { ImplToolItem* pItem = &mpData->m_aItems[nPos]; // Hat sich der Status geaendert if ( pItem->meState != eState ) { // Wenn RadioCheck, dann vorherigen unchecken if ( (eState == STATE_CHECK) && (pItem->mnBits & TIB_AUTOCHECK) && (pItem->mnBits & TIB_RADIOCHECK) ) { ImplToolItem* pGroupItem; sal_uInt16 nGroupPos; sal_uInt16 nItemCount = GetItemCount(); nGroupPos = nPos; while ( nGroupPos ) { pGroupItem = &mpData->m_aItems[nGroupPos-1]; if ( pGroupItem->mnBits & TIB_RADIOCHECK ) { if ( pGroupItem->meState != STATE_NOCHECK ) SetItemState( pGroupItem->mnId, STATE_NOCHECK ); } else break; nGroupPos--; } nGroupPos = nPos+1; while ( nGroupPos < nItemCount ) { pGroupItem = &mpData->m_aItems[nGroupPos]; if ( pGroupItem->mnBits & TIB_RADIOCHECK ) { if ( pGroupItem->meState != STATE_NOCHECK ) SetItemState( pGroupItem->mnId, STATE_NOCHECK ); } else break; nGroupPos++; } } pItem->meState = eState; ImplUpdateItem( nPos ); // Notify button changed event to prepare accessibility bridge ImplCallEventListeners( VCLEVENT_TOOLBOX_BUTTONSTATECHANGED, reinterpret_cast< void* >( nPos ) ); // Notify ImplCallEventListeners( VCLEVENT_TOOLBOX_CLICK, reinterpret_cast< void* >( nPos ) ); } } } // ----------------------------------------------------------------------- TriState ToolBox::GetItemState( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->meState; else return STATE_NOCHECK; } // ----------------------------------------------------------------------- void ToolBox::EnableItem( sal_uInt16 nItemId, sal_Bool bEnable ) { sal_uInt16 nPos = GetItemPos( nItemId ); if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { ImplToolItem* pItem = &mpData->m_aItems[nPos]; if ( bEnable ) bEnable = sal_True; if ( pItem->mbEnabled != bEnable ) { pItem->mbEnabled = bEnable; // Gegebenenfalls das Fenster mit updaten if ( pItem->mpWindow ) pItem->mpWindow->Enable( pItem->mbEnabled ); // Item updaten ImplUpdateItem( nPos ); ImplUpdateInputEnable(); // Notify button changed event to prepare accessibility bridge ImplCallEventListeners( VCLEVENT_TOOLBOX_BUTTONSTATECHANGED, reinterpret_cast< void* >( nPos ) ); ImplCallEventListeners( bEnable ? VCLEVENT_TOOLBOX_ITEMENABLED : VCLEVENT_TOOLBOX_ITEMDISABLED, reinterpret_cast< void* >( nPos ) ); } } } // ----------------------------------------------------------------------- sal_Bool ToolBox::IsItemEnabled( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->mbEnabled; else return sal_False; } // ----------------------------------------------------------------------- void ToolBox::ShowItem( sal_uInt16 nItemId, sal_Bool bVisible ) { sal_uInt16 nPos = GetItemPos( nItemId ); mpData->ImplClearLayoutData(); if ( nPos != TOOLBOX_ITEM_NOTFOUND ) { ImplToolItem* pItem = &mpData->m_aItems[nPos]; if ( pItem->mbVisible != bVisible ) { pItem->mbVisible = bVisible; ImplInvalidate( sal_False ); } } } // ----------------------------------------------------------------------- sal_Bool ToolBox::IsItemVisible( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->mbVisible; else return sal_False; } // ----------------------------------------------------------------------- sal_Bool ToolBox::IsItemReallyVisible( sal_uInt16 nItemId ) const { // is the item on the visible area of the toolbox? sal_Bool bRet = sal_False; Rectangle aRect( mnLeftBorder, mnTopBorder, mnDX-mnRightBorder, mnDY-mnBottomBorder ); ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem && pItem->mbVisible && !pItem->maRect.IsEmpty() && aRect.IsOver( pItem->maRect ) ) { bRet = sal_True; } return bRet; } // ----------------------------------------------------------------------- void ToolBox::SetItemCommand( sal_uInt16 nItemId, const XubString& rCommand ) { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) pItem->maCommandStr = rCommand; } // ----------------------------------------------------------------------- const XubString& ToolBox::GetItemCommand( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->maCommandStr; else return ImplGetSVEmptyStr(); } // ----------------------------------------------------------------------- void ToolBox::SetQuickHelpText( sal_uInt16 nItemId, const XubString& rText ) { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) pItem->maQuickHelpText = rText; } // ----------------------------------------------------------------------- const XubString& ToolBox::GetQuickHelpText( sal_uInt16 nItemId ) const { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) return pItem->maQuickHelpText; else return ImplGetSVEmptyStr(); } // ----------------------------------------------------------------------- void ToolBox::SetHelpText( sal_uInt16 nItemId, const XubString& rText ) { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) pItem->maHelpText = rText; } // ----------------------------------------------------------------------- const XubString& ToolBox::GetHelpText( sal_uInt16 nItemId ) const { return ImplGetHelpText( nItemId ); } // ----------------------------------------------------------------------- void ToolBox::SetHelpId( sal_uInt16 nItemId, const rtl::OString& rHelpId ) { ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) pItem->maHelpId = rHelpId; } // ----------------------------------------------------------------------- rtl::OString ToolBox::GetHelpId( sal_uInt16 nItemId ) const { rtl::OString aRet; ImplToolItem* pItem = ImplGetItem( nItemId ); if ( pItem ) { if ( pItem->maHelpId.getLength() ) aRet = pItem->maHelpId; else aRet = ::rtl::OUStringToOString( pItem->maCommandStr, RTL_TEXTENCODING_UTF8 ); } return aRet; } // ----------------------------------------------------------------------- void ToolBox::SetBorder( long nX, long nY ) { mnBorderX = nX; mnBorderY = nY; ImplInvalidate( sal_True, sal_True ); } // ----------------------------------------------------------------------- void ToolBox::SetOutStyle( sal_uInt16 nNewStyle ) { // always force flat looking toolbars since NWF nNewStyle |= TOOLBOX_STYLE_FLAT; if ( mnOutStyle != nNewStyle ) { mnOutStyle = nNewStyle; ImplDisableFlatButtons(); // Damit das ButtonDevice neu angelegt wird if ( !(mnOutStyle & TOOLBOX_STYLE_FLAT) ) { mnMaxItemWidth = 1; mnMaxItemHeight = 1; } ImplInvalidate( sal_True, sal_True ); } } // ----------------------------------------------------------------------- void ToolBox::RecalcItems() { ImplInvalidate( sal_True ); } // ----------------------------------------------------------------------- // disable key input if all items are disabled void ToolBox::ImplUpdateInputEnable() { for( std::vector< ImplToolItem >::const_iterator it = mpData->m_aItems.begin(); it != mpData->m_aItems.end(); ++it ) { if( it->mbEnabled ) { // at least one useful entry mpData->mbKeyInputDisabled = sal_False; return; } } mpData->mbKeyInputDisabled = sal_True; } // ----------------------------------------------------------------------- void ToolBox::ImplFillLayoutData() const { mpData->m_pLayoutData = new ToolBoxLayoutData(); sal_uInt16 nCount = (sal_uInt16)mpData->m_aItems.size(); for( sal_uInt16 i = 0; i < nCount; i++ ) { ImplToolItem* pItem = &mpData->m_aItems[i]; // Nur malen, wenn Rechteck im PaintRectangle liegt if ( !pItem->maRect.IsEmpty() ) const_cast(this)->ImplDrawItem( i, sal_False, sal_False, sal_True ); } } // ----------------------------------------------------------------------- String ToolBox::GetDisplayText() const { if( ! mpData->m_pLayoutData ) ImplFillLayoutData(); return mpData->m_pLayoutData ? mpData->m_pLayoutData->m_aDisplayText : String(); } // ----------------------------------------------------------------------- Rectangle ToolBox::GetCharacterBounds( sal_uInt16 nItemID, long nIndex ) const { long nItemIndex = -1; if( ! mpData->m_pLayoutData ) ImplFillLayoutData(); if( mpData->m_pLayoutData ) { for( sal_uLong i = 0; i < mpData->m_pLayoutData->m_aLineItemIds.size(); i++ ) { if( mpData->m_pLayoutData->m_aLineItemIds[i] == nItemID ) { nItemIndex = mpData->m_pLayoutData->m_aLineIndices[i]; break; } } } return (mpData->m_pLayoutData && nItemIndex != -1) ? mpData->m_pLayoutData->GetCharacterBounds( nItemIndex+nIndex ) : Rectangle(); } // ----------------------------------------------------------------------- long ToolBox::GetIndexForPoint( const Point& rPoint, sal_uInt16& rItemID ) const { long nIndex = -1; rItemID = 0; if( ! mpData->m_pLayoutData ) ImplFillLayoutData(); if( mpData->m_pLayoutData ) { nIndex = mpData->m_pLayoutData->GetIndexForPoint( rPoint ); for( sal_uLong i = 0; i < mpData->m_pLayoutData->m_aLineIndices.size(); i++ ) { if( mpData->m_pLayoutData->m_aLineIndices[i] <= nIndex && (i == mpData->m_pLayoutData->m_aLineIndices.size()-1 || mpData->m_pLayoutData->m_aLineIndices[i+1] > nIndex) ) { rItemID = mpData->m_pLayoutData->m_aLineItemIds[i]; break; } } } return nIndex; } // ----------------------------------------------------------------------- long ToolBox::GetTextCount() const { if( ! mpData->m_pLayoutData ) ImplFillLayoutData(); return mpData->m_pLayoutData ? mpData->m_pLayoutData->GetLineCount() : 0; } // ----------------------------------------------------------------------- Pair ToolBox::GetTextStartEnd( long nText ) const { if( ! mpData->m_pLayoutData ) ImplFillLayoutData(); return mpData->m_pLayoutData ? mpData->m_pLayoutData->GetLineStartEnd( nText ) : Pair( -1, -1 ); } // ----------------------------------------------------------------------- sal_uInt16 ToolBox::GetDisplayItemId( long nText ) const { sal_uInt16 nItemId = 0; if( ! mpData->m_pLayoutData ) ImplFillLayoutData(); if( mpData->m_pLayoutData && nText >= 0 && (sal_uLong)nText < mpData->m_pLayoutData->m_aLineItemIds.size() ) nItemId = mpData->m_pLayoutData->m_aLineItemIds[nText]; return nItemId; } // ----------------------------------------------------------------------- void ToolBox::SetDropdownClickHdl( const Link& rLink ) { mpData->maDropdownClickHdl = rLink; } const Link& ToolBox::GetDropdownClickHdl() const { return mpData->maDropdownClickHdl; } // ----------------------------------------------------------------------- void ToolBox::SetMenuType( sal_uInt16 aType ) { if( aType != mpData->maMenuType ) { mpData->maMenuType = aType; if( IsFloatingMode() ) { // the menu button may have to be moved into the decoration which changes the layout ImplDockingWindowWrapper *pWrapper = ImplGetDockingManager()->GetDockingWindowWrapper( this ); if( pWrapper ) pWrapper->ShowTitleButton( TITLE_BUTTON_MENU, ( aType & TOOLBOX_MENUTYPE_CUSTOMIZE) ? sal_True : sal_False ); mbFormat = sal_True; ImplFormat(); ImplSetMinMaxFloatSize( this ); } else { // trigger redraw of menu button if( !mpData->maMenubuttonItem.maRect.IsEmpty() ) Invalidate(mpData->maMenubuttonItem.maRect); } } } sal_uInt16 ToolBox::GetMenuType() const { return mpData->maMenuType; } sal_Bool ToolBox::IsMenuEnabled() const { return mpData->maMenuType != TOOLBOX_MENUTYPE_NONE; } PopupMenu* ToolBox::GetMenu() const { return mpData->mpMenu; } void ToolBox::SetMenuButtonHdl( const Link& rLink ) { mpData->maMenuButtonHdl = rLink; } const Link& ToolBox::GetMenuButtonHdl() const { return mpData->maMenuButtonHdl; } // ----------------------------------------------------------------------- sal_Bool ToolBox::ImplHasClippedItems() { // are any items currently clipped ? ImplFormat(); std::vector< ImplToolItem >::const_iterator it = mpData->m_aItems.begin(); while ( it != mpData->m_aItems.end() ) { if( it->IsClipped() ) return sal_True; it++; } return sal_False; } void ToolBox::ImplUpdateCustomMenu() { // fill clipped items into menu if( !IsMenuEnabled() ) return; PopupMenu *pMenu = GetMenu(); sal_uInt16 i = 0; // remove old entries while( i < pMenu->GetItemCount() ) { if( pMenu->GetItemId( i ) >= TOOLBOX_MENUITEM_START ) { pMenu->RemoveItem( i ); i = 0; } else i++; } // add menu items, starting from the end and inserting at pos 0 if ( !mpData->m_aItems.empty() ) { for ( std::vector< ImplToolItem >::reverse_iterator it(mpData->m_aItems.rbegin()); it != mpData->m_aItems.rend(); ++it) { if( it->IsClipped() ) { sal_uInt16 id = it->mnId + TOOLBOX_MENUITEM_START; pMenu->InsertItem( id, it->maText, it->maImage, 0, 0 ); pMenu->EnableItem( id, it->mbEnabled ); pMenu->CheckItem( id, it->meState == STATE_CHECK ); } } } } IMPL_LINK( ToolBox, ImplCustomMenuListener, VclMenuEvent*, pEvent ) { if( pEvent->GetMenu() == GetMenu() && pEvent->GetId() == VCLEVENT_MENU_SELECT ) { sal_uInt16 id = GetMenu()->GetItemId( pEvent->GetItemPos() ); if( id >= TOOLBOX_MENUITEM_START ) TriggerItem( id - TOOLBOX_MENUITEM_START, sal_False, sal_False ); } return 0; } IMPL_LINK( ToolBox, ImplCallExecuteCustomMenu, void*, EMPTYARG ) { mpData->mnEventId = 0; ImplExecuteCustomMenu(); return 0; } void ToolBox::ImplExecuteCustomMenu() { if( IsMenuEnabled() ) { if( GetMenuType() & TOOLBOX_MENUTYPE_CUSTOMIZE ) // call button handler to allow for menu customization mpData->maMenuButtonHdl.Call( this ); // register handler GetMenu()->AddEventListener( LINK( this, ToolBox, ImplCustomMenuListener ) ); // make sure all disabled entries will be shown GetMenu()->SetMenuFlags( GetMenu()->GetMenuFlags() | MENU_FLAG_ALWAYSSHOWDISABLEDENTRIES ); // toolbox might be destroyed during execute ImplDelData aDelData; ImplAddDel( &aDelData ); ImplDelData aBorderDel; bool bBorderDel = false; Window *pWin = this; Rectangle aMenuRect = mpData->maMenubuttonItem.maRect; if( IsFloatingMode() ) { // custom menu is placed in the decoration ImplBorderWindow *pBorderWin = dynamic_cast( GetWindow( WINDOW_BORDER ) ); if( pBorderWin && !pBorderWin->GetMenuRect().IsEmpty() ) { pWin = pBorderWin; aMenuRect = pBorderWin->GetMenuRect(); pWin->ImplAddDel( &aBorderDel ); bBorderDel = true; } } sal_uInt16 uId = GetMenu()->Execute( pWin, Rectangle( ImplGetPopupPosition( aMenuRect, Size() ), Size() ), POPUPMENU_EXECUTE_DOWN | POPUPMENU_NOMOUSEUPCLOSE ); if ( aDelData.IsDelete() ) return; ImplRemoveDel( &aDelData ); if( GetMenu() ) GetMenu()->RemoveEventListener( LINK( this, ToolBox, ImplCustomMenuListener ) ); if( bBorderDel ) { if( aBorderDel.IsDelete() ) return; pWin->ImplRemoveDel( &aBorderDel ); } pWin->Invalidate( aMenuRect ); if( uId ) GrabFocusToDocument(); } } void ToolBox::ExecuteCustomMenu() { if( IsMenuEnabled() ) { // handle custom menu asynchronously // to avoid problems if the toolbox is closed during menu execute ImplUpdateCustomMenu(); Application::PostUserEvent( mpData->mnEventId, LINK( this, ToolBox, ImplCallExecuteCustomMenu ) ); } } // ----------------------------------------------------------------------- // checks override first, useful during calculation of sizes sal_Bool ToolBox::ImplIsFloatingMode() const { DBG_ASSERT( !(mpData->mbAssumeDocked && mpData->mbAssumeFloating), "ToolBox::ImplIsFloatingMode(): cannot assume docked and floating" ); if( mpData->mbAssumeDocked ) return sal_False; else if( mpData->mbAssumeFloating ) return sal_True; else return IsFloatingMode(); } // checks override first, useful during calculation of sizes sal_Bool ToolBox::ImplIsInPopupMode() const { if( mpData->mbAssumePopupMode ) return sal_True; else { ImplDockingWindowWrapper *pWrapper = ImplGetDockingManager()->GetDockingWindowWrapper( this ); return ( pWrapper && pWrapper->GetFloatingWindow() && pWrapper->GetFloatingWindow()->IsInPopupMode() ); } } // ----------------------------------------------------------------------- void ToolBox::Lock( sal_Bool bLock ) { ImplDockingWindowWrapper *pWrapper = ImplGetDockingManager()->GetDockingWindowWrapper( this ); if( !pWrapper ) return; if( mpData->mbIsLocked != bLock ) { mpData->mbIsLocked = bLock; if( !ImplIsFloatingMode() ) { mbCalc = sal_True; mbFormat = sal_True; SetSizePixel( CalcWindowSizePixel(1) ); Invalidate(); } } } // ----------------------------------------------------------------------- sal_Bool ToolBox::AlwaysLocked() { // read config item to determine toolbox behaviour, used for subtoolbars static int nAlwaysLocked = -1; if( nAlwaysLocked == -1 ) { nAlwaysLocked = 0; // ask configuration only once utl::OConfigurationNode aNode = utl::OConfigurationTreeRoot::tryCreateWithServiceFactory( vcl::unohelper::GetMultiServiceFactory(), OUString::createFromAscii( "/org.openoffice.Office.UI.GlobalSettings/Toolbars" ) ); // note: case sensisitive ! if ( aNode.isValid() ) { // feature enabled ? sal_Bool bStatesEnabled = sal_Bool(); ::com::sun::star::uno::Any aValue = aNode.getNodeValue( OUString::createFromAscii( "StatesEnabled" ) ); if( aValue >>= bStatesEnabled ) { if( bStatesEnabled == sal_True ) { // now read the locking state utl::OConfigurationNode aNode2 = utl::OConfigurationTreeRoot::tryCreateWithServiceFactory( vcl::unohelper::GetMultiServiceFactory(), OUString::createFromAscii( "/org.openoffice.Office.UI.GlobalSettings/Toolbars/States" ) ); // note: case sensisitive ! sal_Bool bLocked = sal_Bool(); ::com::sun::star::uno::Any aValue2 = aNode2.getNodeValue( OUString::createFromAscii( "Locked" ) ); if( aValue2 >>= bLocked ) nAlwaysLocked = (bLocked == sal_True) ? 1 : 0; } } } } return nAlwaysLocked == 1 ? sal_True : sal_False; } sal_Bool ToolBox::WillUsePopupMode() const { return mpData->mbWillUsePopupMode; } void ToolBox::WillUsePopupMode( sal_Bool b ) { mpData->mbWillUsePopupMode = b; } void ToolBox::ImplUpdateImageList() { if (mpData->mpImageListProvider != NULL) { sal_Bool bHC = GetSettings().GetStyleSettings().GetHighContrastMode(); try { ImageListType eType = bHC ? vcl::HIGHCONTRAST_YES : vcl::HIGHCONTRAST_NO; if (eType != mpData->meImageListType) { vcl::IImageListProvider* pImageListProvider = mpData->mpImageListProvider; SetImageList( pImageListProvider->getImageList(eType) ); mpData->meImageListType = eType; } } catch (com::sun::star::lang::IllegalArgumentException &) {} } } void ToolBox::SetImageListProvider(vcl::IImageListProvider* _pProvider) { mpData->mpImageListProvider = _pProvider; ImplUpdateImageList(); } // -----------------------------------------------------------------------