summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFrank Schoenheit [fs] <frank.schoenheit@sun.com>2010-03-08 12:37:13 +0100
committerFrank Schoenheit [fs] <frank.schoenheit@sun.com>2010-03-08 12:37:13 +0100
commitfebb0caf8ebd7e2ea0c31666ce513e37b5b09fd9 (patch)
tree9e36833d0010a10d25d4a0929873b1369d4f4621
parente81174982a2cf99c4911391bb5c326789943135c (diff)
slidecopy: buttons for scrolling tab items when they do not fit
-rw-r--r--svtools/source/toolpanel/makefile.mk1
-rw-r--r--svtools/source/toolpanel/paneltabbar.cxx617
-rw-r--r--svtools/source/toolpanel/tabbargeometry.cxx309
-rw-r--r--svtools/source/toolpanel/tabbargeometry.hxx141
-rw-r--r--svtools/source/toolpanel/tabitemdescriptor.hxx90
-rw-r--r--svtools/workben/toolpanel/toolpaneltest.cxx1
6 files changed, 829 insertions, 330 deletions
diff --git a/svtools/source/toolpanel/makefile.mk b/svtools/source/toolpanel/makefile.mk
index 42c12ab99ead..e3dd07fac005 100644
--- a/svtools/source/toolpanel/makefile.mk
+++ b/svtools/source/toolpanel/makefile.mk
@@ -48,6 +48,7 @@ SLOFILES=\
$(SLO)$/paneldecklisteners.obj \
$(SLO)$/paneltabbar.obj \
$(SLO)$/refbase.obj \
+ $(SLO)$/tabbargeometry.obj \
$(SLO)$/tablayouter.obj \
$(SLO)$/toolpanelcollection.obj \
$(SLO)$/toolpaneldeck.obj \
diff --git a/svtools/source/toolpanel/paneltabbar.cxx b/svtools/source/toolpanel/paneltabbar.cxx
index 0aa9b8aca035..f50dfb1c3ff0 100644
--- a/svtools/source/toolpanel/paneltabbar.cxx
+++ b/svtools/source/toolpanel/paneltabbar.cxx
@@ -27,11 +27,14 @@
#include "precompiled_svtools.hxx"
#include "svtools/toolpanel/toolpaneldeck.hxx"
+#include "tabitemdescriptor.hxx"
#include "paneltabbar.hxx"
+#include "tabbargeometry.hxx"
#include <basegfx/range/b2drange.hxx>
#include <basegfx/matrix/b2dhommatrix.hxx>
#include <basegfx/numeric/ftools.hxx>
+#include <vcl/button.hxx>
#include <vcl/help.hxx>
#include <vcl/virdev.hxx>
@@ -40,10 +43,10 @@
// space around an item
#define ITEM_OUTER_SPACE 2 * 2
+// the distance between the button and the control's border, in "upright" direction (what this means depends on the alignment)
+#define BUTTON_UPRIGHT_DISTANCE 2
// spacing before and after (in writing direction, whether this is horizontal or vertical) an item's text
#define ITEM_TEXT_FLOW_SPACE 5
-// distance between two items
-#define ITEM_DISTANCE_PIXEL 0
// space between item icon and icon text
#define ITEM_ICON_TEXT_DISTANCE 4
@@ -62,48 +65,6 @@ namespace svt
#define ITEM_POSITION_LAST 0x10
//==================================================================================================================
- //= ItemDescriptor
- //==================================================================================================================
- struct ItemDescriptor
- {
- PToolPanel pPanel;
- Rectangle aCompleteArea;
- Rectangle aIconOnlyArea;
- Rectangle aTextOnlyArea;
- TabItemContent eContent;
- // content to be used for this particular item. Might differ from item content which has been set
- // up for the complete control, in case not the complete content fits into the available space.
-
- ItemDescriptor()
- :pPanel()
- ,aCompleteArea()
- ,aIconOnlyArea()
- ,aTextOnlyArea()
- ,eContent( TABITEM_IMAGE_AND_TEXT )
- {
- }
-
- const Rectangle& GetRect( const TabItemContent i_eItemContent ) const
- {
- OSL_ENSURE( i_eItemContent != TABITEM_AUTO, "ItemDescriptor::GetRect: illegal value!" );
-
- return ( i_eItemContent == TABITEM_IMAGE_AND_TEXT )
- ? aCompleteArea
- : ( ( i_eItemContent == TABITEM_TEXT_ONLY )
- ? aTextOnlyArea
- : aIconOnlyArea
- );
- }
-
- const Rectangle& GetCurrentRect() const
- {
- return GetRect( eContent );
- }
- };
-
- typedef ::std::vector< ItemDescriptor > ItemDescriptors;
-
- //==================================================================================================================
//= IItemsLayout
//==================================================================================================================
// TODO: originally, IItemsLayout was intended to abstract from the "vertical vs. horizontal" problem. Meanwhile,
@@ -116,6 +77,10 @@ namespace svt
class IItemsLayout
{
public:
+ /** fills the background of our target device
+ */
+ virtual void DrawBackground( const Rectangle& i_rArea ) = 0;
+
/** calculates the size of the area occupied by the item representing the given tool panel
@param i_pPanel
denotes the panel whose item's size should be calculated
@@ -128,36 +93,26 @@ namespace svt
contains, upon return, the area which is available to render the item content. It lies completely
within an assuming bounding rectangle, which starts at corrdinates (0, 0), and has the size returned
in o_rBoundingSize.
-
*/
- virtual void CalculateItemSize(
- const PToolPanel& i_pPanel,
- const TabItemContent i_eItemContent,
- Size& o_rBoundingSize,
- Rectangle& o_rContentArea
- ) const = 0;
-
- /** returns the position where the next item should start, assuming the previous item occupies a given area
- */
- virtual Point GetNextItemPosition( const Rectangle& i_rPreviousItemArea ) const = 0;
-
- /** fills the background of our target device
- */
- virtual void DrawBackground( const Rectangle& i_rArea ) = 0;
+ virtual void CalculateItemSize( const PToolPanel& i_pPanel, const TabItemContent i_eItemContent, Size& o_rBoundingSize, Rectangle& o_rContentArea ) const = 0;
/** draws the item onto the given device, in the givem area
@param i_pPanel
the panel whose item representation is to be drawn
@param i_rPosition
- the position to paint the item to. Starting at this point, an area of the size returned
- as "bounding size" in CalculateItemSize might be used.
+ the position to paint the item to.
@param i_nItemFlags
defines in which state to draw the item
@param i_eItemContent
defines which content to draw on the tab item
*/
- virtual void DrawItem( const PToolPanel& i_pPanel, const Point& i_rPosition,
- const ItemFlags i_nItemFlags, const TabItemContent i_eItemContent ) = 0;
+ virtual void DrawItem(
+ const TabBarGeometry& i_rGeometry,
+ const PToolPanel& i_pPanel,
+ const Point& i_rPosition,
+ const ItemFlags i_nItemFlags,
+ const TabItemContent i_eItemContent
+ ) = 0;
};
typedef ::boost::shared_ptr< IItemsLayout > PItemsLayout;
@@ -171,15 +126,15 @@ namespace svt
VerticalItemLayout( Window& i_rTargetWindow, const bool i_bLeft );
// IItemsLayout overridables
+ virtual void DrawBackground( const Rectangle& i_rArea );
virtual void CalculateItemSize(
const PToolPanel& i_pPanel,
const TabItemContent i_eItemContent,
Size& o_rBoundingSize,
Rectangle& o_rContentArea
) const;
- virtual Point GetNextItemPosition( const Rectangle& i_rPreviousItemArea ) const;
- virtual void DrawBackground( const Rectangle& i_rArea );
virtual void DrawItem(
+ const TabBarGeometry& i_rGeometry,
const PToolPanel& i_pPanel,
const Point& i_rPosition,
const ItemFlags i_nItemFlags,
@@ -203,15 +158,9 @@ namespace svt
const ItemFlags& i_nFlags
);
- /** rotates a pair of bounding/content rect in formation, so they properly keep their relative position,
- and so that the upper left corner of the bounding rect is constant.
- */
- static void
- impl_rotateFormation( Rectangle& io_rBoundingRect, Rectangle& io_rContentRect, const bool i_bLeft );
-
private:
- Window& m_rTargetWindow;
- const bool m_bLeft;
+ Window& m_rTargetWindow;
+ const bool m_bLeft;
enum ItemRenderMode
{
@@ -246,143 +195,6 @@ namespace svt
}
//------------------------------------------------------------------------------------------------------------------
- void VerticalItemLayout::impl_rotateFormation( Rectangle& io_rBoundingRect, Rectangle& io_rContentRect, const bool i_bLeft )
- {
- // move the to-be-upper-left corner to (0,0)
- ::basegfx::B2DHomMatrix aTransformation;
- aTransformation.translate(
- i_bLeft ? -io_rBoundingRect.Right() : -io_rBoundingRect.Left(),
- i_bLeft ? -io_rBoundingRect.Top() : -io_rBoundingRect.Bottom() );
- // rotate by 90 resp. -90 degrees
- aTransformation.rotate( i_bLeft ? -F_PI2 : F_PI2 );
- // note on the screen, the ordinate goes top-down, while basegfx calculates in a system where the ordinate
- // does bottom-up; thus the "wrong" sign before F_PI2 here
- // move back to original coordinates
- aTransformation.translate( io_rBoundingRect.Left(), io_rBoundingRect.Top() );
-
- // apply
- ::basegfx::B2DRange aBoundingRange( io_rBoundingRect.Left(), io_rBoundingRect.Top(), io_rBoundingRect.Right(), io_rBoundingRect.Bottom() );
- aBoundingRange.transform( aTransformation );
- io_rBoundingRect.Left() = long( aBoundingRange.getMinX() );
- io_rBoundingRect.Top() = long( aBoundingRange.getMinY() );
- io_rBoundingRect.Right() = long( aBoundingRange.getMaxX() );
- io_rBoundingRect.Bottom() = long( aBoundingRange.getMaxY() );
-
- ::basegfx::B2DRange aContentRange( io_rContentRect.Left(), io_rContentRect.Top(), io_rContentRect.Right(), io_rContentRect.Bottom() );
- aContentRange.transform( aTransformation );
- io_rContentRect.Left() = long( aContentRange.getMinX() );
- io_rContentRect.Top() = long( aContentRange.getMinY() );
- io_rContentRect.Right() = long( aContentRange.getMaxX() );
- io_rContentRect.Bottom() = long( aContentRange.getMaxY() );
- }
-
- //------------------------------------------------------------------------------------------------------------------
- void VerticalItemLayout::CalculateItemSize( const PToolPanel& i_pPanel, const TabItemContent i_eItemContent,
- Size& o_rBoundingSize, Rectangle& o_rContentArea ) const
- {
- OSL_ENSURE( i_eItemContent != TABITEM_AUTO, "VerticalItemLayout::CalculateItemSize: illegal TabItemContent value!" );
-
- const Image aImage( i_pPanel->GetImage() );
- const bool bUseImage = !!aImage && ( i_eItemContent != TABITEM_TEXT_ONLY );
-
- const ::rtl::OUString sItemText( i_pPanel->GetDisplayName() );
- const bool bUseText = ( sItemText.getLength() != 0 ) && ( i_eItemContent != TABITEM_IMAGE_ONLY );
-
- Size aItemContentSize;
- if ( bUseImage )
- {
- aItemContentSize = aImage.GetSizePixel();
- }
-
- if ( bUseText )
- {
- if ( bUseImage )
- aItemContentSize.Height() += ITEM_ICON_TEXT_DISTANCE;
-
- // add space for vertical text
- const Size aTextSize( m_rTargetWindow.GetCtrlTextWidth( sItemText ), m_rTargetWindow.GetTextHeight() );
- aItemContentSize.Height() += aTextSize.Width();
- aItemContentSize.Width() = ::std::max( aItemContentSize.Width(), aTextSize.Height() );
-
- aItemContentSize.Height() += 2 * ITEM_TEXT_FLOW_SPACE;
- }
-
- aItemContentSize.Width() += 2 * ITEM_OUTER_SPACE;
- aItemContentSize.Height() += 2 * ITEM_OUTER_SPACE;
-
- Region aBoundingRegion, aContentRegion;
- bool bNativeOK = false;
- if ( m_eRenderMode == NWF_TOOLBAR_ITEM )
- {
- // don't ask GetNativeControlRegion, this will not deliver proper results in all cases.
- // Instead, simply assume that both the content and the bounding region are the same.
-// const ImplControlValue aControlValue;
-// bNativeOK = m_rTargetWindow.GetNativeControlRegion(
-// CTRL_TOOLBAR, PART_BUTTON,
-// Rectangle( Point(), aItemContentSize ), CTRL_STATE_ENABLED | CTRL_STATE_ROLLOVER,
-// aControlValue, ::rtl::OUString(),
-// aBoundingRegion, aContentRegion
-// );
- aContentRegion = Rectangle( Point( 1, 1 ), aItemContentSize );
- aBoundingRegion = Rectangle( Point( 0, 0 ), Size( aItemContentSize.Width() + 2, aItemContentSize.Height() + 2 ) );
- bNativeOK = true;
- }
- else if ( m_eRenderMode == NWF_TABBAR_ITEM )
- {
- Rectangle aRotatedContentArea( Point(), aItemContentSize );
- aRotatedContentArea.Transpose();
-
- TabitemValue tiValue;
- ImplControlValue aControlValue( (void*)(&tiValue) );
-
- bNativeOK = m_rTargetWindow.GetNativeControlRegion(
- CTRL_TAB_ITEM, PART_ENTIRE_CONTROL,
- Rectangle( aRotatedContentArea ),
- CTRL_STATE_ENABLED | CTRL_STATE_FOCUSED | CTRL_STATE_ROLLOVER | CTRL_STATE_SELECTED,
- aControlValue, ::rtl::OUString(),
- aBoundingRegion, aContentRegion
- );
- OSL_ENSURE( bNativeOK, "VerticalItemLayout::CalculateItemSize: GetNativeControlRegion not implemented for CTRL_TAB_ITEM?!" );
-
- Rectangle aBoundingRect( aBoundingRegion.GetBoundRect() );
- Rectangle aContentRect( aContentRegion.GetBoundRect() );
-
- impl_rotateFormation( aBoundingRect, aContentRect, m_bLeft );
-
- aBoundingRegion = aBoundingRect;
- aContentRegion = aContentRect;
- }
-
- if ( bNativeOK )
- {
- const Rectangle aBoundingRect( aBoundingRegion.GetBoundRect() );
-
- o_rContentArea = aContentRegion.GetBoundRect();
- o_rBoundingSize = aBoundingRect.GetSize();
-
- // normalize the content area, it is assumed to be relative to a rectangle which starts
- // at (0,0), and has a size of o_rBoundingSize.
- o_rContentArea.Move( -aBoundingRect.Left(), -aBoundingRect.Top() );
- }
- else
- {
- o_rContentArea = Rectangle( Point( 0, 0 ), aItemContentSize );
- o_rBoundingSize = aItemContentSize;
-
- // don't attempt native rendering, again. If it didn't work this time, it won't work in the future.
- const_cast< VerticalItemLayout* >( this )->m_eRenderMode = VCL_BASED;
- }
- }
-
- //------------------------------------------------------------------------------------------------------------------
- Point VerticalItemLayout::GetNextItemPosition( const Rectangle& i_rPreviousItemArea ) const
- {
- Point aNewPos( i_rPreviousItemArea.BottomLeft() );
- aNewPos.Y() += ITEM_DISTANCE_PIXEL;
- return aNewPos;
- }
-
- //------------------------------------------------------------------------------------------------------------------
void VerticalItemLayout::impl_preRender( const Rectangle& i_rBoundingArea, const Rectangle& i_rContentArea, const ItemFlags& i_nItemFlags )
{
ControlState nState = CTRL_STATE_ENABLED;
@@ -413,16 +225,15 @@ namespace svt
{
VirtualDevice aRenderDevice( m_rTargetWindow );
- Rectangle aRotatedBoundingArea( i_rBoundingArea );
- Rectangle aRotatedContentArea( i_rContentArea );
- impl_rotateFormation( aRotatedBoundingArea, aRotatedContentArea, !m_bLeft );
+ NormalizedArea aNormalized( i_rBoundingArea, true );
+ const Rectangle aNormalizedBounds( aNormalized.getReference() );
+ const Rectangle aNormalizedContent( aNormalized.getNormalized( i_rContentArea, m_bLeft ? TABS_LEFT : TABS_RIGHT ) );
- const Size aRotatedBoundingSize( aRotatedBoundingArea.GetSize() );
- aRenderDevice.SetOutputSizePixel( aRotatedBoundingSize );
+ const Size aNormalizedBoundingSize( aNormalizedBounds.GetSize() );
+ aRenderDevice.SetOutputSizePixel( aNormalizedBoundingSize );
- const Point aRotatedContentOffset( aRotatedContentArea.Left() - aRotatedBoundingArea.Left(), aRotatedContentArea.Top() - aRotatedBoundingArea.Top() );
- const Size aRotatedContentSize( aRotatedContentArea.GetSize() );
- const Region aCtrlRegion( Rectangle( aRotatedContentOffset, aRotatedContentSize ) );
+ const Point aNormalizedContentOffset( aNormalizedContent.Left() - aNormalizedBounds.Left(), aNormalizedContent.Top() - aNormalizedBounds.Top() );
+ const Region aCtrlRegion( Rectangle( aNormalizedContentOffset, aNormalizedContent.GetSize() ) );
TabitemValue tiValue;
if ( i_nItemFlags & ITEM_POSITION_FIRST )
@@ -437,8 +248,10 @@ namespace svt
OSL_ENSURE( bNativeOK, "VerticalItemLayout::impl_preRender: inconsistent NWF implementation!" );
// IsNativeControlSupported returned true, previously, otherwise we would not be here ...
- BitmapEx aBitmap( aRenderDevice.GetBitmapEx( Point( 0, 0 ), Size( aRotatedBoundingSize.Width() - 1, aRotatedBoundingSize.Height() - 1 ) ) );
- aBitmap.Rotate( m_bLeft ? 900 : 2700, Color( COL_BLACK ) );
+ BitmapEx aBitmap( aRenderDevice.GetBitmapEx( Point( 0, 0 ), Size( aNormalizedBoundingSize.Width() - 1, aNormalizedBoundingSize.Height() - 1 ) ) );
+ aBitmap.Rotate( 2700, Color( COL_BLACK ) );
+ if ( m_bLeft )
+ aBitmap.Mirror( BMP_MIRROR_HORZ );
m_rTargetWindow.DrawBitmapEx( i_rBoundingArea.TopLeft(), aBitmap );
}
@@ -563,16 +376,110 @@ namespace svt
}
//------------------------------------------------------------------------------------------------------------------
- void VerticalItemLayout::DrawItem( const PToolPanel& i_pPanel, const Point& i_rPosition,
+ void VerticalItemLayout::CalculateItemSize( const PToolPanel& i_pPanel,
+ const TabItemContent i_eItemContent, Size& o_rBoundingSize, Rectangle& o_rContentArea ) const
+ {
+ OSL_ENSURE( i_eItemContent != TABITEM_AUTO, "VerticalItemLayout::CalculateItemSize: illegal TabItemContent value!" );
+
+ const Image aImage( i_pPanel->GetImage() );
+ const bool bUseImage = !!aImage && ( i_eItemContent != TABITEM_TEXT_ONLY );
+
+ const ::rtl::OUString sItemText( i_pPanel->GetDisplayName() );
+ const bool bUseText = ( sItemText.getLength() != 0 ) && ( i_eItemContent != TABITEM_IMAGE_ONLY );
+
+ Size aItemContentSize;
+ if ( bUseImage )
+ {
+ aItemContentSize = aImage.GetSizePixel();
+ }
+
+ if ( bUseText )
+ {
+ if ( bUseImage )
+ aItemContentSize.Width() += ITEM_ICON_TEXT_DISTANCE;
+
+ // add space for text
+ const Size aTextSize( m_rTargetWindow.GetCtrlTextWidth( sItemText ), m_rTargetWindow.GetTextHeight() );
+ aItemContentSize.Width() += aTextSize.Width();
+ aItemContentSize.Height() = ::std::max( aItemContentSize.Height(), aTextSize.Height() );
+
+ aItemContentSize.Width() += 2 * ITEM_TEXT_FLOW_SPACE;
+ }
+
+ aItemContentSize.Width() += 2 * ITEM_OUTER_SPACE;
+ aItemContentSize.Height() += 2 * ITEM_OUTER_SPACE;
+
+ Region aBoundingRegion, aContentRegion;
+ bool bNativeOK = false;
+ if ( m_eRenderMode == NWF_TOOLBAR_ITEM )
+ {
+ // don't ask GetNativeControlRegion, this will not deliver proper results in all cases.
+ // Instead, simply assume that both the content and the bounding region are the same.
+// const ImplControlValue aControlValue;
+// bNativeOK = m_rTargetWindow.GetNativeControlRegion(
+// CTRL_TOOLBAR, PART_BUTTON,
+// Rectangle( Point(), aItemContentSize ), CTRL_STATE_ENABLED | CTRL_STATE_ROLLOVER,
+// aControlValue, ::rtl::OUString(),
+// aBoundingRegion, aContentRegion
+// );
+ aContentRegion = Rectangle( Point( 1, 1 ), aItemContentSize );
+ aBoundingRegion = Rectangle( Point( 0, 0 ), Size( aItemContentSize.Width() + 2, aItemContentSize.Height() + 2 ) );
+ bNativeOK = true;
+ }
+ else if ( m_eRenderMode == NWF_TABBAR_ITEM )
+ {
+ TabitemValue tiValue;
+ ImplControlValue aControlValue( (void*)(&tiValue) );
+
+ bNativeOK = m_rTargetWindow.GetNativeControlRegion(
+ CTRL_TAB_ITEM, PART_ENTIRE_CONTROL,
+ Rectangle( Point(), aItemContentSize ),
+ CTRL_STATE_ENABLED | CTRL_STATE_FOCUSED | CTRL_STATE_ROLLOVER | CTRL_STATE_SELECTED,
+ aControlValue, ::rtl::OUString(),
+ aBoundingRegion, aContentRegion
+ );
+ OSL_ENSURE( bNativeOK, "VerticalItemLayout::CalculateItemSize: GetNativeControlRegion not implemented for CTRL_TAB_ITEM?!" );
+ }
+
+ if ( bNativeOK )
+ {
+ const Rectangle aBoundingRect( aBoundingRegion.GetBoundRect() );
+
+ o_rContentArea = aContentRegion.GetBoundRect();
+ o_rBoundingSize = aBoundingRect.GetSize();
+
+ // normalize the content area, it is assumed to be relative to a rectangle which starts
+ // at (0,0), and has a size of o_rBoundingSize.
+ o_rContentArea.Move( -aBoundingRect.Left(), -aBoundingRect.Top() );
+ }
+ else
+ {
+ o_rContentArea = Rectangle( Point( 0, 0 ), aItemContentSize );
+ o_rBoundingSize = aItemContentSize;
+
+ // don't attempt native rendering, again. If it didn't work this time, it won't work in the future.
+ const_cast< VerticalItemLayout* >( this )->m_eRenderMode = VCL_BASED;
+ }
+ }
+ //------------------------------------------------------------------------------------------------------------------
+ void VerticalItemLayout::DrawItem( const TabBarGeometry& i_rGeometry, const PToolPanel& i_pPanel, const Point& i_rPosition,
const ItemFlags i_nItemFlags, const TabItemContent i_eItemContent )
{
+ // calculate content and bounding size
Rectangle aContentArea;
Size aBoundingSize;
CalculateItemSize( i_pPanel, i_eItemContent, aBoundingSize, aContentArea );
+ // CalculateItemSize returns a normalized size, i.e. one assuming TABS_TOP
+ const NormalizedArea aNormalized( Rectangle( Point(), aBoundingSize ), false );
+ aContentArea = aNormalized.getTransformed( aContentArea, i_rGeometry.getAlignment() );
+ aBoundingSize = Size( aBoundingSize.Height(), aBoundingSize.Width() );
+
+ // move both rects to the given position
aContentArea.Move( i_rPosition.X(), i_rPosition.Y() );
const Rectangle aBoundingArea( i_rPosition, aBoundingSize );
+ // actually draw
impl_preRender( aBoundingArea, aContentArea, i_nItemFlags );
impl_renderContent( i_pPanel, aContentArea, i_eItemContent );
impl_postRender( aBoundingArea, aContentArea, i_nItemFlags );
@@ -584,34 +491,7 @@ namespace svt
class PanelTabBar_Data : public IToolPanelDeckListener
{
public:
- PanelTabBar_Data( PanelTabBar& i_rTabBar, IToolPanelDeck& i_rPanelDeck, const TabAlignment i_eAlignment, const TabItemContent i_eItemContent )
- :rTabBar( i_rTabBar )
- ,rPanelDeck( i_rPanelDeck )
- ,eAlignment( i_eAlignment )
- ,eTabItemContent( i_eItemContent )
- ,pLayout( new VerticalItemLayout( i_rTabBar, i_eAlignment == TABS_LEFT ) )
- ,aHoveredItem()
- ,aFocusedItem()
- ,bMouseButtonDown( false )
- ,aItems()
- ,bItemsDirty( true )
- {
- OSL_ENSURE( ( i_eAlignment == TABS_LEFT ) || ( i_eAlignment == TABS_RIGHT ),
- "PanelTabBar_Data: unsupported alignment!" );
-
- rPanelDeck.AddListener( *this );
-
- if ( i_eAlignment == TABS_LEFT )
- {
- aTopLeftSpace = Size( 2, 2 );
- aBottomRightSpace = Size( 1, 1 );
- }
- else if ( i_eAlignment == TABS_RIGHT )
- {
- aTopLeftSpace = Size( 1, 1 );
- aBottomRightSpace = Size( 2, 2 );
- }
- }
+ PanelTabBar_Data( PanelTabBar& i_rTabBar, IToolPanelDeck& i_rPanelDeck, const TabAlignment i_eAlignment, const TabItemContent i_eItemContent );
~PanelTabBar_Data()
{
@@ -637,12 +517,20 @@ namespace svt
virtual void ActivePanelChanged( const ::boost::optional< size_t >& i_rOldActive, const ::boost::optional< size_t >& i_rNewActive );
virtual void Dying();
+ void UpdateScrollButtons()
+ {
+ m_aScrollBack.Enable( m_nScrollPosition > 0 );
+ m_aScrollForward.Enable( m_nScrollPosition < aItems.size() - 1 );
+ }
+
+ protected:
+ DECL_LINK( OnScroll, const PushButton* );
+
public:
PanelTabBar& rTabBar;
+ TabBarGeometry aGeometry;
IToolPanelDeck& rPanelDeck;
- const TabAlignment eAlignment;
- TabItemContent eTabItemContent;
PItemsLayout pLayout;
::boost::optional< size_t > aHoveredItem;
@@ -652,11 +540,66 @@ namespace svt
ItemDescriptors aItems;
bool bItemsDirty;
- Size aTopLeftSpace;
- Size aBottomRightSpace;
+ PushButton m_aScrollBack;
+ PushButton m_aScrollForward;
+
+ size_t m_nScrollPosition;
};
//==================================================================================================================
+ //= PanelTabBar_Data - implementation
+ //==================================================================================================================
+ //------------------------------------------------------------------------------------------------------------------
+ PanelTabBar_Data::PanelTabBar_Data( PanelTabBar& i_rTabBar, IToolPanelDeck& i_rPanelDeck, const TabAlignment i_eAlignment, const TabItemContent i_eItemContent )
+ :rTabBar( i_rTabBar )
+ ,aGeometry( i_eAlignment, i_eItemContent )
+ ,rPanelDeck( i_rPanelDeck )
+ ,pLayout( new VerticalItemLayout( i_rTabBar, i_eAlignment == TABS_LEFT ) )
+ ,aHoveredItem()
+ ,aFocusedItem()
+ ,bMouseButtonDown( false )
+ ,aItems()
+ ,bItemsDirty( true )
+ ,m_aScrollBack( &i_rTabBar, WB_BEVELBUTTON )
+ ,m_aScrollForward( &i_rTabBar, WB_BEVELBUTTON )
+ ,m_nScrollPosition( 0 )
+ {
+ OSL_ENSURE( ( i_eAlignment == TABS_LEFT ) || ( i_eAlignment == TABS_RIGHT ),
+ "PanelTabBar_Data: unsupported alignment!" );
+
+ rPanelDeck.AddListener( *this );
+
+ m_aScrollBack.SetSymbol( SYMBOL_ARROW_UP );
+ m_aScrollBack.Show();
+ m_aScrollBack.SetClickHdl( LINK( this, PanelTabBar_Data, OnScroll ) );
+
+ m_aScrollForward.SetSymbol( SYMBOL_ARROW_DOWN );
+ m_aScrollForward.Show();
+ m_aScrollForward.SetClickHdl( LINK( this, PanelTabBar_Data, OnScroll ) );
+ }
+
+ //------------------------------------------------------------------------------------------------------------------
+ IMPL_LINK( PanelTabBar_Data, OnScroll, const PushButton*, i_pButton )
+ {
+ if ( i_pButton == &m_aScrollBack )
+ {
+ OSL_ENSURE( m_nScrollPosition > 0, "PanelTabBar_Data::OnScroll: inconsistency!" );
+ --m_nScrollPosition;
+ rTabBar.Invalidate();
+ }
+ else if ( i_pButton == &m_aScrollForward )
+ {
+ OSL_ENSURE( m_nScrollPosition < aItems.size() - 1, "PanelTabBar_Data::OnScroll: inconsistency!" );
+ ++m_nScrollPosition;
+ rTabBar.Invalidate();
+ }
+
+ UpdateScrollButtons();
+
+ return 0L;
+ }
+
+ //==================================================================================================================
//= helper
//==================================================================================================================
namespace
@@ -695,7 +638,7 @@ namespace svt
{
io_rData.aItems.resize(0);
- Point aCompletePos( io_rData.aTopLeftSpace.Width(), io_rData.aTopLeftSpace.Height() );
+ Point aCompletePos( io_rData.aGeometry.getFirstItemPosition() );
Point aIconOnlyPos( aCompletePos );
Point aTextOnlyPos( aCompletePos );
@@ -713,12 +656,15 @@ namespace svt
Size aCompleteSize;
io_rData.pLayout->CalculateItemSize( pPanel, TABITEM_IMAGE_AND_TEXT, aCompleteSize, aContentArea );
+ ::std::swap( aCompleteSize.Width(), aCompleteSize.Height() );
Size aIconOnlySize;
io_rData.pLayout->CalculateItemSize( pPanel, TABITEM_IMAGE_ONLY, aIconOnlySize, aContentArea );
+ ::std::swap( aIconOnlySize.Width(), aIconOnlySize.Height() );
Size aTextOnlySize;
io_rData.pLayout->CalculateItemSize( pPanel, TABITEM_TEXT_ONLY, aTextOnlySize, aContentArea );
+ ::std::swap( aTextOnlySize.Width(), aTextOnlySize.Height() );
// TODO: have one method calculating all sizes?
@@ -728,9 +674,9 @@ namespace svt
io_rData.aItems.push_back( aItem );
- aCompletePos = io_rData.pLayout->GetNextItemPosition( aItem.aCompleteArea );
- aIconOnlyPos = io_rData.pLayout->GetNextItemPosition( aItem.aIconOnlyArea );
- aTextOnlyPos = io_rData.pLayout->GetNextItemPosition( aItem.aTextOnlyArea );
+ aCompletePos = aItem.aCompleteArea.BottomLeft();
+ aIconOnlyPos = aItem.aIconOnlyArea.BottomLeft();
+ aTextOnlyPos = aItem.aTextOnlyArea.BottomLeft();
}
io_rData.bItemsDirty = false;
@@ -750,16 +696,43 @@ namespace svt
}
//--------------------------------------------------------------------------------------------------------------
+ Rectangle lcl_getActualItemRect( const PanelTabBar_Data& i_rData, const Rectangle& i_rLogicalItemRect )
+ {
+ // care for the offset imposed by our geometry, i.e. whether or not we have scroll buttons
+ Rectangle aItemRect( i_rLogicalItemRect );
+
+ const TabBarGeometry& rGeometry( i_rData.aGeometry );
+ aItemRect.Move(
+ rGeometry.isVertical() ? 0 : rGeometry.getItemsRect().Left() - rGeometry.getButtonBackRect().Left(),
+ rGeometry.isVertical() ? rGeometry.getItemsRect().Top() - rGeometry.getButtonBackRect().Top() : 0
+ );
+
+ // care for the current scroll position
+ OSL_ENSURE( i_rData.m_nScrollPosition < i_rData.aItems.size(), "lcl_getActualItemRect: invalid scroll position!" );
+ if ( ( i_rData.m_nScrollPosition > 0 ) && ( i_rData.m_nScrollPosition < i_rData.aItems.size() ) )
+ {
+ long nOffsetX = i_rData.aItems[ i_rData.m_nScrollPosition ].GetCurrentRect().Left() - i_rData.aItems[ 0 ].GetCurrentRect().Left();
+ long nOffsetY = i_rData.aItems[ i_rData.m_nScrollPosition ].GetCurrentRect().Top() - i_rData.aItems[ 0 ].GetCurrentRect().Top();
+ aItemRect.Move( -nOffsetX, -nOffsetY );
+ }
+
+ return aItemRect;
+ }
+
+ //--------------------------------------------------------------------------------------------------------------
static ::boost::optional< size_t > lcl_findItemForPoint( const PanelTabBar_Data& i_rData, const Point& i_rPoint )
{
+ if ( !i_rData.aGeometry.getItemsRect().IsInside( i_rPoint ) )
+ return ::boost::optional< size_t >();
+
size_t i=0;
for ( ItemDescriptors::const_iterator item = i_rData.aItems.begin();
item != i_rData.aItems.end();
++item, ++i
)
{
- const Rectangle& rItemRect( item->GetCurrentRect() );
- if ( rItemRect.IsInside( i_rPoint ) )
+ Rectangle aItemRect( lcl_getActualItemRect( i_rData, item->GetCurrentRect() ) );
+ if ( aItemRect.IsInside( i_rPoint ) )
{
return ::boost::optional< size_t >( i );
}
@@ -768,6 +741,26 @@ namespace svt
}
//--------------------------------------------------------------------------------------------------------------
+ class ClipItemRegion
+ {
+ public:
+ ClipItemRegion( const PanelTabBar_Data& i_rData )
+ :m_rDevice( i_rData.rTabBar )
+ {
+ m_rDevice.Push( PUSH_CLIPREGION );
+ m_rDevice.SetClipRegion( i_rData.aGeometry.getItemsRect() );
+ }
+
+ ~ClipItemRegion()
+ {
+ m_rDevice.Pop();
+ }
+
+ private:
+ OutputDevice& m_rDevice;
+ };
+
+ //--------------------------------------------------------------------------------------------------------------
static void lcl_drawItem( const PanelTabBar_Data& i_rData, const size_t i_nItemIndex )
{
const ItemDescriptor& rItem( i_rData.aItems[ i_nItemIndex ] );
@@ -792,60 +785,28 @@ namespace svt
if ( i_rData.rPanelDeck.GetPanelCount() - 1 == i_nItemIndex )
nItemFlags |= ITEM_POSITION_LAST;
+ // the actual item pos might differ from the saved one, if we have scroll buttons
+ const Point aActualItemPos( lcl_getActualItemRect( i_rData, rItem.GetCurrentRect() ).TopLeft() );
+
i_rData.rTabBar.SetUpdateMode( FALSE );
- i_rData.pLayout->DrawItem( rItem.pPanel, rItem.GetCurrentRect().TopLeft(), nItemFlags, rItem.eContent );
+ i_rData.pLayout->DrawItem( i_rData.aGeometry, rItem.pPanel, aActualItemPos, nItemFlags, rItem.eContent );
i_rData.rTabBar.SetUpdateMode( TRUE );
}
//--------------------------------------------------------------------------------------------------------------
- static void lcl_fitItems( PanelTabBar_Data& io_rData )
+ static void lcl_relayout( PanelTabBar_Data& io_rData )
{
- if ( io_rData.aItems.empty() )
- // nothing to do
- return;
+ io_rData.aGeometry.relayout( io_rData.rTabBar.GetOutputSizePixel(), io_rData.aItems );
- TabItemContent eItemContent( io_rData.eTabItemContent );
- if ( io_rData.eTabItemContent == TABITEM_AUTO )
- {
- // the available size
- Size aOutputSize( io_rData.rTabBar.GetOutputSizePixel() );
- // shrunk by the outer space
- aOutputSize.Width() -= io_rData.aBottomRightSpace.Width();
- aOutputSize.Height() -= io_rData.aBottomRightSpace.Height();
- const Rectangle aFitInto( Point( 0, 0 ), aOutputSize );
-
- // the "content modes" to try
- TabItemContent eTryThis[] =
- {
- TABITEM_IMAGE_ONLY, // assumed to have the smallest rects
- TABITEM_TEXT_ONLY,
- TABITEM_IMAGE_AND_TEXT // assumed to have the largest rects
- };
+ const Rectangle& rButtonBack( io_rData.aGeometry.getButtonBackRect() );
+ io_rData.m_aScrollBack.SetPosSizePixel( rButtonBack.TopLeft(), rButtonBack.GetSize() );
+ io_rData.m_aScrollBack.Show( !rButtonBack.IsEmpty() );
+ const Rectangle& rButtonForward( io_rData.aGeometry.getButtonForwardRect() );
+ io_rData.m_aScrollForward.SetPosSizePixel( rButtonForward.TopLeft(), rButtonForward.GetSize() );
+ io_rData.m_aScrollForward.Show( !rButtonForward.IsEmpty() );
- // determine which of the different version fits
- eItemContent = eTryThis[0];
- size_t nTryIndex = 2;
- while ( nTryIndex > 0 )
- {
- const Point aBottomRight( io_rData.aItems.rbegin()->GetRect( eTryThis[ nTryIndex ] ).BottomRight() );
- if ( aFitInto.IsInside( aBottomRight ) )
- {
- eItemContent = eTryThis[ nTryIndex ];
- break;
- }
- --nTryIndex;
- }
- }
-
- // propagate to the items
- for ( ItemDescriptors::iterator item = io_rData.aItems.begin();
- item != io_rData.aItems.end();
- ++item
- )
- {
- item->eContent = eItemContent;
- }
+ io_rData.UpdateScrollButtons();
}
}
@@ -857,6 +818,7 @@ namespace svt
{
lcl_ensureItemsCache( *this );
+ ClipItemRegion aClipItems( *this );
if ( !!i_rOldActive )
lcl_drawItem( *this, *i_rOldActive );
if ( !!i_rNewActive )
@@ -866,7 +828,7 @@ namespace svt
//------------------------------------------------------------------------------------------------------------------
void PanelTabBar_Data::Dying()
{
- // TODO
+ // not interested in - the notifier is a member of this instance here, so we're dying ourself at the moment
}
//==================================================================================================================
@@ -891,14 +853,14 @@ namespace svt
//------------------------------------------------------------------------------------------------------------------
TabItemContent PanelTabBar::GetTabItemContent() const
{
- return m_pData->eTabItemContent;
+ return m_pData->aGeometry.getItemContent();
}
//------------------------------------------------------------------------------------------------------------------
void PanelTabBar::SetTabItemContent( const TabItemContent& i_eItemContent )
{
- m_pData->eTabItemContent = i_eItemContent;
- lcl_fitItems( *m_pData );
+ m_pData->aGeometry.setItemContent( i_eItemContent );
+ lcl_relayout( *m_pData );
Invalidate();
}
@@ -906,28 +868,14 @@ namespace svt
Size PanelTabBar::GetOptimalSize( WindowSizeType i_eType ) const
{
lcl_ensureItemsCache( *m_pData );
-
- if ( m_pData->aItems.empty() )
- return Size(
- m_pData->aTopLeftSpace.Width() + m_pData->aBottomRightSpace.Width(),
- m_pData->aTopLeftSpace.Height() + m_pData->aBottomRightSpace.Height()
- );
-
- const bool bMinimalSize = ( i_eType == WINDOWSIZE_MINIMUM );
- // the rect of the last item
- const Rectangle& rLastItemRect( bMinimalSize ? m_pData->aItems.rbegin()->aIconOnlyArea : m_pData->aItems.rbegin()->aCompleteArea );
- const Point aBottomRight( rLastItemRect.BottomRight() );
- return Size(
- aBottomRight.X() + 1 + m_pData->aBottomRightSpace.Width(),
- aBottomRight.Y() + 1 + m_pData->aBottomRightSpace.Height()
- );
+ return m_pData->aGeometry.getOptimalSize( m_pData->aItems, i_eType == WINDOWSIZE_MINIMUM );
}
//------------------------------------------------------------------------------------------------------------------
void PanelTabBar::Resize()
{
Control::Resize();
- lcl_fitItems( *m_pData );
+ lcl_relayout( *m_pData );
}
//------------------------------------------------------------------------------------------------------------------
@@ -938,6 +886,9 @@ namespace svt
// background
m_pData->pLayout->DrawBackground( Rectangle( Point(), GetOutputSizePixel() ) );
+ // ensure the items really paint into their own playground only
+ ClipItemRegion aClipItems( *m_pData );
+
// items
size_t i=0;
for ( ItemDescriptors::const_iterator item = m_pData->aItems.begin();
@@ -945,8 +896,8 @@ namespace svt
++item, ++i
)
{
- const Rectangle& rItemRect( item->GetCurrentRect() );
- if ( rItemRect.IsOver( i_rRect ) )
+ Rectangle aItemRect( lcl_getActualItemRect( *m_pData, item->GetCurrentRect() ) );
+ if ( aItemRect.IsOver( i_rRect ) )
{
lcl_drawItem( *m_pData, i );
}
@@ -968,15 +919,13 @@ namespace svt
{
m_pData->aHoveredItem = aNewItem;
+ ClipItemRegion aClipItems( *m_pData );
+
if ( !!aOldItem )
- {
lcl_drawItem( *m_pData, *aOldItem );
- }
if ( !!aNewItem )
- {
lcl_drawItem( *m_pData, *aNewItem );
- }
}
}
@@ -997,6 +946,7 @@ namespace svt
CaptureMouse();
m_pData->bMouseButtonDown = true;
+ ClipItemRegion aClipItems( *m_pData );
lcl_drawItem( *m_pData, *aHitItem );
}
@@ -1016,6 +966,7 @@ namespace svt
if ( !!aHitItem )
{
// re-draw that item now that we're not in mouse-down mode anymore
+ ClipItemRegion aClipItems( *m_pData );
lcl_drawItem( *m_pData, *aHitItem );
// activate the respective panel
m_pData->rPanelDeck.ActivatePanel( *aHitItem );
@@ -1054,6 +1005,7 @@ namespace svt
if ( !!aActivePanel )
{
m_pData->aFocusedItem = aActivePanel;
+ ClipItemRegion aClipItems( *m_pData );
lcl_drawItem( *m_pData, *m_pData->aFocusedItem );
}
}
@@ -1066,8 +1018,12 @@ namespace svt
::boost::optional< size_t > aPreviouslyFocused( m_pData->aFocusedItem );
m_pData->aFocusedItem.reset();
+
if ( !!aPreviouslyFocused )
+ {
+ ClipItemRegion aClipItems( *m_pData );
lcl_drawItem( *m_pData, *aPreviouslyFocused );
+ }
}
//------------------------------------------------------------------------------------------------------------------
@@ -1126,6 +1082,7 @@ namespace svt
m_pData->aFocusedItem.reset( ( *m_pData->aFocusedItem + nPanelCount - 1 ) % nPanelCount );
}
+ ClipItemRegion aClipItems( *m_pData );
lcl_drawItem( *m_pData, nOldFocus );
lcl_drawItem( *m_pData, *m_pData->aFocusedItem );
}
diff --git a/svtools/source/toolpanel/tabbargeometry.cxx b/svtools/source/toolpanel/tabbargeometry.cxx
new file mode 100644
index 000000000000..9d682b141da5
--- /dev/null
+++ b/svtools/source/toolpanel/tabbargeometry.cxx
@@ -0,0 +1,309 @@
+/*************************************************************************
+ * 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
+ * <http://www.openoffice.org/license.html>
+ * for a copy of the LGPLv3 License.
+ *
+************************************************************************/
+
+#include "precompiled_svtools.hxx"
+
+#include "tabbargeometry.hxx"
+
+#include <basegfx/range/b2drange.hxx>
+#include <basegfx/matrix/b2dhommatrix.hxx>
+#include <basegfx/numeric/ftools.hxx>
+
+#include <vcl/window.hxx>
+
+#include <algorithm>
+
+// the width (or height, depending on alignment) of the scroll buttons
+#define BUTTON_FLOW_WIDTH 20
+// the space between the scroll buttons and the items
+#define BUTTON_FLOW_SPACE 2
+// outer space to apply between the tab bar borders and any content. Note that those refer to a "normalized" geometry,
+// i.e. if the tab bar were aligned at the top
+#define OUTER_SPACE_LEFT 2
+#define OUTER_SPACE_TOP 4
+#define OUTER_SPACE_RIGHT 4
+#define OUTER_SPACE_BOTTOM 2
+
+// outer space to apply between the area for the items, and the actual items. They refer to aligned geometry at the moment,
+// they need to be adjusted once we support a horizontal tab bar
+#define ITEMS_SPACE_LEFT 0
+#define ITEMS_SPACE_TOP 2
+#define ITEMS_SPACE_RIGHT 2
+#define ITEMS_SPACE_BOTTOM 4
+
+//......................................................................................................................
+namespace svt
+{
+//......................................................................................................................
+
+ //==================================================================================================================
+ //= helper
+ //==================================================================================================================
+ namespace
+ {
+ //--------------------------------------------------------------------------------------------------------------
+ static void lcl_transform( Rectangle& io_rRect, const ::basegfx::B2DHomMatrix& i_rTransformation )
+ {
+ ::basegfx::B2DRange aRect( io_rRect.Left(), io_rRect.Top(), io_rRect.Right(), io_rRect.Bottom() );
+ aRect.transform( i_rTransformation );
+ io_rRect.Left() = long( aRect.getMinX() );
+ io_rRect.Top() = long( aRect.getMinY() );
+ io_rRect.Right() = long( aRect.getMaxX() );
+ io_rRect.Bottom() = long( aRect.getMaxY() );
+ }
+
+ //--------------------------------------------------------------------------------------------------------------
+ /** transforms the given, possible rotated playground,
+ */
+ void lcl_rotate( const Rectangle& i_rReference, Rectangle& io_rArea, const bool i_bRight )
+ {
+ // step 1: move the to-be-upper-left corner (left/bottom) of the rectangle to (0,0)
+ ::basegfx::B2DHomMatrix aTransformation;
+ aTransformation.translate(
+ i_bRight ? -i_rReference.Left() : -i_rReference.Right(),
+ i_bRight ? -i_rReference.Bottom() : -i_rReference.Top()
+ );
+
+ // step 2: rotate by -90 degrees
+ aTransformation.rotate( i_bRight ? +F_PI2 : -F_PI2 );
+ // note:
+ // on the screen, the ordinate goes top-down, while basegfx calculates in a system where the
+ // ordinate goes bottom-up; thus the "wrong" sign before F_PI2 here
+
+ // step 3: move back to original coordinates
+ aTransformation.translate( i_rReference.Left(), i_rReference.Top() );
+
+ // apply transformation
+ lcl_transform( io_rArea, aTransformation );
+ }
+ }
+
+ //------------------------------------------------------------------------------------------------------------------
+ void lcl_mirrorHorizontally( const Rectangle& i_rReferenceArea, Rectangle& io_rArea )
+ {
+ io_rArea.Left() = i_rReferenceArea.Left() + i_rReferenceArea.Right() - io_rArea.Left();
+ io_rArea.Right() = i_rReferenceArea.Left() + i_rReferenceArea.Right() - io_rArea.Right();
+ ::std::swap( io_rArea.Left(), io_rArea.Right() );
+ }
+
+ //==================================================================================================================
+ //= NormalizedArea
+ //==================================================================================================================
+ //------------------------------------------------------------------------------------------------------------------
+ NormalizedArea::NormalizedArea( const Rectangle& i_rReference, const bool i_bIsVertical )
+ :m_aReference( i_bIsVertical ? Rectangle( i_rReference.TopLeft(), Size( i_rReference.GetHeight(), i_rReference.GetWidth() ) ) : i_rReference )
+ {
+ }
+
+ //------------------------------------------------------------------------------------------------------------------
+ Rectangle NormalizedArea::getTransformed( const Rectangle& i_rArea, const TabAlignment i_eTargetAlignment ) const
+ {
+ OSL_ENSURE( ( i_eTargetAlignment == TABS_LEFT ) || ( i_eTargetAlignment == TABS_RIGHT ),
+ "NormalizedArea::getTransformed: unsupported alignment!" );
+
+ Rectangle aResult( i_rArea );
+
+ if ( ( i_eTargetAlignment == TABS_RIGHT )
+ || ( i_eTargetAlignment == TABS_LEFT )
+ )
+ {
+ lcl_rotate( m_aReference, aResult, true );
+
+ if ( i_eTargetAlignment == TABS_LEFT )
+ {
+ Rectangle aReference( m_aReference );
+ aReference.Transpose();
+ lcl_mirrorHorizontally( aReference, aResult );
+ }
+ }
+ return aResult;
+ }
+
+ //------------------------------------------------------------------------------------------------------------------
+ Rectangle NormalizedArea::getNormalized( const Rectangle& i_rArea, const TabAlignment i_eTargetAlignment ) const
+ {
+ OSL_ENSURE( ( i_eTargetAlignment == TABS_LEFT ) || ( i_eTargetAlignment == TABS_RIGHT ),
+ "NormalizedArea::gerNormalized: unsupported alignment!" );
+
+ Rectangle aResult( i_rArea );
+
+ if ( ( i_eTargetAlignment == TABS_RIGHT )
+ || ( i_eTargetAlignment == TABS_LEFT )
+ )
+ {
+ Rectangle aReference( m_aReference );
+ lcl_rotate( m_aReference, aReference, true );
+
+ if ( i_eTargetAlignment == TABS_LEFT )
+ {
+ lcl_mirrorHorizontally( aReference, aResult );
+ }
+
+ lcl_rotate( aReference, aResult, false );
+ }
+ return aResult;
+ }
+
+ //==================================================================================================================
+ //= TabBarGeometry
+ //==================================================================================================================
+ //------------------------------------------------------------------------------------------------------------------
+ TabBarGeometry::TabBarGeometry( const TabAlignment i_eAlignment, const TabItemContent i_eItemContent )
+ :m_eTabAlignment( i_eAlignment )
+ ,m_eTabItemContent( i_eItemContent )
+ ,m_aNormalizedPlayground( Rectangle(), false )
+ ,m_aButtonBackRect()
+ ,m_aItemsRect()
+ ,m_aButtonForwardRect()
+ {
+ }
+
+ //------------------------------------------------------------------------------------------------------------------
+ TabBarGeometry::~TabBarGeometry()
+ {
+ }
+
+ namespace
+ {
+ //--------------------------------------------------------------------------------------------------------------
+ static bool lcl_fitItems( const TabBarGeometry& i_rGeometry, ItemDescriptors& io_rItems )
+ {
+ if ( io_rItems.empty() )
+ // nothing to do, "no items" perfectly fit into any space we have ...
+ return true;
+
+ // the available size
+ Size aOutputSize( i_rGeometry.getItemsRect().GetSize() );
+ // shrunk by the outer space
+ aOutputSize.Width() -= ITEMS_SPACE_RIGHT;
+ aOutputSize.Height() -= ITEMS_SPACE_BOTTOM;
+ const Rectangle aFitInto( Point( 0, 0 ), aOutputSize );
+
+ TabItemContent eItemContent( i_rGeometry.getItemContent() );
+ if ( eItemContent == TABITEM_AUTO )
+ {
+ // the "content modes" to try
+ TabItemContent eTryThis[] =
+ {
+ TABITEM_IMAGE_ONLY, // assumed to have the smallest rects
+ TABITEM_TEXT_ONLY,
+ TABITEM_IMAGE_AND_TEXT // assumed to have the largest rects
+ };
+
+
+ // determine which of the different version fits
+ eItemContent = eTryThis[0];
+ size_t nTryIndex = 2;
+ while ( nTryIndex > 0 )
+ {
+ const Point aBottomRight( io_rItems.rbegin()->GetRect( eTryThis[ nTryIndex ] ).BottomRight() );
+ if ( aFitInto.IsInside( aBottomRight ) )
+ {
+ eItemContent = eTryThis[ nTryIndex ];
+ break;
+ }
+ --nTryIndex;
+ }
+ }
+
+ // propagate to the items
+ for ( ItemDescriptors::iterator item = io_rItems.begin();
+ item != io_rItems.end();
+ ++item
+ )
+ {
+ item->eContent = eItemContent;
+ }
+
+ const ItemDescriptor& rLastItem( *io_rItems.rbegin() );
+ const Point aLastItemBottomRight( rLastItem.GetCurrentRect().BottomRight() );
+ return aFitInto.IsInside( aLastItemBottomRight );
+ }
+ }
+
+ //------------------------------------------------------------------------------------------------------------------
+ Size TabBarGeometry::getOptimalSize( ItemDescriptors& io_rItems, const bool i_bMinimalSize ) const
+ {
+ if ( io_rItems.empty() )
+ return Size(
+ ITEMS_SPACE_LEFT + ITEMS_SPACE_RIGHT,
+ ITEMS_SPACE_TOP + ITEMS_SPACE_BOTTOM
+ );
+
+ // the rect of the last item
+ const Rectangle& rLastItemRect( i_bMinimalSize ? io_rItems.rbegin()->aIconOnlyArea : io_rItems.rbegin()->aCompleteArea );
+ const Point aBottomRight( rLastItemRect.BottomRight() );
+ return Size(
+ aBottomRight.X() + 1 + ITEMS_SPACE_RIGHT,
+ aBottomRight.Y() + 1 + ITEMS_SPACE_BOTTOM
+ );
+ }
+
+ //------------------------------------------------------------------------------------------------------------------
+ void TabBarGeometry::relayout( const Size& i_rActualOutputSize, ItemDescriptors& io_rItems )
+ {
+ m_aNormalizedPlayground = NormalizedArea( Rectangle( Point(), i_rActualOutputSize ), isVertical() );
+ const Size aNormalizedSize( m_aNormalizedPlayground.getReferenceSize() );
+
+ // assume all items fit
+ Point aButtonBackPos( OUTER_SPACE_LEFT, OUTER_SPACE_TOP );
+ m_aButtonBackRect = m_aNormalizedPlayground.getTransformed( Rectangle( aButtonBackPos, Size( 1, 1 ) ), getAlignment() );
+ m_aButtonBackRect.SetEmpty(); // do this after the transformation, it cannot cope with empty rects
+
+ Point aButtonForwardPos( aNormalizedSize.Width(), OUTER_SPACE_TOP );
+ m_aButtonForwardRect = m_aNormalizedPlayground.getTransformed( Rectangle( aButtonForwardPos, Size( 1, 1 ) ), getAlignment() );
+ m_aButtonBackRect.SetEmpty(); // do this after the transformation, it cannot cope with empty rects
+
+ Point aItemsPos( OUTER_SPACE_LEFT, 0 );
+ Size aItemsSize( aNormalizedSize.Width() - OUTER_SPACE_LEFT - OUTER_SPACE_RIGHT, aNormalizedSize.Height() );
+ m_aItemsRect = m_aNormalizedPlayground.getTransformed( Rectangle( aItemsPos, aItemsSize ), m_eTabAlignment );
+
+ if ( !lcl_fitItems( *this, io_rItems ) )
+ {
+ // assumption was wrong, the items do not fit => calculate rects for the scroll buttons
+ const Size aButtonSize( BUTTON_FLOW_WIDTH, aNormalizedSize.Height() - OUTER_SPACE_TOP - OUTER_SPACE_BOTTOM );
+
+ aButtonBackPos = Point( OUTER_SPACE_LEFT, OUTER_SPACE_TOP );
+ m_aButtonBackRect = m_aNormalizedPlayground.getTransformed( Rectangle( aButtonBackPos, aButtonSize ), m_eTabAlignment );
+
+ aButtonForwardPos = Point( aNormalizedSize.Width() - BUTTON_FLOW_WIDTH - OUTER_SPACE_RIGHT, OUTER_SPACE_TOP );
+ m_aButtonForwardRect = m_aNormalizedPlayground.getTransformed( Rectangle( aButtonForwardPos, aButtonSize ), m_eTabAlignment );
+
+ aItemsPos.X() = aButtonBackPos.X() + aButtonSize.Width() + BUTTON_FLOW_SPACE;
+ aItemsSize.Width() = aButtonForwardPos.X() - BUTTON_FLOW_SPACE - aItemsPos.X();
+ m_aItemsRect = m_aNormalizedPlayground.getTransformed( Rectangle( aItemsPos, aItemsSize ), m_eTabAlignment );
+ }
+ }
+
+ //------------------------------------------------------------------------------------------------------------------
+ Point TabBarGeometry::getFirstItemPosition() const
+ {
+ return Point( ITEMS_SPACE_LEFT, ITEMS_SPACE_TOP );
+ }
+
+//......................................................................................................................
+} // namespace svt
+//......................................................................................................................
diff --git a/svtools/source/toolpanel/tabbargeometry.hxx b/svtools/source/toolpanel/tabbargeometry.hxx
new file mode 100644
index 000000000000..81343b254b6d
--- /dev/null
+++ b/svtools/source/toolpanel/tabbargeometry.hxx
@@ -0,0 +1,141 @@
+/*************************************************************************
+ * 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
+ * <http://www.openoffice.org/license.html>
+ * for a copy of the LGPLv3 License.
+ *
+************************************************************************/
+
+#ifndef TABBARGEOMETRY_HXX
+#define TABBARGEOMETRY_HXX
+
+#include "svtools/toolpanel/tabalignment.hxx"
+
+#include "tabitemdescriptor.hxx"
+
+#include <tools/gen.hxx>
+
+//......................................................................................................................
+namespace svt
+{
+//......................................................................................................................
+
+ //==================================================================================================================
+ //= NormalizedArea
+ //==================================================================================================================
+ /** a rectangle which automatically translates between unrotated and rotated geometry.
+
+ It can be operated as if it were an unrotated area, but is able to provide corrdinates of rotated objects,
+ relative to its playground.
+ */
+ class NormalizedArea
+ {
+ public:
+ NormalizedArea( const Rectangle& i_rReference, const bool i_bIsVertical );
+
+ /** transforms a rectangle, relative to our playground, into a coordinate system defined by the given alignment
+ @param i_rArea
+ the area which is to be transformed.
+ */
+ Rectangle getTransformed(
+ const Rectangle& i_rArea,
+ const TabAlignment i_eTargetAlignment
+ ) const;
+
+ /** normalizes an already transformed rectangle
+ @param i_rArea
+ the area which is to be normalized.
+ */
+ Rectangle getNormalized(
+ const Rectangle& i_rArea,
+ const TabAlignment i_eTargetAlignment
+ ) const;
+
+
+ Size getReferenceSize() const { return m_aReference.GetSize(); }
+ const Rectangle&
+ getReference() const { return m_aReference; }
+
+ private:
+ // the normalized reference area
+ Rectangle m_aReference;
+ };
+
+ //==================================================================================================================
+ //= TabBarGeometry
+ //==================================================================================================================
+ class TabBarGeometry_Impl;
+ class TabBarGeometry
+ {
+ public:
+ TabBarGeometry( const TabAlignment i_eAlignment, const TabItemContent i_eItemContent );
+ ~TabBarGeometry();
+
+ // retrieves the rectangle to be occupied by the button for scrolling backward through the items
+ const Rectangle& getButtonBackRect() const { return m_aButtonBackRect; }
+ // retrieves the rectangle to be occupied by the items
+ const Rectangle& getItemsRect() const { return m_aItemsRect; }
+ // retrieves the rectangle to be occupied by the button for scrolling forward through the items
+ const Rectangle& getButtonForwardRect() const { return m_aButtonForwardRect; }
+
+ inline TabAlignment getAlignment() const { return m_eTabAlignment; }
+
+ inline TabItemContent
+ getItemContent() const { return m_eTabItemContent; }
+ inline void setItemContent( const TabItemContent i_eItemContent ) { m_eTabItemContent = i_eItemContent; }
+
+ inline bool isVertical() const
+ {
+ return ( m_eTabAlignment == TABS_LEFT )
+ || ( m_eTabAlignment == TABS_RIGHT );
+ }
+
+ inline Rectangle getTransformed( const Rectangle& i_rNormalizedRect ) const
+ {
+ return m_aNormalizedPlayground.getTransformed( i_rNormalizedRect, m_eTabAlignment );
+ }
+
+ /** adjusts the sizes of the buttons and the item's playground, plus the sizes of the items
+ */
+ void relayout( const Size& i_rActualOutputSize, ItemDescriptors& io_rItems );
+
+ /** calculates the optimal size of the tab bar, depending on the item's sizes
+ */
+ Size getOptimalSize( ItemDescriptors& io_rItems, const bool i_bMinimalSize ) const;
+
+ /** retrieves the position where the first item should start, relative to the item rect
+ */
+ Point getFirstItemPosition() const;
+
+ private:
+ const TabAlignment m_eTabAlignment;
+ TabItemContent m_eTabItemContent;
+ NormalizedArea m_aNormalizedPlayground;
+ Rectangle m_aButtonBackRect;
+ Rectangle m_aItemsRect;
+ Rectangle m_aButtonForwardRect;
+ };
+
+//......................................................................................................................
+} // namespace svt
+//......................................................................................................................
+
+#endif // TABBARGEOMETRY_HXX
diff --git a/svtools/source/toolpanel/tabitemdescriptor.hxx b/svtools/source/toolpanel/tabitemdescriptor.hxx
new file mode 100644
index 000000000000..3de280a7c92f
--- /dev/null
+++ b/svtools/source/toolpanel/tabitemdescriptor.hxx
@@ -0,0 +1,90 @@
+/*************************************************************************
+ * 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
+ * <http://www.openoffice.org/license.html>
+ * for a copy of the LGPLv3 License.
+ *
+************************************************************************/
+
+#ifndef TABITEMDESCRIPTOR_HXX
+#define TABITEMDESCRIPTOR_HXX
+
+#include "svtools/toolpanel/toolpanel.hxx"
+#include "svtools/toolpanel/tabitemcontent.hxx"
+
+#include <tools/gen.hxx>
+#include <osl/diagnose.h>
+
+#include <vector>
+
+//........................................................................
+namespace svt
+{
+//........................................................................
+
+ //==================================================================================================================
+ //= ItemDescriptor
+ //==================================================================================================================
+ struct ItemDescriptor
+ {
+ PToolPanel pPanel;
+ Rectangle aCompleteArea;
+ Rectangle aIconOnlyArea;
+ Rectangle aTextOnlyArea;
+ TabItemContent eContent;
+ // content to be used for this particular item. Might differ from item content which has been set
+ // up for the complete control, in case not the complete content fits into the available space.
+
+ ItemDescriptor()
+ :pPanel()
+ ,aCompleteArea()
+ ,aIconOnlyArea()
+ ,aTextOnlyArea()
+ ,eContent( TABITEM_IMAGE_AND_TEXT )
+ {
+ }
+
+ const Rectangle& GetRect( const TabItemContent i_eItemContent ) const
+ {
+ OSL_ENSURE( i_eItemContent != TABITEM_AUTO, "ItemDescriptor::GetRect: illegal value!" );
+
+ return ( i_eItemContent == TABITEM_IMAGE_AND_TEXT )
+ ? aCompleteArea
+ : ( ( i_eItemContent == TABITEM_TEXT_ONLY )
+ ? aTextOnlyArea
+ : aIconOnlyArea
+ );
+ }
+
+ const Rectangle& GetCurrentRect() const
+ {
+ return GetRect( eContent );
+ }
+ };
+
+ typedef ::std::vector< ItemDescriptor > ItemDescriptors;
+
+
+//........................................................................
+} // namespace svt
+//........................................................................
+
+#endif // TABITEMDESCRIPTOR_HXX
diff --git a/svtools/workben/toolpanel/toolpaneltest.cxx b/svtools/workben/toolpanel/toolpaneltest.cxx
index 60755a55b516..972d330141e5 100644
--- a/svtools/workben/toolpanel/toolpaneltest.cxx
+++ b/svtools/workben/toolpanel/toolpaneltest.cxx
@@ -597,6 +597,7 @@ PanelDemoMainWindow::PanelDemoMainWindow()
m_aToolPanelDeck.InsertPanel( PToolPanel( new ColoredPanel( m_aToolPanelDeck, Color( COL_RED ), "Red" ) ), m_aToolPanelDeck.GetPanelCount() );
m_aToolPanelDeck.InsertPanel( PToolPanel( new ColoredPanel( m_aToolPanelDeck, Color( COL_GREEN ), "Some flavor of Green" ) ), m_aToolPanelDeck.GetPanelCount() );
m_aToolPanelDeck.InsertPanel( PToolPanel( new ColoredPanel( m_aToolPanelDeck, RGB_COLORDATA( 255, 255, 0 ), "Yellow is ugly" ) ), m_aToolPanelDeck.GetPanelCount() );
+ m_aToolPanelDeck.InsertPanel( PToolPanel( new ColoredPanel( m_aToolPanelDeck, RGB_COLORDATA( 0, 0, 128 ), "Blue is the Color" ) ), m_aToolPanelDeck.GetPanelCount() );
m_aToolPanelDeck.ActivatePanel( 0 );
m_aToolPanelDeck.Show();