blob: 385cf52c3e3c2a1f5369000aa2fcabf024ad8a2b [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.
*
*************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_toolkit.hxx"
#include "defaultgriddatamodel.hxx"
#include <comphelper/stlunosequence.hxx>
#include <comphelper/componentguard.hxx>
#include <toolkit/helper/servicenames.hxx>
#include <tools/diagnose_ex.h>
#include <rtl/ref.hxx>
#include <algorithm>
#include <functional>
//......................................................................................................................
namespace toolkit
//......................................................................................................................
{
/** === begin UNO using === **/
using ::com::sun::star::uno::Reference;
using ::com::sun::star::uno::RuntimeException;
using ::com::sun::star::uno::Sequence;
using ::com::sun::star::uno::UNO_QUERY_THROW;
using ::com::sun::star::uno::UNO_QUERY;
using ::com::sun::star::uno::XInterface;
using ::com::sun::star::lang::XComponent;
using ::com::sun::star::lang::EventObject;
using ::com::sun::star::uno::Exception;
using ::com::sun::star::util::XCloneable;
/** === end UNO using === **/
using ::comphelper::stl_begin;
using ::comphelper::stl_end;
//==================================================================================================================
//= DefaultGridDataModel
//==================================================================================================================
//------------------------------------------------------------------------------------------------------------------
DefaultGridDataModel::DefaultGridDataModel()
:DefaultGridDataModel_Base( m_aMutex )
,m_aRowHeaders()
,m_nColumnCount(0)
{
}
//------------------------------------------------------------------------------------------------------------------
DefaultGridDataModel::DefaultGridDataModel( DefaultGridDataModel const & i_copySource )
:cppu::BaseMutex()
,DefaultGridDataModel_Base( m_aMutex )
,m_aData( i_copySource.m_aData )
,m_aRowHeaders( i_copySource.m_aRowHeaders )
,m_nColumnCount( i_copySource.m_nColumnCount )
{
}
//------------------------------------------------------------------------------------------------------------------
DefaultGridDataModel::~DefaultGridDataModel()
{
}
//------------------------------------------------------------------------------------------------------------------
void DefaultGridDataModel::broadcast( GridDataEvent const & i_event,
void ( SAL_CALL XGridDataListener::*i_listenerMethod )( GridDataEvent const & ), ::comphelper::ComponentGuard & i_instanceLock )
{
::cppu::OInterfaceContainerHelper* pListeners = rBHelper.getContainer( XGridDataListener::static_type() );
if ( !pListeners )
return;
i_instanceLock.clear();
pListeners->notifyEach( i_listenerMethod, i_event );
}
//------------------------------------------------------------------------------------------------------------------
::sal_Int32 SAL_CALL DefaultGridDataModel::getRowCount() throw (::com::sun::star::uno::RuntimeException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
return impl_getRowCount_nolck();
}
//------------------------------------------------------------------------------------------------------------------
::sal_Int32 SAL_CALL DefaultGridDataModel::getColumnCount() throw (::com::sun::star::uno::RuntimeException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
return m_nColumnCount;
}
//------------------------------------------------------------------------------------------------------------------
DefaultGridDataModel::CellData const & DefaultGridDataModel::impl_getCellData_throw( sal_Int32 const i_column, sal_Int32 const i_row ) const
{
if ( ( i_row < 0 ) || ( size_t( i_row ) > m_aData.size() )
|| ( i_column < 0 ) || ( i_column > m_nColumnCount )
)
throw IndexOutOfBoundsException( ::rtl::OUString(), *const_cast< DefaultGridDataModel* >( this ) );
RowData const & rRow( m_aData[ i_row ] );
if ( size_t( i_column ) < rRow.size() )
return rRow[ i_column ];
static CellData s_aEmpty;
return s_aEmpty;
}
//------------------------------------------------------------------------------------------------------------------
DefaultGridDataModel::RowData& DefaultGridDataModel::impl_getRowDataAccess_throw( sal_Int32 const i_rowIndex, size_t const i_requiredColumnCount )
{
OSL_ENSURE( i_requiredColumnCount <= size_t( m_nColumnCount ), "DefaultGridDataModel::impl_getRowDataAccess_throw: invalid column count!" );
if ( ( i_rowIndex < 0 ) || ( size_t( i_rowIndex ) >= m_aData.size() ) )
throw IndexOutOfBoundsException( ::rtl::OUString(), *this );
RowData& rRowData( m_aData[ i_rowIndex ] );
if ( rRowData.size() < i_requiredColumnCount )
rRowData.resize( i_requiredColumnCount );
return rRowData;
}
//------------------------------------------------------------------------------------------------------------------
DefaultGridDataModel::CellData& DefaultGridDataModel::impl_getCellDataAccess_throw( sal_Int32 const i_columnIndex, sal_Int32 const i_rowIndex )
{
if ( ( i_columnIndex < 0 ) || ( i_columnIndex >= m_nColumnCount ) )
throw IndexOutOfBoundsException( ::rtl::OUString(), *this );
RowData& rRowData( impl_getRowDataAccess_throw( i_rowIndex, size_t( i_columnIndex + 1 ) ) );
return rRowData[ i_columnIndex ];
}
//------------------------------------------------------------------------------------------------------------------
Any SAL_CALL DefaultGridDataModel::getCellData( ::sal_Int32 i_column, ::sal_Int32 i_row ) throw (RuntimeException, IndexOutOfBoundsException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
return impl_getCellData_throw( i_column, i_row ).first;
}
//------------------------------------------------------------------------------------------------------------------
Any SAL_CALL DefaultGridDataModel::getCellToolTip( ::sal_Int32 i_column, ::sal_Int32 i_row ) throw (RuntimeException, IndexOutOfBoundsException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
return impl_getCellData_throw( i_column, i_row ).second;
}
//------------------------------------------------------------------------------------------------------------------
Any SAL_CALL DefaultGridDataModel::getRowHeading( ::sal_Int32 i_row ) throw (RuntimeException, IndexOutOfBoundsException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
if ( ( i_row < 0 ) || ( size_t( i_row ) >= m_aRowHeaders.size() ) )
throw IndexOutOfBoundsException( ::rtl::OUString(), *this );
return m_aRowHeaders[ i_row ];
}
//------------------------------------------------------------------------------------------------------------------
Sequence< Any > SAL_CALL DefaultGridDataModel::getRowData( ::sal_Int32 i_rowIndex ) throw (IndexOutOfBoundsException, RuntimeException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
Sequence< Any > resultData( m_nColumnCount );
RowData& rRowData = impl_getRowDataAccess_throw( i_rowIndex, m_nColumnCount );
::std::transform( rRowData.begin(), rRowData.end(), resultData.getArray(), ::std::select1st< CellData >() );
return resultData;
}
//------------------------------------------------------------------------------------------------------------------
void DefaultGridDataModel::impl_insertRow( sal_Int32 const i_position, Any const & i_heading, Sequence< Any > const & i_rowData, sal_Int32 const i_assumedColCount )
{
OSL_PRECOND( ( i_assumedColCount <= 0 ) || ( i_assumedColCount >= i_rowData.getLength() ),
"DefaultGridDataModel::impl_insertRow: invalid column count!" );
// insert heading
m_aRowHeaders.insert( m_aRowHeaders.begin() + i_position, i_heading );
// create new data row
RowData newRow( i_assumedColCount > 0 ? i_assumedColCount : i_rowData.getLength() );
RowData::iterator cellData = newRow.begin();
for ( const Any* pData = stl_begin( i_rowData ); pData != stl_end( i_rowData ); ++pData, ++cellData )
cellData->first = *pData;
// insert data row
m_aData.insert( m_aData.begin() + i_position, newRow );
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::addRow( const Any& i_heading, const Sequence< Any >& i_data ) throw (RuntimeException)
{
insertRow( getRowCount(), i_heading, i_data );
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::addRows( const Sequence< Any >& i_headings, const Sequence< Sequence< Any > >& i_data ) throw (IllegalArgumentException, RuntimeException)
{
insertRows( getRowCount(), i_headings, i_data );
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::insertRow( ::sal_Int32 i_index, const Any& i_heading, const Sequence< Any >& i_data ) throw (RuntimeException, IndexOutOfBoundsException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
if ( ( i_index < 0 ) || ( i_index > impl_getRowCount_nolck() ) )
throw IndexOutOfBoundsException( ::rtl::OUString(), *this );
// actually insert the row
impl_insertRow( i_index, i_heading, i_data );
// update column count
sal_Int32 const columnCount = i_data.getLength();
if ( columnCount > m_nColumnCount )
m_nColumnCount = columnCount;
broadcast(
GridDataEvent( *this, -1, -1, i_index, i_index ),
&XGridDataListener::rowsInserted,
aGuard
);
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::insertRows( ::sal_Int32 i_index, const Sequence< Any>& i_headings, const Sequence< Sequence< Any > >& i_data ) throw (IllegalArgumentException, IndexOutOfBoundsException, RuntimeException)
{
if ( i_headings.getLength() != i_data.getLength() )
throw IllegalArgumentException( ::rtl::OUString(), *this, -1 );
::comphelper::ComponentGuard aGuard( *this, rBHelper );
if ( ( i_index < 0 ) || ( i_index > impl_getRowCount_nolck() ) )
throw IndexOutOfBoundsException( ::rtl::OUString(), *this );
sal_Int32 const rowCount = i_headings.getLength();
if ( rowCount == 0 )
return;
// determine max col count in the new data
sal_Int32 maxColCount = 0;
for ( sal_Int32 row=0; row<rowCount; ++row )
if ( i_data[row].getLength() > maxColCount )
maxColCount = i_data[row].getLength();
if ( maxColCount < m_nColumnCount )
maxColCount = m_nColumnCount;
for ( sal_Int32 row=0; row<rowCount; ++row )
{
impl_insertRow( i_index + row, i_headings[row], i_data[row], maxColCount );
}
if ( maxColCount > m_nColumnCount )
m_nColumnCount = maxColCount;
broadcast(
GridDataEvent( *this, -1, -1, i_index, i_index + rowCount - 1 ),
&XGridDataListener::rowsInserted,
aGuard
);
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::removeRow( ::sal_Int32 i_rowIndex ) throw (IndexOutOfBoundsException, RuntimeException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
if ( ( i_rowIndex < 0 ) || ( size_t( i_rowIndex ) >= m_aData.size() ) )
throw IndexOutOfBoundsException( ::rtl::OUString(), *this );
m_aRowHeaders.erase( m_aRowHeaders.begin() + i_rowIndex );
m_aData.erase( m_aData.begin() + i_rowIndex );
broadcast(
GridDataEvent( *this, -1, -1, i_rowIndex, i_rowIndex ),
&XGridDataListener::rowsRemoved,
aGuard
);
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::removeAllRows( ) throw (RuntimeException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
m_aRowHeaders.clear();
m_aData.clear();
broadcast(
GridDataEvent( *this, -1, -1, -1, -1 ),
&XGridDataListener::rowsRemoved,
aGuard
);
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::updateCellData( ::sal_Int32 i_columnIndex, ::sal_Int32 i_rowIndex, const Any& i_value ) throw (IndexOutOfBoundsException, RuntimeException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
impl_getCellDataAccess_throw( i_columnIndex, i_rowIndex ).first = i_value;
broadcast(
GridDataEvent( *this, i_columnIndex, i_columnIndex, i_rowIndex, i_rowIndex ),
&XGridDataListener::dataChanged,
aGuard
);
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::updateRowData( const Sequence< ::sal_Int32 >& i_columnIndexes, ::sal_Int32 i_rowIndex, const Sequence< Any >& i_values ) throw (IndexOutOfBoundsException, IllegalArgumentException, RuntimeException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
if ( ( i_rowIndex < 0 ) || ( size_t( i_rowIndex ) >= m_aData.size() ) )
throw IndexOutOfBoundsException( ::rtl::OUString(), *this );
if ( i_columnIndexes.getLength() != i_values.getLength() )
throw IllegalArgumentException( ::rtl::OUString(), *this, 1 );
sal_Int32 const columnCount = i_columnIndexes.getLength();
if ( columnCount == 0 )
return;
for ( sal_Int32 col = 0; col < columnCount; ++col )
{
if ( ( i_columnIndexes[col] < 0 ) || ( i_columnIndexes[col] > m_nColumnCount ) )
throw IndexOutOfBoundsException( ::rtl::OUString(), *this );
}
RowData& rDataRow = m_aData[ i_rowIndex ];
for ( sal_Int32 col = 0; col < columnCount; ++col )
{
sal_Int32 const columnIndex = i_columnIndexes[ col ];
if ( size_t( columnIndex ) >= rDataRow.size() )
rDataRow.resize( columnIndex + 1 );
rDataRow[ columnIndex ].first = i_values[ col ];
}
sal_Int32 const firstAffectedColumn = *::std::min_element( stl_begin( i_columnIndexes ), stl_end( i_columnIndexes ) );
sal_Int32 const lastAffectedColumn = *::std::max_element( stl_begin( i_columnIndexes ), stl_end( i_columnIndexes ) );
broadcast(
GridDataEvent( *this, firstAffectedColumn, lastAffectedColumn, i_rowIndex, i_rowIndex ),
&XGridDataListener::dataChanged,
aGuard
);
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::updateRowHeading( ::sal_Int32 i_rowIndex, const Any& i_heading ) throw (IndexOutOfBoundsException, RuntimeException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
if ( ( i_rowIndex < 0 ) || ( size_t( i_rowIndex ) >= m_aRowHeaders.size() ) )
throw IndexOutOfBoundsException( ::rtl::OUString(), *this );
m_aRowHeaders[ i_rowIndex ] = i_heading;
broadcast(
GridDataEvent( *this, -1, -1, i_rowIndex, i_rowIndex ),
&XGridDataListener::rowHeadingChanged,
aGuard
);
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::updateCellToolTip( ::sal_Int32 i_columnIndex, ::sal_Int32 i_rowIndex, const Any& i_value ) throw (IndexOutOfBoundsException, RuntimeException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
impl_getCellDataAccess_throw( i_columnIndex, i_rowIndex ).second = i_value;
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::updateRowToolTip( ::sal_Int32 i_rowIndex, const Any& i_value ) throw (IndexOutOfBoundsException, RuntimeException)
{
::comphelper::ComponentGuard aGuard( *this, rBHelper );
RowData& rRowData = impl_getRowDataAccess_throw( i_rowIndex, m_nColumnCount );
for ( RowData::iterator cell = rRowData.begin(); cell != rRowData.end(); ++cell )
cell->second = i_value;
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::addGridDataListener( const Reference< grid::XGridDataListener >& i_listener ) throw (RuntimeException)
{
rBHelper.addListener( XGridDataListener::static_type(), i_listener );
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::removeGridDataListener( const Reference< grid::XGridDataListener >& i_listener ) throw (RuntimeException)
{
rBHelper.removeListener( XGridDataListener::static_type(), i_listener );
}
//------------------------------------------------------------------------------------------------------------------
void SAL_CALL DefaultGridDataModel::disposing()
{
::com::sun::star::lang::EventObject aEvent;
aEvent.Source.set( *this );
rBHelper.aLC.disposeAndClear( aEvent );
::osl::MutexGuard aGuard( m_aMutex );
GridData aEmptyData;
m_aData.swap( aEmptyData );
::std::vector< Any > aEmptyRowHeaders;
m_aRowHeaders.swap( aEmptyRowHeaders );
m_nColumnCount = 0;
}
//------------------------------------------------------------------------------------------------------------------
::rtl::OUString SAL_CALL DefaultGridDataModel::getImplementationName( ) throw (RuntimeException)
{
static const ::rtl::OUString aImplName( RTL_CONSTASCII_USTRINGPARAM( "toolkit.DefaultGridDataModel" ) );
return aImplName;
}
//------------------------------------------------------------------------------------------------------------------
sal_Bool SAL_CALL DefaultGridDataModel::supportsService( const ::rtl::OUString& ServiceName ) throw (RuntimeException)
{
return ServiceName.equalsAscii( szServiceName_DefaultGridDataModel );
}
//------------------------------------------------------------------------------------------------------------------
Sequence< ::rtl::OUString > SAL_CALL DefaultGridDataModel::getSupportedServiceNames( ) throw (RuntimeException)
{
static const ::rtl::OUString aServiceName( ::rtl::OUString::createFromAscii( szServiceName_DefaultGridDataModel ) );
static const Sequence< ::rtl::OUString > aSeq( &aServiceName, 1 );
return aSeq;
}
//------------------------------------------------------------------------------------------------------------------
Reference< XCloneable > SAL_CALL DefaultGridDataModel::createClone( ) throw (RuntimeException)
{
return new DefaultGridDataModel( *this );
}
//......................................................................................................................
} // namespace toolkit
//......................................................................................................................
Reference< XInterface > SAL_CALL DefaultGridDataModel_CreateInstance( const Reference< XMultiServiceFactory >& )
{
return Reference < XInterface >( ( ::cppu::OWeakObject* ) new ::toolkit::DefaultGridDataModel() );
}