1*e9cbe144SAndrew Rist /************************************************************** 2*e9cbe144SAndrew Rist * 3*e9cbe144SAndrew Rist * Licensed to the Apache Software Foundation (ASF) under one 4*e9cbe144SAndrew Rist * or more contributor license agreements. See the NOTICE file 5*e9cbe144SAndrew Rist * distributed with this work for additional information 6*e9cbe144SAndrew Rist * regarding copyright ownership. The ASF licenses this file 7*e9cbe144SAndrew Rist * to you under the Apache License, Version 2.0 (the 8*e9cbe144SAndrew Rist * "License"); you may not use this file except in compliance 9*e9cbe144SAndrew Rist * with the License. You may obtain a copy of the License at 10*e9cbe144SAndrew Rist * 11*e9cbe144SAndrew Rist * http://www.apache.org/licenses/LICENSE-2.0 12*e9cbe144SAndrew Rist * 13*e9cbe144SAndrew Rist * Unless required by applicable law or agreed to in writing, 14*e9cbe144SAndrew Rist * software distributed under the License is distributed on an 15*e9cbe144SAndrew Rist * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16*e9cbe144SAndrew Rist * KIND, either express or implied. See the License for the 17*e9cbe144SAndrew Rist * specific language governing permissions and limitations 18*e9cbe144SAndrew Rist * under the License. 19*e9cbe144SAndrew Rist * 20*e9cbe144SAndrew Rist *************************************************************/ 21*e9cbe144SAndrew Rist 22*e9cbe144SAndrew Rist 23cdf0e10cSrcweir 24cdf0e10cSrcweir #include "elementlist.hxx" 25cdf0e10cSrcweir 26cdf0e10cSrcweir #include <string.h> 27cdf0e10cSrcweir 28cdf0e10cSrcweir #include <element.hxx> 29cdf0e10cSrcweir #include <document.hxx> 30cdf0e10cSrcweir 31cdf0e10cSrcweir 32cdf0e10cSrcweir namespace DOM 33cdf0e10cSrcweir { 34cdf0e10cSrcweir lcl_initXmlString(::rtl::OUString const & rString)35cdf0e10cSrcweir static xmlChar* lcl_initXmlString(::rtl::OUString const& rString) 36cdf0e10cSrcweir { 37cdf0e10cSrcweir ::rtl::OString const os = 38cdf0e10cSrcweir ::rtl::OUStringToOString(rString, RTL_TEXTENCODING_UTF8); 39cdf0e10cSrcweir xmlChar *const pRet = new xmlChar[os.getLength() + 1]; 40cdf0e10cSrcweir strcpy(reinterpret_cast<char*>(pRet), os.getStr()); 41cdf0e10cSrcweir return pRet; 42cdf0e10cSrcweir } 43cdf0e10cSrcweir CElementList(::rtl::Reference<CElement> const & pElement,::osl::Mutex & rMutex,OUString const & rName,OUString const * const pURI)44cdf0e10cSrcweir CElementList::CElementList(::rtl::Reference<CElement> const& pElement, 45cdf0e10cSrcweir ::osl::Mutex & rMutex, 46cdf0e10cSrcweir OUString const& rName, OUString const*const pURI) 47cdf0e10cSrcweir : m_pElement(pElement) 48cdf0e10cSrcweir , m_rMutex(rMutex) 49cdf0e10cSrcweir , m_pName(lcl_initXmlString(rName)) 50cdf0e10cSrcweir , m_pURI((pURI) ? lcl_initXmlString(*pURI) : 0) 51cdf0e10cSrcweir , m_bRebuild(true) 52cdf0e10cSrcweir { 53cdf0e10cSrcweir if (m_pElement.is()) { 54cdf0e10cSrcweir registerListener(*m_pElement); 55cdf0e10cSrcweir } 56cdf0e10cSrcweir } 57cdf0e10cSrcweir registerListener(CElement & rElement)58cdf0e10cSrcweir void CElementList::registerListener(CElement & rElement) 59cdf0e10cSrcweir { 60cdf0e10cSrcweir try { 61cdf0e10cSrcweir Reference< XEventTarget > const xTarget( 62cdf0e10cSrcweir static_cast<XElement*>(& rElement), UNO_QUERY_THROW); 63cdf0e10cSrcweir OUString aType = OUString::createFromAscii("DOMSubtreeModified"); 64cdf0e10cSrcweir sal_Bool capture = sal_False; 65cdf0e10cSrcweir xTarget->addEventListener(aType, 66cdf0e10cSrcweir Reference< XEventListener >(this), capture); 67cdf0e10cSrcweir } catch (Exception &e){ 68cdf0e10cSrcweir OString aMsg("Exception caught while registering NodeList as listener:\n"); 69cdf0e10cSrcweir aMsg += OUStringToOString(e.Message, RTL_TEXTENCODING_ASCII_US); 70cdf0e10cSrcweir OSL_ENSURE(sal_False, aMsg.getStr()); 71cdf0e10cSrcweir } 72cdf0e10cSrcweir } 73cdf0e10cSrcweir buildlist(xmlNodePtr pNode,sal_Bool start)74cdf0e10cSrcweir void CElementList::buildlist(xmlNodePtr pNode, sal_Bool start) 75cdf0e10cSrcweir { 76cdf0e10cSrcweir // bail out if no rebuild is needed 77cdf0e10cSrcweir if (start) { 78cdf0e10cSrcweir if (!m_bRebuild) 79cdf0e10cSrcweir { 80cdf0e10cSrcweir return; 81cdf0e10cSrcweir } else { 82cdf0e10cSrcweir m_nodevector.erase(m_nodevector.begin(), m_nodevector.end()); 83cdf0e10cSrcweir m_bRebuild = false; // don't rebuild until tree is mutated 84cdf0e10cSrcweir } 85cdf0e10cSrcweir } 86cdf0e10cSrcweir 87cdf0e10cSrcweir while (pNode != NULL ) 88cdf0e10cSrcweir { 89cdf0e10cSrcweir if (pNode->type == XML_ELEMENT_NODE && 90cdf0e10cSrcweir (strcmp((char*)pNode->name, (char*)m_pName.get()) == 0)) 91cdf0e10cSrcweir { 92cdf0e10cSrcweir if (!m_pURI) { 93cdf0e10cSrcweir m_nodevector.push_back(pNode); 94cdf0e10cSrcweir } else { 95cdf0e10cSrcweir if (pNode->ns != NULL && (0 == 96cdf0e10cSrcweir strcmp((char*)pNode->ns->href, (char*)m_pURI.get()))) 97cdf0e10cSrcweir { 98cdf0e10cSrcweir m_nodevector.push_back(pNode); 99cdf0e10cSrcweir } 100cdf0e10cSrcweir } 101cdf0e10cSrcweir } 102cdf0e10cSrcweir if (pNode->children != NULL) buildlist(pNode->children, sal_False); 103cdf0e10cSrcweir 104cdf0e10cSrcweir if (!start) pNode = pNode->next; 105cdf0e10cSrcweir else break; // fold back 106cdf0e10cSrcweir } 107cdf0e10cSrcweir } 108cdf0e10cSrcweir 109cdf0e10cSrcweir /** 110cdf0e10cSrcweir The number of nodes in the list. 111cdf0e10cSrcweir */ getLength()112cdf0e10cSrcweir sal_Int32 SAL_CALL CElementList::getLength() throw (RuntimeException) 113cdf0e10cSrcweir { 114cdf0e10cSrcweir ::osl::MutexGuard const g(m_rMutex); 115cdf0e10cSrcweir 116cdf0e10cSrcweir if (!m_pElement.is()) { return 0; } 117cdf0e10cSrcweir 118cdf0e10cSrcweir // this has to be 'live' 119cdf0e10cSrcweir buildlist(m_pElement->GetNodePtr()); 120cdf0e10cSrcweir return m_nodevector.size(); 121cdf0e10cSrcweir } 122cdf0e10cSrcweir /** 123cdf0e10cSrcweir Returns the indexth item in the collection. 124cdf0e10cSrcweir */ item(sal_Int32 index)125cdf0e10cSrcweir Reference< XNode > SAL_CALL CElementList::item(sal_Int32 index) 126cdf0e10cSrcweir throw (RuntimeException) 127cdf0e10cSrcweir { 128cdf0e10cSrcweir if (index < 0) throw RuntimeException(); 129cdf0e10cSrcweir 130cdf0e10cSrcweir ::osl::MutexGuard const g(m_rMutex); 131cdf0e10cSrcweir 132cdf0e10cSrcweir if (!m_pElement.is()) { return 0; } 133cdf0e10cSrcweir 134cdf0e10cSrcweir buildlist(m_pElement->GetNodePtr()); 135cdf0e10cSrcweir if (m_nodevector.size() <= static_cast<size_t>(index)) { 136cdf0e10cSrcweir throw RuntimeException(); 137cdf0e10cSrcweir } 138cdf0e10cSrcweir Reference< XNode > const xRet( 139cdf0e10cSrcweir m_pElement->GetOwnerDocument().GetCNode(m_nodevector[index]).get()); 140cdf0e10cSrcweir return xRet; 141cdf0e10cSrcweir } 142cdf0e10cSrcweir 143cdf0e10cSrcweir // tree mutations can change the list handleEvent(Reference<XEvent> const &)144cdf0e10cSrcweir void SAL_CALL CElementList::handleEvent(Reference< XEvent > const&) 145cdf0e10cSrcweir throw (RuntimeException) 146cdf0e10cSrcweir { 147cdf0e10cSrcweir ::osl::MutexGuard const g(m_rMutex); 148cdf0e10cSrcweir 149cdf0e10cSrcweir m_bRebuild = true; 150cdf0e10cSrcweir } 151cdf0e10cSrcweir } 152