blob: c0e0e243c95a3b3ec2fc2202efd861b23a933ade [file] [log] [blame]
/**************************************************************
*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*************************************************************/
#include "vclxtabcontrol.hxx"
#include <com/sun/star/awt/PosSize.hpp>
#include <sal/macros.h>
#include <toolkit/helper/property.hxx>
#include <toolkit/helper/vclunohelper.hxx>
#include <vcl/tabctrl.hxx>
#include <vcl/tabpage.hxx>
#include "forward.hxx"
namespace layoutimpl
{
using namespace ::com::sun::star::lang;
using namespace ::com::sun::star::beans;
using namespace ::com::sun::star;
VCLXTabControl::ChildProps::ChildProps( VCLXTabControl::ChildData *pData )
{
addProp( RTL_CONSTASCII_USTRINGPARAM( "Title" ),
::getCppuType( static_cast< const rtl::OUString* >( NULL ) ),
&(pData->maTitle) );
}
VCLXTabControl::ChildData::ChildData( uno::Reference< awt::XLayoutConstrains > const& xChild )
: Box_Base::ChildData( xChild )
, maTitle()
{
}
VCLXTabControl::ChildData*
VCLXTabControl::createChild( uno::Reference< awt::XLayoutConstrains > const& xChild )
{
return new ChildData( xChild );
}
VCLXTabControl::ChildProps*
VCLXTabControl::createChildProps( Box_Base::ChildData *pData )
{
return new ChildProps( static_cast<VCLXTabControl::ChildData*> ( pData ) );
}
DBG_NAME( VCLXTabControl );
#if !defined (__GNUC__)
#define __PRETTY_FUNCTION__ __FUNCTION__
#endif /* !__GNUC__ */
VCLXTabControl::VCLXTabControl()
: VCLXWindow()
, VCLXTabControl_Base()
, Box_Base()
, mTabId (1)
, bRealized (false)
{
#ifndef __SUNPRO_CC
OSL_TRACE ("\n********%s:%x", __PRETTY_FUNCTION__, this);
#endif
DBG_CTOR( VCLXTabControl, NULL );
}
VCLXTabControl::~VCLXTabControl()
{
DBG_DTOR( VCLXTabControl, NULL );
}
IMPLEMENT_2_FORWARD_XINTERFACE2( VCLXTabControl, VCLXWindow, Container, VCLXTabControl_Base );
IMPLEMENT_FORWARD_XTYPEPROVIDER2( VCLXTabControl, VCLXWindow, VCLXTabControl_Base );
void SAL_CALL VCLXTabControl::dispose( ) throw(uno::RuntimeException)
{
{
::vos::OGuard aGuard( GetMutex() );
EventObject aDisposeEvent;
aDisposeEvent.Source = W3K_EXPLICIT_CAST (*this);
// maTabListeners.disposeAndClear( aDisposeEvent );
}
VCLXWindow::dispose();
}
#if 0
void SAL_CALL VCLXTabControl::addTabListener( const Reference< XTabListener >& listener ) throw (uno::RuntimeException)
{
if ( listener.is() )
maTabListeners.addInterface( listener );
}
void SAL_CALL VCLXTabControl::removeTabListener( const Reference< XTabListener >& listener ) throw (uno::RuntimeException)
{
if ( listener.is() )
maTabListeners.removeInterface( listener );
}
#endif
TabControl *VCLXTabControl::getTabControl() const throw (uno::RuntimeException)
{
TabControl *pTabControl = static_cast< TabControl* >( GetWindow() );
if ( pTabControl )
return pTabControl;
throw uno::RuntimeException();
}
sal_Int32 SAL_CALL VCLXTabControl::insertTab() throw (uno::RuntimeException)
{
TabControl *pTabControl = getTabControl();
sal_uInt16 id = sal::static_int_cast< sal_uInt16 >( mTabId++ );
rtl::OUString title (RTL_CONSTASCII_USTRINGPARAM( "" ) );
pTabControl->InsertPage( id, title.getStr(), TAB_APPEND );
pTabControl->SetTabPage( id, new TabPage( pTabControl ) );
return id;
}
void SAL_CALL VCLXTabControl::removeTab( sal_Int32 ID ) throw (uno::RuntimeException, IndexOutOfBoundsException)
{
TabControl *pTabControl = getTabControl();
if ( pTabControl->GetTabPage( sal::static_int_cast< sal_uInt16 >( ID ) ) == NULL )
throw IndexOutOfBoundsException();
pTabControl->RemovePage( sal::static_int_cast< sal_uInt16 >( ID ) );
}
void SAL_CALL VCLXTabControl::activateTab( sal_Int32 ID ) throw (uno::RuntimeException, IndexOutOfBoundsException)
{
TabControl *pTabControl = getTabControl();
if ( pTabControl->GetTabPage( sal::static_int_cast< sal_uInt16 >( ID ) ) == NULL )
throw IndexOutOfBoundsException();
pTabControl->SelectTabPage( sal::static_int_cast< sal_uInt16 >( ID ) );
}
sal_Int32 SAL_CALL VCLXTabControl::getActiveTabID() throw (uno::RuntimeException)
{
return getTabControl()->GetCurPageId( );
}
void SAL_CALL VCLXTabControl::addTabListener( const uno::Reference< awt::XTabListener >& xListener ) throw (uno::RuntimeException)
{
for ( std::list< uno::Reference
< awt::XTabListener > >::const_iterator it
= mxTabListeners.begin(); it != mxTabListeners.end(); it++ )
{
if ( *it == xListener )
// already added
return;
}
mxTabListeners.push_back( xListener );
}
void SAL_CALL VCLXTabControl::removeTabListener( const uno::Reference< awt::XTabListener >& xListener ) throw (uno::RuntimeException)
{
for ( std::list< uno::Reference
< awt::XTabListener > >::iterator it
= mxTabListeners.begin(); it != mxTabListeners.end(); it++ )
{
if ( *it == xListener )
{
mxTabListeners.erase( it );
break;
}
}
}
void SAL_CALL VCLXTabControl::setTabProps( sal_Int32 ID, const uno::Sequence< NamedValue >& Properties ) throw (uno::RuntimeException, IndexOutOfBoundsException)
{
TabControl *pTabControl = getTabControl();
if ( pTabControl->GetTabPage( sal::static_int_cast< sal_uInt16 >( ID ) ) == NULL )
throw IndexOutOfBoundsException();
for ( int i = 0; i < Properties.getLength(); i++ )
{
const rtl::OUString &name = Properties[i].Name;
const uno::Any &value = Properties[i].Value;
if ( name == rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Title" ) ) )
{
rtl::OUString title = value.get<rtl::OUString>();
pTabControl->SetPageText( sal::static_int_cast< sal_uInt16 >( ID ), title.getStr() );
}
}
}
uno::Sequence< NamedValue > SAL_CALL VCLXTabControl::getTabProps( sal_Int32 ID )
throw (IndexOutOfBoundsException, uno::RuntimeException)
{
TabControl *pTabControl = getTabControl();
if ( pTabControl->GetTabPage( sal::static_int_cast< sal_uInt16 >( ID ) ) == NULL )
throw IndexOutOfBoundsException();
#define ADD_PROP( seq, i, name, val ) { \
NamedValue value; \
value.Name = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( name ) ); \
value.Value = uno::makeAny( val ); \
seq[i] = value; \
}
uno::Sequence< NamedValue > props( 2 );
ADD_PROP( props, 0, "Title", rtl::OUString( pTabControl->GetPageText( sal::static_int_cast< sal_uInt16 >( ID ) ) ) );
ADD_PROP( props, 1, "Position", pTabControl->GetPagePos( sal::static_int_cast< sal_uInt16 >( ID ) ) );
#undef ADD_PROP
return props;
}
// TODO: draw tab border here
void SAL_CALL VCLXTabControl::draw( sal_Int32 nX, sal_Int32 nY ) throw(uno::RuntimeException)
{
::vos::OGuard aGuard( GetMutex() );
TabControl *pTabControl = getTabControl();
TabPage *pTabPage = pTabControl->GetTabPage( sal::static_int_cast< sal_uInt16 >( getActiveTabID() ) );
if ( pTabPage )
{
::Point aPos( nX, nY );
::Size aSize = pTabPage->GetSizePixel();
OutputDevice* pDev = VCLUnoHelper::GetOutputDevice( getGraphics() );
aPos = pDev->PixelToLogic( aPos );
aSize = pDev->PixelToLogic( aSize );
pTabPage->Draw( pDev, aPos, aSize, 0 );
}
VCLXWindow::draw( nX, nY );
}
void VCLXTabControl::AddChild (uno::Reference< awt::XLayoutConstrains > const& xChild)
{
#ifndef __SUNPRO_CC
OSL_TRACE ("%s: children: %d", __PRETTY_FUNCTION__, maChildren.size ());
#endif
mIdMap[ xChild ] = mTabId++;
Box_Base::AddChild( xChild );
#ifndef __SUNPRO_CC
OSL_TRACE ("%s: children: %d", __PRETTY_FUNCTION__, maChildren.size ());
#endif
}
void SAL_CALL VCLXTabControl::addChild(
const uno::Reference< awt::XLayoutConstrains > &xChild )
throw (uno::RuntimeException, awt::MaxChildrenException)
{
mIdMap[ xChild ] = insertTab();
Box_Base::addChild( xChild );
}
void SAL_CALL VCLXTabControl::removeChild( const uno::Reference< awt::XLayoutConstrains > &xChild )
throw (uno::RuntimeException)
{
removeTab( mIdMap[xChild] );
mIdMap[ xChild ] = -1;
Box_Base::removeChild( xChild );
}
static void setChildrenVisible( uno::Reference < awt::XLayoutConstrains > xChild, bool visible )
{
uno::Reference< awt::XWindow > xWin( xChild, uno::UNO_QUERY);
if ( xWin.is() )
{
xWin->setVisible( visible );
}
uno::Reference < awt::XLayoutContainer > xCont( xChild, uno::UNO_QUERY );
if ( xCont.is())
{
uno::Sequence< uno::Reference < awt::XLayoutConstrains > > children = xCont->getChildren();
for ( int i = 0; i < children.getLength(); i++ )
{
setChildrenVisible( children[i], visible );
}
}
}
void SAL_CALL VCLXTabControl::allocateArea (awt::Rectangle const &area)
throw (uno::RuntimeException)
{
#ifndef __SUNPRO_CC
OSL_TRACE ("\n%s", __PRETTY_FUNCTION__);
#endif
maAllocation = area;
TabControl *pTabControl = getTabControl();
// FIXME: this is wrong. We just want to set tab controls pos/size for
// the tabs menu, otherwise, it gets events that should go to children
// (I guess we could solve this by making the tabcontrol as the actual
// XWindow parent of its children, when importing...) Not sure about
// TabPage drawing... That doesn't work on gtk+; just ignoring that.
// LATER: Nah, the proper fix is to get the XWindow hierarchy
// straight.
#if 0
setPosSize( area.X, area.Y, area.Width, area.Height, awt::PosSize::POSSIZE );
#else
awt::Size currentSize = getSize();
awt::Size requestedSize (area.Width, area.Height);
// requestedSize.Height = getHeightForWidth( area.Width );
awt::Size minimumSize = getMinimumSize();
if (requestedSize.Width < minimumSize.Width)
requestedSize.Width = minimumSize.Width;
if (requestedSize.Height < minimumSize.Height)
requestedSize.Height = minimumSize.Height;
Size pageSize = static_cast<TabControl*> (GetWindow ())->GetTabPageSizePixel ();
awt::Size pageBasedSize (0, 0);
pageBasedSize.Width = pageSize.Width ();
pageBasedSize.Height = pageSize.Height ();
const int wc = 0;
const int hc = 20;
static int pwc = 0;
static int phc = 40;
if (requestedSize.Width < pageBasedSize.Width)
requestedSize.Width = pageBasedSize.Width + wc;
if (requestedSize.Height < pageBasedSize.Height)
requestedSize.Height = pageBasedSize.Height + hc;
Size windowSize = GetWindow()->GetSizePixel();
Window *parent = GetWindow()->GetParent();
Size parentSize = parent->GetSizePixel();
#ifndef __SUNPRO_CC
#ifdef GCC_MAJOR
OSL_TRACE ("\n%s", __PRETTY_FUNCTION__);
#endif /* GCC_MAJOR */
OSL_TRACE ("%s: cursize: %d ,%d", __FUNCTION__, currentSize.Width, currentSize.Height );
OSL_TRACE ("%s: area: %d, %d", __FUNCTION__, area.Width, area.Height );
OSL_TRACE ("%s: minimum: %d, %d", __FUNCTION__, minimumSize.Width, minimumSize.Height );
OSL_TRACE ("%s: requestedSize: %d, %d", __FUNCTION__, requestedSize.Width, requestedSize.Height );
OSL_TRACE ("%s: pageBasedSize: %d, %d", __FUNCTION__, pageBasedSize.Width, pageBasedSize.Height );
//OSL_TRACE ("%s: parent: %d, %d", __FUNCTION__, parentSize.Width(), parentSize.Height() );
//OSL_TRACE ("%s: window: %d, %d", __FUNCTION__, windowSize.Width(), windowSize.Height() );
#endif
//bRealized = false;
if (!bRealized)
{
setPosSize( area.X, area.Y, requestedSize.Width, requestedSize.Height, awt::PosSize::POSSIZE );
bRealized = true;
}
else
{
if ( requestedSize.Width > currentSize.Width + 10)
setPosSize( 0, 0, requestedSize.Width, 0, awt::PosSize::WIDTH );
if ( requestedSize.Height > currentSize.Height + 10)
setPosSize( 0, 0, 0, requestedSize.Height, awt::PosSize::HEIGHT );
}
#endif
if (pageBasedSize.Width > parentSize.Width ()
|| pageBasedSize.Height > parentSize.Height ())
//parent->SetSizePixel ( Size (pageBasedSize.Width, pageBasedSize.Height));
//parent->SetSizePixel ( Size (pageBasedSize.Width + pwc, pageBasedSize.Height + phc));
parent->SetSizePixel ( Size (requestedSize.Width + pwc, requestedSize.Height + phc));
// FIXME: we can save cycles by setting visibility more sensibly. Having
// it here does makes it easier when changing tabs (just needs a recalc())
unsigned i = 0;
for ( std::list<Box_Base::ChildData *>::const_iterator it
= maChildren.begin(); it != maChildren.end(); it++, i++ )
{
ChildData *child = static_cast<VCLXTabControl::ChildData*> ( *it );
uno::Reference
< awt::XLayoutConstrains > xChild( child->mxChild );
if ( xChild.is() )
{
uno::Reference< awt::XWindow > xWin( xChild, uno::UNO_QUERY );
bool active = (i+1 == (unsigned) getActiveTabID());
// HACK: since our layout:: container don't implement XWindow, we have no easy
// way to set them invisible; lets just set all their children as such :P
#if 0
if ( xWin.is() )
xWin->setVisible( active );
#else
setChildrenVisible( xChild, active );
#endif
if ( active )
{
::Rectangle label_rect = pTabControl->GetTabBounds( sal::static_int_cast< sal_uInt16 >( i+1 ) );
::Rectangle page_rect = pTabControl->GetTabPageBounds( sal::static_int_cast< sal_uInt16 >( i+1 ) );
awt::Rectangle childRect;
childRect.X = page_rect.Left();
childRect.Y = SAL_MAX( label_rect.Bottom(), page_rect.Top() );
childRect.Width = page_rect.Right() - page_rect.Left();
childRect.Height = page_rect.Bottom() - childRect.Y;
allocateChildAt( xChild, childRect );
}
}
}
}
awt::Size SAL_CALL VCLXTabControl::getMinimumSize()
throw(uno::RuntimeException)
{
awt::Size requestedSize = VCLXWindow::getMinimumSize();
awt::Size childrenSize( 0, 0 );
TabControl* pTabControl = static_cast< TabControl* >( GetWindow() );
if ( !pTabControl )
return requestedSize;
// calculate size to accomodate all children
unsigned i = 0;
for ( std::list<Box_Base::ChildData *>::const_iterator it
= maChildren.begin(); it != maChildren.end(); it++, i++ )
{
ChildData *child = static_cast<VCLXTabControl::ChildData*> ( *it );
if ( child->mxChild.is() )
{
// set the title prop here...
pTabControl->SetPageText( sal::static_int_cast< sal_uInt16 >( i+1 ), child->maTitle.getStr() );
awt::Size childSize( child->mxChild->getMinimumSize() );
childrenSize.Width = SAL_MAX( childSize.Width, childrenSize.Width );
childrenSize.Height = SAL_MAX( childSize.Height, childrenSize.Height );
}
}
#ifndef __SUNPRO_CC
#ifdef GCC_MAJOR
OSL_TRACE ("\n%s", __PRETTY_FUNCTION__);
#endif /* GCC_MAJOR */
OSL_TRACE ("%s: children: %d", __FUNCTION__, i);
OSL_TRACE ("%s: childrenSize: %d, %d", __FUNCTION__, childrenSize.Width, childrenSize.Height );
#endif
requestedSize.Width += childrenSize.Width;
requestedSize.Height += childrenSize.Height + 20;
maRequisition = requestedSize;
return requestedSize;
}
void VCLXTabControl::ProcessWindowEvent( const VclWindowEvent& _rVclWindowEvent )
{
::vos::OClearableGuard aGuard( GetMutex() );
TabControl* pTabControl = static_cast< TabControl* >( GetWindow() );
if ( !pTabControl )
return;
switch ( _rVclWindowEvent.GetId() )
{
case VCLEVENT_TABPAGE_ACTIVATE:
forceRecalc();
case VCLEVENT_TABPAGE_DEACTIVATE:
case VCLEVENT_TABPAGE_INSERTED:
case VCLEVENT_TABPAGE_REMOVED:
case VCLEVENT_TABPAGE_REMOVEDALL:
case VCLEVENT_TABPAGE_PAGETEXTCHANGED:
{
sal_uLong page = (sal_uLong) _rVclWindowEvent.GetData();
for ( std::list< uno::Reference
< awt::XTabListener > >::iterator it
= mxTabListeners.begin(); it != mxTabListeners.end(); it++)
{
uno::Reference
< awt::XTabListener > listener = *it;
switch ( _rVclWindowEvent.GetId() )
{
case VCLEVENT_TABPAGE_ACTIVATE:
listener->activated( page );
break;
case VCLEVENT_TABPAGE_DEACTIVATE:
listener->deactivated( page );
break;
case VCLEVENT_TABPAGE_INSERTED:
listener->inserted( page );
break;
case VCLEVENT_TABPAGE_REMOVED:
listener->removed( page );
break;
case VCLEVENT_TABPAGE_REMOVEDALL:
for ( int i = 1; i < mTabId; i++)
{
if ( pTabControl->GetTabPage( sal::static_int_cast< sal_uInt16 >( i ) ) )
listener->removed( i );
}
break;
case VCLEVENT_TABPAGE_PAGETEXTCHANGED:
listener->changed( page, getTabProps( page ) );
break;
}
}
break;
}
default:
aGuard.clear();
VCLXWindow::ProcessWindowEvent( _rVclWindowEvent );
break;
}
}
void SAL_CALL VCLXTabControl::setProperty( const ::rtl::OUString& PropertyName, const uno::Any &Value ) throw(uno::RuntimeException)
{
VCLXWindow::setProperty( PropertyName, Value );
}
uno::Any SAL_CALL VCLXTabControl::getProperty( const ::rtl::OUString& PropertyName ) throw(uno::RuntimeException)
{
return VCLXWindow::getProperty( PropertyName );
}
} // namespace layoutimpl