summaryrefslogtreecommitdiff
path: root/svtools/source/control/breadcrumb.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'svtools/source/control/breadcrumb.cxx')
-rw-r--r--svtools/source/control/breadcrumb.cxx133
1 files changed, 108 insertions, 25 deletions
diff --git a/svtools/source/control/breadcrumb.cxx b/svtools/source/control/breadcrumb.cxx
index c7cbc0064a36..735756acc4eb 100644
--- a/svtools/source/control/breadcrumb.cxx
+++ b/svtools/source/control/breadcrumb.cxx
@@ -12,7 +12,7 @@
Breadcrumb::Breadcrumb( vcl::Window* pParent, WinBits nWinStyle ) : VclHBox( pParent, nWinStyle )
{
m_eMode = SvtBreadcrumbMode::ONLY_CURRENT_PATH;
- set_spacing( 6 );
+ set_spacing( SPACING );
appendField(); // root
}
@@ -51,22 +51,24 @@ void Breadcrumb::SetURL( const OUString& rURL )
{
INetURLObject aURL( rURL );
aURL.setFinalSlash();
+
OUString sPath = aURL.GetURLPath(INetURLObject::DECODE_WITH_CHARSET);
+ OUString sRootPath = INetURLObject::GetScheme( aURL.GetProtocol() ) + aURL.GetHost();
- unsigned int nSegments = aURL.getSegmentCount();
+ int nSegments = aURL.getSegmentCount();
unsigned int nPos = 0;
- unsigned int i;
bool bClear = ( m_eMode == SvtBreadcrumbMode::ONLY_CURRENT_PATH );
+ // root field
+
m_aLinks[0]->SetText( m_sRootName );
- m_aLinks[0]->Show();
m_aLinks[0]->Enable( true );
- m_aLinks[0]->SetURL( INetURLObject::GetScheme( aURL.GetProtocol() )
- + aURL.GetHost() );
- m_aSeparators[0]->Show();
+ m_aLinks[0]->SetURL( sRootPath );
+
+ // fill the other fields
- for( i = 1; i < nSegments + 1; i++ )
+ for( unsigned int i = 1; i < (unsigned int)nSegments + 1; i++ )
{
if( i >= m_aLinks.size() )
appendField();
@@ -80,30 +82,97 @@ void Breadcrumb::SetURL( const OUString& rURL )
bClear = true;
}
-
m_aLinks[i]->SetText( sLabel );
- m_aLinks[i]->SetURL( INetURLObject::GetScheme( aURL.GetProtocol() )
- + aURL.GetHost()
- + OUString( sPath.getStr(), nEnd ) );
- m_aLinks[i]->Show();
+ m_aLinks[i]->SetURL( sRootPath + OUString( sPath.getStr(), nEnd ) );
+ m_aLinks[i]->Hide();
m_aLinks[i]->Enable( true );
- m_aSeparators[i]->Show();
+ m_aSeparators[i]->Hide();
nPos = nEnd;
}
- m_aLinks[i - 1]->Enable( false );
- m_aSeparators[i - 1]->Hide();
+ // clear unused fields
- if( bClear )
+ for( unsigned int i = nSegments + 1; i < m_aLinks.size(); i++ )
{
- clearFields( i );
+ if( bClear )
+ m_aLinks[i]->SetText( "" );
+
+ m_aLinks[i]->Hide();
+ m_aSeparators[i]->Hide();
+ m_aLinks[i]->Enable( true );
}
- else
+
+ // show fields
+
+ Resize();
+ unsigned int nMaxWidth = GetSizePixel().Width();
+ unsigned int nSeparatorWidth = m_aSeparators[0]->GetSizePixel().Width();
+ unsigned int nCurrentWidth = 0;
+ unsigned int nLastVisible = nSegments;
+
+ bool bRight = ( m_eMode == SvtBreadcrumbMode::ALL_VISITED );
+ bool bLeft = true;
+
+ int i = 0;
+
+ while( bLeft || bRight )
{
- for( ; i < m_aLinks.size(); i++ )
- m_aLinks[i]->Enable( true );
+ if( nSegments - i == -1 )
+ bLeft = false;
+
+ if( bLeft )
+ {
+ unsigned int nIndex = nSegments - i;
+
+ if( showField( nIndex, nMaxWidth - nCurrentWidth ) )
+ {
+ nCurrentWidth += m_aLinks[nIndex]->GetSizePixel().Width()
+ + nSeparatorWidth + 2*SPACING;
+ }
+ else
+ {
+ // label is too long
+ if( nSegments != 0 )
+ {
+ m_aLinks[0]->SetText( "..." );
+ m_aLinks[0]->Enable( false );
+ }
+ bLeft = false;
+ }
+ }
+
+ if( nSegments + i == (int)m_aLinks.size() )
+ bRight = false;
+
+ if( i != 0 && bRight )
+ {
+ unsigned int nIndex = nSegments + i;
+
+ if( m_aLinks[nIndex]->GetText() == "" )
+ {
+ bRight = false;
+ }
+ else if( showField( nIndex, nMaxWidth - nCurrentWidth ) )
+ {
+ nCurrentWidth += m_aLinks[nIndex]->GetSizePixel().Width()
+ + nSeparatorWidth + 3*SPACING;
+ nLastVisible = nIndex;
+ }
+ else
+ {
+ bRight = false;
+ }
+ }
+
+ i++;
}
+
+ // current dir should be inactive
+ m_aLinks[nSegments]->Enable( false );
+
+ // hide last separator
+ m_aSeparators[nLastVisible]->Hide();
}
void Breadcrumb::SetMode( SvtBreadcrumbMode eMode )
@@ -122,13 +191,27 @@ void Breadcrumb::appendField()
m_aSeparators[m_aLinks.size() - 1]->Hide();
}
-void Breadcrumb::clearFields( unsigned int nStartIndex )
+bool Breadcrumb::showField( unsigned int nIndex, unsigned int nWidthMax )
{
- for( unsigned int i = nStartIndex; i < m_aLinks.size(); i++ )
+ m_aLinks[nIndex]->Show();
+ m_aSeparators[nIndex]->Show();
+
+ unsigned int nSeparatorWidth = m_aSeparators[0]->GetSizePixel().Width();
+ unsigned int nWidth = m_aLinks[nIndex]->GetSizePixel().Width()
+ + nSeparatorWidth + 3*SPACING;
+
+ if( nWidth > nWidthMax )
{
- m_aLinks[i]->Hide();
- m_aSeparators[i]->Hide();
+ if( nIndex != 0 )
+ {
+ m_aLinks[nIndex]->Hide();
+ m_aSeparators[nIndex]->Hide();
+ }
+
+ return false;
}
+
+ return true;
}
IMPL_LINK ( Breadcrumb, ClickLinkHdl, FixedHyperlink*, pLink )