summaryrefslogtreecommitdiff
path: root/unotools/source/misc/atom.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'unotools/source/misc/atom.cxx')
-rw-r--r--unotools/source/misc/atom.cxx383
1 files changed, 383 insertions, 0 deletions
diff --git a/unotools/source/misc/atom.cxx b/unotools/source/misc/atom.cxx
new file mode 100644
index 000000000000..3d7ccd1dd10e
--- /dev/null
+++ b/unotools/source/misc/atom.cxx
@@ -0,0 +1,383 @@
+/*************************************************************************
+ *
+ * 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
+ * <http://www.openoffice.org/license.html>
+ * for a copy of the LGPLv3 License.
+ *
+ ************************************************************************/
+
+// MARKER(update_precomp.py): autogen include statement, do not remove
+#include "precompiled_unotools.hxx"
+
+#include <unotools/atom.hxx>
+
+using namespace utl;
+using namespace ::com::sun::star::uno;
+using namespace ::com::sun::star::util;
+#define NMSP_UTIL ::com::sun::star::util
+
+AtomProvider::AtomProvider()
+{
+ m_nAtoms = 1;
+}
+
+AtomProvider::~AtomProvider()
+{
+}
+
+int AtomProvider::getAtom( const ::rtl::OUString& rString, sal_Bool bCreate )
+{
+ ::std::hash_map< ::rtl::OUString, int, ::rtl::OUStringHash >::iterator it = m_aAtomMap.find( rString );
+ if( it != m_aAtomMap.end() )
+ return it->second;
+ if( ! bCreate )
+ return INVALID_ATOM;
+ m_aAtomMap[ rString ] = m_nAtoms;
+ m_aStringMap[ m_nAtoms ] = rString;
+ m_nAtoms++;
+ return m_nAtoms-1;
+}
+
+void AtomProvider::getAll( ::std::list< ::utl::AtomDescription >& atoms )
+{
+ atoms.clear();
+ ::std::hash_map< ::rtl::OUString, int, ::rtl::OUStringHash >::const_iterator it = m_aAtomMap.begin();
+
+ ::utl::AtomDescription aDesc;
+ while( it != m_aAtomMap.end() )
+ {
+ aDesc.atom = it->second;
+ aDesc.description = it->first;
+ atoms.push_back( aDesc );
+ ++it;
+ }
+}
+
+void AtomProvider::getRecent( int atom, ::std::list< ::utl::AtomDescription >& atoms )
+{
+ atoms.clear();
+
+ ::std::hash_map< ::rtl::OUString, int, ::rtl::OUStringHash >::const_iterator it = m_aAtomMap.begin();
+
+ ::utl::AtomDescription aDesc;
+ while( it != m_aAtomMap.end() )
+ {
+ if( it->second > atom )
+ {
+ aDesc.atom = it->second;
+ aDesc.description = it->first;
+ atoms.push_back( aDesc );
+ }
+ ++it;
+ }
+}
+
+const ::rtl::OUString& AtomProvider::getString( int nAtom ) const
+{
+ static ::rtl::OUString aEmpty;
+ ::std::hash_map< int, ::rtl::OUString, ::std::hash< int > >::const_iterator it = m_aStringMap.find( nAtom );
+
+ return it == m_aStringMap.end() ? aEmpty : it->second;
+}
+
+void AtomProvider::overrideAtom( int atom, const ::rtl::OUString& description )
+{
+ m_aAtomMap[ description ] = atom;
+ m_aStringMap[ atom ] = description;
+ if( m_nAtoms <= atom )
+ m_nAtoms=atom+1;
+}
+
+sal_Bool AtomProvider::hasAtom( int atom ) const
+{
+ return m_aStringMap.find( atom ) != m_aStringMap.end() ? sal_True : sal_False;
+}
+
+// -----------------------------------------------------------------------
+
+MultiAtomProvider::MultiAtomProvider()
+{
+}
+
+MultiAtomProvider::~MultiAtomProvider()
+{
+ for( ::std::hash_map< int, AtomProvider*, ::std::hash< int > >::iterator it = m_aAtomLists.begin(); it != m_aAtomLists.end(); ++it )
+ delete it->second;
+}
+
+
+sal_Bool MultiAtomProvider::insertAtomClass( int atomClass )
+{
+ ::std::hash_map< int, AtomProvider*, ::std::hash< int > >::iterator it =
+ m_aAtomLists.find( atomClass );
+ if( it != m_aAtomLists.end() )
+ return sal_False;
+ m_aAtomLists[ atomClass ] = new AtomProvider();
+ return sal_True;
+}
+
+int MultiAtomProvider::getAtom( int atomClass, const ::rtl::OUString& rString, sal_Bool bCreate )
+{
+ ::std::hash_map< int, AtomProvider*, ::std::hash< int > >::iterator it =
+ m_aAtomLists.find( atomClass );
+ if( it != m_aAtomLists.end() )
+ return it->second->getAtom( rString, bCreate );
+
+ if( bCreate )
+ {
+ AtomProvider* pNewClass;
+ m_aAtomLists[ atomClass ] = pNewClass = new AtomProvider();
+ return pNewClass->getAtom( rString, bCreate );
+ }
+ return INVALID_ATOM;
+}
+
+int MultiAtomProvider::getLastAtom( int atomClass ) const
+{
+ ::std::hash_map< int, AtomProvider*, ::std::hash< int > >::const_iterator it =
+ m_aAtomLists.find( atomClass );
+
+ return it != m_aAtomLists.end() ? it->second->getLastAtom() : INVALID_ATOM;
+}
+
+void MultiAtomProvider::getRecent( int atomClass, int atom, ::std::list< ::utl::AtomDescription >& atoms )
+{
+ ::std::hash_map< int, AtomProvider*, ::std::hash< int > >::const_iterator it =
+ m_aAtomLists.find( atomClass );
+ if( it != m_aAtomLists.end() )
+ it->second->getRecent( atom, atoms );
+ else
+ atoms.clear();
+}
+
+const ::rtl::OUString& MultiAtomProvider::getString( int atomClass, int atom ) const
+{
+ ::std::hash_map< int, AtomProvider*, ::std::hash< int > >::const_iterator it =
+ m_aAtomLists.find( atomClass );
+ if( it != m_aAtomLists.end() )
+ return it->second->getString( atom );
+
+ static ::rtl::OUString aEmpty;
+ return aEmpty;
+}
+
+sal_Bool MultiAtomProvider::hasAtom( int atomClass, int atom ) const
+{
+ ::std::hash_map< int, AtomProvider*, ::std::hash< int > >::const_iterator it = m_aAtomLists.find( atomClass );
+ return it != m_aAtomLists.end() ? it->second->hasAtom( atom ) : sal_False;
+}
+
+void MultiAtomProvider::getClass( int atomClass, ::std::list< ::utl::AtomDescription >& atoms) const
+{
+ ::std::hash_map< int, AtomProvider*, ::std::hash< int > >::const_iterator it = m_aAtomLists.find( atomClass );
+
+ if( it != m_aAtomLists.end() )
+ it->second->getAll( atoms );
+ else
+ atoms.clear();
+}
+
+void MultiAtomProvider::overrideAtom( int atomClass, int atom, const ::rtl::OUString& description )
+{
+ ::std::hash_map< int, AtomProvider*, ::std::hash< int > >::const_iterator it = m_aAtomLists.find( atomClass );
+ if( it == m_aAtomLists.end() )
+ m_aAtomLists[ atomClass ] = new AtomProvider();
+ m_aAtomLists[ atomClass ]->overrideAtom( atom, description );
+}
+
+// -----------------------------------------------------------------------
+
+AtomServer::AtomServer()
+{
+}
+
+AtomServer::~AtomServer()
+{
+}
+
+sal_Int32 AtomServer::getAtom( sal_Int32 atomClass, const ::rtl::OUString& description, sal_Bool create ) throw()
+{
+ ::osl::Guard< ::osl::Mutex > guard( m_aMutex );
+
+ return m_aProvider.getAtom( atomClass, description, create );
+}
+
+Sequence< Sequence< NMSP_UTIL::AtomDescription > > AtomServer::getClasses( const Sequence< sal_Int32 >& atomClasses ) throw()
+{
+ ::osl::Guard< ::osl::Mutex > guard( m_aMutex );
+
+ Sequence< Sequence< NMSP_UTIL::AtomDescription > > aRet( atomClasses.getLength() );
+ for( int i = 0; i < atomClasses.getLength(); i++ )
+ {
+ aRet.getArray()[i] = getClass( atomClasses.getConstArray()[i] );
+ }
+ return aRet;
+}
+
+Sequence< NMSP_UTIL::AtomDescription > AtomServer::getClass( sal_Int32 atomClass ) throw()
+{
+ ::osl::Guard< ::osl::Mutex > guard( m_aMutex );
+
+ ::std::list< ::utl::AtomDescription > atoms;
+ m_aProvider.getClass( atomClass, atoms );
+
+ Sequence< NMSP_UTIL::AtomDescription > aRet( atoms.size() );
+ for( int i = aRet.getLength()-1; i >= 0; i-- )
+ {
+ aRet.getArray()[i].atom = atoms.back().atom;
+ aRet.getArray()[i].description = atoms.back().description;
+ atoms.pop_back();
+ }
+
+ return aRet;
+}
+
+Sequence< NMSP_UTIL::AtomDescription > AtomServer::getRecentAtoms( sal_Int32 atomClass, sal_Int32 atom ) throw()
+{
+ ::osl::Guard< ::osl::Mutex > guard( m_aMutex );
+
+ ::std::list< ::utl::AtomDescription > atoms;
+ m_aProvider.getRecent( atomClass, atom, atoms );
+
+ Sequence< NMSP_UTIL::AtomDescription > aRet( atoms.size() );
+ for( int i = aRet.getLength()-1; i >= 0; i-- )
+ {
+ aRet.getArray()[i].atom = atoms.back().atom;
+ aRet.getArray()[i].description = atoms.back().description;
+ atoms.pop_back();
+ }
+
+ return aRet;
+}
+
+Sequence< ::rtl::OUString > AtomServer::getAtomDescriptions( const Sequence< AtomClassRequest >& atoms ) throw()
+{
+ ::osl::Guard< ::osl::Mutex > guard( m_aMutex );
+
+ int nStrings = 0, i;
+ for( i = 0; i < atoms.getLength(); i++ )
+ nStrings += atoms.getConstArray()[ i ].atoms.getLength();
+ Sequence< ::rtl::OUString > aRet( nStrings );
+ for( i = 0, nStrings = 0; i < atoms.getLength(); i++ )
+ {
+ const AtomClassRequest& rRequest = atoms.getConstArray()[i];
+ for( int n = 0; n < rRequest.atoms.getLength(); n++ )
+ aRet.getArray()[ nStrings++ ] = m_aProvider.getString( rRequest.atomClass, rRequest.atoms.getConstArray()[ n ] );
+ }
+ return aRet;
+}
+
+// -----------------------------------------------------------------------
+
+AtomClient::AtomClient( const Reference< XAtomServer >& xServer ) :
+ m_xServer( xServer )
+{
+}
+
+AtomClient::~AtomClient()
+{
+}
+
+int AtomClient::getAtom( int atomClass, const ::rtl::OUString& description, sal_Bool bCreate )
+{
+ int nAtom = m_aProvider.getAtom( atomClass, description, sal_False );
+ if( nAtom == INVALID_ATOM && bCreate )
+ {
+ try
+ {
+ nAtom = m_xServer->getAtom( atomClass, description, bCreate );
+ }
+ catch( RuntimeException& )
+ {
+ return INVALID_ATOM;
+ }
+ if( nAtom != INVALID_ATOM )
+ m_aProvider.overrideAtom( atomClass, nAtom, description );
+ }
+ return nAtom;
+}
+
+const ::rtl::OUString& AtomClient::getString( int atomClass, int atom )
+{
+ static ::rtl::OUString aEmpty;
+
+ if( ! m_aProvider.hasAtom( atomClass, atom ) )
+ {
+ Sequence< NMSP_UTIL::AtomDescription > aSeq;
+ try
+ {
+ aSeq = m_xServer->getRecentAtoms( atomClass, m_aProvider.getLastAtom( atomClass ) );
+ }
+ catch( RuntimeException& )
+ {
+ return aEmpty;
+ }
+ const NMSP_UTIL::AtomDescription* pDescriptions = aSeq.getConstArray();
+ for( int i = 0; i < aSeq.getLength(); i++ )
+ m_aProvider.overrideAtom( atomClass,
+ pDescriptions[i].atom,
+ pDescriptions[i].description
+ );
+
+ if( ! m_aProvider.hasAtom( atomClass, atom ) )
+ {
+ // holes may occur by the above procedure!
+ Sequence< AtomClassRequest > aReq( 1 );
+ aReq.getArray()[0].atomClass = atomClass;
+ aReq.getArray()[0].atoms.realloc( 1 );
+ aReq.getArray()[0].atoms.getArray()[0] = atom;
+ Sequence< ::rtl::OUString > aRet;
+ try
+ {
+ aRet = m_xServer->getAtomDescriptions( aReq );
+ }
+ catch( RuntimeException& )
+ {
+ return aEmpty;
+ }
+ if( aRet.getLength() == 1 )
+ m_aProvider.overrideAtom( atomClass, atom, aRet.getConstArray()[0] );
+ }
+ }
+ return m_aProvider.getString( atomClass, atom );
+}
+
+void AtomClient::updateAtomClasses( const Sequence< sal_Int32 >& atomClasses )
+{
+ Sequence< Sequence< NMSP_UTIL::AtomDescription > > aUpdate;
+ try
+ {
+ aUpdate = m_xServer->getClasses( atomClasses );
+ }
+ catch( RuntimeException& )
+ {
+ return;
+ }
+ for( int i = 0; i < atomClasses.getLength(); i++ )
+ {
+ int nClass = atomClasses.getConstArray()[i];
+ const Sequence< NMSP_UTIL::AtomDescription >& rClass = aUpdate.getConstArray()[i];
+ const NMSP_UTIL::AtomDescription* pDesc = rClass.getConstArray();
+ for( int n = 0; n < rClass.getLength(); n++, pDesc++ )
+ m_aProvider.overrideAtom( nClass, pDesc->atom, pDesc->description );
+ }
+}