2000-09-18 23:31:44 +00:00
|
|
|
/*************************************************************************
|
|
|
|
*
|
|
|
|
* $RCSfile: nmspmap.cxx,v $
|
|
|
|
*
|
2001-06-13 13:16:16 +00:00
|
|
|
* $Revision: 1.3 $
|
2000-09-18 23:31:44 +00:00
|
|
|
*
|
2001-06-13 13:16:16 +00:00
|
|
|
* last change: $Author: mtg $ $Date: 2001-06-13 14:16:16 $
|
2000-09-18 23:31:44 +00:00
|
|
|
*
|
|
|
|
* The Contents of this file are made available subject to the terms of
|
|
|
|
* either of the following licenses
|
|
|
|
*
|
|
|
|
* - GNU Lesser General Public License Version 2.1
|
|
|
|
* - Sun Industry Standards Source License Version 1.1
|
|
|
|
*
|
|
|
|
* Sun Microsystems Inc., October, 2000
|
|
|
|
*
|
|
|
|
* GNU Lesser General Public License Version 2.1
|
|
|
|
* =============================================
|
|
|
|
* Copyright 2000 by Sun Microsystems, Inc.
|
|
|
|
* 901 San Antonio Road, Palo Alto, CA 94303, USA
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License version 2.1, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This library 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 for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
|
|
|
|
* MA 02111-1307 USA
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* Sun Industry Standards Source License Version 1.1
|
|
|
|
* =================================================
|
|
|
|
* The contents of this file are subject to the Sun Industry Standards
|
|
|
|
* Source License Version 1.1 (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.openoffice.org/license.html.
|
|
|
|
*
|
|
|
|
* Software provided under this License is provided on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
|
|
|
|
* WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
|
|
|
|
* MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
|
|
|
|
* See the License for the specific provisions governing your rights and
|
|
|
|
* obligations concerning the Software.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is: Sun Microsystems, Inc.
|
|
|
|
*
|
|
|
|
* Copyright: 2000 by Sun Microsystems, Inc.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
*
|
2001-06-13 13:16:16 +00:00
|
|
|
* Contributor(s): Martin Gallwey (gallwey@sun.com)
|
2000-09-18 23:31:44 +00:00
|
|
|
*
|
|
|
|
*
|
|
|
|
************************************************************************/
|
|
|
|
|
|
|
|
#ifndef _TOOLS_DEBUG_HXX //autogen wg. DBG_ASSERT
|
|
|
|
#include <tools/debug.hxx>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef _RTL_USTRING_HXX_
|
|
|
|
#include <rtl/ustring.hxx>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef _RTL_USTRBUF_HXX_
|
|
|
|
#include <rtl/ustrbuf.hxx>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef _XMLKYWD_HXX
|
|
|
|
#include "xmlkywd.hxx"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef _XMLOFF_NMSPMAP_HXX
|
|
|
|
#include <nmspmap.hxx>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
using namespace rtl;
|
|
|
|
|
2001-06-13 13:16:16 +00:00
|
|
|
/* The basic idea of this class is that we have two two ways to search our
|
|
|
|
* data...by prefix and by key. We use an STL hash_map for fast prefix
|
|
|
|
* searching and an STL map for fast key searching.
|
|
|
|
*
|
|
|
|
* The references to an 'Index' refer to an earlier implementation of the
|
|
|
|
* name space map and remain to support code which uses these interfaces.
|
|
|
|
*
|
|
|
|
* In this implementation, key and index should always be the same number.
|
|
|
|
*
|
|
|
|
* Martin 13/06/01
|
|
|
|
*/
|
|
|
|
|
2000-09-18 23:31:44 +00:00
|
|
|
SvXMLNamespaceMap::SvXMLNamespaceMap() :
|
2001-04-18 15:14:13 +00:00
|
|
|
sXMLNS( OUString::createFromAscii(sXML_xmlns) )
|
2000-09-18 23:31:44 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
SvXMLNamespaceMap::SvXMLNamespaceMap( const SvXMLNamespaceMap& rMap ) :
|
2001-04-18 15:14:13 +00:00
|
|
|
sXMLNS( OUString::createFromAscii(sXML_xmlns) )
|
2000-09-18 23:31:44 +00:00
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
aNameHash = rMap.aNameHash;
|
|
|
|
aNameMap = rMap.aNameMap;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SvXMLNamespaceMap::~SvXMLNamespaceMap()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
int SvXMLNamespaceMap::operator ==( const SvXMLNamespaceMap& rCmp ) const
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
return static_cast < int > (aNameHash == rCmp.aNameHash);
|
|
|
|
}
|
2000-09-18 23:31:44 +00:00
|
|
|
|
2001-04-18 15:14:13 +00:00
|
|
|
void SvXMLNamespaceMap::_Add( const OUString& rPrefix, const OUString &rName, USHORT nKey )
|
|
|
|
{
|
|
|
|
NameSpaceEntry *pEntry = new NameSpaceEntry;
|
|
|
|
pEntry->sName = rName;
|
|
|
|
pEntry->nKey = nKey;
|
|
|
|
pEntry->sPrefix = rPrefix;
|
|
|
|
aNameHash[rPrefix] = pEntry;
|
|
|
|
aNameMap[nKey] = pEntry;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
USHORT SvXMLNamespaceMap::Add( const OUString& rPrefix, const OUString& rName,
|
|
|
|
USHORT nKey )
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
sal_Bool bRet = sal_False;
|
|
|
|
|
|
|
|
if( XML_NAMESPACE_UNKNOWN == nKey )
|
2000-09-18 23:31:44 +00:00
|
|
|
nKey = GetKeyByName( rName );
|
|
|
|
|
|
|
|
DBG_ASSERT( XML_NAMESPACE_NONE != nKey,
|
|
|
|
"SvXMLNamespaceMap::Add: invalid namespace key" );
|
|
|
|
if( XML_NAMESPACE_NONE==nKey )
|
|
|
|
return USHRT_MAX;
|
|
|
|
|
2001-04-18 15:14:13 +00:00
|
|
|
if (!(aNameHash.count ( rPrefix ) ) )
|
2000-09-18 23:31:44 +00:00
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
_Add( rPrefix, rName, nKey );
|
|
|
|
bRet = sal_True;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
2001-04-18 15:14:13 +00:00
|
|
|
return bRet;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOL SvXMLNamespaceMap::AddAtIndex( USHORT nIdx, const OUString& rPrefix,
|
|
|
|
const OUString& rName, USHORT nKey )
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
sal_Bool bRet = sal_False;
|
|
|
|
|
|
|
|
if( XML_NAMESPACE_UNKNOWN == nKey )
|
2000-09-18 23:31:44 +00:00
|
|
|
nKey = GetKeyByName( rName );
|
|
|
|
|
|
|
|
DBG_ASSERT( XML_NAMESPACE_NONE != nKey,
|
|
|
|
"SvXMLNamespaceMap::AddAtIndex: invalid namespace key" );
|
2001-04-18 15:14:13 +00:00
|
|
|
if( XML_NAMESPACE_NONE != nKey && ! ( aNameHash.count ( rPrefix ) ) )
|
2000-09-18 23:31:44 +00:00
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
_Add( rPrefix, rName, nKey );
|
|
|
|
bRet = sal_True;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
2001-04-18 15:14:13 +00:00
|
|
|
return bRet;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOL SvXMLNamespaceMap::AddAtIndex( USHORT nIdx, const sal_Char *pPrefix,
|
|
|
|
const sal_Char *pName, USHORT nKey )
|
|
|
|
{
|
|
|
|
OUString sPrefix( OUString::createFromAscii(pPrefix) );
|
|
|
|
OUString sName( OUString::createFromAscii(pName) );
|
|
|
|
|
|
|
|
return AddAtIndex( nIdx, sPrefix, sName, nKey );
|
|
|
|
}
|
|
|
|
|
|
|
|
USHORT SvXMLNamespaceMap::GetIndexByKey( USHORT nKey ) const
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
return nKey;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
USHORT SvXMLNamespaceMap::GetIndexByPrefix( const OUString& rPrefix ) const
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
NameSpaceHash::const_iterator aIter = aNameHash.find(rPrefix);
|
|
|
|
return (aIter != aNameHash.end()) ? (*aIter).second->nKey : USHRT_MAX;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
USHORT SvXMLNamespaceMap::GetKeyByName( const OUString& rName ) const
|
|
|
|
{
|
|
|
|
USHORT nKey = XML_NAMESPACE_UNKNOWN;
|
2001-04-18 15:14:13 +00:00
|
|
|
NameSpaceHash::const_iterator aIter = aNameHash.begin(), aEnd = aNameHash.end();
|
|
|
|
while (aIter != aEnd )
|
2000-09-18 23:31:44 +00:00
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
if ((*aIter).second->sName == rName)
|
2000-09-18 23:31:44 +00:00
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
nKey = (*aIter).second->nKey;
|
2000-09-18 23:31:44 +00:00
|
|
|
break;
|
|
|
|
}
|
2001-04-18 15:14:13 +00:00
|
|
|
aIter++;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
return nKey;
|
|
|
|
}
|
|
|
|
|
|
|
|
const OUString& SvXMLNamespaceMap::GetPrefixByIndex( USHORT nIdx ) const
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
NameSpaceMap::const_iterator aIter = aNameMap.find (nIdx);
|
|
|
|
return (aIter != aNameMap.end()) ? (*aIter).second->sPrefix : sEmpty;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const OUString& SvXMLNamespaceMap::GetNameByIndex( USHORT nIdx ) const
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
NameSpaceMap::const_iterator aIter = aNameMap.find (nIdx);
|
|
|
|
return (aIter != aNameMap.end()) ? (*aIter).second->sName : sEmpty;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
USHORT SvXMLNamespaceMap::GetKeyByIndex( USHORT nIdx ) const
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
return nIdx;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
OUString SvXMLNamespaceMap::GetAttrNameByIndex( USHORT nIdx ) const
|
|
|
|
{
|
|
|
|
OUStringBuffer sAttrName;
|
2001-04-18 15:14:13 +00:00
|
|
|
NameSpaceMap::const_iterator aIter = aNameMap.find ( nIdx );
|
|
|
|
if (aIter != aNameMap.end())
|
2000-09-18 23:31:44 +00:00
|
|
|
{
|
|
|
|
sAttrName.append( sXMLNS );
|
|
|
|
sAttrName.append( sal_Unicode(':') );
|
2001-04-18 15:14:13 +00:00
|
|
|
sAttrName.append( (*aIter).second->sPrefix);
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
return sAttrName.makeStringAndClear();
|
|
|
|
}
|
|
|
|
|
|
|
|
OUString SvXMLNamespaceMap::GetQNameByIndex( USHORT nIdx,
|
|
|
|
const OUString& rLocalName ) const
|
|
|
|
{
|
2001-06-13 13:16:16 +00:00
|
|
|
// We always want to return at least the rLocalName...
|
2000-09-18 23:31:44 +00:00
|
|
|
OUStringBuffer sQName;
|
2001-06-13 13:16:16 +00:00
|
|
|
|
|
|
|
switch ( nIdx )
|
2000-09-18 23:31:44 +00:00
|
|
|
{
|
2001-06-13 13:16:16 +00:00
|
|
|
case XML_NAMESPACE_UNKNOWN:
|
|
|
|
// ...if it's a completely unknown namespace, assert and return the local name
|
|
|
|
DBG_ASSERT( sal_False, "SvXMLNamespaceMap::GetQNameByIndex: invalid namespace key" );
|
|
|
|
case XML_NAMESPACE_NONE:
|
|
|
|
// ...if there isn't one, return the local name
|
|
|
|
sQName.append ( rLocalName );
|
|
|
|
break;
|
|
|
|
case XML_NAMESPACE_XMLNS:
|
|
|
|
{
|
|
|
|
// ...if it's in the xmlns namespace, make the prefix
|
|
|
|
sQName.append ( sXMLNS );
|
|
|
|
sQName.append ( sal_Unicode(':') );
|
|
|
|
sQName.append ( rLocalName );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
NameSpaceMap::const_iterator aIter = aNameMap.find ( nIdx );
|
|
|
|
if ( aIter != aNameMap.end() )
|
|
|
|
{
|
|
|
|
// ...if it's in our map, make the prefix
|
|
|
|
sQName.append ( (*aIter).second->sPrefix);
|
|
|
|
sQName.append ( sal_Unicode(':') );
|
|
|
|
sQName.append ( rLocalName );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// ... if isn't, this is a Bad Thing, assert and return the local name
|
|
|
|
DBG_ASSERT( sal_False, "SvXMLNamespaceMap::GetQNameByIndex: invalid namespace key" );
|
|
|
|
sQName.append ( rLocalName );
|
|
|
|
}
|
|
|
|
}
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
return sQName.makeStringAndClear();;
|
|
|
|
}
|
|
|
|
|
|
|
|
USHORT SvXMLNamespaceMap::GetKeyByAttrName( const OUString& rAttrName,
|
|
|
|
OUString *pPrefix,
|
|
|
|
OUString *pLocalName,
|
|
|
|
OUString *pNamespace,
|
|
|
|
USHORT nIdxGuess ) const
|
|
|
|
{
|
|
|
|
USHORT nKey = XML_NAMESPACE_UNKNOWN;
|
|
|
|
|
|
|
|
sal_Int32 nColonPos = rAttrName.indexOf( sal_Unicode(':') );
|
|
|
|
if( -1L != nColonPos )
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
OUString aPrefix( rAttrName.copy( 0L, nColonPos ) );
|
2000-09-18 23:31:44 +00:00
|
|
|
if( pPrefix )
|
2001-04-18 15:14:13 +00:00
|
|
|
*pPrefix = aPrefix;
|
2000-09-18 23:31:44 +00:00
|
|
|
if( pLocalName )
|
|
|
|
*pLocalName = rAttrName.copy( nColonPos + 1L );
|
|
|
|
|
2001-04-18 15:14:13 +00:00
|
|
|
NameSpaceHash::const_iterator aIter = aNameHash.find( aPrefix );
|
|
|
|
if ( aIter != aNameHash.end() )
|
2000-09-18 23:31:44 +00:00
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
nKey = (*aIter).second->nKey;
|
|
|
|
if ( pNamespace )
|
|
|
|
*pNamespace = (*aIter).second->sName;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
2001-06-13 13:16:16 +00:00
|
|
|
else if ( aPrefix == sXMLNS )
|
|
|
|
nKey = XML_NAMESPACE_XMLNS;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nKey = XML_NAMESPACE_NONE;
|
|
|
|
if( pPrefix )
|
|
|
|
*pPrefix = sEmpty;
|
|
|
|
if( pLocalName )
|
|
|
|
*pLocalName = rAttrName;
|
|
|
|
if( pNamespace )
|
|
|
|
*pNamespace = sEmpty;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nKey;
|
|
|
|
}
|
|
|
|
|
|
|
|
USHORT SvXMLNamespaceMap::GetFirstIndex() const
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
return aNameMap.empty() ? USHRT_MAX : (*aNameMap.begin()).second->nKey;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
USHORT SvXMLNamespaceMap::GetNextIndex( USHORT nOldIdx ) const
|
|
|
|
{
|
2001-04-18 15:14:13 +00:00
|
|
|
NameSpaceMap::const_iterator aIter = aNameMap.find ( nOldIdx );
|
|
|
|
return (++aIter == aNameMap.end()) ? USHRT_MAX : (*aIter).second->nKey;
|
2000-09-18 23:31:44 +00:00
|
|
|
}
|