1*cdf0e10cSrcweir /*************************************************************************
2*cdf0e10cSrcweir  *
3*cdf0e10cSrcweir  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4*cdf0e10cSrcweir  *
5*cdf0e10cSrcweir  * Copyright 2000, 2010 Oracle and/or its affiliates.
6*cdf0e10cSrcweir  *
7*cdf0e10cSrcweir  * OpenOffice.org - a multi-platform office productivity suite
8*cdf0e10cSrcweir  *
9*cdf0e10cSrcweir  * This file is part of OpenOffice.org.
10*cdf0e10cSrcweir  *
11*cdf0e10cSrcweir  * OpenOffice.org is free software: you can redistribute it and/or modify
12*cdf0e10cSrcweir  * it under the terms of the GNU Lesser General Public License version 3
13*cdf0e10cSrcweir  * only, as published by the Free Software Foundation.
14*cdf0e10cSrcweir  *
15*cdf0e10cSrcweir  * OpenOffice.org is distributed in the hope that it will be useful,
16*cdf0e10cSrcweir  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17*cdf0e10cSrcweir  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18*cdf0e10cSrcweir  * GNU Lesser General Public License version 3 for more details
19*cdf0e10cSrcweir  * (a copy is included in the LICENSE file that accompanied this code).
20*cdf0e10cSrcweir  *
21*cdf0e10cSrcweir  * You should have received a copy of the GNU Lesser General Public License
22*cdf0e10cSrcweir  * version 3 along with OpenOffice.org.  If not, see
23*cdf0e10cSrcweir  * <http://www.openoffice.org/license.html>
24*cdf0e10cSrcweir  * for a copy of the LGPLv3 License.
25*cdf0e10cSrcweir  *
26*cdf0e10cSrcweir  ************************************************************************/
27*cdf0e10cSrcweir 
28*cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
29*cdf0e10cSrcweir #include "precompiled_xmloff.hxx"
30*cdf0e10cSrcweir #include "TokenContext.hxx"
31*cdf0e10cSrcweir #include <xmloff/xmltkmap.hxx>
32*cdf0e10cSrcweir #include <xmloff/xmlimp.hxx>
33*cdf0e10cSrcweir #include <xmloff/nmspmap.hxx>
34*cdf0e10cSrcweir #include "xmloff/xmlerror.hxx"
35*cdf0e10cSrcweir 
36*cdf0e10cSrcweir #include <tools/debug.hxx>
37*cdf0e10cSrcweir 
38*cdf0e10cSrcweir using rtl::OUString;
39*cdf0e10cSrcweir using com::sun::star::uno::Reference;
40*cdf0e10cSrcweir using com::sun::star::xml::sax::XAttributeList;
41*cdf0e10cSrcweir 
42*cdf0e10cSrcweir 
43*cdf0e10cSrcweir struct SvXMLTokenMapEntry aEmptyMap[1] =
44*cdf0e10cSrcweir {
45*cdf0e10cSrcweir     XML_TOKEN_MAP_END
46*cdf0e10cSrcweir };
47*cdf0e10cSrcweir 
48*cdf0e10cSrcweir 
49*cdf0e10cSrcweir TokenContext::TokenContext( SvXMLImport& rImport,
50*cdf0e10cSrcweir                             sal_uInt16 nPrefix,
51*cdf0e10cSrcweir                             const OUString& rLocalName,
52*cdf0e10cSrcweir                             const SvXMLTokenMapEntry* pAttributes,
53*cdf0e10cSrcweir                             const SvXMLTokenMapEntry* pChildren )
54*cdf0e10cSrcweir     : SvXMLImportContext( rImport, nPrefix, rLocalName ),
55*cdf0e10cSrcweir       mpAttributes( pAttributes ),
56*cdf0e10cSrcweir       mpChildren( pChildren )
57*cdf0e10cSrcweir {
58*cdf0e10cSrcweir }
59*cdf0e10cSrcweir 
60*cdf0e10cSrcweir TokenContext::~TokenContext()
61*cdf0e10cSrcweir {
62*cdf0e10cSrcweir }
63*cdf0e10cSrcweir 
64*cdf0e10cSrcweir void TokenContext::StartElement(
65*cdf0e10cSrcweir     const Reference<XAttributeList>& xAttributeList )
66*cdf0e10cSrcweir {
67*cdf0e10cSrcweir     // iterate over attributes
68*cdf0e10cSrcweir     // - if in map: call HandleAttribute
69*cdf0e10cSrcweir     // - xmlns:... : ignore
70*cdf0e10cSrcweir     // - other: warning
71*cdf0e10cSrcweir     DBG_ASSERT( mpAttributes != NULL, "no token map for attributes" );
72*cdf0e10cSrcweir     SvXMLTokenMap aMap( mpAttributes );
73*cdf0e10cSrcweir 
74*cdf0e10cSrcweir     sal_Int16 nCount = xAttributeList->getLength();
75*cdf0e10cSrcweir     for( sal_Int16 i = 0; i < nCount; i++ )
76*cdf0e10cSrcweir     {
77*cdf0e10cSrcweir         // get key/local-name pair from namespace map
78*cdf0e10cSrcweir 		OUString sLocalName;
79*cdf0e10cSrcweir 		sal_uInt16 nPrefix = GetImport().GetNamespaceMap().
80*cdf0e10cSrcweir 			GetKeyByAttrName( xAttributeList->getNameByIndex(i), &sLocalName );
81*cdf0e10cSrcweir 
82*cdf0e10cSrcweir         // get token from token map
83*cdf0e10cSrcweir         sal_uInt16 nToken = aMap.Get( nPrefix, sLocalName );
84*cdf0e10cSrcweir 
85*cdf0e10cSrcweir         // and the value...
86*cdf0e10cSrcweir         const OUString& rValue = xAttributeList->getValueByIndex(i);
87*cdf0e10cSrcweir 
88*cdf0e10cSrcweir         if( nToken != XML_TOK_UNKNOWN )
89*cdf0e10cSrcweir         {
90*cdf0e10cSrcweir             HandleAttribute( nToken, rValue );
91*cdf0e10cSrcweir         }
92*cdf0e10cSrcweir         else if( nPrefix != XML_NAMESPACE_XMLNS )
93*cdf0e10cSrcweir         {
94*cdf0e10cSrcweir             // error handling, for all attribute that are not
95*cdf0e10cSrcweir             // namespace declarations
96*cdf0e10cSrcweir             GetImport().SetError( XMLERROR_UNKNOWN_ATTRIBUTE,
97*cdf0e10cSrcweir                                   sLocalName, rValue);
98*cdf0e10cSrcweir         }
99*cdf0e10cSrcweir     }
100*cdf0e10cSrcweir }
101*cdf0e10cSrcweir 
102*cdf0e10cSrcweir SvXMLImportContext* TokenContext::CreateChildContext(
103*cdf0e10cSrcweir     sal_uInt16 nPrefix,
104*cdf0e10cSrcweir     const OUString& rLocalName,
105*cdf0e10cSrcweir     const Reference<XAttributeList>& xAttrList )
106*cdf0e10cSrcweir {
107*cdf0e10cSrcweir     // call HandleChild for elements in token map. Ignore other content.
108*cdf0e10cSrcweir 
109*cdf0e10cSrcweir     SvXMLImportContext* pContext = NULL;
110*cdf0e10cSrcweir 
111*cdf0e10cSrcweir     DBG_ASSERT( mpChildren != NULL, "no token map for child elements" );
112*cdf0e10cSrcweir     SvXMLTokenMap aMap( mpChildren );
113*cdf0e10cSrcweir     sal_uInt16 nToken = aMap.Get( nPrefix, rLocalName );
114*cdf0e10cSrcweir     if( nToken != XML_TOK_UNKNOWN )
115*cdf0e10cSrcweir     {
116*cdf0e10cSrcweir         // call handle child, and pass down arguments
117*cdf0e10cSrcweir         pContext = HandleChild( nToken, nPrefix, rLocalName, xAttrList );
118*cdf0e10cSrcweir     }
119*cdf0e10cSrcweir 
120*cdf0e10cSrcweir     // error handling: create default context and generate warning
121*cdf0e10cSrcweir     if( pContext == NULL )
122*cdf0e10cSrcweir     {
123*cdf0e10cSrcweir         GetImport().SetError( XMLERROR_UNKNOWN_ELEMENT, rLocalName );
124*cdf0e10cSrcweir         pContext = new SvXMLImportContext( GetImport(), nPrefix, rLocalName );
125*cdf0e10cSrcweir     }
126*cdf0e10cSrcweir     return pContext;
127*cdf0e10cSrcweir }
128*cdf0e10cSrcweir 
129*cdf0e10cSrcweir bool lcl_IsWhiteSpace( sal_Unicode c )
130*cdf0e10cSrcweir {
131*cdf0e10cSrcweir     return c == sal_Unicode(  ' ' )
132*cdf0e10cSrcweir         || c == sal_Unicode( 0x09 )
133*cdf0e10cSrcweir         || c == sal_Unicode( 0x0A )
134*cdf0e10cSrcweir         || c == sal_Unicode( 0x0D );
135*cdf0e10cSrcweir }
136*cdf0e10cSrcweir 
137*cdf0e10cSrcweir void TokenContext::Characters( const ::rtl::OUString& rCharacters )
138*cdf0e10cSrcweir {
139*cdf0e10cSrcweir     // get iterators for string data
140*cdf0e10cSrcweir     const sal_Unicode* pBegin = rCharacters.getStr();
141*cdf0e10cSrcweir     const sal_Unicode* pEnd = &( pBegin[ rCharacters.getLength() ] );
142*cdf0e10cSrcweir 
143*cdf0e10cSrcweir     // raise error if non-whitespace character is found
144*cdf0e10cSrcweir     if( ::std::find_if( pBegin, pEnd, ::std::not1(::std::ptr_fun(lcl_IsWhiteSpace)) ) != pEnd )
145*cdf0e10cSrcweir         GetImport().SetError( XMLERROR_UNKNOWN_CHARACTERS, rCharacters );
146*cdf0e10cSrcweir }
147