blob: 0dbddf221b1024b80c258a1c1aa18c260d5433ef [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 "precompiled_reportdesign.hxx"
#include "CondFormat.hxx"
#include "CondFormat.hrc"
#include "uistrings.hrc"
#include "RptResId.hrc"
#include "rptui_slotid.hrc"
#include "ModuleHelper.hxx"
#include "helpids.hrc"
#include "UITools.hxx"
#include "uistrings.hrc"
#include "ReportController.hxx"
#include "Condition.hxx"
/** === begin UNO includes === **/
#include <com/sun/star/beans/XPropertySet.hpp>
#include <com/sun/star/lang/IllegalArgumentException.hpp>
/** === end UNO includes === **/
#include <svx/globlmn.hrc>
#include <svx/svxids.hrc>
#include <toolkit/helper/vclunohelper.hxx>
#include <vcl/msgbox.hxx>
#include <tools/debug.hxx>
#include <tools/diagnose_ex.h>
#include <comphelper/property.hxx>
#include <algorithm>
#include "UndoActions.hxx"
// .............................................................................
namespace rptui
{
// .............................................................................
/** === begin UNO using === **/
using ::com::sun::star::uno::Reference;
using ::com::sun::star::uno::UNO_QUERY_THROW;
using ::com::sun::star::uno::UNO_QUERY;
using ::com::sun::star::uno::Exception;
using ::com::sun::star::lang::IllegalArgumentException;
using ::com::sun::star::uno::Sequence;
using ::com::sun::star::beans::PropertyValue;
using ::com::sun::star::uno::Any;
/** === end UNO using === **/
using namespace ::com::sun::star::report;
//========================================================================
// UpdateLocker
//========================================================================
class UpdateLocker
{
Window& m_rWindow;
public:
UpdateLocker( Window& _rWindow )
:m_rWindow( _rWindow )
{
_rWindow.SetUpdateMode( sal_False );
}
~UpdateLocker()
{
m_rWindow.SetUpdateMode( sal_True );
}
};
//========================================================================
// class ConditionalFormattingDialog
//========================================================================
DBG_NAME(rpt_ConditionalFormattingDialog)
ConditionalFormattingDialog::ConditionalFormattingDialog(
Window* _pParent, const Reference< XReportControlModel >& _rxFormatConditions, ::rptui::OReportController& _rController )
:ModalDialog( _pParent, ModuleRes(RID_CONDFORMAT) )
,m_aConditionPlayground( this, ModuleRes( WND_COND_PLAYGROUND ) )
,m_aSeparator(this, ModuleRes(FL_SEPARATOR1))
,m_aPB_OK(this, ModuleRes(PB_OK))
,m_aPB_CANCEL(this, ModuleRes(PB_CANCEL))
,m_aPB_Help(this, ModuleRes(PB_HELP))
,m_aCondScroll( this, ModuleRes( SB_ALL_CONDITIONS ) )
,m_rController( _rController )
,m_xFormatConditions( _rxFormatConditions )
,m_bDeletingCondition( false )
{
DBG_CTOR(rpt_ConditionalFormattingDialog,NULL);
OSL_ENSURE( m_xFormatConditions.is(), "ConditionalFormattingDialog::ConditionalFormattingDialog: ReportControlModel is NULL -> Prepare for GPF!" );
m_xCopy.set( m_xFormatConditions->createClone(), UNO_QUERY_THROW );
m_aCondScroll.SetScrollHdl( LINK( this, ConditionalFormattingDialog, OnScroll ) );
impl_initializeConditions();
FreeResource();
}
//------------------------------------------------------------------------
ConditionalFormattingDialog::~ConditionalFormattingDialog()
{
m_aConditions.clear();
DBG_DTOR(rpt_ConditionalFormattingDialog,NULL);
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_updateConditionIndicies()
{
sal_Int32 nIndex = 0;
for ( Conditions::const_iterator cond = m_aConditions.begin();
cond != m_aConditions.end();
++cond, ++nIndex
)
{
(*cond)->setConditionIndex( nIndex, impl_getConditionCount() );
}
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_conditionCountChanged()
{
if ( m_aConditions.empty() )
impl_addCondition_nothrow( 0 );
impl_updateScrollBarRange();
impl_updateConditionIndicies();
impl_layoutAll();
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::addCondition( size_t _nAddAfterIndex )
{
OSL_PRECOND( _nAddAfterIndex < impl_getConditionCount(), "ConditionalFormattingDialog::addCondition: illegal condition index!" );
impl_addCondition_nothrow( _nAddAfterIndex + 1 );
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::deleteCondition( size_t _nCondIndex )
{
impl_deleteCondition_nothrow( _nCondIndex );
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_addCondition_nothrow( size_t _nNewCondIndex )
{
UpdateLocker aLockUpdates( *this );
try
{
if ( _nNewCondIndex > (size_t)m_xCopy->getCount() )
throw IllegalArgumentException();
Reference< XFormatCondition > xCond = m_xCopy->createFormatCondition();
::comphelper::copyProperties(m_xCopy.get(),xCond.get());
m_xCopy->insertByIndex( _nNewCondIndex, makeAny( xCond ) );
ConditionPtr pCon( new Condition( &m_aConditionPlayground, *this, m_rController ) );
pCon->setCondition( xCond );
m_aConditions.insert( m_aConditions.begin() + _nNewCondIndex, pCon );
pCon->SetPosSizePixel( 0, 0, impl_getConditionWidth(), 0, WINDOW_POSSIZE_WIDTH );
}
catch( const Exception& )
{
DBG_UNHANDLED_EXCEPTION();
}
impl_conditionCountChanged();
impl_ensureConditionVisible( _nNewCondIndex );
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_focusCondition( size_t _nCondIndex )
{
OSL_PRECOND( _nCondIndex < impl_getConditionCount(),
"ConditionalFormattingDialog::impl_focusCondition: illegal index!" );
impl_ensureConditionVisible( _nCondIndex );
m_aConditions[ _nCondIndex ]->GrabFocus();
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_deleteCondition_nothrow( size_t _nCondIndex )
{
UpdateLocker aLockUpdates( *this );
OSL_PRECOND( _nCondIndex < impl_getConditionCount(),
"ConditionalFormattingDialog::impl_deleteCondition_nothrow: illegal index!" );
bool bLastCondition = ( impl_getConditionCount() == 1 );
bool bSetNewFocus = false;
size_t nNewFocusIndex( _nCondIndex );
try
{
if ( !bLastCondition )
m_xCopy->removeByIndex( _nCondIndex );
Conditions::iterator pos = m_aConditions.begin() + _nCondIndex;
if ( bLastCondition )
{
Reference< XFormatCondition > xFormatCondition( m_xCopy->getByIndex( 0 ), UNO_QUERY_THROW );
xFormatCondition->setFormula( ::rtl::OUString() );
(*pos)->setCondition( xFormatCondition );
}
else
{
bSetNewFocus = (*pos)->HasChildPathFocus();
m_bDeletingCondition = true;
m_aConditions.erase( pos );
m_bDeletingCondition = false;
}
if ( bSetNewFocus )
{
if ( nNewFocusIndex >= impl_getConditionCount() )
nNewFocusIndex = impl_getConditionCount() - 1;
}
}
catch( const Exception& )
{
DBG_UNHANDLED_EXCEPTION();
}
impl_conditionCountChanged();
if ( bSetNewFocus )
impl_focusCondition( nNewFocusIndex );
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_moveCondition_nothrow( size_t _nCondIndex, bool _bMoveUp )
{
size_t nOldConditionIndex( _nCondIndex );
size_t nNewConditionIndex( _bMoveUp ? _nCondIndex - 1 : _nCondIndex + 1 );
// do this in two steps, so we don't become inconsistent if any of the UNO actions fails
Any aMovedCondition;
ConditionPtr pMovedCondition;
try
{
aMovedCondition = m_xCopy->getByIndex( (sal_Int32)nOldConditionIndex );
m_xCopy->removeByIndex( (sal_Int32)nOldConditionIndex );
Conditions::iterator aRemovePos( m_aConditions.begin() + nOldConditionIndex );
pMovedCondition = *aRemovePos;
m_aConditions.erase( aRemovePos );
}
catch( const Exception& )
{
DBG_UNHANDLED_EXCEPTION();
return;
}
try
{
m_xCopy->insertByIndex( (sal_Int32)nNewConditionIndex, aMovedCondition );
m_aConditions.insert( m_aConditions.begin() + nNewConditionIndex, pMovedCondition );
}
catch( const Exception& )
{
DBG_UNHANDLED_EXCEPTION();
}
// at least the two swapped conditions need to know their new index
impl_updateConditionIndicies();
// re-layout all conditions
Point aDummy;
impl_layoutConditions( aDummy );
// ensure the moved condition is visible
impl_ensureConditionVisible( nNewConditionIndex );
}
// -----------------------------------------------------------------------------
long ConditionalFormattingDialog::impl_getConditionWidth() const
{
const Size aDialogSize( GetOutputSizePixel() );
const Size aScrollBarWidth( LogicToPixel( Size( SCROLLBAR_WIDTH + UNRELATED_CONTROLS, 0 ), MAP_APPFONT ) );
return aDialogSize.Width() - aScrollBarWidth.Width();
}
// -----------------------------------------------------------------------------
IMPL_LINK( ConditionalFormattingDialog, OnScroll, ScrollBar*, /*_pNotInterestedIn*/ )
{
size_t nFirstCondIndex( impl_getFirstVisibleConditionIndex() );
size_t nFocusCondIndex = impl_getFocusedConditionIndex( nFirstCondIndex );
Point aDummy;
impl_layoutConditions( aDummy );
if ( nFocusCondIndex < nFirstCondIndex )
impl_focusCondition( nFirstCondIndex );
else if ( nFocusCondIndex >= nFirstCondIndex + MAX_CONDITIONS )
impl_focusCondition( nFirstCondIndex + MAX_CONDITIONS - 1 );
return 0;
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_layoutConditions( Point& _out_rBelowLastVisible )
{
// position the condition's playground
long nConditionWidth = impl_getConditionWidth();
long nConditionHeight = LogicToPixel( Size( 0, CONDITION_HEIGHT ), MAP_APPFONT ).Height();
size_t nVisibleConditions = ::std::min( impl_getConditionCount(), MAX_CONDITIONS );
Size aPlaygroundSize( nConditionWidth, nVisibleConditions * nConditionHeight );
m_aConditionPlayground.SetSizePixel( aPlaygroundSize );
_out_rBelowLastVisible = Point( 0, aPlaygroundSize.Height() );
// position the single conditions
Point aConditionPos( 0, -1 * nConditionHeight * impl_getFirstVisibleConditionIndex() );
for ( Conditions::const_iterator cond = m_aConditions.begin();
cond != m_aConditions.end();
++cond
)
{
(*cond)->SetPosSizePixel( aConditionPos.X(), aConditionPos.Y(), nConditionWidth, nConditionHeight );
aConditionPos.Move( 0, nConditionHeight );
}
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_layoutAll()
{
// condition's positions
Point aPos;
impl_layoutConditions( aPos );
// scrollbar size and visibility
m_aCondScroll.SetPosSizePixel( 0, 0, 0, aPos.Y(), WINDOW_POSSIZE_HEIGHT );
if ( !impl_needScrollBar() )
// normalize the position, so it can, in all situations, be used as top index
m_aCondScroll.SetThumbPos( 0 );
// the separator and the buttons below it
aPos += LogicToPixel( Point( 0 , RELATED_CONTROLS ), MAP_APPFONT );
m_aSeparator.SetPosSizePixel( 0, aPos.Y(), 0, 0, WINDOW_POSSIZE_Y );
aPos += LogicToPixel( Point( 0 , UNRELATED_CONTROLS ), MAP_APPFONT );
Window* pWindows[] = { &m_aPB_OK, &m_aPB_CANCEL, &m_aPB_Help };
for ( size_t i= 0; i < sizeof(pWindows)/sizeof(pWindows[0]); ++i )
{
pWindows[i]->SetPosSizePixel( 0, aPos.Y(), 0, 0, WINDOW_POSSIZE_Y );
}
aPos += LogicToPixel( Point( 0, BUTTON_HEIGHT + RELATED_CONTROLS ), MAP_APPFONT );
SetPosSizePixel( 0, 0, 0, aPos.Y(), WINDOW_POSSIZE_HEIGHT );
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_initializeConditions()
{
try
{
sal_Int32 nCount = m_xCopy->getCount();
for ( sal_Int32 i = 0; i < nCount ; ++i )
{
ConditionPtr pCon( new Condition( &m_aConditionPlayground, *this, m_rController ) );
Reference< XFormatCondition > xCond( m_xCopy->getByIndex(i), UNO_QUERY );
pCon->setCondition( xCond );
pCon->updateToolbar( xCond.get() );
m_aConditions.push_back( pCon );
}
}
catch(Exception&)
{
OSL_ENSURE(0,"Can not access format condition!");
}
impl_conditionCountChanged();
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::applyCommand( size_t _nCondIndex, sal_uInt16 _nCommandId, const ::Color _aColor )
{
OSL_PRECOND( _nCommandId, "ConditionalFormattingDialog::applyCommand: illegal command id!" );
try
{
Reference< XReportControlFormat > xReportControlFormat( m_xCopy->getByIndex( _nCondIndex ), UNO_QUERY_THROW );
Sequence< PropertyValue > aArgs(3);
aArgs[0].Name = REPORTCONTROLFORMAT;
aArgs[0].Value <<= xReportControlFormat;
aArgs[1].Name = CURRENT_WINDOW;
aArgs[1].Value <<= VCLUnoHelper::GetInterface(this);
aArgs[2].Name = PROPERTY_FONTCOLOR;
aArgs[2].Value <<= (sal_uInt32)_aColor.GetColor();
// we use this way to create undo actions
m_rController.executeUnChecked(_nCommandId,aArgs);
m_aConditions[ _nCondIndex ]->updateToolbar(xReportControlFormat);
}
catch( Exception& )
{
DBG_UNHANDLED_EXCEPTION();
}
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::moveConditionUp( size_t _nCondIndex )
{
OSL_PRECOND( _nCondIndex > 0, "ConditionalFormattingDialog::moveConditionUp: cannot move up the first condition!" );
if ( _nCondIndex > 0 )
impl_moveCondition_nothrow( _nCondIndex, true );
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::moveConditionDown( size_t _nCondIndex )
{
OSL_PRECOND( _nCondIndex < impl_getConditionCount(), "ConditionalFormattingDialog::moveConditionDown: cannot move down the last condition!" );
if ( _nCondIndex < impl_getConditionCount() )
impl_moveCondition_nothrow( _nCondIndex, false );
}
// -----------------------------------------------------------------------------
::rtl::OUString ConditionalFormattingDialog::getDataField() const
{
::rtl::OUString sDataField;
try
{
sDataField = m_xFormatConditions->getDataField();
}
catch( const Exception& )
{
DBG_UNHANDLED_EXCEPTION();
}
return sDataField;
}
// -----------------------------------------------------------------------------
short ConditionalFormattingDialog::Execute()
{
short nRet = ModalDialog::Execute();
if ( nRet == RET_OK )
{
const String sUndoAction( ModuleRes( RID_STR_UNDO_CONDITIONAL_FORMATTING ) );
const UndoContext aUndoContext( m_rController.getUndoManager(), sUndoAction );
try
{
sal_Int32 j(0), i(0);;
for ( Conditions::const_iterator cond = m_aConditions.begin();
cond != m_aConditions.end();
++cond, ++i
)
{
Reference< XFormatCondition > xCond( m_xCopy->getByIndex(i), UNO_QUERY_THROW );
(*cond)->fillFormatCondition( xCond );
if ( (*cond)->isEmpty() )
continue;
Reference< XFormatCondition > xNewCond;
sal_Bool bAppend = j >= m_xFormatConditions->getCount();
if ( bAppend )
{
xNewCond = m_xFormatConditions->createFormatCondition();
m_xFormatConditions->insertByIndex( i, makeAny( xNewCond ) );
}
else
xNewCond.set( m_xFormatConditions->getByIndex(j), UNO_QUERY );
++j;
::comphelper::copyProperties(xCond.get(),xNewCond.get());
}
for ( sal_Int32 k = m_xFormatConditions->getCount()-1; k >= j; --k )
m_xFormatConditions->removeByIndex(k);
::comphelper::copyProperties( m_xCopy.get(), m_xFormatConditions.get() );
}
catch ( const Exception& )
{
DBG_UNHANDLED_EXCEPTION();
nRet = RET_NO;
}
}
return nRet;
}
// -----------------------------------------------------------------------------
long ConditionalFormattingDialog::PreNotify( NotifyEvent& _rNEvt )
{
switch ( _rNEvt.GetType() )
{
case EVENT_KEYINPUT:
{
const KeyEvent* pKeyEvent( _rNEvt.GetKeyEvent() );
const KeyCode& rKeyCode = pKeyEvent->GetKeyCode();
if ( rKeyCode.IsMod1() && rKeyCode.IsMod2() )
{
if ( rKeyCode.GetCode() == 0x0508 ) // -
{
impl_deleteCondition_nothrow( impl_getFocusedConditionIndex( 0 ) );
return 1;
}
if ( rKeyCode.GetCode() == 0x0507 ) // +
{
impl_addCondition_nothrow( impl_getFocusedConditionIndex( impl_getConditionCount() - 1 ) + 1 );
return 1;
}
}
}
break;
case EVENT_GETFOCUS:
{
if ( m_bDeletingCondition )
break;
const Window* pGetFocusWindow( _rNEvt.GetWindow() );
// determine whether the new focus window is part of an (currently invisible) condition
const Window* pConditionCandidate = pGetFocusWindow->GetParent();
const Window* pPlaygroundCandidate = pConditionCandidate ? pConditionCandidate->GetParent() : NULL;
while ( ( pPlaygroundCandidate )
&& ( pPlaygroundCandidate != this )
&& ( pPlaygroundCandidate != &m_aConditionPlayground )
)
{
pConditionCandidate = pConditionCandidate->GetParent();
pPlaygroundCandidate = pConditionCandidate ? pConditionCandidate->GetParent() : NULL;
}
if ( pPlaygroundCandidate == &m_aConditionPlayground )
{
impl_ensureConditionVisible( dynamic_cast< const Condition& >( *pConditionCandidate ).getConditionIndex() );
}
}
break;
}
return ModalDialog::PreNotify( _rNEvt );
}
// -----------------------------------------------------------------------------
size_t ConditionalFormattingDialog::impl_getFirstVisibleConditionIndex() const
{
return (size_t)m_aCondScroll.GetThumbPos();
}
// -----------------------------------------------------------------------------
size_t ConditionalFormattingDialog::impl_getLastVisibleConditionIndex() const
{
return ::std::min( impl_getFirstVisibleConditionIndex() + MAX_CONDITIONS, impl_getConditionCount() ) - 1;
}
// -----------------------------------------------------------------------------
size_t ConditionalFormattingDialog::impl_getFocusedConditionIndex( sal_Int32 _nFallBackIfNone ) const
{
size_t nIndex( 0 );
for ( Conditions::const_iterator cond = m_aConditions.begin();
cond != m_aConditions.end();
++cond, ++nIndex
)
{
if ( (*cond)->HasChildPathFocus() )
return nIndex;
}
return _nFallBackIfNone;
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_updateScrollBarRange()
{
long nMax = ( impl_getConditionCount() > MAX_CONDITIONS ) ? impl_getConditionCount() - MAX_CONDITIONS + 1 : 0;
m_aCondScroll.SetRangeMin( 0 );
m_aCondScroll.SetRangeMax( nMax );
m_aCondScroll.SetVisibleSize( 1 );
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_scrollTo( size_t _nTopCondIndex )
{
OSL_PRECOND( _nTopCondIndex + MAX_CONDITIONS <= impl_getConditionCount(),
"ConditionalFormattingDialog::impl_scrollTo: illegal index!" );
m_aCondScroll.SetThumbPos( _nTopCondIndex );
OnScroll( &m_aCondScroll );
}
// -----------------------------------------------------------------------------
void ConditionalFormattingDialog::impl_ensureConditionVisible( size_t _nCondIndex )
{
OSL_PRECOND( _nCondIndex < impl_getConditionCount(),
"ConditionalFormattingDialog::impl_ensureConditionVisible: illegal index!" );
if ( _nCondIndex < impl_getFirstVisibleConditionIndex() )
impl_scrollTo( _nCondIndex );
else if ( _nCondIndex > impl_getLastVisibleConditionIndex() )
impl_scrollTo( _nCondIndex - MAX_CONDITIONS + 1 );
}
// .............................................................................
} // rptui
// .............................................................................