diff options
Diffstat (limited to 'sd/source/ui/table/TableDesignPane.cxx')
-rwxr-xr-x | sd/source/ui/table/TableDesignPane.cxx | 951 |
1 files changed, 951 insertions, 0 deletions
diff --git a/sd/source/ui/table/TableDesignPane.cxx b/sd/source/ui/table/TableDesignPane.cxx new file mode 100755 index 000000000000..cdba12e9887e --- /dev/null +++ b/sd/source/ui/table/TableDesignPane.cxx @@ -0,0 +1,951 @@ +/************************************************************************* + * + * 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. + * + ************************************************************************/ + +// MARKER(update_precomp.py): autogen include statement, do not remove +#include "precompiled_sd.hxx" + +#include "sddll.hxx" + +#include <com/sun/star/beans/XMultiPropertyStates.hpp> +#include <com/sun/star/frame/XController.hpp> +#include <com/sun/star/view/XSelectionSupplier.hpp> +#include <com/sun/star/style/XStyle.hpp> +#include <com/sun/star/style/XStyleFamiliesSupplier.hpp> + +#include <comphelper/processfactory.hxx> +#include <sfx2/viewfrm.hxx> +#include <vcl/bmpacc.hxx> +#include <svl/style.hxx> +#include <sfx2/viewfrm.hxx> +#include <sfx2/bindings.hxx> +#include <sfx2/app.hxx> +#include <sfx2/request.hxx> +#include <sfx2/dispatch.hxx> +#include <svx/svxids.hrc> +#include <svx/svdetc.hxx> +#include <editeng/boxitem.hxx> +#include <editeng/borderline.hxx> +#include <editeng/colritem.hxx> +#include <editeng/eeitem.hxx> +#include <svx/sdr/table/tabledesign.hxx> + +#include "TableDesignPane.hxx" + +#include "DrawDocShell.hxx" +#include "ViewShellBase.hxx" +#include "DrawViewShell.hxx" +#include "DrawController.hxx" +#include "glob.hrc" +#include "sdresid.hxx" +#include "EventMultiplexer.hxx" + +#define C2U(x) OUString( RTL_CONSTASCII_USTRINGPARAM( x ) ) +using ::rtl::OUString; +using namespace ::com::sun::star; +using namespace ::com::sun::star::uno; +using namespace ::com::sun::star::drawing; +using namespace ::com::sun::star::container; +using namespace ::com::sun::star::beans; +using namespace ::com::sun::star::view; +using namespace ::com::sun::star::style; +using namespace ::com::sun::star::frame; +using namespace ::com::sun::star::lang; +using namespace ::com::sun::star::ui; + +namespace sd { + +static const sal_Int32 nPreviewColumns = 5; +static const sal_Int32 nPreviewRows = 5; + +// -------------------------------------------------------------------- + +static const OUString* getPropertyNames() +{ + static const OUString gPropNames[ CB_BANDED_COLUMNS-CB_HEADER_ROW+1 ] = + { + + C2U( "UseFirstRowStyle" ), + C2U( "UseLastRowStyle" ), + C2U( "UseBandingRowStyle" ), + C2U( "UseFirstColumnStyle" ), + C2U( "UseLastColumnStyle" ), + C2U( "UseBandingColumnStyle" ) + }; + return &gPropNames[0]; +} +// -------------------------------------------------------------------- + +TableDesignPane::TableDesignPane( ::Window* pParent, ViewShellBase& rBase, bool bModal ) +: Control( pParent, SdResId(DLG_TABLEDESIGNPANE) ) +, mrBase( rBase ) +, msTableTemplate( RTL_CONSTASCII_USTRINGPARAM( "TableTemplate" ) ) +, mbModal( bModal ) +, mbStyleSelected( false ) +, mbOptionsChanged( false ) +{ + Window* pControlParent = mbModal ? pParent : this; + + mxControls[FL_TABLE_STYLES].reset( new FixedLine( pControlParent, SdResId( FL_TABLE_STYLES + 1 ) ) ); + + ValueSet* pValueSet = new ValueSet( pControlParent, SdResId( CT_TABLE_STYLES+1 ) ); + mxControls[CT_TABLE_STYLES].reset( pValueSet ); + if( !mbModal ) + { + pValueSet->SetStyle( (pValueSet->GetStyle() & ~(WB_ITEMBORDER|WB_BORDER)) | WB_NO_DIRECTSELECT | WB_FLATVALUESET | WB_NOBORDER ); + pValueSet->SetColor(); + pValueSet->SetExtraSpacing(8); + } + else + { + pValueSet->SetColor( Color( COL_WHITE ) ); + pValueSet->SetBackground( Color( COL_WHITE ) ); + } + pValueSet->SetSelectHdl (LINK(this, TableDesignPane, implValueSetHdl)); + + mxControls[FL_STYLE_OPTIONS].reset( new FixedLine( pControlParent, SdResId( FL_STYLE_OPTIONS + 1 ) ) ); + USHORT i; + for( i = CB_HEADER_ROW; i <= CB_BANDED_COLUMNS; ++i ) + { + CheckBox *pCheckBox = new CheckBox( pControlParent, SdResId( i+1 ) ); + mxControls[i].reset( pCheckBox ); + pCheckBox->SetClickHdl( LINK( this, TableDesignPane, implCheckBoxHdl ) ); + } + + for( i = 0; i < DESIGNPANE_CONTROL_COUNT; i++ ) + mnOrgOffsetY[i] = mxControls[i]->GetPosPixel().Y(); + + // get current controller and initialize listeners + try + { + mxView = Reference< XDrawView >::query(mrBase.GetController()); + addListener(); + + Reference< XController > xController( mrBase.GetController(), UNO_QUERY_THROW ); + Reference< XStyleFamiliesSupplier > xFamiliesSupp( xController->getModel(), UNO_QUERY_THROW ); + Reference< XNameAccess > xFamilies( xFamiliesSupp->getStyleFamilies() ); + const OUString sFamilyName( RTL_CONSTASCII_USTRINGPARAM("table" ) ); + mxTableFamily = Reference< XIndexAccess >( xFamilies->getByName( sFamilyName ), UNO_QUERY_THROW ); + + } + catch( Exception& e ) + { + (void)e; + DBG_ERROR( "sd::CustomAnimationPane::CustomAnimationPane(), Exception caught!" ); + } + + onSelectionChanged(); + updateControls(); + + FreeResource(); +} + +// -------------------------------------------------------------------- + +TableDesignPane::~TableDesignPane() +{ + removeListener(); +} + +// -------------------------------------------------------------------- + +void TableDesignPane::DataChanged( const DataChangedEvent& /*rDCEvt*/ ) +{ + updateLayout(); +} + +// -------------------------------------------------------------------- + +void TableDesignPane::Resize() +{ + updateLayout(); +} + +// -------------------------------------------------------------------- + +static SfxBindings* getBindings( ViewShellBase& rBase ) +{ + if( rBase.GetMainViewShell().get() && rBase.GetMainViewShell()->GetViewFrame() ) + return &rBase.GetMainViewShell()->GetViewFrame()->GetBindings(); + else + return 0; +} + +// -------------------------------------------------------------------- + +static SfxDispatcher* getDispatcher( ViewShellBase& rBase ) +{ + if( rBase.GetMainViewShell().get() && rBase.GetMainViewShell()->GetViewFrame() ) + return rBase.GetMainViewShell()->GetViewFrame()->GetDispatcher(); + else + return 0; +} + +// -------------------------------------------------------------------- + +IMPL_LINK( TableDesignPane, implValueSetHdl, Control*, EMPTYARG ) +{ + mbStyleSelected = true; + if( !mbModal ) + ApplyStyle(); + return 0; +} + +// -------------------------------------------------------------------- + +void TableDesignPane::ApplyStyle() +{ + try + { + OUString sStyleName; + ValueSet* pValueSet = static_cast< ValueSet* >( mxControls[CT_TABLE_STYLES].get() ); + sal_Int32 nIndex = static_cast< sal_Int32 >( pValueSet->GetSelectItemId() ) - 1; + + if( (nIndex >= 0) && (nIndex < mxTableFamily->getCount()) ) + { + Reference< XNameAccess > xNames( mxTableFamily, UNO_QUERY_THROW ); + sStyleName = xNames->getElementNames()[nIndex]; + } + + if( sStyleName.getLength() == 0 ) + return; + + SdrView* pView = mrBase.GetDrawView(); + if( mxSelectedTable.is() ) + { + if( pView ) + { + SfxRequest aReq( SID_TABLE_STYLE, SFX_CALLMODE_SYNCHRON, SFX_APP()->GetPool() ); + aReq.AppendItem( SfxStringItem( SID_TABLE_STYLE, sStyleName ) ); + + rtl::Reference< sdr::SelectionController > xController( pView->getSelectionController() ); + if( xController.is() ) + xController->Execute( aReq ); + + SfxBindings* pBindings = getBindings( mrBase ); + if( pBindings ) + { + pBindings->Invalidate( SID_UNDO ); + pBindings->Invalidate( SID_REDO ); + } + } + } + else + { + SfxDispatcher* pDispatcher = getDispatcher( mrBase ); + SfxStringItem aArg( SID_TABLE_STYLE, sStyleName ); + pDispatcher->Execute(SID_INSERT_TABLE, SFX_CALLMODE_ASYNCHRON, &aArg, 0 ); + } + } + catch( Exception& ) + { + DBG_ERROR("TableDesignPane::implValueSetHdl(), exception caught!"); + } +} + +// -------------------------------------------------------------------- + +IMPL_LINK( TableDesignPane, implCheckBoxHdl, Control*, EMPTYARG ) +{ + mbOptionsChanged = true; + + if( !mbModal ) + ApplyOptions(); + + FillDesignPreviewControl(); + return 0; +} + +// -------------------------------------------------------------------- + +void TableDesignPane::ApplyOptions() +{ + static sal_uInt16 gParamIds[CB_BANDED_COLUMNS-CB_HEADER_ROW+1] = + { + ID_VAL_USEFIRSTROWSTYLE, ID_VAL_USELASTROWSTYLE, ID_VAL_USEBANDINGROWSTYLE, + ID_VAL_USEFIRSTCOLUMNSTYLE, ID_VAL_USELASTCOLUMNSTYLE, ID_VAL_USEBANDINGCOLUMNSTYLE + }; + + if( mxSelectedTable.is() ) + { + SfxRequest aReq( SID_TABLE_STYLE_SETTINGS, SFX_CALLMODE_SYNCHRON, SFX_APP()->GetPool() ); + + for( sal_uInt16 i = 0; i < (CB_BANDED_COLUMNS-CB_HEADER_ROW+1); ++i ) + { + aReq.AppendItem( SfxBoolItem( gParamIds[i], static_cast< CheckBox* >( mxControls[CB_HEADER_ROW+i].get() )->IsChecked() ) ); + } + + SdrView* pView = mrBase.GetDrawView(); + if( pView ) + { + rtl::Reference< sdr::SelectionController > xController( pView->getSelectionController() ); + if( xController.is() ) + { + xController->Execute( aReq ); + + SfxBindings* pBindings = getBindings( mrBase ); + if( pBindings ) + { + pBindings->Invalidate( SID_UNDO ); + pBindings->Invalidate( SID_REDO ); + } + } + } + } +} + +// -------------------------------------------------------------------- + +void TableDesignPane::onSelectionChanged() +{ + Reference< XPropertySet > xNewSelection; + + if( mxView.is() ) try + { + Reference< XSelectionSupplier > xSel( mxView, UNO_QUERY_THROW ); + if (xSel.is()) + { + Any aSel( xSel->getSelection() ); + Sequence< XShape > xShapeSeq; + if( aSel >>= xShapeSeq ) + { + if( xShapeSeq.getLength() == 1 ) + aSel <<= xShapeSeq[0]; + } + else + { + Reference< XShapes > xShapes( aSel, UNO_QUERY ); + if( xShapes.is() && (xShapes->getCount() == 1) ) + aSel <<= xShapes->getByIndex(0); + } + + Reference< XShapeDescriptor > xDesc( aSel, UNO_QUERY ); + if( xDesc.is() && + ( xDesc->getShapeType().equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "com.sun.star.drawing.TableShape" ) ) || + xDesc->getShapeType().equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "com.sun.star.presentation.TableShape" ) ) ) ) + { + xNewSelection = Reference< XPropertySet >::query( xDesc ); + } + } + } + catch( Exception& ) + { + DBG_ERROR( "sd::TableDesignPane::onSelectionChanged(), Exception caught!" ); + } + + if( mxSelectedTable != xNewSelection ) + { + mxSelectedTable = xNewSelection; + updateControls(); + } +} + +// -------------------------------------------------------------------- + +void TableDesignPane::updateLayout() +{ + ::Size aPaneSize( GetSizePixel() ); + if(IsVisible() && aPaneSize.Width() > 0) + { + Point aOffset( LogicToPixel( Point(3,3), MAP_APPFONT ) ); + + ValueSet* pValueSet = static_cast< ValueSet* >( mxControls[CT_TABLE_STYLES].get() ); + + Size aValueSetSize; + + if( !mbModal ) + { + const long nOptionsHeight = mnOrgOffsetY[CB_BANDED_COLUMNS] + mxControls[CB_BANDED_COLUMNS]->GetSizePixel().Height() + aOffset.Y(); + + const long nStylesHeight = aPaneSize.Height() - nOptionsHeight; + + // set with of controls to size of pane + for( sal_Int32 nId = 0; nId < DESIGNPANE_CONTROL_COUNT; ++nId ) + { + Size aSize( mxControls[nId]->GetSizePixel() ); + aSize.Width() = aPaneSize.Width() - aOffset.X() - mxControls[nId]->GetPosPixel().X(); + mxControls[nId]->SetSizePixel( aSize ); + mxControls[nId]->SetPaintTransparent(TRUE); + mxControls[nId]->SetBackground(); + } + aValueSetSize = Size( pValueSet->GetSizePixel().Width(), nStylesHeight - mxControls[FL_TABLE_STYLES]->GetSizePixel().Height() - mnOrgOffsetY[FL_TABLE_STYLES] ); + } + else + { + aValueSetSize = pValueSet->GetSizePixel(); + } + + + // Calculate the number of rows and columns. + if( pValueSet->GetItemCount() > 0 ) + { + Image aImage = pValueSet->GetItemImage(pValueSet->GetItemId(0)); + Size aItemSize = pValueSet->CalcItemSizePixel(aImage.GetSizePixel()); + pValueSet->SetItemWidth( aItemSize.Width() ); + pValueSet->SetItemHeight( aItemSize.Height() ); + + aItemSize.Width() += 10; + aItemSize.Height() += 10; + int nColumnCount = (aValueSetSize.Width() - pValueSet->GetScrollWidth()) / aItemSize.Width(); + if (nColumnCount < 1) + nColumnCount = 1; + + int nRowCount = (pValueSet->GetItemCount() + nColumnCount - 1) / nColumnCount; + if (nRowCount < 1) + nRowCount = 1; + + int nVisibleRowCount = (aValueSetSize.Height()+2) / aItemSize.Height(); + + pValueSet->SetLineCount ( (nRowCount < nVisibleRowCount) ? (USHORT)nRowCount : 0 ); + + pValueSet->SetColCount ((USHORT)nColumnCount); + pValueSet->SetLineCount ((USHORT)nRowCount); + + if( !mbModal ) + { + WinBits nStyle = pValueSet->GetStyle() & ~(WB_VSCROLL); + if( nRowCount < nVisibleRowCount ) + { + aValueSetSize.Height() = nRowCount * aItemSize.Height(); + } + else if( nRowCount > nVisibleRowCount ) + { + nStyle |= WB_VSCROLL; + } + pValueSet->SetStyle( nStyle ); + } + } + + if( !mbModal ) + { + pValueSet->SetSizePixel( aValueSetSize ); + pValueSet->SetBackground( GetSettings().GetStyleSettings().GetWindowColor() ); + pValueSet->SetColor( GetSettings().GetStyleSettings().GetWindowColor() ); + + Point aPos( pValueSet->GetPosPixel() ); + + // The following line may look like a no-op but without it the + // control is placed off-screen when RTL is active. + pValueSet->SetPosPixel(pValueSet->GetPosPixel()); + + // shift show options section down + const long nOptionsPos = aPos.Y() + aValueSetSize.Height(); + for( sal_Int32 nId = FL_STYLE_OPTIONS; nId <= CB_BANDED_COLUMNS; ++nId ) + { + Point aCPos( mxControls[nId]->GetPosPixel() ); + aCPos.X() = ( nId == FL_STYLE_OPTIONS ? 1 : 2 ) * aOffset.X(); + aCPos.Y() = mnOrgOffsetY[nId] + nOptionsPos; + mxControls[nId]->SetPosPixel( aCPos ); + } + } + } + + if( !mbModal ) + SetBackground( GetSettings().GetStyleSettings().GetWindowColor() ); +} + +// -------------------------------------------------------------------- + +void TableDesignPane::updateControls() +{ + static sal_Bool gDefaults[CB_BANDED_COLUMNS-CB_HEADER_ROW+1] = { sal_True, sal_False, sal_True, sal_False, sal_False, sal_False }; + + const bool bHasTable = mxSelectedTable.is(); + const OUString* pPropNames = getPropertyNames(); + + for( USHORT i = CB_HEADER_ROW; i <= CB_BANDED_COLUMNS; ++i ) + { + sal_Bool bUse = gDefaults[i-CB_HEADER_ROW]; + if( bHasTable ) try + { + mxSelectedTable->getPropertyValue( *pPropNames++ ) >>= bUse; + } + catch( Exception& ) + { + DBG_ERROR("sd::TableDesignPane::updateControls(), exception caught!"); + } + static_cast< CheckBox* >( mxControls[i].get() )->Check( bUse ? TRUE : FALSE ); + mxControls[i]->Enable(bHasTable ? TRUE : FALSE ); + } + + FillDesignPreviewControl(); + updateLayout(); + + + USHORT nSelection = 0; + if( mxSelectedTable.is() ) + { + Reference< XNamed > xNamed( mxSelectedTable->getPropertyValue( OUString( RTL_CONSTASCII_USTRINGPARAM( "TableTemplate" ) ) ), UNO_QUERY ); + if( xNamed.is() ) + { + const OUString sStyleName( xNamed->getName() ); + + Reference< XNameAccess > xNames( mxTableFamily, UNO_QUERY ); + if( xNames.is() ) + { + Sequence< OUString > aNames( xNames->getElementNames() ); + for( sal_Int32 nIndex = 0; nIndex < aNames.getLength(); nIndex++ ) + { + if( aNames[nIndex] == sStyleName ) + { + nSelection = (USHORT)nIndex+1; + break; + } + } + } + } + } + ValueSet* pValueSet = static_cast< ValueSet* >( mxControls[CT_TABLE_STYLES].get() ); + pValueSet->SelectItem( nSelection ); +} + +// -------------------------------------------------------------------- + +void TableDesignPane::addListener() +{ + Link aLink( LINK(this,TableDesignPane,EventMultiplexerListener) ); + mrBase.GetEventMultiplexer()->AddEventListener ( + aLink, + tools::EventMultiplexerEvent::EID_EDIT_VIEW_SELECTION + | tools::EventMultiplexerEvent::EID_CURRENT_PAGE + | tools::EventMultiplexerEvent::EID_MAIN_VIEW_REMOVED + | tools::EventMultiplexerEvent::EID_MAIN_VIEW_ADDED + | tools::EventMultiplexerEvent::EID_DISPOSING); +} + +// -------------------------------------------------------------------- + +void TableDesignPane::removeListener() +{ + Link aLink( LINK(this,TableDesignPane,EventMultiplexerListener) ); + mrBase.GetEventMultiplexer()->RemoveEventListener( aLink ); +} + +// -------------------------------------------------------------------- + +IMPL_LINK(TableDesignPane,EventMultiplexerListener, + tools::EventMultiplexerEvent*,pEvent) +{ + switch (pEvent->meEventId) + { + case tools::EventMultiplexerEvent::EID_CURRENT_PAGE: + case tools::EventMultiplexerEvent::EID_EDIT_VIEW_SELECTION: + onSelectionChanged(); + break; + + case tools::EventMultiplexerEvent::EID_MAIN_VIEW_REMOVED: + mxView = Reference<XDrawView>(); + onSelectionChanged(); + break; + + case tools::EventMultiplexerEvent::EID_MAIN_VIEW_ADDED: + mxView = Reference<XDrawView>::query( mrBase.GetController() ); + onSelectionChanged(); + break; + } + return 0; +} + +// -------------------------------------------------------------------- + +struct CellInfo +{ + Color maCellColor; + Color maTextColor; + SvxBoxItem maBorder; + + explicit CellInfo( const Reference< XStyle >& xStyle ); +}; + +CellInfo::CellInfo( const Reference< XStyle >& xStyle ) +: maBorder(SDRATTR_TABLE_BORDER) +{ + SfxStyleSheet* pStyleSheet = SfxUnoStyleSheet::getUnoStyleSheet( xStyle ); + if( pStyleSheet ) + { + SfxItemSet& rSet = pStyleSheet->GetItemSet(); + + // get style fill color + if( !GetDraftFillColor(rSet, maCellColor) ) + maCellColor.SetColor( COL_TRANSPARENT ); + + // get style text color + const SvxColorItem* pTextColor = dynamic_cast<const SvxColorItem*>( rSet.GetItem(EE_CHAR_COLOR) ); + if( pTextColor ) + maTextColor = pTextColor->GetValue(); + else + maTextColor.SetColor( COL_TRANSPARENT ); + + // get border + const SvxBoxItem* pBoxItem = dynamic_cast<const SvxBoxItem*>(rSet.GetItem( SDRATTR_TABLE_BORDER ) ); + if( pBoxItem ) + maBorder = *pBoxItem; + } +} + +// -------------------------------------------------------------------- + +typedef std::vector< boost::shared_ptr< CellInfo > > CellInfoVector; +typedef boost::shared_ptr< CellInfo > CellInfoMatrix[nPreviewColumns][nPreviewRows]; + +struct TableStyleSettings +{ + bool mbUseFirstRow; + bool mbUseLastRow; + bool mbUseFirstColumn; + bool mbUseLastColumn; + bool mbUseRowBanding; + bool mbUseColumnBanding; + + TableStyleSettings() + : mbUseFirstRow(true) + , mbUseLastRow(false) + , mbUseFirstColumn(false) + , mbUseLastColumn(false) + , mbUseRowBanding(true) + , mbUseColumnBanding(false) {} +}; + +// -------------------------------------------------------------------- + +static void FillCellInfoVector( const Reference< XIndexAccess >& xTableStyle, CellInfoVector& rVector ) +{ + DBG_ASSERT( xTableStyle.is() && (xTableStyle->getCount() == sdr::table::style_count ), "sd::FillCellInfoVector(), inavlid table style!" ); + if( xTableStyle.is() ) try + { + rVector.resize( sdr::table::style_count ); + + for( sal_Int32 nStyle = 0; nStyle < sdr::table::style_count; ++nStyle ) + { + Reference< XStyle > xStyle( xTableStyle->getByIndex( nStyle ), UNO_QUERY ); + if( xStyle.is() ) + rVector[nStyle].reset( new CellInfo( xStyle ) ); + } + } + catch(Exception&) + { + DBG_ERROR("sd::FillCellInfoVector(), exception caught!"); + } +} + +static void FillCellInfoMatrix( const CellInfoVector& rStyle, const TableStyleSettings& rSettings, CellInfoMatrix& rMatrix ) +{ + for( sal_Int32 nRow = 0; nRow < nPreviewColumns; ++nRow ) + { + const bool bFirstRow = rSettings.mbUseFirstRow && (nRow == 0); + const bool bLastRow = rSettings.mbUseLastRow && (nRow == nPreviewColumns - 1); + + for( sal_Int32 nCol = 0; nCol < nPreviewColumns; ++nCol ) + { + boost::shared_ptr< CellInfo > xCellInfo; + + // first and last row win first, if used and available + if( bFirstRow ) + { + xCellInfo = rStyle[sdr::table::first_row_style]; + } + else if( bLastRow ) + { + xCellInfo = rStyle[sdr::table::last_row_style]; + } + + if( !xCellInfo.get() ) + { + // next come first and last column, if used and available + if( rSettings.mbUseFirstColumn && (nCol == 0) ) + { + xCellInfo = rStyle[sdr::table::first_column_style]; + } + else if( rSettings.mbUseLastColumn && (nCol == nPreviewColumns-1) ) + { + xCellInfo = rStyle[sdr::table::last_column_style]; + } + } + + if( !xCellInfo.get() ) + { + if( rSettings.mbUseRowBanding ) + { + if( (nRow & 1) == 0 ) + { + xCellInfo = rStyle[sdr::table::even_rows_style]; + } + else + { + xCellInfo = rStyle[sdr::table::odd_rows_style]; + } + } + } + + if( !xCellInfo.get() ) + { + if( rSettings.mbUseColumnBanding ) + { + if( (nCol & 1) == 0 ) + { + xCellInfo = rStyle[sdr::table::even_columns_style]; + } + else + { + xCellInfo = rStyle[sdr::table::odd_columns_style]; + } + } + } + + if( !xCellInfo.get() ) + { + // use default cell style if non found yet + xCellInfo = rStyle[sdr::table::body_style]; + } + + rMatrix[nCol][nRow] = xCellInfo; + } + } +} + +// -------------------------------------------------------------------- + +const Bitmap CreateDesignPreview( const Reference< XIndexAccess >& xTableStyle, const TableStyleSettings& rSettings, bool bIsPageDark ) +{ + CellInfoVector aCellInfoVector(sdr::table::style_count); + FillCellInfoVector( xTableStyle, aCellInfoVector ); + + CellInfoMatrix aMatrix; + FillCellInfoMatrix( aCellInfoVector, rSettings, aMatrix ); + +// bbbbbbbbbbbb w = 12 pixel +// bccccccccccb h = 7 pixel +// bccccccccccb b = border color +// bcttttttttcb c = cell color +// bccccccccccb t = text color +// bccccccccccb +// bbbbbbbbbbbb + + + const sal_Int32 nCellWidth = 12; // one pixel is shared with the next cell! + const sal_Int32 nCellHeight = 7; // one pixel is shared with the next cell! + + Bitmap aPreviewBmp( Size( (nCellWidth * nPreviewColumns) - (nPreviewColumns - 1), (nCellHeight * nPreviewRows) - (nPreviewRows - 1)), 24, NULL ); + BitmapWriteAccess* pAccess = aPreviewBmp.AcquireWriteAccess(); + if( pAccess ) + { + pAccess->Erase( Color( bIsPageDark ? COL_BLACK : COL_WHITE ) ); + + // first draw cell background and text line previews + sal_Int32 nY = 0; + sal_Int32 nRow; + for( nRow = 0; nRow < nPreviewRows; ++nRow, nY += nCellHeight-1 ) + { + sal_Int32 nX = 0; + for( sal_Int32 nCol = 0; nCol < nPreviewColumns; ++nCol, nX += nCellWidth-1 ) + { + boost::shared_ptr< CellInfo > xCellInfo( aMatrix[nCol][nRow] ); + + Color aTextColor( COL_AUTO ); + if( xCellInfo.get() ) + { + // fill cell background + const Rectangle aRect( nX, nY, nX + nCellWidth - 1, nY + nCellHeight - 1 ); + + if( xCellInfo->maCellColor.GetColor() != COL_TRANSPARENT ) + { + pAccess->SetFillColor( xCellInfo->maCellColor ); + pAccess->FillRect( aRect ); + } + + aTextColor = xCellInfo->maTextColor; + } + + // draw text preview line + if( aTextColor.GetColor() == COL_AUTO ) + aTextColor.SetColor( bIsPageDark ? COL_WHITE : COL_BLACK ); + pAccess->SetLineColor( aTextColor ); + const Point aPnt1( nX + 2, nY + ((nCellHeight - 1 ) >> 1) ); + const Point aPnt2( nX + nCellWidth - 3, aPnt1.Y() ); + pAccess->DrawLine( aPnt1, aPnt2 ); + } + } + + // second draw border lines + nY = 0; + for( nRow = 0; nRow < nPreviewRows; ++nRow, nY += nCellHeight-1 ) + { + sal_Int32 nX = 0; + for( sal_Int32 nCol = 0; nCol < nPreviewColumns; ++nCol, nX += nCellWidth-1 ) + { + boost::shared_ptr< CellInfo > xCellInfo( aMatrix[nCol][nRow] ); + + if( xCellInfo.get() ) + { + const Point aPntTL( nX, nY ); + const Point aPntTR( nX + nCellWidth - 1, nY ); + const Point aPntBL( nX, nY + nCellHeight - 1 ); + const Point aPntBR( nX + nCellWidth - 1, nY + nCellHeight - 1 ); + + sal_Int32 border_diffs[8] = { 0,-1, 0,1, -1,0, 1,0 }; + sal_Int32* pDiff = &border_diffs[0]; + + // draw top border + for( USHORT nLine = 0; nLine < 4; ++nLine ) + { + const SvxBorderLine* pBorderLine = xCellInfo->maBorder.GetLine(nLine); + if( !pBorderLine || ((pBorderLine->GetOutWidth() == 0) && (pBorderLine->GetInWidth()==0)) ) + continue; + + sal_Int32 nBorderCol = nCol + *pDiff++; + sal_Int32 nBorderRow = nRow + *pDiff++; + if( (nBorderCol >= 0) && (nBorderCol < nPreviewColumns) && (nBorderRow >= 0) && (nBorderRow < nPreviewRows) ) + { + // check border + boost::shared_ptr< CellInfo > xBorderInfo( aMatrix[nBorderCol][nBorderRow] ); + if( xBorderInfo.get() ) + { + const USHORT nOtherLine = nLine ^ 1; + const SvxBorderLine* pBorderLine2 = xBorderInfo->maBorder.GetLine(nOtherLine^1); + if( pBorderLine2 && pBorderLine2->HasPriority(*pBorderLine) ) + continue; // other border line wins + } + } + + pAccess->SetLineColor( pBorderLine->GetColor() ); + switch( nLine ) + { + case 0: pAccess->DrawLine( aPntTL, aPntTR ); break; + case 1: pAccess->DrawLine( aPntBL, aPntBR ); break; + case 2: pAccess->DrawLine( aPntTL, aPntBL ); break; + case 3: pAccess->DrawLine( aPntTR, aPntBR ); break; + } + } + } + } + } + + aPreviewBmp.ReleaseAccess( pAccess ); + } + + return aPreviewBmp; +} + +void TableDesignPane::FillDesignPreviewControl() +{ + ValueSet* pValueSet = static_cast< ValueSet* >( mxControls[CT_TABLE_STYLES].get() ); + + USHORT nSelectedItem = pValueSet->GetSelectItemId(); + pValueSet->Clear(); + try + { + TableStyleSettings aSettings; + if( mxSelectedTable.is() ) + { + aSettings.mbUseFirstRow = static_cast< CheckBox* >(mxControls[CB_HEADER_ROW].get())->IsChecked(); + aSettings.mbUseLastRow = static_cast< CheckBox* >(mxControls[CB_TOTAL_ROW].get())->IsChecked(); + aSettings.mbUseRowBanding = static_cast< CheckBox* >(mxControls[CB_BANDED_ROWS].get())->IsChecked(); + aSettings.mbUseFirstColumn = static_cast< CheckBox* >(mxControls[CB_FIRST_COLUMN].get())->IsChecked(); + aSettings.mbUseLastColumn = static_cast< CheckBox* >(mxControls[CB_LAST_COLUMN].get())->IsChecked(); + aSettings.mbUseColumnBanding = static_cast< CheckBox* >(mxControls[CB_BANDED_COLUMNS].get())->IsChecked(); + } + + sal_Bool bIsPageDark = sal_False; + if( mxView.is() ) + { + Reference< XPropertySet > xPageSet( mxView->getCurrentPage(), UNO_QUERY ); + if( xPageSet.is() ) + { + const OUString sIsBackgroundDark( RTL_CONSTASCII_USTRINGPARAM( "IsBackgroundDark" ) ); + xPageSet->getPropertyValue(sIsBackgroundDark) >>= bIsPageDark; + } + } + + for( sal_Int32 nIndex = 0; nIndex < mxTableFamily->getCount(); nIndex++ ) try + { + Reference< XIndexAccess > xTableStyle( mxTableFamily->getByIndex( nIndex ), UNO_QUERY ); + if( xTableStyle.is() ) + pValueSet->InsertItem( sal::static_int_cast<USHORT>( nIndex + 1 ), Image( CreateDesignPreview( xTableStyle, aSettings, bIsPageDark ) ) ); + } + catch( Exception& ) + { + DBG_ERROR("sd::TableDesignPane::FillDesignPreviewControl(), exception caught!"); + } + } + catch( Exception& ) + { + DBG_ERROR("sd::TableDesignPane::FillDesignPreviewControl(), exception caught!"); + } + pValueSet->SelectItem(nSelectedItem); +} + +// ==================================================================== + +TableDesignDialog::TableDesignDialog(::Window* pParent, ViewShellBase& rBase ) +: ModalDialog( pParent, SdResId( DLG_TABLEDESIGNPANE )) +, mrBase( rBase ) +{ + mxFlSep1.reset( new FixedLine( this, SdResId( FL_SEP1 ) ) ); + mxFlSep2.reset( new FixedLine( this, SdResId( FL_SEP2 ) ) ); + mxHelpButton.reset( new HelpButton( this, SdResId( BTN_HELP ) ) ); + mxOkButton.reset( new OKButton( this, SdResId( BTN_OK ) ) ); + mxCancelButton.reset( new CancelButton( this, SdResId( BTN_CANCEL ) ) ); + FreeResource(); + + mpDesignPane.reset( new TableDesignPane( this, rBase, true ) ); + mpDesignPane->Hide(); +} + +// -------------------------------------------------------------------- + +short TableDesignDialog::Execute() +{ + if( ModalDialog::Execute() ) + { + if( mpDesignPane->isStyleChanged() ) + mpDesignPane->ApplyStyle(); + + if( mpDesignPane->isOptionsChanged() ) + mpDesignPane->ApplyOptions(); + return TRUE; + } + return FALSE; +} + +// ==================================================================== + +::Window * createTableDesignPanel( ::Window* pParent, ViewShellBase& rBase ) +{ + return new TableDesignPane( pParent, rBase, false ); +} + +// ==================================================================== + +void showTableDesignDialog( ::Window* pParent, ViewShellBase& rBase ) +{ + boost::scoped_ptr< TableDesignDialog > xDialog( new TableDesignDialog( pParent, rBase ) ); + xDialog->Execute(); +} + + +} + + |