summaryrefslogtreecommitdiff
path: root/docs/samples/save_custom_writer.cpp
blob: 0c7669c0e57afb98ea0336b22a6037a980819089 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
#include "pugixml.hpp"

#include <string>

#include <stdio.h>

//[code_save_custom_writer
struct xml_string_writer: pugi::xml_writer
{
	std::string result;

	virtual void write(const void* data, size_t size)
	{
		result += std::string(static_cast<const char*>(data), size);
	}
};
//]

struct xml_memory_writer: pugi::xml_writer
{
	char* buffer;
	size_t capacity;

	size_t result;

	xml_memory_writer(): buffer(0), capacity(0), result(0)
	{
	}

	xml_memory_writer(char* buffer, size_t capacity): buffer(buffer), capacity(capacity), result(0)
	{
	}

	size_t written_size() const
	{
		return result < capacity ? result : capacity;
	}

	virtual void write(const void* data, size_t size)
	{
		if (result < capacity)
		{
			size_t chunk = (capacity - result < size) ? capacity - result : size;

			memcpy(buffer + result, data, chunk);
		}

		result += size;
	}
};

std::string node_to_string(pugi::xml_node node)
{
	xml_string_writer writer;
	node.print(writer);

	return writer.result;
}

char* node_to_buffer(pugi::xml_node node, char* buffer, size_t size)
{
	if (size == 0) return buffer;

	// leave one character for null terminator
	xml_memory_writer writer(buffer, size - 1);
	node.print(writer);

	// null terminate
	buffer[writer.written_size()] = 0;

	return buffer;
}

char* node_to_buffer_heap(pugi::xml_node node)
{
	// first pass: get required memory size
	xml_memory_writer counter;
	node.print(counter);

	// allocate necessary size (+1 for null termination)
	char* buffer = new char[counter.result + 1];

	// second pass: actual printing
	xml_memory_writer writer(buffer, counter.result);
	node.print(writer);

	// null terminate
	buffer[writer.written_size()] = 0;

	return buffer;
}

int main()
{
	// get a test document
	pugi::xml_document doc;
	doc.load("<foo bar='baz'>hey</foo>");

	// get contents as std::string (single pass)
	printf("contents: [%s]\n", node_to_string(doc).c_str());

	// get contents into fixed-size buffer (single pass)
	char large_buf[128];
	printf("contents: [%s]\n", node_to_buffer(doc, large_buf, sizeof(large_buf)));

	// get contents into fixed-size buffer (single pass, shows truncating behavior)
	char small_buf[22];
	printf("contents: [%s]\n", node_to_buffer(doc, small_buf, sizeof(small_buf)));

	// get contents into heap-allocated buffer (two passes)
	char* heap_buf = node_to_buffer_heap(doc);
	printf("contents: [%s]\n", heap_buf);
	delete[] heap_buf;
}