/* -*- 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 "XFormsModelExport.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace com::sun::star; using namespace com::sun::star::uno; using namespace xmloff::token; using com::sun::star::beans::XPropertySet; using com::sun::star::beans::XPropertySetInfo; using com::sun::star::container::XIndexAccess; using com::sun::star::container::XNameAccess; using com::sun::star::container::XNameContainer; using com::sun::star::container::XEnumerationAccess; using com::sun::star::container::XEnumeration; using com::sun::star::xml::dom::XDocument; using com::sun::star::form::binding::XBindableValue; using com::sun::star::form::binding::XListEntrySink; using com::sun::star::form::submission::XSubmissionSupplier; using com::sun::star::beans::PropertyValue; using com::sun::star::xforms::XDataTypeRepository; using com::sun::star::xforms::XFormsSupplier; using com::sun::star::util::Duration; void exportXForms( SvXMLExport& rExport ) { Reference xSupplier( rExport.GetModel(), UNO_QUERY ); if( !xSupplier.is() ) return; Reference xForms = xSupplier->getXForms(); if( xForms.is() ) { const Sequence aNames = xForms->getElementNames(); for( const auto& rName : aNames ) { Reference xModel( xForms->getByName( rName ), UNO_QUERY ); exportXFormsModel( rExport, xModel ); } } } static void exportXFormsInstance( SvXMLExport&, const Sequence& ); static void exportXFormsBinding( SvXMLExport&, const Reference& ); static void exportXFormsSubmission( SvXMLExport&, const Reference& ); static void exportXFormsSchemas( SvXMLExport&, const Reference& ); typedef OUString (*convert_t)( const Any& ); namespace { struct ExportTable { const char* pPropertyName; sal_uInt16 const nNamespace; sal_uInt16 nToken; convert_t const aConverter; }; } static void lcl_export( const Reference& rPropertySet, SvXMLExport& rExport, const ExportTable* pTable ); #define TABLE_END { nullptr, 0, 0, nullptr } // any conversion functions static OUString xforms_string( const Any& ); static OUString xforms_bool( const Any& ); static OUString xforms_whitespace( const Any& ); template static OUString xforms_convert( const Any& ); template static OUString xforms_convertRef( const Any& ); static void xforms_formatDate( OUStringBuffer& aBuffer, const util::Date& aDate ); static void xforms_formatTime( OUStringBuffer& aBuffer, const css::util::Time& aTime ); static void xforms_formatDateTime( OUStringBuffer& aBuffer, const util::DateTime& aDateTime ); static void convertNumber(OUStringBuffer & b, sal_Int32 n) { b.append(n); } convert_t const xforms_int32 = &xforms_convert; convert_t const xforms_double = &xforms_convert; convert_t const xforms_dateTime = &xforms_convertRef; convert_t const xforms_date = &xforms_convertRef; convert_t const xforms_time = &xforms_convertRef; // other functions static OUString lcl_getXSDType( SvXMLExport const & rExport, const Reference& xType ); // the model const ExportTable aXFormsModelTable[] = { { "ID", XML_NAMESPACE_NONE, xmloff::token::XML_ID, xforms_string }, { "SchemaRef", XML_NAMESPACE_NONE, xmloff::token::XML_SCHEMA, xforms_string }, TABLE_END }; void exportXFormsModel( SvXMLExport& rExport, const Reference& xModelPropSet ) { // no model -> don't do anything! Reference xModel( xModelPropSet, UNO_QUERY ); if( ! xModel.is() || ! xModelPropSet.is() ) return; lcl_export( xModelPropSet, rExport, aXFormsModelTable ); SvXMLElementExport aModelElement( rExport, XML_NAMESPACE_XFORMS, XML_MODEL, true, true ); // instances Reference xInstances( xModel->getInstances(), UNO_QUERY_THROW); sal_Int32 nCount = xInstances->getCount(); sal_Int32 i = 0; for( i = 0; i < nCount; i++ ) { Sequence aInstance; xInstances->getByIndex( i ) >>= aInstance; exportXFormsInstance( rExport, aInstance ); } // bindings Reference xBindings( xModel->getBindings(), UNO_QUERY_THROW); nCount = xBindings->getCount(); for( i = 0; i < nCount; i++ ) { Reference aBinding( xBindings->getByIndex( i ), UNO_QUERY_THROW ); exportXFormsBinding( rExport, aBinding ); } // submissions Reference xSubmissions( xModel->getSubmissions(), UNO_QUERY_THROW ); nCount = xSubmissions->getCount(); for( i = 0; i < nCount; i++ ) { Reference xSubmission( xSubmissions->getByIndex( i ), UNO_QUERY_THROW ); exportXFormsSubmission( rExport, xSubmission ); } // schemas exportXFormsSchemas( rExport, xModel ); } // the instance void exportXFormsInstance( SvXMLExport& rExport, const Sequence& xInstance ) { OUString sId; OUString sURL; Reference xDoc; for( const auto& rProp : xInstance ) { OUString sName = rProp.Name; const Any& rAny = rProp.Value; if ( sName == "ID" ) rAny >>= sId; else if ( sName == "URL" ) rAny >>= sURL; else if ( sName == "Instance" ) rAny >>= xDoc; } if( !sId.isEmpty() ) rExport.AddAttribute( XML_NAMESPACE_NONE, XML_ID, sId ); if( !sURL.isEmpty() ) rExport.AddAttribute( XML_NAMESPACE_NONE, XML_SRC, sURL ); SvXMLElementExport aElem( rExport, XML_NAMESPACE_XFORMS, XML_INSTANCE, true, true ); rExport.IgnorableWhitespace(); if( xDoc.is() ) { exportDom( rExport, xDoc ); } } // the binding const ExportTable aXFormsBindingTable[] = { { "BindingID", XML_NAMESPACE_NONE, xmloff::token::XML_ID, xforms_string }, { "BindingExpression", XML_NAMESPACE_NONE, xmloff::token::XML_NODESET, xforms_string }, { "ReadonlyExpression", XML_NAMESPACE_NONE, xmloff::token::XML_READONLY, xforms_string }, { "RelevantExpression", XML_NAMESPACE_NONE, xmloff::token::XML_RELEVANT, xforms_string }, { "RequiredExpression", XML_NAMESPACE_NONE, xmloff::token::XML_REQUIRED, xforms_string }, { "ConstraintExpression", XML_NAMESPACE_NONE, xmloff::token::XML_CONSTRAINT, xforms_string }, { "CalculateExpression", XML_NAMESPACE_NONE, xmloff::token::XML_CALCULATE, xforms_string }, // type handled separately, for type name <-> XSD type conversion // { "Type", XML_NAMESPACE_NONE, xmloff::token::XML_TYPE, xforms_string }, TABLE_END }; void exportXFormsBinding( SvXMLExport& rExport, const Reference& xBinding ) { // name check; generate binding ID if necessary { OUString sName; xBinding->getPropertyValue( "BindingID" ) >>= sName; if( sName.isEmpty() ) { // if we don't have a name yet, generate one on the fly sal_Int64 nId = reinterpret_cast( xBinding.get() ); sName = "bind_" + OUString::number( nId , 16 ); xBinding->setPropertyValue( "BindingID", Any(sName)); } } lcl_export( xBinding, rExport, aXFormsBindingTable ); // handle type attribute { OUString sTypeName; xBinding->getPropertyValue( "Type" ) >>= sTypeName; try { // now get type, and determine whether it's a standard type. If // so, export the XSD name Reference xModel( xBinding->getPropertyValue( "Model" ), UNO_QUERY ); Reference xRepository( xModel.is() ? xModel->getDataTypeRepository() : Reference() ); if( xRepository.is() ) { Reference xDataType = xRepository->getDataType( sTypeName ); // if it's a basic data type, write out the XSD name // for the XSD type class bool bIsBasic = false; xDataType->getPropertyValue( "IsBasic" ) >>= bIsBasic; if( bIsBasic ) sTypeName = lcl_getXSDType( rExport, xDataType ); } } catch( Exception& ) { ; // ignore; just use typename } // now that we have the proper type name, write out the attribute if( !sTypeName.isEmpty() ) { rExport.AddAttribute( XML_NAMESPACE_NONE, XML_TYPE, sTypeName ); } } // we need to ensure all the namespaces in the binding will work correctly. // to do so, we will write out all missing namespace declaractions. const SvXMLNamespaceMap& rMap = rExport.GetNamespaceMap(); Reference xNamespaces( xBinding->getPropertyValue( "ModelNamespaces" ), UNO_QUERY); if( xNamespaces.is() ) { // iterate over Prefixes for this binding const Sequence aPrefixes = xNamespaces->getElementNames(); for( const OUString& rPrefix : aPrefixes ) { OUString sURI; xNamespaces->getByName( rPrefix ) >>= sURI; // check whether prefix/URI pair is in map; else write declaration // (we don't need to change the map, since this element has no // other content) sal_uInt16 nKey = rMap.GetKeyByPrefix( rPrefix ); if( nKey == XML_NAMESPACE_UNKNOWN || rMap.GetNameByKey( nKey ) != sURI ) { // add declaration if it doesn't already exist comphelper::AttributeList& rAttrList = rExport.GetAttrList(); OUString sName = "xmlns:" + rPrefix; sal_Int16 nFound = rAttrList.GetIndexByName(sName); // duplicate xmlns:script, http://openoffice.org/2000/script seen assert(nFound == -1 || rAttrList.getValueByIndex(nFound) == sURI); if (nFound != -1) continue; rAttrList.AddAttribute(sName, sURI); } } } SvXMLElementExport aElement( rExport, XML_NAMESPACE_XFORMS, XML_BIND, true, true ); } // the submission const ExportTable aXFormsSubmissionTable[] = { { "ID", XML_NAMESPACE_NONE, xmloff::token::XML_ID, xforms_string }, { "Bind", XML_NAMESPACE_NONE, xmloff::token::XML_BIND, xforms_string }, { "Ref", XML_NAMESPACE_NONE, xmloff::token::XML_REF, xforms_string }, { "Action", XML_NAMESPACE_NONE, xmloff::token::XML_ACTION, xforms_string }, { "Method", XML_NAMESPACE_NONE, xmloff::token::XML_METHOD, xforms_string }, { "Version", XML_NAMESPACE_NONE, xmloff::token::XML_VERSION, xforms_string }, { "Indent", XML_NAMESPACE_NONE, xmloff::token::XML_INDENT, xforms_bool }, { "MediaType", XML_NAMESPACE_NONE, xmloff::token::XML_MEDIATYPE, xforms_string }, { "Encoding", XML_NAMESPACE_NONE, xmloff::token::XML_ENCODING, xforms_string }, { "OmitXmlDeclaration", XML_NAMESPACE_NONE, xmloff::token::XML_OMIT_XML_DECLARATION, xforms_bool }, { "Standalone", XML_NAMESPACE_NONE, xmloff::token::XML_STANDALONE, xforms_bool }, { "CDataSectionElement", XML_NAMESPACE_NONE, xmloff::token::XML_CDATA_SECTION_ELEMENTS, xforms_string }, { "Replace", XML_NAMESPACE_NONE, xmloff::token::XML_REPLACE, xforms_string }, { "Separator", XML_NAMESPACE_NONE, xmloff::token::XML_SEPARATOR, xforms_string }, { "IncludeNamespacePrefixes", XML_NAMESPACE_NONE, xmloff::token::XML_INCLUDENAMESPACEPREFIXES, xforms_string }, TABLE_END }; void exportXFormsSubmission( SvXMLExport& rExport, const Reference& xSubmission ) { lcl_export( xSubmission, rExport, aXFormsSubmissionTable ); SvXMLElementExport aElement( rExport, XML_NAMESPACE_XFORMS, XML_SUBMISSION, true, true ); } // export data types as XSD schema const ExportTable aDataTypeFacetTable[] = { { "Length", XML_NAMESPACE_XSD, xmloff::token::XML_LENGTH, xforms_int32 }, { "MinLength", XML_NAMESPACE_XSD, xmloff::token::XML_MINLENGTH, xforms_int32 }, { "MaxLength", XML_NAMESPACE_XSD, xmloff::token::XML_MAXLENGTH, xforms_int32 }, { "MinInclusiveInt", XML_NAMESPACE_XSD, xmloff::token::XML_MININCLUSIVE, xforms_int32 }, { "MinExclusiveInt", XML_NAMESPACE_XSD, xmloff::token::XML_MINEXCLUSIVE, xforms_int32 }, { "MaxInclusiveInt", XML_NAMESPACE_XSD, xmloff::token::XML_MAXINCLUSIVE, xforms_int32 }, { "MaxExclusiveInt", XML_NAMESPACE_XSD, xmloff::token::XML_MAXEXCLUSIVE, xforms_int32 }, { "MinInclusiveDouble", XML_NAMESPACE_XSD, xmloff::token::XML_MININCLUSIVE, xforms_double }, { "MinExclusiveDouble", XML_NAMESPACE_XSD, xmloff::token::XML_MINEXCLUSIVE, xforms_double }, { "MaxInclusiveDouble", XML_NAMESPACE_XSD, xmloff::token::XML_MAXINCLUSIVE, xforms_double }, { "MaxExclusiveDouble", XML_NAMESPACE_XSD, xmloff::token::XML_MAXEXCLUSIVE, xforms_double }, { "MinInclusiveDate", XML_NAMESPACE_XSD, xmloff::token::XML_MININCLUSIVE, xforms_date }, { "MinExclusiveDate", XML_NAMESPACE_XSD, xmloff::token::XML_MINEXCLUSIVE, xforms_date }, { "MaxInclusiveDate", XML_NAMESPACE_XSD, xmloff::token::XML_MAXINCLUSIVE, xforms_date }, { "MaxExclusiveDate", XML_NAMESPACE_XSD, xmloff::token::XML_MAXEXCLUSIVE, xforms_date }, { "MinInclusiveTime", XML_NAMESPACE_XSD, xmloff::token::XML_MININCLUSIVE, xforms_time }, { "MinExclusiveTime", XML_NAMESPACE_XSD, xmloff::token::XML_MINEXCLUSIVE, xforms_time }, { "MaxInclusiveTime", XML_NAMESPACE_XSD, xmloff::token::XML_MAXINCLUSIVE, xforms_time }, { "MaxExclusiveTime", XML_NAMESPACE_XSD, xmloff::token::XML_MAXEXCLUSIVE, xforms_time }, { "MinInclusiveDateTime", XML_NAMESPACE_XSD, xmloff::token::XML_MININCLUSIVE, xforms_dateTime }, { "MinExclusiveDateTime", XML_NAMESPACE_XSD, xmloff::token::XML_MINEXCLUSIVE, xforms_dateTime }, { "MaxInclusiveDateTime", XML_NAMESPACE_XSD, xmloff::token::XML_MAXINCLUSIVE, xforms_dateTime }, { "MaxExclusiveDateTime", XML_NAMESPACE_XSD, xmloff::token::XML_MAXEXCLUSIVE, xforms_dateTime }, { "Pattern", XML_NAMESPACE_XSD, xmloff::token::XML_PATTERN, xforms_string }, // ??? XML_ENUMERATION, { "WhiteSpace", XML_NAMESPACE_XSD, xmloff::token::XML_WHITESPACE, xforms_whitespace }, { "TotalDigits", XML_NAMESPACE_XSD, xmloff::token::XML_TOTALDIGITS, xforms_int32 }, { "FractionDigits", XML_NAMESPACE_XSD, xmloff::token::XML_FRACTIONDIGITS, xforms_int32 }, TABLE_END }; // export facets through table; use the same table as lcl_export does static void lcl_exportDataTypeFacets( SvXMLExport& rExport, const Reference& rPropertySet, const ExportTable* pTable ) { Reference xInfo = rPropertySet->getPropertySetInfo(); for( const ExportTable* pCurrent = pTable; pCurrent->pPropertyName != nullptr; pCurrent++ ) { OUString sName( OUString::createFromAscii( pCurrent->pPropertyName ) ); if( xInfo->hasPropertyByName( sName ) ) { OUString sValue = (*pCurrent->aConverter)( rPropertySet->getPropertyValue( sName ) ); if( !sValue.isEmpty() ) { rExport.AddAttribute( XML_NAMESPACE_NONE, XML_VALUE, sValue ); SvXMLElementExport aFacet( rExport, pCurrent->nNamespace, static_cast( pCurrent->nToken ), true, true ); } } } } static OUString lcl_getXSDType( SvXMLExport const & rExport, const Reference& xType ) { // we use string as default... XMLTokenEnum eToken = XML_STRING; sal_uInt16 nDataTypeClass = 0; xType->getPropertyValue( "TypeClass" ) >>= nDataTypeClass; switch( nDataTypeClass ) { case css::xsd::DataTypeClass::STRING: eToken = XML_STRING; break; case css::xsd::DataTypeClass::anyURI: eToken = XML_ANYURI; break; case css::xsd::DataTypeClass::DECIMAL: eToken = XML_DECIMAL; break; case css::xsd::DataTypeClass::DOUBLE: eToken = XML_DOUBLE; break; case css::xsd::DataTypeClass::FLOAT: eToken = XML_FLOAT; break; case css::xsd::DataTypeClass::BOOLEAN: eToken = XML_BOOLEAN; break; case css::xsd::DataTypeClass::DATETIME: eToken = XML_DATETIME_XSD; break; case css::xsd::DataTypeClass::TIME: eToken = XML_TIME; break; case css::xsd::DataTypeClass::DATE: eToken = XML_DATE; break; case css::xsd::DataTypeClass::gYear: eToken = XML_YEAR; break; case css::xsd::DataTypeClass::gDay: eToken = XML_DAY; break; case css::xsd::DataTypeClass::gMonth: eToken = XML_MONTH; break; case css::xsd::DataTypeClass::DURATION: case css::xsd::DataTypeClass::gYearMonth: case css::xsd::DataTypeClass::gMonthDay: case css::xsd::DataTypeClass::hexBinary: case css::xsd::DataTypeClass::base64Binary: case css::xsd::DataTypeClass::QName: case css::xsd::DataTypeClass::NOTATION: default: OSL_FAIL( "unknown data type" ); } return rExport.GetNamespaceMap().GetQNameByKey( XML_NAMESPACE_XSD, GetXMLToken( eToken ) ); } static void lcl_exportDataType( SvXMLExport& rExport, const Reference& xType ) { // we do not need to export basic types; exit if we have one bool bIsBasic = false; xType->getPropertyValue( "IsBasic" ) >>= bIsBasic; if( bIsBasic ) return; // no basic type -> export // OUString sName; xType->getPropertyValue( "Name" ) >>= sName; rExport.AddAttribute( XML_NAMESPACE_NONE, XML_NAME, sName ); SvXMLElementExport aSimpleType( rExport, XML_NAMESPACE_XSD, XML_SIMPLETYPE, true, true ); // rExport.AddAttribute( XML_NAMESPACE_NONE, XML_BASE, lcl_getXSDType( rExport, xType ) ); SvXMLElementExport aRestriction( rExport, XML_NAMESPACE_XSD, XML_RESTRICTION, true, true ); // export facets lcl_exportDataTypeFacets( rExport, xType, aDataTypeFacetTable ); } void exportXFormsSchemas( SvXMLExport& rExport, const Reference& xModel ) { // TODO: for now, we'll fake this... { SvXMLElementExport aSchemaElem( rExport, XML_NAMESPACE_XSD, XML_SCHEMA, true, true ); // now get data type repository, and export Reference xTypes = xModel->getDataTypeRepository(); if( xTypes.is() ) { Reference xEnum = xTypes->createEnumeration(); SAL_WARN_IF( !xEnum.is(), "xmloff", "no enum?" ); while( xEnum->hasMoreElements() ) { Reference xType( xEnum->nextElement(), UNO_QUERY ); lcl_exportDataType( rExport, xType ); } } } // export other, 'foreign' schemas Reference xPropSet( xModel, UNO_QUERY ); if( xPropSet.is() ) { Reference xDocument( xPropSet->getPropertyValue( "ForeignSchema" ), UNO_QUERY ); if( xDocument.is() ) exportDom( rExport, xDocument ); } } // helper functions static void lcl_export( const Reference& rPropertySet, SvXMLExport& rExport, const ExportTable* pTable ) { for( const ExportTable* pCurrent = pTable; pCurrent->pPropertyName != nullptr; pCurrent++ ) { Any aAny = rPropertySet->getPropertyValue( OUString::createFromAscii( pCurrent->pPropertyName ) ); OUString sValue = (*pCurrent->aConverter)( aAny ); if( !sValue.isEmpty() ) rExport.AddAttribute( pCurrent->nNamespace, static_cast( pCurrent->nToken ), sValue ); } } // any conversion functions template OUString xforms_convert( const Any& rAny ) { OUStringBuffer aBuffer; T aData = T(); if( rAny >>= aData ) { FUNC( aBuffer, aData ); } return aBuffer.makeStringAndClear(); } template OUString xforms_convertRef( const Any& rAny ) { OUStringBuffer aBuffer; T aData; if( rAny >>= aData ) { FUNC( aBuffer, aData ); } return aBuffer.makeStringAndClear(); } OUString xforms_string( const Any& rAny ) { OUString aResult; rAny >>= aResult; return aResult; } OUString xforms_bool( const Any& rAny ) { bool bResult = bool(); if( rAny >>= bResult ) return GetXMLToken( bResult ? XML_TRUE : XML_FALSE ); OSL_FAIL( "expected boolean value" ); return OUString(); } void xforms_formatDate( OUStringBuffer& aBuffer, const util::Date& rDate ) { aBuffer.append( OUString::number( rDate.Year ) + "-" + OUString::number( rDate.Month ) + "-" + OUString::number( rDate.Day ) ); } void xforms_formatTime( OUStringBuffer& aBuffer, const css::util::Time& rTime ) { Duration aDuration; aDuration.Hours = rTime.Hours; aDuration.Minutes = rTime.Minutes; aDuration.Seconds = rTime.Seconds; aDuration.NanoSeconds = rTime.NanoSeconds; ::sax::Converter::convertDuration( aBuffer, aDuration ); } void xforms_formatDateTime( OUStringBuffer& aBuffer, const util::DateTime& aDateTime ) { ::sax::Converter::convertDateTime(aBuffer, aDateTime, nullptr); } OUString xforms_whitespace( const Any& rAny ) { OUString sResult; sal_uInt16 n = sal_uInt16(); if( rAny >>= n ) { switch( n ) { case css::xsd::WhiteSpaceTreatment::Preserve: sResult = GetXMLToken( XML_PRESERVE ); break; case css::xsd::WhiteSpaceTreatment::Replace: sResult = GetXMLToken( XML_REPLACE ); break; case css::xsd::WhiteSpaceTreatment::Collapse: sResult = GetXMLToken( XML_COLLAPSE ); break; } } return sResult; } /// return name of Binding static OUString lcl_getXFormsBindName( const Reference& xBinding ) { OUString sProp( "BindingID" ); OUString sReturn; if( xBinding.is() && xBinding->getPropertySetInfo()->hasPropertyByName( sProp ) ) { xBinding->getPropertyValue( sProp ) >>= sReturn; } return sReturn; } // return name of binding OUString getXFormsBindName( const Reference& xControl ) { Reference xBindable( xControl, UNO_QUERY ); return xBindable.is() ? lcl_getXFormsBindName( Reference( xBindable->getValueBinding(), UNO_QUERY )) : OUString(); } // return name of list binding OUString getXFormsListBindName( const Reference& xControl ) { Reference xListEntrySink( xControl, UNO_QUERY ); return xListEntrySink.is() ? lcl_getXFormsBindName( Reference( xListEntrySink->getListEntrySource(), UNO_QUERY ) ) : OUString(); } OUString getXFormsSubmissionName( const Reference& xBinding ) { OUString sReturn; Reference xSubmissionSupplier( xBinding, UNO_QUERY ); if( xSubmissionSupplier.is() ) { Reference xPropertySet( xSubmissionSupplier->getSubmission(), UNO_QUERY ); OUString sProp( "ID" ); if( xPropertySet.is() && xPropertySet->getPropertySetInfo()->hasPropertyByName( sProp ) ) { xPropertySet->getPropertyValue( sProp ) >>= sReturn; } } return sReturn; } void getXFormsSettings( const Reference< XNameAccess >& _rXForms, Sequence< PropertyValue >& _out_rSettings ) { _out_rSettings = Sequence< PropertyValue >(); OSL_PRECOND( _rXForms.is(), "getXFormsSettings: invalid XForms container!" ); if ( !_rXForms.is() ) return; try { // we want to export some special properties of our XForms models as config-item-map-named, // which implies we need a PropertyValue whose value is an XNameAccess, whose keys // are the names of the XForm models, and which in turn provides named sequences of // PropertyValues - which denote the actual property values of the given named model. const Sequence< OUString > aModelNames( _rXForms->getElementNames() ); Reference< XNameContainer > xModelSettings = document::NamedPropertyValues::create( comphelper::getProcessComponentContext() ); for ( auto const & modelName : aModelNames ) { Reference< XPropertySet > xModelProps( _rXForms->getByName( modelName ), UNO_QUERY_THROW ); static constexpr OUString sExternalData = u"ExternalData"_ustr; Sequence aModelSettings{ comphelper::makePropertyValue( sExternalData, xModelProps->getPropertyValue(sExternalData)) }; xModelSettings->insertByName( modelName, Any( aModelSettings ) ); } if ( xModelSettings->hasElements() ) { _out_rSettings = { comphelper::makePropertyValue("XFormModels", xModelSettings) }; } } catch( const Exception& ) { DBG_UNHANDLED_EXCEPTION("xmloff"); } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */