/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * 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 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "svx/unoprov.hxx" #include "svx/sdr/table/tabledesign.hxx" #include "svx/dialmgr.hxx" #include "svx/dialogs.hrc" #include "celltypes.hxx" #include #include #include using namespace ::com::sun::star::uno; using namespace ::com::sun::star::style; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::util; using namespace ::com::sun::star::container; using ::osl::MutexGuard; using ::osl::ClearableMutexGuard; using ::cppu::OInterfaceContainerHelper; namespace sdr { namespace table { typedef std::map< OUString, sal_Int32 > CellStyleNameMap; typedef ::cppu::WeakComponentImplHelper< XStyle, XNameReplace, XServiceInfo, XIndexAccess, XModifyBroadcaster, XModifyListener > TableDesignStyleBase; class TableDesignStyle : private ::cppu::BaseMutex, public TableDesignStyleBase { public: TableDesignStyle(); // XServiceInfo virtual OUString SAL_CALL getImplementationName() throw(RuntimeException, std::exception) override; virtual sal_Bool SAL_CALL supportsService( const OUString& ServiceName ) throw(RuntimeException, std::exception) override; virtual Sequence< OUString > SAL_CALL getSupportedServiceNames() throw(RuntimeException, std::exception) override; // XStyle virtual sal_Bool SAL_CALL isUserDefined() throw (RuntimeException, std::exception) override; virtual sal_Bool SAL_CALL isInUse() throw (RuntimeException, std::exception) override; virtual OUString SAL_CALL getParentStyle() throw (RuntimeException, std::exception) override; virtual void SAL_CALL setParentStyle( const OUString& aParentStyle ) throw (NoSuchElementException, RuntimeException, std::exception) override; // XNamed virtual OUString SAL_CALL getName() throw (RuntimeException, std::exception) override; virtual void SAL_CALL setName( const OUString& aName ) throw (RuntimeException, std::exception) override; // XNameAccess virtual Any SAL_CALL getByName( const OUString& aName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException, std::exception) override; virtual Sequence< OUString > SAL_CALL getElementNames() throw(RuntimeException, std::exception) override; virtual sal_Bool SAL_CALL hasByName( const OUString& aName ) throw(RuntimeException, std::exception) override; // XElementAccess virtual css::uno::Type SAL_CALL getElementType() throw(RuntimeException, std::exception) override; virtual sal_Bool SAL_CALL hasElements() throw(RuntimeException, std::exception) override; // XIndexAccess virtual sal_Int32 SAL_CALL getCount() throw(RuntimeException, std::exception) override ; virtual Any SAL_CALL getByIndex( sal_Int32 Index ) throw(IndexOutOfBoundsException, WrappedTargetException, RuntimeException, std::exception) override; // XNameReplace virtual void SAL_CALL replaceByName( const OUString& aName, const Any& aElement ) throw(IllegalArgumentException, NoSuchElementException, WrappedTargetException, RuntimeException, std::exception) override; // XModifyBroadcaster virtual void SAL_CALL addModifyListener( const Reference< XModifyListener >& aListener ) throw (RuntimeException, std::exception) override; virtual void SAL_CALL removeModifyListener( const Reference< XModifyListener >& aListener ) throw (RuntimeException, std::exception) override; // XModifyListener virtual void SAL_CALL modified( const css::lang::EventObject& aEvent ) throw (css::uno::RuntimeException, std::exception) override; virtual void SAL_CALL disposing( const css::lang::EventObject& Source ) throw (css::uno::RuntimeException, std::exception) override; void notifyModifyListener(); // this function is called upon disposing the component virtual void SAL_CALL disposing() override; static const CellStyleNameMap& getCellStyleNameMap(); OUString msName; Reference< XStyle > maCellStyles[style_count]; }; typedef std::vector< Reference< XStyle > > TableDesignStyleVector; class TableDesignFamily : public ::cppu::WeakImplHelper< XNameContainer, XNamed, XIndexAccess, XSingleServiceFactory, XServiceInfo, XComponent, XPropertySet > { public: // XServiceInfo virtual OUString SAL_CALL getImplementationName() throw(RuntimeException, std::exception) override; virtual sal_Bool SAL_CALL supportsService( const OUString& ServiceName ) throw(RuntimeException, std::exception) override; virtual Sequence< OUString > SAL_CALL getSupportedServiceNames() throw(RuntimeException, std::exception) override; // XNamed virtual OUString SAL_CALL getName( ) throw (RuntimeException, std::exception) override; virtual void SAL_CALL setName( const OUString& aName ) throw (RuntimeException, std::exception) override; // XNameAccess virtual Any SAL_CALL getByName( const OUString& aName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException, std::exception) override; virtual Sequence< OUString > SAL_CALL getElementNames() throw(RuntimeException, std::exception) override; virtual sal_Bool SAL_CALL hasByName( const OUString& aName ) throw(RuntimeException, std::exception) override; // XElementAccess virtual Type SAL_CALL getElementType() throw(RuntimeException, std::exception) override; virtual sal_Bool SAL_CALL hasElements() throw(RuntimeException, std::exception) override; // XIndexAccess virtual sal_Int32 SAL_CALL getCount() throw(RuntimeException, std::exception) override ; virtual Any SAL_CALL getByIndex( sal_Int32 Index ) throw(IndexOutOfBoundsException, WrappedTargetException, RuntimeException, std::exception) override; // XNameContainer virtual void SAL_CALL insertByName( const OUString& aName, const Any& aElement ) throw(IllegalArgumentException, ElementExistException, WrappedTargetException, RuntimeException, std::exception) override; virtual void SAL_CALL removeByName( const OUString& Name ) throw(NoSuchElementException, WrappedTargetException, RuntimeException, std::exception) override; // XNameReplace virtual void SAL_CALL replaceByName( const OUString& aName, const Any& aElement ) throw(IllegalArgumentException, NoSuchElementException, WrappedTargetException, RuntimeException, std::exception) override; // XSingleServiceFactory virtual Reference< XInterface > SAL_CALL createInstance( ) throw(Exception, RuntimeException, std::exception) override; virtual Reference< XInterface > SAL_CALL createInstanceWithArguments( const Sequence< Any >& aArguments ) throw(Exception, RuntimeException, std::exception) override; // XComponent virtual void SAL_CALL dispose( ) throw (RuntimeException, std::exception) override; virtual void SAL_CALL addEventListener( const Reference< XEventListener >& xListener ) throw (RuntimeException, std::exception) override; virtual void SAL_CALL removeEventListener( const Reference< XEventListener >& aListener ) throw (RuntimeException, std::exception) override; // XPropertySet virtual Reference SAL_CALL getPropertySetInfo() throw (RuntimeException, std::exception) override; virtual void SAL_CALL setPropertyValue( const OUString& aPropertyName, const Any& aValue ) throw (UnknownPropertyException, PropertyVetoException, IllegalArgumentException, WrappedTargetException, RuntimeException, std::exception) override; virtual Any SAL_CALL getPropertyValue( const OUString& PropertyName ) throw (UnknownPropertyException, WrappedTargetException, RuntimeException, std::exception) override; virtual void SAL_CALL addPropertyChangeListener( const OUString& aPropertyName, const Reference& xListener ) throw (UnknownPropertyException, WrappedTargetException, RuntimeException, std::exception) override; virtual void SAL_CALL removePropertyChangeListener( const OUString& aPropertyName, const Reference& aListener ) throw (UnknownPropertyException, WrappedTargetException, RuntimeException, std::exception) override; virtual void SAL_CALL addVetoableChangeListener(const OUString& PropertyName, const Reference& aListener ) throw (UnknownPropertyException, WrappedTargetException, RuntimeException, std::exception) override; virtual void SAL_CALL removeVetoableChangeListener(const OUString& PropertyName,const Reference&aListener ) throw (UnknownPropertyException, WrappedTargetException, RuntimeException, std::exception) override; TableDesignStyleVector maDesigns; }; TableDesignStyle::TableDesignStyle() : TableDesignStyleBase(m_aMutex) { } const CellStyleNameMap& TableDesignStyle::getCellStyleNameMap() { static CellStyleNameMap aMap; if( aMap.empty() ) { CellStyleNameMap aNewMap; aNewMap[ OUString( "first-row" ) ] = first_row_style; aNewMap[ OUString( "last-row" ) ] = last_row_style; aNewMap[ OUString( "first-column" ) ] = first_column_style; aNewMap[ OUString( "last-column" ) ] = last_column_style; aNewMap[ OUString( "body" ) ] = body_style; aNewMap[ OUString( "even-rows" ) ] = even_rows_style; aNewMap[ OUString( "odd-rows" ) ] = odd_rows_style; aNewMap[ OUString( "even-columns" ) ] = even_columns_style; aNewMap[ OUString( "odd-columns" ) ] = odd_columns_style; aNewMap[ OUString( "background" ) ] = background_style; aMap.swap( aNewMap ); } return aMap; } // XServiceInfo OUString SAL_CALL TableDesignStyle::getImplementationName() throw(RuntimeException, std::exception) { return OUString("TableDesignStyle"); } sal_Bool SAL_CALL TableDesignStyle::supportsService( const OUString& ServiceName ) throw(RuntimeException, std::exception) { return cppu::supportsService( this, ServiceName ); } Sequence< OUString > SAL_CALL TableDesignStyle::getSupportedServiceNames() throw(RuntimeException, std::exception) { OUString aServiceName("com.sun.star.style.Style"); Sequence< OUString > aSeq( &aServiceName, 1 ); return aSeq; } // XStyle sal_Bool SAL_CALL TableDesignStyle::isUserDefined() throw (RuntimeException, std::exception) { return sal_False; } sal_Bool SAL_CALL TableDesignStyle::isInUse() throw (RuntimeException, std::exception) { ClearableMutexGuard aGuard( rBHelper.rMutex ); OInterfaceContainerHelper * pContainer = rBHelper.getContainer( cppu::UnoType::get() ); if( pContainer ) { Sequence< Reference< XInterface > > aListener( pContainer->getElements() ); aGuard.clear(); sal_Int32 nIndex = aListener.getLength(); while( --nIndex >= 0 ) { TableDesignUser* pUser = dynamic_cast< TableDesignUser* >( aListener[nIndex].get() ); if( pUser && pUser->isInUse() ) return sal_True; } } return sal_False; } OUString SAL_CALL TableDesignStyle::getParentStyle() throw (RuntimeException, std::exception) { return OUString(); } void SAL_CALL TableDesignStyle::setParentStyle( const OUString& ) throw (NoSuchElementException, RuntimeException, std::exception) { } // XNamed OUString SAL_CALL TableDesignStyle::getName() throw (RuntimeException, std::exception) { return msName; } void SAL_CALL TableDesignStyle::setName( const OUString& rName ) throw (RuntimeException, std::exception) { msName = rName; } // XNameAccess Any SAL_CALL TableDesignStyle::getByName( const OUString& rName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException, std::exception) { SolarMutexGuard aGuard; const CellStyleNameMap& rMap = getCellStyleNameMap(); CellStyleNameMap::const_iterator iter = rMap.find( rName ); if( iter == rMap.end() ) throw NoSuchElementException(); return Any( maCellStyles[(*iter).second] ); } Sequence< OUString > SAL_CALL TableDesignStyle::getElementNames() throw(RuntimeException, std::exception) { SolarMutexGuard aGuard; return comphelper::mapKeysToSequence( getCellStyleNameMap() ); } sal_Bool SAL_CALL TableDesignStyle::hasByName( const OUString& rName ) throw(RuntimeException, std::exception) { SolarMutexGuard aGuard; const CellStyleNameMap& rMap = getCellStyleNameMap(); CellStyleNameMap::const_iterator iter = rMap.find( rName ); return iter != rMap.end(); } // XElementAccess Type SAL_CALL TableDesignStyle::getElementType() throw(RuntimeException, std::exception) { return cppu::UnoType::get(); } sal_Bool SAL_CALL TableDesignStyle::hasElements() throw(RuntimeException, std::exception) { return sal_True; } // XIndexAccess sal_Int32 SAL_CALL TableDesignStyle::getCount() throw(RuntimeException, std::exception) { return style_count; } Any SAL_CALL TableDesignStyle::getByIndex( sal_Int32 Index ) throw(IndexOutOfBoundsException, WrappedTargetException, RuntimeException, std::exception) { SolarMutexGuard aGuard; if( (Index < 0) || (Index >= style_count) ) throw IndexOutOfBoundsException(); return Any( maCellStyles[Index] ); } // XNameReplace void SAL_CALL TableDesignStyle::replaceByName( const OUString& rName, const Any& aElement ) throw(IllegalArgumentException, NoSuchElementException, WrappedTargetException, RuntimeException, std::exception) { SolarMutexGuard aGuard; const CellStyleNameMap& rMap = getCellStyleNameMap(); CellStyleNameMap::const_iterator iter = rMap.find( rName ); if( iter == rMap.end() ) throw NoSuchElementException(); Reference< XStyle > xNewStyle; if( !(aElement >>= xNewStyle) ) throw IllegalArgumentException(); const sal_Int32 nIndex = (*iter).second; Reference< XStyle > xOldStyle( maCellStyles[nIndex] ); if( xNewStyle != xOldStyle ) { Reference< XModifyListener > xListener( this ); // end listening to old style, if possible Reference< XModifyBroadcaster > xOldBroadcaster( xOldStyle, UNO_QUERY ); if( xOldBroadcaster.is() ) xOldBroadcaster->removeModifyListener( xListener ); // start listening to new style, if possible Reference< XModifyBroadcaster > xNewBroadcaster( xNewStyle, UNO_QUERY ); if( xNewBroadcaster.is() ) xNewBroadcaster->addModifyListener( xListener ); maCellStyles[nIndex] = xNewStyle; } } // XComponent void SAL_CALL TableDesignStyle::disposing() { for( sal_Int32 nIndex = 0; nIndex < style_count; nIndex++ ) maCellStyles[nIndex].clear(); } // XModifyBroadcaster void SAL_CALL TableDesignStyle::addModifyListener( const Reference< XModifyListener >& xListener ) throw (RuntimeException, std::exception) { ClearableMutexGuard aGuard( rBHelper.rMutex ); if (rBHelper.bDisposed || rBHelper.bInDispose) { aGuard.clear(); EventObject aEvt( static_cast< OWeakObject * >( this ) ); xListener->disposing( aEvt ); } else { rBHelper.addListener( cppu::UnoType::get(), xListener ); } } void SAL_CALL TableDesignStyle::removeModifyListener( const Reference< XModifyListener >& xListener ) throw (RuntimeException, std::exception) { rBHelper.removeListener( cppu::UnoType::get(), xListener ); } void TableDesignStyle::notifyModifyListener() { MutexGuard aGuard( rBHelper.rMutex ); OInterfaceContainerHelper * pContainer = rBHelper.getContainer( cppu::UnoType::get() ); if( pContainer ) { EventObject aEvt( static_cast< OWeakObject * >( this ) ); pContainer->forEach( boost::bind( &XModifyListener::modified, _1, boost::cref( aEvt ) ) ); } } // XModifyListener // if we get a modify hint from a style, notify all registered XModifyListener void SAL_CALL TableDesignStyle::modified( const css::lang::EventObject& ) throw (css::uno::RuntimeException, std::exception) { notifyModifyListener(); } void SAL_CALL TableDesignStyle::disposing( const css::lang::EventObject& ) throw (css::uno::RuntimeException, std::exception) { } // TableStyle // XServiceInfo OUString SAL_CALL TableDesignFamily::getImplementationName() throw(RuntimeException, std::exception) { return OUString("TableDesignFamily"); } sal_Bool SAL_CALL TableDesignFamily::supportsService( const OUString& ServiceName ) throw(RuntimeException, std::exception) { return cppu::supportsService( this, ServiceName ); } Sequence< OUString > SAL_CALL TableDesignFamily::getSupportedServiceNames() throw(RuntimeException, std::exception) { OUString aServiceName("com.sun.star.style.StyleFamily"); Sequence< OUString > aSeq( &aServiceName, 1 ); return aSeq; } // XNamed OUString SAL_CALL TableDesignFamily::getName() throw (RuntimeException, std::exception) { return OUString( "table" ); } void SAL_CALL TableDesignFamily::setName( const OUString& ) throw (RuntimeException, std::exception) { } // XNameAccess Any SAL_CALL TableDesignFamily::getByName( const OUString& rName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException, std::exception) { SolarMutexGuard aGuard; const TableDesignStyleVector::const_iterator aEnd( maDesigns.end() ); for( TableDesignStyleVector::const_iterator iter( maDesigns.begin() ); iter != aEnd; ++iter) { if( (*iter)->getName() == rName ) return Any( (*iter) ); } throw NoSuchElementException(); } Sequence< OUString > SAL_CALL TableDesignFamily::getElementNames() throw(RuntimeException, std::exception) { SolarMutexGuard aGuard; Sequence< OUString > aRet( maDesigns.size() ); OUString* pNames = aRet.getArray(); const TableDesignStyleVector::const_iterator aEnd( maDesigns.end() ); for( TableDesignStyleVector::const_iterator iter( maDesigns.begin() ); iter != aEnd; ++iter) *pNames++ = (*iter)->getName(); return aRet; } sal_Bool SAL_CALL TableDesignFamily::hasByName( const OUString& aName ) throw(RuntimeException, std::exception) { SolarMutexGuard aGuard; const TableDesignStyleVector::const_iterator aEnd( maDesigns.end() ); for( TableDesignStyleVector::const_iterator iter( maDesigns.begin() ); iter != aEnd; ++iter) if( (*iter)->getName() == aName ) return sal_True; return sal_False; } // XElementAccess Type SAL_CALL TableDesignFamily::getElementType() throw(RuntimeException, std::exception) { return cppu::UnoType::get(); } sal_Bool SAL_CALL TableDesignFamily::hasElements() throw(RuntimeException, std::exception) { SolarMutexGuard aGuard; return !maDesigns.empty(); } // XIndexAccess sal_Int32 SAL_CALL TableDesignFamily::getCount() throw(RuntimeException, std::exception) { SolarMutexGuard aGuard; return sal::static_int_cast< sal_Int32 >( maDesigns.size() ); } Any SAL_CALL TableDesignFamily::getByIndex( sal_Int32 Index ) throw(IndexOutOfBoundsException, WrappedTargetException, RuntimeException, std::exception) { SolarMutexGuard aGuard; if( (Index >= 0) && (Index < sal::static_int_cast< sal_Int32 >( maDesigns.size() ) ) ) return Any( maDesigns[Index] ); throw IndexOutOfBoundsException(); } // XNameContainer void SAL_CALL TableDesignFamily::insertByName( const OUString& rName, const Any& rElement ) throw(IllegalArgumentException, ElementExistException, WrappedTargetException, RuntimeException, std::exception) { SolarMutexGuard aGuard; Reference< XStyle > xStyle( rElement, UNO_QUERY ); if( !xStyle.is() ) throw IllegalArgumentException(); xStyle->setName( rName ); const TableDesignStyleVector::const_iterator aEnd( maDesigns.end() ); for( TableDesignStyleVector::const_iterator iter( maDesigns.begin() ); iter != aEnd; ++iter) if( (*iter)->getName() == rName ) throw ElementExistException(); maDesigns.push_back( xStyle ); } void SAL_CALL TableDesignFamily::removeByName( const OUString& rName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException, std::exception) { SolarMutexGuard aGuard; const TableDesignStyleVector::const_iterator aEnd( maDesigns.end() ); for( TableDesignStyleVector::iterator iter( maDesigns.begin() ); iter != aEnd; ++iter) { if( (*iter)->getName() == rName ) { maDesigns.erase( iter ); return; } } throw NoSuchElementException(); } // XNameReplace void SAL_CALL TableDesignFamily::replaceByName( const OUString& rName, const Any& aElement ) throw(IllegalArgumentException, NoSuchElementException, WrappedTargetException, RuntimeException, std::exception) { SolarMutexGuard aGuard; Reference< XStyle > xStyle( aElement, UNO_QUERY ); if( !xStyle.is() ) throw IllegalArgumentException(); const TableDesignStyleVector::const_iterator aEnd( maDesigns.end() ); for( TableDesignStyleVector::iterator iter( maDesigns.begin() ); iter != aEnd; ++iter) { if( (*iter)->getName() == rName ) { (*iter) = xStyle; xStyle->setName( rName ); return; } } throw NoSuchElementException(); } // XSingleServiceFactory Reference< XInterface > SAL_CALL TableDesignFamily::createInstance() throw(Exception, RuntimeException, std::exception) { SolarMutexGuard aGuard; return Reference< XInterface >( static_cast< XStyle* >( new TableDesignStyle ) ); } Reference< XInterface > SAL_CALL TableDesignFamily::createInstanceWithArguments( const Sequence< Any >& ) throw(Exception, RuntimeException, std::exception) { return createInstance(); } // XComponent void SAL_CALL TableDesignFamily::dispose( ) throw (RuntimeException, std::exception) { TableDesignStyleVector aDesigns; aDesigns.swap( maDesigns ); for( TableDesignStyleVector::iterator iter( aDesigns.begin() ); iter != aDesigns.end(); ++iter ) { Reference< XComponent > xComp( (*iter), UNO_QUERY ); if( xComp.is() ) xComp->dispose(); } } void SAL_CALL TableDesignFamily::addEventListener( const Reference< XEventListener >& ) throw (RuntimeException, std::exception) { } void SAL_CALL TableDesignFamily::removeEventListener( const Reference< XEventListener >& ) throw (RuntimeException, std::exception) { } // XPropertySet Reference TableDesignFamily::getPropertySetInfo() throw (RuntimeException, std::exception) { OSL_FAIL( "###unexpected!" ); return Reference(); } void TableDesignFamily::setPropertyValue( const OUString& , const Any& ) throw (UnknownPropertyException, PropertyVetoException, IllegalArgumentException, WrappedTargetException, RuntimeException, std::exception) { OSL_FAIL( "###unexpected!" ); } Any TableDesignFamily::getPropertyValue( const OUString& PropertyName ) throw (UnknownPropertyException, WrappedTargetException, RuntimeException, std::exception) { if ( PropertyName == "DisplayName" ) { OUString sDisplayName( SVX_RESSTR( RID_SVXSTR_STYLEFAMILY_TABLEDESIGN ) ); return Any( sDisplayName ); } else { throw UnknownPropertyException( "unknown property: " + PropertyName, static_cast(this) ); } } void TableDesignFamily::addPropertyChangeListener( const OUString& , const Reference& ) throw (UnknownPropertyException, WrappedTargetException, RuntimeException, std::exception) { OSL_FAIL( "###unexpected!" ); } void TableDesignFamily::removePropertyChangeListener( const OUString& , const Reference& ) throw (UnknownPropertyException, WrappedTargetException, RuntimeException, std::exception) { OSL_FAIL( "###unexpected!" ); } void TableDesignFamily::addVetoableChangeListener( const OUString& , const Reference& ) throw (UnknownPropertyException, WrappedTargetException, RuntimeException, std::exception) { OSL_FAIL( "###unexpected!" ); } void TableDesignFamily::removeVetoableChangeListener( const OUString& , const Reference& ) throw (UnknownPropertyException, WrappedTargetException, RuntimeException, std::exception) { OSL_FAIL( "###unexpected!" ); } Reference< XNameAccess > CreateTableDesignFamily() { return new TableDesignFamily(); } } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */