/* -*- 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 using namespace dbaccess; using namespace connectivity; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::sdbcx; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::awt; using namespace ::com::sun::star::io; using namespace ::com::sun::star::container; using namespace ::com::sun::star::util; using namespace ::osl; using namespace ::comphelper; using namespace ::cppu; // OColumn OColumn::OColumn( const bool _bNameIsReadOnly ) :OColumnBase( m_aMutex ) ,::comphelper::OPropertyContainer( OColumnBase::rBHelper ) { registerProperty( PROPERTY_NAME, PROPERTY_ID_NAME, _bNameIsReadOnly ? PropertyAttribute::READONLY : 0, &m_sName, cppu::UnoType::get() ); } OColumn::~OColumn() { } // css::lang::XTypeProvider Sequence< Type > OColumn::getTypes() { return ::comphelper::concatSequences( OColumnBase::getTypes(), getBaseTypes() ); } // css::uno::XInterface IMPLEMENT_FORWARD_XINTERFACE2( OColumn, OColumnBase, ::comphelper::OPropertyContainer ) // css::lang::XServiceInfo OUString OColumn::getImplementationName( ) { return "com.sun.star.sdb.OColumn"; } sal_Bool OColumn::supportsService( const OUString& _rServiceName ) { return cppu::supportsService(this, _rServiceName); } Sequence< OUString > OColumn::getSupportedServiceNames( ) { return { SERVICE_SDBCX_COLUMN }; } // OComponentHelper void OColumn::disposing() { OPropertyContainer::disposing(); } // css::beans::XPropertySet Reference< XPropertySetInfo > OColumn::getPropertySetInfo() { return createPropertySetInfo( getInfoHelper() ) ; } OUString SAL_CALL OColumn::getName( ) { return m_sName; } void SAL_CALL OColumn::setName( const OUString& _rName ) { m_sName = _rName; } void OColumn::registerProperty( const OUString& _rName, sal_Int32 _nHandle, sal_Int32 _nAttributes, void* _pPointerToMember, const Type& _rMemberType ) { ::comphelper::OPropertyContainer::registerProperty( _rName, _nHandle, _nAttributes, _pPointerToMember, _rMemberType ); } void OColumn::registerMayBeVoidProperty( const OUString& _rName, sal_Int32 _nHandle, sal_Int32 _nAttributes, Any* _pPointerToMember, const Type& _rExpectedType ) { ::comphelper::OPropertyContainer::registerMayBeVoidProperty( _rName, _nHandle, _nAttributes, _pPointerToMember, _rExpectedType ); } // OColumns OColumns::OColumns(::cppu::OWeakObject& _rParent, ::osl::Mutex& _rMutex, bool _bCaseSensitive,const std::vector< OUString> &_rVector, IColumnFactory* _pColFactory, ::connectivity::sdbcx::IRefreshableColumns* _pRefresh, bool _bAddColumn, bool _bDropColumn, bool _bUseHardRef) : OColumns_BASE(_rParent,_bCaseSensitive,_rMutex,_rVector,_bUseHardRef) ,m_pMediator(nullptr) ,m_pColFactoryImpl(_pColFactory) ,m_pRefreshColumns(_pRefresh) ,m_bInitialized(false) ,m_bAddColumn(_bAddColumn) ,m_bDropColumn(_bDropColumn) { } OColumns::OColumns(::cppu::OWeakObject& _rParent, ::osl::Mutex& _rMutex, const css::uno::Reference< css::container::XNameAccess >& _rxDrvColumns, bool _bCaseSensitive,const std::vector< OUString> &_rVector, IColumnFactory* _pColFactory, ::connectivity::sdbcx::IRefreshableColumns* _pRefresh, bool _bAddColumn, bool _bDropColumn, bool _bUseHardRef) : OColumns_BASE(_rParent,_bCaseSensitive,_rMutex,_rVector,_bUseHardRef) ,m_pMediator(nullptr) ,m_xDrvColumns(_rxDrvColumns) ,m_pColFactoryImpl(_pColFactory) ,m_pRefreshColumns(_pRefresh) ,m_bInitialized(false) ,m_bAddColumn(_bAddColumn) ,m_bDropColumn(_bDropColumn) { } OColumns::~OColumns() { } // XServiceInfo OUString OColumns::getImplementationName( ) { return "com.sun.star.sdb.OColumns"; } sal_Bool OColumns::supportsService( const OUString& _rServiceName ) { return cppu::supportsService(this, _rServiceName); } Sequence< OUString > OColumns::getSupportedServiceNames( ) { return { SERVICE_SDBCX_CONTAINER }; } void OColumns::append( const OUString& _rName, OColumn* _pColumn ) { MutexGuard aGuard(m_rMutex); OSL_ENSURE( _pColumn, "OColumns::append: invalid column!" ); OSL_ENSURE( !m_pElements->exists( _rName ),"OColumns::append: Column already exists"); _pColumn->m_sName = _rName; // now really insert the column insertElement( _rName, _pColumn ); } void OColumns::clearColumns() { MutexGuard aGuard(m_rMutex); disposing(); } void OColumns::disposing() { MutexGuard aGuard(m_rMutex); m_xDrvColumns = nullptr; m_pMediator = nullptr; m_pColFactoryImpl = nullptr; OColumns_BASE::disposing(); } void OColumns::impl_refresh() { if (m_pRefreshColumns) m_pRefreshColumns->refreshColumns(); } connectivity::sdbcx::ObjectType OColumns::createObject(const OUString& _rName) { OSL_ENSURE(m_pColFactoryImpl, "OColumns::createObject: no column factory!"); connectivity::sdbcx::ObjectType xRet; if ( m_pColFactoryImpl ) { xRet = m_pColFactoryImpl->createColumn(_rName); Reference xChild(xRet,UNO_QUERY); if ( xChild.is() ) xChild->setParent(static_cast(static_cast(this))); } Reference xDest(xRet,UNO_QUERY); if ( m_pMediator && xDest.is() ) m_pMediator->notifyElementCreated(_rName,xDest); return xRet; } Reference< XPropertySet > OColumns::createDescriptor() { if ( m_pColFactoryImpl ) { Reference xRet = m_pColFactoryImpl->createColumnDescriptor(); Reference xChild(xRet,UNO_QUERY); if ( xChild.is() ) xChild->setParent(static_cast(static_cast(this))); return xRet; } else return Reference< XPropertySet >(); } Any SAL_CALL OColumns::queryInterface( const Type & rType ) { Any aRet; if(m_xDrvColumns.is()) { aRet = m_xDrvColumns->queryInterface(rType); if ( aRet.hasValue() ) aRet = OColumns_BASE::queryInterface( rType); if ( !aRet.hasValue() ) aRet = TXChild::queryInterface( rType); return aRet; } else if(!m_pTable || !m_pTable->isNew()) { if(!m_bAddColumn && rType == cppu::UnoType::get()) return Any(); if(!m_bDropColumn && rType == cppu::UnoType::get()) return Any(); } aRet = OColumns_BASE::queryInterface( rType); if ( !aRet.hasValue() ) aRet = TXChild::queryInterface( rType); return aRet; } Sequence< Type > SAL_CALL OColumns::getTypes( ) { bool bAppendFound = false,bDropFound = false; sal_Int32 nSize = 0; Type aAppendType = cppu::UnoType::get(); Type aDropType = cppu::UnoType::get(); if(m_xDrvColumns.is()) { Reference xTypes(m_xDrvColumns,UNO_QUERY); Sequence< Type > aTypes(xTypes->getTypes()); const Type* pBegin = aTypes.getConstArray(); const Type* pEnd = pBegin + aTypes.getLength(); for (;pBegin != pEnd ; ++pBegin) { if(aAppendType == *pBegin) bAppendFound = true; else if(aDropType == *pBegin) bDropFound = true; } nSize = (bDropFound ? (bAppendFound ? 0 : 1) : (bAppendFound ? 1 : 2)); } else { if (m_pTable && m_pTable->isNew()) nSize = 0; else if (m_bDropColumn) nSize = m_bAddColumn ? 0 : 1; else nSize = m_bAddColumn ? 1 : 2; bDropFound = (m_pTable && m_pTable->isNew()) || m_bDropColumn; bAppendFound = (m_pTable && m_pTable->isNew()) || m_bAddColumn; } Sequence< Type > aTypes(::comphelper::concatSequences(OColumns_BASE::getTypes(),TXChild::getTypes())); Sequence< Type > aRet(aTypes.getLength() - nSize); const Type* pBegin = aTypes.getConstArray(); const Type* pEnd = pBegin + aTypes.getLength(); for(sal_Int32 i=0;pBegin != pEnd ;++pBegin) { if(*pBegin != aAppendType && *pBegin != aDropType) aRet.getArray()[i++] = *pBegin; else if(bDropFound && *pBegin == aDropType) aRet.getArray()[i++] = *pBegin; else if(bAppendFound && *pBegin == aAppendType) aRet.getArray()[i++] = *pBegin; } return aRet; } // XAppend sdbcx::ObjectType OColumns::appendObject( const OUString& _rForName, const Reference< XPropertySet >& descriptor ) { sdbcx::ObjectType xReturn; Reference< XAppend > xAppend( m_xDrvColumns, UNO_QUERY ); if ( xAppend.is() ) { xAppend->appendByDescriptor(descriptor); xReturn = createObject( _rForName ); } else if ( m_pTable && !m_pTable->isNew() ) { if ( m_bAddColumn ) { Reference< css::sdb::tools::XTableAlteration> xAlterService = m_pTable->getAlterService(); if ( xAlterService.is() ) { xAlterService->addColumn(m_pTable,descriptor); xReturn = createObject( _rForName ); } else xReturn = OColumns_BASE::appendObject( _rForName, descriptor ); } else ::dbtools::throwGenericSQLException( DBA_RES( RID_STR_NO_COLUMN_ADD ), static_cast(static_cast(this)) ); } else xReturn = cloneDescriptor( descriptor ); if ( m_pColFactoryImpl ) m_pColFactoryImpl->columnAppended( descriptor ); ::dbaccess::notifyDataSourceModified(m_xParent); return xReturn; } // XDrop void OColumns::dropObject(sal_Int32 _nPos, const OUString& _sElementName) { Reference< XDrop > xDrop( m_xDrvColumns, UNO_QUERY ); if ( xDrop.is() ) { xDrop->dropByName( _sElementName ); } else if ( m_pTable && !m_pTable->isNew() ) { if ( m_bDropColumn ) { Reference< css::sdb::tools::XTableAlteration> xAlterService = m_pTable->getAlterService(); if ( xAlterService.is() ) xAlterService->dropColumn(m_pTable,_sElementName); else OColumns_BASE::dropObject(_nPos,_sElementName); } else ::dbtools::throwGenericSQLException( DBA_RES( RID_STR_NO_COLUMN_DROP ), static_cast(static_cast(this)) ); } if ( m_pColFactoryImpl ) m_pColFactoryImpl->columnDropped(_sElementName); ::dbaccess::notifyDataSourceModified(m_xParent); } Reference< XInterface > SAL_CALL OColumns::getParent( ) { ::osl::MutexGuard aGuard(m_rMutex); return m_xParent; } void SAL_CALL OColumns::setParent( const Reference< XInterface >& _xParent ) { ::osl::MutexGuard aGuard(m_rMutex); m_xParent = _xParent; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */