|           Line data    Source code 
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : 
      21             : #include "saxeventkeeperimpl.hxx"
      22             : #include <com/sun/star/lang/XMultiServiceFactory.hpp>
      23             : #include <com/sun/star/xml/sax/XDocumentHandler.hpp>
      24             : #include <com/sun/star/xml/crypto/sax/ConstOfSecurityId.hpp>
      25             : #include <cppuhelper/supportsservice.hxx>
      26             : 
      27             : namespace cssu = com::sun::star::uno;
      28             : namespace cssl = com::sun::star::lang;
      29             : namespace cssxc = com::sun::star::xml::crypto;
      30             : namespace cssxcsax = com::sun::star::xml::csax;
      31             : namespace cssxw = com::sun::star::xml::wrapper;
      32             : namespace cssxs = com::sun::star::xml::sax;
      33             : 
      34             : #define SERVICE_NAME "com.sun.star.xml.crypto.sax.SAXEventKeeper"
      35             : #define IMPLEMENTATION_NAME "com.sun.star.xml.security.framework.SAXEventKeeperImpl"
      36             : 
      37             : #define _USECOMPRESSEDDOCUMENTHANDLER
      38             : 
      39           0 : SAXEventKeeperImpl::SAXEventKeeperImpl( )
      40             :     :m_pRootBufferNode(NULL),
      41             :      m_pCurrentBufferNode(NULL),
      42             :      m_nNextElementMarkId(1),
      43             :      m_pNewBlocker(NULL),
      44             :      m_pCurrentBlockingBufferNode(NULL),
      45             :      m_bIsReleasing(false),
      46           0 :      m_bIsForwarding(false)
      47             : {
      48           0 :     m_vElementMarkBuffers.reserve(2);
      49           0 :     m_vNewElementCollectors.reserve(2);
      50           0 :     m_vReleasedElementMarkBuffers.reserve(2);
      51           0 : }
      52             : 
      53           0 : SAXEventKeeperImpl::~SAXEventKeeperImpl()
      54             : {
      55             :     /*
      56             :      * delete the BufferNode tree
      57             :      */
      58           0 :     if (m_pRootBufferNode != NULL)
      59             :     {
      60           0 :         m_pRootBufferNode->freeAllChildren();
      61           0 :         delete m_pRootBufferNode;
      62             :     }
      63             : 
      64           0 :     m_pRootBufferNode = m_pCurrentBufferNode = m_pCurrentBlockingBufferNode = NULL;
      65             : 
      66             :     /*
      67             :      * delete all unfreed ElementMarks
      68             :      */
      69           0 :     m_vNewElementCollectors.clear();
      70           0 :     m_pNewBlocker = NULL;
      71             : 
      72           0 :     std::vector< const ElementMark* >::const_iterator ii = m_vElementMarkBuffers.begin();
      73           0 :     for( ; ii != m_vElementMarkBuffers.end(); ++ii )
      74             :     {
      75           0 :         delete (*ii);
      76             :     }
      77           0 :     m_vElementMarkBuffers.clear();
      78           0 : }
      79             : 
      80           0 : void SAXEventKeeperImpl::setCurrentBufferNode(BufferNode* pBufferNode)
      81             : /****** SAXEventKeeperImpl/setCurrentBufferNode ******************************
      82             :  *
      83             :  *   NAME
      84             :  *  setCurrentBufferNode -- set a new active BufferNode.
      85             :  *
      86             :  *   SYNOPSIS
      87             :  *  setCurrentBufferNode( pBufferNode );
      88             :  *
      89             :  *   FUNCTION
      90             :  *  connects this BufferNode into the BufferNode tree as a child of the
      91             :  *  current active BufferNode. Then makes this BufferNode as the current
      92             :  *  active BufferNode.
      93             :  *  If the previous active BufferNode points to the root
      94             :  *  BufferNode, which means that no buffering operation was proceeding,
      95             :  *  then notifies the status change listener that buffering  operation
      96             :  *  will begin at once.
      97             :  *
      98             :  *   INPUTS
      99             :  *  pBufferNode - a BufferNode which will be the new active BufferNode
     100             :  *
     101             :  *   RESULT
     102             :  *  empty
     103             :  *
     104             :  *   AUTHOR
     105             :  *  Michael Mi
     106             :  *  Email: michael.mi@sun.com
     107             :  ******************************************************************************/
     108             : {
     109           0 :     if (pBufferNode != m_pCurrentBufferNode)
     110             :     {
     111           0 :         if ( m_pCurrentBufferNode == m_pRootBufferNode &&
     112           0 :              m_xSAXEventKeeperStatusChangeListener.is())
     113             :         {
     114           0 :             m_xSAXEventKeeperStatusChangeListener->collectionStatusChanged(sal_True);
     115             :         }
     116             : 
     117           0 :         if (pBufferNode->getParent() == NULL)
     118             :         {
     119           0 :             m_pCurrentBufferNode->addChild(pBufferNode);
     120           0 :             pBufferNode->setParent(m_pCurrentBufferNode);
     121             :         }
     122             : 
     123           0 :         m_pCurrentBufferNode = pBufferNode;
     124             :     }
     125           0 : }
     126             : 
     127           0 : BufferNode* SAXEventKeeperImpl::addNewElementMarkBuffers()
     128             : /****** SAXEventKeeperImpl/addNewElementMarkBuffers **************************
     129             :  *
     130             :  *   NAME
     131             :  *  addNewElementMarkBuffers -- add new ElementCollectors and new Blocker.
     132             :  *
     133             :  *   SYNOPSIS
     134             :  *  pBufferNode = addNewElementMarkBuffers( );
     135             :  *
     136             :  *   FUNCTION
     137             :  *  if there are new ElementCollector or new Blocker to be added, then
     138             :  *  connect all of them with the current BufferNode. In case of the
     139             :  *  current BufferNode doesn't exist, creates one.
     140             :  *  Clears up the new ElementCollector list and the new Blocker pointer.
     141             :  *
     142             :  *   INPUTS
     143             :  *  empty
     144             :  *
     145             :  *   RESULT
     146             :  *  pBufferNode - the BufferNode that has been connected with both new
     147             :  *                ElementCollectors and new Blocker.
     148             :  *
     149             :  *   AUTHOR
     150             :  *  Michael Mi
     151             :  *  Email: michael.mi@sun.com
     152             :  ******************************************************************************/
     153             : {
     154           0 :     BufferNode* pBufferNode = NULL;
     155             : 
     156           0 :     if (m_pNewBlocker || !m_vNewElementCollectors.empty() )
     157             :     {
     158             :         /*
     159             :          * When the current BufferNode is right pointing to the current
     160             :          * working element in the XMLDocumentWrapper component, then
     161             :          * no new BufferNode is needed to create.
     162             :          * This situation can only happen in the "Forwarding" mode.
     163             :          */
     164           0 :         if ( (m_pCurrentBufferNode != NULL) &&
     165           0 :              (m_xXMLDocument->isCurrent(m_pCurrentBufferNode->getXMLElement())))
     166             :         {
     167           0 :             pBufferNode = m_pCurrentBufferNode;
     168             :         }
     169             :         else
     170             :         {
     171           0 :             pBufferNode = new BufferNode(m_xXMLDocument->getCurrentElement());
     172             :         }
     173             : 
     174           0 :         if (m_pNewBlocker != NULL)
     175             :         {
     176           0 :             pBufferNode->setBlocker(m_pNewBlocker);
     177             : 
     178             :             /*
     179             :              * If no blocking before, then notify the status change listener that
     180             :              * the SAXEventKeeper has entered "blocking" status, during which, no
     181             :              * SAX events will be forwarded to the next document handler.
     182             :              */
     183           0 :             if (m_pCurrentBlockingBufferNode == NULL)
     184             :             {
     185           0 :                 m_pCurrentBlockingBufferNode = pBufferNode;
     186             : 
     187           0 :                 if (m_xSAXEventKeeperStatusChangeListener.is())
     188             :                 {
     189           0 :                     m_xSAXEventKeeperStatusChangeListener->blockingStatusChanged(sal_True);
     190             :                 }
     191             :             }
     192             : 
     193           0 :             m_pNewBlocker = NULL;
     194             :         }
     195             : 
     196           0 :         if (!m_vNewElementCollectors.empty())
     197             :         {
     198           0 :             std::vector< const ElementCollector* >::const_iterator ii = m_vNewElementCollectors.begin();
     199             : 
     200           0 :             for( ; ii != m_vNewElementCollectors.end(); ++ii )
     201             :             {
     202           0 :                 pBufferNode->addElementCollector(*ii);
     203             :             }
     204             : 
     205           0 :             m_vNewElementCollectors.clear();
     206             :         }
     207             :     }
     208             : 
     209           0 :     return pBufferNode;
     210             : }
     211             : 
     212           0 : ElementMark* SAXEventKeeperImpl::findElementMarkBuffer(sal_Int32 nId) const
     213             : /****** SAXEventKeeperImpl/findElementMarkBuffer *****************************
     214             :  *
     215             :  *   NAME
     216             :  *  findElementMarkBuffer -- finds an ElementMark.
     217             :  *
     218             :  *   SYNOPSIS
     219             :  *  pElementMark = findElementMarkBuffer( nId );
     220             :  *
     221             :  *   FUNCTION
     222             :  *  searches an ElementMark with the particular Id in the ElementMark
     223             :  *  list.
     224             :  *
     225             :  *   INPUTS
     226             :  *  nId - the Id of the ElementMark to be searched.
     227             :  *
     228             :  *   RESULT
     229             :  *  pElementMark - the ElementMark with the particular Id, or NULL when
     230             :  *                 no such Id exists.
     231             :  *
     232             :  *   AUTHOR
     233             :  *  Michael Mi
     234             :  *  Email: michael.mi@sun.com
     235             :  ******************************************************************************/
     236             : {
     237           0 :     ElementMark* pElementMark = NULL;
     238             : 
     239           0 :     std::vector< const ElementMark* >::const_iterator ii = m_vElementMarkBuffers.begin();
     240             : 
     241           0 :     for( ; ii != m_vElementMarkBuffers.end(); ++ii )
     242             :     {
     243           0 :         if ( nId == (*ii)->getBufferId())
     244             :         {
     245           0 :             pElementMark = (ElementMark*)*ii;
     246           0 :             break;
     247             :         }
     248             :     }
     249             : 
     250           0 :     return pElementMark;
     251             : }
     252             : 
     253           0 : void SAXEventKeeperImpl::removeElementMarkBuffer(sal_Int32 nId)
     254             : /****** SAXEventKeeperImpl/removeElementMarkBuffer ***************************
     255             :  *
     256             :  *   NAME
     257             :  *  removeElementMarkBuffer -- removes an ElementMark
     258             :  *
     259             :  *   SYNOPSIS
     260             :  *  removeElementMarkBuffer( nId );
     261             :  *
     262             :  *   FUNCTION
     263             :  *  removes an ElementMark with the particular Id in the ElementMark list.
     264             :  *
     265             :  *   INPUTS
     266             :  *  nId - the Id of the ElementMark to be removed.
     267             :  *
     268             :  *   RESULT
     269             :  *  empty
     270             :  *
     271             :  *   AUTHOR
     272             :  *  Michael Mi
     273             :  *  Email: michael.mi@sun.com
     274             :  ******************************************************************************/
     275             : {
     276           0 :     std::vector< const ElementMark* >::iterator ii = m_vElementMarkBuffers.begin();
     277             : 
     278           0 :     for( ; ii != m_vElementMarkBuffers.end(); ++ii )
     279             :     {
     280           0 :         if ( nId == (*ii)->getBufferId())
     281             :         {
     282             :             /*
     283             :              * checks whether this ElementMark still in the new ElementCollect array
     284             :              */
     285           0 :             std::vector< const ElementCollector* >::iterator jj = m_vNewElementCollectors.begin();
     286           0 :             for( ; jj != m_vNewElementCollectors.end(); ++jj )
     287             :             {
     288           0 :                 if ((*ii) == (*jj))
     289             :                 {
     290           0 :                     m_vNewElementCollectors.erase(jj);
     291           0 :                     break;
     292             :                 }
     293             :             }
     294             : 
     295             :             /*
     296             :              * checks whether this ElementMark is the new Blocker
     297             :              */
     298           0 :             if ((*ii) == m_pNewBlocker)
     299             :             {
     300           0 :                 m_pNewBlocker = NULL;
     301             :             }
     302             : 
     303             :             /*
     304             :              * destory the ElementMark
     305             :              */
     306           0 :             delete (*ii);
     307             : 
     308           0 :             m_vElementMarkBuffers.erase( ii );
     309           0 :             break;
     310             :         }
     311             :     }
     312           0 : }
     313             : 
     314           0 : OUString SAXEventKeeperImpl::printBufferNode(
     315             :     BufferNode* pBufferNode, sal_Int32 nIndent) const
     316             : /****** SAXEventKeeperImpl/printBufferNode ***********************************
     317             :  *
     318             :  *   NAME
     319             :  *  printBufferNode -- retrieves the information of a BufferNode and its
     320             :  *  branch.
     321             :  *
     322             :  *   SYNOPSIS
     323             :  *  info = printBufferNode( pBufferNode, nIndent );
     324             :  *
     325             :  *   FUNCTION
     326             :  *  all retrieved information includes:
     327             :  *  1. whether it is the current BufferNode;
     328             :  *  2. whether it is the current blocking BufferNode;
     329             :  *  3. the name of the parent element;
     330             :  *  4. the name of this element;
     331             :  *  5. all ElementCollectors working on this BufferNode;
     332             :  *  6. the Blocker working on this BufferNode;
     333             :  *  7. all child BufferNodes' information.
     334             :  *
     335             :  *   INPUTS
     336             :  *  pBufferNode -   the BufferNode from where information will be retrieved.
     337             :  *  nIndent -   how many space characters prefixed before the output
     338             :  *              message.
     339             :  *
     340             :  *   RESULT
     341             :  *  info - the information string
     342             :  *
     343             :  *   AUTHOR
     344             :  *  Michael Mi
     345             :  *  Email: michael.mi@sun.com
     346             :  ******************************************************************************/
     347             : {
     348           0 :     OUString rc;
     349             : 
     350           0 :     for ( int i=0; i<nIndent; ++i )
     351             :     {
     352           0 :         rc += OUString( " " );
     353             :     }
     354             : 
     355           0 :     if (pBufferNode == m_pCurrentBufferNode)
     356             :     {
     357           0 :         rc += OUString( "[%]" );
     358             :     }
     359             : 
     360           0 :     if (pBufferNode == m_pCurrentBlockingBufferNode)
     361             :     {
     362           0 :         rc += OUString( "[B]" );
     363             :     }
     364             : 
     365           0 :     rc += OUString( " " );
     366           0 :     rc += m_xXMLDocument->getNodeName(pBufferNode->getXMLElement());
     367             : 
     368           0 :     BufferNode* pParent = (BufferNode*)pBufferNode->getParent();
     369           0 :     if (pParent != NULL)
     370             :     {
     371           0 :         rc += OUString( "[" );
     372           0 :         rc += m_xXMLDocument->getNodeName(pParent->getXMLElement());
     373           0 :         rc += OUString( "]" );
     374             :     }
     375             : 
     376           0 :     rc += OUString( ":EC=" );
     377           0 :     rc += pBufferNode->printChildren();
     378           0 :     rc += OUString( " BR=" );
     379             : 
     380           0 :     ElementMark * pBlocker = pBufferNode->getBlocker();
     381           0 :     if (pBlocker != NULL)
     382             :     {
     383           0 :         rc += OUString::number( pBlocker->getBufferId() );
     384           0 :         rc += OUString( "(SecId=" );
     385           0 :         rc += OUString::number( pBlocker->getSecurityId() );
     386           0 :         rc += OUString( ")" );
     387           0 :         rc += OUString( " " );
     388             :     }
     389           0 :     rc += OUString( "\n" );
     390             : 
     391           0 :     std::vector< const BufferNode* >* vChildren = pBufferNode->getChildren();
     392           0 :     std::vector< const BufferNode* >::const_iterator jj = vChildren->begin();
     393           0 :     for( ; jj != vChildren->end(); ++jj )
     394             :     {
     395           0 :         rc += printBufferNode((BufferNode *)*jj, nIndent+4);
     396             :     }
     397             : 
     398           0 :     delete vChildren;
     399             : 
     400           0 :     return rc;
     401             : }
     402             : 
     403             : cssu::Sequence< cssu::Reference< cssxw::XXMLElementWrapper > >
     404           0 :     SAXEventKeeperImpl::collectChildWorkingElement(BufferNode* pBufferNode) const
     405             : /****** SAXEventKeeperImpl/collectChildWorkingElement ************************
     406             :  *
     407             :  *   NAME
     408             :  *  collectChildWorkingElement -- collects a BufferNode's all child
     409             :  *  Elements.
     410             :  *
     411             :  *   SYNOPSIS
     412             :  *  list = collectChildWorkingElement( pBufferNode );
     413             :  *
     414             :  *   FUNCTION
     415             :  *  see NAME.
     416             :  *
     417             :  *   INPUTS
     418             :  *  pBufferNode - the BufferNode whose child Elements will be collected.
     419             :  *
     420             :  *   RESULT
     421             :  *  list - the child Elements list.
     422             :  *
     423             :  *   AUTHOR
     424             :  *  Michael Mi
     425             :  *  Email: michael.mi@sun.com
     426             :  ******************************************************************************/
     427             : {
     428           0 :     std::vector< const BufferNode* >* vChildren = pBufferNode->getChildren();
     429             : 
     430             :     cssu::Sequence < cssu::Reference<
     431           0 :         cssxw::XXMLElementWrapper > > aChildrenCollection ( vChildren->size());
     432             : 
     433           0 :     std::vector< const BufferNode* >::const_iterator ii = vChildren->begin();
     434             : 
     435           0 :     sal_Int32 nIndex = 0;
     436           0 :     for( ; ii != vChildren->end(); ++ii )
     437             :     {
     438           0 :         aChildrenCollection[nIndex] = (*ii)->getXMLElement();
     439           0 :         nIndex++;
     440             :     }
     441             : 
     442           0 :     delete vChildren;
     443             : 
     444           0 :     return aChildrenCollection;
     445             : }
     446             : 
     447           0 : void SAXEventKeeperImpl::smashBufferNode(
     448             :     BufferNode* pBufferNode, bool bClearRoot) const
     449             : /****** SAXEventKeeperImpl/smashBufferNode ***********************************
     450             :  *
     451             :  *   NAME
     452             :  *  smashBufferNode -- removes a BufferNode along with its working
     453             :  *  element.
     454             :  *
     455             :  *   SYNOPSIS
     456             :  *  smashBufferNode( pBufferNode, bClearRoot );
     457             :  *
     458             :  *   FUNCTION
     459             :  *  removes the BufferNode's working element from the DOM document, while
     460             :  *  reserves all ancestor paths for its child BufferNodes.
     461             :  *  when any of the BufferNode's ancestor element is useless, removes it
     462             :  *  too.
     463             :  *  removes the BufferNode from the BufferNode tree.
     464             :  *
     465             :  *   INPUTS
     466             :  *  pBufferNode -   the BufferNode to be removed
     467             :  *  bClearRoot -    whether the root element also needs to be cleared up.
     468             :  *
     469             :  *   RESULT
     470             :  *  empty
     471             :  *
     472             :  *   NOTES
     473             :  *  when removeing a Blocker's BufferNode, the bClearRoot flag should be
     474             :  *  true. Because a Blocker can buffer many SAX events which are not used
     475             :  *  by any other ElementCollector or Blocker.
     476             :  *  When the bClearRoot is set to true, the root BufferNode will be first
     477             :  *  cleared, with a stop flag seting at the next Blocking BufferNode. This
     478             :  *  operation can delete all useless bufferred SAX events which are only
     479             :  *  needed by the Blocker to be deleted.
     480             :  *
     481             :  *   AUTHOR
     482             :  *  Michael Mi
     483             :  *  Email: michael.mi@sun.com
     484             :  ******************************************************************************/
     485             : {
     486           0 :     if (!pBufferNode->hasAnything())
     487             :     {
     488           0 :         BufferNode* pParent = (BufferNode*)pBufferNode->getParent();
     489             : 
     490             :             /*
     491             :              * delete the XML data
     492             :              */
     493           0 :         if (pParent == m_pRootBufferNode)
     494             :         {
     495           0 :             bool bIsNotBlocking = (m_pCurrentBlockingBufferNode == NULL);
     496           0 :             bool bIsBlockInside = false;
     497           0 :             bool bIsBlockingAfterward = false;
     498             : 
     499             :                 /*
     500             :                  * If this is a blocker, then remove any out-element data
     501             :                  * which caused by blocking. The removal process will stop
     502             :                  * at the next blokcer to avoid removing any useful data.
     503             :                  */
     504           0 :             if (bClearRoot)
     505             :             {
     506             :                 cssu::Sequence< cssu::Reference< cssxw::XXMLElementWrapper > >
     507           0 :                     aChildElements = collectChildWorkingElement(m_pRootBufferNode);
     508             : 
     509             :                     /*
     510             :                      * the clearUselessData only clearup the content in the
     511             :                      * node, not the node itself.
     512             :                      */
     513           0 :                 m_xXMLDocument->clearUselessData(m_pRootBufferNode->getXMLElement(),
     514             :                     aChildElements,
     515             :                     bIsNotBlocking?(NULL):
     516           0 :                                    (m_pCurrentBlockingBufferNode->getXMLElement()));
     517             : 
     518             :                     /*
     519             :                      * remove the node if it is empty, then if its parent is also
     520             :                      * empty, remove it, then if the next parent is also empty,
     521             :                      * remove it,..., until parent become null.
     522             :                      */
     523           0 :                 m_xXMLDocument->collapse( m_pRootBufferNode->getXMLElement() );
     524             :             }
     525             : 
     526             :             /*
     527             :              * if blocking, check the relationship between this BufferNode and
     528             :              * the current blocking BufferNode.
     529             :              */
     530           0 :             if ( !bIsNotBlocking )
     531             :             {
     532             :                 /*
     533             :                  * the current blocking BufferNode is a descendant of this BufferNode.
     534             :                  */
     535           0 :                 bIsBlockInside = (NULL != pBufferNode->isAncestor(m_pCurrentBlockingBufferNode));
     536             : 
     537             :                 /*
     538             :                  * the current blocking BufferNode locates behind this BufferNode in tree
     539             :                  * order.
     540             :                  */
     541           0 :                 bIsBlockingAfterward = pBufferNode->isPrevious(m_pCurrentBlockingBufferNode);
     542             :             }
     543             : 
     544             :             /*
     545             :              * this BufferNode's working element needs to be deleted only when
     546             :              * 1. there is no blocking, or
     547             :              * 2. the current blocking BufferNode is a descendant of this BufferNode,
     548             :              *    (then in the BufferNode's working element, the useless data before the blocking
     549             :              *     element should be deleted.) or
     550             :              * 3. the current blocking BufferNode is locates behind this BufferNode in tree,
     551             :              *    (then the useless data between the blocking element and the working element
     552             :              *     should be deleted.).
     553             :              * Otherwise, this working element should not be deleted.
     554             :              */
     555           0 :             if ( bIsNotBlocking || bIsBlockInside || bIsBlockingAfterward )
     556             :             {
     557             :                 cssu::Sequence< cssu::Reference< cssxw::XXMLElementWrapper > >
     558           0 :                     aChildElements = collectChildWorkingElement(pBufferNode);
     559             : 
     560             :                     /*
     561             :                      * the clearUselessData only clearup the content in the
     562             :                      * node, not the node itself.
     563             :                      */
     564           0 :                 m_xXMLDocument->clearUselessData(pBufferNode->getXMLElement(),
     565             :                     aChildElements,
     566             :                     bIsBlockInside?(m_pCurrentBlockingBufferNode->getXMLElement()):
     567           0 :                                (NULL));
     568             : 
     569             :                     /*
     570             :                      * remove the node if it is empty, then if its parent is also
     571             :                      * empty, remove it, then if the next parent is also empty,
     572             :                      * remove it,..., until parent become null.
     573             :                      */
     574           0 :                 m_xXMLDocument->collapse( pBufferNode->getXMLElement() );
     575             :             }
     576             :         }
     577             : 
     578           0 :         sal_Int32 nIndex = pParent->indexOfChild(pBufferNode);
     579             : 
     580           0 :         std::vector< const BufferNode* >* vChildren = pBufferNode->getChildren();
     581           0 :         pParent->removeChild(pBufferNode);
     582           0 :         pBufferNode->setParent(NULL);
     583             : 
     584           0 :         std::vector< const BufferNode * >::const_iterator ii = vChildren->begin();
     585           0 :         for( ; ii != vChildren->end(); ++ii )
     586             :         {
     587           0 :             ((BufferNode *)(*ii))->setParent(pParent);
     588           0 :             pParent->addChild(*ii, nIndex);
     589           0 :             nIndex++;
     590             :         }
     591             : 
     592           0 :         delete vChildren;
     593             : 
     594             :         /*
     595             :          * delete the BufferNode
     596             :          */
     597           0 :         delete pBufferNode;
     598             :     }
     599           0 : }
     600             : 
     601           0 : BufferNode* SAXEventKeeperImpl::findNextBlockingBufferNode(
     602             :     BufferNode* pStartBufferNode) const
     603             : /****** SAXEventKeeperImpl/findNextBlockingBufferNode ************************
     604             :  *
     605             :  *   NAME
     606             :  *  findNextBlockingBufferNode -- finds the next blocking BufferNode
     607             :  *  behind the particular BufferNode.
     608             :  *
     609             :  *   SYNOPSIS
     610             :  *  pBufferNode = findNextBlockingBufferNode( pStartBufferNode );
     611             :  *
     612             :  *   FUNCTION
     613             :  *  see NAME.
     614             :  *
     615             :  *   INPUTS
     616             :  *  pStartBufferNode - the BufferNode from where to search the next
     617             :  *                     blocking BufferNode.
     618             :  *
     619             :  *   RESULT
     620             :  *  pBufferNode - the next blocking BufferNode, or NULL if no such
     621             :  *                BufferNode exists.
     622             :  *
     623             :  *   AUTHOR
     624             :  *  Michael Mi
     625             :  *  Email: michael.mi@sun.com
     626             :  ******************************************************************************/
     627             : {
     628           0 :     BufferNode* pNext = NULL;
     629             : 
     630           0 :     if (pStartBufferNode != NULL)
     631             :     {
     632           0 :         pNext = pStartBufferNode;
     633             : 
     634           0 :         while (NULL != (pNext = (BufferNode*)pNext->getNextNodeByTreeOrder()))
     635             :         {
     636           0 :             if (pNext->getBlocker() != NULL)
     637             :             {
     638           0 :                 break;
     639             :             }
     640             :         }
     641             :     }
     642             : 
     643           0 :     return pNext;
     644             : }
     645             : 
     646           0 : void SAXEventKeeperImpl::diffuse(BufferNode* pBufferNode) const
     647             : /****** SAXEventKeeperImpl/diffuse *******************************************
     648             :  *
     649             :  *   NAME
     650             :  *  diffuse -- diffuse the notification.
     651             :  *
     652             :  *   SYNOPSIS
     653             :  *  diffuse( pBufferNode );
     654             :  *
     655             :  *   FUNCTION
     656             :  *  diffuse the collecting completion notification from the specific
     657             :  *  BufferNode along its parent link, until an ancestor which is not
     658             :  *  completely received is met.
     659             :  *
     660             :  *   INPUTS
     661             :  *  pBufferNode - the BufferNode from which the notification will be
     662             :  *                diffused.
     663             :  *
     664             :  *   RESULT
     665             :  *  empty
     666             :  *
     667             :  *   AUTHOR
     668             :  *  Michael Mi
     669             :  *  Email: michael.mi@sun.com
     670             :  ******************************************************************************/
     671             : {
     672           0 :     BufferNode* pParent = pBufferNode;
     673             : 
     674           0 :     while(pParent->isAllReceived())
     675             :     {
     676           0 :         pParent->elementCollectorNotify();
     677           0 :         pParent = (BufferNode*)pParent->getParent();
     678             :     }
     679           0 : }
     680             : 
     681           0 : void SAXEventKeeperImpl::releaseElementMarkBuffer()
     682             : /****** SAXEventKeeperImpl/releaseElementMarkBuffer **************************
     683             :  *
     684             :  *   NAME
     685             :  *  releaseElementMarkBuffer -- releases useless ElementMarks
     686             :  *
     687             :  *   SYNOPSIS
     688             :  *  releaseElementMarkBuffer( );
     689             :  *
     690             :  *   FUNCTION
     691             :  *  releases each ElementMark in the releasing list
     692             :  *  m_vReleasedElementMarkBuffers.
     693             :  *  The operation differs between an ElementCollector and a Blocker.
     694             :  *
     695             :  *   INPUTS
     696             :  *  empty
     697             :  *
     698             :  *   RESULT
     699             :  *  empty
     700             :  *
     701             :  *   AUTHOR
     702             :  *  Michael Mi
     703             :  *  Email: michael.mi@sun.com
     704             :  ******************************************************************************/
     705             : {
     706           0 :     m_bIsReleasing = true;
     707           0 :     while (!m_vReleasedElementMarkBuffers.empty())
     708             :     {
     709           0 :         std::vector< sal_Int32 >::iterator pId = m_vReleasedElementMarkBuffers.begin();
     710           0 :         sal_Int32 nId = *pId;
     711           0 :         m_vReleasedElementMarkBuffers.erase( pId );
     712             : 
     713           0 :         ElementMark* pElementMark = findElementMarkBuffer(nId);
     714             : 
     715           0 :         if (pElementMark != NULL)
     716             :         {
     717           0 :             if (cssxc::sax::ElementMarkType_ELEMENTCOLLECTOR
     718           0 :                 == pElementMark->getType())
     719             :             /*
     720             :              * it is a EC
     721             :              */
     722             :             {
     723           0 :                 ElementCollector* pElementCollector = static_cast<ElementCollector*>(pElementMark);
     724             : 
     725           0 :                 cssxc::sax::ElementMarkPriority nPriority = pElementCollector->getPriority();
     726             :                 /*
     727             :                      * Delete the EC from the buffer node.
     728             :                      */
     729           0 :                 BufferNode* pBufferNode = pElementCollector->getBufferNode();
     730           0 :                 pBufferNode->removeElementCollector(pElementCollector);
     731             : 
     732           0 :                 if ( nPriority == cssxc::sax::ElementMarkPriority_BEFOREMODIFY)
     733             :                 {
     734           0 :                     pBufferNode->notifyBranch();
     735             :                 }
     736             : 
     737             :                 /*
     738             :                  * delete the ElementMark
     739             :                  */
     740           0 :                 pElementCollector = NULL;
     741           0 :                 pElementMark = NULL;
     742           0 :                 removeElementMarkBuffer(nId);
     743             : 
     744             :                 /*
     745             :                  * delete the BufferNode
     746             :                  */
     747           0 :                 diffuse(pBufferNode);
     748           0 :                 smashBufferNode(pBufferNode, false);
     749             :             }
     750             :             else
     751             :             /*
     752             :              * it is a Blocker
     753             :              */
     754             :             {
     755             :                     /*
     756             :                      * Delete the TH from the buffer node.
     757             :                      */
     758           0 :                 BufferNode *pBufferNode = pElementMark->getBufferNode();
     759           0 :                 pBufferNode->setBlocker(NULL);
     760             : 
     761             :                     /*
     762             :                      * If there is a following handler and no blocking now, then
     763             :                      * forward this event
     764             :                      */
     765           0 :                 if (m_pCurrentBlockingBufferNode == pBufferNode)
     766             :                 {
     767             :                         /*
     768             :                          * Before forwarding, the next blocking point needs to be
     769             :                          * found.
     770             :                          */
     771           0 :                     m_pCurrentBlockingBufferNode = findNextBlockingBufferNode(pBufferNode);
     772             : 
     773             :                         /*
     774             :                          * Forward the blocked events between these two STHs.
     775             :                          */
     776           0 :                            if (m_xNextHandler.is())
     777             :                            {
     778           0 :                                BufferNode* pTempCurrentBufferNode = m_pCurrentBufferNode;
     779           0 :                                BufferNode* pTempCurrentBlockingBufferNode = m_pCurrentBlockingBufferNode;
     780             : 
     781           0 :                                m_pCurrentBufferNode = pBufferNode;
     782           0 :                                m_pCurrentBlockingBufferNode = NULL;
     783             : 
     784           0 :                         m_bIsForwarding = true;
     785             : 
     786           0 :                         m_xXMLDocument->generateSAXEvents(
     787             :                             m_xNextHandler,
     788             :                             this,
     789             :                             pBufferNode->getXMLElement(),
     790           0 :                             (pTempCurrentBlockingBufferNode == NULL)?NULL:(pTempCurrentBlockingBufferNode->getXMLElement()));
     791             : 
     792           0 :                         m_bIsForwarding = false;
     793             : 
     794           0 :                         m_pCurrentBufferNode = pTempCurrentBufferNode;
     795           0 :                         if (m_pCurrentBlockingBufferNode == NULL)
     796             :                         {
     797           0 :                             m_pCurrentBlockingBufferNode = pTempCurrentBlockingBufferNode;
     798             :                         }
     799             :                     }
     800             : 
     801           0 :                     if (m_pCurrentBlockingBufferNode == NULL &&
     802           0 :                         m_xSAXEventKeeperStatusChangeListener.is())
     803             :                     {
     804           0 :                         m_xSAXEventKeeperStatusChangeListener->blockingStatusChanged(sal_False);
     805             :                     }
     806             :                 }
     807             : 
     808             :                 /*
     809             :                  * delete the ElementMark
     810             :                  */
     811           0 :                 pElementMark = NULL;
     812           0 :                 removeElementMarkBuffer(nId);
     813             : 
     814             :                 /*
     815             :                  * delete the BufferNode
     816             :                  */
     817           0 :                 diffuse(pBufferNode);
     818           0 :                 smashBufferNode(pBufferNode, true);
     819             :             }
     820             :         }
     821             :     }
     822             : 
     823           0 :     m_bIsReleasing = false;
     824             : 
     825           0 :     if (!m_pRootBufferNode->hasAnything() &&
     826           0 :         !m_pRootBufferNode->hasChildren() &&
     827           0 :         m_xSAXEventKeeperStatusChangeListener.is())
     828             :     {
     829           0 :         m_xSAXEventKeeperStatusChangeListener->bufferStatusChanged(sal_True);
     830             :     }
     831           0 : }
     832             : 
     833           0 : void SAXEventKeeperImpl::markElementMarkBuffer(sal_Int32 nId)
     834             : /****** SAXEventKeeperImpl/markElementMarkBuffer *****************************
     835             :  *
     836             :  *   NAME
     837             :  *  markElementMarkBuffer -- marks an ElementMark to be released
     838             :  *
     839             :  *   SYNOPSIS
     840             :  *  markElementMarkBuffer( nId );
     841             :  *
     842             :  *   FUNCTION
     843             :  *  puts the ElementMark with the particular Id into the releasing list,
     844             :  *  checks whether the releasing process is runing, if not then launch
     845             :  *  this process.
     846             :  *
     847             :  *   INPUTS
     848             :  *  nId - the Id of the ElementMark which will be released
     849             :  *
     850             :  *   RESULT
     851             :  *  empty
     852             :  *
     853             :  *   AUTHOR
     854             :  *  Michael Mi
     855             :  *  Email: michael.mi@sun.com
     856             :  ******************************************************************************/
     857             : {
     858           0 :     m_vReleasedElementMarkBuffers.push_back( nId );
     859           0 :     if ( !m_bIsReleasing )
     860             :     {
     861           0 :         releaseElementMarkBuffer();
     862             :     }
     863           0 : }
     864             : 
     865           0 : sal_Int32 SAXEventKeeperImpl::createElementCollector(
     866             :     sal_Int32 nSecurityId,
     867             :     cssxc::sax::ElementMarkPriority nPriority,
     868             :     bool bModifyElement,
     869             :     const cssu::Reference< cssxc::sax::XReferenceResolvedListener >& xReferenceResolvedListener)
     870             : /****** SAXEventKeeperImpl/createElementCollector ****************************
     871             :  *
     872             :  *   NAME
     873             :  *  createElementCollector -- creates a new ElementCollector on the
     874             :  *  incoming element.
     875             :  *
     876             :  *   SYNOPSIS
     877             :  *  nId = createElementCollector( nSecurityId, nPriority,
     878             :  *                               bModifyElement,
     879             :  *                               xReferenceResolvedListener );
     880             :  *
     881             :  *   FUNCTION
     882             :  *  allocs a new Id, then create an ElementCollector with this Id value.
     883             :  *  Add the new created ElementCollector to the new ElementCollecotor list.
     884             :  *
     885             :  *   INPUTS
     886             :  *  nSecurityId -   the security Id of the new ElementCollector
     887             :  *  nPriority -     the prirority of the new ElementCollector
     888             :  *  bModifyElement -whether this BufferNode will modify the content of
     889             :  *                  the corresponding element it works on
     890             :  *  xReferenceResolvedListener - the listener for the new ElementCollector.
     891             :  *
     892             :  *   RESULT
     893             :  *  nId - the Id of the new ElementCollector
     894             :  *
     895             :  *   AUTHOR
     896             :  *  Michael Mi
     897             :  *  Email: michael.mi@sun.com
     898             :  ******************************************************************************/
     899             : {
     900           0 :     sal_Int32 nId = m_nNextElementMarkId;
     901           0 :     m_nNextElementMarkId ++;
     902             : 
     903             :     ElementCollector* pElementCollector
     904             :         = new ElementCollector(
     905             :             nSecurityId,
     906             :             nId,
     907             :             nPriority,
     908             :             bModifyElement,
     909           0 :             xReferenceResolvedListener);
     910             : 
     911           0 :     m_vElementMarkBuffers.push_back( pElementCollector );
     912             : 
     913             :         /*
     914             :          * All the new EC to initial EC array.
     915             :          */
     916           0 :     m_vNewElementCollectors.push_back( pElementCollector );
     917             : 
     918           0 :     return nId;
     919             : }
     920             : 
     921             : 
     922           0 : sal_Int32 SAXEventKeeperImpl::createBlocker(sal_Int32 nSecurityId)
     923             : /****** SAXEventKeeperImpl/createBlocker *************************************
     924             :  *
     925             :  *   NAME
     926             :  *  createBlocker -- creates a new Blocker on the incoming element.
     927             :  *
     928             :  *   SYNOPSIS
     929             :  *  nId = createBlocker( nSecurityId );
     930             :  *
     931             :  *   FUNCTION
     932             :  *  see NAME.
     933             :  *
     934             :  *   INPUTS
     935             :  *  nSecurityId -   the security Id of the new Blocker
     936             :  *
     937             :  *   RESULT
     938             :  *  nId - the Id of the new Blocker
     939             :  *
     940             :  *   AUTHOR
     941             :  *  Michael Mi
     942             :  *  Email: michael.mi@sun.com
     943             :  ******************************************************************************/
     944             : {
     945           0 :     sal_Int32 nId = m_nNextElementMarkId;
     946           0 :     m_nNextElementMarkId ++;
     947             : 
     948             :     OSL_ASSERT(m_pNewBlocker == NULL);
     949             : 
     950           0 :     m_pNewBlocker = new ElementMark(nSecurityId, nId);
     951           0 :     m_vElementMarkBuffers.push_back( m_pNewBlocker );
     952             : 
     953           0 :     return nId;
     954             : }
     955             : 
     956             : /* XSAXEventKeeper */
     957           0 : sal_Int32 SAL_CALL SAXEventKeeperImpl::addElementCollector(  )
     958             :     throw (cssu::RuntimeException, std::exception)
     959             : {
     960             :     return createElementCollector(
     961             :         cssxc::sax::ConstOfSecurityId::UNDEFINEDSECURITYID,
     962             :         cssxc::sax::ElementMarkPriority_AFTERMODIFY,
     963             :         false,
     964           0 :         NULL);
     965             : }
     966             : 
     967           0 : void SAL_CALL SAXEventKeeperImpl::removeElementCollector( sal_Int32 id )
     968             :     throw (cssu::RuntimeException, std::exception)
     969             : {
     970           0 :     markElementMarkBuffer(id);
     971           0 : }
     972             : 
     973           0 : sal_Int32 SAL_CALL SAXEventKeeperImpl::addBlocker(  )
     974             :     throw (cssu::RuntimeException, std::exception)
     975             : {
     976           0 :     return createBlocker(cssxc::sax::ConstOfSecurityId::UNDEFINEDSECURITYID);
     977             : }
     978             : 
     979           0 : void SAL_CALL SAXEventKeeperImpl::removeBlocker( sal_Int32 id )
     980             :     throw (cssu::RuntimeException, std::exception)
     981             : {
     982           0 :     markElementMarkBuffer(id);
     983           0 : }
     984             : 
     985           0 : sal_Bool SAL_CALL SAXEventKeeperImpl::isBlocking(  )
     986             :     throw (cssu::RuntimeException, std::exception)
     987             : {
     988           0 :     return (m_pCurrentBlockingBufferNode != NULL);
     989             : }
     990             : 
     991             : cssu::Reference< cssxw::XXMLElementWrapper > SAL_CALL
     992           0 :     SAXEventKeeperImpl::getElement( sal_Int32 id )
     993             :     throw (cssu::RuntimeException, std::exception)
     994             : {
     995           0 :     cssu::Reference< cssxw::XXMLElementWrapper > rc;
     996             : 
     997           0 :     ElementMark* pElementMark = findElementMarkBuffer(id);
     998           0 :     if (pElementMark != NULL)
     999             :     {
    1000           0 :         rc = pElementMark->getBufferNode()->getXMLElement();
    1001             :     }
    1002             : 
    1003           0 :     return rc;
    1004             : }
    1005             : 
    1006           0 : void SAL_CALL SAXEventKeeperImpl::setElement(
    1007             :     sal_Int32 id,
    1008             :     const cssu::Reference< cssxw::XXMLElementWrapper >& aElement )
    1009             :     throw (cssu::RuntimeException, std::exception)
    1010             : {
    1011           0 :     if (aElement.is())
    1012             :     {
    1013           0 :         m_xXMLDocument->rebuildIDLink(aElement);
    1014             : 
    1015           0 :         ElementMark* pElementMark = findElementMarkBuffer(id);
    1016             : 
    1017           0 :         if (pElementMark != NULL)
    1018             :         {
    1019           0 :             BufferNode* pBufferNode = pElementMark->getBufferNode();
    1020           0 :             if (pBufferNode != NULL)
    1021             :             {
    1022           0 :                     bool bIsCurrent = m_xXMLDocument->isCurrent(pBufferNode->getXMLElement());
    1023           0 :                 pBufferNode->setXMLElement(aElement);
    1024             : 
    1025           0 :                 if (bIsCurrent)
    1026             :                 {
    1027           0 :                     m_xXMLDocument->setCurrentElement(aElement);
    1028             :                 }
    1029             :             }
    1030             :         }
    1031             :     }
    1032             :     else
    1033             :     {
    1034           0 :         removeElementCollector( id );
    1035             :     }
    1036           0 : }
    1037             : 
    1038           0 : cssu::Reference< cssxs::XDocumentHandler > SAL_CALL SAXEventKeeperImpl::setNextHandler(
    1039             :     const cssu::Reference< cssxs::XDocumentHandler >& xNewHandler )
    1040             :     throw (cssu::RuntimeException, std::exception)
    1041             : {
    1042           0 :     cssu::Reference< cssxs::XDocumentHandler > xOldHandler = m_xNextHandler;
    1043             : 
    1044           0 :     m_xNextHandler = xNewHandler;
    1045           0 :     return xOldHandler;
    1046             : }
    1047             : 
    1048           0 : OUString SAL_CALL SAXEventKeeperImpl::printBufferNodeTree()
    1049             :     throw (cssu::RuntimeException, std::exception)
    1050             : {
    1051           0 :     OUString rc;
    1052             : 
    1053           0 :     rc += OUString( "ElementMarkBuffers: size = " );
    1054           0 :     rc += OUString::number(m_vElementMarkBuffers.size());
    1055           0 :     rc += OUString( "\nCurrentBufferNode: " );
    1056           0 :     rc += m_xXMLDocument->getNodeName(m_pCurrentBufferNode->getXMLElement());
    1057           0 :     rc += OUString( "\n" );
    1058           0 :     rc += printBufferNode(m_pRootBufferNode, 0);
    1059             : 
    1060           0 :     return rc;
    1061             : }
    1062             : 
    1063           0 : cssu::Reference< cssxw::XXMLElementWrapper > SAL_CALL SAXEventKeeperImpl::getCurrentBlockingNode()
    1064             :     throw (cssu::RuntimeException, std::exception)
    1065             : {
    1066           0 :     cssu::Reference< cssxw::XXMLElementWrapper > rc;
    1067             : 
    1068           0 :     if (m_pCurrentBlockingBufferNode != NULL)
    1069             :     {
    1070           0 :         rc = m_pCurrentBlockingBufferNode->getXMLElement();
    1071             :     }
    1072             : 
    1073           0 :     return rc;
    1074             : }
    1075             : 
    1076             : /* XSecuritySAXEventKeeper */
    1077           0 : sal_Int32 SAL_CALL SAXEventKeeperImpl::addSecurityElementCollector(
    1078             :     cssxc::sax::ElementMarkPriority priority,
    1079             :     sal_Bool modifyElement )
    1080             :     throw (cssu::RuntimeException, std::exception)
    1081             : {
    1082             :     return createElementCollector(
    1083             :         cssxc::sax::ConstOfSecurityId::UNDEFINEDSECURITYID,
    1084             :         priority,
    1085             :         modifyElement,
    1086           0 :         NULL);
    1087             : }
    1088             : 
    1089           0 : sal_Int32 SAL_CALL SAXEventKeeperImpl::cloneElementCollector(
    1090             :     sal_Int32 referenceId,
    1091             :     cssxc::sax::ElementMarkPriority priority )
    1092             :     throw (cssu::RuntimeException, std::exception)
    1093             : {
    1094           0 :     sal_Int32 nId = -1;
    1095             : 
    1096           0 :     ElementCollector* pElementCollector = static_cast<ElementCollector*>(findElementMarkBuffer(referenceId));
    1097           0 :     if (pElementCollector != NULL)
    1098             :     {
    1099           0 :         nId = m_nNextElementMarkId;
    1100           0 :         m_nNextElementMarkId ++;
    1101             : 
    1102             :         ElementCollector* pClonedOne
    1103           0 :             = pElementCollector->clone(nId, priority);
    1104             : 
    1105             :             /*
    1106             :              * add this EC into the security data buffer array.
    1107             :              */
    1108           0 :         m_vElementMarkBuffers.push_back(pClonedOne);
    1109             : 
    1110             :             /*
    1111             :              * If the reference EC is still in initial EC array, add
    1112             :              * this cloned one into the initial EC array too.
    1113             :              */
    1114           0 :             if (pElementCollector->getBufferNode() == NULL)
    1115             :         {
    1116           0 :             m_vNewElementCollectors.push_back(pClonedOne);
    1117             :         }
    1118             :     }
    1119             : 
    1120           0 :     return nId;
    1121             : }
    1122             : 
    1123           0 : void SAL_CALL SAXEventKeeperImpl::setSecurityId( sal_Int32 id, sal_Int32 securityId )
    1124             :     throw (cssu::RuntimeException, std::exception)
    1125             : {
    1126           0 :     ElementMark* pElementMark = findElementMarkBuffer(id);
    1127           0 :     if (pElementMark != NULL)
    1128             :     {
    1129           0 :         pElementMark->setSecurityId(securityId);
    1130             :     }
    1131           0 : }
    1132             : 
    1133             : 
    1134             : /* XReferenceResolvedBroadcaster */
    1135           0 : void SAL_CALL SAXEventKeeperImpl::addReferenceResolvedListener(
    1136             :     sal_Int32 referenceId,
    1137             :     const cssu::Reference< cssxc::sax::XReferenceResolvedListener >& listener )
    1138             :     throw (cssu::RuntimeException, std::exception)
    1139             : {
    1140           0 :     ElementCollector* pElementCollector = static_cast<ElementCollector*>(findElementMarkBuffer(referenceId));
    1141           0 :     if (pElementCollector != NULL)
    1142             :     {
    1143           0 :         pElementCollector->setReferenceResolvedListener(listener);
    1144             :     }
    1145           0 : }
    1146             : 
    1147           0 : void SAL_CALL SAXEventKeeperImpl::removeReferenceResolvedListener(
    1148             :     sal_Int32 /*referenceId*/,
    1149             :     const cssu::Reference< cssxc::sax::XReferenceResolvedListener >&)
    1150             :     throw (cssu::RuntimeException, std::exception)
    1151             : {
    1152           0 : }
    1153             : 
    1154             : /* XSAXEventKeeperStatusChangeBroadcaster */
    1155           0 : void SAL_CALL SAXEventKeeperImpl::addSAXEventKeeperStatusChangeListener(
    1156             :     const cssu::Reference< cssxc::sax::XSAXEventKeeperStatusChangeListener >& listener )
    1157             :     throw (cssu::RuntimeException, std::exception)
    1158             : {
    1159           0 :     m_xSAXEventKeeperStatusChangeListener = listener;
    1160           0 : }
    1161             : 
    1162           0 : void SAL_CALL SAXEventKeeperImpl::removeSAXEventKeeperStatusChangeListener(
    1163             :     const cssu::Reference< cssxc::sax::XSAXEventKeeperStatusChangeListener >&)
    1164             :     throw (cssu::RuntimeException, std::exception)
    1165             : {
    1166           0 : }
    1167             : 
    1168             : /* XDocumentHandler */
    1169           0 : void SAL_CALL SAXEventKeeperImpl::startDocument(  )
    1170             :     throw (cssxs::SAXException, cssu::RuntimeException, std::exception)
    1171             : {
    1172           0 :     if ( m_xNextHandler.is())
    1173             :     {
    1174           0 :         m_xNextHandler->startDocument();
    1175             :     }
    1176           0 : }
    1177             : 
    1178           0 : void SAL_CALL SAXEventKeeperImpl::endDocument(  )
    1179             :     throw (cssxs::SAXException, cssu::RuntimeException, std::exception)
    1180             : {
    1181           0 :     if ( m_xNextHandler.is())
    1182             :     {
    1183           0 :         m_xNextHandler->endDocument();
    1184             :     }
    1185           0 : }
    1186             : 
    1187           0 : void SAL_CALL SAXEventKeeperImpl::startElement(
    1188             :     const OUString& aName,
    1189             :     const cssu::Reference< cssxs::XAttributeList >& xAttribs )
    1190             :     throw (cssxs::SAXException, cssu::RuntimeException, std::exception)
    1191             : {
    1192             :         /*
    1193             :          * If there is a following handler and no blocking now, then
    1194             :          * forward this event
    1195             :          */
    1196           0 :     if ((m_pCurrentBlockingBufferNode == NULL) &&
    1197           0 :         (m_xNextHandler.is()) &&
    1198           0 :         (!m_bIsForwarding) &&
    1199           0 :         (m_pNewBlocker == NULL))
    1200             :     {
    1201           0 :         m_xNextHandler->startElement(aName, xAttribs);
    1202             :     }
    1203             : 
    1204             :         /*
    1205             :          * If not forwarding, buffer this startElement.
    1206             :          */
    1207           0 :            if (!m_bIsForwarding)
    1208             :            {
    1209             :     #ifndef _USECOMPRESSEDDOCUMENTHANDLER
    1210             :         m_xDocumentHandler->startElement(aName, xAttribs);
    1211             :     #else
    1212           0 :         sal_Int32 nLength = xAttribs->getLength();
    1213           0 :         cssu::Sequence< cssxcsax::XMLAttribute > aAttributes (nLength);
    1214             : 
    1215           0 :         for ( int i = 0; i<nLength; ++i )
    1216             :         {
    1217           0 :             aAttributes[i].sName = xAttribs->getNameByIndex((short)i);
    1218           0 :             aAttributes[i].sValue =xAttribs->getValueByIndex((short)i);
    1219             :         }
    1220             : 
    1221           0 :         m_xCompressedDocumentHandler->compressedStartElement(aName, aAttributes);
    1222             :     #endif
    1223             : 
    1224             :     }
    1225             : 
    1226           0 :     BufferNode* pBufferNode = addNewElementMarkBuffers();
    1227           0 :         if (pBufferNode != NULL)
    1228             :         {
    1229           0 :         setCurrentBufferNode(pBufferNode);
    1230             :     }
    1231           0 : }
    1232             : 
    1233           0 : void SAL_CALL SAXEventKeeperImpl::endElement( const OUString& aName )
    1234             :     throw (cssxs::SAXException, cssu::RuntimeException, std::exception)
    1235             : {
    1236           0 :         bool bIsCurrent = m_xXMLDocument->isCurrent(m_pCurrentBufferNode->getXMLElement());
    1237             : 
    1238             :         /*
    1239             :          * If there is a following handler and no blocking now, then
    1240             :          * forward this event
    1241             :          */
    1242           0 :     if ((m_pCurrentBlockingBufferNode == NULL) &&
    1243           0 :         (m_xNextHandler.is()) &&
    1244           0 :         (!m_bIsForwarding))
    1245             :     {
    1246           0 :         m_xNextHandler->endElement(aName);
    1247             :     }
    1248             : 
    1249           0 :     if ((m_pCurrentBlockingBufferNode != NULL) ||
    1250           0 :         (m_pCurrentBufferNode != m_pRootBufferNode) ||
    1251           0 :         (!m_xXMLDocument->isCurrentElementEmpty()))
    1252             :     {
    1253           0 :             if (!m_bIsForwarding)
    1254             :             {
    1255             :         #ifndef _USECOMPRESSEDDOCUMENTHANDLER
    1256             :             m_xDocumentHandler->endElement(aName);
    1257             :         #else
    1258           0 :             m_xCompressedDocumentHandler->compressedEndElement(aName);
    1259             :         #endif
    1260             :         }
    1261             : 
    1262             :         /*
    1263             :         * If the current buffer node has not notified yet, and
    1264             :         * the current buffer node is waiting for the current element,
    1265             :         * then let it notify.
    1266             :         */
    1267           0 :            if (bIsCurrent && (m_pCurrentBufferNode != m_pRootBufferNode))
    1268             :         {
    1269           0 :             BufferNode* pOldCurrentBufferNode = m_pCurrentBufferNode;
    1270           0 :             m_pCurrentBufferNode = (BufferNode*)m_pCurrentBufferNode->getParent();
    1271             : 
    1272           0 :             pOldCurrentBufferNode->setReceivedAll();
    1273             : 
    1274           0 :             if ((m_pCurrentBufferNode == m_pRootBufferNode) &&
    1275           0 :                 m_xSAXEventKeeperStatusChangeListener.is())
    1276             :             {
    1277           0 :                 m_xSAXEventKeeperStatusChangeListener->collectionStatusChanged(sal_False);
    1278             :             }
    1279             :         }
    1280             :     }
    1281             :     else
    1282             :     {
    1283           0 :         if (!m_bIsForwarding)
    1284             :         {
    1285           0 :             m_xXMLDocument->removeCurrentElement();
    1286             :         }
    1287             :     }
    1288           0 : }
    1289             : 
    1290           0 : void SAL_CALL SAXEventKeeperImpl::characters( const OUString& aChars )
    1291             :     throw (cssxs::SAXException, cssu::RuntimeException, std::exception)
    1292             : {
    1293           0 :     if (!m_bIsForwarding)
    1294             :     {
    1295           0 :         if ((m_pCurrentBlockingBufferNode == NULL) && m_xNextHandler.is())
    1296             :         {
    1297           0 :             m_xNextHandler->characters(aChars);
    1298             :         }
    1299             : 
    1300           0 :         if ((m_pCurrentBlockingBufferNode != NULL) ||
    1301           0 :             (m_pCurrentBufferNode != m_pRootBufferNode))
    1302             :         {
    1303             :         #ifndef _USECOMPRESSEDDOCUMENTHANDLER
    1304             :                 m_xDocumentHandler->characters(aChars);
    1305             :         #else
    1306           0 :             m_xCompressedDocumentHandler->compressedCharacters(aChars);
    1307             :         #endif
    1308             :             }
    1309             :         }
    1310           0 : }
    1311             : 
    1312           0 : void SAL_CALL SAXEventKeeperImpl::ignorableWhitespace( const OUString& aWhitespaces )
    1313             :     throw (cssxs::SAXException, cssu::RuntimeException, std::exception)
    1314             : {
    1315           0 :     characters( aWhitespaces );
    1316           0 : }
    1317             : 
    1318           0 : void SAL_CALL SAXEventKeeperImpl::processingInstruction(
    1319             :     const OUString& aTarget, const OUString& aData )
    1320             :     throw (cssxs::SAXException, cssu::RuntimeException, std::exception)
    1321             : {
    1322           0 :     if (!m_bIsForwarding)
    1323             :     {
    1324           0 :         if ((m_pCurrentBlockingBufferNode == NULL) && m_xNextHandler.is())
    1325             :         {
    1326           0 :             m_xNextHandler->processingInstruction(aTarget, aData);
    1327             :         }
    1328             : 
    1329           0 :         if ((m_pCurrentBlockingBufferNode != NULL) ||
    1330           0 :             (m_pCurrentBufferNode != m_pRootBufferNode))
    1331             :         {
    1332             :         #ifndef _USECOMPRESSEDDOCUMENTHANDLER
    1333             :             m_xDocumentHandler->processingInstruction(aTarget, aData);
    1334             :         #else
    1335           0 :             m_xCompressedDocumentHandler->compressedProcessingInstruction(aTarget, aData);
    1336             :         #endif
    1337             :             }
    1338             :         }
    1339           0 : }
    1340             : 
    1341           0 : void SAL_CALL SAXEventKeeperImpl::setDocumentLocator( const cssu::Reference< cssxs::XLocator >&)
    1342             :     throw (cssxs::SAXException, cssu::RuntimeException, std::exception)
    1343             : {
    1344           0 : }
    1345             : 
    1346             : /* XInitialization */
    1347           0 : void SAL_CALL SAXEventKeeperImpl::initialize( const cssu::Sequence< cssu::Any >& aArguments )
    1348             :     throw (cssu::Exception, cssu::RuntimeException, std::exception)
    1349             : {
    1350             :     OSL_ASSERT(aArguments.getLength() == 1);
    1351             : 
    1352           0 :     aArguments[0] >>= m_xXMLDocument;
    1353           0 :     m_xDocumentHandler = cssu::Reference< cssxs::XDocumentHandler >(
    1354           0 :         m_xXMLDocument, cssu::UNO_QUERY );
    1355           0 :     m_xCompressedDocumentHandler = cssu::Reference< cssxcsax::XCompressedDocumentHandler >(
    1356           0 :         m_xXMLDocument, cssu::UNO_QUERY );
    1357             : 
    1358           0 :     m_pRootBufferNode = new BufferNode(m_xXMLDocument->getCurrentElement());
    1359           0 :     m_pCurrentBufferNode = m_pRootBufferNode;
    1360           0 : }
    1361             : 
    1362           4 : OUString SAXEventKeeperImpl_getImplementationName ()
    1363             :     throw (cssu::RuntimeException)
    1364             : {
    1365           4 :     return OUString ( IMPLEMENTATION_NAME );
    1366             : }
    1367             : 
    1368           0 : cssu::Sequence< OUString > SAL_CALL SAXEventKeeperImpl_getSupportedServiceNames(  )
    1369             :     throw (cssu::RuntimeException)
    1370             : {
    1371           0 :     cssu::Sequence < OUString > aRet(1);
    1372           0 :     OUString* pArray = aRet.getArray();
    1373           0 :     pArray[0] =  OUString ( SERVICE_NAME );
    1374           0 :     return aRet;
    1375             : }
    1376             : #undef SERVICE_NAME
    1377             : 
    1378           0 : cssu::Reference< cssu::XInterface > SAL_CALL SAXEventKeeperImpl_createInstance(
    1379             :     const cssu::Reference< cssl::XMultiServiceFactory > &)
    1380             :     throw( cssu::Exception )
    1381             : {
    1382           0 :     return (cppu::OWeakObject*) new SAXEventKeeperImpl();
    1383             : }
    1384             : 
    1385             : /* XServiceInfo */
    1386           0 : OUString SAL_CALL SAXEventKeeperImpl::getImplementationName(  )
    1387             :     throw (cssu::RuntimeException, std::exception)
    1388             : {
    1389           0 :     return SAXEventKeeperImpl_getImplementationName();
    1390             : }
    1391             : 
    1392           0 : sal_Bool SAL_CALL SAXEventKeeperImpl::supportsService( const OUString& rServiceName )
    1393             :     throw (cssu::RuntimeException, std::exception)
    1394             : {
    1395           0 :     return cppu::supportsService(this, rServiceName);
    1396             : }
    1397             : 
    1398           0 : cssu::Sequence< OUString > SAL_CALL SAXEventKeeperImpl::getSupportedServiceNames(  )
    1399             :     throw (cssu::RuntimeException, std::exception)
    1400             : {
    1401           0 :     return SAXEventKeeperImpl_getSupportedServiceNames();
    1402             : }
    1403             : 
    1404             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
 |