/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_svtools.hxx" #include "tabbar.hxx" #include #include #include #include #include #include #include #include #include "svtaccessiblefactory.hxx" #include "svtools/svtools.hrc" #include "svtools/svtdata.hxx" #include // ======================================================================= #define TABBAR_OFFSET_X 7 #define TABBAR_OFFSET_X2 2 #define TABBAR_DRAG_SCROLLOFF 5 #define TABBAR_MINSIZE 5 const USHORT ADDNEWPAGE_AREAWIDTH = 10; // ======================================================================= struct ImplTabBarItem { USHORT mnId; TabBarPageBits mnBits; XubString maText; XubString maHelpText; Rectangle maRect; long mnWidth; ULONG mnHelpId; BOOL mbShort; BOOL mbSelect; BOOL mbEnable; Color maTabBgColor; Color maTabTextColor; ImplTabBarItem( USHORT nItemId, const XubString& rText, TabBarPageBits nPageBits ) : maText( rText ) { mnId = nItemId; mnBits = nPageBits; mnWidth = 0; mnHelpId = 0; mbShort = FALSE; mbSelect = FALSE; mbEnable = TRUE; maTabBgColor = Color( COL_AUTO ); maTabTextColor = Color( COL_AUTO ); } bool IsDefaultTabBgColor() const { return maTabBgColor == Color(COL_AUTO); } bool IsDefaultTabTextColor() const { return maTabTextColor == Color(COL_AUTO); } bool IsSelected(ImplTabBarItem* pCurItem) const { return mbSelect || (pCurItem == this); } }; // ======================================================================= // ----------------- // - ImplTabButton - // ----------------- class ImplTabButton : public PushButton { public: ImplTabButton( TabBar* pParent, WinBits nWinStyle = 0 ) : PushButton( pParent, nWinStyle | WB_RECTSTYLE | WB_SMALLSTYLE | WB_NOLIGHTBORDER | WB_NOPOINTERFOCUS ) {} TabBar* GetParent() const { return (TabBar*)Window::GetParent(); } virtual long PreNotify( NotifyEvent& rNEvt ); }; // ======================================================================= long ImplTabButton::PreNotify( NotifyEvent& rNEvt ) { if ( rNEvt.GetType() == EVENT_MOUSEBUTTONDOWN ) { if ( GetParent()->IsInEditMode() ) { GetParent()->EndEditMode(); return TRUE; } } return PushButton::PreNotify( rNEvt ); } // ======================================================================= // ---------------- // - ImplTabSizer - // ---------------- class ImplTabSizer : public Window { public: ImplTabSizer( TabBar* pParent, WinBits nWinStyle = 0 ); TabBar* GetParent() const { return (TabBar*)Window::GetParent(); } private: void ImplTrack( const Point& rScreenPos ); virtual void MouseButtonDown( const MouseEvent& rMEvt ); virtual void Tracking( const TrackingEvent& rTEvt ); virtual void Paint( const Rectangle& rRect ); Point maStartPos; long mnStartWidth; }; // ----------------------------------------------------------------------- ImplTabSizer::ImplTabSizer( TabBar* pParent, WinBits nWinStyle ) : Window( pParent, nWinStyle & WB_3DLOOK ) , mnStartWidth(0) { SetPointer( Pointer( POINTER_HSIZEBAR ) ); SetSizePixel( Size( 7, 0 ) ); } // ----------------------------------------------------------------------- void ImplTabSizer::ImplTrack( const Point& rScreenPos ) { TabBar* pParent = GetParent(); long nDiff = rScreenPos.X() - maStartPos.X(); pParent->mnSplitSize = mnStartWidth + (pParent->IsMirrored() ? -nDiff : nDiff); if ( pParent->mnSplitSize < TABBAR_MINSIZE ) pParent->mnSplitSize = TABBAR_MINSIZE; pParent->Split(); pParent->Update(); } // ----------------------------------------------------------------------- void ImplTabSizer::MouseButtonDown( const MouseEvent& rMEvt ) { if ( GetParent()->IsInEditMode() ) { GetParent()->EndEditMode(); return; } if ( rMEvt.IsLeft() ) { maStartPos = OutputToScreenPixel( rMEvt.GetPosPixel() ); mnStartWidth = GetParent()->GetSizePixel().Width(); StartTracking(); } } // ----------------------------------------------------------------------- void ImplTabSizer::Tracking( const TrackingEvent& rTEvt ) { if ( rTEvt.IsTrackingEnded() ) { if ( rTEvt.IsTrackingCanceled() ) ImplTrack( maStartPos ); GetParent()->mnSplitSize = 0; } else ImplTrack( OutputToScreenPixel( rTEvt.GetMouseEvent().GetPosPixel() ) ); } // ----------------------------------------------------------------------- void ImplTabSizer::Paint( const Rectangle& ) { const StyleSettings& rStyleSettings = GetSettings().GetStyleSettings(); DecorationView aDecoView( this ); long nOffX = 0; Size aOutputSize = GetOutputSizePixel(); if ( !(rStyleSettings.GetOptions() & STYLE_OPTION_MONO) ) { SetLineColor( rStyleSettings.GetDarkShadowColor() ); DrawLine( Point( 0, 0 ), Point( 0, aOutputSize.Height()-1 ) ); nOffX++; aOutputSize.Width()--; } aDecoView.DrawButton( Rectangle( Point( nOffX, 0 ), aOutputSize ), BUTTON_DRAW_NOLIGHTBORDER ); } // ======================================================================= // Heisst nicht Impl, da evtl. mal von aussen benutz- und ueberladbar // -------------- // - TabBarEdit - // -------------- class TabBarEdit : public Edit { private: Timer maLoseFocusTimer; BOOL mbPostEvt; DECL_LINK( ImplEndEditHdl, void* ); DECL_LINK( ImplEndTimerHdl, void* ); public: TabBarEdit( TabBar* pParent, WinBits nWinStyle = 0 ); TabBar* GetParent() const { return (TabBar*)Window::GetParent(); } void SetPostEvent() { mbPostEvt = TRUE; } void ResetPostEvent() { mbPostEvt = FALSE; } virtual long PreNotify( NotifyEvent& rNEvt ); virtual void LoseFocus(); }; // ----------------------------------------------------------------------- TabBarEdit::TabBarEdit( TabBar* pParent, WinBits nWinStyle ) : Edit( pParent, nWinStyle ) { mbPostEvt = FALSE; } // ----------------------------------------------------------------------- long TabBarEdit::PreNotify( NotifyEvent& rNEvt ) { if ( rNEvt.GetType() == EVENT_KEYINPUT ) { const KeyEvent* pKEvt = rNEvt.GetKeyEvent(); if ( !pKEvt->GetKeyCode().GetModifier() ) { if ( pKEvt->GetKeyCode().GetCode() == KEY_RETURN ) { if ( !mbPostEvt ) { if ( PostUserEvent( LINK( this, TabBarEdit, ImplEndEditHdl ), (void*)FALSE ) ) mbPostEvt = TRUE; } return TRUE; } else if ( pKEvt->GetKeyCode().GetCode() == KEY_ESCAPE ) { if ( !mbPostEvt ) { if ( PostUserEvent( LINK( this, TabBarEdit, ImplEndEditHdl ), (void*)TRUE ) ) mbPostEvt = TRUE; } return TRUE; } } } return Edit::PreNotify( rNEvt ); } // ----------------------------------------------------------------------- void TabBarEdit::LoseFocus() { if ( !mbPostEvt ) { if ( PostUserEvent( LINK( this, TabBarEdit, ImplEndEditHdl ), (void*)FALSE ) ) mbPostEvt = TRUE; } Edit::LoseFocus(); } // ----------------------------------------------------------------------- IMPL_LINK( TabBarEdit, ImplEndEditHdl, void*, pCancel ) { ResetPostEvent(); maLoseFocusTimer.Stop(); // We need this query, because the edit get a losefous event, // when it shows the context menu or the insert symbol dialog if ( !HasFocus() && HasChildPathFocus( TRUE ) ) { maLoseFocusTimer.SetTimeout( 30 ); maLoseFocusTimer.SetTimeoutHdl( LINK( this, TabBarEdit, ImplEndTimerHdl ) ); maLoseFocusTimer.Start(); } else GetParent()->EndEditMode( pCancel != 0 ); return 0; } // ----------------------------------------------------------------------- IMPL_LINK( TabBarEdit, ImplEndTimerHdl, void*, EMPTYARG ) { if ( HasFocus() ) return 0; // We need this query, because the edit get a losefous event, // when it shows the context menu or the insert symbol dialog if ( HasChildPathFocus( TRUE ) ) maLoseFocusTimer.Start(); else GetParent()->EndEditMode( TRUE ); return 0; } // ======================================================================= struct TabBar_Impl { ImplTabSizer* mpSizer; ::svt::AccessibleFactoryAccess maAccessibleFactory; TabBar_Impl() :mpSizer( NULL ) { } ~TabBar_Impl() { delete mpSizer; } }; // ======================================================================= const sal_uInt16 TabBar::APPEND = ::std::numeric_limits::max(); const sal_uInt16 TabBar::PAGE_NOT_FOUND = ::std::numeric_limits::max(); const sal_uInt16 TabBar::INSERT_TAB_POS = ::std::numeric_limits::max() - 1; void TabBar::ImplInit( WinBits nWinStyle ) { mpItemList = new ImplTabBarList; mpFirstBtn = NULL; mpPrevBtn = NULL; mpNextBtn = NULL; mpLastBtn = NULL; mpImpl = new TabBar_Impl; mpEdit = NULL; mnMaxPageWidth = 0; mnCurMaxWidth = 0; mnOffX = 0; mnOffY = 0; mnLastOffX = 0; mnSplitSize = 0; mnSwitchTime = 0; mnWinStyle = nWinStyle; mnCurPageId = 0; mnFirstPos = 0; mnDropPos = 0; mnSwitchId = 0; mnEditId = 0; mbFormat = TRUE; mbFirstFormat = TRUE; mbSizeFormat = TRUE; mbAutoMaxWidth = TRUE; mbInSwitching = FALSE; mbAutoEditMode = FALSE; mbEditCanceled = FALSE; mbDropPos = FALSE; mbInSelect = FALSE; mbSelColor = FALSE; mbSelTextColor = FALSE; mbMirrored = FALSE; if ( nWinStyle & WB_3DTAB ) mnOffY++; ImplInitControls(); SetSizePixel( Size( 100, CalcWindowSizePixel().Height() ) ); ImplInitSettings( TRUE, TRUE ); } // ----------------------------------------------------------------------- TabBar::TabBar( Window* pParent, WinBits nWinStyle ) : Window( pParent, (nWinStyle & WB_3DLOOK) | WB_CLIPCHILDREN ) { ImplInit( nWinStyle ); maCurrentItemList = 0; } // ----------------------------------------------------------------------- TabBar::~TabBar() { EndEditMode( TRUE ); // Controls loeschen if ( mpPrevBtn ) delete mpPrevBtn; if ( mpNextBtn ) delete mpNextBtn; if ( mpFirstBtn ) delete mpFirstBtn; if ( mpLastBtn ) delete mpLastBtn; delete mpImpl; for ( size_t i = 0, n = mpItemList->size(); i < n; ++i ) { delete (*mpItemList)[ i ]; } delete mpItemList; } // ----------------------------------------------------------------------- ImplTabBarItem* TabBar::seek( size_t i ) { if ( i < mpItemList->size() ) { maCurrentItemList = i; return (*mpItemList)[ maCurrentItemList ]; } return NULL; } ImplTabBarItem* TabBar::prev() { if ( maCurrentItemList > 0 ) { return (*mpItemList)[ --maCurrentItemList ]; } return NULL; } ImplTabBarItem* TabBar::next() { if ( maCurrentItemList+1 < mpItemList->size() ) { return (*mpItemList)[ ++maCurrentItemList ]; } return NULL; } // ----------------------------------------------------------------------- void TabBar::ImplInitSettings( BOOL bFont, BOOL bBackground ) { const StyleSettings& rStyleSettings = GetSettings().GetStyleSettings(); if ( bFont ) { Font aToolFont; aToolFont = rStyleSettings.GetToolFont(); if ( IsControlFont() ) aToolFont.Merge( GetControlFont() ); aToolFont.SetWeight( WEIGHT_BOLD ); SetZoomedPointFont( aToolFont ); // Font in der groesse Anpassen, wenn Fenster zu klein? while ( GetTextHeight() > (GetOutputSizePixel().Height()-1) ) { Font aFont = GetFont(); if ( aFont.GetHeight() <= 6 ) break; aFont.SetHeight( aFont.GetHeight()-1 ); SetFont( aFont ); } } if ( bBackground ) { Color aColor; if ( IsControlBackground() ) aColor = GetControlBackground(); else aColor = rStyleSettings.GetFaceColor(); SetBackground( aColor ); } } // ----------------------------------------------------------------------- void TabBar::ImplGetColors( Color& rFaceColor, Color& rFaceTextColor, Color& rSelectColor, Color& rSelectTextColor ) { const StyleSettings& rStyleSettings = GetSettings().GetStyleSettings(); if ( IsControlBackground() ) rFaceColor = GetControlBackground(); else rFaceColor = rStyleSettings.GetInactiveTabColor(); if ( IsControlForeground() ) rFaceTextColor = GetControlForeground(); else rFaceTextColor = rStyleSettings.GetButtonTextColor(); if ( mbSelColor ) rSelectColor = maSelColor; else rSelectColor = rStyleSettings.GetActiveTabColor(); if ( mbSelTextColor ) rSelectTextColor = maSelTextColor; else rSelectTextColor = rStyleSettings.GetWindowTextColor(); // Bei 3D-Tabs wird Selektions- und Face-Farbe umgedreht, da die // selektierten Tabs in 3D erscheinen sollen if ( mnWinStyle & WB_3DTAB ) { Color aTempColor = rFaceColor; rFaceColor = rSelectColor; rSelectColor = aTempColor; aTempColor = rFaceTextColor; rFaceTextColor = rSelectTextColor; rSelectTextColor = rFaceTextColor; } } // ----------------------------------------------------------------------- BOOL TabBar::ImplCalcWidth() { // Groessen muessen nur ermittelt werden, wenn sich Text aendert oder // wenn der Font geaendert wurde if ( !mbSizeFormat ) return FALSE; // Breiten der Tabs mit dem fetten Font ermitteln Font aFont = GetFont(); if ( aFont.GetWeight() != WEIGHT_BOLD ) { aFont.SetWeight( WEIGHT_BOLD ); SetFont( aFont ); } if ( mnMaxPageWidth ) mnCurMaxWidth = mnMaxPageWidth; else if ( mbAutoMaxWidth ) { mnCurMaxWidth = mnLastOffX-mnOffX- TABBAR_OFFSET_X-TABBAR_OFFSET_X- TABBAR_OFFSET_X2-TABBAR_OFFSET_X2-TABBAR_OFFSET_X2; if ( mnCurMaxWidth < 1 ) mnCurMaxWidth = 1; } else mnCurMaxWidth = 0; BOOL bChanged = FALSE; for ( size_t i = 0, n = mpItemList->size(); i < n; ++i ) { ImplTabBarItem* pItem = (*mpItemList)[ i ]; long nNewWidth = GetTextWidth( pItem->maText ); if ( mnCurMaxWidth && (nNewWidth > mnCurMaxWidth) ) { pItem->mbShort = TRUE; nNewWidth = mnCurMaxWidth; } else pItem->mbShort = FALSE; nNewWidth += TABBAR_OFFSET_X+TABBAR_OFFSET_X2; if ( pItem->mnWidth != nNewWidth ) { pItem->mnWidth = nNewWidth; if ( !pItem->maRect.IsEmpty() ) bChanged = TRUE; } } mbSizeFormat = FALSE; mbFormat = TRUE; return bChanged; } // ----------------------------------------------------------------------- void TabBar::ImplFormat() { ImplCalcWidth(); if ( !mbFormat ) return; USHORT n = 0; long x = mnOffX; for ( size_t i = 0, nL = mpItemList->size(); i < nL; ++i ) { ImplTabBarItem* pItem = (*mpItemList)[ i ]; // Bei allen nicht sichtbaren Tabs, wird ein leeres Rechteck // gesetzt if ( (n+1 < mnFirstPos) || (x > mnLastOffX) ) pItem->maRect.SetEmpty(); else { // Etwas von der Tab vor der ersten sichtbaren Page // muss auch zu sehen sein if ( n+1 == mnFirstPos ) pItem->maRect.Left() = x-pItem->mnWidth; else { pItem->maRect.Left() = x; x += pItem->mnWidth; } pItem->maRect.Right() = x+TABBAR_OFFSET_X+TABBAR_OFFSET_X2; pItem->maRect.Bottom() = maWinSize.Height()-1; if( mbMirrored ) { long nTmp = mnOffX + mnLastOffX - pItem->maRect.Right(); pItem->maRect.Right() = mnOffX + mnLastOffX - pItem->maRect.Left(); pItem->maRect.Left() = nTmp; } } n++; } mbFormat = FALSE; // Button enablen/disablen ImplEnableControls(); } // ----------------------------------------------------------------------- USHORT TabBar::ImplGetLastFirstPos() { USHORT nCount = (USHORT)(mpItemList->size()); if ( !nCount || mbSizeFormat || mbFormat ) return 0; USHORT nLastFirstPos = nCount-1; long nWinWidth = mnLastOffX-mnOffX-TABBAR_OFFSET_X-ADDNEWPAGE_AREAWIDTH; long nWidth = (*mpItemList)[ nLastFirstPos ]->mnWidth; while ( nLastFirstPos && (nWidth < nWinWidth) ) { nLastFirstPos--; nWidth += (*mpItemList)[ nLastFirstPos ]->mnWidth; } if ( (nLastFirstPos != (USHORT)(mpItemList->size()-1)) && (nWidth > nWinWidth) ) nLastFirstPos++; return nLastFirstPos; } // ----------------------------------------------------------------------- void TabBar::ImplInitControls() { if ( mnWinStyle & WB_SIZEABLE ) { if ( !mpImpl->mpSizer ) mpImpl->mpSizer = new ImplTabSizer( this, mnWinStyle & (WB_DRAG | WB_3DLOOK) ); mpImpl->mpSizer->Show(); } else { DELETEZ( mpImpl->mpSizer ); } Link aLink = LINK( this, TabBar, ImplClickHdl ); if ( mnWinStyle & (WB_MINSCROLL | WB_SCROLL) ) { if ( !mpPrevBtn ) { mpPrevBtn = new ImplTabButton( this, WB_REPEAT ); mpPrevBtn->SetClickHdl( aLink ); } mpPrevBtn->SetSymbol( mbMirrored ? SYMBOL_NEXT : SYMBOL_PREV ); mpPrevBtn->Show(); if ( !mpNextBtn ) { mpNextBtn = new ImplTabButton( this, WB_REPEAT ); mpNextBtn->SetClickHdl( aLink ); } mpNextBtn->SetSymbol( mbMirrored ? SYMBOL_PREV : SYMBOL_NEXT ); mpNextBtn->Show(); } else { DELETEZ( mpPrevBtn ); DELETEZ( mpNextBtn ); } if ( mnWinStyle & WB_SCROLL ) { if ( !mpFirstBtn ) { mpFirstBtn = new ImplTabButton( this ); mpFirstBtn->SetClickHdl( aLink ); } mpFirstBtn->SetSymbol( mbMirrored ? SYMBOL_LAST : SYMBOL_FIRST ); mpFirstBtn->Show(); if ( !mpLastBtn ) { mpLastBtn = new ImplTabButton( this ); mpLastBtn->SetClickHdl( aLink ); } mpLastBtn->SetSymbol( mbMirrored ? SYMBOL_FIRST : SYMBOL_LAST ); mpLastBtn->Show(); } else { DELETEZ( mpFirstBtn ); DELETEZ( mpLastBtn ); } mbHasInsertTab = (mnWinStyle & WB_INSERTTAB); } // ----------------------------------------------------------------------- void TabBar::ImplEnableControls() { if ( mbSizeFormat || mbFormat ) return; // Buttons enablen/disblen BOOL bEnableBtn = mnFirstPos > 0; if ( mpFirstBtn ) mpFirstBtn->Enable( bEnableBtn ); if ( mpPrevBtn ) mpPrevBtn->Enable( bEnableBtn ); bEnableBtn = mnFirstPos < ImplGetLastFirstPos(); if ( mpNextBtn ) mpNextBtn->Enable( bEnableBtn ); if ( mpLastBtn ) mpLastBtn->Enable( bEnableBtn ); } // ----------------------------------------------------------------------- void TabBar::ImplShowPage( USHORT nPos ) { // Breite berechnen long nWidth = GetOutputSizePixel().Width(); if ( nWidth >= TABBAR_OFFSET_X ) nWidth -= TABBAR_OFFSET_X; ImplTabBarItem* pItem = (*mpItemList)[ nPos ]; if ( nPos < mnFirstPos ) SetFirstPageId( pItem->mnId ); else if ( pItem->maRect.Right() > nWidth ) { while ( pItem->maRect.Right() > nWidth ) { USHORT nNewPos = mnFirstPos+1; SetFirstPageId( GetPageId( nNewPos ) ); ImplFormat(); if ( nNewPos != mnFirstPos ) break; } } } // ----------------------------------------------------------------------- IMPL_LINK( TabBar, ImplClickHdl, ImplTabButton*, pBtn ) { EndEditMode(); USHORT nNewPos = mnFirstPos; if ( pBtn == mpFirstBtn ) nNewPos = 0; else if ( pBtn == mpPrevBtn ) { if ( mnFirstPos ) nNewPos = mnFirstPos-1; } else if ( pBtn == mpNextBtn ) { USHORT nCount = GetPageCount(); if ( mnFirstPos < nCount ) nNewPos = mnFirstPos+1; } else { USHORT nCount = GetPageCount(); if ( nCount ) nNewPos = nCount-1; } if ( nNewPos != mnFirstPos ) SetFirstPageId( GetPageId( nNewPos ) ); return 0; } // ----------------------------------------------------------------------- void TabBar::MouseMove( const MouseEvent& rMEvt ) { if ( rMEvt.IsLeaveWindow() ) mbInSelect = FALSE; Window::MouseMove( rMEvt ); } // ----------------------------------------------------------------------- void TabBar::MouseButtonDown( const MouseEvent& rMEvt ) { // Bei Klick in unser Fenster EditModus nur beenden und Klick nicht // ausfuehren if ( IsInEditMode() ) { EndEditMode(); return; } ImplTabBarItem* pItem; USHORT nSelId = GetPageId( rMEvt.GetPosPixel() ); if ( !rMEvt.IsLeft() ) { Window::MouseButtonDown( rMEvt ); if ( (nSelId > 0) && (nSelId != mnCurPageId) ) { USHORT nPos = GetPagePos( nSelId ); pItem = (*mpItemList)[ nPos ]; if ( pItem->mbEnable ) { if ( ImplDeactivatePage() ) { SetCurPageId( nSelId ); Update(); ImplActivatePage(); ImplSelect(); } mbInSelect = TRUE; } } return; } if ( rMEvt.IsMod2() && mbAutoEditMode && nSelId ) { if ( StartEditMode( nSelId ) ) return; } if ( (rMEvt.GetMode() & (MOUSE_MULTISELECT | MOUSE_RANGESELECT)) && (rMEvt.GetClicks() == 1) ) { if ( nSelId ) { USHORT nPos = GetPagePos( nSelId ); BOOL bSelectTab = FALSE; pItem = (*mpItemList)[ nPos ]; if ( pItem->mbEnable ) { if ( (rMEvt.GetMode() & MOUSE_MULTISELECT) && (mnWinStyle & WB_MULTISELECT) ) { if ( nSelId != mnCurPageId ) { SelectPage( nSelId, !IsPageSelected( nSelId ) ); bSelectTab = TRUE; } } else if ( mnWinStyle & (WB_MULTISELECT | WB_RANGESELECT) ) { bSelectTab = TRUE; USHORT n; BOOL bSelect; USHORT nCurPos = GetPagePos( mnCurPageId ); if ( nPos <= nCurPos ) { // Alle Tabs bis zur angeklickten Tab deselektieren // und alle Tabs von der angeklickten Tab bis // zur aktuellen Position selektieren n = 0; while ( n < nCurPos ) { pItem = (*mpItemList)[ n ]; if ( n < nPos ) bSelect = FALSE; else bSelect = TRUE; if ( pItem->mbSelect != bSelect ) { pItem->mbSelect = bSelect; if ( !pItem->maRect.IsEmpty() ) Invalidate( pItem->maRect ); } n++; } } if ( nPos >= nCurPos ) { // Alle Tabs von der aktuellen bis zur angeklickten // Tab selektieren und alle Tabs von der angeklickten // Tab bis zur letzten Tab deselektieren USHORT nCount = (USHORT)mpItemList->size(); n = nCurPos; while ( n < nCount ) { pItem = (*mpItemList)[ n ]; if ( n <= nPos ) bSelect = TRUE; else bSelect = FALSE; if ( pItem->mbSelect != bSelect ) { pItem->mbSelect = bSelect; if ( !pItem->maRect.IsEmpty() ) Invalidate( pItem->maRect ); } n++; } } } // Gegebenenfalls muss die selektierte Tab gescrollt werden if ( bSelectTab ) { ImplShowPage( nPos ); Update(); ImplSelect(); } } else ImplShowPage( nPos ); mbInSelect = TRUE; return; } } else if ( rMEvt.GetClicks() == 2 ) { // Gegebenenfalls den Double-Click-Handler rufen if ( !rMEvt.GetModifier() && (!nSelId || (nSelId == mnCurPageId)) ) { USHORT nOldCurId = mnCurPageId; mnCurPageId = nSelId; DoubleClick(); // Abfrage, da im DoubleClick-Handler die aktuelle Seite // umgeschaltet werden konnte if ( mnCurPageId == nSelId ) mnCurPageId = nOldCurId; } return; } else { if ( nSelId ) { // Nur Select ausfuehren, wenn noch nicht aktuelle Page if ( nSelId != mnCurPageId ) { USHORT nPos = GetPagePos( nSelId ); pItem = (*mpItemList)[ nPos ]; if ( pItem->mbEnable ) { if ( !pItem->mbSelect ) { // Muss invalidiert werden BOOL bUpdate = FALSE; if ( IsReallyVisible() && IsUpdateMode() ) bUpdate = TRUE; // Alle selektierten Items deselektieren for ( size_t i = 0, n = mpItemList->size(); i < n; ++i ) { pItem = (*mpItemList)[ i ]; if ( pItem->mbSelect || (pItem->mnId == mnCurPageId) ) { pItem->mbSelect = FALSE; if ( bUpdate ) Invalidate( pItem->maRect ); } } } if ( ImplDeactivatePage() ) { SetCurPageId( nSelId ); Update(); ImplActivatePage(); ImplSelect(); } } else ImplShowPage( nPos ); mbInSelect = TRUE; } return; } } Window::MouseButtonDown( rMEvt ); } // ----------------------------------------------------------------------- void TabBar::MouseButtonUp( const MouseEvent& rMEvt ) { mbInSelect = FALSE; Window::MouseButtonUp( rMEvt ); } // ----------------------------------------------------------------------- namespace { class TabBarPaintGuard { public: explicit TabBarPaintGuard(TabBar& rParent) : mrParent(rParent), maFont(rParent.GetFont()) { // #i36013# exclude push buttons from painting area mrParent.SetClipRegion( Region(mrParent.GetPageArea()) ); } ~TabBarPaintGuard() { // Restore original font. mrParent.SetFont(maFont); // remove clip region mrParent.SetClipRegion(); } private: TabBar& mrParent; Font maFont; }; class TabDrawer { public: explicit TabDrawer(TabBar& rParent) : mrParent(rParent), mpStyleSettings(&mrParent.GetSettings().GetStyleSettings()), maPoly(4), mbSelected(false), mbCustomColored(false), mbSpecialTab(false), mbEnabled(false) { } void drawOutputAreaBorder() { WinBits nWinStyle = mrParent.GetStyle(); // Bei Border oben und unten einen Strich extra malen if ( (nWinStyle & WB_BORDER) || (nWinStyle & WB_TOPBORDER) ) { Size aOutputSize = mrParent.GetOutputSizePixel(); Rectangle aOutRect = mrParent.GetPageArea(); // Bei 3D-Tabs wird auch der Border in 3D gemalt if ( nWinStyle & WB_3DTAB ) { mrParent.SetLineColor( mpStyleSettings->GetShadowColor() ); mrParent.DrawLine( Point( aOutRect.Left(), 0 ), Point( aOutputSize.Width(), 0 ) ); } // Border malen (Strich oben und Strich unten) mrParent.SetLineColor( mpStyleSettings->GetDarkShadowColor() ); mrParent.DrawLine( aOutRect.TopLeft(), Point( aOutputSize.Width()-1, aOutRect.Top() ) ); } } void drawOuterFrame() { mrParent.DrawPolygon(maPoly); } void drawLeftShadow() { Point p1 = maPoly[0], p2 = maPoly[1]; p1.X()++; p2.X()++; p2.Y()--; mrParent.DrawLine(p1, p2); } void drawRightShadow() { Point p1 = maPoly[2]; Point p2 = maPoly[3]; p1.X()--; p2.X()--; mrParent.DrawLine(p1, p2); } void drawTopInnerShadow() { Point p1 = maPoly[0], p2 = maPoly[3]; p1.Y()++; p2.Y()++; mrParent.DrawLine(p1, p2); } void drawBottomShadow(bool bColored) { Point p1 = maPoly[1], p2 = maPoly[2]; p1.X() += 1; p1.Y() -= 1; p2.X() -= 1; p2.Y() -= 1; mrParent.DrawLine(p1, p2); if (bColored) { p1 += Point(-1, -1); p2 += Point(1, -1); mrParent.DrawLine(p1, p2); } } void drawText(const String& aText) { Rectangle aRect = maRect; long nTextWidth = mrParent.GetTextWidth(aText); long nTextHeight = mrParent.GetTextHeight(); Point aPos = aRect.TopLeft(); aPos.X() += (aRect.getWidth() - nTextWidth) / 2; aPos.Y() += (aRect.getHeight() - nTextHeight) / 2; if (mbEnabled) mrParent.DrawText(aPos, aText); else mrParent.DrawCtrlText( aPos, aText, 0, STRING_LEN, (TEXT_DRAW_DISABLE | TEXT_DRAW_MNEMONIC)); } void drawOverTopBorder(bool b3DTab) { Point p1 = maPoly[0], p2 = maPoly[3]; p1.X() += 1; p2.X() -= 1; Rectangle aDelRect(p1, p2); mrParent.DrawRect(aDelRect); if (b3DTab) { aDelRect.Top()--; mrParent.DrawRect(aDelRect); } } void drawTab() { mrParent.SetLineColor(mpStyleSettings->GetDarkShadowColor()); // Je nach Status die richtige FillInBrush setzen // Set the correct FillInBrush depending upon status if ( mbSelected ) { // Currently selected Tab mrParent.SetFillColor( maSelectedColor ); } else if ( mbCustomColored ) { mrParent.SetFillColor( maCustomColor ); } else { mrParent.SetFillColor( maUnselectedColor ); } drawOuterFrame(); // If this is the current tab, draw the left inner shadow the default color, // otherwise make it the same as the custom background color Color aColor = mpStyleSettings->GetLightColor(); if (mbCustomColored && !mbSelected) aColor = maCustomColor; mrParent.SetLineColor(aColor); drawLeftShadow(); if ( !mbSelected ) drawTopInnerShadow(); mrParent.SetLineColor( mpStyleSettings->GetShadowColor() ); drawRightShadow(); if ( mbCustomColored && mbSelected ) { mrParent.SetLineColor(maCustomColor); drawBottomShadow(true); } else drawBottomShadow(false); // Draw the outer frame once more. In some environments, the outer frame // gets overpainted. mrParent.SetLineColor( mpStyleSettings->GetDarkShadowColor() ); mrParent.SetFillColor(); drawOuterFrame(); } void drawPlusImage() { Image aPlusImg( SvtResId(BMP_LIST_ADD) ); // Center the image within the bounding rectangle. Size aSize = aPlusImg.GetSizePixel(); Point pt = maRect.TopLeft(); long nXOffSet = (maRect.GetWidth() - aSize.Width()) / 2; long nYOffset = (maRect.GetHeight() - aSize.Height()) / 2; pt += Point(nXOffSet, nYOffset); pt.X() += 1; mrParent.DrawImage(pt, aPlusImg); } void setRect(const Rectangle& rRect) { maRect = rRect; long nOffY = mrParent.GetPageArea().getY(); // Zuerst geben wir das Polygon gefuellt aus maPoly[0] = Point( rRect.Left(), nOffY ); maPoly[1] = Point( rRect.Left()+TABBAR_OFFSET_X, rRect.Bottom() ); maPoly[2] = Point( rRect.Right()-TABBAR_OFFSET_X, rRect.Bottom() ); maPoly[3] = Point( rRect.Right(), nOffY ); } void setSelected(bool b) { mbSelected = b; } void setCustomColored(bool b) { mbCustomColored = b; } void setSpecialTab(bool b) { mbSpecialTab = b; } void setEnabled(bool b) { mbEnabled = b; } void setSelectedFillColor(const Color& rColor) { maSelectedColor = rColor; } void setUnselectedFillColor(const Color& rColor) { maUnselectedColor = rColor; } void setCustomColor(const Color& rColor) { maCustomColor = rColor; } private: TabBar& mrParent; const StyleSettings* mpStyleSettings; Rectangle maRect; Polygon maPoly; Color maSelectedColor; Color maCustomColor; Color maUnselectedColor; bool mbSelected:1; bool mbCustomColored:1; bool mbSpecialTab:1; bool mbEnabled:1; }; } void TabBar::Paint( const Rectangle& ) { // Items berechnen und ausgeben USHORT nItemCount = (USHORT)mpItemList->size(); if (!nItemCount) return; ImplPrePaint(); Color aFaceColor, aSelectColor, aFaceTextColor, aSelectTextColor; ImplGetColors( aFaceColor, aFaceTextColor, aSelectColor, aSelectTextColor ); // Font selektieren Font aFont = GetFont(); Font aLightFont = aFont; aLightFont.SetWeight( WEIGHT_NORMAL ); TabBarPaintGuard aGuard(*this); TabDrawer aDrawer(*this); aDrawer.setSelectedFillColor(aSelectColor); aDrawer.setUnselectedFillColor(aFaceColor); aDrawer.drawOutputAreaBorder(); // Now, start drawing the tabs. ImplTabBarItem* pItem = ImplGetLastTabBarItem(nItemCount); if (pItem && mbHasInsertTab) { // Draw the insert tab at the right end. Rectangle aRect = ImplGetInsertTabRect(pItem); aDrawer.setRect(aRect); aDrawer.drawTab(); aDrawer.drawPlusImage(); } const StyleSettings& rStyleSettings = GetSettings().GetStyleSettings(); ImplTabBarItem* pCurItem = NULL; while ( pItem ) { // CurrentItem als letztes ausgeben, da es alle anderen ueberdeckt if ( !pCurItem && (pItem->mnId == mnCurPageId) ) { pCurItem = pItem; pItem = prev(); if ( !pItem ) pItem = pCurItem; continue; } bool bCurrent = pItem == pCurItem; if ( !pItem->maRect.IsEmpty() ) { Rectangle aRect = pItem->maRect; bool bSelected = pItem->IsSelected(pCurItem); // We disable custom background color in high contrast mode. bool bCustomBgColor = !pItem->IsDefaultTabBgColor() && !rStyleSettings.GetHighContrastMode(); bool bSpecialTab = (pItem->mnBits & TPB_SPECIAL); bool bEnabled = pItem->mbEnable; String aText = pItem->mbShort ? GetEllipsisString(pItem->maText, mnCurMaxWidth, TEXT_DRAW_ENDELLIPSIS) : pItem->maText; aDrawer.setRect(aRect); aDrawer.setSelected(bSelected); aDrawer.setCustomColored(bCustomBgColor); aDrawer.setSpecialTab(bSpecialTab); aDrawer.setEnabled(bEnabled); aDrawer.setCustomColor(pItem->maTabBgColor); aDrawer.drawTab(); // Aktuelle Page wird mit einem fetten Font ausgegeben if ( bCurrent ) SetFont( aFont ); else SetFont( aLightFont ); // Je nach Status die richtige FillInBrush setzen // Set the correct FillInBrush depending upon status if ( bSelected ) SetTextColor( aSelectTextColor ); else if ( bCustomBgColor ) SetTextColor( pItem->maTabTextColor ); else SetTextColor( aFaceTextColor ); // This tab is "special", and a special tab needs a blue text. if (bSpecialTab) SetTextColor(Color(COL_LIGHTBLUE)); aDrawer.drawText(aText); if ( bCurrent ) { SetLineColor(); SetFillColor(aSelectColor); aDrawer.drawOverTopBorder(mnWinStyle & WB_3DTAB); return; } pItem = prev(); } else { if ( bCurrent ) return; pItem = NULL; } if ( !pItem ) pItem = pCurItem; } } // ----------------------------------------------------------------------- void TabBar::Resize() { Size aNewSize = GetOutputSizePixel(); long nSizerWidth = 0; long nButtonWidth = 0; // Sizer anordnen if ( mpImpl->mpSizer ) { Size aSizerSize = mpImpl->mpSizer->GetSizePixel(); Point aNewSizerPos( mbMirrored ? 0 : (aNewSize.Width()-aSizerSize.Width()), 0 ); Size aNewSizerSize( aSizerSize.Width(), aNewSize.Height() ); mpImpl->mpSizer->SetPosSizePixel( aNewSizerPos, aNewSizerSize ); nSizerWidth = aSizerSize.Width(); } // Scroll-Buttons anordnen long nHeight = aNewSize.Height(); // Font in der groesse Anpassen? ImplInitSettings( TRUE, FALSE ); long nX = mbMirrored ? (aNewSize.Width()-nHeight) : 0; long nXDiff = mbMirrored ? -nHeight : nHeight; Size aBtnSize( nHeight, nHeight ); if ( mpFirstBtn ) { mpFirstBtn->SetPosSizePixel( Point( nX, 0 ), aBtnSize ); nX += nXDiff; nButtonWidth += nHeight; } if ( mpPrevBtn ) { mpPrevBtn->SetPosSizePixel( Point( nX, 0 ), aBtnSize ); nX += nXDiff; nButtonWidth += nHeight; } if ( mpNextBtn ) { mpNextBtn->SetPosSizePixel( Point( nX, 0 ), aBtnSize ); nX += nXDiff; nButtonWidth += nHeight; } if ( mpLastBtn ) { mpLastBtn->SetPosSizePixel( Point( nX, 0 ), aBtnSize ); nX += nXDiff; nButtonWidth += nHeight; } // Groesse merken maWinSize = aNewSize; if( mbMirrored ) { mnOffX = nSizerWidth; mnLastOffX = maWinSize.Width() - nButtonWidth - 1; } else { mnOffX = nButtonWidth; mnLastOffX = maWinSize.Width() - nSizerWidth - 1; } // Neu formatieren mbSizeFormat = TRUE; if ( IsReallyVisible() ) { if ( ImplCalcWidth() ) Invalidate(); ImplFormat(); } // Button enablen/disablen ImplEnableControls(); } // ----------------------------------------------------------------------- void TabBar::RequestHelp( const HelpEvent& rHEvt ) { USHORT nItemId = GetPageId( ScreenToOutputPixel( rHEvt.GetMousePosPixel() ) ); if ( nItemId ) { if ( rHEvt.GetMode() & HELPMODE_BALLOON ) { XubString aStr = GetHelpText( nItemId ); if ( aStr.Len() ) { Rectangle aItemRect = GetPageRect( nItemId ); Point aPt = OutputToScreenPixel( aItemRect.TopLeft() ); aItemRect.Left() = aPt.X(); aItemRect.Top() = aPt.Y(); aPt = OutputToScreenPixel( aItemRect.BottomRight() ); aItemRect.Right() = aPt.X(); aItemRect.Bottom() = aPt.Y(); Help::ShowBalloon( this, aItemRect.Center(), aItemRect, aStr ); return; } } else if ( rHEvt.GetMode() & HELPMODE_EXTENDED ) { ULONG nHelpId = GetHelpId( nItemId ); if ( nHelpId ) { // Wenn eine Hilfe existiert, dann ausloesen Help* pHelp = Application::GetHelp(); if ( pHelp ) pHelp->Start( nHelpId, this ); return; } } // Bei Quick- oder Ballloon-Help zeigen wir den Text an, // wenn dieser abgeschnitten oder nicht voll sichtbar ist if ( rHEvt.GetMode() & (HELPMODE_QUICK | HELPMODE_BALLOON) ) { USHORT nPos = GetPagePos( nItemId ); ImplTabBarItem* pItem = (*mpItemList)[ nPos ]; if ( pItem->mbShort || (pItem->maRect.Right()-TABBAR_OFFSET_X-5 > mnLastOffX) ) { Rectangle aItemRect = GetPageRect( nItemId ); Point aPt = OutputToScreenPixel( aItemRect.TopLeft() ); aItemRect.Left() = aPt.X(); aItemRect.Top() = aPt.Y(); aPt = OutputToScreenPixel( aItemRect.BottomRight() ); aItemRect.Right() = aPt.X(); aItemRect.Bottom() = aPt.Y(); XubString aStr = (*mpItemList)[ nPos ]->maText; if ( aStr.Len() ) { if ( rHEvt.GetMode() & HELPMODE_BALLOON ) Help::ShowBalloon( this, aItemRect.Center(), aItemRect, aStr ); else Help::ShowQuickHelp( this, aItemRect, aStr ); return; } } } } Window::RequestHelp( rHEvt ); } // ----------------------------------------------------------------------- void TabBar::StateChanged( StateChangedType nType ) { Window::StateChanged( nType ); if ( nType == STATE_CHANGE_INITSHOW ) { if ( (mbSizeFormat || mbFormat) && !mpItemList->empty() ) ImplFormat(); } else if ( (nType == STATE_CHANGE_ZOOM) || (nType == STATE_CHANGE_CONTROLFONT) ) { ImplInitSettings( TRUE, FALSE ); Invalidate(); } else if ( nType == STATE_CHANGE_CONTROLFOREGROUND ) Invalidate(); else if ( nType == STATE_CHANGE_CONTROLBACKGROUND ) { ImplInitSettings( FALSE, TRUE ); Invalidate(); } else if ( nType == STATE_CHANGE_MIRRORING ) { // reacts on calls of EnableRTL, have to mirror all child controls if( mpFirstBtn ) mpFirstBtn->EnableRTL( IsRTLEnabled() ); if( mpPrevBtn ) mpPrevBtn->EnableRTL( IsRTLEnabled() ); if( mpNextBtn ) mpNextBtn->EnableRTL( IsRTLEnabled() ); if( mpLastBtn ) mpLastBtn->EnableRTL( IsRTLEnabled() ); if( mpImpl->mpSizer ) mpImpl->mpSizer->EnableRTL( IsRTLEnabled() ); if( mpEdit ) mpEdit->EnableRTL( IsRTLEnabled() ); } } // ----------------------------------------------------------------------- void TabBar::DataChanged( const DataChangedEvent& rDCEvt ) { Window::DataChanged( rDCEvt ); if ( (rDCEvt.GetType() == DATACHANGED_FONTS) || (rDCEvt.GetType() == DATACHANGED_FONTSUBSTITUTION) || ((rDCEvt.GetType() == DATACHANGED_SETTINGS) && (rDCEvt.GetFlags() & SETTINGS_STYLE)) ) { ImplInitSettings( TRUE, TRUE ); Invalidate(); } } // ----------------------------------------------------------------------- void TabBar::ImplSelect() { Select(); CallEventListeners( VCLEVENT_TABBAR_PAGESELECTED, reinterpret_cast(sal::static_int_cast(mnCurPageId)) ); } // ----------------------------------------------------------------------- void TabBar::Select() { maSelectHdl.Call( this ); } // ----------------------------------------------------------------------- void TabBar::DoubleClick() { maDoubleClickHdl.Call( this ); } // ----------------------------------------------------------------------- void TabBar::Split() { maSplitHdl.Call( this ); } // ----------------------------------------------------------------------- void TabBar::ImplActivatePage() { ActivatePage(); CallEventListeners( VCLEVENT_TABBAR_PAGEACTIVATED, reinterpret_cast(sal::static_int_cast(mnCurPageId)) ); } // ----------------------------------------------------------------------- void TabBar::ActivatePage() { maActivatePageHdl.Call( this ); } // ----------------------------------------------------------------------- long TabBar::ImplDeactivatePage() { long nRet = DeactivatePage(); CallEventListeners( VCLEVENT_TABBAR_PAGEDEACTIVATED, reinterpret_cast(sal::static_int_cast(mnCurPageId)) ); return nRet; } void TabBar::ImplPrePaint() { USHORT nItemCount = (USHORT)mpItemList->size(); if (!nItemCount) return; ImplTabBarItem* pItem; // TabBar muss formatiert sein ImplFormat(); // Beim ersten Format auch dafuer sorgen, das aktuelle TabPage // sichtbar wird if ( mbFirstFormat ) { mbFirstFormat = FALSE; if ( mnCurPageId && (mnFirstPos == 0) && !mbDropPos ) { pItem = (*mpItemList)[ GetPagePos( mnCurPageId ) ]; if ( pItem->maRect.IsEmpty() ) { // mbDropPos setzen (bzw. misbrauchen) um Invalidate() // zu unterbinden mbDropPos = TRUE; SetFirstPageId( mnCurPageId ); mbDropPos = FALSE; if ( mnFirstPos != 0 ) ImplFormat(); } } } } ImplTabBarItem* TabBar::ImplGetLastTabBarItem( USHORT nItemCount ) { // letzten sichtbaren Eintrag suchen USHORT n = mnFirstPos+1; if ( n >= nItemCount ) n = nItemCount-1; ImplTabBarItem* pItem = seek( n ); while ( pItem ) { if ( !pItem->maRect.IsEmpty() ) { n++; pItem = next(); } else break; } // Alle Tabs ausgeben (von hinten nach vorn und aktuellen zuletzt) if ( pItem ) n--; else if ( n >= nItemCount ) n = nItemCount-1; pItem = seek( n ); return pItem; } Rectangle TabBar::ImplGetInsertTabRect(ImplTabBarItem* pItem) const { if (mbHasInsertTab && pItem) { Rectangle aInsTabRect = pItem->maRect; aInsTabRect.setX( aInsTabRect.getX() + aInsTabRect.getWidth() - TABBAR_OFFSET_X - TABBAR_OFFSET_X2); aInsTabRect.setWidth(32); return aInsTabRect; } return Rectangle(); } // ----------------------------------------------------------------------- long TabBar::DeactivatePage() { if ( maDeactivatePageHdl.IsSet() ) return maDeactivatePageHdl.Call( this ); else return TRUE; } // ----------------------------------------------------------------------- long TabBar::StartRenaming() { if ( maStartRenamingHdl.IsSet() ) return maStartRenamingHdl.Call( this ); else return TRUE; } // ----------------------------------------------------------------------- long TabBar::AllowRenaming() { if ( maAllowRenamingHdl.IsSet() ) return maAllowRenamingHdl.Call( this ); else return TRUE; } // ----------------------------------------------------------------------- void TabBar::EndRenaming() { maEndRenamingHdl.Call( this ); } // ----------------------------------------------------------------------- void TabBar::Mirror() { } // ----------------------------------------------------------------------- void TabBar::InsertPage( USHORT nPageId, const XubString& rText, TabBarPageBits nBits, USHORT nPos ) { DBG_ASSERT( nPageId, "TabBar::InsertPage(): PageId == 0" ); DBG_ASSERT( GetPagePos( nPageId ) == PAGE_NOT_FOUND, "TabBar::InsertPage(): PageId already exists" ); DBG_ASSERT( nBits <= TPB_SPECIAL, "TabBar::InsertPage(): nBits is wrong" ); // PageItem anlegen und in die Item-Liste eintragen ImplTabBarItem* pItem = new ImplTabBarItem( nPageId, rText, nBits ); if ( nPos < mpItemList->size() ) { ImplTabBarList::iterator it = mpItemList->begin(); ::std::advance( it, nPos ); mpItemList->insert( it, pItem ); } else { mpItemList->push_back( pItem ); } mbSizeFormat = TRUE; // CurPageId gegebenenfalls setzen if ( !mnCurPageId ) mnCurPageId = nPageId; // Leiste neu ausgeben if ( IsReallyVisible() && IsUpdateMode() ) Invalidate(); CallEventListeners( VCLEVENT_TABBAR_PAGEINSERTED, reinterpret_cast(sal::static_int_cast(nPageId)) ); } // ----------------------------------------------------------------------- Color TabBar::GetTabBgColor( USHORT nPageId ) const { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) return (*mpItemList)[ nPos ]->maTabBgColor; else return Color( COL_AUTO ); } void TabBar::SetTabBgColor( USHORT nPageId, const Color& aTabBgColor ) { USHORT nPos = GetPagePos( nPageId ); ImplTabBarItem* pItem; if ( nPos != PAGE_NOT_FOUND ) { pItem = (*mpItemList)[ nPos ]; if ( aTabBgColor != Color( COL_AUTO ) ) { pItem->maTabBgColor = aTabBgColor; if ( aTabBgColor.GetLuminance() <= 128 ) //Do not use aTabBgColor.IsDark(), because that threshold is way too low... pItem->maTabTextColor = Color( COL_WHITE ); else pItem->maTabTextColor = Color( COL_BLACK ); } else { pItem->maTabBgColor = Color( COL_AUTO ); pItem->maTabTextColor = Color( COL_AUTO ); } } } // ----------------------------------------------------------------------- void TabBar::RemovePage( USHORT nPageId ) { USHORT nPos = GetPagePos( nPageId ); // Existiert Item if ( nPos != PAGE_NOT_FOUND ) { if ( mnCurPageId == nPageId ) mnCurPageId = 0; // Testen, ob erste sichtbare Seite verschoben werden muss if ( mnFirstPos > nPos ) mnFirstPos--; // Item-Daten loeschen ImplTabBarList::iterator it = mpItemList->begin(); ::std::advance( it, nPos ); delete *it; mpItemList->erase( it ); mbFormat = TRUE; // Leiste neu ausgeben if ( IsReallyVisible() && IsUpdateMode() ) Invalidate(); CallEventListeners( VCLEVENT_TABBAR_PAGEREMOVED, reinterpret_cast(sal::static_int_cast(nPageId)) ); } } // ----------------------------------------------------------------------- void TabBar::MovePage( USHORT nPageId, USHORT nNewPos ) { USHORT nPos = GetPagePos( nPageId ); Pair aPair( nPos, nNewPos ); if ( nPos < nNewPos ) nNewPos--; if ( nPos == nNewPos ) return; // Existiert Item if ( nPos != PAGE_NOT_FOUND ) { // TabBar-Item in der Liste verschieben ImplTabBarList::iterator it = mpItemList->begin(); ::std::advance( it, nPos ); ImplTabBarItem* pItem = *it; mpItemList->erase( it ); if ( nNewPos < mpItemList->size() ) { it = mpItemList->begin(); ::std::advance( it, nNewPos ); mpItemList->insert( it, pItem ); } else { mpItemList->push_back( pItem ); } mbFormat = TRUE; // Leiste neu ausgeben if ( IsReallyVisible() && IsUpdateMode() ) Invalidate(); CallEventListeners( VCLEVENT_TABBAR_PAGEMOVED, (void*) &aPair ); } } // ----------------------------------------------------------------------- void TabBar::Clear() { // Alle Items loeschen for ( size_t i = 0, n = mpItemList->size(); i < n; ++i ) { delete (*mpItemList)[ i ]; } mpItemList->clear(); // Items aus der Liste loeschen mbSizeFormat = TRUE; mnCurPageId = 0; mnFirstPos = 0; maCurrentItemList = 0; // Leiste neu ausgeben if ( IsReallyVisible() && IsUpdateMode() ) Invalidate(); CallEventListeners( VCLEVENT_TABBAR_PAGEREMOVED, (void*) PAGE_NOT_FOUND ); } // ----------------------------------------------------------------------- void TabBar::EnablePage( USHORT nPageId, BOOL bEnable ) { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) { ImplTabBarItem* pItem = (*mpItemList)[ nPos ]; if ( pItem->mbEnable != bEnable ) { pItem->mbEnable = bEnable; // Leiste neu ausgeben if ( IsReallyVisible() && IsUpdateMode() ) Invalidate( pItem->maRect ); CallEventListeners( bEnable ? VCLEVENT_TABBAR_PAGEENABLED : VCLEVENT_TABBAR_PAGEDISABLED, reinterpret_cast(sal::static_int_cast(nPageId)) ); } } } // ----------------------------------------------------------------------- BOOL TabBar::IsPageEnabled( USHORT nPageId ) const { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) return (*mpItemList)[ nPos ]->mbEnable; else return FALSE; } // ----------------------------------------------------------------------- void TabBar::SetPageBits( USHORT nPageId, TabBarPageBits nBits ) { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) { ImplTabBarItem* pItem = (*mpItemList)[ nPos ]; if ( pItem->mnBits != nBits ) { pItem->mnBits = nBits; // Leiste neu ausgeben if ( IsReallyVisible() && IsUpdateMode() ) Invalidate( pItem->maRect ); } } } // ----------------------------------------------------------------------- TabBarPageBits TabBar::GetPageBits( USHORT nPageId ) const { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) return (*mpItemList)[ nPos ]->mnBits; else return FALSE; } // ----------------------------------------------------------------------- USHORT TabBar::GetPageCount() const { return (USHORT)mpItemList->size(); } // ----------------------------------------------------------------------- USHORT TabBar::GetPageId( USHORT nPos ) const { return ( nPos < mpItemList->size() ) ? (*mpItemList)[ nPos ]->mnId : 0; } // ----------------------------------------------------------------------- USHORT TabBar::GetPagePos( USHORT nPageId ) const { for ( size_t i = 0, n = mpItemList->size(); i < n; ++i ) { if ( (*mpItemList)[ i ]->mnId == nPageId ) { return USHORT( i ); } } return PAGE_NOT_FOUND; } // ----------------------------------------------------------------------- sal_uInt16 TabBar::GetPageId( const Point& rPos, bool bCheckInsTab ) const { for ( size_t i = 0, n = mpItemList->size(); i < n; ++i ) { ImplTabBarItem* pItem = (*mpItemList)[ i ]; if ( pItem->maRect.IsInside( rPos ) ) return pItem->mnId; } if (bCheckInsTab && mbHasInsertTab && !mpItemList->empty()) { ImplTabBarItem* pItem = mpItemList->back(); if (ImplGetInsertTabRect(pItem).IsInside(rPos)) return INSERT_TAB_POS; } return 0; } // ----------------------------------------------------------------------- Rectangle TabBar::GetPageRect( USHORT nPageId ) const { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) return (*mpItemList)[ nPos ]->maRect; else return Rectangle(); } // ----------------------------------------------------------------------- void TabBar::SetCurPageId( USHORT nPageId ) { USHORT nPos = GetPagePos( nPageId ); // Wenn Item nicht existiert, dann nichts machen if ( nPos != PAGE_NOT_FOUND ) { // Wenn sich aktuelle Page nicht geaendert hat, dann muessen wir // jetzt nichts mehr machen if ( nPageId == mnCurPageId ) return; // Muss invalidiert werden BOOL bUpdate = FALSE; if ( IsReallyVisible() && IsUpdateMode() ) bUpdate = TRUE; ImplTabBarItem* pItem = (*mpItemList)[ nPos ]; ImplTabBarItem* pOldItem; if ( mnCurPageId ) pOldItem = (*mpItemList)[ GetPagePos( mnCurPageId ) ]; else pOldItem = NULL; // Wenn Page nicht selektiert, dann vorher selektierte Seite // deselktieren, wenn dies die einzige selektierte Seite ist if ( !pItem->mbSelect && pOldItem ) { USHORT nSelPageCount = GetSelectPageCount(); if ( nSelPageCount == 1 ) pOldItem->mbSelect = FALSE; pItem->mbSelect = TRUE; } mnCurPageId = nPageId; mbFormat = TRUE; // Dafuer sorgen, das aktuelle Page sichtbar wird if ( IsReallyVisible() ) { if ( nPos < mnFirstPos ) SetFirstPageId( nPageId ); else { // sichtbare Breite berechnen long nWidth = mnLastOffX; if ( nWidth > TABBAR_OFFSET_X ) nWidth -= TABBAR_OFFSET_X; if ( nWidth > ADDNEWPAGE_AREAWIDTH ) nWidth -= ADDNEWPAGE_AREAWIDTH; if ( pItem->maRect.IsEmpty() ) ImplFormat(); while ( (mbMirrored ? (pItem->maRect.Left() < mnOffX) : (pItem->maRect.Right() > nWidth)) || pItem->maRect.IsEmpty() ) { USHORT nNewPos = mnFirstPos+1; // Dafuer sorgen, das min. die aktuelle TabPages als // erste TabPage sichtbar ist if ( nNewPos >= nPos ) { SetFirstPageId( nPageId ); break; } else SetFirstPageId( GetPageId( nNewPos ) ); ImplFormat(); // Falls erste Seite nicht weitergeschaltet wird, dann // koennen wir abbrechen if ( nNewPos != mnFirstPos ) break; } } } // Leiste neu ausgeben if ( bUpdate ) { Invalidate( pItem->maRect ); if ( pOldItem ) Invalidate( pOldItem->maRect ); } } } // ----------------------------------------------------------------------- void TabBar::MakeVisible( USHORT nPageId ) { if ( !IsReallyVisible() ) return; USHORT nPos = GetPagePos( nPageId ); // Wenn Item nicht existiert, dann nichts machen if ( nPos != PAGE_NOT_FOUND ) { if ( nPos < mnFirstPos ) SetFirstPageId( nPageId ); else { ImplTabBarItem* pItem = (*mpItemList)[ nPos ]; // sichtbare Breite berechnen long nWidth = mnLastOffX; if ( nWidth > TABBAR_OFFSET_X ) nWidth -= TABBAR_OFFSET_X; if ( mbFormat || pItem->maRect.IsEmpty() ) { mbFormat = TRUE; ImplFormat(); } while ( (pItem->maRect.Right() > nWidth) || pItem->maRect.IsEmpty() ) { USHORT nNewPos = mnFirstPos+1; // Dafuer sorgen, das min. die aktuelle TabPages als // erste TabPage sichtbar ist if ( nNewPos >= nPos ) { SetFirstPageId( nPageId ); break; } else SetFirstPageId( GetPageId( nNewPos ) ); ImplFormat(); // Falls erste Seite nicht weitergeschaltet wird, dann // koennen wir abbrechen if ( nNewPos != mnFirstPos ) break; } } } } // ----------------------------------------------------------------------- void TabBar::SetFirstPageId( USHORT nPageId ) { USHORT nPos = GetPagePos( nPageId ); // Wenn Item nicht existiert, dann FALSE zurueckgeben if ( nPos != PAGE_NOT_FOUND ) { if ( nPos != mnFirstPos ) { // Dafuer sorgen, das nach Moeglichkteit soviele Pages wie // moeglich sichtbar sind ImplFormat(); USHORT nLastFirstPos = ImplGetLastFirstPos(); USHORT nNewPos; if ( nPos > nLastFirstPos ) nNewPos = nLastFirstPos; else nNewPos = nPos; if ( nNewPos != mnFirstPos ) { mnFirstPos = nNewPos; mbFormat = TRUE; // Leiste neu ausgeben (Achtung: mbDropPos beachten, da wenn // dieses Flag gesetzt ist, wird direkt gepaintet) if ( IsReallyVisible() && IsUpdateMode() && !mbDropPos ) Invalidate(); } } } } // ----------------------------------------------------------------------- void TabBar::SelectPage( USHORT nPageId, BOOL bSelect ) { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) { ImplTabBarItem* pItem = (*mpItemList)[ nPos ]; if ( pItem->mbSelect != bSelect ) { pItem->mbSelect = bSelect; // Leiste neu ausgeben if ( IsReallyVisible() && IsUpdateMode() ) Invalidate( pItem->maRect ); } } } // ----------------------------------------------------------------------- void TabBar::SelectPageRange( BOOL bSelect, USHORT nStartPos, USHORT nEndPos ) { Rectangle aPaintRect; USHORT nPos = nStartPos; ImplTabBarItem* pItem = seek( nPos ); while ( pItem && (nPos <= nEndPos) ) { if ( (pItem->mbSelect != bSelect) && (pItem->mnId != mnCurPageId) ) { pItem->mbSelect = bSelect; aPaintRect.Union( pItem->maRect ); } nPos++; pItem = next(); } // Leiste neu ausgeben if ( IsReallyVisible() && IsUpdateMode() && !aPaintRect.IsEmpty() ) Invalidate( aPaintRect ); } // ----------------------------------------------------------------------- USHORT TabBar::GetSelectPage( USHORT nSelIndex ) const { USHORT nSelected = 0; for ( size_t i = 0, n = mpItemList->size(); i < n; ++i ) { ImplTabBarItem* pItem = (*mpItemList)[ i ]; if ( pItem->mbSelect ) nSelected++; if ( nSelected == nSelIndex ) return pItem->mnId; } return 0; } // ----------------------------------------------------------------------- USHORT TabBar::GetSelectPageCount() const { USHORT nSelected = 0; for ( size_t i = 0, n = mpItemList->size(); i < n; ++i ) { ImplTabBarItem* pItem = (*mpItemList)[ i ]; if ( pItem->mbSelect ) nSelected++; } return nSelected; } // ----------------------------------------------------------------------- BOOL TabBar::IsPageSelected( USHORT nPageId ) const { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) return (*mpItemList)[ nPos ]->mbSelect; else return FALSE; } // ----------------------------------------------------------------------- BOOL TabBar::StartEditMode( USHORT nPageId ) { USHORT nPos = GetPagePos( nPageId ); if ( mpEdit || (nPos == PAGE_NOT_FOUND) || (mnLastOffX < 8) ) return FALSE; mnEditId = nPageId; if ( StartRenaming() ) { ImplShowPage( nPos ); ImplFormat(); Update(); mpEdit = new TabBarEdit( this, WB_CENTER ); Rectangle aRect = GetPageRect( mnEditId ); long nX = aRect.Left()+TABBAR_OFFSET_X+(TABBAR_OFFSET_X2/2); long nWidth = aRect.GetWidth()-(TABBAR_OFFSET_X*2)-TABBAR_OFFSET_X2; if ( mnEditId != GetCurPageId() ) nX += 1; if ( nX+nWidth > mnLastOffX ) nWidth = mnLastOffX-nX; if ( nWidth < 3 ) { nX = aRect.Left(); nWidth = aRect.GetWidth(); } mpEdit->SetText( GetPageText( mnEditId ) ); mpEdit->SetPosSizePixel( nX, aRect.Top()+mnOffY+1, nWidth, aRect.GetHeight()-3 ); Font aFont = GetPointFont(); Color aForegroundColor; Color aBackgroundColor; Color aFaceColor; Color aSelectColor; Color aFaceTextColor; Color aSelectTextColor; ImplGetColors( aFaceColor, aFaceTextColor, aSelectColor, aSelectTextColor ); if ( mnEditId != GetCurPageId() ) aFont.SetWeight( WEIGHT_LIGHT ); if ( IsPageSelected( mnEditId ) || (mnEditId == GetCurPageId()) ) { aForegroundColor = aSelectTextColor; aBackgroundColor = aSelectColor; } else { aForegroundColor = aFaceTextColor; aBackgroundColor = aFaceColor; } if ( GetPageBits( mnEditId ) & TPB_SPECIAL ) aForegroundColor = Color( COL_LIGHTBLUE ); mpEdit->SetControlFont( aFont ); mpEdit->SetControlForeground( aForegroundColor ); mpEdit->SetControlBackground( aBackgroundColor ); mpEdit->GrabFocus(); mpEdit->SetSelection( Selection( 0, mpEdit->GetText().Len() ) ); mpEdit->Show(); return TRUE; } else { mnEditId = 0; return FALSE; } } // ----------------------------------------------------------------------- void TabBar::EndEditMode( BOOL bCancel ) { if ( mpEdit ) { // call hdl BOOL bEnd = TRUE; mbEditCanceled = bCancel; maEditText = mpEdit->GetText(); mpEdit->SetPostEvent(); if ( !bCancel ) { long nAllowRenaming = AllowRenaming(); if ( nAllowRenaming == TABBAR_RENAMING_YES ) SetPageText( mnEditId, maEditText ); else if ( nAllowRenaming == TABBAR_RENAMING_NO ) bEnd = FALSE; else // nAllowRenaming == TABBAR_RENAMING_CANCEL mbEditCanceled = TRUE; } // renaming not allowed, than reset edit data if ( !bEnd ) { mpEdit->ResetPostEvent(); mpEdit->GrabFocus(); } else { // close edit and call end hdl delete mpEdit; mpEdit = NULL; EndRenaming(); mnEditId = 0; } // reset maEditText.Erase(); mbEditCanceled = FALSE; } } // ----------------------------------------------------------------------- void TabBar::SetMirrored( BOOL bMirrored ) { if( mbMirrored != bMirrored ) { mbMirrored = bMirrored; mbSizeFormat = TRUE; ImplInitControls(); // for button images Resize(); // recalculates control positions Mirror(); } } void TabBar::SetEffectiveRTL( BOOL bRTL ) { SetMirrored( bRTL != Application::GetSettings().GetLayoutRTL() ); } BOOL TabBar::IsEffectiveRTL() const { return IsMirrored() != Application::GetSettings().GetLayoutRTL(); } // ----------------------------------------------------------------------- void TabBar::SetMaxPageWidth( long nMaxWidth ) { if ( mnMaxPageWidth != nMaxWidth ) { mnMaxPageWidth = nMaxWidth; mbSizeFormat = TRUE; // Leiste neu ausgeben if ( IsReallyVisible() && IsUpdateMode() ) Invalidate(); } } // ----------------------------------------------------------------------- void TabBar::SetSelectColor() { if ( mbSelColor ) { maSelColor = Color( COL_TRANSPARENT ); mbSelColor = FALSE; Invalidate(); } } // ----------------------------------------------------------------------- void TabBar::SetSelectColor( const Color& rColor ) { if ( rColor.GetTransparency() ) { if ( mbSelColor ) { maSelColor = Color( COL_TRANSPARENT ); mbSelColor = FALSE; Invalidate(); } } else { if ( maSelColor != rColor ) { maSelColor = rColor; mbSelColor = TRUE; Invalidate(); } } } // ----------------------------------------------------------------------- void TabBar::SetSelectTextColor() { if ( mbSelTextColor ) { maSelTextColor = Color( COL_TRANSPARENT ); mbSelTextColor = FALSE; Invalidate(); } } // ----------------------------------------------------------------------- void TabBar::SetSelectTextColor( const Color& rColor ) { if ( rColor.GetTransparency() ) { if ( mbSelTextColor ) { maSelTextColor = Color( COL_TRANSPARENT ); mbSelTextColor = FALSE; Invalidate(); } } else { if ( maSelTextColor != rColor ) { maSelTextColor = rColor; mbSelTextColor = TRUE; Invalidate(); } } } // ----------------------------------------------------------------------- void TabBar::SetPageText( USHORT nPageId, const XubString& rText ) { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) { (*mpItemList)[ nPos ]->maText = rText; mbSizeFormat = TRUE; // Leiste neu ausgeben if ( IsReallyVisible() && IsUpdateMode() ) Invalidate(); CallEventListeners( VCLEVENT_TABBAR_PAGETEXTCHANGED, reinterpret_cast(sal::static_int_cast(nPageId)) ); } } // ----------------------------------------------------------------------- XubString TabBar::GetPageText( USHORT nPageId ) const { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) return (*mpItemList)[ nPos ]->maText; else return XubString(); } // ----------------------------------------------------------------------- void TabBar::SetHelpText( USHORT nPageId, const XubString& rText ) { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) (*mpItemList)[ nPos ]->maHelpText = rText; } // ----------------------------------------------------------------------- XubString TabBar::GetHelpText( USHORT nPageId ) const { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) { ImplTabBarItem* pItem = (*mpItemList)[ nPos ]; if ( !pItem->maHelpText.Len() && pItem->mnHelpId ) { Help* pHelp = Application::GetHelp(); if ( pHelp ) pItem->maHelpText = pHelp->GetHelpText( pItem->mnHelpId, this ); } return pItem->maHelpText; } else return XubString(); } // ----------------------------------------------------------------------- void TabBar::SetHelpId( USHORT nPageId, ULONG nHelpId ) { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) (*mpItemList)[ nPos ]->mnHelpId = nHelpId; } // ----------------------------------------------------------------------- ULONG TabBar::GetHelpId( USHORT nPageId ) const { USHORT nPos = GetPagePos( nPageId ); if ( nPos != PAGE_NOT_FOUND ) return (*mpItemList)[ nPos ]->mnHelpId; else return 0; } // ----------------------------------------------------------------------- long TabBar::GetMinSize() const { long nMinSize = TABBAR_MINSIZE + TABBAR_OFFSET_X; if ( mnWinStyle & WB_MINSCROLL ) nMinSize += mpPrevBtn->GetSizePixel().Width()*2; else if ( mnWinStyle & WB_SCROLL ) nMinSize += mpFirstBtn->GetSizePixel().Width()*4; return nMinSize; } // ----------------------------------------------------------------------- BOOL TabBar::StartDrag( const CommandEvent& rCEvt, Region& rRegion ) { if ( !(mnWinStyle & WB_DRAG) || (rCEvt.GetCommand() != COMMAND_STARTDRAG) ) return FALSE; // Testen, ob angeklickte Seite selektiert ist. Falls dies nicht // der Fall ist, setzen wir ihn als aktuellen Eintrag. Falls Drag and // Drop auch mal ueber Tastatur ausgeloest werden kann, testen wir // dies nur bei einer Mausaktion. // Ausserdem machen wir das nur, wenn kein Select() ausgeloest wurde, // da der Select schon den Bereich gescrollt haben kann if ( rCEvt.IsMouseEvent() && !mbInSelect ) { USHORT nSelId = GetPageId( rCEvt.GetMousePosPixel() ); // Falls kein Eintrag angeklickt wurde, starten wir kein Dragging if ( !nSelId ) return FALSE; // Testen, ob Seite selektiertiert ist. Falls nicht, als aktuelle // Seite setzen und Select rufen. if ( !IsPageSelected( nSelId ) ) { if ( ImplDeactivatePage() ) { SetCurPageId( nSelId ); Update(); ImplActivatePage(); ImplSelect(); } else return FALSE; } } mbInSelect = FALSE; Region aRegion; // Region zuweisen rRegion = aRegion; return TRUE; } // ----------------------------------------------------------------------- USHORT TabBar::ShowDropPos( const Point& rPos ) { ImplTabBarItem* pItem; USHORT nDropId; USHORT nNewDropPos; USHORT nItemCount = (USHORT)mpItemList->size(); short nScroll = 0; if ( rPos.X() > mnLastOffX-TABBAR_DRAG_SCROLLOFF ) { pItem = (*mpItemList)[ mpItemList->size()-1 ]; if ( !pItem->maRect.IsEmpty() && (rPos.X() > pItem->maRect.Right()) ) nNewDropPos = (USHORT)mpItemList->size(); else { nNewDropPos = mnFirstPos+1; nScroll = 1; } } else if ( (rPos.X() <= mnOffX) || (!mnOffX && (rPos.X() <= TABBAR_DRAG_SCROLLOFF)) ) { if ( mnFirstPos ) { nNewDropPos = mnFirstPos; nScroll = -1; } else nNewDropPos = 0; } else { nDropId = GetPageId( rPos ); if ( nDropId ) { nNewDropPos = GetPagePos( nDropId ); if ( mnFirstPos && (nNewDropPos == mnFirstPos-1) ) nScroll = -1; } else nNewDropPos = nItemCount; } if ( mbDropPos && (nNewDropPos == mnDropPos) && !nScroll ) return mnDropPos; if ( mbDropPos ) HideDropPos(); mbDropPos = TRUE; mnDropPos = nNewDropPos; if ( nScroll ) { USHORT nOldFirstPos = mnFirstPos; SetFirstPageId( GetPageId( mnFirstPos+nScroll ) ); // Direkt ausgeben, da kein Paint bei Drag and Drop moeglich if ( nOldFirstPos != mnFirstPos ) { Rectangle aRect( mnOffX, 0, mnLastOffX, maWinSize.Height() ); SetFillColor( GetBackground().GetColor() ); DrawRect( aRect ); Paint( aRect ); } } // Drop-Position-Pfeile ausgeben Color aBlackColor( COL_BLACK ); long nX; long nY = (maWinSize.Height()/2)-1; USHORT nCurPos = GetPagePos( mnCurPageId ); SetLineColor( aBlackColor ); if ( mnDropPos < nItemCount ) { pItem = (*mpItemList)[ mnDropPos ]; nX = pItem->maRect.Left()+TABBAR_OFFSET_X; if ( mnDropPos == nCurPos ) nX--; else nX++; if ( !pItem->IsDefaultTabBgColor() && !pItem->mbSelect) SetLineColor( pItem->maTabTextColor ); DrawLine( Point( nX, nY ), Point( nX, nY ) ); DrawLine( Point( nX+1, nY-1 ), Point( nX+1, nY+1 ) ); DrawLine( Point( nX+2, nY-2 ), Point( nX+2, nY+2 ) ); SetLineColor( aBlackColor ); } if ( (mnDropPos > 0) && (mnDropPos < nItemCount+1) ) { pItem = (*mpItemList)[ mnDropPos-1 ]; nX = pItem->maRect.Right()-TABBAR_OFFSET_X; if ( mnDropPos == nCurPos ) nX++; if ( !pItem->IsDefaultTabBgColor() && !pItem->mbSelect) SetLineColor( pItem->maTabTextColor ); DrawLine( Point( nX, nY ), Point( nX, nY ) ); DrawLine( Point( nX-1, nY-1 ), Point( nX-1, nY+1 ) ); DrawLine( Point( nX-2, nY-2 ), Point( nX-2, nY+2 ) ); } return mnDropPos; } // ----------------------------------------------------------------------- void TabBar::HideDropPos() { if ( mbDropPos ) { ImplTabBarItem* pItem; long nX; long nY1 = (maWinSize.Height()/2)-3; long nY2 = nY1 + 5; USHORT nItemCount = (USHORT)mpItemList->size(); if ( mnDropPos < nItemCount ) { pItem = (*mpItemList)[ mnDropPos ]; nX = pItem->maRect.Left()+TABBAR_OFFSET_X; // Paint direkt aufrufen, da bei Drag and Drop kein Paint // moeglich Rectangle aRect( nX-1, nY1, nX+3, nY2 ); Region aRegion( aRect ); SetClipRegion( aRegion ); Paint( aRect ); SetClipRegion(); } if ( (mnDropPos > 0) && (mnDropPos < nItemCount+1) ) { pItem = (*mpItemList)[ mnDropPos-1 ]; nX = pItem->maRect.Right()-TABBAR_OFFSET_X; // Paint direkt aufrufen, da bei Drag and Drop kein Paint // moeglich Rectangle aRect( nX-2, nY1, nX+1, nY2 ); Region aRegion( aRect ); SetClipRegion( aRegion ); Paint( aRect ); SetClipRegion(); } mbDropPos = FALSE; mnDropPos = 0; } } // ----------------------------------------------------------------------- BOOL TabBar::SwitchPage( const Point& rPos ) { BOOL bSwitch = FALSE; USHORT nSwitchId = GetPageId( rPos ); if ( !nSwitchId ) EndSwitchPage(); else { if ( nSwitchId != mnSwitchId ) { mnSwitchId = nSwitchId; mnSwitchTime = Time::GetSystemTicks(); } else { // Erst nach 500 ms umschalten if ( mnSwitchId != GetCurPageId() ) { if ( Time::GetSystemTicks() > mnSwitchTime+500 ) { mbInSwitching = TRUE; if ( ImplDeactivatePage() ) { SetCurPageId( mnSwitchId ); Update(); ImplActivatePage(); ImplSelect(); bSwitch = TRUE; } mbInSwitching = FALSE; } } } } return bSwitch; } // ----------------------------------------------------------------------- void TabBar::EndSwitchPage() { mnSwitchTime = 0; mnSwitchId = 0; } // ----------------------------------------------------------------------- void TabBar::SetStyle( WinBits nStyle ) { mnWinStyle = nStyle; ImplInitControls(); // Evt. Controls neu anordnen if ( IsReallyVisible() && IsUpdateMode() ) Resize(); } // ----------------------------------------------------------------------- Size TabBar::CalcWindowSizePixel() const { long nWidth = 0; if ( mpItemList->size() ) { ((TabBar*)this)->ImplCalcWidth(); for ( size_t i = 0, n = mpItemList->size(); i < n; ++i ) { ImplTabBarItem* pItem = (*mpItemList)[ i ]; nWidth += pItem->mnWidth; } nWidth += TABBAR_OFFSET_X+TABBAR_OFFSET_X2; } return Size( nWidth, GetSettings().GetStyleSettings().GetScrollBarSize() ); } // ----------------------------------------------------------------------- Rectangle TabBar::GetPageArea() const { return Rectangle( Point( mnOffX, mnOffY ), Size( mnLastOffX-mnOffX+1, GetSizePixel().Height()-mnOffY ) ); } // ----------------------------------------------------------------------- ::com::sun::star::uno::Reference< ::com::sun::star::accessibility::XAccessible > TabBar::CreateAccessible() { return mpImpl->maAccessibleFactory.getFactory().createAccessibleTabBar( *this ); } // ----------------------------------------------------------------------- /* vim:set shiftwidth=4 softtabstop=4 expandtab: */