/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_xmloff.hxx" #include #include #include #include #include "XMLPropertyBackpatcher.hxx" #include // XMLTextImportHelper partially implemented here using ::rtl::OUString; using ::std::vector; using ::std::map; using ::com::sun::star::uno::Reference; using ::com::sun::star::uno::Any; using ::com::sun::star::beans::XPropertySet; template XMLPropertyBackpatcher::XMLPropertyBackpatcher( const ::rtl::OUString& sPropName) : sPropertyName(sPropName) , bDefaultHandling(sal_False) , bPreserveProperty(sal_False) , sPreservePropertyName() { } template XMLPropertyBackpatcher::XMLPropertyBackpatcher( const OUString& sPropName, const OUString& sPreserveName, sal_Bool bDefault, A aDef) : sPropertyName(sPropName) , bDefaultHandling(bDefault) , bPreserveProperty(sPreserveName.getLength()>0) , sPreservePropertyName(sPreserveName) , aDefault(aDef) { } template XMLPropertyBackpatcher::XMLPropertyBackpatcher( const sal_Char* pPropName) : bDefaultHandling(sal_False) , bPreserveProperty(sal_False) { DBG_ASSERT(pPropName != NULL, "need property name"); sPropertyName = OUString::createFromAscii(pPropName); } template XMLPropertyBackpatcher::XMLPropertyBackpatcher( const sal_Char* pPropName, const sal_Char* pPreservePropName, sal_Bool bDefault, A aDef) : bDefaultHandling(bDefault) , bPreserveProperty(pPreservePropName != NULL) , aDefault(aDef) { DBG_ASSERT(pPropName != NULL, "need property name"); sPropertyName = OUString::createFromAscii(pPropName); if (pPreservePropName != NULL) { sPreservePropertyName = OUString::createFromAscii(pPreservePropName); } } template XMLPropertyBackpatcher::~XMLPropertyBackpatcher() { SetDefault(); } template void XMLPropertyBackpatcher::ResolveId( const OUString& sName, A aValue) { // insert ID into ID map aIDMap[sName] = aValue; // backpatch old references, if backpatch list exists if (aBackpatchListMap.count(sName)) { // aah, we have a backpatch list! BackpatchListType* pList = (BackpatchListType*)aBackpatchListMap[sName]; // a) remove list from list map aBackpatchListMap.erase(sName); // b) for every item, set SequenceNumber // (and preserve Property, if appropriate) Any aAny; aAny <<= aValue; if (bPreserveProperty) { // preserve version for(BackpatchListType::iterator aIter = pList->begin(); aIter != pList->end(); aIter++) { Reference xProp = (*aIter); Any aPres = xProp->getPropertyValue(sPreservePropertyName); xProp->setPropertyValue(sPropertyName, aAny); xProp->setPropertyValue(sPreservePropertyName, aPres); } } else { // without preserve for(BackpatchListType::iterator aIter = pList->begin(); aIter != pList->end(); aIter++) { (*aIter)->setPropertyValue(sPropertyName, aAny); } } // c) delete list delete pList; } // else: no backpatch list -> then we're finished } template void XMLPropertyBackpatcher::SetProperty( const Reference & xPropSet, const OUString& sName) { Reference xNonConstPropSet(xPropSet); SetProperty(xNonConstPropSet, sName); } template void XMLPropertyBackpatcher::SetProperty( Reference & xPropSet, const OUString& sName) { if (aIDMap.count(sName)) { // we know this ID -> set property Any aAny; aAny <<= aIDMap[sName]; xPropSet->setPropertyValue(sPropertyName, aAny); } else { // ID unknown -> into backpatch list for later fixup if (! aBackpatchListMap.count(sName)) { // create backpatch list for this name BackpatchListType* pTmp = new BackpatchListType() ; aBackpatchListMap[sName] = (void*)pTmp; } // insert footnote ((BackpatchListType*)aBackpatchListMap[sName])->push_back(xPropSet); } } template void XMLPropertyBackpatcher::SetDefault() { if (bDefaultHandling) { // not implemented yet } } // force instantiation of templates template class XMLPropertyBackpatcher; template class XMLPropertyBackpatcher; struct SAL_DLLPRIVATE XMLTextImportHelper::BackpatcherImpl { /// backpatcher for references to footnotes and endnotes ::std::auto_ptr< XMLPropertyBackpatcher > m_pFootnoteBackpatcher; /// backpatchers for references to sequences ::std::auto_ptr< XMLPropertyBackpatcher > m_pSequenceIdBackpatcher; ::std::auto_ptr< XMLPropertyBackpatcher< ::rtl::OUString> > m_pSequenceNameBackpatcher; }; ::boost::shared_ptr XMLTextImportHelper::MakeBackpatcherImpl() { // n.b.: the shared_ptr stores the dtor! return ::boost::shared_ptr(new BackpatcherImpl); } static ::rtl::OUString const& GetSequenceNumber() { static ::rtl::OUString s_SequenceNumber( RTL_CONSTASCII_USTRINGPARAM("SequenceNumber")); return s_SequenceNumber; } // // XMLTextImportHelper // // Code from XMLTextImportHelper using the XMLPropertyBackpatcher is // implemented here. The reason is that in the unxsols2 environment, // all templates are instatiated as file local (switch // -instances=static), and thus are not accessible from the outside. // // The previous solution was to force additional instantiation of // XMLPropertyBackpatcher in txtimp.cxx. This solution combines all // usage of the XMLPropertyBackpatcher in XMLPropertyBackpatcher.cxx // instead. // XMLPropertyBackpatcher& XMLTextImportHelper::GetFootnoteBP() { if (!m_pBackpatcherImpl->m_pFootnoteBackpatcher.get()) { m_pBackpatcherImpl->m_pFootnoteBackpatcher.reset( new XMLPropertyBackpatcher(GetSequenceNumber())); } return *m_pBackpatcherImpl->m_pFootnoteBackpatcher; } XMLPropertyBackpatcher& XMLTextImportHelper::GetSequenceIdBP() { if (!m_pBackpatcherImpl->m_pSequenceIdBackpatcher.get()) { m_pBackpatcherImpl->m_pSequenceIdBackpatcher.reset( new XMLPropertyBackpatcher(GetSequenceNumber())); } return *m_pBackpatcherImpl->m_pSequenceIdBackpatcher; } XMLPropertyBackpatcher& XMLTextImportHelper::GetSequenceNameBP() { static ::rtl::OUString s_SourceName( RTL_CONSTASCII_USTRINGPARAM("SourceName")); if (!m_pBackpatcherImpl->m_pSequenceNameBackpatcher.get()) { m_pBackpatcherImpl->m_pSequenceNameBackpatcher.reset( new XMLPropertyBackpatcher(s_SourceName)); } return *m_pBackpatcherImpl->m_pSequenceNameBackpatcher; } void XMLTextImportHelper::InsertFootnoteID( const OUString& sXMLId, sal_Int16 nAPIId) { GetFootnoteBP().ResolveId(sXMLId, nAPIId); } void XMLTextImportHelper::ProcessFootnoteReference( const OUString& sXMLId, const Reference & xPropSet) { GetFootnoteBP().SetProperty(xPropSet, sXMLId); } void XMLTextImportHelper::InsertSequenceID( const OUString& sXMLId, const OUString& sName, sal_Int16 nAPIId) { GetSequenceIdBP().ResolveId(sXMLId, nAPIId); GetSequenceNameBP().ResolveId(sXMLId, sName); } void XMLTextImportHelper::ProcessSequenceReference( const OUString& sXMLId, const Reference & xPropSet) { GetSequenceIdBP().SetProperty(xPropSet, sXMLId); GetSequenceNameBP().SetProperty(xPropSet, sXMLId); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */