diff options
author | Matt Mower <mowerm@gmail.com> | 2016-02-10 07:19:34 +0100 |
---|---|---|
committer | Dees Troy <dees_troy@teamw.in> | 2016-02-29 17:14:17 +0100 |
commit | 64ffe730a663dda9dec46b7165209cdedaac06e9 (patch) | |
tree | bc5e503af779df57d72dc144e896192830d1c7fd /gui | |
parent | minuitwrp: retry opening fb0 if it failed (diff) | |
download | android_bootable_recovery-64ffe730a663dda9dec46b7165209cdedaac06e9.tar android_bootable_recovery-64ffe730a663dda9dec46b7165209cdedaac06e9.tar.gz android_bootable_recovery-64ffe730a663dda9dec46b7165209cdedaac06e9.tar.bz2 android_bootable_recovery-64ffe730a663dda9dec46b7165209cdedaac06e9.tar.lz android_bootable_recovery-64ffe730a663dda9dec46b7165209cdedaac06e9.tar.xz android_bootable_recovery-64ffe730a663dda9dec46b7165209cdedaac06e9.tar.zst android_bootable_recovery-64ffe730a663dda9dec46b7165209cdedaac06e9.zip |
Diffstat (limited to 'gui')
-rw-r--r-- | gui/rapidxml.hpp | 5232 | ||||
-rw-r--r--[-rwxr-xr-x] | gui/theme/common/languages/de.xml | 1338 | ||||
-rw-r--r--[-rwxr-xr-x] | gui/theme/common/languages/ru.xml | 0 | ||||
-rw-r--r-- | gui/theme/landscape_hdpi/splash.xml | 102 | ||||
-rw-r--r-- | gui/theme/landscape_mdpi/splash.xml | 102 | ||||
-rw-r--r-- | gui/theme/portrait_mdpi/splash.xml | 102 | ||||
-rw-r--r-- | gui/theme/watch_mdpi/splash.xml | 102 |
7 files changed, 3489 insertions, 3489 deletions
diff --git a/gui/rapidxml.hpp b/gui/rapidxml.hpp index c0a7be44e..33a886014 100644 --- a/gui/rapidxml.hpp +++ b/gui/rapidxml.hpp @@ -1,2616 +1,2616 @@ -#ifndef RAPIDXML_HPP_INCLUDED
-#define RAPIDXML_HPP_INCLUDED
-
-#define RAPIDXML_NO_EXCEPTIONS
-
-// Copyright (C) 2006, 2009 Marcin Kalicinski
-// Version 1.13
-// Revision $DateTime: 2009/05/13 01:46:17 $
-//! \file rapidxml.hpp This file contains rapidxml parser and DOM implementation
-
-// If standard library is disabled, user must provide implementations of required functions and typedefs
-#if !defined(RAPIDXML_NO_STDLIB)
- #include <cstdlib> // For std::size_t
- #include <cassert> // For assert
- #include <new> // For placement new
-#endif
-
-// On MSVC, disable "conditional expression is constant" warning (level 4).
-// This warning is almost impossible to avoid with certain types of templated code
-#ifdef _MSC_VER
- #pragma warning(push)
- #pragma warning(disable:4127) // Conditional expression is constant
-#endif
-
-///////////////////////////////////////////////////////////////////////////
-// RAPIDXML_PARSE_ERROR
-
-#if defined(RAPIDXML_NO_EXCEPTIONS)
-
-#define RAPIDXML_PARSE_ERROR(what, where) { parse_error_handler(what, where); assert(0); }
-
-namespace rapidxml
-{
- //! When exceptions are disabled by defining RAPIDXML_NO_EXCEPTIONS,
- //! this function is called to notify user about the error.
- //! It must be defined by the user.
- //! <br><br>
- //! This function cannot return. If it does, the results are undefined.
- //! <br><br>
- //! A very simple definition might look like that:
- //! <pre>
- //! void %rapidxml::%parse_error_handler(const char *what, void *where)
- //! {
- //! std::cout << "Parse error: " << what << "\n";
- //! std::abort();
- //! }
- //! </pre>
- //! \param what Human readable description of the error.
- //! \param where Pointer to character data where error was detected.
- void parse_error_handler(const char *what, void *where);
-}
-
-#else
-
-#include <exception> // For std::exception
-
-#define RAPIDXML_PARSE_ERROR(what, where) throw parse_error(what, where)
-
-namespace rapidxml
-{
-
- //! Parse error exception.
- //! This exception is thrown by the parser when an error occurs.
- //! Use what() function to get human-readable error message.
- //! Use where() function to get a pointer to position within source text where error was detected.
- //! <br><br>
- //! If throwing exceptions by the parser is undesirable,
- //! it can be disabled by defining RAPIDXML_NO_EXCEPTIONS macro before rapidxml.hpp is included.
- //! This will cause the parser to call rapidxml::parse_error_handler() function instead of throwing an exception.
- //! This function must be defined by the user.
- //! <br><br>
- //! This class derives from <code>std::exception</code> class.
- class parse_error: public std::exception
- {
-
- public:
-
- //! Constructs parse error
- parse_error(const char *what, void *where)
- : m_what(what)
- , m_where(where)
- {
- }
-
- //! Gets human readable description of error.
- //! \return Pointer to null terminated description of the error.
- virtual const char *what() const throw()
- {
- return m_what;
- }
-
- //! Gets pointer to character data where error happened.
- //! Ch should be the same as char type of xml_document that produced the error.
- //! \return Pointer to location within the parsed string where error occured.
- template<class Ch>
- Ch *where() const
- {
- return reinterpret_cast<Ch *>(m_where);
- }
-
- private:
-
- const char *m_what;
- void *m_where;
-
- };
-}
-
-#endif
-
-///////////////////////////////////////////////////////////////////////////
-// Pool sizes
-
-#ifndef RAPIDXML_STATIC_POOL_SIZE
- // Size of static memory block of memory_pool.
- // Define RAPIDXML_STATIC_POOL_SIZE before including rapidxml.hpp if you want to override the default value.
- // No dynamic memory allocations are performed by memory_pool until static memory is exhausted.
- #define RAPIDXML_STATIC_POOL_SIZE (64 * 1024)
-#endif
-
-#ifndef RAPIDXML_DYNAMIC_POOL_SIZE
- // Size of dynamic memory block of memory_pool.
- // Define RAPIDXML_DYNAMIC_POOL_SIZE before including rapidxml.hpp if you want to override the default value.
- // After the static block is exhausted, dynamic blocks with approximately this size are allocated by memory_pool.
- #define RAPIDXML_DYNAMIC_POOL_SIZE (64 * 1024)
-#endif
-
-#ifndef RAPIDXML_ALIGNMENT
- // Memory allocation alignment.
- // Define RAPIDXML_ALIGNMENT before including rapidxml.hpp if you want to override the default value, which is the size of pointer.
- // All memory allocations for nodes, attributes and strings will be aligned to this value.
- // This must be a power of 2 and at least 1, otherwise memory_pool will not work.
- #define RAPIDXML_ALIGNMENT sizeof(void *)
-#endif
-
-namespace rapidxml
-{
- // Forward declarations
- template<class Ch> class xml_node;
- template<class Ch> class xml_attribute;
- template<class Ch> class xml_document;
-
- //! Enumeration listing all node types produced by the parser.
- //! Use xml_node::type() function to query node type.
- enum node_type
- {
- node_document, //!< A document node. Name and value are empty.
- node_element, //!< An element node. Name contains element name. Value contains text of first data node.
- node_data, //!< A data node. Name is empty. Value contains data text.
- node_cdata, //!< A CDATA node. Name is empty. Value contains data text.
- node_comment, //!< A comment node. Name is empty. Value contains comment text.
- node_declaration, //!< A declaration node. Name and value are empty. Declaration parameters (version, encoding and standalone) are in node attributes.
- node_doctype, //!< A DOCTYPE node. Name is empty. Value contains DOCTYPE text.
- node_pi //!< A PI node. Name contains target. Value contains instructions.
- };
-
- ///////////////////////////////////////////////////////////////////////
- // Parsing flags
-
- //! Parse flag instructing the parser to not create data nodes.
- //! Text of first data node will still be placed in value of parent element, unless rapidxml::parse_no_element_values flag is also specified.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_no_data_nodes = 0x1;
-
- //! Parse flag instructing the parser to not use text of first data node as a value of parent element.
- //! Can be combined with other flags by use of | operator.
- //! Note that child data nodes of element node take precendence over its value when printing.
- //! That is, if element has one or more child data nodes <em>and</em> a value, the value will be ignored.
- //! Use rapidxml::parse_no_data_nodes flag to prevent creation of data nodes if you want to manipulate data using values of elements.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_no_element_values = 0x2;
-
- //! Parse flag instructing the parser to not place zero terminators after strings in the source text.
- //! By default zero terminators are placed, modifying source text.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_no_string_terminators = 0x4;
-
- //! Parse flag instructing the parser to not translate entities in the source text.
- //! By default entities are translated, modifying source text.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_no_entity_translation = 0x8;
-
- //! Parse flag instructing the parser to disable UTF-8 handling and assume plain 8 bit characters.
- //! By default, UTF-8 handling is enabled.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_no_utf8 = 0x10;
-
- //! Parse flag instructing the parser to create XML declaration node.
- //! By default, declaration node is not created.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_declaration_node = 0x20;
-
- //! Parse flag instructing the parser to create comments nodes.
- //! By default, comment nodes are not created.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_comment_nodes = 0x40;
-
- //! Parse flag instructing the parser to create DOCTYPE node.
- //! By default, doctype node is not created.
- //! Although W3C specification allows at most one DOCTYPE node, RapidXml will silently accept documents with more than one.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_doctype_node = 0x80;
-
- //! Parse flag instructing the parser to create PI nodes.
- //! By default, PI nodes are not created.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_pi_nodes = 0x100;
-
- //! Parse flag instructing the parser to validate closing tag names.
- //! If not set, name inside closing tag is irrelevant to the parser.
- //! By default, closing tags are not validated.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_validate_closing_tags = 0x200;
-
- //! Parse flag instructing the parser to trim all leading and trailing whitespace of data nodes.
- //! By default, whitespace is not trimmed.
- //! This flag does not cause the parser to modify source text.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_trim_whitespace = 0x400;
-
- //! Parse flag instructing the parser to condense all whitespace runs of data nodes to a single space character.
- //! Trimming of leading and trailing whitespace of data is controlled by rapidxml::parse_trim_whitespace flag.
- //! By default, whitespace is not normalized.
- //! If this flag is specified, source text will be modified.
- //! Can be combined with other flags by use of | operator.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_normalize_whitespace = 0x800;
-
- // Compound flags
-
- //! Parse flags which represent default behaviour of the parser.
- //! This is always equal to 0, so that all other flags can be simply ored together.
- //! Normally there is no need to inconveniently disable flags by anding with their negated (~) values.
- //! This also means that meaning of each flag is a <i>negation</i> of the default setting.
- //! For example, if flag name is rapidxml::parse_no_utf8, it means that utf-8 is <i>enabled</i> by default,
- //! and using the flag will disable it.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_default = 0;
-
- //! A combination of parse flags that forbids any modifications of the source text.
- //! This also results in faster parsing. However, note that the following will occur:
- //! <ul>
- //! <li>names and values of nodes will not be zero terminated, you have to use xml_base::name_size() and xml_base::value_size() functions to determine where name and value ends</li>
- //! <li>entities will not be translated</li>
- //! <li>whitespace will not be normalized</li>
- //! </ul>
- //! See xml_document::parse() function.
- const int parse_non_destructive = parse_no_string_terminators | parse_no_entity_translation;
-
- //! A combination of parse flags resulting in fastest possible parsing, without sacrificing important data.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_fastest = parse_non_destructive | parse_no_data_nodes;
-
- //! A combination of parse flags resulting in largest amount of data being extracted.
- //! This usually results in slowest parsing.
- //! <br><br>
- //! See xml_document::parse() function.
- const int parse_full = parse_declaration_node | parse_comment_nodes | parse_doctype_node | parse_pi_nodes | parse_validate_closing_tags;
-
- ///////////////////////////////////////////////////////////////////////
- // Internals
-
- //! \cond internal
- namespace internal
- {
-
- // Struct that contains lookup tables for the parser
- // It must be a template to allow correct linking (because it has static data members, which are defined in a header file).
- template<int Dummy>
- struct lookup_tables
- {
- static const unsigned char lookup_whitespace[256]; // Whitespace table
- static const unsigned char lookup_node_name[256]; // Node name table
- static const unsigned char lookup_text[256]; // Text table
- static const unsigned char lookup_text_pure_no_ws[256]; // Text table
- static const unsigned char lookup_text_pure_with_ws[256]; // Text table
- static const unsigned char lookup_attribute_name[256]; // Attribute name table
- static const unsigned char lookup_attribute_data_1[256]; // Attribute data table with single quote
- static const unsigned char lookup_attribute_data_1_pure[256]; // Attribute data table with single quote
- static const unsigned char lookup_attribute_data_2[256]; // Attribute data table with double quotes
- static const unsigned char lookup_attribute_data_2_pure[256]; // Attribute data table with double quotes
- static const unsigned char lookup_digits[256]; // Digits
- static const unsigned char lookup_upcase[256]; // To uppercase conversion table for ASCII characters
- };
-
- // Find length of the string
- template<class Ch>
- inline std::size_t measure(const Ch *p)
- {
- const Ch *tmp = p;
- while (*tmp)
- ++tmp;
- return tmp - p;
- }
-
- // Compare strings for equality
- template<class Ch>
- inline bool compare(const Ch *p1, std::size_t size1, const Ch *p2, std::size_t size2, bool case_sensitive)
- {
- if (size1 != size2)
- return false;
- if (case_sensitive)
- {
- for (const Ch *end = p1 + size1; p1 < end; ++p1, ++p2)
- if (*p1 != *p2)
- return false;
- }
- else
- {
- for (const Ch *end = p1 + size1; p1 < end; ++p1, ++p2)
- if (lookup_tables<0>::lookup_upcase[static_cast<unsigned char>(*p1)] != lookup_tables<0>::lookup_upcase[static_cast<unsigned char>(*p2)])
- return false;
- }
- return true;
- }
- }
- //! \endcond
-
- ///////////////////////////////////////////////////////////////////////
- // Memory pool
-
- //! This class is used by the parser to create new nodes and attributes, without overheads of dynamic memory allocation.
- //! In most cases, you will not need to use this class directly.
- //! However, if you need to create nodes manually or modify names/values of nodes,
- //! you are encouraged to use memory_pool of relevant xml_document to allocate the memory.
- //! Not only is this faster than allocating them by using <code>new</code> operator,
- //! but also their lifetime will be tied to the lifetime of document,
- //! possibly simplyfing memory management.
- //! <br><br>
- //! Call allocate_node() or allocate_attribute() functions to obtain new nodes or attributes from the pool.
- //! You can also call allocate_string() function to allocate strings.
- //! Such strings can then be used as names or values of nodes without worrying about their lifetime.
- //! Note that there is no <code>free()</code> function -- all allocations are freed at once when clear() function is called,
- //! or when the pool is destroyed.
- //! <br><br>
- //! It is also possible to create a standalone memory_pool, and use it
- //! to allocate nodes, whose lifetime will not be tied to any document.
- //! <br><br>
- //! Pool maintains <code>RAPIDXML_STATIC_POOL_SIZE</code> bytes of statically allocated memory.
- //! Until static memory is exhausted, no dynamic memory allocations are done.
- //! When static memory is exhausted, pool allocates additional blocks of memory of size <code>RAPIDXML_DYNAMIC_POOL_SIZE</code> each,
- //! by using global <code>new[]</code> and <code>delete[]</code> operators.
- //! This behaviour can be changed by setting custom allocation routines.
- //! Use set_allocator() function to set them.
- //! <br><br>
- //! Allocations for nodes, attributes and strings are aligned at <code>RAPIDXML_ALIGNMENT</code> bytes.
- //! This value defaults to the size of pointer on target architecture.
- //! <br><br>
- //! To obtain absolutely top performance from the parser,
- //! it is important that all nodes are allocated from a single, contiguous block of memory.
- //! Otherwise, cache misses when jumping between two (or more) disjoint blocks of memory can slow down parsing quite considerably.
- //! If required, you can tweak <code>RAPIDXML_STATIC_POOL_SIZE</code>, <code>RAPIDXML_DYNAMIC_POOL_SIZE</code> and <code>RAPIDXML_ALIGNMENT</code>
- //! to obtain best wasted memory to performance compromise.
- //! To do it, define their values before rapidxml.hpp file is included.
- //! \param Ch Character type of created nodes.
- template<class Ch = char>
- class memory_pool
- {
-
- public:
-
- //! \cond internal
- typedef void *(alloc_func)(std::size_t); // Type of user-defined function used to allocate memory
- typedef void (free_func)(void *); // Type of user-defined function used to free memory
- //! \endcond
-
- //! Constructs empty pool with default allocator functions.
- memory_pool()
- : m_alloc_func(0)
- , m_free_func(0)
- {
- init();
- }
-
- //! Destroys pool and frees all the memory.
- //! This causes memory occupied by nodes allocated by the pool to be freed.
- //! Nodes allocated from the pool are no longer valid.
- ~memory_pool()
- {
- clear();
- }
-
- //! Allocates a new node from the pool, and optionally assigns name and value to it.
- //! If the allocation request cannot be accomodated, this function will throw <code>std::bad_alloc</code>.
- //! If exceptions are disabled by defining RAPIDXML_NO_EXCEPTIONS, this function
- //! will call rapidxml::parse_error_handler() function.
- //! \param type Type of node to create.
- //! \param name Name to assign to the node, or 0 to assign no name.
- //! \param value Value to assign to the node, or 0 to assign no value.
- //! \param name_size Size of name to assign, or 0 to automatically calculate size from name string.
- //! \param value_size Size of value to assign, or 0 to automatically calculate size from value string.
- //! \return Pointer to allocated node. This pointer will never be NULL.
- xml_node<Ch> *allocate_node(node_type type,
- const Ch *name = 0, const Ch *value = 0,
- std::size_t name_size = 0, std::size_t value_size = 0)
- {
- void *memory = allocate_aligned(sizeof(xml_node<Ch>));
- xml_node<Ch> *node = new(memory) xml_node<Ch>(type);
- if (name)
- {
- if (name_size > 0)
- node->name(name, name_size);
- else
- node->name(name);
- }
- if (value)
- {
- if (value_size > 0)
- node->value(value, value_size);
- else
- node->value(value);
- }
- return node;
- }
-
- //! Allocates a new attribute from the pool, and optionally assigns name and value to it.
- //! If the allocation request cannot be accomodated, this function will throw <code>std::bad_alloc</code>.
- //! If exceptions are disabled by defining RAPIDXML_NO_EXCEPTIONS, this function
- //! will call rapidxml::parse_error_handler() function.
- //! \param name Name to assign to the attribute, or 0 to assign no name.
- //! \param value Value to assign to the attribute, or 0 to assign no value.
- //! \param name_size Size of name to assign, or 0 to automatically calculate size from name string.
- //! \param value_size Size of value to assign, or 0 to automatically calculate size from value string.
- //! \return Pointer to allocated attribute. This pointer will never be NULL.
- xml_attribute<Ch> *allocate_attribute(const Ch *name = 0, const Ch *value = 0,
- std::size_t name_size = 0, std::size_t value_size = 0)
- {
- void *memory = allocate_aligned(sizeof(xml_attribute<Ch>));
- xml_attribute<Ch> *attribute = new(memory) xml_attribute<Ch>;
- if (name)
- {
- if (name_size > 0)
- attribute->name(name, name_size);
- else
- attribute->name(name);
- }
- if (value)
- {
- if (value_size > 0)
- attribute->value(value, value_size);
- else
- attribute->value(value);
- }
- return attribute;
- }
-
- //! Allocates a char array of given size from the pool, and optionally copies a given string to it.
- //! If the allocation request cannot be accomodated, this function will throw <code>std::bad_alloc</code>.
- //! If exceptions are disabled by defining RAPIDXML_NO_EXCEPTIONS, this function
- //! will call rapidxml::parse_error_handler() function.
- //! \param source String to initialize the allocated memory with, or 0 to not initialize it.
- //! \param size Number of characters to allocate, or zero to calculate it automatically from source string length; if size is 0, source string must be specified and null terminated.
- //! \return Pointer to allocated char array. This pointer will never be NULL.
- Ch *allocate_string(const Ch *source = 0, std::size_t size = 0)
- {
- assert(source || size); // Either source or size (or both) must be specified
- if (size == 0)
- size = internal::measure(source) + 1;
- Ch *result = static_cast<Ch *>(allocate_aligned(size * sizeof(Ch)));
- if (source)
- for (std::size_t i = 0; i < size; ++i)
- result[i] = source[i];
- return result;
- }
-
- //! Clones an xml_node and its hierarchy of child nodes and attributes.
- //! Nodes and attributes are allocated from this memory pool.
- //! Names and values are not cloned, they are shared between the clone and the source.
- //! Result node can be optionally specified as a second parameter,
- //! in which case its contents will be replaced with cloned source node.
- //! This is useful when you want to clone entire document.
- //! \param source Node to clone.
- //! \param result Node to put results in, or 0 to automatically allocate result node
- //! \return Pointer to cloned node. This pointer will never be NULL.
- xml_node<Ch> *clone_node(const xml_node<Ch> *source, xml_node<Ch> *result = 0)
- {
- // Prepare result node
- if (result)
- {
- result->remove_all_attributes();
- result->remove_all_nodes();
- result->type(source->type());
- }
- else
- result = allocate_node(source->type());
-
- // Clone name and value
- result->name(source->name(), source->name_size());
- result->value(source->value(), source->value_size());
-
- // Clone child nodes and attributes
- for (xml_node<Ch> *child = source->first_node(); child; child = child->next_sibling())
- result->append_node(clone_node(child));
- for (xml_attribute<Ch> *attr = source->first_attribute(); attr; attr = attr->next_attribute())
- result->append_attribute(allocate_attribute(attr->name(), attr->value(), attr->name_size(), attr->value_size()));
-
- return result;
- }
-
- //! Clears the pool.
- //! This causes memory occupied by nodes allocated by the pool to be freed.
- //! Any nodes or strings allocated from the pool will no longer be valid.
- void clear()
- {
- while (m_begin != m_static_memory)
- {
- char *previous_begin = reinterpret_cast<header *>(align(m_begin))->previous_begin;
- if (m_free_func)
- m_free_func(m_begin);
- else
- delete[] m_begin;
- m_begin = previous_begin;
- }
- init();
- }
-
- //! Sets or resets the user-defined memory allocation functions for the pool.
- //! This can only be called when no memory is allocated from the pool yet, otherwise results are undefined.
- //! Allocation function must not return invalid pointer on failure. It should either throw,
- //! stop the program, or use <code>longjmp()</code> function to pass control to other place of program.
- //! If it returns invalid pointer, results are undefined.
- //! <br><br>
- //! User defined allocation functions must have the following forms:
- //! <br><code>
- //! <br>void *allocate(std::size_t size);
- //! <br>void free(void *pointer);
- //! </code><br>
- //! \param af Allocation function, or 0 to restore default function
- //! \param ff Free function, or 0 to restore default function
- void set_allocator(alloc_func *af, free_func *ff)
- {
- assert(m_begin == m_static_memory && m_ptr == align(m_begin)); // Verify that no memory is allocated yet
- m_alloc_func = af;
- m_free_func = ff;
- }
-
- private:
-
- struct header
- {
- char *previous_begin;
- };
-
- void init()
- {
- m_begin = m_static_memory;
- m_ptr = align(m_begin);
- m_end = m_static_memory + sizeof(m_static_memory);
- }
-
- char *align(char *ptr)
- {
- std::size_t alignment = ((RAPIDXML_ALIGNMENT - (std::size_t(ptr) & (RAPIDXML_ALIGNMENT - 1))) & (RAPIDXML_ALIGNMENT - 1));
- return ptr + alignment;
- }
-
- char *allocate_raw(std::size_t size)
- {
- // Allocate
- void *memory;
- if (m_alloc_func) // Allocate memory using either user-specified allocation function or global operator new[]
- {
- memory = m_alloc_func(size);
- assert(memory); // Allocator is not allowed to return 0, on failure it must either throw, stop the program or use longjmp
- }
- else
- {
- memory = new char[size];
-#ifdef RAPIDXML_NO_EXCEPTIONS
- if (!memory) // If exceptions are disabled, verify memory allocation, because new will not be able to throw bad_alloc
- RAPIDXML_PARSE_ERROR("out of memory", 0);
-#endif
- }
- return static_cast<char *>(memory);
- }
-
- void *allocate_aligned(std::size_t size)
- {
- // Calculate aligned pointer
- char *result = align(m_ptr);
-
- // If not enough memory left in current pool, allocate a new pool
- if (result + size > m_end)
- {
- // Calculate required pool size (may be bigger than RAPIDXML_DYNAMIC_POOL_SIZE)
- std::size_t pool_size = RAPIDXML_DYNAMIC_POOL_SIZE;
- if (pool_size < size)
- pool_size = size;
-
- // Allocate
- std::size_t alloc_size = sizeof(header) + (2 * RAPIDXML_ALIGNMENT - 2) + pool_size; // 2 alignments required in worst case: one for header, one for actual allocation
- char *raw_memory = allocate_raw(alloc_size);
-
- // Setup new pool in allocated memory
- char *pool = align(raw_memory);
- header *new_header = reinterpret_cast<header *>(pool);
- new_header->previous_begin = m_begin;
- m_begin = raw_memory;
- m_ptr = pool + sizeof(header);
- m_end = raw_memory + alloc_size;
-
- // Calculate aligned pointer again using new pool
- result = align(m_ptr);
- }
-
- // Update pool and return aligned pointer
- m_ptr = result + size;
- return result;
- }
-
- char *m_begin; // Start of raw memory making up current pool
- char *m_ptr; // First free byte in current pool
- char *m_end; // One past last available byte in current pool
- char m_static_memory[RAPIDXML_STATIC_POOL_SIZE]; // Static raw memory
- alloc_func *m_alloc_func; // Allocator function, or 0 if default is to be used
- free_func *m_free_func; // Free function, or 0 if default is to be used
- };
-
- ///////////////////////////////////////////////////////////////////////////
- // XML base
-
- //! Base class for xml_node and xml_attribute implementing common functions:
- //! name(), name_size(), value(), value_size() and parent().
- //! \param Ch Character type to use
- template<class Ch = char>
- class xml_base
- {
-
- public:
-
- ///////////////////////////////////////////////////////////////////////////
- // Construction & destruction
-
- // Construct a base with empty name, value and parent
- xml_base()
- : m_name(0)
- , m_value(0)
- , m_parent(0)
- {
- }
-
- ///////////////////////////////////////////////////////////////////////////
- // Node data access
-
- //! Gets name of the node.
- //! Interpretation of name depends on type of node.
- //! Note that name will not be zero-terminated if rapidxml::parse_no_string_terminators option was selected during parse.
- //! <br><br>
- //! Use name_size() function to determine length of the name.
- //! \return Name of node, or empty string if node has no name.
- Ch *name() const
- {
- return m_name ? m_name : nullstr();
- }
-
- //! Gets size of node name, not including terminator character.
- //! This function works correctly irrespective of whether name is or is not zero terminated.
- //! \return Size of node name, in characters.
- std::size_t name_size() const
- {
- return m_name ? m_name_size : 0;
- }
-
- //! Gets value of node.
- //! Interpretation of value depends on type of node.
- //! Note that value will not be zero-terminated if rapidxml::parse_no_string_terminators option was selected during parse.
- //! <br><br>
- //! Use value_size() function to determine length of the value.
- //! \return Value of node, or empty string if node has no value.
- Ch *value() const
- {
- return m_value ? m_value : nullstr();
- }
-
- //! Gets size of node value, not including terminator character.
- //! This function works correctly irrespective of whether value is or is not zero terminated.
- //! \return Size of node value, in characters.
- std::size_t value_size() const
- {
- return m_value ? m_value_size : 0;
- }
-
- ///////////////////////////////////////////////////////////////////////////
- // Node modification
-
- //! Sets name of node to a non zero-terminated string.
- //! See \ref ownership_of_strings.
- //! <br><br>
- //! Note that node does not own its name or value, it only stores a pointer to it.
- //! It will not delete or otherwise free the pointer on destruction.
- //! It is reponsibility of the user to properly manage lifetime of the string.
- //! The easiest way to achieve it is to use memory_pool of the document to allocate the string -
- //! on destruction of the document the string will be automatically freed.
- //! <br><br>
- //! Size of name must be specified separately, because name does not have to be zero terminated.
- //! Use name(const Ch *) function to have the length automatically calculated (string must be zero terminated).
- //! \param name Name of node to set. Does not have to be zero terminated.
- //! \param size Size of name, in characters. This does not include zero terminator, if one is present.
- void name(const Ch *name, std::size_t size)
- {
- m_name = const_cast<Ch *>(name);
- m_name_size = size;
- }
-
- //! Sets name of node to a zero-terminated string.
- //! See also \ref ownership_of_strings and xml_node::name(const Ch *, std::size_t).
- //! \param name Name of node to set. Must be zero terminated.
- void name(const Ch *name)
- {
- this->name(name, internal::measure(name));
- }
-
- //! Sets value of node to a non zero-terminated string.
- //! See \ref ownership_of_strings.
- //! <br><br>
- //! Note that node does not own its name or value, it only stores a pointer to it.
- //! It will not delete or otherwise free the pointer on destruction.
- //! It is reponsibility of the user to properly manage lifetime of the string.
- //! The easiest way to achieve it is to use memory_pool of the document to allocate the string -
- //! on destruction of the document the string will be automatically freed.
- //! <br><br>
- //! Size of value must be specified separately, because it does not have to be zero terminated.
- //! Use value(const Ch *) function to have the length automatically calculated (string must be zero terminated).
- //! <br><br>
- //! If an element has a child node of type node_data, it will take precedence over element value when printing.
- //! If you want to manipulate data of elements using values, use parser flag rapidxml::parse_no_data_nodes to prevent creation of data nodes by the parser.
- //! \param value value of node to set. Does not have to be zero terminated.
- //! \param size Size of value, in characters. This does not include zero terminator, if one is present.
- void value(const Ch *value, std::size_t size)
- {
- m_value = const_cast<Ch *>(value);
- m_value_size = size;
- }
-
- //! Sets value of node to a zero-terminated string.
- //! See also \ref ownership_of_strings and xml_node::value(const Ch *, std::size_t).
- //! \param value Vame of node to set. Must be zero terminated.
- void value(const Ch *value)
- {
- this->value(value, internal::measure(value));
- }
-
- ///////////////////////////////////////////////////////////////////////////
- // Related nodes access
-
- //! Gets node parent.
- //! \return Pointer to parent node, or 0 if there is no parent.
- xml_node<Ch> *parent() const
- {
- return m_parent;
- }
-
- protected:
-
- // Return empty string
- static Ch *nullstr()
- {
- static Ch zero = Ch('\0');
- return &zero;
- }
-
- Ch *m_name; // Name of node, or 0 if no name
- Ch *m_value; // Value of node, or 0 if no value
- std::size_t m_name_size; // Length of node name, or undefined of no name
- std::size_t m_value_size; // Length of node value, or undefined if no value
- xml_node<Ch> *m_parent; // Pointer to parent node, or 0 if none
-
- };
-
- //! Class representing attribute node of XML document.
- //! Each attribute has name and value strings, which are available through name() and value() functions (inherited from xml_base).
- //! Note that after parse, both name and value of attribute will point to interior of source text used for parsing.
- //! Thus, this text must persist in memory for the lifetime of attribute.
- //! \param Ch Character type to use.
- template<class Ch = char>
- class xml_attribute: public xml_base<Ch>
- {
-
- friend class xml_node<Ch>;
-
- public:
-
- ///////////////////////////////////////////////////////////////////////////
- // Construction & destruction
-
- //! Constructs an empty attribute with the specified type.
- //! Consider using memory_pool of appropriate xml_document if allocating attributes manually.
- xml_attribute()
- {
- }
-
- ///////////////////////////////////////////////////////////////////////////
- // Related nodes access
-
- //! Gets document of which attribute is a child.
- //! \return Pointer to document that contains this attribute, or 0 if there is no parent document.
- xml_document<Ch> *document() const
- {
- if (xml_node<Ch> *node = this->parent())
- {
- while (node->parent())
- node = node->parent();
- return node->type() == node_document ? static_cast<xml_document<Ch> *>(node) : 0;
- }
- else
- return 0;
- }
-
- //! Gets previous attribute, optionally matching attribute name.
- //! \param name Name of attribute to find, or 0 to return previous attribute regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero
- //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string
- //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters
- //! \return Pointer to found attribute, or 0 if not found.
- xml_attribute<Ch> *previous_attribute(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const
- {
- if (name)
- {
- if (name_size == 0)
- name_size = internal::measure(name);
- for (xml_attribute<Ch> *attribute = m_prev_attribute; attribute; attribute = attribute->m_prev_attribute)
- if (internal::compare(attribute->name(), attribute->name_size(), name, name_size, case_sensitive))
- return attribute;
- return 0;
- }
- else
- return this->m_parent ? m_prev_attribute : 0;
- }
-
- //! Gets next attribute, optionally matching attribute name.
- //! \param name Name of attribute to find, or 0 to return next attribute regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero
- //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string
- //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters
- //! \return Pointer to found attribute, or 0 if not found.
- xml_attribute<Ch> *next_attribute(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const
- {
- if (name)
- {
- if (name_size == 0)
- name_size = internal::measure(name);
- for (xml_attribute<Ch> *attribute = m_next_attribute; attribute; attribute = attribute->m_next_attribute)
- if (internal::compare(attribute->name(), attribute->name_size(), name, name_size, case_sensitive))
- return attribute;
- return 0;
- }
- else
- return this->m_parent ? m_next_attribute : 0;
- }
-
- private:
-
- xml_attribute<Ch> *m_prev_attribute; // Pointer to previous sibling of attribute, or 0 if none; only valid if parent is non-zero
- xml_attribute<Ch> *m_next_attribute; // Pointer to next sibling of attribute, or 0 if none; only valid if parent is non-zero
-
- };
-
- ///////////////////////////////////////////////////////////////////////////
- // XML node
-
- //! Class representing a node of XML document.
- //! Each node may have associated name and value strings, which are available through name() and value() functions.
- //! Interpretation of name and value depends on type of the node.
- //! Type of node can be determined by using type() function.
- //! <br><br>
- //! Note that after parse, both name and value of node, if any, will point interior of source text used for parsing.
- //! Thus, this text must persist in the memory for the lifetime of node.
- //! \param Ch Character type to use.
- template<class Ch = char>
- class xml_node: public xml_base<Ch>
- {
-
- public:
-
- ///////////////////////////////////////////////////////////////////////////
- // Construction & destruction
-
- //! Constructs an empty node with the specified type.
- //! Consider using memory_pool of appropriate document to allocate nodes manually.
- //! \param type Type of node to construct.
- xml_node(node_type type)
- : m_type(type)
- , m_first_node(0)
- , m_first_attribute(0)
- {
- }
-
- ///////////////////////////////////////////////////////////////////////////
- // Node data access
-
- //! Gets type of node.
- //! \return Type of node.
- node_type type() const
- {
- return m_type;
- }
-
- ///////////////////////////////////////////////////////////////////////////
- // Related nodes access
-
- //! Gets document of which node is a child.
- //! \return Pointer to document that contains this node, or 0 if there is no parent document.
- xml_document<Ch> *document() const
- {
- xml_node<Ch> *node = const_cast<xml_node<Ch> *>(this);
- while (node->parent())
- node = node->parent();
- return node->type() == node_document ? static_cast<xml_document<Ch> *>(node) : 0;
- }
-
- //! Gets first child node, optionally matching node name.
- //! \param name Name of child to find, or 0 to return first child regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero
- //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string
- //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters
- //! \return Pointer to found child, or 0 if not found.
- xml_node<Ch> *first_node(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const
- {
- if (name)
- {
- if (name_size == 0)
- name_size = internal::measure(name);
- for (xml_node<Ch> *child = m_first_node; child; child = child->next_sibling())
- if (internal::compare(child->name(), child->name_size(), name, name_size, case_sensitive))
- return child;
- return 0;
- }
- else
- return m_first_node;
- }
-
- //! Gets last child node, optionally matching node name.
- //! Behaviour is undefined if node has no children.
- //! Use first_node() to test if node has children.
- //! \param name Name of child to find, or 0 to return last child regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero
- //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string
- //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters
- //! \return Pointer to found child, or 0 if not found.
- xml_node<Ch> *last_node(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const
- {
- assert(m_first_node); // Cannot query for last child if node has no children
- if (name)
- {
- if (name_size == 0)
- name_size = internal::measure(name);
- for (xml_node<Ch> *child = m_last_node; child; child = child->previous_sibling())
- if (internal::compare(child->name(), child->name_size(), name, name_size, case_sensitive))
- return child;
- return 0;
- }
- else
- return m_last_node;
- }
-
- //! Gets previous sibling node, optionally matching node name.
- //! Behaviour is undefined if node has no parent.
- //! Use parent() to test if node has a parent.
- //! \param name Name of sibling to find, or 0 to return previous sibling regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero
- //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string
- //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters
- //! \return Pointer to found sibling, or 0 if not found.
- xml_node<Ch> *previous_sibling(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const
- {
- assert(this->m_parent); // Cannot query for siblings if node has no parent
- if (name)
- {
- if (name_size == 0)
- name_size = internal::measure(name);
- for (xml_node<Ch> *sibling = m_prev_sibling; sibling; sibling = sibling->m_prev_sibling)
- if (internal::compare(sibling->name(), sibling->name_size(), name, name_size, case_sensitive))
- return sibling;
- return 0;
- }
- else
- return m_prev_sibling;
- }
-
- //! Gets next sibling node, optionally matching node name.
- //! Behaviour is undefined if node has no parent.
- //! Use parent() to test if node has a parent.
- //! \param name Name of sibling to find, or 0 to return next sibling regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero
- //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string
- //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters
- //! \return Pointer to found sibling, or 0 if not found.
- xml_node<Ch> *next_sibling(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const
- {
- assert(this->m_parent); // Cannot query for siblings if node has no parent
- if (name)
- {
- if (name_size == 0)
- name_size = internal::measure(name);
- for (xml_node<Ch> *sibling = m_next_sibling; sibling; sibling = sibling->m_next_sibling)
- if (internal::compare(sibling->name(), sibling->name_size(), name, name_size, case_sensitive))
- return sibling;
- return 0;
- }
- else
- return m_next_sibling;
- }
-
- //! Gets first attribute of node, optionally matching attribute name.
- //! \param name Name of attribute to find, or 0 to return first attribute regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero
- //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string
- //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters
- //! \return Pointer to found attribute, or 0 if not found.
- xml_attribute<Ch> *first_attribute(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const
- {
- if (name)
- {
- if (name_size == 0)
- name_size = internal::measure(name);
- for (xml_attribute<Ch> *attribute = m_first_attribute; attribute; attribute = attribute->m_next_attribute)
- if (internal::compare(attribute->name(), attribute->name_size(), name, name_size, case_sensitive))
- return attribute;
- return 0;
- }
- else
- return m_first_attribute;
- }
-
- //! Gets last attribute of node, optionally matching attribute name.
- //! \param name Name of attribute to find, or 0 to return last attribute regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero
- //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string
- //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters
- //! \return Pointer to found attribute, or 0 if not found.
- xml_attribute<Ch> *last_attribute(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const
- {
- if (name)
- {
- if (name_size == 0)
- name_size = internal::measure(name);
- for (xml_attribute<Ch> *attribute = m_last_attribute; attribute; attribute = attribute->m_prev_attribute)
- if (internal::compare(attribute->name(), attribute->name_size(), name, name_size, case_sensitive))
- return attribute;
- return 0;
- }
- else
- return m_first_attribute ? m_last_attribute : 0;
- }
-
- ///////////////////////////////////////////////////////////////////////////
- // Node modification
-
- //! Sets type of node.
- //! \param type Type of node to set.
- void type(node_type type)
- {
- m_type = type;
- }
-
- ///////////////////////////////////////////////////////////////////////////
- // Node manipulation
-
- //! Prepends a new child node.
- //! The prepended child becomes the first child, and all existing children are moved one position back.
- //! \param child Node to prepend.
- void prepend_node(xml_node<Ch> *child)
- {
- assert(child && !child->parent() && child->type() != node_document);
- if (first_node())
- {
- child->m_next_sibling = m_first_node;
- m_first_node->m_prev_sibling = child;
- }
- else
- {
- child->m_next_sibling = 0;
- m_last_node = child;
- }
- m_first_node = child;
- child->m_parent = this;
- child->m_prev_sibling = 0;
- }
-
- //! Appends a new child node.
- //! The appended child becomes the last child.
- //! \param child Node to append.
- void append_node(xml_node<Ch> *child)
- {
- assert(child && !child->parent() && child->type() != node_document);
- if (first_node())
- {
- child->m_prev_sibling = m_last_node;
- m_last_node->m_next_sibling = child;
- }
- else
- {
- child->m_prev_sibling = 0;
- m_first_node = child;
- }
- m_last_node = child;
- child->m_parent = this;
- child->m_next_sibling = 0;
- }
-
- //! Inserts a new child node at specified place inside the node.
- //! All children after and including the specified node are moved one position back.
- //! \param where Place where to insert the child, or 0 to insert at the back.
- //! \param child Node to insert.
- void insert_node(xml_node<Ch> *where, xml_node<Ch> *child)
- {
- assert(!where || where->parent() == this);
- assert(child && !child->parent() && child->type() != node_document);
- if (where == m_first_node)
- prepend_node(child);
- else if (where == 0)
- append_node(child);
- else
- {
- child->m_prev_sibling = where->m_prev_sibling;
- child->m_next_sibling = where;
- where->m_prev_sibling->m_next_sibling = child;
- where->m_prev_sibling = child;
- child->m_parent = this;
- }
- }
-
- //! Removes first child node.
- //! If node has no children, behaviour is undefined.
- //! Use first_node() to test if node has children.
- void remove_first_node()
- {
- assert(first_node());
- xml_node<Ch> *child = m_first_node;
- m_first_node = child->m_next_sibling;
- if (child->m_next_sibling)
- child->m_next_sibling->m_prev_sibling = 0;
- else
- m_last_node = 0;
- child->m_parent = 0;
- }
-
- //! Removes last child of the node.
- //! If node has no children, behaviour is undefined.
- //! Use first_node() to test if node has children.
- void remove_last_node()
- {
- assert(first_node());
- xml_node<Ch> *child = m_last_node;
- if (child->m_prev_sibling)
- {
- m_last_node = child->m_prev_sibling;
- child->m_prev_sibling->m_next_sibling = 0;
- }
- else
- m_first_node = 0;
- child->m_parent = 0;
- }
-
- //! Removes specified child from the node
- // \param where Pointer to child to be removed.
- void remove_node(xml_node<Ch> *where)
- {
- assert(where && where->parent() == this);
- assert(first_node());
- if (where == m_first_node)
- remove_first_node();
- else if (where == m_last_node)
- remove_last_node();
- else
- {
- where->m_prev_sibling->m_next_sibling = where->m_next_sibling;
- where->m_next_sibling->m_prev_sibling = where->m_prev_sibling;
- where->m_parent = 0;
- }
- }
-
- //! Removes all child nodes (but not attributes).
- void remove_all_nodes()
- {
- for (xml_node<Ch> *node = first_node(); node; node = node->m_next_sibling)
- node->m_parent = 0;
- m_first_node = 0;
- }
-
- //! Prepends a new attribute to the node.
- //! \param attribute Attribute to prepend.
- void prepend_attribute(xml_attribute<Ch> *attribute)
- {
- assert(attribute && !attribute->parent());
- if (first_attribute())
- {
- attribute->m_next_attribute = m_first_attribute;
- m_first_attribute->m_prev_attribute = attribute;
- }
- else
- {
- attribute->m_next_attribute = 0;
- m_last_attribute = attribute;
- }
- m_first_attribute = attribute;
- attribute->m_parent = this;
- attribute->m_prev_attribute = 0;
- }
-
- //! Appends a new attribute to the node.
- //! \param attribute Attribute to append.
- void append_attribute(xml_attribute<Ch> *attribute)
- {
- assert(attribute && !attribute->parent());
- if (first_attribute())
- {
- attribute->m_prev_attribute = m_last_attribute;
- m_last_attribute->m_next_attribute = attribute;
- }
- else
- {
- attribute->m_prev_attribute = 0;
- m_first_attribute = attribute;
- }
- m_last_attribute = attribute;
- attribute->m_parent = this;
- attribute->m_next_attribute = 0;
- }
-
- //! Inserts a new attribute at specified place inside the node.
- //! All attributes after and including the specified attribute are moved one position back.
- //! \param where Place where to insert the attribute, or 0 to insert at the back.
- //! \param attribute Attribute to insert.
- void insert_attribute(xml_attribute<Ch> *where, xml_attribute<Ch> *attribute)
- {
- assert(!where || where->parent() == this);
- assert(attribute && !attribute->parent());
- if (where == m_first_attribute)
- prepend_attribute(attribute);
- else if (where == 0)
- append_attribute(attribute);
- else
- {
- attribute->m_prev_attribute = where->m_prev_attribute;
- attribute->m_next_attribute = where;
- where->m_prev_attribute->m_next_attribute = attribute;
- where->m_prev_attribute = attribute;
- attribute->m_parent = this;
- }
- }
-
- //! Removes first attribute of the node.
- //! If node has no attributes, behaviour is undefined.
- //! Use first_attribute() to test if node has attributes.
- void remove_first_attribute()
- {
- assert(first_attribute());
- xml_attribute<Ch> *attribute = m_first_attribute;
- if (attribute->m_next_attribute)
- {
- attribute->m_next_attribute->m_prev_attribute = 0;
- }
- else
- m_last_attribute = 0;
- attribute->m_parent = 0;
- m_first_attribute = attribute->m_next_attribute;
- }
-
- //! Removes last attribute of the node.
- //! If node has no attributes, behaviour is undefined.
- //! Use first_attribute() to test if node has attributes.
- void remove_last_attribute()
- {
- assert(first_attribute());
- xml_attribute<Ch> *attribute = m_last_attribute;
- if (attribute->m_prev_attribute)
- {
- attribute->m_prev_attribute->m_next_attribute = 0;
- m_last_attribute = attribute->m_prev_attribute;
- }
- else
- m_first_attribute = 0;
- attribute->m_parent = 0;
- }
-
- //! Removes specified attribute from node.
- //! \param where Pointer to attribute to be removed.
- void remove_attribute(xml_attribute<Ch> *where)
- {
- assert(first_attribute() && where->parent() == this);
- if (where == m_first_attribute)
- remove_first_attribute();
- else if (where == m_last_attribute)
- remove_last_attribute();
- else
- {
- where->m_prev_attribute->m_next_attribute = where->m_next_attribute;
- where->m_next_attribute->m_prev_attribute = where->m_prev_attribute;
- where->m_parent = 0;
- }
- }
-
- //! Removes all attributes of node.
- void remove_all_attributes()
- {
- for (xml_attribute<Ch> *attribute = first_attribute(); attribute; attribute = attribute->m_next_attribute)
- attribute->m_parent = 0;
- m_first_attribute = 0;
- }
-
- private:
-
- ///////////////////////////////////////////////////////////////////////////
- // Restrictions
-
- // No copying
- xml_node(const xml_node &);
- void operator =(const xml_node &);
-
- ///////////////////////////////////////////////////////////////////////////
- // Data members
-
- // Note that some of the pointers below have UNDEFINED values if certain other pointers are 0.
- // This is required for maximum performance, as it allows the parser to omit initialization of
- // unneded/redundant values.
- //
- // The rules are as follows:
- // 1. first_node and first_attribute contain valid pointers, or 0 if node has no children/attributes respectively
- // 2. last_node and last_attribute are valid only if node has at least one child/attribute respectively, otherwise they contain garbage
- // 3. prev_sibling and next_sibling are valid only if node has a parent, otherwise they contain garbage
-
- node_type m_type; // Type of node; always valid
- xml_node<Ch> *m_first_node; // Pointer to first child node, or 0 if none; always valid
- xml_node<Ch> *m_last_node; // Pointer to last child node, or 0 if none; this value is only valid if m_first_node is non-zero
- xml_attribute<Ch> *m_first_attribute; // Pointer to first attribute of node, or 0 if none; always valid
- xml_attribute<Ch> *m_last_attribute; // Pointer to last attribute of node, or 0 if none; this value is only valid if m_first_attribute is non-zero
- xml_node<Ch> *m_prev_sibling; // Pointer to previous sibling of node, or 0 if none; this value is only valid if m_parent is non-zero
- xml_node<Ch> *m_next_sibling; // Pointer to next sibling of node, or 0 if none; this value is only valid if m_parent is non-zero
-
- };
-
- ///////////////////////////////////////////////////////////////////////////
- // XML document
-
- //! This class represents root of the DOM hierarchy.
- //! It is also an xml_node and a memory_pool through public inheritance.
- //! Use parse() function to build a DOM tree from a zero-terminated XML text string.
- //! parse() function allocates memory for nodes and attributes by using functions of xml_document,
- //! which are inherited from memory_pool.
- //! To access root node of the document, use the document itself, as if it was an xml_node.
- //! \param Ch Character type to use.
- template<class Ch = char>
- class xml_document: public xml_node<Ch>, public memory_pool<Ch>
- {
-
- public:
-
- //! Constructs empty XML document
- xml_document()
- : xml_node<Ch>(node_document)
- {
- }
-
- //! Parses zero-terminated XML string according to given flags.
- //! Passed string will be modified by the parser, unless rapidxml::parse_non_destructive flag is used.
- //! The string must persist for the lifetime of the document.
- //! In case of error, rapidxml::parse_error exception will be thrown.
- //! <br><br>
- //! If you want to parse contents of a file, you must first load the file into the memory, and pass pointer to its beginning.
- //! Make sure that data is zero-terminated.
- //! <br><br>
- //! Document can be parsed into multiple times.
- //! Each new call to parse removes previous nodes and attributes (if any), but does not clear memory pool.
- //! \param text XML data to parse; pointer is non-const to denote fact that this data may be modified by the parser.
- template<int Flags>
- void parse(Ch *text)
- {
- assert(text);
-
- // Remove current contents
- this->remove_all_nodes();
- this->remove_all_attributes();
-
- // Parse BOM, if any
- parse_bom<Flags>(text);
-
- // Parse children
- while (1)
- {
- // Skip whitespace before node
- skip<whitespace_pred, Flags>(text);
- if (*text == 0)
- break;
-
- // Parse and append new child
- if (*text == Ch('<'))
- {
- ++text; // Skip '<'
- if (xml_node<Ch> *node = parse_node<Flags>(text))
- this->append_node(node);
- }
- else
- RAPIDXML_PARSE_ERROR("expected <", text);
- }
-
- }
-
- //! Clears the document by deleting all nodes and clearing the memory pool.
- //! All nodes owned by document pool are destroyed.
- void clear()
- {
- this->remove_all_nodes();
- this->remove_all_attributes();
- memory_pool<Ch>::clear();
- }
-
- private:
-
- ///////////////////////////////////////////////////////////////////////
- // Internal character utility functions
-
- // Detect whitespace character
- struct whitespace_pred
- {
- static unsigned char test(Ch ch)
- {
- return internal::lookup_tables<0>::lookup_whitespace[static_cast<unsigned char>(ch)];
- }
- };
-
- // Detect node name character
- struct node_name_pred
- {
- static unsigned char test(Ch ch)
- {
- return internal::lookup_tables<0>::lookup_node_name[static_cast<unsigned char>(ch)];
- }
- };
-
- // Detect attribute name character
- struct attribute_name_pred
- {
- static unsigned char test(Ch ch)
- {
- return internal::lookup_tables<0>::lookup_attribute_name[static_cast<unsigned char>(ch)];
- }
- };
-
- // Detect text character (PCDATA)
- struct text_pred
- {
- static unsigned char test(Ch ch)
- {
- return internal::lookup_tables<0>::lookup_text[static_cast<unsigned char>(ch)];
- }
- };
-
- // Detect text character (PCDATA) that does not require processing
- struct text_pure_no_ws_pred
- {
- static unsigned char test(Ch ch)
- {
- return internal::lookup_tables<0>::lookup_text_pure_no_ws[static_cast<unsigned char>(ch)];
- }
- };
-
- // Detect text character (PCDATA) that does not require processing
- struct text_pure_with_ws_pred
- {
- static unsigned char test(Ch ch)
- {
- return internal::lookup_tables<0>::lookup_text_pure_with_ws[static_cast<unsigned char>(ch)];
- }
- };
-
- // Detect attribute value character
- template<Ch Quote>
- struct attribute_value_pred
- {
- static unsigned char test(Ch ch)
- {
- if (Quote == Ch('\''))
- return internal::lookup_tables<0>::lookup_attribute_data_1[static_cast<unsigned char>(ch)];
- if (Quote == Ch('\"'))
- return internal::lookup_tables<0>::lookup_attribute_data_2[static_cast<unsigned char>(ch)];
- return 0; // Should never be executed, to avoid warnings on Comeau
- }
- };
-
- // Detect attribute value character
- template<Ch Quote>
- struct attribute_value_pure_pred
- {
- static unsigned char test(Ch ch)
- {
- if (Quote == Ch('\''))
- return internal::lookup_tables<0>::lookup_attribute_data_1_pure[static_cast<unsigned char>(ch)];
- if (Quote == Ch('\"'))
- return internal::lookup_tables<0>::lookup_attribute_data_2_pure[static_cast<unsigned char>(ch)];
- return 0; // Should never be executed, to avoid warnings on Comeau
- }
- };
-
- // Insert coded character, using UTF8 or 8-bit ASCII
- template<int Flags>
- static void insert_coded_character(Ch *&text, unsigned long code)
- {
- if (Flags & parse_no_utf8)
- {
- // Insert 8-bit ASCII character
- // Todo: possibly verify that code is less than 256 and use replacement char otherwise?
- text[0] = static_cast<unsigned char>(code);
- text += 1;
- }
- else
- {
- // Insert UTF8 sequence
- if (code < 0x80) // 1 byte sequence
- {
- text[0] = static_cast<unsigned char>(code);
- text += 1;
- }
- else if (code < 0x800) // 2 byte sequence
- {
- text[1] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6;
- text[0] = static_cast<unsigned char>(code | 0xC0);
- text += 2;
- }
- else if (code < 0x10000) // 3 byte sequence
- {
- text[2] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6;
- text[1] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6;
- text[0] = static_cast<unsigned char>(code | 0xE0);
- text += 3;
- }
- else if (code < 0x110000) // 4 byte sequence
- {
- text[3] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6;
- text[2] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6;
- text[1] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6;
- text[0] = static_cast<unsigned char>(code | 0xF0);
- text += 4;
- }
- else // Invalid, only codes up to 0x10FFFF are allowed in Unicode
- {
- RAPIDXML_PARSE_ERROR("invalid numeric character entity", text);
- }
- }
- }
-
- // Skip characters until predicate evaluates to true
- template<class StopPred, int Flags>
- static void skip(Ch *&text)
- {
- Ch *tmp = text;
- while (StopPred::test(*tmp))
- ++tmp;
- text = tmp;
- }
-
- // Skip characters until predicate evaluates to true while doing the following:
- // - replacing XML character entity references with proper characters (' & " < > &#...;)
- // - condensing whitespace sequences to single space character
- template<class StopPred, class StopPredPure, int Flags>
- static Ch *skip_and_expand_character_refs(Ch *&text)
- {
- // If entity translation, whitespace condense and whitespace trimming is disabled, use plain skip
- if (Flags & parse_no_entity_translation &&
- !(Flags & parse_normalize_whitespace) &&
- !(Flags & parse_trim_whitespace))
- {
- skip<StopPred, Flags>(text);
- return text;
- }
-
- // Use simple skip until first modification is detected
- skip<StopPredPure, Flags>(text);
-
- // Use translation skip
- Ch *src = text;
- Ch *dest = src;
- while (StopPred::test(*src))
- {
- // If entity translation is enabled
- if (!(Flags & parse_no_entity_translation))
- {
- // Test if replacement is needed
- if (src[0] == Ch('&'))
- {
- switch (src[1])
- {
-
- // & '
- case Ch('a'):
- if (src[2] == Ch('m') && src[3] == Ch('p') && src[4] == Ch(';'))
- {
- *dest = Ch('&');
- ++dest;
- src += 5;
- continue;
- }
- if (src[2] == Ch('p') && src[3] == Ch('o') && src[4] == Ch('s') && src[5] == Ch(';'))
- {
- *dest = Ch('\'');
- ++dest;
- src += 6;
- continue;
- }
- break;
-
- // "
- case Ch('q'):
- if (src[2] == Ch('u') && src[3] == Ch('o') && src[4] == Ch('t') && src[5] == Ch(';'))
- {
- *dest = Ch('"');
- ++dest;
- src += 6;
- continue;
- }
- break;
-
- // >
- case Ch('g'):
- if (src[2] == Ch('t') && src[3] == Ch(';'))
- {
- *dest = Ch('>');
- ++dest;
- src += 4;
- continue;
- }
- break;
-
- // <
- case Ch('l'):
- if (src[2] == Ch('t') && src[3] == Ch(';'))
- {
- *dest = Ch('<');
- ++dest;
- src += 4;
- continue;
- }
- break;
-
- // &#...; - assumes ASCII
- case Ch('#'):
- if (src[2] == Ch('x'))
- {
- unsigned long code = 0;
- src += 3; // Skip &#x
- while (1)
- {
- unsigned char digit = internal::lookup_tables<0>::lookup_digits[static_cast<unsigned char>(*src)];
- if (digit == 0xFF)
- break;
- code = code * 16 + digit;
- ++src;
- }
- insert_coded_character<Flags>(dest, code); // Put character in output
- }
- else
- {
- unsigned long code = 0;
- src += 2; // Skip &#
- while (1)
- {
- unsigned char digit = internal::lookup_tables<0>::lookup_digits[static_cast<unsigned char>(*src)];
- if (digit == 0xFF)
- break;
- code = code * 10 + digit;
- ++src;
- }
- insert_coded_character<Flags>(dest, code); // Put character in output
- }
- if (*src == Ch(';'))
- ++src;
- else
- RAPIDXML_PARSE_ERROR("expected ;", src);
- continue;
-
- // Something else
- default:
- // Ignore, just copy '&' verbatim
- break;
-
- }
- }
- }
-
- // If whitespace condensing is enabled
- if (Flags & parse_normalize_whitespace)
- {
- // Test if condensing is needed
- if (whitespace_pred::test(*src))
- {
- *dest = Ch(' '); ++dest; // Put single space in dest
- ++src; // Skip first whitespace char
- // Skip remaining whitespace chars
- while (whitespace_pred::test(*src))
- ++src;
- continue;
- }
- }
-
- // No replacement, only copy character
- *dest++ = *src++;
-
- }
-
- // Return new end
- text = src;
- return dest;
-
- }
-
- ///////////////////////////////////////////////////////////////////////
- // Internal parsing functions
-
- // Parse BOM, if any
- template<int Flags>
- void parse_bom(Ch *&text)
- {
- // UTF-8?
- if (static_cast<unsigned char>(text[0]) == 0xEF &&
- static_cast<unsigned char>(text[1]) == 0xBB &&
- static_cast<unsigned char>(text[2]) == 0xBF)
- {
- text += 3; // Skup utf-8 bom
- }
- }
-
- // Parse XML declaration (<?xml...)
- template<int Flags>
- xml_node<Ch> *parse_xml_declaration(Ch *&text)
- {
- // If parsing of declaration is disabled
- if (!(Flags & parse_declaration_node))
- {
- // Skip until end of declaration
- while (text[0] != Ch('?') || text[1] != Ch('>'))
- {
- if (!text[0]) {
- RAPIDXML_PARSE_ERROR("unexpected end of data", text);
- return 0;
- }
- ++text;
- }
- text += 2; // Skip '?>'
- return 0;
- }
-
- // Create declaration
- xml_node<Ch> *declaration = this->allocate_node(node_declaration);
-
- // Skip whitespace before attributes or ?>
- skip<whitespace_pred, Flags>(text);
-
- // Parse declaration attributes
- parse_node_attributes<Flags>(text, declaration);
-
- // Skip ?>
- if (text[0] != Ch('?') || text[1] != Ch('>'))
- RAPIDXML_PARSE_ERROR("expected ?>", text);
- text += 2;
-
- return declaration;
- }
-
- // Parse XML comment (<!--...)
- template<int Flags>
- xml_node<Ch> *parse_comment(Ch *&text)
- {
- // If parsing of comments is disabled
- if (!(Flags & parse_comment_nodes))
- {
- // Skip until end of comment
- while (text[0] != Ch('-') || text[1] != Ch('-') || text[2] != Ch('>'))
- {
- if (!text[0]) {
- RAPIDXML_PARSE_ERROR("unexpected end of data", text);
- return 0;
- }
- ++text;
- }
- text += 3; // Skip '-->'
- return 0; // Do not produce comment node
- }
-
- // Remember value start
- Ch *value = text;
-
- // Skip until end of comment
- while (text[0] != Ch('-') || text[1] != Ch('-') || text[2] != Ch('>'))
- {
- if (!text[0]) {
- RAPIDXML_PARSE_ERROR("unexpected end of data", text);
- return 0;
- }
- ++text;
- }
-
- // Create comment node
- xml_node<Ch> *comment = this->allocate_node(node_comment);
- comment->value(value, text - value);
-
- // Place zero terminator after comment value
- if (!(Flags & parse_no_string_terminators))
- *text = Ch('\0');
-
- text += 3; // Skip '-->'
- return comment;
- }
-
- // Parse DOCTYPE
- template<int Flags>
- xml_node<Ch> *parse_doctype(Ch *&text)
- {
- // Remember value start
- Ch *value = text;
-
- // Skip to >
- while (*text != Ch('>'))
- {
- // Determine character type
- switch (*text)
- {
-
- // If '[' encountered, scan for matching ending ']' using naive algorithm with depth
- // This works for all W3C test files except for 2 most wicked
- case Ch('['):
- {
- ++text; // Skip '['
- int depth = 1;
- while (depth > 0)
- {
- switch (*text)
- {
- case Ch('['): ++depth; break;
- case Ch(']'): --depth; break;
- case 0: RAPIDXML_PARSE_ERROR("unexpected end of data", text); return 0;
- }
- ++text;
- }
- break;
- }
-
- // Error on end of text
- case Ch('\0'):
- RAPIDXML_PARSE_ERROR("unexpected end of data", text);
- return 0;
-
- // Other character, skip it
- default:
- ++text;
-
- }
- }
-
- // If DOCTYPE nodes enabled
- if (Flags & parse_doctype_node)
- {
- // Create a new doctype node
- xml_node<Ch> *doctype = this->allocate_node(node_doctype);
- doctype->value(value, text - value);
-
- // Place zero terminator after value
- if (!(Flags & parse_no_string_terminators))
- *text = Ch('\0');
-
- text += 1; // skip '>'
- return doctype;
- }
- else
- {
- text += 1; // skip '>'
- return 0;
- }
-
- }
-
- // Parse PI
- template<int Flags>
- xml_node<Ch> *parse_pi(Ch *&text)
- {
- // If creation of PI nodes is enabled
- if (Flags & parse_pi_nodes)
- {
- // Create pi node
- xml_node<Ch> *pi = this->allocate_node(node_pi);
-
- // Extract PI target name
- Ch *name = text;
- skip<node_name_pred, Flags>(text);
- if (text == name)
- RAPIDXML_PARSE_ERROR("expected PI target", text);
- pi->name(name, text - name);
-
- // Skip whitespace between pi target and pi
- skip<whitespace_pred, Flags>(text);
-
- // Remember start of pi
- Ch *value = text;
-
- // Skip to '?>'
- while (text[0] != Ch('?') || text[1] != Ch('>'))
- {
- if (*text == Ch('\0')) {
- RAPIDXML_PARSE_ERROR("unexpected end of data", text);
- return 0;
- }
- ++text;
- }
-
- // Set pi value (verbatim, no entity expansion or whitespace normalization)
- pi->value(value, text - value);
-
- // Place zero terminator after name and value
- if (!(Flags & parse_no_string_terminators))
- {
- pi->name()[pi->name_size()] = Ch('\0');
- pi->value()[pi->value_size()] = Ch('\0');
- }
-
- text += 2; // Skip '?>'
- return pi;
- }
- else
- {
- // Skip to '?>'
- while (text[0] != Ch('?') || text[1] != Ch('>'))
- {
- if (*text == Ch('\0')) {
- RAPIDXML_PARSE_ERROR("unexpected end of data", text);
- return 0;
- }
- ++text;
- }
- text += 2; // Skip '?>'
- return 0;
- }
- }
-
- // Parse and append data
- // Return character that ends data.
- // This is necessary because this character might have been overwritten by a terminating 0
- template<int Flags>
- Ch parse_and_append_data(xml_node<Ch> *node, Ch *&text, Ch *contents_start)
- {
- // Backup to contents start if whitespace trimming is disabled
- if (!(Flags & parse_trim_whitespace))
- text = contents_start;
-
- // Skip until end of data
- Ch *value = text, *end;
- if (Flags & parse_normalize_whitespace)
- end = skip_and_expand_character_refs<text_pred, text_pure_with_ws_pred, Flags>(text);
- else
- end = skip_and_expand_character_refs<text_pred, text_pure_no_ws_pred, Flags>(text);
-
- // Trim trailing whitespace if flag is set; leading was already trimmed by whitespace skip after >
- if (Flags & parse_trim_whitespace)
- {
- if (Flags & parse_normalize_whitespace)
- {
- // Whitespace is already condensed to single space characters by skipping function, so just trim 1 char off the end
- if (*(end - 1) == Ch(' '))
- --end;
- }
- else
- {
- // Backup until non-whitespace character is found
- while (whitespace_pred::test(*(end - 1)))
- --end;
- }
- }
-
- // If characters are still left between end and value (this test is only necessary if normalization is enabled)
- // Create new data node
- if (!(Flags & parse_no_data_nodes))
- {
- xml_node<Ch> *data = this->allocate_node(node_data);
- data->value(value, end - value);
- node->append_node(data);
- }
-
- // Add data to parent node if no data exists yet
- if (!(Flags & parse_no_element_values))
- if (*node->value() == Ch('\0'))
- node->value(value, end - value);
-
- // Place zero terminator after value
- if (!(Flags & parse_no_string_terminators))
- {
- Ch ch = *text;
- *end = Ch('\0');
- return ch; // Return character that ends data; this is required because zero terminator overwritten it
- }
-
- // Return character that ends data
- return *text;
- }
-
- // Parse CDATA
- template<int Flags>
- xml_node<Ch> *parse_cdata(Ch *&text)
- {
- // If CDATA is disabled
- if (Flags & parse_no_data_nodes)
- {
- // Skip until end of cdata
- while (text[0] != Ch(']') || text[1] != Ch(']') || text[2] != Ch('>'))
- {
- if (!text[0]) {
- RAPIDXML_PARSE_ERROR("unexpected end of data", text);
- return 0;
- }
- ++text;
- }
- text += 3; // Skip ]]>
- return 0; // Do not produce CDATA node
- }
-
- // Skip until end of cdata
- Ch *value = text;
- while (text[0] != Ch(']') || text[1] != Ch(']') || text[2] != Ch('>'))
- {
- if (!text[0]) {
- RAPIDXML_PARSE_ERROR("unexpected end of data", text);
- return 0;
- }
- ++text;
- }
-
- // Create new cdata node
- xml_node<Ch> *cdata = this->allocate_node(node_cdata);
- cdata->value(value, text - value);
-
- // Place zero terminator after value
- if (!(Flags & parse_no_string_terminators))
- *text = Ch('\0');
-
- text += 3; // Skip ]]>
- return cdata;
- }
-
- // Parse element node
- template<int Flags>
- xml_node<Ch> *parse_element(Ch *&text)
- {
- // Create element node
- xml_node<Ch> *element = this->allocate_node(node_element);
-
- // Extract element name
- Ch *name = text;
- skip<node_name_pred, Flags>(text);
- if (text == name)
- RAPIDXML_PARSE_ERROR("expected element name", text);
- element->name(name, text - name);
-
- // Skip whitespace between element name and attributes or >
- skip<whitespace_pred, Flags>(text);
-
- // Parse attributes, if any
- parse_node_attributes<Flags>(text, element);
-
- // Determine ending type
- if (*text == Ch('>'))
- {
- ++text;
- parse_node_contents<Flags>(text, element);
- }
- else if (*text == Ch('/'))
- {
- ++text;
- if (*text != Ch('>'))
- RAPIDXML_PARSE_ERROR("expected >", text);
- ++text;
- }
- else
- RAPIDXML_PARSE_ERROR("expected >", text);
-
- // Place zero terminator after name
- if (!(Flags & parse_no_string_terminators))
- element->name()[element->name_size()] = Ch('\0');
-
- // Return parsed element
- return element;
- }
-
- // Determine node type, and parse it
- template<int Flags>
- xml_node<Ch> *parse_node(Ch *&text)
- {
- // Parse proper node type
- switch (text[0])
- {
-
- // <...
- default:
- // Parse and append element node
- return parse_element<Flags>(text);
-
- // <?...
- case Ch('?'):
- ++text; // Skip ?
- if ((text[0] == Ch('x') || text[0] == Ch('X')) &&
- (text[1] == Ch('m') || text[1] == Ch('M')) &&
- (text[2] == Ch('l') || text[2] == Ch('L')) &&
- whitespace_pred::test(text[3]))
- {
- // '<?xml ' - xml declaration
- text += 4; // Skip 'xml '
- return parse_xml_declaration<Flags>(text);
- }
- else
- {
- // Parse PI
- return parse_pi<Flags>(text);
- }
-
- // <!...
- case Ch('!'):
-
- // Parse proper subset of <! node
- switch (text[1])
- {
-
- // <!-
- case Ch('-'):
- if (text[2] == Ch('-'))
- {
- // '<!--' - xml comment
- text += 3; // Skip '!--'
- return parse_comment<Flags>(text);
- }
- break;
-
- // <![
- case Ch('['):
- if (text[2] == Ch('C') && text[3] == Ch('D') && text[4] == Ch('A') &&
- text[5] == Ch('T') && text[6] == Ch('A') && text[7] == Ch('['))
- {
- // '<![CDATA[' - cdata
- text += 8; // Skip '![CDATA['
- return parse_cdata<Flags>(text);
- }
- break;
-
- // <!D
- case Ch('D'):
- if (text[2] == Ch('O') && text[3] == Ch('C') && text[4] == Ch('T') &&
- text[5] == Ch('Y') && text[6] == Ch('P') && text[7] == Ch('E') &&
- whitespace_pred::test(text[8]))
- {
- // '<!DOCTYPE ' - doctype
- text += 9; // skip '!DOCTYPE '
- return parse_doctype<Flags>(text);
- }
-
- } // switch
-
- // Attempt to skip other, unrecognized node types starting with <!
- ++text; // Skip !
- while (*text != Ch('>'))
- {
- if (*text == 0) {
- RAPIDXML_PARSE_ERROR("unexpected end of data", text);
- return 0;
- }
- ++text;
- }
- ++text; // Skip '>'
- return 0; // No node recognized
-
- }
- }
-
- // Parse contents of the node - children, data etc.
- template<int Flags>
- void parse_node_contents(Ch *&text, xml_node<Ch> *node)
- {
- // For all children and text
- while (1)
- {
- // Skip whitespace between > and node contents
- Ch *contents_start = text; // Store start of node contents before whitespace is skipped
- skip<whitespace_pred, Flags>(text);
- Ch next_char = *text;
-
- // After data nodes, instead of continuing the loop, control jumps here.
- // This is because zero termination inside parse_and_append_data() function
- // would wreak havoc with the above code.
- // Also, skipping whitespace after data nodes is unnecessary.
- after_data_node:
-
- // Determine what comes next: node closing, child node, data node, or 0?
- switch (next_char)
- {
-
- // Node closing or child node
- case Ch('<'):
- if (text[1] == Ch('/'))
- {
- // Node closing
- text += 2; // Skip '</'
- if (Flags & parse_validate_closing_tags)
- {
- // Skip and validate closing tag name
- Ch *closing_name = text;
- skip<node_name_pred, Flags>(text);
- if (!internal::compare(node->name(), node->name_size(), closing_name, text - closing_name, true))
- RAPIDXML_PARSE_ERROR("invalid closing tag name", text);
- }
- else
- {
- // No validation, just skip name
- skip<node_name_pred, Flags>(text);
- }
- // Skip remaining whitespace after node name
- skip<whitespace_pred, Flags>(text);
- if (*text != Ch('>'))
- RAPIDXML_PARSE_ERROR("expected >", text);
- ++text; // Skip '>'
- return; // Node closed, finished parsing contents
- }
- else
- {
- // Child node
- ++text; // Skip '<'
- if (xml_node<Ch> *child = parse_node<Flags>(text))
- node->append_node(child);
- }
- break;
-
- // End of data - error
- case Ch('\0'):
- RAPIDXML_PARSE_ERROR("unexpected end of data", text);
- return;
-
- // Data node
- default:
- next_char = parse_and_append_data<Flags>(node, text, contents_start);
- goto after_data_node; // Bypass regular processing after data nodes
-
- }
- }
- }
-
- // Parse XML attributes of the node
- template<int Flags>
- void parse_node_attributes(Ch *&text, xml_node<Ch> *node)
- {
- // For all attributes
- while (attribute_name_pred::test(*text))
- {
- // Extract attribute name
- Ch *name = text;
- ++text; // Skip first character of attribute name
- skip<attribute_name_pred, Flags>(text);
- if (text == name)
- RAPIDXML_PARSE_ERROR("expected attribute name", name);
-
- // Create new attribute
- xml_attribute<Ch> *attribute = this->allocate_attribute();
- attribute->name(name, text - name);
- node->append_attribute(attribute);
-
- // Skip whitespace after attribute name
- skip<whitespace_pred, Flags>(text);
-
- // Skip =
- if (*text != Ch('='))
- RAPIDXML_PARSE_ERROR("expected =", text);
- ++text;
-
- // Add terminating zero after name
- if (!(Flags & parse_no_string_terminators))
- attribute->name()[attribute->name_size()] = 0;
-
- // Skip whitespace after =
- skip<whitespace_pred, Flags>(text);
-
- // Skip quote and remember if it was ' or "
- Ch quote = *text;
- if (quote != Ch('\'') && quote != Ch('"'))
- RAPIDXML_PARSE_ERROR("expected ' or \"", text);
- ++text;
-
- // Extract attribute value and expand char refs in it
- Ch *value = text, *end;
- const int AttFlags = Flags & ~parse_normalize_whitespace; // No whitespace normalization in attributes
- if (quote == Ch('\''))
- end = skip_and_expand_character_refs<attribute_value_pred<Ch('\'')>, attribute_value_pure_pred<Ch('\'')>, AttFlags>(text);
- else
- end = skip_and_expand_character_refs<attribute_value_pred<Ch('"')>, attribute_value_pure_pred<Ch('"')>, AttFlags>(text);
-
- // Set attribute value
- attribute->value(value, end - value);
-
- // Make sure that end quote is present
- if (*text != quote)
- RAPIDXML_PARSE_ERROR("expected ' or \"", text);
- ++text; // Skip quote
-
- // Add terminating zero after value
- if (!(Flags & parse_no_string_terminators))
- attribute->value()[attribute->value_size()] = 0;
-
- // Skip whitespace after attribute value
- skip<whitespace_pred, Flags>(text);
- }
- }
-
- };
-
- //! \cond internal
- namespace internal
- {
-
- // Whitespace (space \n \r \t)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_whitespace[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 1, 0, 0, // 0
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 1
- 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 2
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 3
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 4
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 5
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 6
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 7
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 8
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 9
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // A
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // B
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // C
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // D
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // E
- 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 // F
- };
-
- // Node name (anything but space \n \r \t / > ? \0)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_node_name[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 0, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, // 0
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1
- 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, // 2
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, // 3
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F
- };
-
- // Text (i.e. PCDATA) (anything but < \0)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_text[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, // 3
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F
- };
-
- // Text (i.e. PCDATA) that does not require processing when ws normalization is disabled
- // (anything but < \0 &)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_text_pure_no_ws[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1
- 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, // 3
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F
- };
-
- // Text (i.e. PCDATA) that does not require processing when ws normalizationis is enabled
- // (anything but < \0 & space \n \r \t)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_text_pure_with_ws[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 0, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, // 0
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1
- 0, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, // 3
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F
- };
-
- // Attribute name (anything but space \n \r \t / < > = ? ! \0)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_attribute_name[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 0, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, // 0
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1
- 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, // 2
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, // 3
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F
- };
-
- // Attribute data with single quote (anything but ' \0)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_attribute_data_1[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1
- 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, // 2
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 3
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F
- };
-
- // Attribute data with single quote that does not require processing (anything but ' \0 &)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_attribute_data_1_pure[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1
- 1, 1, 1, 1, 1, 1, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, // 2
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 3
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F
- };
-
- // Attribute data with double quote (anything but " \0)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_attribute_data_2[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1
- 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 3
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F
- };
-
- // Attribute data with double quote that does not require processing (anything but " \0 &)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_attribute_data_2_pure[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1
- 1, 1, 0, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 3
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F
- };
-
- // Digits (dec and hex, 255 denotes end of numeric character reference)
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_digits[256] =
- {
- // 0 1 2 3 4 5 6 7 8 9 A B C D E F
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 0
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 1
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 2
- 0, 1, 2, 3, 4, 5, 6, 7, 8, 9,255,255,255,255,255,255, // 3
- 255, 10, 11, 12, 13, 14, 15,255,255,255,255,255,255,255,255,255, // 4
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 5
- 255, 10, 11, 12, 13, 14, 15,255,255,255,255,255,255,255,255,255, // 6
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 7
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 8
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 9
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // A
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // B
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // C
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // D
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // E
- 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255 // F
- };
-
- // Upper case conversion
- template<int Dummy>
- const unsigned char lookup_tables<Dummy>::lookup_upcase[256] =
- {
- // 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, A B C D E F
- 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, // 0
- 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, // 1
- 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, // 2
- 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, // 3
- 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, // 4
- 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95, // 5
- 96, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, // 6
- 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 123,124,125,126,127, // 7
- 128,129,130,131,132,133,134,135,136,137,138,139,140,141,142,143, // 8
- 144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159, // 9
- 160,161,162,163,164,165,166,167,168,169,170,171,172,173,174,175, // A
- 176,177,178,179,180,181,182,183,184,185,186,187,188,189,190,191, // B
- 192,193,194,195,196,197,198,199,200,201,202,203,204,205,206,207, // C
- 208,209,210,211,212,213,214,215,216,217,218,219,220,221,222,223, // D
- 224,225,226,227,228,229,230,231,232,233,234,235,236,237,238,239, // E
- 240,241,242,243,244,245,246,247,248,249,250,251,252,253,254,255 // F
- };
- }
- //! \endcond
-
-}
-
-// Undefine internal macros
-#undef RAPIDXML_PARSE_ERROR
-
-// On MSVC, restore warnings state
-#ifdef _MSC_VER
- #pragma warning(pop)
-#endif
-
-#endif
+#ifndef RAPIDXML_HPP_INCLUDED +#define RAPIDXML_HPP_INCLUDED + +#define RAPIDXML_NO_EXCEPTIONS + +// Copyright (C) 2006, 2009 Marcin Kalicinski +// Version 1.13 +// Revision $DateTime: 2009/05/13 01:46:17 $ +//! \file rapidxml.hpp This file contains rapidxml parser and DOM implementation + +// If standard library is disabled, user must provide implementations of required functions and typedefs +#if !defined(RAPIDXML_NO_STDLIB) + #include <cstdlib> // For std::size_t + #include <cassert> // For assert + #include <new> // For placement new +#endif + +// On MSVC, disable "conditional expression is constant" warning (level 4). +// This warning is almost impossible to avoid with certain types of templated code +#ifdef _MSC_VER + #pragma warning(push) + #pragma warning(disable:4127) // Conditional expression is constant +#endif + +/////////////////////////////////////////////////////////////////////////// +// RAPIDXML_PARSE_ERROR + +#if defined(RAPIDXML_NO_EXCEPTIONS) + +#define RAPIDXML_PARSE_ERROR(what, where) { parse_error_handler(what, where); assert(0); } + +namespace rapidxml +{ + //! When exceptions are disabled by defining RAPIDXML_NO_EXCEPTIONS, + //! this function is called to notify user about the error. + //! It must be defined by the user. + //! <br><br> + //! This function cannot return. If it does, the results are undefined. + //! <br><br> + //! A very simple definition might look like that: + //! <pre> + //! void %rapidxml::%parse_error_handler(const char *what, void *where) + //! { + //! std::cout << "Parse error: " << what << "\n"; + //! std::abort(); + //! } + //! </pre> + //! \param what Human readable description of the error. + //! \param where Pointer to character data where error was detected. + void parse_error_handler(const char *what, void *where); +} + +#else + +#include <exception> // For std::exception + +#define RAPIDXML_PARSE_ERROR(what, where) throw parse_error(what, where) + +namespace rapidxml +{ + + //! Parse error exception. + //! This exception is thrown by the parser when an error occurs. + //! Use what() function to get human-readable error message. + //! Use where() function to get a pointer to position within source text where error was detected. + //! <br><br> + //! If throwing exceptions by the parser is undesirable, + //! it can be disabled by defining RAPIDXML_NO_EXCEPTIONS macro before rapidxml.hpp is included. + //! This will cause the parser to call rapidxml::parse_error_handler() function instead of throwing an exception. + //! This function must be defined by the user. + //! <br><br> + //! This class derives from <code>std::exception</code> class. + class parse_error: public std::exception + { + + public: + + //! Constructs parse error + parse_error(const char *what, void *where) + : m_what(what) + , m_where(where) + { + } + + //! Gets human readable description of error. + //! \return Pointer to null terminated description of the error. + virtual const char *what() const throw() + { + return m_what; + } + + //! Gets pointer to character data where error happened. + //! Ch should be the same as char type of xml_document that produced the error. + //! \return Pointer to location within the parsed string where error occured. + template<class Ch> + Ch *where() const + { + return reinterpret_cast<Ch *>(m_where); + } + + private: + + const char *m_what; + void *m_where; + + }; +} + +#endif + +/////////////////////////////////////////////////////////////////////////// +// Pool sizes + +#ifndef RAPIDXML_STATIC_POOL_SIZE + // Size of static memory block of memory_pool. + // Define RAPIDXML_STATIC_POOL_SIZE before including rapidxml.hpp if you want to override the default value. + // No dynamic memory allocations are performed by memory_pool until static memory is exhausted. + #define RAPIDXML_STATIC_POOL_SIZE (64 * 1024) +#endif + +#ifndef RAPIDXML_DYNAMIC_POOL_SIZE + // Size of dynamic memory block of memory_pool. + // Define RAPIDXML_DYNAMIC_POOL_SIZE before including rapidxml.hpp if you want to override the default value. + // After the static block is exhausted, dynamic blocks with approximately this size are allocated by memory_pool. + #define RAPIDXML_DYNAMIC_POOL_SIZE (64 * 1024) +#endif + +#ifndef RAPIDXML_ALIGNMENT + // Memory allocation alignment. + // Define RAPIDXML_ALIGNMENT before including rapidxml.hpp if you want to override the default value, which is the size of pointer. + // All memory allocations for nodes, attributes and strings will be aligned to this value. + // This must be a power of 2 and at least 1, otherwise memory_pool will not work. + #define RAPIDXML_ALIGNMENT sizeof(void *) +#endif + +namespace rapidxml +{ + // Forward declarations + template<class Ch> class xml_node; + template<class Ch> class xml_attribute; + template<class Ch> class xml_document; + + //! Enumeration listing all node types produced by the parser. + //! Use xml_node::type() function to query node type. + enum node_type + { + node_document, //!< A document node. Name and value are empty. + node_element, //!< An element node. Name contains element name. Value contains text of first data node. + node_data, //!< A data node. Name is empty. Value contains data text. + node_cdata, //!< A CDATA node. Name is empty. Value contains data text. + node_comment, //!< A comment node. Name is empty. Value contains comment text. + node_declaration, //!< A declaration node. Name and value are empty. Declaration parameters (version, encoding and standalone) are in node attributes. + node_doctype, //!< A DOCTYPE node. Name is empty. Value contains DOCTYPE text. + node_pi //!< A PI node. Name contains target. Value contains instructions. + }; + + /////////////////////////////////////////////////////////////////////// + // Parsing flags + + //! Parse flag instructing the parser to not create data nodes. + //! Text of first data node will still be placed in value of parent element, unless rapidxml::parse_no_element_values flag is also specified. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_no_data_nodes = 0x1; + + //! Parse flag instructing the parser to not use text of first data node as a value of parent element. + //! Can be combined with other flags by use of | operator. + //! Note that child data nodes of element node take precendence over its value when printing. + //! That is, if element has one or more child data nodes <em>and</em> a value, the value will be ignored. + //! Use rapidxml::parse_no_data_nodes flag to prevent creation of data nodes if you want to manipulate data using values of elements. + //! <br><br> + //! See xml_document::parse() function. + const int parse_no_element_values = 0x2; + + //! Parse flag instructing the parser to not place zero terminators after strings in the source text. + //! By default zero terminators are placed, modifying source text. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_no_string_terminators = 0x4; + + //! Parse flag instructing the parser to not translate entities in the source text. + //! By default entities are translated, modifying source text. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_no_entity_translation = 0x8; + + //! Parse flag instructing the parser to disable UTF-8 handling and assume plain 8 bit characters. + //! By default, UTF-8 handling is enabled. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_no_utf8 = 0x10; + + //! Parse flag instructing the parser to create XML declaration node. + //! By default, declaration node is not created. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_declaration_node = 0x20; + + //! Parse flag instructing the parser to create comments nodes. + //! By default, comment nodes are not created. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_comment_nodes = 0x40; + + //! Parse flag instructing the parser to create DOCTYPE node. + //! By default, doctype node is not created. + //! Although W3C specification allows at most one DOCTYPE node, RapidXml will silently accept documents with more than one. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_doctype_node = 0x80; + + //! Parse flag instructing the parser to create PI nodes. + //! By default, PI nodes are not created. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_pi_nodes = 0x100; + + //! Parse flag instructing the parser to validate closing tag names. + //! If not set, name inside closing tag is irrelevant to the parser. + //! By default, closing tags are not validated. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_validate_closing_tags = 0x200; + + //! Parse flag instructing the parser to trim all leading and trailing whitespace of data nodes. + //! By default, whitespace is not trimmed. + //! This flag does not cause the parser to modify source text. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_trim_whitespace = 0x400; + + //! Parse flag instructing the parser to condense all whitespace runs of data nodes to a single space character. + //! Trimming of leading and trailing whitespace of data is controlled by rapidxml::parse_trim_whitespace flag. + //! By default, whitespace is not normalized. + //! If this flag is specified, source text will be modified. + //! Can be combined with other flags by use of | operator. + //! <br><br> + //! See xml_document::parse() function. + const int parse_normalize_whitespace = 0x800; + + // Compound flags + + //! Parse flags which represent default behaviour of the parser. + //! This is always equal to 0, so that all other flags can be simply ored together. + //! Normally there is no need to inconveniently disable flags by anding with their negated (~) values. + //! This also means that meaning of each flag is a <i>negation</i> of the default setting. + //! For example, if flag name is rapidxml::parse_no_utf8, it means that utf-8 is <i>enabled</i> by default, + //! and using the flag will disable it. + //! <br><br> + //! See xml_document::parse() function. + const int parse_default = 0; + + //! A combination of parse flags that forbids any modifications of the source text. + //! This also results in faster parsing. However, note that the following will occur: + //! <ul> + //! <li>names and values of nodes will not be zero terminated, you have to use xml_base::name_size() and xml_base::value_size() functions to determine where name and value ends</li> + //! <li>entities will not be translated</li> + //! <li>whitespace will not be normalized</li> + //! </ul> + //! See xml_document::parse() function. + const int parse_non_destructive = parse_no_string_terminators | parse_no_entity_translation; + + //! A combination of parse flags resulting in fastest possible parsing, without sacrificing important data. + //! <br><br> + //! See xml_document::parse() function. + const int parse_fastest = parse_non_destructive | parse_no_data_nodes; + + //! A combination of parse flags resulting in largest amount of data being extracted. + //! This usually results in slowest parsing. + //! <br><br> + //! See xml_document::parse() function. + const int parse_full = parse_declaration_node | parse_comment_nodes | parse_doctype_node | parse_pi_nodes | parse_validate_closing_tags; + + /////////////////////////////////////////////////////////////////////// + // Internals + + //! \cond internal + namespace internal + { + + // Struct that contains lookup tables for the parser + // It must be a template to allow correct linking (because it has static data members, which are defined in a header file). + template<int Dummy> + struct lookup_tables + { + static const unsigned char lookup_whitespace[256]; // Whitespace table + static const unsigned char lookup_node_name[256]; // Node name table + static const unsigned char lookup_text[256]; // Text table + static const unsigned char lookup_text_pure_no_ws[256]; // Text table + static const unsigned char lookup_text_pure_with_ws[256]; // Text table + static const unsigned char lookup_attribute_name[256]; // Attribute name table + static const unsigned char lookup_attribute_data_1[256]; // Attribute data table with single quote + static const unsigned char lookup_attribute_data_1_pure[256]; // Attribute data table with single quote + static const unsigned char lookup_attribute_data_2[256]; // Attribute data table with double quotes + static const unsigned char lookup_attribute_data_2_pure[256]; // Attribute data table with double quotes + static const unsigned char lookup_digits[256]; // Digits + static const unsigned char lookup_upcase[256]; // To uppercase conversion table for ASCII characters + }; + + // Find length of the string + template<class Ch> + inline std::size_t measure(const Ch *p) + { + const Ch *tmp = p; + while (*tmp) + ++tmp; + return tmp - p; + } + + // Compare strings for equality + template<class Ch> + inline bool compare(const Ch *p1, std::size_t size1, const Ch *p2, std::size_t size2, bool case_sensitive) + { + if (size1 != size2) + return false; + if (case_sensitive) + { + for (const Ch *end = p1 + size1; p1 < end; ++p1, ++p2) + if (*p1 != *p2) + return false; + } + else + { + for (const Ch *end = p1 + size1; p1 < end; ++p1, ++p2) + if (lookup_tables<0>::lookup_upcase[static_cast<unsigned char>(*p1)] != lookup_tables<0>::lookup_upcase[static_cast<unsigned char>(*p2)]) + return false; + } + return true; + } + } + //! \endcond + + /////////////////////////////////////////////////////////////////////// + // Memory pool + + //! This class is used by the parser to create new nodes and attributes, without overheads of dynamic memory allocation. + //! In most cases, you will not need to use this class directly. + //! However, if you need to create nodes manually or modify names/values of nodes, + //! you are encouraged to use memory_pool of relevant xml_document to allocate the memory. + //! Not only is this faster than allocating them by using <code>new</code> operator, + //! but also their lifetime will be tied to the lifetime of document, + //! possibly simplyfing memory management. + //! <br><br> + //! Call allocate_node() or allocate_attribute() functions to obtain new nodes or attributes from the pool. + //! You can also call allocate_string() function to allocate strings. + //! Such strings can then be used as names or values of nodes without worrying about their lifetime. + //! Note that there is no <code>free()</code> function -- all allocations are freed at once when clear() function is called, + //! or when the pool is destroyed. + //! <br><br> + //! It is also possible to create a standalone memory_pool, and use it + //! to allocate nodes, whose lifetime will not be tied to any document. + //! <br><br> + //! Pool maintains <code>RAPIDXML_STATIC_POOL_SIZE</code> bytes of statically allocated memory. + //! Until static memory is exhausted, no dynamic memory allocations are done. + //! When static memory is exhausted, pool allocates additional blocks of memory of size <code>RAPIDXML_DYNAMIC_POOL_SIZE</code> each, + //! by using global <code>new[]</code> and <code>delete[]</code> operators. + //! This behaviour can be changed by setting custom allocation routines. + //! Use set_allocator() function to set them. + //! <br><br> + //! Allocations for nodes, attributes and strings are aligned at <code>RAPIDXML_ALIGNMENT</code> bytes. + //! This value defaults to the size of pointer on target architecture. + //! <br><br> + //! To obtain absolutely top performance from the parser, + //! it is important that all nodes are allocated from a single, contiguous block of memory. + //! Otherwise, cache misses when jumping between two (or more) disjoint blocks of memory can slow down parsing quite considerably. + //! If required, you can tweak <code>RAPIDXML_STATIC_POOL_SIZE</code>, <code>RAPIDXML_DYNAMIC_POOL_SIZE</code> and <code>RAPIDXML_ALIGNMENT</code> + //! to obtain best wasted memory to performance compromise. + //! To do it, define their values before rapidxml.hpp file is included. + //! \param Ch Character type of created nodes. + template<class Ch = char> + class memory_pool + { + + public: + + //! \cond internal + typedef void *(alloc_func)(std::size_t); // Type of user-defined function used to allocate memory + typedef void (free_func)(void *); // Type of user-defined function used to free memory + //! \endcond + + //! Constructs empty pool with default allocator functions. + memory_pool() + : m_alloc_func(0) + , m_free_func(0) + { + init(); + } + + //! Destroys pool and frees all the memory. + //! This causes memory occupied by nodes allocated by the pool to be freed. + //! Nodes allocated from the pool are no longer valid. + ~memory_pool() + { + clear(); + } + + //! Allocates a new node from the pool, and optionally assigns name and value to it. + //! If the allocation request cannot be accomodated, this function will throw <code>std::bad_alloc</code>. + //! If exceptions are disabled by defining RAPIDXML_NO_EXCEPTIONS, this function + //! will call rapidxml::parse_error_handler() function. + //! \param type Type of node to create. + //! \param name Name to assign to the node, or 0 to assign no name. + //! \param value Value to assign to the node, or 0 to assign no value. + //! \param name_size Size of name to assign, or 0 to automatically calculate size from name string. + //! \param value_size Size of value to assign, or 0 to automatically calculate size from value string. + //! \return Pointer to allocated node. This pointer will never be NULL. + xml_node<Ch> *allocate_node(node_type type, + const Ch *name = 0, const Ch *value = 0, + std::size_t name_size = 0, std::size_t value_size = 0) + { + void *memory = allocate_aligned(sizeof(xml_node<Ch>)); + xml_node<Ch> *node = new(memory) xml_node<Ch>(type); + if (name) + { + if (name_size > 0) + node->name(name, name_size); + else + node->name(name); + } + if (value) + { + if (value_size > 0) + node->value(value, value_size); + else + node->value(value); + } + return node; + } + + //! Allocates a new attribute from the pool, and optionally assigns name and value to it. + //! If the allocation request cannot be accomodated, this function will throw <code>std::bad_alloc</code>. + //! If exceptions are disabled by defining RAPIDXML_NO_EXCEPTIONS, this function + //! will call rapidxml::parse_error_handler() function. + //! \param name Name to assign to the attribute, or 0 to assign no name. + //! \param value Value to assign to the attribute, or 0 to assign no value. + //! \param name_size Size of name to assign, or 0 to automatically calculate size from name string. + //! \param value_size Size of value to assign, or 0 to automatically calculate size from value string. + //! \return Pointer to allocated attribute. This pointer will never be NULL. + xml_attribute<Ch> *allocate_attribute(const Ch *name = 0, const Ch *value = 0, + std::size_t name_size = 0, std::size_t value_size = 0) + { + void *memory = allocate_aligned(sizeof(xml_attribute<Ch>)); + xml_attribute<Ch> *attribute = new(memory) xml_attribute<Ch>; + if (name) + { + if (name_size > 0) + attribute->name(name, name_size); + else + attribute->name(name); + } + if (value) + { + if (value_size > 0) + attribute->value(value, value_size); + else + attribute->value(value); + } + return attribute; + } + + //! Allocates a char array of given size from the pool, and optionally copies a given string to it. + //! If the allocation request cannot be accomodated, this function will throw <code>std::bad_alloc</code>. + //! If exceptions are disabled by defining RAPIDXML_NO_EXCEPTIONS, this function + //! will call rapidxml::parse_error_handler() function. + //! \param source String to initialize the allocated memory with, or 0 to not initialize it. + //! \param size Number of characters to allocate, or zero to calculate it automatically from source string length; if size is 0, source string must be specified and null terminated. + //! \return Pointer to allocated char array. This pointer will never be NULL. + Ch *allocate_string(const Ch *source = 0, std::size_t size = 0) + { + assert(source || size); // Either source or size (or both) must be specified + if (size == 0) + size = internal::measure(source) + 1; + Ch *result = static_cast<Ch *>(allocate_aligned(size * sizeof(Ch))); + if (source) + for (std::size_t i = 0; i < size; ++i) + result[i] = source[i]; + return result; + } + + //! Clones an xml_node and its hierarchy of child nodes and attributes. + //! Nodes and attributes are allocated from this memory pool. + //! Names and values are not cloned, they are shared between the clone and the source. + //! Result node can be optionally specified as a second parameter, + //! in which case its contents will be replaced with cloned source node. + //! This is useful when you want to clone entire document. + //! \param source Node to clone. + //! \param result Node to put results in, or 0 to automatically allocate result node + //! \return Pointer to cloned node. This pointer will never be NULL. + xml_node<Ch> *clone_node(const xml_node<Ch> *source, xml_node<Ch> *result = 0) + { + // Prepare result node + if (result) + { + result->remove_all_attributes(); + result->remove_all_nodes(); + result->type(source->type()); + } + else + result = allocate_node(source->type()); + + // Clone name and value + result->name(source->name(), source->name_size()); + result->value(source->value(), source->value_size()); + + // Clone child nodes and attributes + for (xml_node<Ch> *child = source->first_node(); child; child = child->next_sibling()) + result->append_node(clone_node(child)); + for (xml_attribute<Ch> *attr = source->first_attribute(); attr; attr = attr->next_attribute()) + result->append_attribute(allocate_attribute(attr->name(), attr->value(), attr->name_size(), attr->value_size())); + + return result; + } + + //! Clears the pool. + //! This causes memory occupied by nodes allocated by the pool to be freed. + //! Any nodes or strings allocated from the pool will no longer be valid. + void clear() + { + while (m_begin != m_static_memory) + { + char *previous_begin = reinterpret_cast<header *>(align(m_begin))->previous_begin; + if (m_free_func) + m_free_func(m_begin); + else + delete[] m_begin; + m_begin = previous_begin; + } + init(); + } + + //! Sets or resets the user-defined memory allocation functions for the pool. + //! This can only be called when no memory is allocated from the pool yet, otherwise results are undefined. + //! Allocation function must not return invalid pointer on failure. It should either throw, + //! stop the program, or use <code>longjmp()</code> function to pass control to other place of program. + //! If it returns invalid pointer, results are undefined. + //! <br><br> + //! User defined allocation functions must have the following forms: + //! <br><code> + //! <br>void *allocate(std::size_t size); + //! <br>void free(void *pointer); + //! </code><br> + //! \param af Allocation function, or 0 to restore default function + //! \param ff Free function, or 0 to restore default function + void set_allocator(alloc_func *af, free_func *ff) + { + assert(m_begin == m_static_memory && m_ptr == align(m_begin)); // Verify that no memory is allocated yet + m_alloc_func = af; + m_free_func = ff; + } + + private: + + struct header + { + char *previous_begin; + }; + + void init() + { + m_begin = m_static_memory; + m_ptr = align(m_begin); + m_end = m_static_memory + sizeof(m_static_memory); + } + + char *align(char *ptr) + { + std::size_t alignment = ((RAPIDXML_ALIGNMENT - (std::size_t(ptr) & (RAPIDXML_ALIGNMENT - 1))) & (RAPIDXML_ALIGNMENT - 1)); + return ptr + alignment; + } + + char *allocate_raw(std::size_t size) + { + // Allocate + void *memory; + if (m_alloc_func) // Allocate memory using either user-specified allocation function or global operator new[] + { + memory = m_alloc_func(size); + assert(memory); // Allocator is not allowed to return 0, on failure it must either throw, stop the program or use longjmp + } + else + { + memory = new char[size]; +#ifdef RAPIDXML_NO_EXCEPTIONS + if (!memory) // If exceptions are disabled, verify memory allocation, because new will not be able to throw bad_alloc + RAPIDXML_PARSE_ERROR("out of memory", 0); +#endif + } + return static_cast<char *>(memory); + } + + void *allocate_aligned(std::size_t size) + { + // Calculate aligned pointer + char *result = align(m_ptr); + + // If not enough memory left in current pool, allocate a new pool + if (result + size > m_end) + { + // Calculate required pool size (may be bigger than RAPIDXML_DYNAMIC_POOL_SIZE) + std::size_t pool_size = RAPIDXML_DYNAMIC_POOL_SIZE; + if (pool_size < size) + pool_size = size; + + // Allocate + std::size_t alloc_size = sizeof(header) + (2 * RAPIDXML_ALIGNMENT - 2) + pool_size; // 2 alignments required in worst case: one for header, one for actual allocation + char *raw_memory = allocate_raw(alloc_size); + + // Setup new pool in allocated memory + char *pool = align(raw_memory); + header *new_header = reinterpret_cast<header *>(pool); + new_header->previous_begin = m_begin; + m_begin = raw_memory; + m_ptr = pool + sizeof(header); + m_end = raw_memory + alloc_size; + + // Calculate aligned pointer again using new pool + result = align(m_ptr); + } + + // Update pool and return aligned pointer + m_ptr = result + size; + return result; + } + + char *m_begin; // Start of raw memory making up current pool + char *m_ptr; // First free byte in current pool + char *m_end; // One past last available byte in current pool + char m_static_memory[RAPIDXML_STATIC_POOL_SIZE]; // Static raw memory + alloc_func *m_alloc_func; // Allocator function, or 0 if default is to be used + free_func *m_free_func; // Free function, or 0 if default is to be used + }; + + /////////////////////////////////////////////////////////////////////////// + // XML base + + //! Base class for xml_node and xml_attribute implementing common functions: + //! name(), name_size(), value(), value_size() and parent(). + //! \param Ch Character type to use + template<class Ch = char> + class xml_base + { + + public: + + /////////////////////////////////////////////////////////////////////////// + // Construction & destruction + + // Construct a base with empty name, value and parent + xml_base() + : m_name(0) + , m_value(0) + , m_parent(0) + { + } + + /////////////////////////////////////////////////////////////////////////// + // Node data access + + //! Gets name of the node. + //! Interpretation of name depends on type of node. + //! Note that name will not be zero-terminated if rapidxml::parse_no_string_terminators option was selected during parse. + //! <br><br> + //! Use name_size() function to determine length of the name. + //! \return Name of node, or empty string if node has no name. + Ch *name() const + { + return m_name ? m_name : nullstr(); + } + + //! Gets size of node name, not including terminator character. + //! This function works correctly irrespective of whether name is or is not zero terminated. + //! \return Size of node name, in characters. + std::size_t name_size() const + { + return m_name ? m_name_size : 0; + } + + //! Gets value of node. + //! Interpretation of value depends on type of node. + //! Note that value will not be zero-terminated if rapidxml::parse_no_string_terminators option was selected during parse. + //! <br><br> + //! Use value_size() function to determine length of the value. + //! \return Value of node, or empty string if node has no value. + Ch *value() const + { + return m_value ? m_value : nullstr(); + } + + //! Gets size of node value, not including terminator character. + //! This function works correctly irrespective of whether value is or is not zero terminated. + //! \return Size of node value, in characters. + std::size_t value_size() const + { + return m_value ? m_value_size : 0; + } + + /////////////////////////////////////////////////////////////////////////// + // Node modification + + //! Sets name of node to a non zero-terminated string. + //! See \ref ownership_of_strings. + //! <br><br> + //! Note that node does not own its name or value, it only stores a pointer to it. + //! It will not delete or otherwise free the pointer on destruction. + //! It is reponsibility of the user to properly manage lifetime of the string. + //! The easiest way to achieve it is to use memory_pool of the document to allocate the string - + //! on destruction of the document the string will be automatically freed. + //! <br><br> + //! Size of name must be specified separately, because name does not have to be zero terminated. + //! Use name(const Ch *) function to have the length automatically calculated (string must be zero terminated). + //! \param name Name of node to set. Does not have to be zero terminated. + //! \param size Size of name, in characters. This does not include zero terminator, if one is present. + void name(const Ch *name, std::size_t size) + { + m_name = const_cast<Ch *>(name); + m_name_size = size; + } + + //! Sets name of node to a zero-terminated string. + //! See also \ref ownership_of_strings and xml_node::name(const Ch *, std::size_t). + //! \param name Name of node to set. Must be zero terminated. + void name(const Ch *name) + { + this->name(name, internal::measure(name)); + } + + //! Sets value of node to a non zero-terminated string. + //! See \ref ownership_of_strings. + //! <br><br> + //! Note that node does not own its name or value, it only stores a pointer to it. + //! It will not delete or otherwise free the pointer on destruction. + //! It is reponsibility of the user to properly manage lifetime of the string. + //! The easiest way to achieve it is to use memory_pool of the document to allocate the string - + //! on destruction of the document the string will be automatically freed. + //! <br><br> + //! Size of value must be specified separately, because it does not have to be zero terminated. + //! Use value(const Ch *) function to have the length automatically calculated (string must be zero terminated). + //! <br><br> + //! If an element has a child node of type node_data, it will take precedence over element value when printing. + //! If you want to manipulate data of elements using values, use parser flag rapidxml::parse_no_data_nodes to prevent creation of data nodes by the parser. + //! \param value value of node to set. Does not have to be zero terminated. + //! \param size Size of value, in characters. This does not include zero terminator, if one is present. + void value(const Ch *value, std::size_t size) + { + m_value = const_cast<Ch *>(value); + m_value_size = size; + } + + //! Sets value of node to a zero-terminated string. + //! See also \ref ownership_of_strings and xml_node::value(const Ch *, std::size_t). + //! \param value Vame of node to set. Must be zero terminated. + void value(const Ch *value) + { + this->value(value, internal::measure(value)); + } + + /////////////////////////////////////////////////////////////////////////// + // Related nodes access + + //! Gets node parent. + //! \return Pointer to parent node, or 0 if there is no parent. + xml_node<Ch> *parent() const + { + return m_parent; + } + + protected: + + // Return empty string + static Ch *nullstr() + { + static Ch zero = Ch('\0'); + return &zero; + } + + Ch *m_name; // Name of node, or 0 if no name + Ch *m_value; // Value of node, or 0 if no value + std::size_t m_name_size; // Length of node name, or undefined of no name + std::size_t m_value_size; // Length of node value, or undefined if no value + xml_node<Ch> *m_parent; // Pointer to parent node, or 0 if none + + }; + + //! Class representing attribute node of XML document. + //! Each attribute has name and value strings, which are available through name() and value() functions (inherited from xml_base). + //! Note that after parse, both name and value of attribute will point to interior of source text used for parsing. + //! Thus, this text must persist in memory for the lifetime of attribute. + //! \param Ch Character type to use. + template<class Ch = char> + class xml_attribute: public xml_base<Ch> + { + + friend class xml_node<Ch>; + + public: + + /////////////////////////////////////////////////////////////////////////// + // Construction & destruction + + //! Constructs an empty attribute with the specified type. + //! Consider using memory_pool of appropriate xml_document if allocating attributes manually. + xml_attribute() + { + } + + /////////////////////////////////////////////////////////////////////////// + // Related nodes access + + //! Gets document of which attribute is a child. + //! \return Pointer to document that contains this attribute, or 0 if there is no parent document. + xml_document<Ch> *document() const + { + if (xml_node<Ch> *node = this->parent()) + { + while (node->parent()) + node = node->parent(); + return node->type() == node_document ? static_cast<xml_document<Ch> *>(node) : 0; + } + else + return 0; + } + + //! Gets previous attribute, optionally matching attribute name. + //! \param name Name of attribute to find, or 0 to return previous attribute regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero + //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string + //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters + //! \return Pointer to found attribute, or 0 if not found. + xml_attribute<Ch> *previous_attribute(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const + { + if (name) + { + if (name_size == 0) + name_size = internal::measure(name); + for (xml_attribute<Ch> *attribute = m_prev_attribute; attribute; attribute = attribute->m_prev_attribute) + if (internal::compare(attribute->name(), attribute->name_size(), name, name_size, case_sensitive)) + return attribute; + return 0; + } + else + return this->m_parent ? m_prev_attribute : 0; + } + + //! Gets next attribute, optionally matching attribute name. + //! \param name Name of attribute to find, or 0 to return next attribute regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero + //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string + //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters + //! \return Pointer to found attribute, or 0 if not found. + xml_attribute<Ch> *next_attribute(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const + { + if (name) + { + if (name_size == 0) + name_size = internal::measure(name); + for (xml_attribute<Ch> *attribute = m_next_attribute; attribute; attribute = attribute->m_next_attribute) + if (internal::compare(attribute->name(), attribute->name_size(), name, name_size, case_sensitive)) + return attribute; + return 0; + } + else + return this->m_parent ? m_next_attribute : 0; + } + + private: + + xml_attribute<Ch> *m_prev_attribute; // Pointer to previous sibling of attribute, or 0 if none; only valid if parent is non-zero + xml_attribute<Ch> *m_next_attribute; // Pointer to next sibling of attribute, or 0 if none; only valid if parent is non-zero + + }; + + /////////////////////////////////////////////////////////////////////////// + // XML node + + //! Class representing a node of XML document. + //! Each node may have associated name and value strings, which are available through name() and value() functions. + //! Interpretation of name and value depends on type of the node. + //! Type of node can be determined by using type() function. + //! <br><br> + //! Note that after parse, both name and value of node, if any, will point interior of source text used for parsing. + //! Thus, this text must persist in the memory for the lifetime of node. + //! \param Ch Character type to use. + template<class Ch = char> + class xml_node: public xml_base<Ch> + { + + public: + + /////////////////////////////////////////////////////////////////////////// + // Construction & destruction + + //! Constructs an empty node with the specified type. + //! Consider using memory_pool of appropriate document to allocate nodes manually. + //! \param type Type of node to construct. + xml_node(node_type type) + : m_type(type) + , m_first_node(0) + , m_first_attribute(0) + { + } + + /////////////////////////////////////////////////////////////////////////// + // Node data access + + //! Gets type of node. + //! \return Type of node. + node_type type() const + { + return m_type; + } + + /////////////////////////////////////////////////////////////////////////// + // Related nodes access + + //! Gets document of which node is a child. + //! \return Pointer to document that contains this node, or 0 if there is no parent document. + xml_document<Ch> *document() const + { + xml_node<Ch> *node = const_cast<xml_node<Ch> *>(this); + while (node->parent()) + node = node->parent(); + return node->type() == node_document ? static_cast<xml_document<Ch> *>(node) : 0; + } + + //! Gets first child node, optionally matching node name. + //! \param name Name of child to find, or 0 to return first child regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero + //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string + //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters + //! \return Pointer to found child, or 0 if not found. + xml_node<Ch> *first_node(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const + { + if (name) + { + if (name_size == 0) + name_size = internal::measure(name); + for (xml_node<Ch> *child = m_first_node; child; child = child->next_sibling()) + if (internal::compare(child->name(), child->name_size(), name, name_size, case_sensitive)) + return child; + return 0; + } + else + return m_first_node; + } + + //! Gets last child node, optionally matching node name. + //! Behaviour is undefined if node has no children. + //! Use first_node() to test if node has children. + //! \param name Name of child to find, or 0 to return last child regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero + //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string + //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters + //! \return Pointer to found child, or 0 if not found. + xml_node<Ch> *last_node(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const + { + assert(m_first_node); // Cannot query for last child if node has no children + if (name) + { + if (name_size == 0) + name_size = internal::measure(name); + for (xml_node<Ch> *child = m_last_node; child; child = child->previous_sibling()) + if (internal::compare(child->name(), child->name_size(), name, name_size, case_sensitive)) + return child; + return 0; + } + else + return m_last_node; + } + + //! Gets previous sibling node, optionally matching node name. + //! Behaviour is undefined if node has no parent. + //! Use parent() to test if node has a parent. + //! \param name Name of sibling to find, or 0 to return previous sibling regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero + //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string + //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters + //! \return Pointer to found sibling, or 0 if not found. + xml_node<Ch> *previous_sibling(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const + { + assert(this->m_parent); // Cannot query for siblings if node has no parent + if (name) + { + if (name_size == 0) + name_size = internal::measure(name); + for (xml_node<Ch> *sibling = m_prev_sibling; sibling; sibling = sibling->m_prev_sibling) + if (internal::compare(sibling->name(), sibling->name_size(), name, name_size, case_sensitive)) + return sibling; + return 0; + } + else + return m_prev_sibling; + } + + //! Gets next sibling node, optionally matching node name. + //! Behaviour is undefined if node has no parent. + //! Use parent() to test if node has a parent. + //! \param name Name of sibling to find, or 0 to return next sibling regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero + //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string + //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters + //! \return Pointer to found sibling, or 0 if not found. + xml_node<Ch> *next_sibling(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const + { + assert(this->m_parent); // Cannot query for siblings if node has no parent + if (name) + { + if (name_size == 0) + name_size = internal::measure(name); + for (xml_node<Ch> *sibling = m_next_sibling; sibling; sibling = sibling->m_next_sibling) + if (internal::compare(sibling->name(), sibling->name_size(), name, name_size, case_sensitive)) + return sibling; + return 0; + } + else + return m_next_sibling; + } + + //! Gets first attribute of node, optionally matching attribute name. + //! \param name Name of attribute to find, or 0 to return first attribute regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero + //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string + //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters + //! \return Pointer to found attribute, or 0 if not found. + xml_attribute<Ch> *first_attribute(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const + { + if (name) + { + if (name_size == 0) + name_size = internal::measure(name); + for (xml_attribute<Ch> *attribute = m_first_attribute; attribute; attribute = attribute->m_next_attribute) + if (internal::compare(attribute->name(), attribute->name_size(), name, name_size, case_sensitive)) + return attribute; + return 0; + } + else + return m_first_attribute; + } + + //! Gets last attribute of node, optionally matching attribute name. + //! \param name Name of attribute to find, or 0 to return last attribute regardless of its name; this string doesn't have to be zero-terminated if name_size is non-zero + //! \param name_size Size of name, in characters, or 0 to have size calculated automatically from string + //! \param case_sensitive Should name comparison be case-sensitive; non case-sensitive comparison works properly only for ASCII characters + //! \return Pointer to found attribute, or 0 if not found. + xml_attribute<Ch> *last_attribute(const Ch *name = 0, std::size_t name_size = 0, bool case_sensitive = true) const + { + if (name) + { + if (name_size == 0) + name_size = internal::measure(name); + for (xml_attribute<Ch> *attribute = m_last_attribute; attribute; attribute = attribute->m_prev_attribute) + if (internal::compare(attribute->name(), attribute->name_size(), name, name_size, case_sensitive)) + return attribute; + return 0; + } + else + return m_first_attribute ? m_last_attribute : 0; + } + + /////////////////////////////////////////////////////////////////////////// + // Node modification + + //! Sets type of node. + //! \param type Type of node to set. + void type(node_type type) + { + m_type = type; + } + + /////////////////////////////////////////////////////////////////////////// + // Node manipulation + + //! Prepends a new child node. + //! The prepended child becomes the first child, and all existing children are moved one position back. + //! \param child Node to prepend. + void prepend_node(xml_node<Ch> *child) + { + assert(child && !child->parent() && child->type() != node_document); + if (first_node()) + { + child->m_next_sibling = m_first_node; + m_first_node->m_prev_sibling = child; + } + else + { + child->m_next_sibling = 0; + m_last_node = child; + } + m_first_node = child; + child->m_parent = this; + child->m_prev_sibling = 0; + } + + //! Appends a new child node. + //! The appended child becomes the last child. + //! \param child Node to append. + void append_node(xml_node<Ch> *child) + { + assert(child && !child->parent() && child->type() != node_document); + if (first_node()) + { + child->m_prev_sibling = m_last_node; + m_last_node->m_next_sibling = child; + } + else + { + child->m_prev_sibling = 0; + m_first_node = child; + } + m_last_node = child; + child->m_parent = this; + child->m_next_sibling = 0; + } + + //! Inserts a new child node at specified place inside the node. + //! All children after and including the specified node are moved one position back. + //! \param where Place where to insert the child, or 0 to insert at the back. + //! \param child Node to insert. + void insert_node(xml_node<Ch> *where, xml_node<Ch> *child) + { + assert(!where || where->parent() == this); + assert(child && !child->parent() && child->type() != node_document); + if (where == m_first_node) + prepend_node(child); + else if (where == 0) + append_node(child); + else + { + child->m_prev_sibling = where->m_prev_sibling; + child->m_next_sibling = where; + where->m_prev_sibling->m_next_sibling = child; + where->m_prev_sibling = child; + child->m_parent = this; + } + } + + //! Removes first child node. + //! If node has no children, behaviour is undefined. + //! Use first_node() to test if node has children. + void remove_first_node() + { + assert(first_node()); + xml_node<Ch> *child = m_first_node; + m_first_node = child->m_next_sibling; + if (child->m_next_sibling) + child->m_next_sibling->m_prev_sibling = 0; + else + m_last_node = 0; + child->m_parent = 0; + } + + //! Removes last child of the node. + //! If node has no children, behaviour is undefined. + //! Use first_node() to test if node has children. + void remove_last_node() + { + assert(first_node()); + xml_node<Ch> *child = m_last_node; + if (child->m_prev_sibling) + { + m_last_node = child->m_prev_sibling; + child->m_prev_sibling->m_next_sibling = 0; + } + else + m_first_node = 0; + child->m_parent = 0; + } + + //! Removes specified child from the node + // \param where Pointer to child to be removed. + void remove_node(xml_node<Ch> *where) + { + assert(where && where->parent() == this); + assert(first_node()); + if (where == m_first_node) + remove_first_node(); + else if (where == m_last_node) + remove_last_node(); + else + { + where->m_prev_sibling->m_next_sibling = where->m_next_sibling; + where->m_next_sibling->m_prev_sibling = where->m_prev_sibling; + where->m_parent = 0; + } + } + + //! Removes all child nodes (but not attributes). + void remove_all_nodes() + { + for (xml_node<Ch> *node = first_node(); node; node = node->m_next_sibling) + node->m_parent = 0; + m_first_node = 0; + } + + //! Prepends a new attribute to the node. + //! \param attribute Attribute to prepend. + void prepend_attribute(xml_attribute<Ch> *attribute) + { + assert(attribute && !attribute->parent()); + if (first_attribute()) + { + attribute->m_next_attribute = m_first_attribute; + m_first_attribute->m_prev_attribute = attribute; + } + else + { + attribute->m_next_attribute = 0; + m_last_attribute = attribute; + } + m_first_attribute = attribute; + attribute->m_parent = this; + attribute->m_prev_attribute = 0; + } + + //! Appends a new attribute to the node. + //! \param attribute Attribute to append. + void append_attribute(xml_attribute<Ch> *attribute) + { + assert(attribute && !attribute->parent()); + if (first_attribute()) + { + attribute->m_prev_attribute = m_last_attribute; + m_last_attribute->m_next_attribute = attribute; + } + else + { + attribute->m_prev_attribute = 0; + m_first_attribute = attribute; + } + m_last_attribute = attribute; + attribute->m_parent = this; + attribute->m_next_attribute = 0; + } + + //! Inserts a new attribute at specified place inside the node. + //! All attributes after and including the specified attribute are moved one position back. + //! \param where Place where to insert the attribute, or 0 to insert at the back. + //! \param attribute Attribute to insert. + void insert_attribute(xml_attribute<Ch> *where, xml_attribute<Ch> *attribute) + { + assert(!where || where->parent() == this); + assert(attribute && !attribute->parent()); + if (where == m_first_attribute) + prepend_attribute(attribute); + else if (where == 0) + append_attribute(attribute); + else + { + attribute->m_prev_attribute = where->m_prev_attribute; + attribute->m_next_attribute = where; + where->m_prev_attribute->m_next_attribute = attribute; + where->m_prev_attribute = attribute; + attribute->m_parent = this; + } + } + + //! Removes first attribute of the node. + //! If node has no attributes, behaviour is undefined. + //! Use first_attribute() to test if node has attributes. + void remove_first_attribute() + { + assert(first_attribute()); + xml_attribute<Ch> *attribute = m_first_attribute; + if (attribute->m_next_attribute) + { + attribute->m_next_attribute->m_prev_attribute = 0; + } + else + m_last_attribute = 0; + attribute->m_parent = 0; + m_first_attribute = attribute->m_next_attribute; + } + + //! Removes last attribute of the node. + //! If node has no attributes, behaviour is undefined. + //! Use first_attribute() to test if node has attributes. + void remove_last_attribute() + { + assert(first_attribute()); + xml_attribute<Ch> *attribute = m_last_attribute; + if (attribute->m_prev_attribute) + { + attribute->m_prev_attribute->m_next_attribute = 0; + m_last_attribute = attribute->m_prev_attribute; + } + else + m_first_attribute = 0; + attribute->m_parent = 0; + } + + //! Removes specified attribute from node. + //! \param where Pointer to attribute to be removed. + void remove_attribute(xml_attribute<Ch> *where) + { + assert(first_attribute() && where->parent() == this); + if (where == m_first_attribute) + remove_first_attribute(); + else if (where == m_last_attribute) + remove_last_attribute(); + else + { + where->m_prev_attribute->m_next_attribute = where->m_next_attribute; + where->m_next_attribute->m_prev_attribute = where->m_prev_attribute; + where->m_parent = 0; + } + } + + //! Removes all attributes of node. + void remove_all_attributes() + { + for (xml_attribute<Ch> *attribute = first_attribute(); attribute; attribute = attribute->m_next_attribute) + attribute->m_parent = 0; + m_first_attribute = 0; + } + + private: + + /////////////////////////////////////////////////////////////////////////// + // Restrictions + + // No copying + xml_node(const xml_node &); + void operator =(const xml_node &); + + /////////////////////////////////////////////////////////////////////////// + // Data members + + // Note that some of the pointers below have UNDEFINED values if certain other pointers are 0. + // This is required for maximum performance, as it allows the parser to omit initialization of + // unneded/redundant values. + // + // The rules are as follows: + // 1. first_node and first_attribute contain valid pointers, or 0 if node has no children/attributes respectively + // 2. last_node and last_attribute are valid only if node has at least one child/attribute respectively, otherwise they contain garbage + // 3. prev_sibling and next_sibling are valid only if node has a parent, otherwise they contain garbage + + node_type m_type; // Type of node; always valid + xml_node<Ch> *m_first_node; // Pointer to first child node, or 0 if none; always valid + xml_node<Ch> *m_last_node; // Pointer to last child node, or 0 if none; this value is only valid if m_first_node is non-zero + xml_attribute<Ch> *m_first_attribute; // Pointer to first attribute of node, or 0 if none; always valid + xml_attribute<Ch> *m_last_attribute; // Pointer to last attribute of node, or 0 if none; this value is only valid if m_first_attribute is non-zero + xml_node<Ch> *m_prev_sibling; // Pointer to previous sibling of node, or 0 if none; this value is only valid if m_parent is non-zero + xml_node<Ch> *m_next_sibling; // Pointer to next sibling of node, or 0 if none; this value is only valid if m_parent is non-zero + + }; + + /////////////////////////////////////////////////////////////////////////// + // XML document + + //! This class represents root of the DOM hierarchy. + //! It is also an xml_node and a memory_pool through public inheritance. + //! Use parse() function to build a DOM tree from a zero-terminated XML text string. + //! parse() function allocates memory for nodes and attributes by using functions of xml_document, + //! which are inherited from memory_pool. + //! To access root node of the document, use the document itself, as if it was an xml_node. + //! \param Ch Character type to use. + template<class Ch = char> + class xml_document: public xml_node<Ch>, public memory_pool<Ch> + { + + public: + + //! Constructs empty XML document + xml_document() + : xml_node<Ch>(node_document) + { + } + + //! Parses zero-terminated XML string according to given flags. + //! Passed string will be modified by the parser, unless rapidxml::parse_non_destructive flag is used. + //! The string must persist for the lifetime of the document. + //! In case of error, rapidxml::parse_error exception will be thrown. + //! <br><br> + //! If you want to parse contents of a file, you must first load the file into the memory, and pass pointer to its beginning. + //! Make sure that data is zero-terminated. + //! <br><br> + //! Document can be parsed into multiple times. + //! Each new call to parse removes previous nodes and attributes (if any), but does not clear memory pool. + //! \param text XML data to parse; pointer is non-const to denote fact that this data may be modified by the parser. + template<int Flags> + void parse(Ch *text) + { + assert(text); + + // Remove current contents + this->remove_all_nodes(); + this->remove_all_attributes(); + + // Parse BOM, if any + parse_bom<Flags>(text); + + // Parse children + while (1) + { + // Skip whitespace before node + skip<whitespace_pred, Flags>(text); + if (*text == 0) + break; + + // Parse and append new child + if (*text == Ch('<')) + { + ++text; // Skip '<' + if (xml_node<Ch> *node = parse_node<Flags>(text)) + this->append_node(node); + } + else + RAPIDXML_PARSE_ERROR("expected <", text); + } + + } + + //! Clears the document by deleting all nodes and clearing the memory pool. + //! All nodes owned by document pool are destroyed. + void clear() + { + this->remove_all_nodes(); + this->remove_all_attributes(); + memory_pool<Ch>::clear(); + } + + private: + + /////////////////////////////////////////////////////////////////////// + // Internal character utility functions + + // Detect whitespace character + struct whitespace_pred + { + static unsigned char test(Ch ch) + { + return internal::lookup_tables<0>::lookup_whitespace[static_cast<unsigned char>(ch)]; + } + }; + + // Detect node name character + struct node_name_pred + { + static unsigned char test(Ch ch) + { + return internal::lookup_tables<0>::lookup_node_name[static_cast<unsigned char>(ch)]; + } + }; + + // Detect attribute name character + struct attribute_name_pred + { + static unsigned char test(Ch ch) + { + return internal::lookup_tables<0>::lookup_attribute_name[static_cast<unsigned char>(ch)]; + } + }; + + // Detect text character (PCDATA) + struct text_pred + { + static unsigned char test(Ch ch) + { + return internal::lookup_tables<0>::lookup_text[static_cast<unsigned char>(ch)]; + } + }; + + // Detect text character (PCDATA) that does not require processing + struct text_pure_no_ws_pred + { + static unsigned char test(Ch ch) + { + return internal::lookup_tables<0>::lookup_text_pure_no_ws[static_cast<unsigned char>(ch)]; + } + }; + + // Detect text character (PCDATA) that does not require processing + struct text_pure_with_ws_pred + { + static unsigned char test(Ch ch) + { + return internal::lookup_tables<0>::lookup_text_pure_with_ws[static_cast<unsigned char>(ch)]; + } + }; + + // Detect attribute value character + template<Ch Quote> + struct attribute_value_pred + { + static unsigned char test(Ch ch) + { + if (Quote == Ch('\'')) + return internal::lookup_tables<0>::lookup_attribute_data_1[static_cast<unsigned char>(ch)]; + if (Quote == Ch('\"')) + return internal::lookup_tables<0>::lookup_attribute_data_2[static_cast<unsigned char>(ch)]; + return 0; // Should never be executed, to avoid warnings on Comeau + } + }; + + // Detect attribute value character + template<Ch Quote> + struct attribute_value_pure_pred + { + static unsigned char test(Ch ch) + { + if (Quote == Ch('\'')) + return internal::lookup_tables<0>::lookup_attribute_data_1_pure[static_cast<unsigned char>(ch)]; + if (Quote == Ch('\"')) + return internal::lookup_tables<0>::lookup_attribute_data_2_pure[static_cast<unsigned char>(ch)]; + return 0; // Should never be executed, to avoid warnings on Comeau + } + }; + + // Insert coded character, using UTF8 or 8-bit ASCII + template<int Flags> + static void insert_coded_character(Ch *&text, unsigned long code) + { + if (Flags & parse_no_utf8) + { + // Insert 8-bit ASCII character + // Todo: possibly verify that code is less than 256 and use replacement char otherwise? + text[0] = static_cast<unsigned char>(code); + text += 1; + } + else + { + // Insert UTF8 sequence + if (code < 0x80) // 1 byte sequence + { + text[0] = static_cast<unsigned char>(code); + text += 1; + } + else if (code < 0x800) // 2 byte sequence + { + text[1] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6; + text[0] = static_cast<unsigned char>(code | 0xC0); + text += 2; + } + else if (code < 0x10000) // 3 byte sequence + { + text[2] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6; + text[1] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6; + text[0] = static_cast<unsigned char>(code | 0xE0); + text += 3; + } + else if (code < 0x110000) // 4 byte sequence + { + text[3] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6; + text[2] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6; + text[1] = static_cast<unsigned char>((code | 0x80) & 0xBF); code >>= 6; + text[0] = static_cast<unsigned char>(code | 0xF0); + text += 4; + } + else // Invalid, only codes up to 0x10FFFF are allowed in Unicode + { + RAPIDXML_PARSE_ERROR("invalid numeric character entity", text); + } + } + } + + // Skip characters until predicate evaluates to true + template<class StopPred, int Flags> + static void skip(Ch *&text) + { + Ch *tmp = text; + while (StopPred::test(*tmp)) + ++tmp; + text = tmp; + } + + // Skip characters until predicate evaluates to true while doing the following: + // - replacing XML character entity references with proper characters (' & " < > &#...;) + // - condensing whitespace sequences to single space character + template<class StopPred, class StopPredPure, int Flags> + static Ch *skip_and_expand_character_refs(Ch *&text) + { + // If entity translation, whitespace condense and whitespace trimming is disabled, use plain skip + if (Flags & parse_no_entity_translation && + !(Flags & parse_normalize_whitespace) && + !(Flags & parse_trim_whitespace)) + { + skip<StopPred, Flags>(text); + return text; + } + + // Use simple skip until first modification is detected + skip<StopPredPure, Flags>(text); + + // Use translation skip + Ch *src = text; + Ch *dest = src; + while (StopPred::test(*src)) + { + // If entity translation is enabled + if (!(Flags & parse_no_entity_translation)) + { + // Test if replacement is needed + if (src[0] == Ch('&')) + { + switch (src[1]) + { + + // & ' + case Ch('a'): + if (src[2] == Ch('m') && src[3] == Ch('p') && src[4] == Ch(';')) + { + *dest = Ch('&'); + ++dest; + src += 5; + continue; + } + if (src[2] == Ch('p') && src[3] == Ch('o') && src[4] == Ch('s') && src[5] == Ch(';')) + { + *dest = Ch('\''); + ++dest; + src += 6; + continue; + } + break; + + // " + case Ch('q'): + if (src[2] == Ch('u') && src[3] == Ch('o') && src[4] == Ch('t') && src[5] == Ch(';')) + { + *dest = Ch('"'); + ++dest; + src += 6; + continue; + } + break; + + // > + case Ch('g'): + if (src[2] == Ch('t') && src[3] == Ch(';')) + { + *dest = Ch('>'); + ++dest; + src += 4; + continue; + } + break; + + // < + case Ch('l'): + if (src[2] == Ch('t') && src[3] == Ch(';')) + { + *dest = Ch('<'); + ++dest; + src += 4; + continue; + } + break; + + // &#...; - assumes ASCII + case Ch('#'): + if (src[2] == Ch('x')) + { + unsigned long code = 0; + src += 3; // Skip &#x + while (1) + { + unsigned char digit = internal::lookup_tables<0>::lookup_digits[static_cast<unsigned char>(*src)]; + if (digit == 0xFF) + break; + code = code * 16 + digit; + ++src; + } + insert_coded_character<Flags>(dest, code); // Put character in output + } + else + { + unsigned long code = 0; + src += 2; // Skip &# + while (1) + { + unsigned char digit = internal::lookup_tables<0>::lookup_digits[static_cast<unsigned char>(*src)]; + if (digit == 0xFF) + break; + code = code * 10 + digit; + ++src; + } + insert_coded_character<Flags>(dest, code); // Put character in output + } + if (*src == Ch(';')) + ++src; + else + RAPIDXML_PARSE_ERROR("expected ;", src); + continue; + + // Something else + default: + // Ignore, just copy '&' verbatim + break; + + } + } + } + + // If whitespace condensing is enabled + if (Flags & parse_normalize_whitespace) + { + // Test if condensing is needed + if (whitespace_pred::test(*src)) + { + *dest = Ch(' '); ++dest; // Put single space in dest + ++src; // Skip first whitespace char + // Skip remaining whitespace chars + while (whitespace_pred::test(*src)) + ++src; + continue; + } + } + + // No replacement, only copy character + *dest++ = *src++; + + } + + // Return new end + text = src; + return dest; + + } + + /////////////////////////////////////////////////////////////////////// + // Internal parsing functions + + // Parse BOM, if any + template<int Flags> + void parse_bom(Ch *&text) + { + // UTF-8? + if (static_cast<unsigned char>(text[0]) == 0xEF && + static_cast<unsigned char>(text[1]) == 0xBB && + static_cast<unsigned char>(text[2]) == 0xBF) + { + text += 3; // Skup utf-8 bom + } + } + + // Parse XML declaration (<?xml...) + template<int Flags> + xml_node<Ch> *parse_xml_declaration(Ch *&text) + { + // If parsing of declaration is disabled + if (!(Flags & parse_declaration_node)) + { + // Skip until end of declaration + while (text[0] != Ch('?') || text[1] != Ch('>')) + { + if (!text[0]) { + RAPIDXML_PARSE_ERROR("unexpected end of data", text); + return 0; + } + ++text; + } + text += 2; // Skip '?>' + return 0; + } + + // Create declaration + xml_node<Ch> *declaration = this->allocate_node(node_declaration); + + // Skip whitespace before attributes or ?> + skip<whitespace_pred, Flags>(text); + + // Parse declaration attributes + parse_node_attributes<Flags>(text, declaration); + + // Skip ?> + if (text[0] != Ch('?') || text[1] != Ch('>')) + RAPIDXML_PARSE_ERROR("expected ?>", text); + text += 2; + + return declaration; + } + + // Parse XML comment (<!--...) + template<int Flags> + xml_node<Ch> *parse_comment(Ch *&text) + { + // If parsing of comments is disabled + if (!(Flags & parse_comment_nodes)) + { + // Skip until end of comment + while (text[0] != Ch('-') || text[1] != Ch('-') || text[2] != Ch('>')) + { + if (!text[0]) { + RAPIDXML_PARSE_ERROR("unexpected end of data", text); + return 0; + } + ++text; + } + text += 3; // Skip '-->' + return 0; // Do not produce comment node + } + + // Remember value start + Ch *value = text; + + // Skip until end of comment + while (text[0] != Ch('-') || text[1] != Ch('-') || text[2] != Ch('>')) + { + if (!text[0]) { + RAPIDXML_PARSE_ERROR("unexpected end of data", text); + return 0; + } + ++text; + } + + // Create comment node + xml_node<Ch> *comment = this->allocate_node(node_comment); + comment->value(value, text - value); + + // Place zero terminator after comment value + if (!(Flags & parse_no_string_terminators)) + *text = Ch('\0'); + + text += 3; // Skip '-->' + return comment; + } + + // Parse DOCTYPE + template<int Flags> + xml_node<Ch> *parse_doctype(Ch *&text) + { + // Remember value start + Ch *value = text; + + // Skip to > + while (*text != Ch('>')) + { + // Determine character type + switch (*text) + { + + // If '[' encountered, scan for matching ending ']' using naive algorithm with depth + // This works for all W3C test files except for 2 most wicked + case Ch('['): + { + ++text; // Skip '[' + int depth = 1; + while (depth > 0) + { + switch (*text) + { + case Ch('['): ++depth; break; + case Ch(']'): --depth; break; + case 0: RAPIDXML_PARSE_ERROR("unexpected end of data", text); return 0; + } + ++text; + } + break; + } + + // Error on end of text + case Ch('\0'): + RAPIDXML_PARSE_ERROR("unexpected end of data", text); + return 0; + + // Other character, skip it + default: + ++text; + + } + } + + // If DOCTYPE nodes enabled + if (Flags & parse_doctype_node) + { + // Create a new doctype node + xml_node<Ch> *doctype = this->allocate_node(node_doctype); + doctype->value(value, text - value); + + // Place zero terminator after value + if (!(Flags & parse_no_string_terminators)) + *text = Ch('\0'); + + text += 1; // skip '>' + return doctype; + } + else + { + text += 1; // skip '>' + return 0; + } + + } + + // Parse PI + template<int Flags> + xml_node<Ch> *parse_pi(Ch *&text) + { + // If creation of PI nodes is enabled + if (Flags & parse_pi_nodes) + { + // Create pi node + xml_node<Ch> *pi = this->allocate_node(node_pi); + + // Extract PI target name + Ch *name = text; + skip<node_name_pred, Flags>(text); + if (text == name) + RAPIDXML_PARSE_ERROR("expected PI target", text); + pi->name(name, text - name); + + // Skip whitespace between pi target and pi + skip<whitespace_pred, Flags>(text); + + // Remember start of pi + Ch *value = text; + + // Skip to '?>' + while (text[0] != Ch('?') || text[1] != Ch('>')) + { + if (*text == Ch('\0')) { + RAPIDXML_PARSE_ERROR("unexpected end of data", text); + return 0; + } + ++text; + } + + // Set pi value (verbatim, no entity expansion or whitespace normalization) + pi->value(value, text - value); + + // Place zero terminator after name and value + if (!(Flags & parse_no_string_terminators)) + { + pi->name()[pi->name_size()] = Ch('\0'); + pi->value()[pi->value_size()] = Ch('\0'); + } + + text += 2; // Skip '?>' + return pi; + } + else + { + // Skip to '?>' + while (text[0] != Ch('?') || text[1] != Ch('>')) + { + if (*text == Ch('\0')) { + RAPIDXML_PARSE_ERROR("unexpected end of data", text); + return 0; + } + ++text; + } + text += 2; // Skip '?>' + return 0; + } + } + + // Parse and append data + // Return character that ends data. + // This is necessary because this character might have been overwritten by a terminating 0 + template<int Flags> + Ch parse_and_append_data(xml_node<Ch> *node, Ch *&text, Ch *contents_start) + { + // Backup to contents start if whitespace trimming is disabled + if (!(Flags & parse_trim_whitespace)) + text = contents_start; + + // Skip until end of data + Ch *value = text, *end; + if (Flags & parse_normalize_whitespace) + end = skip_and_expand_character_refs<text_pred, text_pure_with_ws_pred, Flags>(text); + else + end = skip_and_expand_character_refs<text_pred, text_pure_no_ws_pred, Flags>(text); + + // Trim trailing whitespace if flag is set; leading was already trimmed by whitespace skip after > + if (Flags & parse_trim_whitespace) + { + if (Flags & parse_normalize_whitespace) + { + // Whitespace is already condensed to single space characters by skipping function, so just trim 1 char off the end + if (*(end - 1) == Ch(' ')) + --end; + } + else + { + // Backup until non-whitespace character is found + while (whitespace_pred::test(*(end - 1))) + --end; + } + } + + // If characters are still left between end and value (this test is only necessary if normalization is enabled) + // Create new data node + if (!(Flags & parse_no_data_nodes)) + { + xml_node<Ch> *data = this->allocate_node(node_data); + data->value(value, end - value); + node->append_node(data); + } + + // Add data to parent node if no data exists yet + if (!(Flags & parse_no_element_values)) + if (*node->value() == Ch('\0')) + node->value(value, end - value); + + // Place zero terminator after value + if (!(Flags & parse_no_string_terminators)) + { + Ch ch = *text; + *end = Ch('\0'); + return ch; // Return character that ends data; this is required because zero terminator overwritten it + } + + // Return character that ends data + return *text; + } + + // Parse CDATA + template<int Flags> + xml_node<Ch> *parse_cdata(Ch *&text) + { + // If CDATA is disabled + if (Flags & parse_no_data_nodes) + { + // Skip until end of cdata + while (text[0] != Ch(']') || text[1] != Ch(']') || text[2] != Ch('>')) + { + if (!text[0]) { + RAPIDXML_PARSE_ERROR("unexpected end of data", text); + return 0; + } + ++text; + } + text += 3; // Skip ]]> + return 0; // Do not produce CDATA node + } + + // Skip until end of cdata + Ch *value = text; + while (text[0] != Ch(']') || text[1] != Ch(']') || text[2] != Ch('>')) + { + if (!text[0]) { + RAPIDXML_PARSE_ERROR("unexpected end of data", text); + return 0; + } + ++text; + } + + // Create new cdata node + xml_node<Ch> *cdata = this->allocate_node(node_cdata); + cdata->value(value, text - value); + + // Place zero terminator after value + if (!(Flags & parse_no_string_terminators)) + *text = Ch('\0'); + + text += 3; // Skip ]]> + return cdata; + } + + // Parse element node + template<int Flags> + xml_node<Ch> *parse_element(Ch *&text) + { + // Create element node + xml_node<Ch> *element = this->allocate_node(node_element); + + // Extract element name + Ch *name = text; + skip<node_name_pred, Flags>(text); + if (text == name) + RAPIDXML_PARSE_ERROR("expected element name", text); + element->name(name, text - name); + + // Skip whitespace between element name and attributes or > + skip<whitespace_pred, Flags>(text); + + // Parse attributes, if any + parse_node_attributes<Flags>(text, element); + + // Determine ending type + if (*text == Ch('>')) + { + ++text; + parse_node_contents<Flags>(text, element); + } + else if (*text == Ch('/')) + { + ++text; + if (*text != Ch('>')) + RAPIDXML_PARSE_ERROR("expected >", text); + ++text; + } + else + RAPIDXML_PARSE_ERROR("expected >", text); + + // Place zero terminator after name + if (!(Flags & parse_no_string_terminators)) + element->name()[element->name_size()] = Ch('\0'); + + // Return parsed element + return element; + } + + // Determine node type, and parse it + template<int Flags> + xml_node<Ch> *parse_node(Ch *&text) + { + // Parse proper node type + switch (text[0]) + { + + // <... + default: + // Parse and append element node + return parse_element<Flags>(text); + + // <?... + case Ch('?'): + ++text; // Skip ? + if ((text[0] == Ch('x') || text[0] == Ch('X')) && + (text[1] == Ch('m') || text[1] == Ch('M')) && + (text[2] == Ch('l') || text[2] == Ch('L')) && + whitespace_pred::test(text[3])) + { + // '<?xml ' - xml declaration + text += 4; // Skip 'xml ' + return parse_xml_declaration<Flags>(text); + } + else + { + // Parse PI + return parse_pi<Flags>(text); + } + + // <!... + case Ch('!'): + + // Parse proper subset of <! node + switch (text[1]) + { + + // <!- + case Ch('-'): + if (text[2] == Ch('-')) + { + // '<!--' - xml comment + text += 3; // Skip '!--' + return parse_comment<Flags>(text); + } + break; + + // <![ + case Ch('['): + if (text[2] == Ch('C') && text[3] == Ch('D') && text[4] == Ch('A') && + text[5] == Ch('T') && text[6] == Ch('A') && text[7] == Ch('[')) + { + // '<![CDATA[' - cdata + text += 8; // Skip '![CDATA[' + return parse_cdata<Flags>(text); + } + break; + + // <!D + case Ch('D'): + if (text[2] == Ch('O') && text[3] == Ch('C') && text[4] == Ch('T') && + text[5] == Ch('Y') && text[6] == Ch('P') && text[7] == Ch('E') && + whitespace_pred::test(text[8])) + { + // '<!DOCTYPE ' - doctype + text += 9; // skip '!DOCTYPE ' + return parse_doctype<Flags>(text); + } + + } // switch + + // Attempt to skip other, unrecognized node types starting with <! + ++text; // Skip ! + while (*text != Ch('>')) + { + if (*text == 0) { + RAPIDXML_PARSE_ERROR("unexpected end of data", text); + return 0; + } + ++text; + } + ++text; // Skip '>' + return 0; // No node recognized + + } + } + + // Parse contents of the node - children, data etc. + template<int Flags> + void parse_node_contents(Ch *&text, xml_node<Ch> *node) + { + // For all children and text + while (1) + { + // Skip whitespace between > and node contents + Ch *contents_start = text; // Store start of node contents before whitespace is skipped + skip<whitespace_pred, Flags>(text); + Ch next_char = *text; + + // After data nodes, instead of continuing the loop, control jumps here. + // This is because zero termination inside parse_and_append_data() function + // would wreak havoc with the above code. + // Also, skipping whitespace after data nodes is unnecessary. + after_data_node: + + // Determine what comes next: node closing, child node, data node, or 0? + switch (next_char) + { + + // Node closing or child node + case Ch('<'): + if (text[1] == Ch('/')) + { + // Node closing + text += 2; // Skip '</' + if (Flags & parse_validate_closing_tags) + { + // Skip and validate closing tag name + Ch *closing_name = text; + skip<node_name_pred, Flags>(text); + if (!internal::compare(node->name(), node->name_size(), closing_name, text - closing_name, true)) + RAPIDXML_PARSE_ERROR("invalid closing tag name", text); + } + else + { + // No validation, just skip name + skip<node_name_pred, Flags>(text); + } + // Skip remaining whitespace after node name + skip<whitespace_pred, Flags>(text); + if (*text != Ch('>')) + RAPIDXML_PARSE_ERROR("expected >", text); + ++text; // Skip '>' + return; // Node closed, finished parsing contents + } + else + { + // Child node + ++text; // Skip '<' + if (xml_node<Ch> *child = parse_node<Flags>(text)) + node->append_node(child); + } + break; + + // End of data - error + case Ch('\0'): + RAPIDXML_PARSE_ERROR("unexpected end of data", text); + return; + + // Data node + default: + next_char = parse_and_append_data<Flags>(node, text, contents_start); + goto after_data_node; // Bypass regular processing after data nodes + + } + } + } + + // Parse XML attributes of the node + template<int Flags> + void parse_node_attributes(Ch *&text, xml_node<Ch> *node) + { + // For all attributes + while (attribute_name_pred::test(*text)) + { + // Extract attribute name + Ch *name = text; + ++text; // Skip first character of attribute name + skip<attribute_name_pred, Flags>(text); + if (text == name) + RAPIDXML_PARSE_ERROR("expected attribute name", name); + + // Create new attribute + xml_attribute<Ch> *attribute = this->allocate_attribute(); + attribute->name(name, text - name); + node->append_attribute(attribute); + + // Skip whitespace after attribute name + skip<whitespace_pred, Flags>(text); + + // Skip = + if (*text != Ch('=')) + RAPIDXML_PARSE_ERROR("expected =", text); + ++text; + + // Add terminating zero after name + if (!(Flags & parse_no_string_terminators)) + attribute->name()[attribute->name_size()] = 0; + + // Skip whitespace after = + skip<whitespace_pred, Flags>(text); + + // Skip quote and remember if it was ' or " + Ch quote = *text; + if (quote != Ch('\'') && quote != Ch('"')) + RAPIDXML_PARSE_ERROR("expected ' or \"", text); + ++text; + + // Extract attribute value and expand char refs in it + Ch *value = text, *end; + const int AttFlags = Flags & ~parse_normalize_whitespace; // No whitespace normalization in attributes + if (quote == Ch('\'')) + end = skip_and_expand_character_refs<attribute_value_pred<Ch('\'')>, attribute_value_pure_pred<Ch('\'')>, AttFlags>(text); + else + end = skip_and_expand_character_refs<attribute_value_pred<Ch('"')>, attribute_value_pure_pred<Ch('"')>, AttFlags>(text); + + // Set attribute value + attribute->value(value, end - value); + + // Make sure that end quote is present + if (*text != quote) + RAPIDXML_PARSE_ERROR("expected ' or \"", text); + ++text; // Skip quote + + // Add terminating zero after value + if (!(Flags & parse_no_string_terminators)) + attribute->value()[attribute->value_size()] = 0; + + // Skip whitespace after attribute value + skip<whitespace_pred, Flags>(text); + } + } + + }; + + //! \cond internal + namespace internal + { + + // Whitespace (space \n \r \t) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_whitespace[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 1, 0, 0, // 0 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 1 + 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 2 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 3 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 4 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 5 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 6 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 7 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 8 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 9 + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // A + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // B + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // C + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // D + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // E + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 // F + }; + + // Node name (anything but space \n \r \t / > ? \0) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_node_name[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 0, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, // 0 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1 + 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, // 2 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, // 3 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F + }; + + // Text (i.e. PCDATA) (anything but < \0) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_text[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, // 3 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F + }; + + // Text (i.e. PCDATA) that does not require processing when ws normalization is disabled + // (anything but < \0 &) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_text_pure_no_ws[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1 + 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, // 3 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F + }; + + // Text (i.e. PCDATA) that does not require processing when ws normalizationis is enabled + // (anything but < \0 & space \n \r \t) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_text_pure_with_ws[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 0, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, // 0 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1 + 0, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, // 3 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F + }; + + // Attribute name (anything but space \n \r \t / < > = ? ! \0) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_attribute_name[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 0, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, // 0 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1 + 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, // 2 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, // 3 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F + }; + + // Attribute data with single quote (anything but ' \0) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_attribute_data_1[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1 + 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, // 2 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 3 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F + }; + + // Attribute data with single quote that does not require processing (anything but ' \0 &) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_attribute_data_1_pure[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1 + 1, 1, 1, 1, 1, 1, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, // 2 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 3 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F + }; + + // Attribute data with double quote (anything but " \0) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_attribute_data_2[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1 + 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 3 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F + }; + + // Attribute data with double quote that does not require processing (anything but " \0 &) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_attribute_data_2_pure[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 0 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 1 + 1, 1, 0, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 2 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 3 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 4 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 5 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 6 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 7 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 8 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 9 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // C + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // D + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // E + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F + }; + + // Digits (dec and hex, 255 denotes end of numeric character reference) + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_digits[256] = + { + // 0 1 2 3 4 5 6 7 8 9 A B C D E F + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 0 + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 1 + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 2 + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9,255,255,255,255,255,255, // 3 + 255, 10, 11, 12, 13, 14, 15,255,255,255,255,255,255,255,255,255, // 4 + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 5 + 255, 10, 11, 12, 13, 14, 15,255,255,255,255,255,255,255,255,255, // 6 + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 7 + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 8 + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // 9 + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // A + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // B + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // C + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // D + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255, // E + 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255 // F + }; + + // Upper case conversion + template<int Dummy> + const unsigned char lookup_tables<Dummy>::lookup_upcase[256] = + { + // 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, A B C D E F + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, // 0 + 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, // 1 + 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, // 2 + 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, // 3 + 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, // 4 + 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95, // 5 + 96, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, // 6 + 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 123,124,125,126,127, // 7 + 128,129,130,131,132,133,134,135,136,137,138,139,140,141,142,143, // 8 + 144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159, // 9 + 160,161,162,163,164,165,166,167,168,169,170,171,172,173,174,175, // A + 176,177,178,179,180,181,182,183,184,185,186,187,188,189,190,191, // B + 192,193,194,195,196,197,198,199,200,201,202,203,204,205,206,207, // C + 208,209,210,211,212,213,214,215,216,217,218,219,220,221,222,223, // D + 224,225,226,227,228,229,230,231,232,233,234,235,236,237,238,239, // E + 240,241,242,243,244,245,246,247,248,249,250,251,252,253,254,255 // F + }; + } + //! \endcond + +} + +// Undefine internal macros +#undef RAPIDXML_PARSE_ERROR + +// On MSVC, restore warnings state +#ifdef _MSC_VER + #pragma warning(pop) +#endif + +#endif diff --git a/gui/theme/common/languages/de.xml b/gui/theme/common/languages/de.xml index b09852cec..438951e5f 100755..100644 --- a/gui/theme/common/languages/de.xml +++ b/gui/theme/common/languages/de.xml @@ -1,670 +1,670 @@ -<?xml version="1.0"?>
-
-<language>
- <display>Deutsch</display>
-
- <resources>
- <!-- Font overrides - only change these if your language requires special characters -->
- <resource name="font_l" type="fontoverride" filename="RobotoCondensed-Regular.ttf" scale="100" />
- <resource name="font_m" type="fontoverride" filename="RobotoCondensed-Regular.ttf" scale="100" />
- <resource name="font_s" type="fontoverride" filename="RobotoCondensed-Regular.ttf" scale="100" />
- <resource name="fixed" type="fontoverride" filename="DroidSansMono.ttf" scale="100" />
-
- <!-- Partition display names -->
- <string name="system">System</string>
- <string name="system_image">System-Image</string>
- <string name="vendor">Anbieter</string>
- <string name="vendor_image">Anbieter-Image</string>
- <string name="boot">Boot</string>
- <string name="recovery">Recovery</string>
- <string name="cache">Cache</string>
- <string name="data">Daten</string>
- <string name="sdcard">SD-Karte</string>
- <string name="internal">Interner Speicher</string>
- <string name="microsd">Micro SD-Karte</string>
- <string name="usbotg">USB-OTG</string>
- <string name="android_secure">Android Secure</string>
- <string name="dalvik">Dalvik / ART Cache</string>
- <!-- This is a rarely used partition on a Micro SD card for a very old app2sd system -->
- <string name="sdext">SD-EXT</string>
- <string name="adopted_data">Eingegliederte Daten-Partition</string>
- <string name="adopted_storage">Eingegliederte Storage-Partition</string>
-
- <!-- GUI XML strings -->
- <string name="twrp_header">Team Win Recovery Project</string>
- <string name="twrp_watch_header">TWRP %tw_version%</string>
- <string name="cpu_temp">CPU: %tw_cpu_temp% °C</string>
- <string name="battery_pct">Akku: %tw_battery%</string>
- <string name="sort_by_name">Nach Namen sortieren</string>
- <string name="sort_by_date">Nach Datum sortieren</string>
- <string name="sort_by_size">Nach Größe sortieren</string>
- <string name="sort_by_name_only">Name</string>
- <string name="sort_by_date_only">Datum</string>
- <string name="sort_by_size_only">Größe</string>
- <string name="tab_general">ALLGEMEIN</string>
- <string name="tab_options">OPTIONEN</string>
- <string name="tab_backup">SICHERUNG</string>
- <string name="tab_time_zone">ZEITZONE</string>
- <string name="tab_screen">BILDSCHIRM</string>
- <string name="tab_vibration">VIBRATION</string>
- <string name="tab_language">SPRACHE</string>
-
- <string name="install_btn">Installieren</string>
- <string name="wipe_btn">Löschen</string>
- <string name="backup_btn">Sichern</string>
- <string name="restore_btn">Wiederherstellen</string>
- <string name="mount_btn">Einhängen</string>
- <string name="settings_btn">Einstellungen</string>
- <string name="advanced_btn">Erweitert</string>
- <string name="reboot_btn">Neustart</string>
- <string name="files_btn">Dateien</string>
- <string name="copy_log_btn">Log kopieren</string>
- <string name="select_type_hdr">Typ wählen</string>
- <string name="install_zip_hdr">ZIP installieren</string>
- <string name="install_zip_btn">ZIP installieren</string>
- <string name="install_image_hdr">Image installieren</string>
- <string name="install_image_btn">Image installieren</string>
- <string name="install_select_file_hdr">Datei auswählen</string>
- <string name="file_selector_folders_hdr">Ordner</string>
- <string name="select_file_from_storage">Speicher: %tw_storage_display_name% (%tw_storage_free_size% MB)</string>
- <string name="adb_sideload_btn">ADB Sideload</string>
- <string name="install_hdr">Installieren</string>
- <string name="select_storage_hdr">Speicher auswählen</string>
- <string name="select_storage_btn">Speicher auswählen</string>
- <string name="queue_hdr">Einreihen</string>
- <string name="zip_queue_count">%tw_zip_queue_count% von 10 Dateien eingereiht</string>
- <string name="zip_queue_count_s">File %tw_zip_queue_count% von 10:</string>
- <string name="zip_warn1">Dieser Vorgang kann möglicherweise inkompatible</string>
- <string name="zip_warn2">Software installieren und das Gerät unbrauchbar machen.</string>
- <string name="zip_back_cancel">Zurück-Taste drücken, um den Vorgang abzubrechen.</string>
- <string name="zip_back_clear">Zurück-Taste drücken, um Warteschlange zu löschen.</string>
- <string name="folder">Ordner:</string>
- <string name="file">Datei:</string>
- <string name="zip_sig_chk">ZIP-Signaturprüfung</string>
- <string name="inject_twrp_chk">TWRP nach der Installation injizieren</string>
- <string name="options_hdr">Einstellungen</string>
- <string name="confirm_flash_hdr">Flashvorgang bestätigen</string>
- <string name="zip_queue">Warteschlange:</string>
- <string name="options">Einstellungen:</string>
- <string name="swipe_confirm">Bestätigen</string>
- <string name="zip_add_btn">Mehr ZIPs hinzufügen</string>
- <string name="zip_clear_btn">ZIP-Stapel löschen</string>
- <string name="install_zip_count_hdr">Installiere ZIP %tw_zip_index% von %tw_zip_queue_count%</string>
- <string name="installing_zip_xml">Installiere ZIP: %tw_file%</string>
- <string name="failed">Fehlgeschlagen</string>
- <string name="successful">Erfolgreich</string>
- <string name="install_failed">Installation fehlgeschlagen</string>
- <string name="install_successful">Installation erfolgreich</string>
- <string name="wipe_cache_dalvik_btn">Cache/Dalvik löschen</string>
- <string name="reboot_system_btn">System neustarten</string>
- <string name="install_sel_target">Ziel-Partition auswählen</string>
- <string name="flash_image_select">Partition wählen, um das Image zu einzuspielen:</string>
- <string name="target_partition">Ziel-Partition:</string>
- <string name="flashing_image">Spiele Image ein...</string>
- <string name="image_flashed">Image eingespielt</string>
- <string name="wipe_cache_dalvik_confirm">Cache & Dalvik löschen?</string>
- <string name="wiping_cache_dalvik">Lösche Cache & Dalvik...</string>
- <string name="wipe_cache_dalvik_complete">Cache & Dalvik gelöscht</string>
- <string name="swipe_wipe">Löschen bestätigen</string>
- <string name="swipe_wipe_s">Löschen</string>
- <string name="no_os1">Kein OS installiert! Sind Sie</string>
- <string name="no_osrb">sicher, dass Sie neu starten möchten?</string>
- <string name="no_ospo">sicher, dass Sie ausschalten möchten?</string>
- <string name="rebooting">Neustart...</string>
- <string name="swipe_reboot">Neustart bestätigen</string>
- <string name="swipe_reboot_s">Neustart</string>
- <string name="swipe_flash">Installation bestätigen</string>
- <string name="confirm_action">Aktion bestätigen</string>
- <string name="back_cancel">Zurück-Taste drücken, um den Vorgang abzubrechen.</string>
- <string name="cancel_btn">Abbruch</string>
- <string name="wipe_hdr">Löschen</string>
- <string name="factory_reset_hdr">Werkseinstellungen herstellen</string>
- <string name="factory_reset_btn">Werkseinstellungen</string>
- <string name="factory_reset1">Löscht Daten, Cache und Dalvik</string>
- <string name="factory_reset2">(ohne internen Speicher)</string>
- <string name="factory_reset3">Diese Lösch-Aktion ist in der Regel</string>
- <string name="factory_reset4">die einzige die benötigt wird.</string>
- <string name="factory_resetting">Werkseinstellungen herstellen...</string>
- <string name="advanced_wipe_hdr">Erweitertes Löschen</string>
- <string name="advanced_wipe_btn">Erweitertes Löschen</string>
- <string name="wipe_enc_confirm">Verschlüsselung der Daten-Partition enfernen?</string>
- <string name="formatting_data">Data wird formatiert...</string>
- <string name="swipe_format_data">Daten-Partition formatieren</string>
- <string name="swipe_format_data_s">Formatieren </string>
- <string name="factory_reset_complete">Werkseinstellungen wiederhergestellt</string>
- <string name="sel_part_hdr">Partitionen auswählen</string>
- <string name="wipe_sel_confirm">Ausgewählte Partition(en) löschen?</string>
- <string name="wiping_part">Lösche Partition(en)...</string>
- <string name="wipe_complete">Löschen abgeschlossen </string>
- <string name="sel_part_wipe">Partitionen zum Löschen auswählen:</string>
- <string name="invalid_part_sel">Ungültige Partitionsauswahl</string>
- <string name="format_data_hdr">Daten-Partition formatieren</string>
- <string name="format_data_btn">Daten formatieren</string>
- <string name="format_data_ptr1">Formatieren der Daten-Partition löscht alle Apps,</string>
- <string name="format_data_ptr2">Sicherungen, Bilder, Videos, sowie Medien und</string>
- <string name="format_data_ptr3">entfernt die Verschlüsselung des internen Speichers.</string>
- <string name="format_data_adopted">Betrifft auch die eingegliederte Storage-Partition.</string>
- <string name="format_data_lcp1">Formatieren der Daten-Partition löscht alle Apps, Backups, Bilder, Videos, sowie Medien und</string>
- <string name="format_data_lcp2">entfernt die Verschlüsselung des internen Speichers.</string>
- <string name="format_data_wtc1">Formatieren der Daten-Partition löscht alle Apps,</string>
- <string name="format_data_wtc2">Sicherungen und Medien. Dies kann nicht rückgängig gemacht werden.</string>
- <string name="format_data_undo">Dies kann nicht rückgängig gemacht werden.</string>
- <string name="format_data_complete">Formatieren der Daten-Partition abgeschlossen</string>
- <!-- Translator note: the word "yes" must remain English! -->
- <string name="yes_continue">"yes" eingeben um fortzufahren. Zurück drücken um abzubrechen.</string>
- <string name="part_opt_hdr">Optionen für %tw_partition_name%-Partition</string>
- <string name="sel_act_hdr">Aktion auswählen</string>
- <string name="file_sys_opt">Dateisystem-Optionen</string>
- <string name="partition">Partition: %tw_partition_name%</string>
- <string name="part_mount_point">Einhängepunkt: %tw_partition_mount_point%</string>
- <string name="part_curr_fs">Dateisystem: %tw_partition_file_system%</string>
- <string name="part_present_yes">Vorhanden: Ja</string>
- <string name="part_present_no">Vorhanden: Nein</string>
- <string name="part_removable_yes">Wechselbar: Ja</string>
- <string name="part_removable_no">Wechselbar: Nein</string>
- <string name="part_size">Größe: %tw_partition_size%MB</string>
- <string name="part_used">Verwendet: %tw_partition_used%MB</string>
- <string name="part_free">Frei: %tw_partition_free%MB</string>
- <string name="part_backup_size">Backup-Größe: %tw_partition_backup_size%MB</string>
- <string name="resize_btn">Größe ändern</string>
- <string name="resize_btn_s">Größe ändern</string>
- <string name="resize_confirm">Größe von %tw_partition_name% ändern?</string>
- <string name="resizing">Ändere Größe...</string>
- <string name="resize_complete">Größenänderung abgeschlossen</string>
- <string name="swipe_resize">Größenänderung bestätigen</string>
- <string name="swipe_resize_s">Größe ändern</string>
- <string name="repair_btn">Dateisystem reparieren</string>
- <string name="repair_btn_s">Reparatur</string>
- <string name="repair_confirm">Repariere %tw_partition_name%?</string>
- <string name="repairing">Reparieren...</string>
- <string name="repair_complete">Reparatur abgeschlossen</string>
- <string name="swipe_repair">Partition reparieren</string>
- <string name="swipe_repair_s">Reparieren</string>
- <string name="change_fs_btn">Dateisystem ändern</string>
- <string name="change_fs_btn_s">Ändern</string>
- <string name="change_fs_for_hdr">Dateisystem für %tw_partition_name% ändern</string>
- <string name="change_fs_for_hdr_s">Partition: %tw_partition_name% > Dateisystem auswählen</string>
- <string name="change_fs_warn1">Einige ROMs oder Kernel unterstützen möglicherweise manche</string>
- <string name="change_fs_warn2">Dateisysteme nicht. Überlegt handeln!</string>
- <string name="change_fs_confirm">%tw_partition_name% ändern?</string>
- <string name="formatting">Formatierung...</string>
- <string name="format_complete">Formatieren abgeschlossen</string>
- <string name="swipe_change_fs">Dateisystem ändern</string>
- <string name="swipe_change_s">Ändern</string>
- <string name="back_btn">Zurück</string>
- <string name="wipe_enc_btn">Verschlüsselung entfernen</string>
- <string name="swipe_factory_reset">Werkseinstellungen herstellen</string>
- <string name="repair_change_btn">Dateisystem reparieren oder ändern</string>
- <string name="storage_hdr">Speicher: %tw_storage_display_name% (%tw_storage_free_size% MB)</string>
- <string name="backup_hdr">Sicherung</string>
- <string name="backup_confirm_hdr">Sicherung bestätigen</string>
- <string name="encryption_tab">VERSCHLÜSSELUNG</string>
- <string name="encryption">Verschlüsselung:</string>
- <string name="name">Name:</string>
- <string name="sel_part_backup">Zu sichernde Partitionen auswählen:</string>
- <string name="storage">Speicherort:</string>
- <string name="enc_disabled">deaktiviert - Passwort festlegen um zu aktivieren</string>
- <string name="enc_enabled">aktiviert</string>
- <string name="enable_backup_comp_chk">Komprimierung aktivieren</string>
- <string name="skip_md5_backup_chk">MD5-Erstellung für die Sicherung überspringen</string>
- <string name="disable_backup_space_chk">Prüfung auf freien Speicher deaktivieren</string>
- <string name="refresh_sizes_btn">Größen aktualisieren</string>
- <string name="swipe_backup">Sicherung erstellen</string>
- <string name="append_date_btn">Datum anhängen</string>
- <string name="backup_name_exists">Eine Sicherung mit diesem Namen existiert bereits!</string>
- <string name="encrypt_backup">Sicherung verschlüsseln?</string>
- <string name="enter_pass">Passwort eingeben:</string>
- <string name="enter_pass2">Passwort erneut eingeben:</string>
- <string name="pass_not_match">Die Passwörter stimmen nicht überein!</string>
- <string name="partitions">Partitionen:</string>
- <string name="disabled">Deaktiviert</string>
- <string name="enabled">Aktiviert</string>
- <string name="backup_name_hdr">Sicherung benennen</string>
- <string name="progress">Fortschritt:</string>
- <string name="backup_complete">Sicherung abgeschlossen</string>
- <string name="restore_hdr">Wiederherstellen</string>
- <string name="sel_backup_hdr">Sicherung auswählen</string>
- <string name="restore_sel_store_hdr">Speicher: %tw_storage_display_name% (%tw_storage_free_size% MB)</string>
- <string name="restore_sel_pack_fs">Wiederherzustellende Sicherung:</string>
- <string name="restore_enc_backup_hdr">Verschlüsselte Sicherung</string>
- <string name="restore_dec_fail">Passwort falsch, bitte erneut versuchen!</string>
- <string name="del_backup_btn">Sicherung löschen</string>
- <string name="del_backup_confirm">Sicherung löschen?</string>
- <string name="del_backup_confirm2">Dies kann nicht rückgängig gemacht werden!</string>
- <string name="deleting_backup">Sicherung wird gelöscht...</string>
- <string name="backup_deleted">Sicherung gelöscht</string>
- <string name="swipe_delete">Löschen bestätigen</string>
- <string name="swipe_delete_s">Löschen</string>
- <string name="restore_try_decrypt">Verschlüsselte Sicherung - Versuche Entschlüsselung</string>
- <string name="restore_try_decrypt_s">Versuche Entschlüsselung</string>
- <string name="restore_backup_date">Sicherung am %tw_restore_file_date% erstellt</string>
- <string name="restore_sel_part">Wiederherzustellende Partitionen:</string>
- <string name="restore_enable_md5_chk">MD5-Prüfung von Sicherungs-Dateien aktivieren</string>
- <string name="restore_complete">Wiederherstellen abgeschlossen</string>
- <string name="swipe_restore">Sicherung wiederherstellen</string>
- <string name="swipe_restore_s">Wiederherstellen</string>
- <string name="rename_backup_hdr">Sicherung umbenennen</string>
- <string name="rename_backup_confirm">Sicherung umbenennen?</string>
- <string name="rename_backup_confirm2">Dies kann nicht rückgängig gemacht werden!</string>
- <string name="renaming_backup">Sicherung wird umbenannt...</string>
- <string name="rename_backup_complete">Sicherung umbenannt</string>
- <string name="swipe_to_rename">Name ändern</string>
- <string name="swipe_rename">Umbenennen</string>
- <string name="confirm_hdr">Bestätigen</string>
- <string name="mount_hdr">Einhängen</string>
- <string name="mount_sel_part">Einzuhängende Partitionen:</string>
- <string name="mount_sys_ro_chk">System-Partition schreibgeschützt einhängen</string>
- <string name="mount_sys_ro_s_chk">System nur lesend</string>
- <string name="decrypt_data_btn">Daten-Partition entschlüsseln</string>
- <string name="disable_mtp_btn">MTP deaktivieren</string>
- <string name="enable_mtp_btn">MTP aktivieren</string>
- <string name="mount_usb_storage_btn">USB-Speicher einhängen</string>
- <string name="usb_storage_hdr">USB-Speicher</string>
- <string name="usb_stor_mnt1">USB-Speicher eingehängt</string>
- <string name="usb_stor_mnt2">Funktion "USB-Speicher auswerfen" auf Computer</string>
- <string name="usb_stor_mnt3">ausführen, bevor die Verbindung getrennt wird!</string>
- <string name="unmount_btn">Auswerfen</string>
- <string name="rb_system_btn">System</string>
- <string name="rb_poweroff_btn">Ausschalten</string>
- <string name="rb_recovery_btn">Recovery</string>
- <string name="rb_bootloader_btn">Bootloader</string>
- <string name="rb_download_btn">Download</string>
- <string name="turning_off">Ausschalten...</string>
- <string name="swipe_power_off">Gerät ausschalten</string>
- <string name="swipe_power_off_s">Ausschalten</string>
- <string name="sys_ro_hdr">Unveränderte Systempartition</string>
- <string name="sys_ro_keep">System-Partition schreibgeschützt einhängen?</string>
- <string name="sys_rop1">TWRP kann die System Partition unverändert lassen,</string>
- <string name="sys_rop2">um das Aufspielen offizieller Updates zu erleichtern.</string>
- <string name="sys_rop3">Jedoch kann dann das Hersteller-ROM TWRP wieder</string>
- <string name="sys_rop4">überschreiben und TWRP kann keinen Root-Zugriff einrichten.</string>
- <string name="sys_rop5">Das Installieren von ZIPs oder ADB-Vorgänge könnten jedoch</string>
- <string name="sys_rop6">trotzdem die System-Partition verändern.</string>
- <string name="sys_rol1">TWRP kann die System-Partition unverändert lassen, um das Aufspielen offizieller Updates zu erleichtern.</string>
- <string name="sys_rol2">Jedoch kann dann das Hersteller-ROM TWRP wieder überschreiben und TWRP kann keinen Root-Zugriff einrichten.</string>
- <string name="sys_rol3">Das Installieren von ZIPs oder ADB-Vorgänge könnten jedoch trotzdem die System-Partition verändern.</string>
- <string name="sys_ro_never_show_chk">Diesen Bildschirm nicht mehr anzeigen</string>
- <string name="sys_ro_keep_ro_btn">Nur-Lesend belassen</string>
- <string name="swipe_allow_mod">Veränderungen zulassen</string>
- <string name="swipe_allow_mod_s">Zulassen</string>
- <string name="settings_hdr">Einstellungen</string>
- <string name="settings_gen_hdr">Allgemein</string>
- <string name="settings_gen_s_hdr">Allgemein</string>
- <string name="settings_gen_btn">Allgemein</string>
- <string name="use_rmrf_chk">rm -rf anstatt formatieren verwenden</string>
- <string name="use24clock_chk">24-Stunden-Format</string>
- <string name="rev_navbar_chk">Navigationsleiste umkehren</string>
- <string name="simact_chk">Benutzeraktionen für Design-Test simulieren</string>
- <string name="simfail_chk">Fehlschlag von Benutzeraktionen simulieren</string>
- <string name="ctr_navbar_rdo">Navigationsleisten-Schaltflächen zentrieren</string>
- <string name="lft_navbar_rdo">Navigationsleisten-Schaltflächen linksbündig</string>
- <string name="rht_navbar_rdo">Navigationsleisten-Schaltflächen rechtsbündig</string>
- <string name="restore_defaults_btn">Standard herstellen</string>
- <string name="settings_tz_btn">Zeitzone</string>
- <string name="settings_screen_btn">Bildschirm</string>
- <string name="settings_screen_bright_btn">Helligkeit</string>
- <string name="settings_vibration_btn">Vibration</string>
- <string name="settings_language_btn">Sprache</string>
- <string name="time_zone_hdr">Zeitzone</string>
- <string name="sel_tz_list">Zeitzone auswählen:</string>
- <!-- Translator note: if it does not make sense to translate the locations or if it makes more sense,
- feel free to change the location listed or drop the location entirely and just call it UTC -6 -->
- <string name="utcm11">(UTC -11) Samoa, Midwayinseln</string>
- <string name="utcm10">(UTC -10) Hawaii</string>
- <string name="utcm9">(UTC -9) Alaska</string>
- <string name="utcm8">(UTC -8) Pacific Time</string>
- <string name="utcm7">(UTC -7) Mountain Time (USA)</string>
- <string name="utcm6">(UTC -6) Central Time (USA)</string>
- <string name="utcm5">(UTC -5) Eastern Time (USA)</string>
- <string name="utcm4">(UTC -4) Atlantic Time (USA)</string>
- <string name="utcm3">(UTC -3) Brasilien, Buenos Aires</string>
- <string name="utcm2">(UTC -2) Mittelatlantik</string>
- <string name="utcm1">(UTC -1) Azores, Cape Verde</string>
- <string name="utc0">(UTC 0) London, Dublin, Lissabon</string>
- <string name="utcp1">(UTC +1) Berlin, Brüssel, Paris</string>
- <string name="utcp2">(UTC +2) Athen, Istanbul, Südafrika</string>
- <string name="utcp3">(UTC +3) Moskau, Bagdad</string>
- <string name="utcp4">(UTC +4) Abu Dhabi, Tiflis, Muscat</string>
- <string name="utcp5">(UTC +5) Ekaterinburg, Islamabad</string>
- <string name="utcp6">(UTC +6) Almaty, Dhaka, Colombo</string>
- <string name="utcp7">(UTC +7) Bangkok, Hanoi, Jakarta</string>
- <string name="utcp8">(UTC +8) Peking, Singapur, Hong Kong</string>
- <string name="utcp9">(UTC +9) Tokio, Seoul, Yakutsk</string>
- <string name="utcp10">(UTC +10) Ostaustralien, Guam</string>
- <string name="utcp11">(UTC +11) Vladivostok, Salomonen</string>
- <string name="utcp12">(UTC +12) Auckland, Wellington, Fidschi</string>
- <string name="sel_tz_offset">Zeitverschiebung: %tw_time_zone_guioffset%</string>
- <string name="tz_offset_none">Keine</string>
- <string name="tz_offset_0">0</string>
- <string name="tz_offset_15">15</string>
- <string name="tz_offset_30">30</string>
- <string name="tz_offset_45">45</string>
- <string name="use_dst_chk">Sommerzeit verwenden</string>
- <string name="curr_tz">Aktuelle Zeitzone: %tw_time_zone%</string>
- <string name="curr_tz_s">Aktuelle Zeitzone:</string>
- <string name="set_tz_btn">Zeitzone einstellen</string>
- <string name="settings_screen_hdr">Bildschirm</string>
- <string name="settings_screen_timeout_hdr">Bildschirm-Timeout</string>
- <string name="enable_timeout_chk">Bildschirm automatisch ausschalten</string>
- <string name="screen_to_slider">Zeit bis zum Ausschalten (in Sekunden):</string>
- <string name="screen_to_slider_s">Ausschalten nach %tw_screen_timeout_secs% Sekunden (0 = deaktiviert): </string>
- <string name="screen_to_na">Bildschirm-Abschaltung nicht verfügbar</string>
- <string name="screen_bright_slider">Helligkeit: %tw_brightness_pct% %</string>
- <string name="screen_bright_na">Helligkeit nicht verfügbar</string>
- <string name="vibration_hdr">Vibration</string>
- <string name="button_vibration_hdr">Tasten-Vibration</string>
- <string name="kb_vibration_hdr">Tastatur-Vibration</string>
- <string name="act_vibration_hdr">Aktion-Vibration</string>
- <string name="button_vibration">Tasten-Vibration:</string>
- <string name="kb_vibration">Tastatur-Vibration:</string>
- <string name="act_vibration">Aktion-Vibration:</string>
- <string name="select_language">Sprache auswählen:</string>
- <string name="sel_lang_btn">Sprache auswählen</string>
- <string name="set_language_btn">Sprache einstellen</string>
- <string name="advanced_hdr">Erweitert</string>
- <string name="copy_log_confirm">Log auf SD-Karte kopieren?</string>
- <string name="copying_log">Kopiere Log auf SD-Karte...</string>
- <string name="copy_log_complete">Log erfolgreich kopiert</string>
- <string name="fix_context_btn">SELinux Kontexte</string>
- <string name="part_sd_btn">SD-Karte partitionieren</string>
- <string name="part_sd_s_btn">SD-Karte</string>
- <string name="file_manager_btn">Dateimanager</string>
- <string name="language_hdr">Sprache</string>
- <string name="terminal_btn">Terminal</string>
- <string name="reload_theme_btn">Theme neu laden</string>
- <string name="dumlock_btn">HTC Dumlock</string>
- <string name="inject_twrp_btn">TWRP injizieren</string>
- <string name="inject_twrp_confirm">TWRP wieder injizieren?</string>
- <string name="injecting_twrp">TWRP wird wieder injiziert...</string>
- <string name="inject_twrp_complete">TWRP-Injektion abgeschlossen</string>
- <string name="swipe_to_confirm">Aktion bestätigen</string>
- <string name="part_sd_hdr">SD-Karte partitionieren</string>
- <string name="invalid_partsd_sel">Es muss ein Wechseldatenträger ausgewählt sein!</string>
- <string name="part_sd_lose">Es werden alle Dateien von der SD-Karte gelöscht!</string>
- <string name="part_sd_undo">Diese Aktion kann nicht rückgängig gemacht werden!</string>
- <string name="part_sd_ext_sz">EXT Grösse:</string>
- <string name="part_sd_swap_sz">Grösse der Auslagerungsdatei:</string>
- <string name="part_sd_m">-</string>
- <string name="part_sd_p">+</string>
- <string name="file_system">Dateisystem:</string>
- <string name="swipe_part_sd">SD-Karte partitionieren</string>
- <string name="swipe_part_sd_s">Partitionieren</string>
- <string name="partitioning_sd">Partitioniere SD-Karte...</string>
- <string name="partitioning_sd2">Dies wird einige Minuten dauern.</string>
- <string name="part_sd_complete">Partitionierung abgeschlossen</string>
- <string name="dumlock_hdr">HTC Dumlock</string>
- <string name="dumlock_restore_btn">Orginale Boot-Partition wiederherstellen</string>
- <string name="dumlock_restore_confirm">Orginales Image der Boot-Partition wiederherstellen?</string>
- <string name="dumlock_restoring">Stelle originale Boot-Partition wieder her...</string>
- <string name="dumlock_restore_complete">Wiederherstellen der orginalen Boot-Partition abgeschlossen</string>
- <string name="dumlock_reflash_btn">Recovery erneut einspielen</string>
- <string name="dumlock_reflash_confirm">Recovery in Boot-Partition einspielen?</string>
- <string name="dumlock_reflashing">Spiele Recovery in Boot-Partition ein...</string>
- <string name="dumlock_reflash_complete">Einspielen der Recovery in die Boot-Partition abgeschlossen</string>
- <string name="dumlock_install_btn">HTC Dumlock installieren</string>
- <string name="dumlock_install_confirm">HTC Dumlock Dateien ins ROM installieren?</string>
- <string name="dumlock_installing">Installiere HTC Dumlock...</string>
- <string name="dumlock_install_complete">HTC Dumlock Installation abgeschlossen</string>
- <string name="swipe_to_unlock">Bildschirm entsperren</string>
- <string name="swipe_unlock">Entsperren</string>
- <string name="fm_hdr">Dateimanager</string>
- <string name="fm_sel_file">Datei oder Ordner auswählen</string>
- <string name="fm_type_folder">Ordner</string>
- <string name="fm_type_file">Datei</string>
- <string name="fm_choose_act">Aktion auswählen</string>
- <string name="fm_selected">%tw_fm_type%:</string>
- <string name="fm_copy_btn">Kopieren</string>
- <string name="fm_copy_file_btn">Datei kopieren</string>
- <string name="fm_copy_folder_btn">Ordner kopieren</string>
- <string name="fm_copying">Kopiere</string>
- <string name="fm_move_btn">Verschieben</string>
- <string name="fm_moving">Verschiebe</string>
- <string name="fm_chmod755_btn">chmod 755</string>
- <string name="fm_chmod755ing">chmod 755</string>
- <string name="fm_chmod_btn">chmod</string>
- <string name="fm_delete_btn">Löschen</string>
- <string name="fm_deleting">Wird gelöscht</string>
- <string name="fm_rename_btn">Umbenennen</string>
- <string name="fm_rename_file_btn">Datei umbennen</string>
- <string name="fm_rename_folder_btn">Ordner umbenennen</string>
- <string name="fm_renaming">Umbenennen</string>
- <string name="fm_sel_dest">Zielordner auswählen</string>
- <string name="fm_sel_curr_folder">Aktuellen Ordner auswählen</string>
- <string name="fm_rename_hdr">Umbenennen</string>
- <string name="fm_set_perms_hdr">Berechtigungen setzen</string>
- <string name="fm_perms">Berechtigungen:</string>
- <string name="fm_complete">Dateivorgang beendet</string>
- <string name="decrypt_data_hdr">Daten-Partition entschlüsseln</string>
- <string name="decrypt_data_enter_pass">Passwort eingeben:</string>
- <string name="decrypt_data_failed">Passwort falsch, bitte erneut versuchen!</string>
- <string name="decrypt_data_failed_pattern">Muster falsch, bitte erneut versuchen!</string>
- <string name="decrypt_data_enter_pattern">Muster eingeben.</string>
- <string name="decrypt_data_trying">Versuche Entschlüsselung</string>
- <string name="term_hdr">Terminal</string>
- <string name="term_s_hdr">Terminal</string>
- <string name="term_kill_btn">BEENDEN</string>
- <string name="term_sel_folder_hdr">Zum Stamm-Ordner navigieren</string>
- <string name="adb_sideload_hdr">ADB Sideload</string>
- <string name="sideload_wipe_dalvik_chk">Dalvik Cache löschen</string>
- <string name="sideload_wipe_cache_chk">Cache löschen</string>
- <string name="swipe_to_sideload">ADB Sideload starten</string>
- <string name="swipe_sideload">Start</string>
- <string name="sideload_confirm">ADB Sideload</string>
- <string name="sideload_usage">Syntax: adb sideload Dateiname.zip</string>
- <string name="sideload_complete">ADB Sideload abgeschlossen</string>
- <string name="fix_contexts_hdr">SELinux Kontexte</string>
- <string name="fix_contexts_note1">Das Korrigieren der Kontexte wird selten benötigt!</string>
- <string name="fix_contexts_note2">Das Korrigieren der SELinux-Kontexte kann</string>
- <string name="fix_contexts_note3">Startprobleme verursachen.</string>
- <string name="swipe_to_fix_contexts">SELinux-Kontexte korrigieren</string>
- <string name="swipe_fix_contexts">Korrigieren</string>
- <string name="fixing_contexts">Korrigiere Kontexte...</string>
- <string name="fix_contexts_complete">Korrektur der Kontexte abgeschlossen</string>
- <string name="reboot_hdr">Neustart</string>
- <string name="su_hdr">SuperSU-Check</string>
- <string name="su_note1">Anscheinend besteht kein Root-Zugriff auf das Gerät.</string>
- <string name="su_note2">SuperSU jetzt installieren?</string>
- <string name="su_note3">Dies wird Root-Zugriff auf dem Gerät einrichten.</string>
- <string name="su_cancel">Nicht installieren</string>
- <string name="swipe_su_to_install">Installation starten</string>
- <string name="swipe_su_install">Installieren</string>
- <string name="su_installing">Installiere SuperSU</string>
- <string name="sel_storage_list">Speicher auswählen</string>
- <string name="ok_btn">OK</string>
-
- <!-- Various console messages - these consist of user displayed messages, oftentimes errors -->
- <string name="no_kernel_selinux">Dieser Kernel hat keine Leseunterstützung für SELinux-Kontexte.</string>
- <string name="full_selinux">Volle SELinux-Unterstützung ist vorhanden.</string>
- <string name="no_selinux">Keine SELinux-Unterstützung vorhanden (kein libselinux).</string>
- <string name="mtp_enabled">MTP aktiviert</string>
- <string name="mtp_crash">MTP abgestürzt, MTP wird künftig während Bootvorgang nicht gestartet.</string>
- <string name="decrypt_success">Erfolgreich mit dem Standardpasswort entschlüsselt.</string>
- <string name="unable_to_decrypt">Entschlüsselung mit Standardpasswort nicht möglich. Eventuell muss die Daten-Partition formatiert werden.</string>
- <string name="generating_md51">Generiere MD5</string>
- <!-- Message displayed during a backup if we are generating an MD5, ideally, leave the leading " * " to help align and separate this text from other console text -->
- <string name="generating_md52"> * generiere MD5...</string>
- <string name="md5_created"> * MD5 erstellt.</string>
- <string name="md5_error"> * MD5-Fehler!</string>
- <string name="md5_compute_error"> * Fehler bei MD5-Berechnung.</string>
- <string name="current_date">(Aktuelles Datum)</string>
- <string name="auto_generate">(Automatisch generiert)</string>
- <string name="unable_to_locate_partition">Kann '{1}' Partition für Sicherungsberechnungen nicht finden.</string>
- <string name="no_partition_selected">Keine Partitionen für die Sicherung ausgewählt.</string>
- <string name="total_partitions_backup"> * Anzahl der zu sichernden Partitionen: {1}</string>
- <string name="total_backup_size"> * Gesamtgröße aller Daten: {1}MB</string>
- <string name="available_space"> * Verfügbare Speicherplatz: {1}MB</string>
- <string name="unable_locate_storage">Kann Sicherungsgerät nicht finden.</string>
- <string name="no_space">Zu wenig freier Speicherplatz.</string>
- <string name="backup_started">[SICHERUNG GESTARTET]</string>
- <string name="backup_folder"> * Ordner für Sicherung: {1}</string>
- <string name="fail_backup_folder">Ordner für Sicherung konnte nicht erstellt werden.</string>
- <string name="avg_backup_fs">Durchschnittliche Sicherungsgeschwindigkeit für Dateisysteme: {1} MB/sek</string>
- <string name="avg_backup_img">Durchschnittliche Sicherungsgeschwindigkeit für abgebildete Laufwerke: {1} MB/sek</string>
- <string name="total_backed_size">[INSGESAMT {1} MB GESICHERT]</string>
- <string name="backup_completed">[SICHERUNG ABGESCHLOSSEN IN {1} SEKUNDEN]</string>
- <string name="restore_started">[WIEDERHERSTELLEN GESTARTET]</string>
- <string name="restore_folder">Wiederherstellen aus Ordner: '{1}'</string>
- <!-- {1} is the partition display name and {2} is the number of seconds -->
- <string name="restore_part_done">[{1} fertiggestellt ({2} Sekunden)]</string>
- <string name="verifying_md5">Überprüfe MD5</string>
- <string name="skip_md5">Überspringe MD5-Prüfung aufgrund Benutzereinstellungen.</string>
- <string name="calc_restore">Berechne Wiederherstellungsinformationen...</string>
- <string name="restore_read_only">Kann {1} nicht wiederherstellen -- Partition schreibgeschützt.</string>
- <string name="restore_unable_locate">Kann wiederherzustellende '{1}'-Partition nicht finden.</string>
- <string name="no_part_restore">Keine Partitionen zur Wiederherstellung ausgewählt.</string>
- <string name="restore_part_count">{1} Partitionen wiederherstellen...</string>
- <string name="total_restore_size">Insgesamt wiederherzustellen: {1}MB</string>
- <string name="updating_system_details">Aktualisieren der System-Details</string>
- <string name="restore_completed">[WIEDERHERSTELLEN ABGESCHLOSSEN IN {1} SEKUNDEN]</string>
- <!-- {1} is the path we could not open, {2} is strerror output -->
- <string name="error_opening_strerr">Fehler beim Öffnen: '{1}' ({2})</string>
- <string name="unable_locate_part_backup_name">Partition nicht anhand des Sicherungsnamens ermittelbar: '{1}'</string>
- <string name="unable_find_part_path">Partition für Pfad '{1}' nicht gefunden</string>
- <string name="update_part_details">Partitions-Informationen werden aktualisiert...</string>
- <string name="update_part_details_done">...Fertig</string>
- <string name="wiping_dalvik">Dalvik Cache-Verzeichnisse bereinigen...</string>
- <string name="cleaned">Bereinigt: {1}...</string>
- <string name="dalvik_done">-- Dalvik Cache-Verzeichnisse bereinigt!</string>
- <string name="no_andsec">Keine "Android Secure"-Partitionen gefunden.</string>
- <string name="unable_to_locate">{1} nicht gefunden.</string>
- <string name="wiping_datamedia">Lösche internen Speicher -- /data/media...</string>
- <string name="unable_to_mount">Kann {1} nicht einhängen</string>
- <string name="unable_to_mount_internal">Kann internen Speicher nicht einhängen</string>
- <string name="unable_to_mount_storage">Kann Speicher nicht einhängen</string>
- <string name="fail_decrypt">Entschlüsselung der Daten-Partition fehlgeschlagen.</string>
- <string name="no_crypto_support">In diese Version wurde keine Krypto-Unterstützung eingebunden.</string>
- <string name="decrypt_success_dev">Erfolgreich mit Standardpasswort entschlüsselt.</string>
- <string name="done">Abgeschlossen.</string>
- <string name="start_partition_sd">Partitioniere SD Karte...</string>
- <string name="partition_sd_locate">Zu partitionierendes Gerät nicht gefunden.</string>
- <string name="ext_swap_size">EXT + Auslagerung ist grösser als die SD-Karte.</string>
- <string name="remove_part_table">Entferne Partitionstabelle...</string>
- <string name="unable_rm_part">Partitionstabelle kann nicht entfernt werden.</string>
- <string name="create_part">Erstelle {1}-Partition...</string>
- <string name="unable_to_create_part">{1}-Partition kann nicht erstellt werden.</string>
- <string name="format_sdext_as">Formatiere sd-ext als {1}...</string>
- <string name="part_complete">Partitionierung abgeschlossen.</string>
- <string name="unable_to_open">'{1}' kann nicht geöffnet werden.</string>
- <string name="mtp_already_enabled">MTP bereits aktiviert</string>
- <string name="mtp_fail">Fehler beim Aktivieren von MTP</string>
- <string name="no_mtp">MTP-Unterstützung nicht integriert</string>
- <string name="image_flash_start">[IMAGE-DATEI WIRD AUFGESPIELT]</string>
- <string name="img_to_flash">Aufzuspielendes Image: '{1}'</string>
- <string name="flash_unable_locate">'{1}'-Partition wurde nicht gefunden.</string>
- <string name="no_part_flash">Keine Partitionen ausgewählt.</string>
- <string name="too_many_flash">Zu viele Partitionen ausgewählt.</string>
- <string name="invalid_flash">Ungültige Partitions-Auswahl.</string>
- <string name="flash_done">[IMAGE EINSPIELEN ABGESCHLOSSEN]</string>
- <string name="wiping">Lösche {1}</string>
- <string name="repair_not_exist">{1} existiert nicht! Reparieren nicht möglich!</string>
- <string name="repairing_using">Reparatur von {1} mit {2}...</string>
- <string name="unable_repair">{1} kann nicht repariert werden.</string>
- <string name="mount_data_footer">/data konnte nicht eingehängt werden und Krypto-Signatur wurde nicht gefunden.</string>
- <!-- {1} is the folder name that we could not create, {2} is strerror output -->
- <string name="create_folder_strerr">Ordner '{1}' konnte nicht angelegt werden ({2}).</string>
- <!-- {1} is the folder name that we could not mount, {2} is strerror output -->
- <string name="fail_mount">'{1}' konnte nicht eingehängt werden ({2})</string>
- <!-- {1} is the folder name that we could not unmount, {2} is strerror output -->
- <string name="fail_unmount">'{1}' konnte nicht eingehängt werden ({2})</string>
- <string name="cannot_resize">Größe von {1} kann nicht geändert werden.</string>
- <string name="repair_resize">{1} wird repariert. Danach wird die Größe geändert.</string>
- <string name="unable_resize">Größe von {1} kann nicht geändert werden.</string>
- <string name="no_md5_found">Keine MD5-Datei für '{1}' gefunden. Bitte MD5-Prüfung für Wiederherstellung deaktivieren.</string>
- <string name="md5_fail_match">MD5-Prüfung für '{1}' fehlgeschlagen.</string>
- <string name="fail_decrypt_tar">TAR-Datei '{1}' konnte nicht entschlüsselt werden.</string>
- <string name="format_data_msg">Ein Neustart von TWRP kann notwendig sein, damit /data wieder verwendet werden kann.</string>
- <string name="format_data_err">Formatierung zum Entfernen der Verschlüsselung kann nicht durchgeführt werden.</string>
- <string name="formatting_using">Formatiere {1} mit {2}...</string>
- <string name="unable_to_wipe">{1} kann nicht gelöscht werden.</string>
- <string name="cannot_wipe">Partition {1} kann nicht gelöscht werden.</string>
- <string name="remove_all">Entferne alle Dateien unter '{1}'</string>
- <string name="wiping_data">Lösche Daten, aber verschone internen Speicher...</string>
- <string name="backing_up">Sichere {1}...</string>
- <string name="backing">Sichere</string>
- <string name="backup_size">Sicherung von '{1}' hat 0 Byte.</string>
- <string name="datamedia_fs_restore">WARNUNG: Diese Sicherung wurde mit dem Dateisystem {1} erstellt! Es kann sein, dass wieder zu {1} gewechselt werden muss, damit das Gerät nach der Wiederherstellung auch startet.</string>
- <string name="restoring">Wiederherstellung läuft</string>
- <string name="restoring_hdr">Wiederherstellung läuft</string>
- <string name="recreate_folder_err">Ordner {1} kann nicht wiederhergestellt werden.</string>
- <string name="img_size_err">Image ist zu groß für das Gerät</string>
- <string name="flashing">Einspielen von {1}...</string>
- <string name="backup_folder_set"> * Ordner für Sicherung: {1}</string>
- <string name="locate_backup_err">Sicherung '{1}' nicht gefunden</string>
- <string name="set_restore_opt">Setze Wiederherstellungs-Optionen: '{1}':</string>
- <string name="md5_check_skip">MD5-Prüfung ist deaktiviert</string>
- <string name="ors_encrypt_restore_err">Eine verschlüsselte Sicherung kann nicht per OpenRecoveryScript wiederhergestellt werden.</string>
- <string name="mounting">Einhängen</string>
- <string name="unmounting">Auswerfen</string>
- <string name="mounted">'{1}' eingehängt</string>
- <string name="unmounted">'{1}' ausgeworfen</string>
- <string name="setting">Setze '{1}' auf '{2}'</string>
- <string name="setting_empty">Setze '{1}' auf leer</string>
- <string name="making_dir1">Erstelle Verzeichnis</string>
- <string name="making_dir2">Erstelle Verzeichnis: '{1}'</string>
- <string name="running_command">Führe Befehl aus</string>
- <string name="sideload">ADB Sideload</string>
- <string name="start_sideload">Starte ADB Sideload...</string>
- <string name="need_new_adb">Für dieses Gerät wird ADB 1.0.32 oder neuer benötigt.</string>
- <string name="no_pwd">Kein Passwort angegeben.</string>
- <string name="done_ors">Skript wurde verarbeitet</string>
- <string name="injecttwrp">Injiziere TWRP in das Boot-Image...</string>
- <string name="zip_err">Fehler beim Installieren von ZIP '{1}'</string>
- <string name="installing_zip">Installiere Zip: %tw_file%</string>
- <string name="select_backup_opt">Setze Sicherungs-Optionen:</string>
- <string name="compression_on">Komprimierung ist aktiviert</string>
- <string name="md5_off">MD5-Generierung ist deaktiviert</string>
- <string name="backup_fail">Sicherung fehlgeschlagen</string>
- <string name="backup_clean">Sicherung fehlgeschlagen, bereinige Sicherungs-Verzeichnis</string>
- <string name="running_recovery_commands">Führe Recovery-Befehle aus</string>
- <string name="recovery_commands_complete">Recovery-Befehle ausgeführt</string>
- <string name="running_ors">Führe OpenRecoveryScript aus</string>
- <string name="ors_complete">OpenRecoveryScript ausgeführt</string>
- <string name="no_updater_binary">'{1}' konnte in der ZIP nicht gefunden werden.</string>
- <string name="check_for_md5">Suche nach MD5-Datei...</string>
- <string name="fail_sysmap">'{1}' kann nicht zugeordnet werden</string>
- <string name="verify_zip_sig">Überprüfe ZIP-Signatur...</string>
- <string name="verify_zip_fail">Prüfung der ZIP-Signatur fehlgeschlagen!</string>
- <string name="verify_zip_done">Prüfung der ZIP-Signatur erfolgreich.</string>
- <string name="zip_corrupt">ZIP-Datei ist beschädigt!</string>
- <string name="no_md5">MD5-Prüfung übersprungen: keine MD5-Datei gefunden</string>
- <string name="md5_fail">MD5 stimmt nicht überein</string>
- <string name="md5_match">MD5 stimmt überein</string>
- <string name="pid_signal">Prozess {1} endete mit Meldung: {2}</string>
- <string name="pid_error">Prozess {1} endete mit FEHLER: {2}</string>
- <string name="install_dumlock">Installiere HTC Dumlock in System-Partition...</string>
- <string name="dumlock_restore">Stelle originale Boot-Partition wieder her...</string>
- <string name="dumlock_reflash">Recovery auf Boot-Partition einspielen...</string>
- <string name="run_script">Führe {1}-Skript aus...</string>
- <string name="rename_stock">Hersteller-Recovery in "/system" wurde umbenannt, damit das Hersteller-ROM TWRP nicht überschreibt.</string>
- <string name="split_backup">Sicherungs-Datei wird in mehrere Archive aufgeteilt...</string>
- <string name="backup_error">Fehler beim Erstellen der Sicherung.</string>
- <string name="restore_error">Fehler während der Wiederherstellung.</string>
- <string name="split_thread">Teile Thread-ID {1} in Archiv {2}</string>
- <!-- These 2 items are saved in the data manager instead of resource manager, so %llu, etc is correct instead of {1} -->
- <string name="file_progress">%llu von %llu Dateien, %i%%</string>
- <string name="size_progress">%lluMB von %lluMB, %i%%</string>
- <string name="decrypt_cmd">Versuche die Daten-Partition per Kommandozeile zu entschlüsseln.</string>
- <string name="base_pkg_err">Basis-Pakete konnten nicht geladen werden.</string>
- <string name="simulating">Simuliere Aktionen...</string>
- <string name="backup_cancel">Sicherung abgebrochen</string>
- <string name="config_twrp">Konfiguriere TWRP...</string>
- <string name="config_twrp_err">Konfigurieren von TWRP mit diesem Kernel nicht möglich.</string>
- <string name="copy_log">Recovery-Log wurde nach {1} kopiert.</string>
- <string name="max_queue">Maximale Anzahl an ZIP-Dateien erreicht!</string>
- <string name="min_queue">Minimale Anzahl an ZIP-Dateien erreicht!</string>
- <string name="screenshot_saved">Bildschirmfoto gespeichert unter {1}</string>
- <string name="screenshot_err">Bildschirmfoto konnte nicht erstellt werden!</string>
- <string name="zip_wipe_cache">Eine oder mehrere ZIP-Dateien wollen den Cache löschen -- Lösche Cache jetzt.</string>
- <string name="and_sec_wipe_err">Android Secure kann nicht gelöscht werden</string>
- <string name="dalvik_wipe_err">Löschen von Dalvik fehlgeschlagen</string>
- <string name="auto_gen">(automatisch erstellt)</string>
- <string name="curr_date">(aktuelles Datum)</string>
- <string name="backup_name_len">Der Name der Sicherung ist zu lang.</string>
- <string name="backup_name_invalid">Sicherungs-Name '{1}' enthält ungültige Zeichen: '{1}'</string>
- <string name="no_real_sdcard">Dieses Gerät verwendet keine SD-Karte! Abbruch!</string>
- <string name="cancel_sideload">ADB Sideload wird abgebrochen...</string>
- <string name="change_fs_err">Fehler beim Ändern des Dateisystems.</string>
- <string name="theme_ver_err">Theme-Version inkompatibel zu TWRP-Version. Standard-Theme wird verwendet.</string>
- <string name="up_a_level">(Übergeordneter Ordner)</string>
- </resources>
+<?xml version="1.0"?> + +<language> + <display>Deutsch</display> + + <resources> + <!-- Font overrides - only change these if your language requires special characters --> + <resource name="font_l" type="fontoverride" filename="RobotoCondensed-Regular.ttf" scale="100" /> + <resource name="font_m" type="fontoverride" filename="RobotoCondensed-Regular.ttf" scale="100" /> + <resource name="font_s" type="fontoverride" filename="RobotoCondensed-Regular.ttf" scale="100" /> + <resource name="fixed" type="fontoverride" filename="DroidSansMono.ttf" scale="100" /> + + <!-- Partition display names --> + <string name="system">System</string> + <string name="system_image">System-Image</string> + <string name="vendor">Anbieter</string> + <string name="vendor_image">Anbieter-Image</string> + <string name="boot">Boot</string> + <string name="recovery">Recovery</string> + <string name="cache">Cache</string> + <string name="data">Daten</string> + <string name="sdcard">SD-Karte</string> + <string name="internal">Interner Speicher</string> + <string name="microsd">Micro SD-Karte</string> + <string name="usbotg">USB-OTG</string> + <string name="android_secure">Android Secure</string> + <string name="dalvik">Dalvik / ART Cache</string> + <!-- This is a rarely used partition on a Micro SD card for a very old app2sd system --> + <string name="sdext">SD-EXT</string> + <string name="adopted_data">Eingegliederte Daten-Partition</string> + <string name="adopted_storage">Eingegliederte Storage-Partition</string> + + <!-- GUI XML strings --> + <string name="twrp_header">Team Win Recovery Project</string> + <string name="twrp_watch_header">TWRP %tw_version%</string> + <string name="cpu_temp">CPU: %tw_cpu_temp% °C</string> + <string name="battery_pct">Akku: %tw_battery%</string> + <string name="sort_by_name">Nach Namen sortieren</string> + <string name="sort_by_date">Nach Datum sortieren</string> + <string name="sort_by_size">Nach Größe sortieren</string> + <string name="sort_by_name_only">Name</string> + <string name="sort_by_date_only">Datum</string> + <string name="sort_by_size_only">Größe</string> + <string name="tab_general">ALLGEMEIN</string> + <string name="tab_options">OPTIONEN</string> + <string name="tab_backup">SICHERUNG</string> + <string name="tab_time_zone">ZEITZONE</string> + <string name="tab_screen">BILDSCHIRM</string> + <string name="tab_vibration">VIBRATION</string> + <string name="tab_language">SPRACHE</string> + + <string name="install_btn">Installieren</string> + <string name="wipe_btn">Löschen</string> + <string name="backup_btn">Sichern</string> + <string name="restore_btn">Wiederherstellen</string> + <string name="mount_btn">Einhängen</string> + <string name="settings_btn">Einstellungen</string> + <string name="advanced_btn">Erweitert</string> + <string name="reboot_btn">Neustart</string> + <string name="files_btn">Dateien</string> + <string name="copy_log_btn">Log kopieren</string> + <string name="select_type_hdr">Typ wählen</string> + <string name="install_zip_hdr">ZIP installieren</string> + <string name="install_zip_btn">ZIP installieren</string> + <string name="install_image_hdr">Image installieren</string> + <string name="install_image_btn">Image installieren</string> + <string name="install_select_file_hdr">Datei auswählen</string> + <string name="file_selector_folders_hdr">Ordner</string> + <string name="select_file_from_storage">Speicher: %tw_storage_display_name% (%tw_storage_free_size% MB)</string> + <string name="adb_sideload_btn">ADB Sideload</string> + <string name="install_hdr">Installieren</string> + <string name="select_storage_hdr">Speicher auswählen</string> + <string name="select_storage_btn">Speicher auswählen</string> + <string name="queue_hdr">Einreihen</string> + <string name="zip_queue_count">%tw_zip_queue_count% von 10 Dateien eingereiht</string> + <string name="zip_queue_count_s">File %tw_zip_queue_count% von 10:</string> + <string name="zip_warn1">Dieser Vorgang kann möglicherweise inkompatible</string> + <string name="zip_warn2">Software installieren und das Gerät unbrauchbar machen.</string> + <string name="zip_back_cancel">Zurück-Taste drücken, um den Vorgang abzubrechen.</string> + <string name="zip_back_clear">Zurück-Taste drücken, um Warteschlange zu löschen.</string> + <string name="folder">Ordner:</string> + <string name="file">Datei:</string> + <string name="zip_sig_chk">ZIP-Signaturprüfung</string> + <string name="inject_twrp_chk">TWRP nach der Installation injizieren</string> + <string name="options_hdr">Einstellungen</string> + <string name="confirm_flash_hdr">Flashvorgang bestätigen</string> + <string name="zip_queue">Warteschlange:</string> + <string name="options">Einstellungen:</string> + <string name="swipe_confirm">Bestätigen</string> + <string name="zip_add_btn">Mehr ZIPs hinzufügen</string> + <string name="zip_clear_btn">ZIP-Stapel löschen</string> + <string name="install_zip_count_hdr">Installiere ZIP %tw_zip_index% von %tw_zip_queue_count%</string> + <string name="installing_zip_xml">Installiere ZIP: %tw_file%</string> + <string name="failed">Fehlgeschlagen</string> + <string name="successful">Erfolgreich</string> + <string name="install_failed">Installation fehlgeschlagen</string> + <string name="install_successful">Installation erfolgreich</string> + <string name="wipe_cache_dalvik_btn">Cache/Dalvik löschen</string> + <string name="reboot_system_btn">System neustarten</string> + <string name="install_sel_target">Ziel-Partition auswählen</string> + <string name="flash_image_select">Partition wählen, um das Image zu einzuspielen:</string> + <string name="target_partition">Ziel-Partition:</string> + <string name="flashing_image">Spiele Image ein...</string> + <string name="image_flashed">Image eingespielt</string> + <string name="wipe_cache_dalvik_confirm">Cache & Dalvik löschen?</string> + <string name="wiping_cache_dalvik">Lösche Cache & Dalvik...</string> + <string name="wipe_cache_dalvik_complete">Cache & Dalvik gelöscht</string> + <string name="swipe_wipe">Löschen bestätigen</string> + <string name="swipe_wipe_s">Löschen</string> + <string name="no_os1">Kein OS installiert! Sind Sie</string> + <string name="no_osrb">sicher, dass Sie neu starten möchten?</string> + <string name="no_ospo">sicher, dass Sie ausschalten möchten?</string> + <string name="rebooting">Neustart...</string> + <string name="swipe_reboot">Neustart bestätigen</string> + <string name="swipe_reboot_s">Neustart</string> + <string name="swipe_flash">Installation bestätigen</string> + <string name="confirm_action">Aktion bestätigen</string> + <string name="back_cancel">Zurück-Taste drücken, um den Vorgang abzubrechen.</string> + <string name="cancel_btn">Abbruch</string> + <string name="wipe_hdr">Löschen</string> + <string name="factory_reset_hdr">Werkseinstellungen herstellen</string> + <string name="factory_reset_btn">Werkseinstellungen</string> + <string name="factory_reset1">Löscht Daten, Cache und Dalvik</string> + <string name="factory_reset2">(ohne internen Speicher)</string> + <string name="factory_reset3">Diese Lösch-Aktion ist in der Regel</string> + <string name="factory_reset4">die einzige die benötigt wird.</string> + <string name="factory_resetting">Werkseinstellungen herstellen...</string> + <string name="advanced_wipe_hdr">Erweitertes Löschen</string> + <string name="advanced_wipe_btn">Erweitertes Löschen</string> + <string name="wipe_enc_confirm">Verschlüsselung der Daten-Partition enfernen?</string> + <string name="formatting_data">Data wird formatiert...</string> + <string name="swipe_format_data">Daten-Partition formatieren</string> + <string name="swipe_format_data_s">Formatieren </string> + <string name="factory_reset_complete">Werkseinstellungen wiederhergestellt</string> + <string name="sel_part_hdr">Partitionen auswählen</string> + <string name="wipe_sel_confirm">Ausgewählte Partition(en) löschen?</string> + <string name="wiping_part">Lösche Partition(en)...</string> + <string name="wipe_complete">Löschen abgeschlossen </string> + <string name="sel_part_wipe">Partitionen zum Löschen auswählen:</string> + <string name="invalid_part_sel">Ungültige Partitionsauswahl</string> + <string name="format_data_hdr">Daten-Partition formatieren</string> + <string name="format_data_btn">Daten formatieren</string> + <string name="format_data_ptr1">Formatieren der Daten-Partition löscht alle Apps,</string> + <string name="format_data_ptr2">Sicherungen, Bilder, Videos, sowie Medien und</string> + <string name="format_data_ptr3">entfernt die Verschlüsselung des internen Speichers.</string> + <string name="format_data_adopted">Betrifft auch die eingegliederte Storage-Partition.</string> + <string name="format_data_lcp1">Formatieren der Daten-Partition löscht alle Apps, Backups, Bilder, Videos, sowie Medien und</string> + <string name="format_data_lcp2">entfernt die Verschlüsselung des internen Speichers.</string> + <string name="format_data_wtc1">Formatieren der Daten-Partition löscht alle Apps,</string> + <string name="format_data_wtc2">Sicherungen und Medien. Dies kann nicht rückgängig gemacht werden.</string> + <string name="format_data_undo">Dies kann nicht rückgängig gemacht werden.</string> + <string name="format_data_complete">Formatieren der Daten-Partition abgeschlossen</string> + <!-- Translator note: the word "yes" must remain English! --> + <string name="yes_continue">"yes" eingeben um fortzufahren. Zurück drücken um abzubrechen.</string> + <string name="part_opt_hdr">Optionen für %tw_partition_name%-Partition</string> + <string name="sel_act_hdr">Aktion auswählen</string> + <string name="file_sys_opt">Dateisystem-Optionen</string> + <string name="partition">Partition: %tw_partition_name%</string> + <string name="part_mount_point">Einhängepunkt: %tw_partition_mount_point%</string> + <string name="part_curr_fs">Dateisystem: %tw_partition_file_system%</string> + <string name="part_present_yes">Vorhanden: Ja</string> + <string name="part_present_no">Vorhanden: Nein</string> + <string name="part_removable_yes">Wechselbar: Ja</string> + <string name="part_removable_no">Wechselbar: Nein</string> + <string name="part_size">Größe: %tw_partition_size%MB</string> + <string name="part_used">Verwendet: %tw_partition_used%MB</string> + <string name="part_free">Frei: %tw_partition_free%MB</string> + <string name="part_backup_size">Backup-Größe: %tw_partition_backup_size%MB</string> + <string name="resize_btn">Größe ändern</string> + <string name="resize_btn_s">Größe ändern</string> + <string name="resize_confirm">Größe von %tw_partition_name% ändern?</string> + <string name="resizing">Ändere Größe...</string> + <string name="resize_complete">Größenänderung abgeschlossen</string> + <string name="swipe_resize">Größenänderung bestätigen</string> + <string name="swipe_resize_s">Größe ändern</string> + <string name="repair_btn">Dateisystem reparieren</string> + <string name="repair_btn_s">Reparatur</string> + <string name="repair_confirm">Repariere %tw_partition_name%?</string> + <string name="repairing">Reparieren...</string> + <string name="repair_complete">Reparatur abgeschlossen</string> + <string name="swipe_repair">Partition reparieren</string> + <string name="swipe_repair_s">Reparieren</string> + <string name="change_fs_btn">Dateisystem ändern</string> + <string name="change_fs_btn_s">Ändern</string> + <string name="change_fs_for_hdr">Dateisystem für %tw_partition_name% ändern</string> + <string name="change_fs_for_hdr_s">Partition: %tw_partition_name% > Dateisystem auswählen</string> + <string name="change_fs_warn1">Einige ROMs oder Kernel unterstützen möglicherweise manche</string> + <string name="change_fs_warn2">Dateisysteme nicht. Überlegt handeln!</string> + <string name="change_fs_confirm">%tw_partition_name% ändern?</string> + <string name="formatting">Formatierung...</string> + <string name="format_complete">Formatieren abgeschlossen</string> + <string name="swipe_change_fs">Dateisystem ändern</string> + <string name="swipe_change_s">Ändern</string> + <string name="back_btn">Zurück</string> + <string name="wipe_enc_btn">Verschlüsselung entfernen</string> + <string name="swipe_factory_reset">Werkseinstellungen herstellen</string> + <string name="repair_change_btn">Dateisystem reparieren oder ändern</string> + <string name="storage_hdr">Speicher: %tw_storage_display_name% (%tw_storage_free_size% MB)</string> + <string name="backup_hdr">Sicherung</string> + <string name="backup_confirm_hdr">Sicherung bestätigen</string> + <string name="encryption_tab">VERSCHLÜSSELUNG</string> + <string name="encryption">Verschlüsselung:</string> + <string name="name">Name:</string> + <string name="sel_part_backup">Zu sichernde Partitionen auswählen:</string> + <string name="storage">Speicherort:</string> + <string name="enc_disabled">deaktiviert - Passwort festlegen um zu aktivieren</string> + <string name="enc_enabled">aktiviert</string> + <string name="enable_backup_comp_chk">Komprimierung aktivieren</string> + <string name="skip_md5_backup_chk">MD5-Erstellung für die Sicherung überspringen</string> + <string name="disable_backup_space_chk">Prüfung auf freien Speicher deaktivieren</string> + <string name="refresh_sizes_btn">Größen aktualisieren</string> + <string name="swipe_backup">Sicherung erstellen</string> + <string name="append_date_btn">Datum anhängen</string> + <string name="backup_name_exists">Eine Sicherung mit diesem Namen existiert bereits!</string> + <string name="encrypt_backup">Sicherung verschlüsseln?</string> + <string name="enter_pass">Passwort eingeben:</string> + <string name="enter_pass2">Passwort erneut eingeben:</string> + <string name="pass_not_match">Die Passwörter stimmen nicht überein!</string> + <string name="partitions">Partitionen:</string> + <string name="disabled">Deaktiviert</string> + <string name="enabled">Aktiviert</string> + <string name="backup_name_hdr">Sicherung benennen</string> + <string name="progress">Fortschritt:</string> + <string name="backup_complete">Sicherung abgeschlossen</string> + <string name="restore_hdr">Wiederherstellen</string> + <string name="sel_backup_hdr">Sicherung auswählen</string> + <string name="restore_sel_store_hdr">Speicher: %tw_storage_display_name% (%tw_storage_free_size% MB)</string> + <string name="restore_sel_pack_fs">Wiederherzustellende Sicherung:</string> + <string name="restore_enc_backup_hdr">Verschlüsselte Sicherung</string> + <string name="restore_dec_fail">Passwort falsch, bitte erneut versuchen!</string> + <string name="del_backup_btn">Sicherung löschen</string> + <string name="del_backup_confirm">Sicherung löschen?</string> + <string name="del_backup_confirm2">Dies kann nicht rückgängig gemacht werden!</string> + <string name="deleting_backup">Sicherung wird gelöscht...</string> + <string name="backup_deleted">Sicherung gelöscht</string> + <string name="swipe_delete">Löschen bestätigen</string> + <string name="swipe_delete_s">Löschen</string> + <string name="restore_try_decrypt">Verschlüsselte Sicherung - Versuche Entschlüsselung</string> + <string name="restore_try_decrypt_s">Versuche Entschlüsselung</string> + <string name="restore_backup_date">Sicherung am %tw_restore_file_date% erstellt</string> + <string name="restore_sel_part">Wiederherzustellende Partitionen:</string> + <string name="restore_enable_md5_chk">MD5-Prüfung von Sicherungs-Dateien aktivieren</string> + <string name="restore_complete">Wiederherstellen abgeschlossen</string> + <string name="swipe_restore">Sicherung wiederherstellen</string> + <string name="swipe_restore_s">Wiederherstellen</string> + <string name="rename_backup_hdr">Sicherung umbenennen</string> + <string name="rename_backup_confirm">Sicherung umbenennen?</string> + <string name="rename_backup_confirm2">Dies kann nicht rückgängig gemacht werden!</string> + <string name="renaming_backup">Sicherung wird umbenannt...</string> + <string name="rename_backup_complete">Sicherung umbenannt</string> + <string name="swipe_to_rename">Name ändern</string> + <string name="swipe_rename">Umbenennen</string> + <string name="confirm_hdr">Bestätigen</string> + <string name="mount_hdr">Einhängen</string> + <string name="mount_sel_part">Einzuhängende Partitionen:</string> + <string name="mount_sys_ro_chk">System-Partition schreibgeschützt einhängen</string> + <string name="mount_sys_ro_s_chk">System nur lesend</string> + <string name="decrypt_data_btn">Daten-Partition entschlüsseln</string> + <string name="disable_mtp_btn">MTP deaktivieren</string> + <string name="enable_mtp_btn">MTP aktivieren</string> + <string name="mount_usb_storage_btn">USB-Speicher einhängen</string> + <string name="usb_storage_hdr">USB-Speicher</string> + <string name="usb_stor_mnt1">USB-Speicher eingehängt</string> + <string name="usb_stor_mnt2">Funktion "USB-Speicher auswerfen" auf Computer</string> + <string name="usb_stor_mnt3">ausführen, bevor die Verbindung getrennt wird!</string> + <string name="unmount_btn">Auswerfen</string> + <string name="rb_system_btn">System</string> + <string name="rb_poweroff_btn">Ausschalten</string> + <string name="rb_recovery_btn">Recovery</string> + <string name="rb_bootloader_btn">Bootloader</string> + <string name="rb_download_btn">Download</string> + <string name="turning_off">Ausschalten...</string> + <string name="swipe_power_off">Gerät ausschalten</string> + <string name="swipe_power_off_s">Ausschalten</string> + <string name="sys_ro_hdr">Unveränderte Systempartition</string> + <string name="sys_ro_keep">System-Partition schreibgeschützt einhängen?</string> + <string name="sys_rop1">TWRP kann die System Partition unverändert lassen,</string> + <string name="sys_rop2">um das Aufspielen offizieller Updates zu erleichtern.</string> + <string name="sys_rop3">Jedoch kann dann das Hersteller-ROM TWRP wieder</string> + <string name="sys_rop4">überschreiben und TWRP kann keinen Root-Zugriff einrichten.</string> + <string name="sys_rop5">Das Installieren von ZIPs oder ADB-Vorgänge könnten jedoch</string> + <string name="sys_rop6">trotzdem die System-Partition verändern.</string> + <string name="sys_rol1">TWRP kann die System-Partition unverändert lassen, um das Aufspielen offizieller Updates zu erleichtern.</string> + <string name="sys_rol2">Jedoch kann dann das Hersteller-ROM TWRP wieder überschreiben und TWRP kann keinen Root-Zugriff einrichten.</string> + <string name="sys_rol3">Das Installieren von ZIPs oder ADB-Vorgänge könnten jedoch trotzdem die System-Partition verändern.</string> + <string name="sys_ro_never_show_chk">Diesen Bildschirm nicht mehr anzeigen</string> + <string name="sys_ro_keep_ro_btn">Nur-Lesend belassen</string> + <string name="swipe_allow_mod">Veränderungen zulassen</string> + <string name="swipe_allow_mod_s">Zulassen</string> + <string name="settings_hdr">Einstellungen</string> + <string name="settings_gen_hdr">Allgemein</string> + <string name="settings_gen_s_hdr">Allgemein</string> + <string name="settings_gen_btn">Allgemein</string> + <string name="use_rmrf_chk">rm -rf anstatt formatieren verwenden</string> + <string name="use24clock_chk">24-Stunden-Format</string> + <string name="rev_navbar_chk">Navigationsleiste umkehren</string> + <string name="simact_chk">Benutzeraktionen für Design-Test simulieren</string> + <string name="simfail_chk">Fehlschlag von Benutzeraktionen simulieren</string> + <string name="ctr_navbar_rdo">Navigationsleisten-Schaltflächen zentrieren</string> + <string name="lft_navbar_rdo">Navigationsleisten-Schaltflächen linksbündig</string> + <string name="rht_navbar_rdo">Navigationsleisten-Schaltflächen rechtsbündig</string> + <string name="restore_defaults_btn">Standard herstellen</string> + <string name="settings_tz_btn">Zeitzone</string> + <string name="settings_screen_btn">Bildschirm</string> + <string name="settings_screen_bright_btn">Helligkeit</string> + <string name="settings_vibration_btn">Vibration</string> + <string name="settings_language_btn">Sprache</string> + <string name="time_zone_hdr">Zeitzone</string> + <string name="sel_tz_list">Zeitzone auswählen:</string> + <!-- Translator note: if it does not make sense to translate the locations or if it makes more sense, + feel free to change the location listed or drop the location entirely and just call it UTC -6 --> + <string name="utcm11">(UTC -11) Samoa, Midwayinseln</string> + <string name="utcm10">(UTC -10) Hawaii</string> + <string name="utcm9">(UTC -9) Alaska</string> + <string name="utcm8">(UTC -8) Pacific Time</string> + <string name="utcm7">(UTC -7) Mountain Time (USA)</string> + <string name="utcm6">(UTC -6) Central Time (USA)</string> + <string name="utcm5">(UTC -5) Eastern Time (USA)</string> + <string name="utcm4">(UTC -4) Atlantic Time (USA)</string> + <string name="utcm3">(UTC -3) Brasilien, Buenos Aires</string> + <string name="utcm2">(UTC -2) Mittelatlantik</string> + <string name="utcm1">(UTC -1) Azores, Cape Verde</string> + <string name="utc0">(UTC 0) London, Dublin, Lissabon</string> + <string name="utcp1">(UTC +1) Berlin, Brüssel, Paris</string> + <string name="utcp2">(UTC +2) Athen, Istanbul, Südafrika</string> + <string name="utcp3">(UTC +3) Moskau, Bagdad</string> + <string name="utcp4">(UTC +4) Abu Dhabi, Tiflis, Muscat</string> + <string name="utcp5">(UTC +5) Ekaterinburg, Islamabad</string> + <string name="utcp6">(UTC +6) Almaty, Dhaka, Colombo</string> + <string name="utcp7">(UTC +7) Bangkok, Hanoi, Jakarta</string> + <string name="utcp8">(UTC +8) Peking, Singapur, Hong Kong</string> + <string name="utcp9">(UTC +9) Tokio, Seoul, Yakutsk</string> + <string name="utcp10">(UTC +10) Ostaustralien, Guam</string> + <string name="utcp11">(UTC +11) Vladivostok, Salomonen</string> + <string name="utcp12">(UTC +12) Auckland, Wellington, Fidschi</string> + <string name="sel_tz_offset">Zeitverschiebung: %tw_time_zone_guioffset%</string> + <string name="tz_offset_none">Keine</string> + <string name="tz_offset_0">0</string> + <string name="tz_offset_15">15</string> + <string name="tz_offset_30">30</string> + <string name="tz_offset_45">45</string> + <string name="use_dst_chk">Sommerzeit verwenden</string> + <string name="curr_tz">Aktuelle Zeitzone: %tw_time_zone%</string> + <string name="curr_tz_s">Aktuelle Zeitzone:</string> + <string name="set_tz_btn">Zeitzone einstellen</string> + <string name="settings_screen_hdr">Bildschirm</string> + <string name="settings_screen_timeout_hdr">Bildschirm-Timeout</string> + <string name="enable_timeout_chk">Bildschirm automatisch ausschalten</string> + <string name="screen_to_slider">Zeit bis zum Ausschalten (in Sekunden):</string> + <string name="screen_to_slider_s">Ausschalten nach %tw_screen_timeout_secs% Sekunden (0 = deaktiviert): </string> + <string name="screen_to_na">Bildschirm-Abschaltung nicht verfügbar</string> + <string name="screen_bright_slider">Helligkeit: %tw_brightness_pct% %</string> + <string name="screen_bright_na">Helligkeit nicht verfügbar</string> + <string name="vibration_hdr">Vibration</string> + <string name="button_vibration_hdr">Tasten-Vibration</string> + <string name="kb_vibration_hdr">Tastatur-Vibration</string> + <string name="act_vibration_hdr">Aktion-Vibration</string> + <string name="button_vibration">Tasten-Vibration:</string> + <string name="kb_vibration">Tastatur-Vibration:</string> + <string name="act_vibration">Aktion-Vibration:</string> + <string name="select_language">Sprache auswählen:</string> + <string name="sel_lang_btn">Sprache auswählen</string> + <string name="set_language_btn">Sprache einstellen</string> + <string name="advanced_hdr">Erweitert</string> + <string name="copy_log_confirm">Log auf SD-Karte kopieren?</string> + <string name="copying_log">Kopiere Log auf SD-Karte...</string> + <string name="copy_log_complete">Log erfolgreich kopiert</string> + <string name="fix_context_btn">SELinux Kontexte</string> + <string name="part_sd_btn">SD-Karte partitionieren</string> + <string name="part_sd_s_btn">SD-Karte</string> + <string name="file_manager_btn">Dateimanager</string> + <string name="language_hdr">Sprache</string> + <string name="terminal_btn">Terminal</string> + <string name="reload_theme_btn">Theme neu laden</string> + <string name="dumlock_btn">HTC Dumlock</string> + <string name="inject_twrp_btn">TWRP injizieren</string> + <string name="inject_twrp_confirm">TWRP wieder injizieren?</string> + <string name="injecting_twrp">TWRP wird wieder injiziert...</string> + <string name="inject_twrp_complete">TWRP-Injektion abgeschlossen</string> + <string name="swipe_to_confirm">Aktion bestätigen</string> + <string name="part_sd_hdr">SD-Karte partitionieren</string> + <string name="invalid_partsd_sel">Es muss ein Wechseldatenträger ausgewählt sein!</string> + <string name="part_sd_lose">Es werden alle Dateien von der SD-Karte gelöscht!</string> + <string name="part_sd_undo">Diese Aktion kann nicht rückgängig gemacht werden!</string> + <string name="part_sd_ext_sz">EXT Grösse:</string> + <string name="part_sd_swap_sz">Grösse der Auslagerungsdatei:</string> + <string name="part_sd_m">-</string> + <string name="part_sd_p">+</string> + <string name="file_system">Dateisystem:</string> + <string name="swipe_part_sd">SD-Karte partitionieren</string> + <string name="swipe_part_sd_s">Partitionieren</string> + <string name="partitioning_sd">Partitioniere SD-Karte...</string> + <string name="partitioning_sd2">Dies wird einige Minuten dauern.</string> + <string name="part_sd_complete">Partitionierung abgeschlossen</string> + <string name="dumlock_hdr">HTC Dumlock</string> + <string name="dumlock_restore_btn">Orginale Boot-Partition wiederherstellen</string> + <string name="dumlock_restore_confirm">Orginales Image der Boot-Partition wiederherstellen?</string> + <string name="dumlock_restoring">Stelle originale Boot-Partition wieder her...</string> + <string name="dumlock_restore_complete">Wiederherstellen der orginalen Boot-Partition abgeschlossen</string> + <string name="dumlock_reflash_btn">Recovery erneut einspielen</string> + <string name="dumlock_reflash_confirm">Recovery in Boot-Partition einspielen?</string> + <string name="dumlock_reflashing">Spiele Recovery in Boot-Partition ein...</string> + <string name="dumlock_reflash_complete">Einspielen der Recovery in die Boot-Partition abgeschlossen</string> + <string name="dumlock_install_btn">HTC Dumlock installieren</string> + <string name="dumlock_install_confirm">HTC Dumlock Dateien ins ROM installieren?</string> + <string name="dumlock_installing">Installiere HTC Dumlock...</string> + <string name="dumlock_install_complete">HTC Dumlock Installation abgeschlossen</string> + <string name="swipe_to_unlock">Bildschirm entsperren</string> + <string name="swipe_unlock">Entsperren</string> + <string name="fm_hdr">Dateimanager</string> + <string name="fm_sel_file">Datei oder Ordner auswählen</string> + <string name="fm_type_folder">Ordner</string> + <string name="fm_type_file">Datei</string> + <string name="fm_choose_act">Aktion auswählen</string> + <string name="fm_selected">%tw_fm_type%:</string> + <string name="fm_copy_btn">Kopieren</string> + <string name="fm_copy_file_btn">Datei kopieren</string> + <string name="fm_copy_folder_btn">Ordner kopieren</string> + <string name="fm_copying">Kopiere</string> + <string name="fm_move_btn">Verschieben</string> + <string name="fm_moving">Verschiebe</string> + <string name="fm_chmod755_btn">chmod 755</string> + <string name="fm_chmod755ing">chmod 755</string> + <string name="fm_chmod_btn">chmod</string> + <string name="fm_delete_btn">Löschen</string> + <string name="fm_deleting">Wird gelöscht</string> + <string name="fm_rename_btn">Umbenennen</string> + <string name="fm_rename_file_btn">Datei umbennen</string> + <string name="fm_rename_folder_btn">Ordner umbenennen</string> + <string name="fm_renaming">Umbenennen</string> + <string name="fm_sel_dest">Zielordner auswählen</string> + <string name="fm_sel_curr_folder">Aktuellen Ordner auswählen</string> + <string name="fm_rename_hdr">Umbenennen</string> + <string name="fm_set_perms_hdr">Berechtigungen setzen</string> + <string name="fm_perms">Berechtigungen:</string> + <string name="fm_complete">Dateivorgang beendet</string> + <string name="decrypt_data_hdr">Daten-Partition entschlüsseln</string> + <string name="decrypt_data_enter_pass">Passwort eingeben:</string> + <string name="decrypt_data_failed">Passwort falsch, bitte erneut versuchen!</string> + <string name="decrypt_data_failed_pattern">Muster falsch, bitte erneut versuchen!</string> + <string name="decrypt_data_enter_pattern">Muster eingeben.</string> + <string name="decrypt_data_trying">Versuche Entschlüsselung</string> + <string name="term_hdr">Terminal</string> + <string name="term_s_hdr">Terminal</string> + <string name="term_kill_btn">BEENDEN</string> + <string name="term_sel_folder_hdr">Zum Stamm-Ordner navigieren</string> + <string name="adb_sideload_hdr">ADB Sideload</string> + <string name="sideload_wipe_dalvik_chk">Dalvik Cache löschen</string> + <string name="sideload_wipe_cache_chk">Cache löschen</string> + <string name="swipe_to_sideload">ADB Sideload starten</string> + <string name="swipe_sideload">Start</string> + <string name="sideload_confirm">ADB Sideload</string> + <string name="sideload_usage">Syntax: adb sideload Dateiname.zip</string> + <string name="sideload_complete">ADB Sideload abgeschlossen</string> + <string name="fix_contexts_hdr">SELinux Kontexte</string> + <string name="fix_contexts_note1">Das Korrigieren der Kontexte wird selten benötigt!</string> + <string name="fix_contexts_note2">Das Korrigieren der SELinux-Kontexte kann</string> + <string name="fix_contexts_note3">Startprobleme verursachen.</string> + <string name="swipe_to_fix_contexts">SELinux-Kontexte korrigieren</string> + <string name="swipe_fix_contexts">Korrigieren</string> + <string name="fixing_contexts">Korrigiere Kontexte...</string> + <string name="fix_contexts_complete">Korrektur der Kontexte abgeschlossen</string> + <string name="reboot_hdr">Neustart</string> + <string name="su_hdr">SuperSU-Check</string> + <string name="su_note1">Anscheinend besteht kein Root-Zugriff auf das Gerät.</string> + <string name="su_note2">SuperSU jetzt installieren?</string> + <string name="su_note3">Dies wird Root-Zugriff auf dem Gerät einrichten.</string> + <string name="su_cancel">Nicht installieren</string> + <string name="swipe_su_to_install">Installation starten</string> + <string name="swipe_su_install">Installieren</string> + <string name="su_installing">Installiere SuperSU</string> + <string name="sel_storage_list">Speicher auswählen</string> + <string name="ok_btn">OK</string> + + <!-- Various console messages - these consist of user displayed messages, oftentimes errors --> + <string name="no_kernel_selinux">Dieser Kernel hat keine Leseunterstützung für SELinux-Kontexte.</string> + <string name="full_selinux">Volle SELinux-Unterstützung ist vorhanden.</string> + <string name="no_selinux">Keine SELinux-Unterstützung vorhanden (kein libselinux).</string> + <string name="mtp_enabled">MTP aktiviert</string> + <string name="mtp_crash">MTP abgestürzt, MTP wird künftig während Bootvorgang nicht gestartet.</string> + <string name="decrypt_success">Erfolgreich mit dem Standardpasswort entschlüsselt.</string> + <string name="unable_to_decrypt">Entschlüsselung mit Standardpasswort nicht möglich. Eventuell muss die Daten-Partition formatiert werden.</string> + <string name="generating_md51">Generiere MD5</string> + <!-- Message displayed during a backup if we are generating an MD5, ideally, leave the leading " * " to help align and separate this text from other console text --> + <string name="generating_md52"> * generiere MD5...</string> + <string name="md5_created"> * MD5 erstellt.</string> + <string name="md5_error"> * MD5-Fehler!</string> + <string name="md5_compute_error"> * Fehler bei MD5-Berechnung.</string> + <string name="current_date">(Aktuelles Datum)</string> + <string name="auto_generate">(Automatisch generiert)</string> + <string name="unable_to_locate_partition">Kann '{1}' Partition für Sicherungsberechnungen nicht finden.</string> + <string name="no_partition_selected">Keine Partitionen für die Sicherung ausgewählt.</string> + <string name="total_partitions_backup"> * Anzahl der zu sichernden Partitionen: {1}</string> + <string name="total_backup_size"> * Gesamtgröße aller Daten: {1}MB</string> + <string name="available_space"> * Verfügbare Speicherplatz: {1}MB</string> + <string name="unable_locate_storage">Kann Sicherungsgerät nicht finden.</string> + <string name="no_space">Zu wenig freier Speicherplatz.</string> + <string name="backup_started">[SICHERUNG GESTARTET]</string> + <string name="backup_folder"> * Ordner für Sicherung: {1}</string> + <string name="fail_backup_folder">Ordner für Sicherung konnte nicht erstellt werden.</string> + <string name="avg_backup_fs">Durchschnittliche Sicherungsgeschwindigkeit für Dateisysteme: {1} MB/sek</string> + <string name="avg_backup_img">Durchschnittliche Sicherungsgeschwindigkeit für abgebildete Laufwerke: {1} MB/sek</string> + <string name="total_backed_size">[INSGESAMT {1} MB GESICHERT]</string> + <string name="backup_completed">[SICHERUNG ABGESCHLOSSEN IN {1} SEKUNDEN]</string> + <string name="restore_started">[WIEDERHERSTELLEN GESTARTET]</string> + <string name="restore_folder">Wiederherstellen aus Ordner: '{1}'</string> + <!-- {1} is the partition display name and {2} is the number of seconds --> + <string name="restore_part_done">[{1} fertiggestellt ({2} Sekunden)]</string> + <string name="verifying_md5">Überprüfe MD5</string> + <string name="skip_md5">Überspringe MD5-Prüfung aufgrund Benutzereinstellungen.</string> + <string name="calc_restore">Berechne Wiederherstellungsinformationen...</string> + <string name="restore_read_only">Kann {1} nicht wiederherstellen -- Partition schreibgeschützt.</string> + <string name="restore_unable_locate">Kann wiederherzustellende '{1}'-Partition nicht finden.</string> + <string name="no_part_restore">Keine Partitionen zur Wiederherstellung ausgewählt.</string> + <string name="restore_part_count">{1} Partitionen wiederherstellen...</string> + <string name="total_restore_size">Insgesamt wiederherzustellen: {1}MB</string> + <string name="updating_system_details">Aktualisieren der System-Details</string> + <string name="restore_completed">[WIEDERHERSTELLEN ABGESCHLOSSEN IN {1} SEKUNDEN]</string> + <!-- {1} is the path we could not open, {2} is strerror output --> + <string name="error_opening_strerr">Fehler beim Öffnen: '{1}' ({2})</string> + <string name="unable_locate_part_backup_name">Partition nicht anhand des Sicherungsnamens ermittelbar: '{1}'</string> + <string name="unable_find_part_path">Partition für Pfad '{1}' nicht gefunden</string> + <string name="update_part_details">Partitions-Informationen werden aktualisiert...</string> + <string name="update_part_details_done">...Fertig</string> + <string name="wiping_dalvik">Dalvik Cache-Verzeichnisse bereinigen...</string> + <string name="cleaned">Bereinigt: {1}...</string> + <string name="dalvik_done">-- Dalvik Cache-Verzeichnisse bereinigt!</string> + <string name="no_andsec">Keine "Android Secure"-Partitionen gefunden.</string> + <string name="unable_to_locate">{1} nicht gefunden.</string> + <string name="wiping_datamedia">Lösche internen Speicher -- /data/media...</string> + <string name="unable_to_mount">Kann {1} nicht einhängen</string> + <string name="unable_to_mount_internal">Kann internen Speicher nicht einhängen</string> + <string name="unable_to_mount_storage">Kann Speicher nicht einhängen</string> + <string name="fail_decrypt">Entschlüsselung der Daten-Partition fehlgeschlagen.</string> + <string name="no_crypto_support">In diese Version wurde keine Krypto-Unterstützung eingebunden.</string> + <string name="decrypt_success_dev">Erfolgreich mit Standardpasswort entschlüsselt.</string> + <string name="done">Abgeschlossen.</string> + <string name="start_partition_sd">Partitioniere SD Karte...</string> + <string name="partition_sd_locate">Zu partitionierendes Gerät nicht gefunden.</string> + <string name="ext_swap_size">EXT + Auslagerung ist grösser als die SD-Karte.</string> + <string name="remove_part_table">Entferne Partitionstabelle...</string> + <string name="unable_rm_part">Partitionstabelle kann nicht entfernt werden.</string> + <string name="create_part">Erstelle {1}-Partition...</string> + <string name="unable_to_create_part">{1}-Partition kann nicht erstellt werden.</string> + <string name="format_sdext_as">Formatiere sd-ext als {1}...</string> + <string name="part_complete">Partitionierung abgeschlossen.</string> + <string name="unable_to_open">'{1}' kann nicht geöffnet werden.</string> + <string name="mtp_already_enabled">MTP bereits aktiviert</string> + <string name="mtp_fail">Fehler beim Aktivieren von MTP</string> + <string name="no_mtp">MTP-Unterstützung nicht integriert</string> + <string name="image_flash_start">[IMAGE-DATEI WIRD AUFGESPIELT]</string> + <string name="img_to_flash">Aufzuspielendes Image: '{1}'</string> + <string name="flash_unable_locate">'{1}'-Partition wurde nicht gefunden.</string> + <string name="no_part_flash">Keine Partitionen ausgewählt.</string> + <string name="too_many_flash">Zu viele Partitionen ausgewählt.</string> + <string name="invalid_flash">Ungültige Partitions-Auswahl.</string> + <string name="flash_done">[IMAGE EINSPIELEN ABGESCHLOSSEN]</string> + <string name="wiping">Lösche {1}</string> + <string name="repair_not_exist">{1} existiert nicht! Reparieren nicht möglich!</string> + <string name="repairing_using">Reparatur von {1} mit {2}...</string> + <string name="unable_repair">{1} kann nicht repariert werden.</string> + <string name="mount_data_footer">/data konnte nicht eingehängt werden und Krypto-Signatur wurde nicht gefunden.</string> + <!-- {1} is the folder name that we could not create, {2} is strerror output --> + <string name="create_folder_strerr">Ordner '{1}' konnte nicht angelegt werden ({2}).</string> + <!-- {1} is the folder name that we could not mount, {2} is strerror output --> + <string name="fail_mount">'{1}' konnte nicht eingehängt werden ({2})</string> + <!-- {1} is the folder name that we could not unmount, {2} is strerror output --> + <string name="fail_unmount">'{1}' konnte nicht eingehängt werden ({2})</string> + <string name="cannot_resize">Größe von {1} kann nicht geändert werden.</string> + <string name="repair_resize">{1} wird repariert. Danach wird die Größe geändert.</string> + <string name="unable_resize">Größe von {1} kann nicht geändert werden.</string> + <string name="no_md5_found">Keine MD5-Datei für '{1}' gefunden. Bitte MD5-Prüfung für Wiederherstellung deaktivieren.</string> + <string name="md5_fail_match">MD5-Prüfung für '{1}' fehlgeschlagen.</string> + <string name="fail_decrypt_tar">TAR-Datei '{1}' konnte nicht entschlüsselt werden.</string> + <string name="format_data_msg">Ein Neustart von TWRP kann notwendig sein, damit /data wieder verwendet werden kann.</string> + <string name="format_data_err">Formatierung zum Entfernen der Verschlüsselung kann nicht durchgeführt werden.</string> + <string name="formatting_using">Formatiere {1} mit {2}...</string> + <string name="unable_to_wipe">{1} kann nicht gelöscht werden.</string> + <string name="cannot_wipe">Partition {1} kann nicht gelöscht werden.</string> + <string name="remove_all">Entferne alle Dateien unter '{1}'</string> + <string name="wiping_data">Lösche Daten, aber verschone internen Speicher...</string> + <string name="backing_up">Sichere {1}...</string> + <string name="backing">Sichere</string> + <string name="backup_size">Sicherung von '{1}' hat 0 Byte.</string> + <string name="datamedia_fs_restore">WARNUNG: Diese Sicherung wurde mit dem Dateisystem {1} erstellt! Es kann sein, dass wieder zu {1} gewechselt werden muss, damit das Gerät nach der Wiederherstellung auch startet.</string> + <string name="restoring">Wiederherstellung läuft</string> + <string name="restoring_hdr">Wiederherstellung läuft</string> + <string name="recreate_folder_err">Ordner {1} kann nicht wiederhergestellt werden.</string> + <string name="img_size_err">Image ist zu groß für das Gerät</string> + <string name="flashing">Einspielen von {1}...</string> + <string name="backup_folder_set"> * Ordner für Sicherung: {1}</string> + <string name="locate_backup_err">Sicherung '{1}' nicht gefunden</string> + <string name="set_restore_opt">Setze Wiederherstellungs-Optionen: '{1}':</string> + <string name="md5_check_skip">MD5-Prüfung ist deaktiviert</string> + <string name="ors_encrypt_restore_err">Eine verschlüsselte Sicherung kann nicht per OpenRecoveryScript wiederhergestellt werden.</string> + <string name="mounting">Einhängen</string> + <string name="unmounting">Auswerfen</string> + <string name="mounted">'{1}' eingehängt</string> + <string name="unmounted">'{1}' ausgeworfen</string> + <string name="setting">Setze '{1}' auf '{2}'</string> + <string name="setting_empty">Setze '{1}' auf leer</string> + <string name="making_dir1">Erstelle Verzeichnis</string> + <string name="making_dir2">Erstelle Verzeichnis: '{1}'</string> + <string name="running_command">Führe Befehl aus</string> + <string name="sideload">ADB Sideload</string> + <string name="start_sideload">Starte ADB Sideload...</string> + <string name="need_new_adb">Für dieses Gerät wird ADB 1.0.32 oder neuer benötigt.</string> + <string name="no_pwd">Kein Passwort angegeben.</string> + <string name="done_ors">Skript wurde verarbeitet</string> + <string name="injecttwrp">Injiziere TWRP in das Boot-Image...</string> + <string name="zip_err">Fehler beim Installieren von ZIP '{1}'</string> + <string name="installing_zip">Installiere Zip: %tw_file%</string> + <string name="select_backup_opt">Setze Sicherungs-Optionen:</string> + <string name="compression_on">Komprimierung ist aktiviert</string> + <string name="md5_off">MD5-Generierung ist deaktiviert</string> + <string name="backup_fail">Sicherung fehlgeschlagen</string> + <string name="backup_clean">Sicherung fehlgeschlagen, bereinige Sicherungs-Verzeichnis</string> + <string name="running_recovery_commands">Führe Recovery-Befehle aus</string> + <string name="recovery_commands_complete">Recovery-Befehle ausgeführt</string> + <string name="running_ors">Führe OpenRecoveryScript aus</string> + <string name="ors_complete">OpenRecoveryScript ausgeführt</string> + <string name="no_updater_binary">'{1}' konnte in der ZIP nicht gefunden werden.</string> + <string name="check_for_md5">Suche nach MD5-Datei...</string> + <string name="fail_sysmap">'{1}' kann nicht zugeordnet werden</string> + <string name="verify_zip_sig">Überprüfe ZIP-Signatur...</string> + <string name="verify_zip_fail">Prüfung der ZIP-Signatur fehlgeschlagen!</string> + <string name="verify_zip_done">Prüfung der ZIP-Signatur erfolgreich.</string> + <string name="zip_corrupt">ZIP-Datei ist beschädigt!</string> + <string name="no_md5">MD5-Prüfung übersprungen: keine MD5-Datei gefunden</string> + <string name="md5_fail">MD5 stimmt nicht überein</string> + <string name="md5_match">MD5 stimmt überein</string> + <string name="pid_signal">Prozess {1} endete mit Meldung: {2}</string> + <string name="pid_error">Prozess {1} endete mit FEHLER: {2}</string> + <string name="install_dumlock">Installiere HTC Dumlock in System-Partition...</string> + <string name="dumlock_restore">Stelle originale Boot-Partition wieder her...</string> + <string name="dumlock_reflash">Recovery auf Boot-Partition einspielen...</string> + <string name="run_script">Führe {1}-Skript aus...</string> + <string name="rename_stock">Hersteller-Recovery in "/system" wurde umbenannt, damit das Hersteller-ROM TWRP nicht überschreibt.</string> + <string name="split_backup">Sicherungs-Datei wird in mehrere Archive aufgeteilt...</string> + <string name="backup_error">Fehler beim Erstellen der Sicherung.</string> + <string name="restore_error">Fehler während der Wiederherstellung.</string> + <string name="split_thread">Teile Thread-ID {1} in Archiv {2}</string> + <!-- These 2 items are saved in the data manager instead of resource manager, so %llu, etc is correct instead of {1} --> + <string name="file_progress">%llu von %llu Dateien, %i%%</string> + <string name="size_progress">%lluMB von %lluMB, %i%%</string> + <string name="decrypt_cmd">Versuche die Daten-Partition per Kommandozeile zu entschlüsseln.</string> + <string name="base_pkg_err">Basis-Pakete konnten nicht geladen werden.</string> + <string name="simulating">Simuliere Aktionen...</string> + <string name="backup_cancel">Sicherung abgebrochen</string> + <string name="config_twrp">Konfiguriere TWRP...</string> + <string name="config_twrp_err">Konfigurieren von TWRP mit diesem Kernel nicht möglich.</string> + <string name="copy_log">Recovery-Log wurde nach {1} kopiert.</string> + <string name="max_queue">Maximale Anzahl an ZIP-Dateien erreicht!</string> + <string name="min_queue">Minimale Anzahl an ZIP-Dateien erreicht!</string> + <string name="screenshot_saved">Bildschirmfoto gespeichert unter {1}</string> + <string name="screenshot_err">Bildschirmfoto konnte nicht erstellt werden!</string> + <string name="zip_wipe_cache">Eine oder mehrere ZIP-Dateien wollen den Cache löschen -- Lösche Cache jetzt.</string> + <string name="and_sec_wipe_err">Android Secure kann nicht gelöscht werden</string> + <string name="dalvik_wipe_err">Löschen von Dalvik fehlgeschlagen</string> + <string name="auto_gen">(automatisch erstellt)</string> + <string name="curr_date">(aktuelles Datum)</string> + <string name="backup_name_len">Der Name der Sicherung ist zu lang.</string> + <string name="backup_name_invalid">Sicherungs-Name '{1}' enthält ungültige Zeichen: '{1}'</string> + <string name="no_real_sdcard">Dieses Gerät verwendet keine SD-Karte! Abbruch!</string> + <string name="cancel_sideload">ADB Sideload wird abgebrochen...</string> + <string name="change_fs_err">Fehler beim Ändern des Dateisystems.</string> + <string name="theme_ver_err">Theme-Version inkompatibel zu TWRP-Version. Standard-Theme wird verwendet.</string> + <string name="up_a_level">(Übergeordneter Ordner)</string> + </resources> </language>
\ No newline at end of file diff --git a/gui/theme/common/languages/ru.xml b/gui/theme/common/languages/ru.xml index 57a99656d..57a99656d 100755..100644 --- a/gui/theme/common/languages/ru.xml +++ b/gui/theme/common/languages/ru.xml diff --git a/gui/theme/landscape_hdpi/splash.xml b/gui/theme/landscape_hdpi/splash.xml index cbadcde72..6616038dd 100644 --- a/gui/theme/landscape_hdpi/splash.xml +++ b/gui/theme/landscape_hdpi/splash.xml @@ -1,51 +1,51 @@ -<?xml version="1.0"?>
-<recovery>
- <details>
- <resolution width="1920" height="1200"/>
- <author>TeamWin</author>
- <title>TWRP</title>
- <description>splash screen</description>
- <themeversion>1</themeversion>
- </details>
-
- <resources>
- <font name="font_l" filename="RobotoCondensed-Regular.ttf" size="52"/>
- <image name="splashlogo" filename="splashlogo" retainaspect="1"/>
- <image name="splashteamwin" filename="splashteamwin" retainaspect="1"/>
- </resources>
-
- <variables>
- <variable name="screen_width" value="1920"/>
- <variable name="screen_height" value="1200"/>
- <variable name="background_color" value="#222222"/>
- <variable name="header_color" value="#555555"/>
- <variable name="accent_color" value="#0090CA"/>
- </variables>
-
- <pages>
- <page name="splash">
- <background color="%background_color%"/>
-
- <fill color="%header_color%">
- <placement x="0" y="0" w="%screen_width%" h="320"/>
- </fill>
-
- <image>
- <image resource="splashlogo"/>
- <placement x="960" y="320" placement="4"/>
- </image>
-
- <image>
- <image resource="splashteamwin"/>
- <placement x="960" y="920" placement="4"/>
- </image>
-
- <text color="%header_color%">
- <font resource="font_l"/>
- <placement x="960" y="970" placement="5"/>
- <text>Recovery Project %tw_version%</text>
- </text>
- </page>
- </pages>
-</recovery>
-
+<?xml version="1.0"?> +<recovery> + <details> + <resolution width="1920" height="1200"/> + <author>TeamWin</author> + <title>TWRP</title> + <description>splash screen</description> + <themeversion>1</themeversion> + </details> + + <resources> + <font name="font_l" filename="RobotoCondensed-Regular.ttf" size="52"/> + <image name="splashlogo" filename="splashlogo" retainaspect="1"/> + <image name="splashteamwin" filename="splashteamwin" retainaspect="1"/> + </resources> + + <variables> + <variable name="screen_width" value="1920"/> + <variable name="screen_height" value="1200"/> + <variable name="background_color" value="#222222"/> + <variable name="header_color" value="#555555"/> + <variable name="accent_color" value="#0090CA"/> + </variables> + + <pages> + <page name="splash"> + <background color="%background_color%"/> + + <fill color="%header_color%"> + <placement x="0" y="0" w="%screen_width%" h="320"/> + </fill> + + <image> + <image resource="splashlogo"/> + <placement x="960" y="320" placement="4"/> + </image> + + <image> + <image resource="splashteamwin"/> + <placement x="960" y="920" placement="4"/> + </image> + + <text color="%header_color%"> + <font resource="font_l"/> + <placement x="960" y="970" placement="5"/> + <text>Recovery Project %tw_version%</text> + </text> + </page> + </pages> +</recovery> + diff --git a/gui/theme/landscape_mdpi/splash.xml b/gui/theme/landscape_mdpi/splash.xml index d4f16fcd3..115d5bd90 100644 --- a/gui/theme/landscape_mdpi/splash.xml +++ b/gui/theme/landscape_mdpi/splash.xml @@ -1,51 +1,51 @@ -<?xml version="1.0"?>
-<recovery>
- <details>
- <resolution width="800" height="480"/>
- <author>TeamWin</author>
- <title>TWRP</title>
- <description>splash screen</description>
- <themeversion>1</themeversion>
- </details>
-
- <resources>
- <font name="font_l" filename="RobotoCondensed-Regular.ttf" size="24"/>
- <image name="splashlogo" filename="splashlogo" retainaspect="1"/>
- <image name="splashteamwin" filename="splashteamwin" retainaspect="1"/>
- </resources>
-
- <variables>
- <variable name="screen_width" value="800"/>
- <variable name="screen_height" value="480"/>
- <variable name="background_color" value="#222222"/>
- <variable name="header_color" value="#555555"/>
- <variable name="accent_color" value="#0090CA"/>
- </variables>
-
- <pages>
- <page name="splash">
- <background color="%background_color%"/>
-
- <fill color="%header_color%">
- <placement x="0" y="0" w="%screen_width%" h="140"/>
- </fill>
-
- <image>
- <image resource="splashlogo"/>
- <placement x="400" y="140" placement="4"/>
- </image>
-
- <image>
- <image resource="splashteamwin"/>
- <placement x="400" y="366" placement="4"/>
- </image>
-
- <text color="%header_color%">
- <font resource="font_l"/>
- <placement x="400" y="386" placement="5"/>
- <text>Recovery Project %tw_version%</text>
- </text>
- </page>
- </pages>
-</recovery>
-
+<?xml version="1.0"?> +<recovery> + <details> + <resolution width="800" height="480"/> + <author>TeamWin</author> + <title>TWRP</title> + <description>splash screen</description> + <themeversion>1</themeversion> + </details> + + <resources> + <font name="font_l" filename="RobotoCondensed-Regular.ttf" size="24"/> + <image name="splashlogo" filename="splashlogo" retainaspect="1"/> + <image name="splashteamwin" filename="splashteamwin" retainaspect="1"/> + </resources> + + <variables> + <variable name="screen_width" value="800"/> + <variable name="screen_height" value="480"/> + <variable name="background_color" value="#222222"/> + <variable name="header_color" value="#555555"/> + <variable name="accent_color" value="#0090CA"/> + </variables> + + <pages> + <page name="splash"> + <background color="%background_color%"/> + + <fill color="%header_color%"> + <placement x="0" y="0" w="%screen_width%" h="140"/> + </fill> + + <image> + <image resource="splashlogo"/> + <placement x="400" y="140" placement="4"/> + </image> + + <image> + <image resource="splashteamwin"/> + <placement x="400" y="366" placement="4"/> + </image> + + <text color="%header_color%"> + <font resource="font_l"/> + <placement x="400" y="386" placement="5"/> + <text>Recovery Project %tw_version%</text> + </text> + </page> + </pages> +</recovery> + diff --git a/gui/theme/portrait_mdpi/splash.xml b/gui/theme/portrait_mdpi/splash.xml index dad53cce9..e238ce15b 100644 --- a/gui/theme/portrait_mdpi/splash.xml +++ b/gui/theme/portrait_mdpi/splash.xml @@ -1,51 +1,51 @@ -<?xml version="1.0"?>
-<recovery>
- <details>
- <resolution width="480" height="800"/>
- <author>TeamWin</author>
- <title>TWRP</title>
- <description>splash screen</description>
- <themeversion>1</themeversion>
- </details>
-
- <resources>
- <font name="font_l" filename="RobotoCondensed-Regular.ttf" size="24"/>
- <image name="splashlogo" filename="splashlogo" retainaspect="1"/>
- <image name="splashteamwin" filename="splashteamwin" retainaspect="1"/>
- </resources>
-
- <variables>
- <variable name="screen_width" value="480"/>
- <variable name="screen_height" value="800"/>
- <variable name="background_color" value="#222222"/>
- <variable name="header_color" value="#555555"/>
- <variable name="accent_color" value="#0090CA"/>
- </variables>
-
- <pages>
- <page name="splash">
- <background color="%background_color%"/>
-
- <fill color="%header_color%">
- <placement x="0" y="0" w="%screen_width%" h="200"/>
- </fill>
-
- <image>
- <image resource="splashlogo"/>
- <placement x="240" y="200" placement="4"/>
- </image>
-
- <image>
- <image resource="splashteamwin"/>
- <placement x="240" y="660" placement="4"/>
- </image>
-
- <text color="%header_color%">
- <font resource="font_l"/>
- <placement x="240" y="680" placement="5"/>
- <text>Recovery Project %tw_version%</text>
- </text>
- </page>
- </pages>
-</recovery>
-
+<?xml version="1.0"?> +<recovery> + <details> + <resolution width="480" height="800"/> + <author>TeamWin</author> + <title>TWRP</title> + <description>splash screen</description> + <themeversion>1</themeversion> + </details> + + <resources> + <font name="font_l" filename="RobotoCondensed-Regular.ttf" size="24"/> + <image name="splashlogo" filename="splashlogo" retainaspect="1"/> + <image name="splashteamwin" filename="splashteamwin" retainaspect="1"/> + </resources> + + <variables> + <variable name="screen_width" value="480"/> + <variable name="screen_height" value="800"/> + <variable name="background_color" value="#222222"/> + <variable name="header_color" value="#555555"/> + <variable name="accent_color" value="#0090CA"/> + </variables> + + <pages> + <page name="splash"> + <background color="%background_color%"/> + + <fill color="%header_color%"> + <placement x="0" y="0" w="%screen_width%" h="200"/> + </fill> + + <image> + <image resource="splashlogo"/> + <placement x="240" y="200" placement="4"/> + </image> + + <image> + <image resource="splashteamwin"/> + <placement x="240" y="660" placement="4"/> + </image> + + <text color="%header_color%"> + <font resource="font_l"/> + <placement x="240" y="680" placement="5"/> + <text>Recovery Project %tw_version%</text> + </text> + </page> + </pages> +</recovery> + diff --git a/gui/theme/watch_mdpi/splash.xml b/gui/theme/watch_mdpi/splash.xml index ff9bc9859..2c2538a93 100644 --- a/gui/theme/watch_mdpi/splash.xml +++ b/gui/theme/watch_mdpi/splash.xml @@ -1,51 +1,51 @@ -<?xml version="1.0"?>
-<recovery>
- <details>
- <resolution width="320" height="320"/>
- <author>TeamWin</author>
- <title>TWRP</title>
- <description>splash screen</description>
- <themeversion>1</themeversion>
- </details>
-
- <resources>
- <font name="font_l" filename="RobotoCondensed-Regular.ttf" size="24"/>
- <image name="splashlogo" filename="splashlogo" retainaspect="1"/>
- <image name="splashteamwin" filename="splashteamwin" retainaspect="1"/>
- </resources>
-
- <variables>
- <variable name="screen_width" value="320"/>
- <variable name="screen_height" value="320"/>
- <variable name="background_color" value="#222222"/>
- <variable name="header_color" value="#555555"/>
- <variable name="accent_color" value="#0090CA"/>
- </variables>
-
- <pages>
- <page name="splash">
- <background color="%background_color%"/>
-
- <fill color="%header_color%">
- <placement x="0" y="0" w="%screen_width%" h="120"/>
- </fill>
-
- <image>
- <image resource="splashlogo"/>
- <placement x="160" y="120" placement="4"/>
- </image>
-
- <image>
- <image resource="splashteamwin"/>
- <placement x="160" y="270" placement="4"/>
- </image>
-
- <text color="%header_color%">
- <font resource="font_l"/>
- <placement x="160" y="290" placement="5"/>
- <text>Recovery Project %tw_version%</text>
- </text>
- </page>
- </pages>
-</recovery>
-
+<?xml version="1.0"?> +<recovery> + <details> + <resolution width="320" height="320"/> + <author>TeamWin</author> + <title>TWRP</title> + <description>splash screen</description> + <themeversion>1</themeversion> + </details> + + <resources> + <font name="font_l" filename="RobotoCondensed-Regular.ttf" size="24"/> + <image name="splashlogo" filename="splashlogo" retainaspect="1"/> + <image name="splashteamwin" filename="splashteamwin" retainaspect="1"/> + </resources> + + <variables> + <variable name="screen_width" value="320"/> + <variable name="screen_height" value="320"/> + <variable name="background_color" value="#222222"/> + <variable name="header_color" value="#555555"/> + <variable name="accent_color" value="#0090CA"/> + </variables> + + <pages> + <page name="splash"> + <background color="%background_color%"/> + + <fill color="%header_color%"> + <placement x="0" y="0" w="%screen_width%" h="120"/> + </fill> + + <image> + <image resource="splashlogo"/> + <placement x="160" y="120" placement="4"/> + </image> + + <image> + <image resource="splashteamwin"/> + <placement x="160" y="270" placement="4"/> + </image> + + <text color="%header_color%"> + <font resource="font_l"/> + <placement x="160" y="290" placement="5"/> + <text>Recovery Project %tw_version%</text> + </text> + </page> + </pages> +</recovery> + |