/************************************************************** * * 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_extensions.hxx" #include "xsdvalidationhelper.hxx" #include "xsddatatypes.hxx" #include "formstrings.hxx" /** === begin UNO includes === **/ #include <com/sun/star/lang/XServiceInfo.hpp> #include <com/sun/star/xsd/DataTypeClass.hpp> #include <com/sun/star/util/NumberFormat.hpp> #include <com/sun/star/util/XNumberFormatTypes.hpp> #include <com/sun/star/util/XNumberFormatsSupplier.hpp> #include <com/sun/star/xforms/XDataTypeRepository.hpp> /** === end UNO includes === **/ #include <unotools/syslocale.hxx> #include <tools/diagnose_ex.h> //........................................................................ namespace pcr { //........................................................................ using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::xsd; using namespace ::com::sun::star::util; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::xforms; namespace NumberFormat = ::com::sun::star::util::NumberFormat; //==================================================================== //= XSDValidationHelper //==================================================================== //-------------------------------------------------------------------- XSDValidationHelper::XSDValidationHelper( ::osl::Mutex& _rMutex, const Reference< XPropertySet >& _rxIntrospectee, const Reference< frame::XModel >& _rxContextDocument ) :EFormsHelper( _rMutex, _rxIntrospectee, _rxContextDocument ) ,m_bInspectingFormattedField( false ) { try { Reference< XPropertySetInfo > xPSI; Reference< XServiceInfo > xSI( _rxIntrospectee, UNO_QUERY ); if ( m_xControlModel.is() ) xPSI = m_xControlModel->getPropertySetInfo(); if ( xPSI.is() && xPSI->hasPropertyByName( PROPERTY_FORMATKEY ) && xPSI->hasPropertyByName( PROPERTY_FORMATSSUPPLIER ) && xSI.is() && xSI->supportsService( SERVICE_COMPONENT_FORMATTEDFIELD ) ) m_bInspectingFormattedField = true; } catch( const Exception& ) { OSL_ENSURE( sal_False, "XSDValidationHelper::XSDValidationHelper: caught an exception while examining the introspectee!" ); } } //-------------------------------------------------------------------- void XSDValidationHelper::getAvailableDataTypeNames( ::std::vector< ::rtl::OUString >& /* [out] */ _rNames ) const SAL_THROW(()) { _rNames.resize( 0 ); try { Reference< XDataTypeRepository > xRepository = getDataTypeRepository(); Sequence< ::rtl::OUString > aElements; if ( xRepository.is() ) aElements = xRepository->getElementNames(); _rNames.resize( aElements.getLength() ); ::std::copy( aElements.getConstArray(), aElements.getConstArray() + aElements.getLength(), _rNames.begin() ); } catch( const Exception& ) { OSL_ENSURE( sal_False, "XSDValidationHelper::getAvailableDataTypeNames: caught an exception!" ); } } //-------------------------------------------------------------------- Reference< XDataTypeRepository > XSDValidationHelper::getDataTypeRepository() const SAL_THROW((Exception)) { Reference< XDataTypeRepository > xRepository; Reference< xforms::XModel > xModel( getCurrentFormModel( ) ); if ( xModel.is() ) xRepository = xModel->getDataTypeRepository(); return xRepository; } //-------------------------------------------------------------------- Reference< XDataTypeRepository > XSDValidationHelper::getDataTypeRepository( const ::rtl::OUString& _rModelName ) const SAL_THROW((Exception)) { Reference< XDataTypeRepository > xRepository; Reference< xforms::XModel > xModel( getFormModelByName( _rModelName ) ); if ( xModel.is() ) xRepository = xModel->getDataTypeRepository(); return xRepository; } //-------------------------------------------------------------------- Reference< XDataType > XSDValidationHelper::getDataType( const ::rtl::OUString& _rName ) const SAL_THROW((Exception)) { Reference< XDataType > xDataType; if ( _rName.getLength() ) { Reference< XDataTypeRepository > xRepository = getDataTypeRepository(); if ( xRepository.is() ) xDataType = xRepository->getDataType( _rName ); } return xDataType; } //-------------------------------------------------------------------- ::rtl::OUString XSDValidationHelper::getValidatingDataTypeName( ) const SAL_THROW(()) { ::rtl::OUString sDataTypeName; try { Reference< XPropertySet > xBinding( getCurrentBinding() ); // it's allowed here to not (yet) have a binding if ( xBinding.is() ) { OSL_VERIFY( xBinding->getPropertyValue( PROPERTY_XSD_DATA_TYPE ) >>= sDataTypeName ); } } catch( const Exception& ) { OSL_ENSURE( sal_False, "XSDValidationHelper::getValidatingDataTypeName: caught an exception!" ); } return sDataTypeName; } //-------------------------------------------------------------------- ::rtl::Reference< XSDDataType > XSDValidationHelper::getDataTypeByName( const ::rtl::OUString& _rName ) const SAL_THROW(()) { ::rtl::Reference< XSDDataType > pReturn; try { Reference< XDataType > xValidatedAgainst; if ( _rName.getLength() ) xValidatedAgainst = getDataType( _rName ); if ( xValidatedAgainst.is() ) pReturn = new XSDDataType( xValidatedAgainst ); } catch( const Exception& ) { OSL_ENSURE( sal_False, "XSDValidationHelper::getDataTypeByName: caught an exception!" ); } return pReturn; } //-------------------------------------------------------------------- ::rtl::Reference< XSDDataType > XSDValidationHelper::getValidatingDataType( ) const SAL_THROW(()) { return getDataTypeByName( getValidatingDataTypeName() ); } //-------------------------------------------------------------------- bool XSDValidationHelper::cloneDataType( const ::rtl::Reference< XSDDataType >& _pDataType, const ::rtl::OUString& _rNewName ) const SAL_THROW(()) { OSL_ENSURE( _pDataType.is(), "XSDValidationHelper::removeDataTypeFromRepository: invalid data type!" ); if ( !_pDataType.is() ) return false; try { Reference< XDataTypeRepository > xRepository( getDataTypeRepository() ); OSL_ENSURE( xRepository.is(), "XSDValidationHelper::removeDataTypeFromRepository: invalid data type repository!" ); if ( !xRepository.is() ) return false; Reference< XDataType > xDataType( _pDataType->getUnoDataType() ); OSL_ENSURE( xDataType.is(), "XSDValidationHelper::removeDataTypeFromRepository: invalid data type (II)!" ); if ( !xDataType.is() ) return false; xRepository->cloneDataType( xDataType->getName(), _rNewName ); } catch( const Exception& ) { OSL_ENSURE( sal_False, "XSDValidationHelper::cloneDataType: caught an exception!" ); } return true; } //-------------------------------------------------------------------- bool XSDValidationHelper::removeDataTypeFromRepository( const ::rtl::OUString& _rName ) const SAL_THROW(()) { try { Reference< XDataTypeRepository > xRepository( getDataTypeRepository() ); OSL_ENSURE( xRepository.is(), "XSDValidationHelper::removeDataTypeFromRepository: invalid data type repository!" ); if ( !xRepository.is() ) return false; if ( !xRepository->hasByName( _rName ) ) { OSL_ENSURE( sal_False, "XSDValidationHelper::removeDataTypeFromRepository: invalid repository and/or data type!" ); return false; } xRepository->revokeDataType( _rName ); } catch( const Exception& ) { OSL_ENSURE( sal_False, "XSDValidationHelper::removeDataTypeFromRepository: caught an exception!" ); return false; } return true; } //-------------------------------------------------------------------- void XSDValidationHelper::setValidatingDataTypeByName( const ::rtl::OUString& _rName ) const SAL_THROW(()) { try { Reference< XPropertySet > xBinding( getCurrentBinding() ); OSL_ENSURE( xBinding.is(), "XSDValidationHelper::setValidatingDataTypeByName: no active binding - how this?" ); if ( xBinding.is() ) { // get the old data type - this is necessary for notifying property changes ::rtl::OUString sOldDataTypeName; OSL_VERIFY( xBinding->getPropertyValue( PROPERTY_XSD_DATA_TYPE ) >>= sOldDataTypeName ); Reference< XPropertySet > xOldType; try { xOldType = xOldType.query( getDataType( sOldDataTypeName ) ); } catch( const Exception& ) { } // set the new data type name xBinding->setPropertyValue( PROPERTY_XSD_DATA_TYPE, makeAny( _rName ) ); // retrieve the new data type object Reference< XPropertySet > xNewType( getDataType( _rName ), UNO_QUERY ); // fire any changes in the properties which result from this new type std::set< ::rtl::OUString > aFilter; aFilter.insert( PROPERTY_NAME ); firePropertyChanges( xOldType, xNewType, aFilter ); // fire the change in the Data Type property ::rtl::OUString sNewDataTypeName; OSL_VERIFY( xBinding->getPropertyValue( PROPERTY_XSD_DATA_TYPE ) >>= sNewDataTypeName ); firePropertyChange( PROPERTY_XSD_DATA_TYPE, makeAny( sOldDataTypeName ), makeAny( sNewDataTypeName ) ); } } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION(); } } //-------------------------------------------------------------------- void XSDValidationHelper::copyDataType( const ::rtl::OUString& _rFromModel, const ::rtl::OUString& _rToModel, const ::rtl::OUString& _rDataTypeName ) const SAL_THROW(()) { if ( _rFromModel == _rToModel ) // nothing to do (me thinks) return; try { Reference< XDataTypeRepository > xFromRepository, xToRepository; if ( _rFromModel.getLength() ) xFromRepository = getDataTypeRepository( _rFromModel ); if ( _rToModel.getLength() ) xToRepository = getDataTypeRepository( _rToModel ); if ( !xFromRepository.is() || !xToRepository.is() ) return; if ( !xFromRepository->hasByName( _rDataTypeName ) || xToRepository->hasByName( _rDataTypeName ) ) // not existent in the source, or already existent (by name) in the destination return; // determine the built-in type belonging to the source type ::rtl::Reference< XSDDataType > pSourceType = new XSDDataType( xFromRepository->getDataType( _rDataTypeName ) ); ::rtl::OUString sTargetBaseType = getBasicTypeNameForClass( pSourceType->classify(), xToRepository ); // create the target type Reference< XDataType > xTargetType = xToRepository->cloneDataType( sTargetBaseType, _rDataTypeName ); ::rtl::Reference< XSDDataType > pTargetType = new XSDDataType( xTargetType ); // copy the facets pTargetType->copyFacetsFrom( pSourceType ); } catch( const Exception& ) { OSL_ENSURE( sal_False, "XSDValidationHelper::copyDataType: caught an exception!" ); } } //-------------------------------------------------------------------- void XSDValidationHelper::findDefaultFormatForIntrospectee() SAL_THROW(()) { try { ::rtl::Reference< XSDDataType > xDataType = getValidatingDataType(); if ( xDataType.is() ) { // find a NumberFormat type corresponding to the DataTypeClass sal_Int16 nNumberFormatType = NumberFormat::NUMBER; switch ( xDataType->classify() ) { case DataTypeClass::DATETIME: nNumberFormatType = NumberFormat::DATETIME; break; case DataTypeClass::DATE: nNumberFormatType = NumberFormat::DATE; break; case DataTypeClass::TIME: nNumberFormatType = NumberFormat::TIME; break; case DataTypeClass::STRING: case DataTypeClass::anyURI: case DataTypeClass::QName: case DataTypeClass::NOTATION: nNumberFormatType = NumberFormat::TEXT; break; } // get the number formatter from the introspectee Reference< XNumberFormatsSupplier > xSupplier; Reference< XNumberFormatTypes > xFormatTypes; OSL_VERIFY( m_xControlModel->getPropertyValue( PROPERTY_FORMATSSUPPLIER ) >>= xSupplier ); if ( xSupplier.is() ) xFormatTypes = xFormatTypes.query( xSupplier->getNumberFormats() ); OSL_ENSURE( xFormatTypes.is(), "XSDValidationHelper::findDefaultFormatForIntrospectee: no number formats for the introspectee!" ); if ( !xFormatTypes.is() ) return; // and the standard format for the given NumberFormat type sal_Int32 nDesiredFormat = xFormatTypes->getStandardFormat( nNumberFormatType, SvtSysLocale().GetLocaleData().getLocale() ); // set this at the introspectee m_xControlModel->setPropertyValue( PROPERTY_FORMATKEY, makeAny( nDesiredFormat ) ); } } catch( const Exception& ) { OSL_ENSURE( sal_False, "XSDValidationHelper::findDefaultFormatForIntrospectee: caught an exception!" ); } } //-------------------------------------------------------------------- ::rtl::OUString XSDValidationHelper::getBasicTypeNameForClass( sal_Int16 _nClass ) const SAL_THROW(()) { return getBasicTypeNameForClass( _nClass, getDataTypeRepository() ); } //-------------------------------------------------------------------- ::rtl::OUString XSDValidationHelper::getBasicTypeNameForClass( sal_Int16 _nClass, Reference< XDataTypeRepository > _rxRepository ) const SAL_THROW(()) { ::rtl::OUString sReturn; OSL_ENSURE( _rxRepository.is(), "XSDValidationHelper::getBasicTypeNameForClass: invalid repository!" ); if ( !_rxRepository.is() ) return sReturn; try { Reference< XDataType > xDataType = _rxRepository->getBasicDataType( _nClass ); OSL_ENSURE( xDataType.is(), "XSDValidationHelper::getBasicTypeNameForClass: invalid data type returned!" ); if ( xDataType.is() ) sReturn = xDataType->getName(); } catch( const Exception& ) { OSL_ENSURE( sal_False, "XSDValidationHelper::getBasicTypeNameForClass: caught an exception!" ); } return sReturn; } //........................................................................ } // namespace pcr //........................................................................