Files
libreoffice/vcl/source/control/ctrl.cxx

489 lines
14 KiB
C++
Raw Normal View History

2000-09-18 16:07:07 +00:00
/*************************************************************************
*
* OpenOffice.org - a multi-platform office productivity suite
2000-09-18 16:07:07 +00:00
*
* $RCSfile: ctrl.cxx,v $
2000-09-18 16:07:07 +00:00
*
* $Revision: 1.24 $
2000-09-18 16:07:07 +00:00
*
* last change: $Author: ihi $ $Date: 2008-01-14 13:04:59 $
2000-09-18 16:07:07 +00:00
*
* The Contents of this file are made available subject to
* the terms of GNU Lesser General Public License Version 2.1.
2000-09-18 16:07:07 +00:00
*
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2005 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
2000-09-18 16:07:07 +00:00
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1, as published by the Free Software Foundation.
2000-09-18 16:07:07 +00:00
*
* This library 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 for more details.
2000-09-18 16:07:07 +00:00
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
2000-09-18 16:07:07 +00:00
*
************************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_vcl.hxx"
2000-09-18 16:07:07 +00:00
#ifndef _SV_RC_H
#include <tools/rc.h>
2000-09-18 16:07:07 +00:00
#endif
2001-08-27 08:45:10 +00:00
#ifndef _SV_SVDATA_HXX
#include <vcl/svdata.hxx>
2001-08-27 08:45:10 +00:00
#endif
2000-09-18 16:07:07 +00:00
#ifndef _SV_SVAPP_HXX
#include <vcl/svapp.hxx>
2000-09-18 16:07:07 +00:00
#endif
2001-07-06 15:12:47 +00:00
#ifndef _SV_EVENT_HXX
#include <vcl/event.hxx>
2001-07-06 15:12:47 +00:00
#endif
2000-09-18 16:07:07 +00:00
#ifndef _SV_CTRL_HXX
#include <vcl/ctrl.hxx>
2000-09-18 16:07:07 +00:00
#endif
#ifndef _SV_DECOVIEW_HXX
#include <vcl/decoview.hxx>
#endif
#ifndef _VCL_CONTROLLAYOUT_HXX
#include <vcl/controllayout.hxx>
#endif
#ifndef _SV_NATIVEWIDGETS_HXX
#include <vcl/salnativewidgets.hxx>
#endif
2000-09-18 16:07:07 +00:00
2000-09-18 16:07:07 +00:00
using namespace vcl;
2000-09-18 16:07:07 +00:00
// =======================================================================
void Control::ImplInitControlData()
2000-09-18 16:07:07 +00:00
{
mbHasFocus = FALSE;
mpLayoutData = NULL;
2000-09-18 16:07:07 +00:00
}
// -----------------------------------------------------------------------
Control::Control( WindowType nType ) :
Window( nType )
{
ImplInitControlData();
2000-09-18 16:07:07 +00:00
}
// -----------------------------------------------------------------------
Control::Control( Window* pParent, WinBits nStyle ) :
Window( WINDOW_CONTROL )
{
ImplInitControlData();
2000-09-18 16:07:07 +00:00
Window::ImplInit( pParent, nStyle, NULL );
}
// -----------------------------------------------------------------------
Control::Control( Window* pParent, const ResId& rResId ) :
Window( WINDOW_CONTROL )
{
ImplInitControlData();
2000-09-18 16:07:07 +00:00
rResId.SetRT( RSC_CONTROL );
WinBits nStyle = ImplInitRes( rResId );
ImplInit( pParent, nStyle, NULL );
ImplLoadRes( rResId );
if ( !(nStyle & WB_HIDE) )
Show();
}
// -----------------------------------------------------------------------
Control::~Control()
{
delete mpLayoutData, mpLayoutData = NULL;
}
// -----------------------------------------------------------------------
2000-09-18 16:07:07 +00:00
void Control::GetFocus()
{
Window::GetFocus();
}
// -----------------------------------------------------------------------
void Control::LoseFocus()
{
Window::LoseFocus();
}
// -----------------------------------------------------------------------
void Control::Resize()
{
delete mpLayoutData, mpLayoutData = NULL;
Window::Resize();
}
// -----------------------------------------------------------------------
void Control::FillLayoutData() const
{
}
// -----------------------------------------------------------------------
void Control::SetText( const String& rStr )
{
delete mpLayoutData;
mpLayoutData = NULL;
Window::SetText( rStr );
}
// -----------------------------------------------------------------------
Rectangle ControlLayoutData::GetCharacterBounds( long nIndex ) const
{
return (nIndex >= 0 && nIndex < (long) m_aUnicodeBoundRects.size()) ? m_aUnicodeBoundRects[ nIndex ] : Rectangle();
}
// -----------------------------------------------------------------------
Rectangle Control::GetCharacterBounds( long nIndex ) const
{
if( ! mpLayoutData )
FillLayoutData();
return mpLayoutData ? mpLayoutData->GetCharacterBounds( nIndex ) : Rectangle();
}
// -----------------------------------------------------------------------
long ControlLayoutData::GetIndexForPoint( const Point& rPoint ) const
{
long nIndex = -1;
for( long i = m_aUnicodeBoundRects.size()-1; i >= 0; i-- )
{
if( m_aUnicodeBoundRects[ i ].IsInside( rPoint ) )
{
nIndex = i;
break;
}
}
return nIndex;
}
// -----------------------------------------------------------------------
long Control::GetIndexForPoint( const Point& rPoint ) const
{
if( ! mpLayoutData )
FillLayoutData();
return mpLayoutData ? mpLayoutData->GetIndexForPoint( rPoint ) : -1;
}
// -----------------------------------------------------------------------
long ControlLayoutData::GetLineCount() const
{
long nLines = m_aLineIndices.size();
if( nLines == 0 && m_aDisplayText.Len() )
nLines = 1;
return nLines;
}
// -----------------------------------------------------------------------
long Control::GetLineCount() const
{
if( ! mpLayoutData )
FillLayoutData();
return mpLayoutData ? mpLayoutData->GetLineCount() : 0;
}
// -----------------------------------------------------------------------
Pair ControlLayoutData::GetLineStartEnd( long nLine ) const
{
Pair aPair( -1, -1 );
int nDisplayLines = m_aLineIndices.size();
if( nLine >= 0 && nLine < nDisplayLines )
{
aPair.A() = m_aLineIndices[nLine];
if( nLine+1 < nDisplayLines )
aPair.B() = m_aLineIndices[nLine+1]-1;
else
aPair.B() = m_aDisplayText.Len()-1;
}
else if( nLine == 0 && nDisplayLines == 0 && m_aDisplayText.Len() )
{
// special case for single line controls so the implementations
// in that case do not have to fill in the line indices
aPair.A() = 0;
aPair.B() = m_aDisplayText.Len()-1;
}
return aPair;
}
// -----------------------------------------------------------------------
Pair Control::GetLineStartEnd( long nLine ) const
{
if( ! mpLayoutData )
FillLayoutData();
return mpLayoutData ? mpLayoutData->GetLineStartEnd( nLine ) : Pair( -1, -1 );
}
// -----------------------------------------------------------------------
long ControlLayoutData::ToRelativeLineIndex( long nIndex ) const
{
// is the index sensible at all ?
if( nIndex >= 0 && nIndex < m_aDisplayText.Len() )
{
int nDisplayLines = m_aLineIndices.size();
// if only 1 line exists, then absolute and relative index are
// identical -> do nothing
if( nDisplayLines > 1 )
{
int nLine;
for( nLine = nDisplayLines-1; nLine >= 0; nLine-- )
{
if( m_aLineIndices[nLine] <= nIndex )
{
nIndex -= m_aLineIndices[nLine];
break;
}
}
if( nLine < 0 )
{
DBG_ASSERT( nLine >= 0, "ToRelativeLineIndex failed" );
nIndex = -1;
}
}
}
else
nIndex = -1;
return nIndex;
}
// -----------------------------------------------------------------------
long Control::ToRelativeLineIndex( long nIndex ) const
{
if( ! mpLayoutData )
FillLayoutData();
return mpLayoutData ? mpLayoutData->ToRelativeLineIndex( nIndex ) : -1;
}
// -----------------------------------------------------------------------
String Control::GetDisplayText() const
{
if( ! mpLayoutData )
FillLayoutData();
return mpLayoutData ? mpLayoutData->m_aDisplayText : GetText();
}
// -----------------------------------------------------------------------
2000-09-18 16:07:07 +00:00
long Control::Notify( NotifyEvent& rNEvt )
{
if ( rNEvt.GetType() == EVENT_GETFOCUS )
{
if ( !mbHasFocus )
{
mbHasFocus = TRUE;
if ( ImplCallEventListenersAndHandler( VCLEVENT_CONTROL_GETFOCUS, maGetFocusHdl, this ) )
// been destroyed within the handler
2001-11-27 08:54:45 +00:00
return TRUE;
2000-09-18 16:07:07 +00:00
}
}
else
{
if ( rNEvt.GetType() == EVENT_LOSEFOCUS )
{
Window* pFocusWin = Application::GetFocusWindow();
if ( !pFocusWin || !ImplIsWindowOrChild( pFocusWin ) )
{
mbHasFocus = FALSE;
if ( ImplCallEventListenersAndHandler( VCLEVENT_CONTROL_LOSEFOCUS, maLoseFocusHdl, this ) )
// been destroyed within the handler
2001-08-27 08:45:10 +00:00
return TRUE;
2000-09-18 16:07:07 +00:00
}
}
}
return Window::Notify( rNEvt );
}
// -----------------------------------------------------------------------
void Control::StateChanged( StateChangedType nStateChange )
{
if( nStateChange == STATE_CHANGE_INITSHOW ||
nStateChange == STATE_CHANGE_VISIBLE ||
nStateChange == STATE_CHANGE_FORMAT ||
nStateChange == STATE_CHANGE_ZOOM ||
nStateChange == STATE_CHANGE_BORDER ||
nStateChange == STATE_CHANGE_CONTROLFONT
)
{
delete mpLayoutData;
mpLayoutData = NULL;
}
Window::StateChanged( nStateChange );
}
// -----------------------------------------------------------------------
void Control::AppendLayoutData( const Control& rSubControl ) const
{
if( ! rSubControl.mpLayoutData )
rSubControl.FillLayoutData();
if( ! rSubControl.mpLayoutData || ! rSubControl.mpLayoutData->m_aDisplayText.Len() )
return;
long nCurrentIndex = mpLayoutData->m_aDisplayText.Len();
mpLayoutData->m_aDisplayText.Append( rSubControl.mpLayoutData->m_aDisplayText );
int nLines = rSubControl.mpLayoutData->m_aLineIndices.size();
int n;
mpLayoutData->m_aLineIndices.push_back( nCurrentIndex );
for( n = 1; n < nLines; n++ )
mpLayoutData->m_aLineIndices.push_back( rSubControl.mpLayoutData->m_aLineIndices[n] + nCurrentIndex );
int nRectangles = rSubControl.mpLayoutData->m_aUnicodeBoundRects.size();
Rectangle aRel = const_cast<Control&>(rSubControl).GetWindowExtentsRelative( const_cast<Control*>(this) );
for( n = 0; n < nRectangles; n++ )
{
Rectangle aRect = rSubControl.mpLayoutData->m_aUnicodeBoundRects[n];
aRect.Move( aRel.Left(), aRel.Top() );
mpLayoutData->m_aUnicodeBoundRects.push_back( aRect );
}
}
// -----------------------------------------------------------------
BOOL Control::ImplCallEventListenersAndHandler( ULONG nEvent, const Link& rHandler, void* pCaller )
{
ImplDelData aCheckDelete;
ImplAddDel( &aCheckDelete );
ImplCallEventListeners( nEvent );
if ( !aCheckDelete.IsDelete() )
{
rHandler.Call( pCaller );
if ( !aCheckDelete.IsDelete() )
{
ImplRemoveDel( &aCheckDelete );
return FALSE;
}
}
return TRUE;
}
// -----------------------------------------------------------------
void Control::SetLayoutDataParent( const Control* pParent ) const
{
if( mpLayoutData )
mpLayoutData->m_pParent = pParent;
}
// -----------------------------------------------------------------
void Control::ImplClearLayoutData() const
{
delete mpLayoutData, mpLayoutData = NULL;
}
// -----------------------------------------------------------------------
void Control::ImplDrawFrame( OutputDevice* pDev, Rectangle& rRect )
{
// use a deco view to draw the frame
// However, since there happens a lot of magic there, we need to fake some (style) settings
// on the device
AllSettings aOriginalSettings( pDev->GetSettings() );
AllSettings aNewSettings( aOriginalSettings );
StyleSettings aStyle( aNewSettings.GetStyleSettings() );
// The *only known* clients of the Draw methods of the various VCL-controls are form controls:
// During print preview, and during printing, Draw is called. Thus, drawing always happens with a
// mono (colored) border
aStyle.SetOptions( aStyle.GetOptions() | STYLE_OPTION_MONO );
aStyle.SetMonoColor( GetSettings().GetStyleSettings().GetMonoColor() );
aNewSettings.SetStyleSettings( aStyle );
// #i67023# do not call data changed listeners for this fake
// since they may understandably invalidate on settings changed
pDev->OutputDevice::SetSettings( aNewSettings );
DecorationView aDecoView( pDev );
rRect = aDecoView.DrawFrame( rRect, FRAME_DRAW_WINDOWBORDER );
pDev->OutputDevice::SetSettings( aOriginalSettings );
}
// -----------------------------------------------------------------------
void Control::DataChanged( const DataChangedEvent& rDCEvt)
{
// we don't want to loose some style settings for controls created with the
// toolkit
if ( IsCreatedWithToolkit() &&
(rDCEvt.GetType() == DATACHANGED_SETTINGS) &&
(rDCEvt.GetFlags() & SETTINGS_STYLE) )
{
AllSettings aSettings = GetSettings();
StyleSettings aStyleSettings = aSettings.GetStyleSettings();
ULONG nOldOptions = rDCEvt.GetOldSettings()->GetStyleSettings().GetOptions();
ULONG nNewOptions = aStyleSettings.GetOptions();
if ( !(nNewOptions & STYLE_OPTION_MONO) && ( nOldOptions & STYLE_OPTION_MONO ) )
{
nNewOptions |= STYLE_OPTION_MONO;
aStyleSettings.SetOptions( nNewOptions );
aStyleSettings.SetMonoColor( rDCEvt.GetOldSettings()->GetStyleSettings().GetMonoColor() );
aSettings.SetStyleSettings( aStyleSettings );
SetSettings( aSettings );
}
}
}
// -----------------------------------------------------------------
ControlLayoutData::~ControlLayoutData()
{
if( m_pParent )
m_pParent->ImplClearLayoutData();
}
// -----------------------------------------------------------------
Size Control::GetOptimalSize(WindowSizeType eType) const
{
switch (eType) {
case WINDOWSIZE_MINIMUM:
return Size( GetTextWidth( GetText() ) + 2 * 12,
GetTextHeight() + 2 * 6 );
case WINDOWSIZE_PREFERRED:
return GetOptimalSize( WINDOWSIZE_MINIMUM );
case WINDOWSIZE_MAXIMUM:
default:
return Size( LONG_MAX, LONG_MAX );
}
}