forked from mirrors/gecko-dev
		
	 03c8e8c2dd
			
		
	
	
		03c8e8c2dd
		
	
	
	
	
		
			
			clang-format-8 upstream had some improvements wrt macros See: https://reviews.llvm.org/D33440 This is why the diff is bigger than usual # ignore-this-changeset Differential Revision: https://phabricator.services.mozilla.com/D26098 --HG-- extra : moz-landing-system : lando
		
			
				
	
	
		
			164 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			164 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 | |
| /* This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | |
| 
 | |
| /*
 | |
|  * XML utility classes
 | |
|  */
 | |
| 
 | |
| #include "txXMLUtils.h"
 | |
| #include "nsString.h"
 | |
| #include "nsReadableUtils.h"
 | |
| #include "nsGkAtoms.h"
 | |
| #include "txStringUtils.h"
 | |
| #include "txNamespaceMap.h"
 | |
| #include "txXPathTreeWalker.h"
 | |
| #include "nsContentUtils.h"
 | |
| 
 | |
| //------------------------------/
 | |
| //- Implementation of XMLUtils -/
 | |
| //------------------------------/
 | |
| 
 | |
| // static
 | |
| nsresult XMLUtils::splitExpatName(const char16_t* aExpatName, nsAtom** aPrefix,
 | |
|                                   nsAtom** aLocalName, int32_t* aNameSpaceID) {
 | |
|   /**
 | |
|    *  Expat can send the following:
 | |
|    *    localName
 | |
|    *    namespaceURI<separator>localName
 | |
|    *    namespaceURI<separator>localName<separator>prefix
 | |
|    */
 | |
| 
 | |
|   const char16_t* uriEnd = nullptr;
 | |
|   const char16_t* nameEnd = nullptr;
 | |
|   const char16_t* pos;
 | |
|   for (pos = aExpatName; *pos; ++pos) {
 | |
|     if (*pos == kExpatSeparatorChar) {
 | |
|       if (uriEnd) {
 | |
|         nameEnd = pos;
 | |
|       } else {
 | |
|         uriEnd = pos;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const char16_t* nameStart;
 | |
|   if (uriEnd) {
 | |
|     *aNameSpaceID = txNamespaceManager::getNamespaceID(
 | |
|         nsDependentSubstring(aExpatName, uriEnd));
 | |
|     if (*aNameSpaceID == kNameSpaceID_Unknown) {
 | |
|       return NS_ERROR_FAILURE;
 | |
|     }
 | |
| 
 | |
|     nameStart = (uriEnd + 1);
 | |
|     if (nameEnd) {
 | |
|       const char16_t* prefixStart = nameEnd + 1;
 | |
|       *aPrefix = NS_Atomize(Substring(prefixStart, pos)).take();
 | |
|       if (!*aPrefix) {
 | |
|         return NS_ERROR_OUT_OF_MEMORY;
 | |
|       }
 | |
|     } else {
 | |
|       nameEnd = pos;
 | |
|       *aPrefix = nullptr;
 | |
|     }
 | |
|   } else {
 | |
|     *aNameSpaceID = kNameSpaceID_None;
 | |
|     nameStart = aExpatName;
 | |
|     nameEnd = pos;
 | |
|     *aPrefix = nullptr;
 | |
|   }
 | |
| 
 | |
|   *aLocalName = NS_Atomize(Substring(nameStart, nameEnd)).take();
 | |
| 
 | |
|   return *aLocalName ? NS_OK : NS_ERROR_OUT_OF_MEMORY;
 | |
| }
 | |
| 
 | |
| nsresult XMLUtils::splitQName(const nsAString& aName, nsAtom** aPrefix,
 | |
|                               nsAtom** aLocalName) {
 | |
|   const char16_t* colon;
 | |
|   bool valid = XMLUtils::isValidQName(aName, &colon);
 | |
|   if (!valid) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   if (colon) {
 | |
|     const char16_t* end;
 | |
|     aName.EndReading(end);
 | |
| 
 | |
|     *aPrefix = NS_Atomize(Substring(aName.BeginReading(), colon)).take();
 | |
|     *aLocalName = NS_Atomize(Substring(colon + 1, end)).take();
 | |
|   } else {
 | |
|     *aPrefix = nullptr;
 | |
|     *aLocalName = NS_Atomize(aName).take();
 | |
|   }
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Returns true if the given string has only whitespace characters
 | |
|  */
 | |
| bool XMLUtils::isWhitespace(const nsAString& aText) {
 | |
|   nsString::const_char_iterator start, end;
 | |
|   aText.BeginReading(start);
 | |
|   aText.EndReading(end);
 | |
|   for (; start != end; ++start) {
 | |
|     if (!isWhitespace(*start)) {
 | |
|       return false;
 | |
|     }
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Normalizes the value of a XML processing instruction
 | |
|  **/
 | |
| void XMLUtils::normalizePIValue(nsAString& piValue) {
 | |
|   nsAutoString origValue(piValue);
 | |
|   uint32_t origLength = origValue.Length();
 | |
|   uint32_t conversionLoop = 0;
 | |
|   char16_t prevCh = 0;
 | |
|   piValue.Truncate();
 | |
| 
 | |
|   while (conversionLoop < origLength) {
 | |
|     char16_t ch = origValue.CharAt(conversionLoop);
 | |
|     switch (ch) {
 | |
|       case '>': {
 | |
|         if (prevCh == '?') {
 | |
|           piValue.Append(char16_t(' '));
 | |
|         }
 | |
|         break;
 | |
|       }
 | |
|       default: {
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|     piValue.Append(ch);
 | |
|     prevCh = ch;
 | |
|     ++conversionLoop;
 | |
|   }
 | |
| }
 | |
| 
 | |
| // static
 | |
| bool XMLUtils::isValidQName(const nsAString& aQName, const char16_t** aColon) {
 | |
|   return NS_SUCCEEDED(nsContentUtils::CheckQName(aQName, true, aColon));
 | |
| }
 | |
| 
 | |
| // static
 | |
| bool XMLUtils::getXMLSpacePreserve(const txXPathNode& aNode) {
 | |
|   nsAutoString value;
 | |
|   txXPathTreeWalker walker(aNode);
 | |
|   do {
 | |
|     if (walker.getAttr(nsGkAtoms::space, kNameSpaceID_XML, value)) {
 | |
|       if (TX_StringEqualsAtom(value, nsGkAtoms::preserve)) {
 | |
|         return true;
 | |
|       }
 | |
|       if (TX_StringEqualsAtom(value, nsGkAtoms::_default)) {
 | |
|         return false;
 | |
|       }
 | |
|     }
 | |
|   } while (walker.moveToParent());
 | |
| 
 | |
|   return false;
 | |
| }
 |