xref: /aosp_15_r20/external/cronet/third_party/libxml/src/HTMLtree.c (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker /*
2*6777b538SAndroid Build Coastguard Worker  * HTMLtree.c : implementation of access function for an HTML tree.
3*6777b538SAndroid Build Coastguard Worker  *
4*6777b538SAndroid Build Coastguard Worker  * See Copyright for the status of this software.
5*6777b538SAndroid Build Coastguard Worker  *
6*6777b538SAndroid Build Coastguard Worker  * [email protected]
7*6777b538SAndroid Build Coastguard Worker  */
8*6777b538SAndroid Build Coastguard Worker 
9*6777b538SAndroid Build Coastguard Worker 
10*6777b538SAndroid Build Coastguard Worker #define IN_LIBXML
11*6777b538SAndroid Build Coastguard Worker #include "libxml.h"
12*6777b538SAndroid Build Coastguard Worker #ifdef LIBXML_HTML_ENABLED
13*6777b538SAndroid Build Coastguard Worker 
14*6777b538SAndroid Build Coastguard Worker #include <string.h> /* for memset() only ! */
15*6777b538SAndroid Build Coastguard Worker #include <ctype.h>
16*6777b538SAndroid Build Coastguard Worker #include <stdlib.h>
17*6777b538SAndroid Build Coastguard Worker 
18*6777b538SAndroid Build Coastguard Worker #include <libxml/xmlmemory.h>
19*6777b538SAndroid Build Coastguard Worker #include <libxml/HTMLparser.h>
20*6777b538SAndroid Build Coastguard Worker #include <libxml/HTMLtree.h>
21*6777b538SAndroid Build Coastguard Worker #include <libxml/entities.h>
22*6777b538SAndroid Build Coastguard Worker #include <libxml/xmlerror.h>
23*6777b538SAndroid Build Coastguard Worker #include <libxml/parserInternals.h>
24*6777b538SAndroid Build Coastguard Worker #include <libxml/uri.h>
25*6777b538SAndroid Build Coastguard Worker 
26*6777b538SAndroid Build Coastguard Worker #include "private/buf.h"
27*6777b538SAndroid Build Coastguard Worker #include "private/error.h"
28*6777b538SAndroid Build Coastguard Worker #include "private/io.h"
29*6777b538SAndroid Build Coastguard Worker #include "private/save.h"
30*6777b538SAndroid Build Coastguard Worker 
31*6777b538SAndroid Build Coastguard Worker /************************************************************************
32*6777b538SAndroid Build Coastguard Worker  *									*
33*6777b538SAndroid Build Coastguard Worker  *		Getting/Setting encoding meta tags			*
34*6777b538SAndroid Build Coastguard Worker  *									*
35*6777b538SAndroid Build Coastguard Worker  ************************************************************************/
36*6777b538SAndroid Build Coastguard Worker 
37*6777b538SAndroid Build Coastguard Worker /**
38*6777b538SAndroid Build Coastguard Worker  * htmlGetMetaEncoding:
39*6777b538SAndroid Build Coastguard Worker  * @doc:  the document
40*6777b538SAndroid Build Coastguard Worker  *
41*6777b538SAndroid Build Coastguard Worker  * Encoding definition lookup in the Meta tags
42*6777b538SAndroid Build Coastguard Worker  *
43*6777b538SAndroid Build Coastguard Worker  * Returns the current encoding as flagged in the HTML source
44*6777b538SAndroid Build Coastguard Worker  */
45*6777b538SAndroid Build Coastguard Worker const xmlChar *
htmlGetMetaEncoding(htmlDocPtr doc)46*6777b538SAndroid Build Coastguard Worker htmlGetMetaEncoding(htmlDocPtr doc) {
47*6777b538SAndroid Build Coastguard Worker     htmlNodePtr cur;
48*6777b538SAndroid Build Coastguard Worker     const xmlChar *content;
49*6777b538SAndroid Build Coastguard Worker     const xmlChar *encoding;
50*6777b538SAndroid Build Coastguard Worker 
51*6777b538SAndroid Build Coastguard Worker     if (doc == NULL)
52*6777b538SAndroid Build Coastguard Worker 	return(NULL);
53*6777b538SAndroid Build Coastguard Worker     cur = doc->children;
54*6777b538SAndroid Build Coastguard Worker 
55*6777b538SAndroid Build Coastguard Worker     /*
56*6777b538SAndroid Build Coastguard Worker      * Search the html
57*6777b538SAndroid Build Coastguard Worker      */
58*6777b538SAndroid Build Coastguard Worker     while (cur != NULL) {
59*6777b538SAndroid Build Coastguard Worker 	if ((cur->type == XML_ELEMENT_NODE) && (cur->name != NULL)) {
60*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrEqual(cur->name, BAD_CAST"html"))
61*6777b538SAndroid Build Coastguard Worker 		break;
62*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrEqual(cur->name, BAD_CAST"head"))
63*6777b538SAndroid Build Coastguard Worker 		goto found_head;
64*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrEqual(cur->name, BAD_CAST"meta"))
65*6777b538SAndroid Build Coastguard Worker 		goto found_meta;
66*6777b538SAndroid Build Coastguard Worker 	}
67*6777b538SAndroid Build Coastguard Worker 	cur = cur->next;
68*6777b538SAndroid Build Coastguard Worker     }
69*6777b538SAndroid Build Coastguard Worker     if (cur == NULL)
70*6777b538SAndroid Build Coastguard Worker 	return(NULL);
71*6777b538SAndroid Build Coastguard Worker     cur = cur->children;
72*6777b538SAndroid Build Coastguard Worker 
73*6777b538SAndroid Build Coastguard Worker     /*
74*6777b538SAndroid Build Coastguard Worker      * Search the head
75*6777b538SAndroid Build Coastguard Worker      */
76*6777b538SAndroid Build Coastguard Worker     while (cur != NULL) {
77*6777b538SAndroid Build Coastguard Worker 	if ((cur->type == XML_ELEMENT_NODE) && (cur->name != NULL)) {
78*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrEqual(cur->name, BAD_CAST"head"))
79*6777b538SAndroid Build Coastguard Worker 		break;
80*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrEqual(cur->name, BAD_CAST"meta"))
81*6777b538SAndroid Build Coastguard Worker 		goto found_meta;
82*6777b538SAndroid Build Coastguard Worker 	}
83*6777b538SAndroid Build Coastguard Worker 	cur = cur->next;
84*6777b538SAndroid Build Coastguard Worker     }
85*6777b538SAndroid Build Coastguard Worker     if (cur == NULL)
86*6777b538SAndroid Build Coastguard Worker 	return(NULL);
87*6777b538SAndroid Build Coastguard Worker found_head:
88*6777b538SAndroid Build Coastguard Worker     cur = cur->children;
89*6777b538SAndroid Build Coastguard Worker 
90*6777b538SAndroid Build Coastguard Worker     /*
91*6777b538SAndroid Build Coastguard Worker      * Search the meta elements
92*6777b538SAndroid Build Coastguard Worker      */
93*6777b538SAndroid Build Coastguard Worker found_meta:
94*6777b538SAndroid Build Coastguard Worker     while (cur != NULL) {
95*6777b538SAndroid Build Coastguard Worker 	if ((cur->type == XML_ELEMENT_NODE) && (cur->name != NULL)) {
96*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrEqual(cur->name, BAD_CAST"meta")) {
97*6777b538SAndroid Build Coastguard Worker 		xmlAttrPtr attr = cur->properties;
98*6777b538SAndroid Build Coastguard Worker 		int http;
99*6777b538SAndroid Build Coastguard Worker 		const xmlChar *value;
100*6777b538SAndroid Build Coastguard Worker 
101*6777b538SAndroid Build Coastguard Worker 		content = NULL;
102*6777b538SAndroid Build Coastguard Worker 		http = 0;
103*6777b538SAndroid Build Coastguard Worker 		while (attr != NULL) {
104*6777b538SAndroid Build Coastguard Worker 		    if ((attr->children != NULL) &&
105*6777b538SAndroid Build Coastguard Worker 		        (attr->children->type == XML_TEXT_NODE) &&
106*6777b538SAndroid Build Coastguard Worker 		        (attr->children->next == NULL)) {
107*6777b538SAndroid Build Coastguard Worker 			value = attr->children->content;
108*6777b538SAndroid Build Coastguard Worker 			if ((!xmlStrcasecmp(attr->name, BAD_CAST"http-equiv"))
109*6777b538SAndroid Build Coastguard Worker 			 && (!xmlStrcasecmp(value, BAD_CAST"Content-Type")))
110*6777b538SAndroid Build Coastguard Worker 			    http = 1;
111*6777b538SAndroid Build Coastguard Worker 			else if ((value != NULL)
112*6777b538SAndroid Build Coastguard Worker 			 && (!xmlStrcasecmp(attr->name, BAD_CAST"content")))
113*6777b538SAndroid Build Coastguard Worker 			    content = value;
114*6777b538SAndroid Build Coastguard Worker 			if ((http != 0) && (content != NULL))
115*6777b538SAndroid Build Coastguard Worker 			    goto found_content;
116*6777b538SAndroid Build Coastguard Worker 		    }
117*6777b538SAndroid Build Coastguard Worker 		    attr = attr->next;
118*6777b538SAndroid Build Coastguard Worker 		}
119*6777b538SAndroid Build Coastguard Worker 	    }
120*6777b538SAndroid Build Coastguard Worker 	}
121*6777b538SAndroid Build Coastguard Worker 	cur = cur->next;
122*6777b538SAndroid Build Coastguard Worker     }
123*6777b538SAndroid Build Coastguard Worker     return(NULL);
124*6777b538SAndroid Build Coastguard Worker 
125*6777b538SAndroid Build Coastguard Worker found_content:
126*6777b538SAndroid Build Coastguard Worker     encoding = xmlStrstr(content, BAD_CAST"charset=");
127*6777b538SAndroid Build Coastguard Worker     if (encoding == NULL)
128*6777b538SAndroid Build Coastguard Worker 	encoding = xmlStrstr(content, BAD_CAST"Charset=");
129*6777b538SAndroid Build Coastguard Worker     if (encoding == NULL)
130*6777b538SAndroid Build Coastguard Worker 	encoding = xmlStrstr(content, BAD_CAST"CHARSET=");
131*6777b538SAndroid Build Coastguard Worker     if (encoding != NULL) {
132*6777b538SAndroid Build Coastguard Worker 	encoding += 8;
133*6777b538SAndroid Build Coastguard Worker     } else {
134*6777b538SAndroid Build Coastguard Worker 	encoding = xmlStrstr(content, BAD_CAST"charset =");
135*6777b538SAndroid Build Coastguard Worker 	if (encoding == NULL)
136*6777b538SAndroid Build Coastguard Worker 	    encoding = xmlStrstr(content, BAD_CAST"Charset =");
137*6777b538SAndroid Build Coastguard Worker 	if (encoding == NULL)
138*6777b538SAndroid Build Coastguard Worker 	    encoding = xmlStrstr(content, BAD_CAST"CHARSET =");
139*6777b538SAndroid Build Coastguard Worker 	if (encoding != NULL)
140*6777b538SAndroid Build Coastguard Worker 	    encoding += 9;
141*6777b538SAndroid Build Coastguard Worker     }
142*6777b538SAndroid Build Coastguard Worker     if (encoding != NULL) {
143*6777b538SAndroid Build Coastguard Worker 	while ((*encoding == ' ') || (*encoding == '\t')) encoding++;
144*6777b538SAndroid Build Coastguard Worker     }
145*6777b538SAndroid Build Coastguard Worker     return(encoding);
146*6777b538SAndroid Build Coastguard Worker }
147*6777b538SAndroid Build Coastguard Worker 
148*6777b538SAndroid Build Coastguard Worker /**
149*6777b538SAndroid Build Coastguard Worker  * htmlSetMetaEncoding:
150*6777b538SAndroid Build Coastguard Worker  * @doc:  the document
151*6777b538SAndroid Build Coastguard Worker  * @encoding:  the encoding string
152*6777b538SAndroid Build Coastguard Worker  *
153*6777b538SAndroid Build Coastguard Worker  * Sets the current encoding in the Meta tags
154*6777b538SAndroid Build Coastguard Worker  * NOTE: this will not change the document content encoding, just
155*6777b538SAndroid Build Coastguard Worker  * the META flag associated.
156*6777b538SAndroid Build Coastguard Worker  *
157*6777b538SAndroid Build Coastguard Worker  * Returns 0 in case of success and -1 in case of error
158*6777b538SAndroid Build Coastguard Worker  */
159*6777b538SAndroid Build Coastguard Worker int
htmlSetMetaEncoding(htmlDocPtr doc,const xmlChar * encoding)160*6777b538SAndroid Build Coastguard Worker htmlSetMetaEncoding(htmlDocPtr doc, const xmlChar *encoding) {
161*6777b538SAndroid Build Coastguard Worker     htmlNodePtr cur, meta = NULL, head = NULL;
162*6777b538SAndroid Build Coastguard Worker     const xmlChar *content = NULL;
163*6777b538SAndroid Build Coastguard Worker     char newcontent[100];
164*6777b538SAndroid Build Coastguard Worker 
165*6777b538SAndroid Build Coastguard Worker     newcontent[0] = 0;
166*6777b538SAndroid Build Coastguard Worker 
167*6777b538SAndroid Build Coastguard Worker     if (doc == NULL)
168*6777b538SAndroid Build Coastguard Worker 	return(-1);
169*6777b538SAndroid Build Coastguard Worker 
170*6777b538SAndroid Build Coastguard Worker     /* html isn't a real encoding it's just libxml2 way to get entities */
171*6777b538SAndroid Build Coastguard Worker     if (!xmlStrcasecmp(encoding, BAD_CAST "html"))
172*6777b538SAndroid Build Coastguard Worker         return(-1);
173*6777b538SAndroid Build Coastguard Worker 
174*6777b538SAndroid Build Coastguard Worker     if (encoding != NULL) {
175*6777b538SAndroid Build Coastguard Worker 	snprintf(newcontent, sizeof(newcontent), "text/html; charset=%s",
176*6777b538SAndroid Build Coastguard Worker                 (char *)encoding);
177*6777b538SAndroid Build Coastguard Worker 	newcontent[sizeof(newcontent) - 1] = 0;
178*6777b538SAndroid Build Coastguard Worker     }
179*6777b538SAndroid Build Coastguard Worker 
180*6777b538SAndroid Build Coastguard Worker     cur = doc->children;
181*6777b538SAndroid Build Coastguard Worker 
182*6777b538SAndroid Build Coastguard Worker     /*
183*6777b538SAndroid Build Coastguard Worker      * Search the html
184*6777b538SAndroid Build Coastguard Worker      */
185*6777b538SAndroid Build Coastguard Worker     while (cur != NULL) {
186*6777b538SAndroid Build Coastguard Worker 	if ((cur->type == XML_ELEMENT_NODE) && (cur->name != NULL)) {
187*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrcasecmp(cur->name, BAD_CAST"html") == 0)
188*6777b538SAndroid Build Coastguard Worker 		break;
189*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrcasecmp(cur->name, BAD_CAST"head") == 0)
190*6777b538SAndroid Build Coastguard Worker 		goto found_head;
191*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrcasecmp(cur->name, BAD_CAST"meta") == 0)
192*6777b538SAndroid Build Coastguard Worker 		goto found_meta;
193*6777b538SAndroid Build Coastguard Worker 	}
194*6777b538SAndroid Build Coastguard Worker 	cur = cur->next;
195*6777b538SAndroid Build Coastguard Worker     }
196*6777b538SAndroid Build Coastguard Worker     if (cur == NULL)
197*6777b538SAndroid Build Coastguard Worker 	return(-1);
198*6777b538SAndroid Build Coastguard Worker     cur = cur->children;
199*6777b538SAndroid Build Coastguard Worker 
200*6777b538SAndroid Build Coastguard Worker     /*
201*6777b538SAndroid Build Coastguard Worker      * Search the head
202*6777b538SAndroid Build Coastguard Worker      */
203*6777b538SAndroid Build Coastguard Worker     while (cur != NULL) {
204*6777b538SAndroid Build Coastguard Worker 	if ((cur->type == XML_ELEMENT_NODE) && (cur->name != NULL)) {
205*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrcasecmp(cur->name, BAD_CAST"head") == 0)
206*6777b538SAndroid Build Coastguard Worker 		break;
207*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrcasecmp(cur->name, BAD_CAST"meta") == 0) {
208*6777b538SAndroid Build Coastguard Worker                 head = cur->parent;
209*6777b538SAndroid Build Coastguard Worker 		goto found_meta;
210*6777b538SAndroid Build Coastguard Worker             }
211*6777b538SAndroid Build Coastguard Worker 	}
212*6777b538SAndroid Build Coastguard Worker 	cur = cur->next;
213*6777b538SAndroid Build Coastguard Worker     }
214*6777b538SAndroid Build Coastguard Worker     if (cur == NULL)
215*6777b538SAndroid Build Coastguard Worker 	return(-1);
216*6777b538SAndroid Build Coastguard Worker found_head:
217*6777b538SAndroid Build Coastguard Worker     head = cur;
218*6777b538SAndroid Build Coastguard Worker     if (cur->children == NULL)
219*6777b538SAndroid Build Coastguard Worker         goto create;
220*6777b538SAndroid Build Coastguard Worker     cur = cur->children;
221*6777b538SAndroid Build Coastguard Worker 
222*6777b538SAndroid Build Coastguard Worker found_meta:
223*6777b538SAndroid Build Coastguard Worker     /*
224*6777b538SAndroid Build Coastguard Worker      * Search and update all the remaining the meta elements carrying
225*6777b538SAndroid Build Coastguard Worker      * encoding information
226*6777b538SAndroid Build Coastguard Worker      */
227*6777b538SAndroid Build Coastguard Worker     while (cur != NULL) {
228*6777b538SAndroid Build Coastguard Worker 	if ((cur->type == XML_ELEMENT_NODE) && (cur->name != NULL)) {
229*6777b538SAndroid Build Coastguard Worker 	    if (xmlStrcasecmp(cur->name, BAD_CAST"meta") == 0) {
230*6777b538SAndroid Build Coastguard Worker 		xmlAttrPtr attr = cur->properties;
231*6777b538SAndroid Build Coastguard Worker 		int http;
232*6777b538SAndroid Build Coastguard Worker 		const xmlChar *value;
233*6777b538SAndroid Build Coastguard Worker 
234*6777b538SAndroid Build Coastguard Worker 		content = NULL;
235*6777b538SAndroid Build Coastguard Worker 		http = 0;
236*6777b538SAndroid Build Coastguard Worker 		while (attr != NULL) {
237*6777b538SAndroid Build Coastguard Worker 		    if ((attr->children != NULL) &&
238*6777b538SAndroid Build Coastguard Worker 		        (attr->children->type == XML_TEXT_NODE) &&
239*6777b538SAndroid Build Coastguard Worker 		        (attr->children->next == NULL)) {
240*6777b538SAndroid Build Coastguard Worker 			value = attr->children->content;
241*6777b538SAndroid Build Coastguard Worker 			if ((!xmlStrcasecmp(attr->name, BAD_CAST"http-equiv"))
242*6777b538SAndroid Build Coastguard Worker 			 && (!xmlStrcasecmp(value, BAD_CAST"Content-Type")))
243*6777b538SAndroid Build Coastguard Worker 			    http = 1;
244*6777b538SAndroid Build Coastguard Worker 			else
245*6777b538SAndroid Build Coastguard Worker                         {
246*6777b538SAndroid Build Coastguard Worker                            if ((value != NULL) &&
247*6777b538SAndroid Build Coastguard Worker                                (!xmlStrcasecmp(attr->name, BAD_CAST"content")))
248*6777b538SAndroid Build Coastguard Worker 			       content = value;
249*6777b538SAndroid Build Coastguard Worker                         }
250*6777b538SAndroid Build Coastguard Worker 		        if ((http != 0) && (content != NULL))
251*6777b538SAndroid Build Coastguard Worker 			    break;
252*6777b538SAndroid Build Coastguard Worker 		    }
253*6777b538SAndroid Build Coastguard Worker 		    attr = attr->next;
254*6777b538SAndroid Build Coastguard Worker 		}
255*6777b538SAndroid Build Coastguard Worker 		if ((http != 0) && (content != NULL)) {
256*6777b538SAndroid Build Coastguard Worker 		    meta = cur;
257*6777b538SAndroid Build Coastguard Worker 		    break;
258*6777b538SAndroid Build Coastguard Worker 		}
259*6777b538SAndroid Build Coastguard Worker 
260*6777b538SAndroid Build Coastguard Worker 	    }
261*6777b538SAndroid Build Coastguard Worker 	}
262*6777b538SAndroid Build Coastguard Worker 	cur = cur->next;
263*6777b538SAndroid Build Coastguard Worker     }
264*6777b538SAndroid Build Coastguard Worker create:
265*6777b538SAndroid Build Coastguard Worker     if (meta == NULL) {
266*6777b538SAndroid Build Coastguard Worker         if ((encoding != NULL) && (head != NULL)) {
267*6777b538SAndroid Build Coastguard Worker             /*
268*6777b538SAndroid Build Coastguard Worker              * Create a new Meta element with the right attributes
269*6777b538SAndroid Build Coastguard Worker              */
270*6777b538SAndroid Build Coastguard Worker 
271*6777b538SAndroid Build Coastguard Worker             meta = xmlNewDocNode(doc, NULL, BAD_CAST"meta", NULL);
272*6777b538SAndroid Build Coastguard Worker             if (head->children == NULL)
273*6777b538SAndroid Build Coastguard Worker                 xmlAddChild(head, meta);
274*6777b538SAndroid Build Coastguard Worker             else
275*6777b538SAndroid Build Coastguard Worker                 xmlAddPrevSibling(head->children, meta);
276*6777b538SAndroid Build Coastguard Worker             xmlNewProp(meta, BAD_CAST"http-equiv", BAD_CAST"Content-Type");
277*6777b538SAndroid Build Coastguard Worker             xmlNewProp(meta, BAD_CAST"content", BAD_CAST newcontent);
278*6777b538SAndroid Build Coastguard Worker         }
279*6777b538SAndroid Build Coastguard Worker     } else {
280*6777b538SAndroid Build Coastguard Worker         /* remove the meta tag if NULL is passed */
281*6777b538SAndroid Build Coastguard Worker         if (encoding == NULL) {
282*6777b538SAndroid Build Coastguard Worker             xmlUnlinkNode(meta);
283*6777b538SAndroid Build Coastguard Worker             xmlFreeNode(meta);
284*6777b538SAndroid Build Coastguard Worker         }
285*6777b538SAndroid Build Coastguard Worker         /* change the document only if there is a real encoding change */
286*6777b538SAndroid Build Coastguard Worker         else if (xmlStrcasestr(content, encoding) == NULL) {
287*6777b538SAndroid Build Coastguard Worker             xmlSetProp(meta, BAD_CAST"content", BAD_CAST newcontent);
288*6777b538SAndroid Build Coastguard Worker         }
289*6777b538SAndroid Build Coastguard Worker     }
290*6777b538SAndroid Build Coastguard Worker 
291*6777b538SAndroid Build Coastguard Worker 
292*6777b538SAndroid Build Coastguard Worker     return(0);
293*6777b538SAndroid Build Coastguard Worker }
294*6777b538SAndroid Build Coastguard Worker 
295*6777b538SAndroid Build Coastguard Worker /**
296*6777b538SAndroid Build Coastguard Worker  * booleanHTMLAttrs:
297*6777b538SAndroid Build Coastguard Worker  *
298*6777b538SAndroid Build Coastguard Worker  * These are the HTML attributes which will be output
299*6777b538SAndroid Build Coastguard Worker  * in minimized form, i.e. <option selected="selected"> will be
300*6777b538SAndroid Build Coastguard Worker  * output as <option selected>, as per XSLT 1.0 16.2 "HTML Output Method"
301*6777b538SAndroid Build Coastguard Worker  *
302*6777b538SAndroid Build Coastguard Worker  */
303*6777b538SAndroid Build Coastguard Worker static const char* const htmlBooleanAttrs[] = {
304*6777b538SAndroid Build Coastguard Worker   "checked", "compact", "declare", "defer", "disabled", "ismap",
305*6777b538SAndroid Build Coastguard Worker   "multiple", "nohref", "noresize", "noshade", "nowrap", "readonly",
306*6777b538SAndroid Build Coastguard Worker   "selected", NULL
307*6777b538SAndroid Build Coastguard Worker };
308*6777b538SAndroid Build Coastguard Worker 
309*6777b538SAndroid Build Coastguard Worker 
310*6777b538SAndroid Build Coastguard Worker /**
311*6777b538SAndroid Build Coastguard Worker  * htmlIsBooleanAttr:
312*6777b538SAndroid Build Coastguard Worker  * @name:  the name of the attribute to check
313*6777b538SAndroid Build Coastguard Worker  *
314*6777b538SAndroid Build Coastguard Worker  * Determine if a given attribute is a boolean attribute.
315*6777b538SAndroid Build Coastguard Worker  *
316*6777b538SAndroid Build Coastguard Worker  * returns: false if the attribute is not boolean, true otherwise.
317*6777b538SAndroid Build Coastguard Worker  */
318*6777b538SAndroid Build Coastguard Worker int
htmlIsBooleanAttr(const xmlChar * name)319*6777b538SAndroid Build Coastguard Worker htmlIsBooleanAttr(const xmlChar *name)
320*6777b538SAndroid Build Coastguard Worker {
321*6777b538SAndroid Build Coastguard Worker     int i = 0;
322*6777b538SAndroid Build Coastguard Worker 
323*6777b538SAndroid Build Coastguard Worker     while (htmlBooleanAttrs[i] != NULL) {
324*6777b538SAndroid Build Coastguard Worker         if (xmlStrcasecmp((const xmlChar *)htmlBooleanAttrs[i], name) == 0)
325*6777b538SAndroid Build Coastguard Worker             return 1;
326*6777b538SAndroid Build Coastguard Worker         i++;
327*6777b538SAndroid Build Coastguard Worker     }
328*6777b538SAndroid Build Coastguard Worker     return 0;
329*6777b538SAndroid Build Coastguard Worker }
330*6777b538SAndroid Build Coastguard Worker 
331*6777b538SAndroid Build Coastguard Worker #ifdef LIBXML_OUTPUT_ENABLED
332*6777b538SAndroid Build Coastguard Worker /************************************************************************
333*6777b538SAndroid Build Coastguard Worker  *									*
334*6777b538SAndroid Build Coastguard Worker  *			Output error handlers				*
335*6777b538SAndroid Build Coastguard Worker  *									*
336*6777b538SAndroid Build Coastguard Worker  ************************************************************************/
337*6777b538SAndroid Build Coastguard Worker 
338*6777b538SAndroid Build Coastguard Worker /**
339*6777b538SAndroid Build Coastguard Worker  * htmlSaveErr:
340*6777b538SAndroid Build Coastguard Worker  * @code:  the error number
341*6777b538SAndroid Build Coastguard Worker  * @node:  the location of the error.
342*6777b538SAndroid Build Coastguard Worker  * @extra:  extra information
343*6777b538SAndroid Build Coastguard Worker  *
344*6777b538SAndroid Build Coastguard Worker  * Handle an out of memory condition
345*6777b538SAndroid Build Coastguard Worker  */
346*6777b538SAndroid Build Coastguard Worker static void
htmlSaveErr(int code,xmlNodePtr node,const char * extra)347*6777b538SAndroid Build Coastguard Worker htmlSaveErr(int code, xmlNodePtr node, const char *extra)
348*6777b538SAndroid Build Coastguard Worker {
349*6777b538SAndroid Build Coastguard Worker     const char *msg = NULL;
350*6777b538SAndroid Build Coastguard Worker     int res;
351*6777b538SAndroid Build Coastguard Worker 
352*6777b538SAndroid Build Coastguard Worker     switch(code) {
353*6777b538SAndroid Build Coastguard Worker         case XML_SAVE_NOT_UTF8:
354*6777b538SAndroid Build Coastguard Worker 	    msg = "string is not in UTF-8\n";
355*6777b538SAndroid Build Coastguard Worker 	    break;
356*6777b538SAndroid Build Coastguard Worker 	case XML_SAVE_CHAR_INVALID:
357*6777b538SAndroid Build Coastguard Worker 	    msg = "invalid character value\n";
358*6777b538SAndroid Build Coastguard Worker 	    break;
359*6777b538SAndroid Build Coastguard Worker 	case XML_SAVE_UNKNOWN_ENCODING:
360*6777b538SAndroid Build Coastguard Worker 	    msg = "unknown encoding %s\n";
361*6777b538SAndroid Build Coastguard Worker 	    break;
362*6777b538SAndroid Build Coastguard Worker 	case XML_SAVE_NO_DOCTYPE:
363*6777b538SAndroid Build Coastguard Worker 	    msg = "HTML has no DOCTYPE\n";
364*6777b538SAndroid Build Coastguard Worker 	    break;
365*6777b538SAndroid Build Coastguard Worker 	default:
366*6777b538SAndroid Build Coastguard Worker 	    msg = "unexpected error number\n";
367*6777b538SAndroid Build Coastguard Worker     }
368*6777b538SAndroid Build Coastguard Worker 
369*6777b538SAndroid Build Coastguard Worker     res = __xmlRaiseError(NULL, NULL, NULL, NULL, node,
370*6777b538SAndroid Build Coastguard Worker                           XML_FROM_OUTPUT, code, XML_ERR_ERROR, NULL, 0,
371*6777b538SAndroid Build Coastguard Worker                           extra, NULL, NULL, 0, 0,
372*6777b538SAndroid Build Coastguard Worker                           msg, extra);
373*6777b538SAndroid Build Coastguard Worker     if (res < 0)
374*6777b538SAndroid Build Coastguard Worker         xmlRaiseMemoryError(NULL, NULL, NULL, XML_FROM_OUTPUT, NULL);
375*6777b538SAndroid Build Coastguard Worker }
376*6777b538SAndroid Build Coastguard Worker 
377*6777b538SAndroid Build Coastguard Worker /************************************************************************
378*6777b538SAndroid Build Coastguard Worker  *									*
379*6777b538SAndroid Build Coastguard Worker  *		Dumping HTML tree content to a simple buffer		*
380*6777b538SAndroid Build Coastguard Worker  *									*
381*6777b538SAndroid Build Coastguard Worker  ************************************************************************/
382*6777b538SAndroid Build Coastguard Worker 
383*6777b538SAndroid Build Coastguard Worker static xmlCharEncodingHandler *
htmlFindOutputEncoder(const char * encoding)384*6777b538SAndroid Build Coastguard Worker htmlFindOutputEncoder(const char *encoding) {
385*6777b538SAndroid Build Coastguard Worker     xmlCharEncodingHandler *handler = NULL;
386*6777b538SAndroid Build Coastguard Worker 
387*6777b538SAndroid Build Coastguard Worker     if (encoding != NULL) {
388*6777b538SAndroid Build Coastguard Worker 	xmlCharEncoding enc;
389*6777b538SAndroid Build Coastguard Worker 
390*6777b538SAndroid Build Coastguard Worker 	enc = xmlParseCharEncoding(encoding);
391*6777b538SAndroid Build Coastguard Worker 	if (enc != XML_CHAR_ENCODING_UTF8) {
392*6777b538SAndroid Build Coastguard Worker 	    xmlOpenCharEncodingHandler(encoding, /* output */ 1, &handler);
393*6777b538SAndroid Build Coastguard Worker 	    if (handler == NULL)
394*6777b538SAndroid Build Coastguard Worker 		htmlSaveErr(XML_SAVE_UNKNOWN_ENCODING, NULL, encoding);
395*6777b538SAndroid Build Coastguard Worker 	}
396*6777b538SAndroid Build Coastguard Worker     } else {
397*6777b538SAndroid Build Coastguard Worker         /*
398*6777b538SAndroid Build Coastguard Worker          * Fallback to HTML or ASCII when the encoding is unspecified
399*6777b538SAndroid Build Coastguard Worker          */
400*6777b538SAndroid Build Coastguard Worker         if (handler == NULL)
401*6777b538SAndroid Build Coastguard Worker             xmlOpenCharEncodingHandler("HTML", /* output */ 1, &handler);
402*6777b538SAndroid Build Coastguard Worker         if (handler == NULL)
403*6777b538SAndroid Build Coastguard Worker             xmlOpenCharEncodingHandler("ascii", /* output */ 1, &handler);
404*6777b538SAndroid Build Coastguard Worker     }
405*6777b538SAndroid Build Coastguard Worker 
406*6777b538SAndroid Build Coastguard Worker     return(handler);
407*6777b538SAndroid Build Coastguard Worker }
408*6777b538SAndroid Build Coastguard Worker 
409*6777b538SAndroid Build Coastguard Worker /**
410*6777b538SAndroid Build Coastguard Worker  * htmlBufNodeDumpFormat:
411*6777b538SAndroid Build Coastguard Worker  * @buf:  the xmlBufPtr output
412*6777b538SAndroid Build Coastguard Worker  * @doc:  the document
413*6777b538SAndroid Build Coastguard Worker  * @cur:  the current node
414*6777b538SAndroid Build Coastguard Worker  * @format:  should formatting spaces been added
415*6777b538SAndroid Build Coastguard Worker  *
416*6777b538SAndroid Build Coastguard Worker  * Dump an HTML node, recursive behaviour,children are printed too.
417*6777b538SAndroid Build Coastguard Worker  *
418*6777b538SAndroid Build Coastguard Worker  * Returns the number of byte written or -1 in case of error
419*6777b538SAndroid Build Coastguard Worker  */
420*6777b538SAndroid Build Coastguard Worker static size_t
htmlBufNodeDumpFormat(xmlBufPtr buf,xmlDocPtr doc,xmlNodePtr cur,int format)421*6777b538SAndroid Build Coastguard Worker htmlBufNodeDumpFormat(xmlBufPtr buf, xmlDocPtr doc, xmlNodePtr cur,
422*6777b538SAndroid Build Coastguard Worker 	           int format) {
423*6777b538SAndroid Build Coastguard Worker     size_t use;
424*6777b538SAndroid Build Coastguard Worker     int ret;
425*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferPtr outbuf;
426*6777b538SAndroid Build Coastguard Worker 
427*6777b538SAndroid Build Coastguard Worker     if (cur == NULL) {
428*6777b538SAndroid Build Coastguard Worker 	return (-1);
429*6777b538SAndroid Build Coastguard Worker     }
430*6777b538SAndroid Build Coastguard Worker     if (buf == NULL) {
431*6777b538SAndroid Build Coastguard Worker 	return (-1);
432*6777b538SAndroid Build Coastguard Worker     }
433*6777b538SAndroid Build Coastguard Worker     outbuf = (xmlOutputBufferPtr) xmlMalloc(sizeof(xmlOutputBuffer));
434*6777b538SAndroid Build Coastguard Worker     if (outbuf == NULL)
435*6777b538SAndroid Build Coastguard Worker 	return (-1);
436*6777b538SAndroid Build Coastguard Worker     memset(outbuf, 0, sizeof(xmlOutputBuffer));
437*6777b538SAndroid Build Coastguard Worker     outbuf->buffer = buf;
438*6777b538SAndroid Build Coastguard Worker     outbuf->encoder = NULL;
439*6777b538SAndroid Build Coastguard Worker     outbuf->writecallback = NULL;
440*6777b538SAndroid Build Coastguard Worker     outbuf->closecallback = NULL;
441*6777b538SAndroid Build Coastguard Worker     outbuf->context = NULL;
442*6777b538SAndroid Build Coastguard Worker     outbuf->written = 0;
443*6777b538SAndroid Build Coastguard Worker 
444*6777b538SAndroid Build Coastguard Worker     use = xmlBufUse(buf);
445*6777b538SAndroid Build Coastguard Worker     htmlNodeDumpFormatOutput(outbuf, doc, cur, NULL, format);
446*6777b538SAndroid Build Coastguard Worker     xmlFree(outbuf);
447*6777b538SAndroid Build Coastguard Worker     ret = xmlBufUse(buf) - use;
448*6777b538SAndroid Build Coastguard Worker     return (ret);
449*6777b538SAndroid Build Coastguard Worker }
450*6777b538SAndroid Build Coastguard Worker 
451*6777b538SAndroid Build Coastguard Worker /**
452*6777b538SAndroid Build Coastguard Worker  * htmlNodeDump:
453*6777b538SAndroid Build Coastguard Worker  * @buf:  the HTML buffer output
454*6777b538SAndroid Build Coastguard Worker  * @doc:  the document
455*6777b538SAndroid Build Coastguard Worker  * @cur:  the current node
456*6777b538SAndroid Build Coastguard Worker  *
457*6777b538SAndroid Build Coastguard Worker  * Dump an HTML node, recursive behaviour,children are printed too,
458*6777b538SAndroid Build Coastguard Worker  * and formatting returns are added.
459*6777b538SAndroid Build Coastguard Worker  *
460*6777b538SAndroid Build Coastguard Worker  * Returns the number of byte written or -1 in case of error
461*6777b538SAndroid Build Coastguard Worker  */
462*6777b538SAndroid Build Coastguard Worker int
htmlNodeDump(xmlBufferPtr buf,xmlDocPtr doc,xmlNodePtr cur)463*6777b538SAndroid Build Coastguard Worker htmlNodeDump(xmlBufferPtr buf, xmlDocPtr doc, xmlNodePtr cur) {
464*6777b538SAndroid Build Coastguard Worker     xmlBufPtr buffer;
465*6777b538SAndroid Build Coastguard Worker     size_t ret;
466*6777b538SAndroid Build Coastguard Worker 
467*6777b538SAndroid Build Coastguard Worker     if ((buf == NULL) || (cur == NULL))
468*6777b538SAndroid Build Coastguard Worker         return(-1);
469*6777b538SAndroid Build Coastguard Worker 
470*6777b538SAndroid Build Coastguard Worker     xmlInitParser();
471*6777b538SAndroid Build Coastguard Worker     buffer = xmlBufFromBuffer(buf);
472*6777b538SAndroid Build Coastguard Worker     if (buffer == NULL)
473*6777b538SAndroid Build Coastguard Worker         return(-1);
474*6777b538SAndroid Build Coastguard Worker 
475*6777b538SAndroid Build Coastguard Worker     ret = htmlBufNodeDumpFormat(buffer, doc, cur, 1);
476*6777b538SAndroid Build Coastguard Worker 
477*6777b538SAndroid Build Coastguard Worker     xmlBufBackToBuffer(buffer);
478*6777b538SAndroid Build Coastguard Worker 
479*6777b538SAndroid Build Coastguard Worker     if (ret > INT_MAX)
480*6777b538SAndroid Build Coastguard Worker         return(-1);
481*6777b538SAndroid Build Coastguard Worker     return((int) ret);
482*6777b538SAndroid Build Coastguard Worker }
483*6777b538SAndroid Build Coastguard Worker 
484*6777b538SAndroid Build Coastguard Worker /**
485*6777b538SAndroid Build Coastguard Worker  * htmlNodeDumpFileFormat:
486*6777b538SAndroid Build Coastguard Worker  * @out:  the FILE pointer
487*6777b538SAndroid Build Coastguard Worker  * @doc:  the document
488*6777b538SAndroid Build Coastguard Worker  * @cur:  the current node
489*6777b538SAndroid Build Coastguard Worker  * @encoding: the document encoding
490*6777b538SAndroid Build Coastguard Worker  * @format:  should formatting spaces been added
491*6777b538SAndroid Build Coastguard Worker  *
492*6777b538SAndroid Build Coastguard Worker  * Dump an HTML node, recursive behaviour,children are printed too.
493*6777b538SAndroid Build Coastguard Worker  *
494*6777b538SAndroid Build Coastguard Worker  * TODO: if encoding == NULL try to save in the doc encoding
495*6777b538SAndroid Build Coastguard Worker  *
496*6777b538SAndroid Build Coastguard Worker  * returns: the number of byte written or -1 in case of failure.
497*6777b538SAndroid Build Coastguard Worker  */
498*6777b538SAndroid Build Coastguard Worker int
htmlNodeDumpFileFormat(FILE * out,xmlDocPtr doc,xmlNodePtr cur,const char * encoding,int format)499*6777b538SAndroid Build Coastguard Worker htmlNodeDumpFileFormat(FILE *out, xmlDocPtr doc,
500*6777b538SAndroid Build Coastguard Worker 	               xmlNodePtr cur, const char *encoding, int format) {
501*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferPtr buf;
502*6777b538SAndroid Build Coastguard Worker     xmlCharEncodingHandlerPtr handler;
503*6777b538SAndroid Build Coastguard Worker     int ret;
504*6777b538SAndroid Build Coastguard Worker 
505*6777b538SAndroid Build Coastguard Worker     xmlInitParser();
506*6777b538SAndroid Build Coastguard Worker 
507*6777b538SAndroid Build Coastguard Worker     /*
508*6777b538SAndroid Build Coastguard Worker      * save the content to a temp buffer.
509*6777b538SAndroid Build Coastguard Worker      */
510*6777b538SAndroid Build Coastguard Worker     handler = htmlFindOutputEncoder(encoding);
511*6777b538SAndroid Build Coastguard Worker     buf = xmlOutputBufferCreateFile(out, handler);
512*6777b538SAndroid Build Coastguard Worker     if (buf == NULL) return(0);
513*6777b538SAndroid Build Coastguard Worker 
514*6777b538SAndroid Build Coastguard Worker     htmlNodeDumpFormatOutput(buf, doc, cur, NULL, format);
515*6777b538SAndroid Build Coastguard Worker 
516*6777b538SAndroid Build Coastguard Worker     ret = xmlOutputBufferClose(buf);
517*6777b538SAndroid Build Coastguard Worker     return(ret);
518*6777b538SAndroid Build Coastguard Worker }
519*6777b538SAndroid Build Coastguard Worker 
520*6777b538SAndroid Build Coastguard Worker /**
521*6777b538SAndroid Build Coastguard Worker  * htmlNodeDumpFile:
522*6777b538SAndroid Build Coastguard Worker  * @out:  the FILE pointer
523*6777b538SAndroid Build Coastguard Worker  * @doc:  the document
524*6777b538SAndroid Build Coastguard Worker  * @cur:  the current node
525*6777b538SAndroid Build Coastguard Worker  *
526*6777b538SAndroid Build Coastguard Worker  * Dump an HTML node, recursive behaviour,children are printed too,
527*6777b538SAndroid Build Coastguard Worker  * and formatting returns are added.
528*6777b538SAndroid Build Coastguard Worker  */
529*6777b538SAndroid Build Coastguard Worker void
htmlNodeDumpFile(FILE * out,xmlDocPtr doc,xmlNodePtr cur)530*6777b538SAndroid Build Coastguard Worker htmlNodeDumpFile(FILE *out, xmlDocPtr doc, xmlNodePtr cur) {
531*6777b538SAndroid Build Coastguard Worker     htmlNodeDumpFileFormat(out, doc, cur, NULL, 1);
532*6777b538SAndroid Build Coastguard Worker }
533*6777b538SAndroid Build Coastguard Worker 
534*6777b538SAndroid Build Coastguard Worker /**
535*6777b538SAndroid Build Coastguard Worker  * htmlDocDumpMemoryFormat:
536*6777b538SAndroid Build Coastguard Worker  * @cur:  the document
537*6777b538SAndroid Build Coastguard Worker  * @mem:  OUT: the memory pointer
538*6777b538SAndroid Build Coastguard Worker  * @size:  OUT: the memory length
539*6777b538SAndroid Build Coastguard Worker  * @format:  should formatting spaces been added
540*6777b538SAndroid Build Coastguard Worker  *
541*6777b538SAndroid Build Coastguard Worker  * Dump an HTML document in memory and return the xmlChar * and it's size.
542*6777b538SAndroid Build Coastguard Worker  * It's up to the caller to free the memory.
543*6777b538SAndroid Build Coastguard Worker  */
544*6777b538SAndroid Build Coastguard Worker void
htmlDocDumpMemoryFormat(xmlDocPtr cur,xmlChar ** mem,int * size,int format)545*6777b538SAndroid Build Coastguard Worker htmlDocDumpMemoryFormat(xmlDocPtr cur, xmlChar**mem, int *size, int format) {
546*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferPtr buf;
547*6777b538SAndroid Build Coastguard Worker     xmlCharEncodingHandlerPtr handler = NULL;
548*6777b538SAndroid Build Coastguard Worker     const char *encoding;
549*6777b538SAndroid Build Coastguard Worker 
550*6777b538SAndroid Build Coastguard Worker     xmlInitParser();
551*6777b538SAndroid Build Coastguard Worker 
552*6777b538SAndroid Build Coastguard Worker     if ((mem == NULL) || (size == NULL))
553*6777b538SAndroid Build Coastguard Worker         return;
554*6777b538SAndroid Build Coastguard Worker     if (cur == NULL) {
555*6777b538SAndroid Build Coastguard Worker 	*mem = NULL;
556*6777b538SAndroid Build Coastguard Worker 	*size = 0;
557*6777b538SAndroid Build Coastguard Worker 	return;
558*6777b538SAndroid Build Coastguard Worker     }
559*6777b538SAndroid Build Coastguard Worker 
560*6777b538SAndroid Build Coastguard Worker     encoding = (const char *) htmlGetMetaEncoding(cur);
561*6777b538SAndroid Build Coastguard Worker     handler = htmlFindOutputEncoder(encoding);
562*6777b538SAndroid Build Coastguard Worker     buf = xmlAllocOutputBufferInternal(handler);
563*6777b538SAndroid Build Coastguard Worker     if (buf == NULL) {
564*6777b538SAndroid Build Coastguard Worker 	*mem = NULL;
565*6777b538SAndroid Build Coastguard Worker 	*size = 0;
566*6777b538SAndroid Build Coastguard Worker 	return;
567*6777b538SAndroid Build Coastguard Worker     }
568*6777b538SAndroid Build Coastguard Worker 
569*6777b538SAndroid Build Coastguard Worker     htmlDocContentDumpFormatOutput(buf, cur, NULL, format);
570*6777b538SAndroid Build Coastguard Worker 
571*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferFlush(buf);
572*6777b538SAndroid Build Coastguard Worker     if (buf->conv != NULL) {
573*6777b538SAndroid Build Coastguard Worker 	*size = xmlBufUse(buf->conv);
574*6777b538SAndroid Build Coastguard Worker 	*mem = xmlStrndup(xmlBufContent(buf->conv), *size);
575*6777b538SAndroid Build Coastguard Worker     } else {
576*6777b538SAndroid Build Coastguard Worker 	*size = xmlBufUse(buf->buffer);
577*6777b538SAndroid Build Coastguard Worker 	*mem = xmlStrndup(xmlBufContent(buf->buffer), *size);
578*6777b538SAndroid Build Coastguard Worker     }
579*6777b538SAndroid Build Coastguard Worker     (void)xmlOutputBufferClose(buf);
580*6777b538SAndroid Build Coastguard Worker }
581*6777b538SAndroid Build Coastguard Worker 
582*6777b538SAndroid Build Coastguard Worker /**
583*6777b538SAndroid Build Coastguard Worker  * htmlDocDumpMemory:
584*6777b538SAndroid Build Coastguard Worker  * @cur:  the document
585*6777b538SAndroid Build Coastguard Worker  * @mem:  OUT: the memory pointer
586*6777b538SAndroid Build Coastguard Worker  * @size:  OUT: the memory length
587*6777b538SAndroid Build Coastguard Worker  *
588*6777b538SAndroid Build Coastguard Worker  * Dump an HTML document in memory and return the xmlChar * and it's size.
589*6777b538SAndroid Build Coastguard Worker  * It's up to the caller to free the memory.
590*6777b538SAndroid Build Coastguard Worker  */
591*6777b538SAndroid Build Coastguard Worker void
htmlDocDumpMemory(xmlDocPtr cur,xmlChar ** mem,int * size)592*6777b538SAndroid Build Coastguard Worker htmlDocDumpMemory(xmlDocPtr cur, xmlChar**mem, int *size) {
593*6777b538SAndroid Build Coastguard Worker 	htmlDocDumpMemoryFormat(cur, mem, size, 1);
594*6777b538SAndroid Build Coastguard Worker }
595*6777b538SAndroid Build Coastguard Worker 
596*6777b538SAndroid Build Coastguard Worker 
597*6777b538SAndroid Build Coastguard Worker /************************************************************************
598*6777b538SAndroid Build Coastguard Worker  *									*
599*6777b538SAndroid Build Coastguard Worker  *		Dumping HTML tree content to an I/O output buffer	*
600*6777b538SAndroid Build Coastguard Worker  *									*
601*6777b538SAndroid Build Coastguard Worker  ************************************************************************/
602*6777b538SAndroid Build Coastguard Worker 
603*6777b538SAndroid Build Coastguard Worker /**
604*6777b538SAndroid Build Coastguard Worker  * htmlDtdDumpOutput:
605*6777b538SAndroid Build Coastguard Worker  * @buf:  the HTML buffer output
606*6777b538SAndroid Build Coastguard Worker  * @doc:  the document
607*6777b538SAndroid Build Coastguard Worker  * @encoding:  the encoding string
608*6777b538SAndroid Build Coastguard Worker  *
609*6777b538SAndroid Build Coastguard Worker  * TODO: check whether encoding is needed
610*6777b538SAndroid Build Coastguard Worker  *
611*6777b538SAndroid Build Coastguard Worker  * Dump the HTML document DTD, if any.
612*6777b538SAndroid Build Coastguard Worker  */
613*6777b538SAndroid Build Coastguard Worker static void
htmlDtdDumpOutput(xmlOutputBufferPtr buf,xmlDocPtr doc,const char * encoding ATTRIBUTE_UNUSED)614*6777b538SAndroid Build Coastguard Worker htmlDtdDumpOutput(xmlOutputBufferPtr buf, xmlDocPtr doc,
615*6777b538SAndroid Build Coastguard Worker 	          const char *encoding ATTRIBUTE_UNUSED) {
616*6777b538SAndroid Build Coastguard Worker     xmlDtdPtr cur = doc->intSubset;
617*6777b538SAndroid Build Coastguard Worker 
618*6777b538SAndroid Build Coastguard Worker     if (cur == NULL) {
619*6777b538SAndroid Build Coastguard Worker 	htmlSaveErr(XML_SAVE_NO_DOCTYPE, (xmlNodePtr) doc, NULL);
620*6777b538SAndroid Build Coastguard Worker 	return;
621*6777b538SAndroid Build Coastguard Worker     }
622*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferWriteString(buf, "<!DOCTYPE ");
623*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferWriteString(buf, (const char *)cur->name);
624*6777b538SAndroid Build Coastguard Worker     if (cur->ExternalID != NULL) {
625*6777b538SAndroid Build Coastguard Worker 	xmlOutputBufferWriteString(buf, " PUBLIC ");
626*6777b538SAndroid Build Coastguard Worker 	xmlBufWriteQuotedString(buf->buffer, cur->ExternalID);
627*6777b538SAndroid Build Coastguard Worker 	if (cur->SystemID != NULL) {
628*6777b538SAndroid Build Coastguard Worker 	    xmlOutputBufferWriteString(buf, " ");
629*6777b538SAndroid Build Coastguard Worker 	    xmlBufWriteQuotedString(buf->buffer, cur->SystemID);
630*6777b538SAndroid Build Coastguard Worker 	}
631*6777b538SAndroid Build Coastguard Worker     } else if (cur->SystemID != NULL &&
632*6777b538SAndroid Build Coastguard Worker 	       xmlStrcmp(cur->SystemID, BAD_CAST "about:legacy-compat")) {
633*6777b538SAndroid Build Coastguard Worker 	xmlOutputBufferWriteString(buf, " SYSTEM ");
634*6777b538SAndroid Build Coastguard Worker 	xmlBufWriteQuotedString(buf->buffer, cur->SystemID);
635*6777b538SAndroid Build Coastguard Worker     }
636*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferWriteString(buf, ">\n");
637*6777b538SAndroid Build Coastguard Worker }
638*6777b538SAndroid Build Coastguard Worker 
639*6777b538SAndroid Build Coastguard Worker /**
640*6777b538SAndroid Build Coastguard Worker  * htmlAttrDumpOutput:
641*6777b538SAndroid Build Coastguard Worker  * @buf:  the HTML buffer output
642*6777b538SAndroid Build Coastguard Worker  * @doc:  the document
643*6777b538SAndroid Build Coastguard Worker  * @cur:  the attribute pointer
644*6777b538SAndroid Build Coastguard Worker  *
645*6777b538SAndroid Build Coastguard Worker  * Dump an HTML attribute
646*6777b538SAndroid Build Coastguard Worker  */
647*6777b538SAndroid Build Coastguard Worker static void
htmlAttrDumpOutput(xmlOutputBufferPtr buf,xmlDocPtr doc,xmlAttrPtr cur)648*6777b538SAndroid Build Coastguard Worker htmlAttrDumpOutput(xmlOutputBufferPtr buf, xmlDocPtr doc, xmlAttrPtr cur) {
649*6777b538SAndroid Build Coastguard Worker     xmlChar *value;
650*6777b538SAndroid Build Coastguard Worker 
651*6777b538SAndroid Build Coastguard Worker     /*
652*6777b538SAndroid Build Coastguard Worker      * The html output method should not escape a & character
653*6777b538SAndroid Build Coastguard Worker      * occurring in an attribute value immediately followed by
654*6777b538SAndroid Build Coastguard Worker      * a { character (see Section B.7.1 of the HTML 4.0 Recommendation).
655*6777b538SAndroid Build Coastguard Worker      * This is implemented in xmlEncodeEntitiesReentrant
656*6777b538SAndroid Build Coastguard Worker      */
657*6777b538SAndroid Build Coastguard Worker 
658*6777b538SAndroid Build Coastguard Worker     if (cur == NULL) {
659*6777b538SAndroid Build Coastguard Worker 	return;
660*6777b538SAndroid Build Coastguard Worker     }
661*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferWriteString(buf, " ");
662*6777b538SAndroid Build Coastguard Worker     if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
663*6777b538SAndroid Build Coastguard Worker         xmlOutputBufferWriteString(buf, (const char *)cur->ns->prefix);
664*6777b538SAndroid Build Coastguard Worker 	xmlOutputBufferWriteString(buf, ":");
665*6777b538SAndroid Build Coastguard Worker     }
666*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferWriteString(buf, (const char *)cur->name);
667*6777b538SAndroid Build Coastguard Worker     if ((cur->children != NULL) && (!htmlIsBooleanAttr(cur->name))) {
668*6777b538SAndroid Build Coastguard Worker 	value = xmlNodeListGetString(doc, cur->children, 0);
669*6777b538SAndroid Build Coastguard Worker 	if (value) {
670*6777b538SAndroid Build Coastguard Worker 	    xmlOutputBufferWriteString(buf, "=");
671*6777b538SAndroid Build Coastguard Worker 	    if ((cur->ns == NULL) && (cur->parent != NULL) &&
672*6777b538SAndroid Build Coastguard Worker 		(cur->parent->ns == NULL) &&
673*6777b538SAndroid Build Coastguard Worker 		((!xmlStrcasecmp(cur->name, BAD_CAST "href")) ||
674*6777b538SAndroid Build Coastguard Worker 	         (!xmlStrcasecmp(cur->name, BAD_CAST "action")) ||
675*6777b538SAndroid Build Coastguard Worker 		 (!xmlStrcasecmp(cur->name, BAD_CAST "src")) ||
676*6777b538SAndroid Build Coastguard Worker 		 ((!xmlStrcasecmp(cur->name, BAD_CAST "name")) &&
677*6777b538SAndroid Build Coastguard Worker 		  (!xmlStrcasecmp(cur->parent->name, BAD_CAST "a"))))) {
678*6777b538SAndroid Build Coastguard Worker 		xmlChar *escaped;
679*6777b538SAndroid Build Coastguard Worker 		xmlChar *tmp = value;
680*6777b538SAndroid Build Coastguard Worker 
681*6777b538SAndroid Build Coastguard Worker 		while (IS_BLANK_CH(*tmp)) tmp++;
682*6777b538SAndroid Build Coastguard Worker 
683*6777b538SAndroid Build Coastguard Worker 		/*
684*6777b538SAndroid Build Coastguard Worker                  * Angle brackets are technically illegal in URIs, but they're
685*6777b538SAndroid Build Coastguard Worker                  * used in server side includes, for example. Curly brackets
686*6777b538SAndroid Build Coastguard Worker                  * are illegal as well and often used in templates.
687*6777b538SAndroid Build Coastguard Worker                  * Don't escape non-whitespace, printable ASCII chars for
688*6777b538SAndroid Build Coastguard Worker                  * improved interoperability. Only escape space, control
689*6777b538SAndroid Build Coastguard Worker                  * and non-ASCII chars.
690*6777b538SAndroid Build Coastguard Worker 		 */
691*6777b538SAndroid Build Coastguard Worker 		escaped = xmlURIEscapeStr(tmp,
692*6777b538SAndroid Build Coastguard Worker                         BAD_CAST "\"#$%&+,/:;<=>?@[\\]^`{|}");
693*6777b538SAndroid Build Coastguard Worker 		if (escaped != NULL) {
694*6777b538SAndroid Build Coastguard Worker 		    xmlBufWriteQuotedString(buf->buffer, escaped);
695*6777b538SAndroid Build Coastguard Worker 		    xmlFree(escaped);
696*6777b538SAndroid Build Coastguard Worker 		} else {
697*6777b538SAndroid Build Coastguard Worker                     buf->error = XML_ERR_NO_MEMORY;
698*6777b538SAndroid Build Coastguard Worker 		}
699*6777b538SAndroid Build Coastguard Worker 	    } else {
700*6777b538SAndroid Build Coastguard Worker 		xmlBufWriteQuotedString(buf->buffer, value);
701*6777b538SAndroid Build Coastguard Worker 	    }
702*6777b538SAndroid Build Coastguard Worker 	    xmlFree(value);
703*6777b538SAndroid Build Coastguard Worker 	} else  {
704*6777b538SAndroid Build Coastguard Worker             buf->error = XML_ERR_NO_MEMORY;
705*6777b538SAndroid Build Coastguard Worker 	}
706*6777b538SAndroid Build Coastguard Worker     }
707*6777b538SAndroid Build Coastguard Worker }
708*6777b538SAndroid Build Coastguard Worker 
709*6777b538SAndroid Build Coastguard Worker /**
710*6777b538SAndroid Build Coastguard Worker  * htmlNodeDumpFormatOutput:
711*6777b538SAndroid Build Coastguard Worker  * @buf:  the HTML buffer output
712*6777b538SAndroid Build Coastguard Worker  * @doc:  the document
713*6777b538SAndroid Build Coastguard Worker  * @cur:  the current node
714*6777b538SAndroid Build Coastguard Worker  * @encoding:  the encoding string (unused)
715*6777b538SAndroid Build Coastguard Worker  * @format:  should formatting spaces been added
716*6777b538SAndroid Build Coastguard Worker  *
717*6777b538SAndroid Build Coastguard Worker  * Dump an HTML node, recursive behaviour,children are printed too.
718*6777b538SAndroid Build Coastguard Worker  */
719*6777b538SAndroid Build Coastguard Worker void
htmlNodeDumpFormatOutput(xmlOutputBufferPtr buf,xmlDocPtr doc,xmlNodePtr cur,const char * encoding ATTRIBUTE_UNUSED,int format)720*6777b538SAndroid Build Coastguard Worker htmlNodeDumpFormatOutput(xmlOutputBufferPtr buf, xmlDocPtr doc,
721*6777b538SAndroid Build Coastguard Worker 	                 xmlNodePtr cur, const char *encoding ATTRIBUTE_UNUSED,
722*6777b538SAndroid Build Coastguard Worker                          int format) {
723*6777b538SAndroid Build Coastguard Worker     xmlNodePtr root, parent;
724*6777b538SAndroid Build Coastguard Worker     xmlAttrPtr attr;
725*6777b538SAndroid Build Coastguard Worker     const htmlElemDesc * info;
726*6777b538SAndroid Build Coastguard Worker 
727*6777b538SAndroid Build Coastguard Worker     xmlInitParser();
728*6777b538SAndroid Build Coastguard Worker 
729*6777b538SAndroid Build Coastguard Worker     if ((cur == NULL) || (buf == NULL)) {
730*6777b538SAndroid Build Coastguard Worker 	return;
731*6777b538SAndroid Build Coastguard Worker     }
732*6777b538SAndroid Build Coastguard Worker 
733*6777b538SAndroid Build Coastguard Worker     root = cur;
734*6777b538SAndroid Build Coastguard Worker     parent = cur->parent;
735*6777b538SAndroid Build Coastguard Worker     while (1) {
736*6777b538SAndroid Build Coastguard Worker         switch (cur->type) {
737*6777b538SAndroid Build Coastguard Worker         case XML_HTML_DOCUMENT_NODE:
738*6777b538SAndroid Build Coastguard Worker         case XML_DOCUMENT_NODE:
739*6777b538SAndroid Build Coastguard Worker             if (((xmlDocPtr) cur)->intSubset != NULL) {
740*6777b538SAndroid Build Coastguard Worker                 htmlDtdDumpOutput(buf, (xmlDocPtr) cur, NULL);
741*6777b538SAndroid Build Coastguard Worker             }
742*6777b538SAndroid Build Coastguard Worker             if (cur->children != NULL) {
743*6777b538SAndroid Build Coastguard Worker                 /* Always validate cur->parent when descending. */
744*6777b538SAndroid Build Coastguard Worker                 if (cur->parent == parent) {
745*6777b538SAndroid Build Coastguard Worker                     parent = cur;
746*6777b538SAndroid Build Coastguard Worker                     cur = cur->children;
747*6777b538SAndroid Build Coastguard Worker                     continue;
748*6777b538SAndroid Build Coastguard Worker                 }
749*6777b538SAndroid Build Coastguard Worker             } else {
750*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, "\n");
751*6777b538SAndroid Build Coastguard Worker             }
752*6777b538SAndroid Build Coastguard Worker             break;
753*6777b538SAndroid Build Coastguard Worker 
754*6777b538SAndroid Build Coastguard Worker         case XML_ELEMENT_NODE:
755*6777b538SAndroid Build Coastguard Worker             /*
756*6777b538SAndroid Build Coastguard Worker              * Some users like lxml are known to pass nodes with a corrupted
757*6777b538SAndroid Build Coastguard Worker              * tree structure. Fall back to a recursive call to handle this
758*6777b538SAndroid Build Coastguard Worker              * case.
759*6777b538SAndroid Build Coastguard Worker              */
760*6777b538SAndroid Build Coastguard Worker             if ((cur->parent != parent) && (cur->children != NULL)) {
761*6777b538SAndroid Build Coastguard Worker                 htmlNodeDumpFormatOutput(buf, doc, cur, encoding, format);
762*6777b538SAndroid Build Coastguard Worker                 break;
763*6777b538SAndroid Build Coastguard Worker             }
764*6777b538SAndroid Build Coastguard Worker 
765*6777b538SAndroid Build Coastguard Worker             /*
766*6777b538SAndroid Build Coastguard Worker              * Get specific HTML info for that node.
767*6777b538SAndroid Build Coastguard Worker              */
768*6777b538SAndroid Build Coastguard Worker             if (cur->ns == NULL)
769*6777b538SAndroid Build Coastguard Worker                 info = htmlTagLookup(cur->name);
770*6777b538SAndroid Build Coastguard Worker             else
771*6777b538SAndroid Build Coastguard Worker                 info = NULL;
772*6777b538SAndroid Build Coastguard Worker 
773*6777b538SAndroid Build Coastguard Worker             xmlOutputBufferWriteString(buf, "<");
774*6777b538SAndroid Build Coastguard Worker             if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
775*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, (const char *)cur->ns->prefix);
776*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, ":");
777*6777b538SAndroid Build Coastguard Worker             }
778*6777b538SAndroid Build Coastguard Worker             xmlOutputBufferWriteString(buf, (const char *)cur->name);
779*6777b538SAndroid Build Coastguard Worker             if (cur->nsDef)
780*6777b538SAndroid Build Coastguard Worker                 xmlNsListDumpOutput(buf, cur->nsDef);
781*6777b538SAndroid Build Coastguard Worker             attr = cur->properties;
782*6777b538SAndroid Build Coastguard Worker             while (attr != NULL) {
783*6777b538SAndroid Build Coastguard Worker                 htmlAttrDumpOutput(buf, doc, attr);
784*6777b538SAndroid Build Coastguard Worker                 attr = attr->next;
785*6777b538SAndroid Build Coastguard Worker             }
786*6777b538SAndroid Build Coastguard Worker 
787*6777b538SAndroid Build Coastguard Worker             if ((info != NULL) && (info->empty)) {
788*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, ">");
789*6777b538SAndroid Build Coastguard Worker             } else if (cur->children == NULL) {
790*6777b538SAndroid Build Coastguard Worker                 if ((info != NULL) && (info->saveEndTag != 0) &&
791*6777b538SAndroid Build Coastguard Worker                     (xmlStrcmp(BAD_CAST info->name, BAD_CAST "html")) &&
792*6777b538SAndroid Build Coastguard Worker                     (xmlStrcmp(BAD_CAST info->name, BAD_CAST "body"))) {
793*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf, ">");
794*6777b538SAndroid Build Coastguard Worker                 } else {
795*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf, "></");
796*6777b538SAndroid Build Coastguard Worker                     if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
797*6777b538SAndroid Build Coastguard Worker                         xmlOutputBufferWriteString(buf,
798*6777b538SAndroid Build Coastguard Worker                                 (const char *)cur->ns->prefix);
799*6777b538SAndroid Build Coastguard Worker                         xmlOutputBufferWriteString(buf, ":");
800*6777b538SAndroid Build Coastguard Worker                     }
801*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf, (const char *)cur->name);
802*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf, ">");
803*6777b538SAndroid Build Coastguard Worker                 }
804*6777b538SAndroid Build Coastguard Worker             } else {
805*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, ">");
806*6777b538SAndroid Build Coastguard Worker                 if ((format) && (info != NULL) && (!info->isinline) &&
807*6777b538SAndroid Build Coastguard Worker                     (cur->children->type != HTML_TEXT_NODE) &&
808*6777b538SAndroid Build Coastguard Worker                     (cur->children->type != HTML_ENTITY_REF_NODE) &&
809*6777b538SAndroid Build Coastguard Worker                     (cur->children != cur->last) &&
810*6777b538SAndroid Build Coastguard Worker                     (cur->name != NULL) &&
811*6777b538SAndroid Build Coastguard Worker                     (cur->name[0] != 'p')) /* p, pre, param */
812*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf, "\n");
813*6777b538SAndroid Build Coastguard Worker                 parent = cur;
814*6777b538SAndroid Build Coastguard Worker                 cur = cur->children;
815*6777b538SAndroid Build Coastguard Worker                 continue;
816*6777b538SAndroid Build Coastguard Worker             }
817*6777b538SAndroid Build Coastguard Worker 
818*6777b538SAndroid Build Coastguard Worker             if ((format) && (cur->next != NULL) &&
819*6777b538SAndroid Build Coastguard Worker                 (info != NULL) && (!info->isinline)) {
820*6777b538SAndroid Build Coastguard Worker                 if ((cur->next->type != HTML_TEXT_NODE) &&
821*6777b538SAndroid Build Coastguard Worker                     (cur->next->type != HTML_ENTITY_REF_NODE) &&
822*6777b538SAndroid Build Coastguard Worker                     (parent != NULL) &&
823*6777b538SAndroid Build Coastguard Worker                     (parent->name != NULL) &&
824*6777b538SAndroid Build Coastguard Worker                     (parent->name[0] != 'p')) /* p, pre, param */
825*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf, "\n");
826*6777b538SAndroid Build Coastguard Worker             }
827*6777b538SAndroid Build Coastguard Worker 
828*6777b538SAndroid Build Coastguard Worker             break;
829*6777b538SAndroid Build Coastguard Worker 
830*6777b538SAndroid Build Coastguard Worker         case XML_ATTRIBUTE_NODE:
831*6777b538SAndroid Build Coastguard Worker             htmlAttrDumpOutput(buf, doc, (xmlAttrPtr) cur);
832*6777b538SAndroid Build Coastguard Worker             break;
833*6777b538SAndroid Build Coastguard Worker 
834*6777b538SAndroid Build Coastguard Worker         case HTML_TEXT_NODE:
835*6777b538SAndroid Build Coastguard Worker             if (cur->content == NULL)
836*6777b538SAndroid Build Coastguard Worker                 break;
837*6777b538SAndroid Build Coastguard Worker             if (((cur->name == (const xmlChar *)xmlStringText) ||
838*6777b538SAndroid Build Coastguard Worker                  (cur->name != (const xmlChar *)xmlStringTextNoenc)) &&
839*6777b538SAndroid Build Coastguard Worker                 ((parent == NULL) ||
840*6777b538SAndroid Build Coastguard Worker                  ((xmlStrcasecmp(parent->name, BAD_CAST "script")) &&
841*6777b538SAndroid Build Coastguard Worker                   (xmlStrcasecmp(parent->name, BAD_CAST "style"))))) {
842*6777b538SAndroid Build Coastguard Worker                 xmlChar *buffer;
843*6777b538SAndroid Build Coastguard Worker 
844*6777b538SAndroid Build Coastguard Worker                 buffer = xmlEncodeEntitiesReentrant(doc, cur->content);
845*6777b538SAndroid Build Coastguard Worker                 if (buffer == NULL) {
846*6777b538SAndroid Build Coastguard Worker                     buf->error = XML_ERR_NO_MEMORY;
847*6777b538SAndroid Build Coastguard Worker                     return;
848*6777b538SAndroid Build Coastguard Worker                 }
849*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, (const char *)buffer);
850*6777b538SAndroid Build Coastguard Worker                 xmlFree(buffer);
851*6777b538SAndroid Build Coastguard Worker             } else {
852*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, (const char *)cur->content);
853*6777b538SAndroid Build Coastguard Worker             }
854*6777b538SAndroid Build Coastguard Worker             break;
855*6777b538SAndroid Build Coastguard Worker 
856*6777b538SAndroid Build Coastguard Worker         case HTML_COMMENT_NODE:
857*6777b538SAndroid Build Coastguard Worker             if (cur->content != NULL) {
858*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, "<!--");
859*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, (const char *)cur->content);
860*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, "-->");
861*6777b538SAndroid Build Coastguard Worker             }
862*6777b538SAndroid Build Coastguard Worker             break;
863*6777b538SAndroid Build Coastguard Worker 
864*6777b538SAndroid Build Coastguard Worker         case HTML_PI_NODE:
865*6777b538SAndroid Build Coastguard Worker             if (cur->name != NULL) {
866*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, "<?");
867*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, (const char *)cur->name);
868*6777b538SAndroid Build Coastguard Worker                 if (cur->content != NULL) {
869*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf, " ");
870*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf,
871*6777b538SAndroid Build Coastguard Worker                             (const char *)cur->content);
872*6777b538SAndroid Build Coastguard Worker                 }
873*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, ">");
874*6777b538SAndroid Build Coastguard Worker             }
875*6777b538SAndroid Build Coastguard Worker             break;
876*6777b538SAndroid Build Coastguard Worker 
877*6777b538SAndroid Build Coastguard Worker         case HTML_ENTITY_REF_NODE:
878*6777b538SAndroid Build Coastguard Worker             xmlOutputBufferWriteString(buf, "&");
879*6777b538SAndroid Build Coastguard Worker             xmlOutputBufferWriteString(buf, (const char *)cur->name);
880*6777b538SAndroid Build Coastguard Worker             xmlOutputBufferWriteString(buf, ";");
881*6777b538SAndroid Build Coastguard Worker             break;
882*6777b538SAndroid Build Coastguard Worker 
883*6777b538SAndroid Build Coastguard Worker         case HTML_PRESERVE_NODE:
884*6777b538SAndroid Build Coastguard Worker             if (cur->content != NULL) {
885*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, (const char *)cur->content);
886*6777b538SAndroid Build Coastguard Worker             }
887*6777b538SAndroid Build Coastguard Worker             break;
888*6777b538SAndroid Build Coastguard Worker 
889*6777b538SAndroid Build Coastguard Worker         default:
890*6777b538SAndroid Build Coastguard Worker             break;
891*6777b538SAndroid Build Coastguard Worker         }
892*6777b538SAndroid Build Coastguard Worker 
893*6777b538SAndroid Build Coastguard Worker         while (1) {
894*6777b538SAndroid Build Coastguard Worker             if (cur == root)
895*6777b538SAndroid Build Coastguard Worker                 return;
896*6777b538SAndroid Build Coastguard Worker             if (cur->next != NULL) {
897*6777b538SAndroid Build Coastguard Worker                 cur = cur->next;
898*6777b538SAndroid Build Coastguard Worker                 break;
899*6777b538SAndroid Build Coastguard Worker             }
900*6777b538SAndroid Build Coastguard Worker 
901*6777b538SAndroid Build Coastguard Worker             cur = parent;
902*6777b538SAndroid Build Coastguard Worker             /* cur->parent was validated when descending. */
903*6777b538SAndroid Build Coastguard Worker             parent = cur->parent;
904*6777b538SAndroid Build Coastguard Worker 
905*6777b538SAndroid Build Coastguard Worker             if ((cur->type == XML_HTML_DOCUMENT_NODE) ||
906*6777b538SAndroid Build Coastguard Worker                 (cur->type == XML_DOCUMENT_NODE)) {
907*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, "\n");
908*6777b538SAndroid Build Coastguard Worker             } else {
909*6777b538SAndroid Build Coastguard Worker                 if ((format) && (cur->ns == NULL))
910*6777b538SAndroid Build Coastguard Worker                     info = htmlTagLookup(cur->name);
911*6777b538SAndroid Build Coastguard Worker                 else
912*6777b538SAndroid Build Coastguard Worker                     info = NULL;
913*6777b538SAndroid Build Coastguard Worker 
914*6777b538SAndroid Build Coastguard Worker                 if ((format) && (info != NULL) && (!info->isinline) &&
915*6777b538SAndroid Build Coastguard Worker                     (cur->last->type != HTML_TEXT_NODE) &&
916*6777b538SAndroid Build Coastguard Worker                     (cur->last->type != HTML_ENTITY_REF_NODE) &&
917*6777b538SAndroid Build Coastguard Worker                     (cur->children != cur->last) &&
918*6777b538SAndroid Build Coastguard Worker                     (cur->name != NULL) &&
919*6777b538SAndroid Build Coastguard Worker                     (cur->name[0] != 'p')) /* p, pre, param */
920*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf, "\n");
921*6777b538SAndroid Build Coastguard Worker 
922*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, "</");
923*6777b538SAndroid Build Coastguard Worker                 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
924*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf, (const char *)cur->ns->prefix);
925*6777b538SAndroid Build Coastguard Worker                     xmlOutputBufferWriteString(buf, ":");
926*6777b538SAndroid Build Coastguard Worker                 }
927*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, (const char *)cur->name);
928*6777b538SAndroid Build Coastguard Worker                 xmlOutputBufferWriteString(buf, ">");
929*6777b538SAndroid Build Coastguard Worker 
930*6777b538SAndroid Build Coastguard Worker                 if ((format) && (info != NULL) && (!info->isinline) &&
931*6777b538SAndroid Build Coastguard Worker                     (cur->next != NULL)) {
932*6777b538SAndroid Build Coastguard Worker                     if ((cur->next->type != HTML_TEXT_NODE) &&
933*6777b538SAndroid Build Coastguard Worker                         (cur->next->type != HTML_ENTITY_REF_NODE) &&
934*6777b538SAndroid Build Coastguard Worker                         (parent != NULL) &&
935*6777b538SAndroid Build Coastguard Worker                         (parent->name != NULL) &&
936*6777b538SAndroid Build Coastguard Worker                         (parent->name[0] != 'p')) /* p, pre, param */
937*6777b538SAndroid Build Coastguard Worker                         xmlOutputBufferWriteString(buf, "\n");
938*6777b538SAndroid Build Coastguard Worker                 }
939*6777b538SAndroid Build Coastguard Worker             }
940*6777b538SAndroid Build Coastguard Worker         }
941*6777b538SAndroid Build Coastguard Worker     }
942*6777b538SAndroid Build Coastguard Worker }
943*6777b538SAndroid Build Coastguard Worker 
944*6777b538SAndroid Build Coastguard Worker /**
945*6777b538SAndroid Build Coastguard Worker  * htmlNodeDumpOutput:
946*6777b538SAndroid Build Coastguard Worker  * @buf:  the HTML buffer output
947*6777b538SAndroid Build Coastguard Worker  * @doc:  the document
948*6777b538SAndroid Build Coastguard Worker  * @cur:  the current node
949*6777b538SAndroid Build Coastguard Worker  * @encoding:  the encoding string (unused)
950*6777b538SAndroid Build Coastguard Worker  *
951*6777b538SAndroid Build Coastguard Worker  * Dump an HTML node, recursive behaviour,children are printed too,
952*6777b538SAndroid Build Coastguard Worker  * and formatting returns/spaces are added.
953*6777b538SAndroid Build Coastguard Worker  */
954*6777b538SAndroid Build Coastguard Worker void
htmlNodeDumpOutput(xmlOutputBufferPtr buf,xmlDocPtr doc,xmlNodePtr cur,const char * encoding ATTRIBUTE_UNUSED)955*6777b538SAndroid Build Coastguard Worker htmlNodeDumpOutput(xmlOutputBufferPtr buf, xmlDocPtr doc,
956*6777b538SAndroid Build Coastguard Worker 	           xmlNodePtr cur, const char *encoding ATTRIBUTE_UNUSED) {
957*6777b538SAndroid Build Coastguard Worker     htmlNodeDumpFormatOutput(buf, doc, cur, NULL, 1);
958*6777b538SAndroid Build Coastguard Worker }
959*6777b538SAndroid Build Coastguard Worker 
960*6777b538SAndroid Build Coastguard Worker /**
961*6777b538SAndroid Build Coastguard Worker  * htmlDocContentDumpFormatOutput:
962*6777b538SAndroid Build Coastguard Worker  * @buf:  the HTML buffer output
963*6777b538SAndroid Build Coastguard Worker  * @cur:  the document
964*6777b538SAndroid Build Coastguard Worker  * @encoding:  the encoding string (unused)
965*6777b538SAndroid Build Coastguard Worker  * @format:  should formatting spaces been added
966*6777b538SAndroid Build Coastguard Worker  *
967*6777b538SAndroid Build Coastguard Worker  * Dump an HTML document.
968*6777b538SAndroid Build Coastguard Worker  */
969*6777b538SAndroid Build Coastguard Worker void
htmlDocContentDumpFormatOutput(xmlOutputBufferPtr buf,xmlDocPtr cur,const char * encoding ATTRIBUTE_UNUSED,int format)970*6777b538SAndroid Build Coastguard Worker htmlDocContentDumpFormatOutput(xmlOutputBufferPtr buf, xmlDocPtr cur,
971*6777b538SAndroid Build Coastguard Worker 	                       const char *encoding ATTRIBUTE_UNUSED,
972*6777b538SAndroid Build Coastguard Worker                                int format) {
973*6777b538SAndroid Build Coastguard Worker     int type = 0;
974*6777b538SAndroid Build Coastguard Worker     if (cur) {
975*6777b538SAndroid Build Coastguard Worker         type = cur->type;
976*6777b538SAndroid Build Coastguard Worker         cur->type = XML_HTML_DOCUMENT_NODE;
977*6777b538SAndroid Build Coastguard Worker     }
978*6777b538SAndroid Build Coastguard Worker     htmlNodeDumpFormatOutput(buf, cur, (xmlNodePtr) cur, NULL, format);
979*6777b538SAndroid Build Coastguard Worker     if (cur)
980*6777b538SAndroid Build Coastguard Worker         cur->type = (xmlElementType) type;
981*6777b538SAndroid Build Coastguard Worker }
982*6777b538SAndroid Build Coastguard Worker 
983*6777b538SAndroid Build Coastguard Worker /**
984*6777b538SAndroid Build Coastguard Worker  * htmlDocContentDumpOutput:
985*6777b538SAndroid Build Coastguard Worker  * @buf:  the HTML buffer output
986*6777b538SAndroid Build Coastguard Worker  * @cur:  the document
987*6777b538SAndroid Build Coastguard Worker  * @encoding:  the encoding string (unused)
988*6777b538SAndroid Build Coastguard Worker  *
989*6777b538SAndroid Build Coastguard Worker  * Dump an HTML document. Formatting return/spaces are added.
990*6777b538SAndroid Build Coastguard Worker  */
991*6777b538SAndroid Build Coastguard Worker void
htmlDocContentDumpOutput(xmlOutputBufferPtr buf,xmlDocPtr cur,const char * encoding ATTRIBUTE_UNUSED)992*6777b538SAndroid Build Coastguard Worker htmlDocContentDumpOutput(xmlOutputBufferPtr buf, xmlDocPtr cur,
993*6777b538SAndroid Build Coastguard Worker 	                 const char *encoding ATTRIBUTE_UNUSED) {
994*6777b538SAndroid Build Coastguard Worker     htmlNodeDumpFormatOutput(buf, cur, (xmlNodePtr) cur, NULL, 1);
995*6777b538SAndroid Build Coastguard Worker }
996*6777b538SAndroid Build Coastguard Worker 
997*6777b538SAndroid Build Coastguard Worker /************************************************************************
998*6777b538SAndroid Build Coastguard Worker  *									*
999*6777b538SAndroid Build Coastguard Worker  *		Saving functions front-ends				*
1000*6777b538SAndroid Build Coastguard Worker  *									*
1001*6777b538SAndroid Build Coastguard Worker  ************************************************************************/
1002*6777b538SAndroid Build Coastguard Worker 
1003*6777b538SAndroid Build Coastguard Worker /**
1004*6777b538SAndroid Build Coastguard Worker  * htmlDocDump:
1005*6777b538SAndroid Build Coastguard Worker  * @f:  the FILE*
1006*6777b538SAndroid Build Coastguard Worker  * @cur:  the document
1007*6777b538SAndroid Build Coastguard Worker  *
1008*6777b538SAndroid Build Coastguard Worker  * Dump an HTML document to an open FILE.
1009*6777b538SAndroid Build Coastguard Worker  *
1010*6777b538SAndroid Build Coastguard Worker  * returns: the number of byte written or -1 in case of failure.
1011*6777b538SAndroid Build Coastguard Worker  */
1012*6777b538SAndroid Build Coastguard Worker int
htmlDocDump(FILE * f,xmlDocPtr cur)1013*6777b538SAndroid Build Coastguard Worker htmlDocDump(FILE *f, xmlDocPtr cur) {
1014*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferPtr buf;
1015*6777b538SAndroid Build Coastguard Worker     xmlCharEncodingHandlerPtr handler = NULL;
1016*6777b538SAndroid Build Coastguard Worker     const char *encoding;
1017*6777b538SAndroid Build Coastguard Worker     int ret;
1018*6777b538SAndroid Build Coastguard Worker 
1019*6777b538SAndroid Build Coastguard Worker     xmlInitParser();
1020*6777b538SAndroid Build Coastguard Worker 
1021*6777b538SAndroid Build Coastguard Worker     if ((cur == NULL) || (f == NULL)) {
1022*6777b538SAndroid Build Coastguard Worker 	return(-1);
1023*6777b538SAndroid Build Coastguard Worker     }
1024*6777b538SAndroid Build Coastguard Worker 
1025*6777b538SAndroid Build Coastguard Worker     encoding = (const char *) htmlGetMetaEncoding(cur);
1026*6777b538SAndroid Build Coastguard Worker     handler = htmlFindOutputEncoder(encoding);
1027*6777b538SAndroid Build Coastguard Worker     buf = xmlOutputBufferCreateFile(f, handler);
1028*6777b538SAndroid Build Coastguard Worker     if (buf == NULL) return(-1);
1029*6777b538SAndroid Build Coastguard Worker     htmlDocContentDumpOutput(buf, cur, NULL);
1030*6777b538SAndroid Build Coastguard Worker 
1031*6777b538SAndroid Build Coastguard Worker     ret = xmlOutputBufferClose(buf);
1032*6777b538SAndroid Build Coastguard Worker     return(ret);
1033*6777b538SAndroid Build Coastguard Worker }
1034*6777b538SAndroid Build Coastguard Worker 
1035*6777b538SAndroid Build Coastguard Worker /**
1036*6777b538SAndroid Build Coastguard Worker  * htmlSaveFile:
1037*6777b538SAndroid Build Coastguard Worker  * @filename:  the filename (or URL)
1038*6777b538SAndroid Build Coastguard Worker  * @cur:  the document
1039*6777b538SAndroid Build Coastguard Worker  *
1040*6777b538SAndroid Build Coastguard Worker  * Dump an HTML document to a file. If @filename is "-" the stdout file is
1041*6777b538SAndroid Build Coastguard Worker  * used.
1042*6777b538SAndroid Build Coastguard Worker  * returns: the number of byte written or -1 in case of failure.
1043*6777b538SAndroid Build Coastguard Worker  */
1044*6777b538SAndroid Build Coastguard Worker int
htmlSaveFile(const char * filename,xmlDocPtr cur)1045*6777b538SAndroid Build Coastguard Worker htmlSaveFile(const char *filename, xmlDocPtr cur) {
1046*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferPtr buf;
1047*6777b538SAndroid Build Coastguard Worker     xmlCharEncodingHandlerPtr handler = NULL;
1048*6777b538SAndroid Build Coastguard Worker     const char *encoding;
1049*6777b538SAndroid Build Coastguard Worker     int ret;
1050*6777b538SAndroid Build Coastguard Worker 
1051*6777b538SAndroid Build Coastguard Worker     if ((cur == NULL) || (filename == NULL))
1052*6777b538SAndroid Build Coastguard Worker         return(-1);
1053*6777b538SAndroid Build Coastguard Worker 
1054*6777b538SAndroid Build Coastguard Worker     xmlInitParser();
1055*6777b538SAndroid Build Coastguard Worker 
1056*6777b538SAndroid Build Coastguard Worker     encoding = (const char *) htmlGetMetaEncoding(cur);
1057*6777b538SAndroid Build Coastguard Worker     handler = htmlFindOutputEncoder(encoding);
1058*6777b538SAndroid Build Coastguard Worker     buf = xmlOutputBufferCreateFilename(filename, handler, cur->compression);
1059*6777b538SAndroid Build Coastguard Worker     if (buf == NULL) return(0);
1060*6777b538SAndroid Build Coastguard Worker 
1061*6777b538SAndroid Build Coastguard Worker     htmlDocContentDumpOutput(buf, cur, NULL);
1062*6777b538SAndroid Build Coastguard Worker 
1063*6777b538SAndroid Build Coastguard Worker     ret = xmlOutputBufferClose(buf);
1064*6777b538SAndroid Build Coastguard Worker     return(ret);
1065*6777b538SAndroid Build Coastguard Worker }
1066*6777b538SAndroid Build Coastguard Worker 
1067*6777b538SAndroid Build Coastguard Worker /**
1068*6777b538SAndroid Build Coastguard Worker  * htmlSaveFileFormat:
1069*6777b538SAndroid Build Coastguard Worker  * @filename:  the filename
1070*6777b538SAndroid Build Coastguard Worker  * @cur:  the document
1071*6777b538SAndroid Build Coastguard Worker  * @format:  should formatting spaces been added
1072*6777b538SAndroid Build Coastguard Worker  * @encoding: the document encoding
1073*6777b538SAndroid Build Coastguard Worker  *
1074*6777b538SAndroid Build Coastguard Worker  * Dump an HTML document to a file using a given encoding.
1075*6777b538SAndroid Build Coastguard Worker  *
1076*6777b538SAndroid Build Coastguard Worker  * returns: the number of byte written or -1 in case of failure.
1077*6777b538SAndroid Build Coastguard Worker  */
1078*6777b538SAndroid Build Coastguard Worker int
htmlSaveFileFormat(const char * filename,xmlDocPtr cur,const char * encoding,int format)1079*6777b538SAndroid Build Coastguard Worker htmlSaveFileFormat(const char *filename, xmlDocPtr cur,
1080*6777b538SAndroid Build Coastguard Worker 	           const char *encoding, int format) {
1081*6777b538SAndroid Build Coastguard Worker     xmlOutputBufferPtr buf;
1082*6777b538SAndroid Build Coastguard Worker     xmlCharEncodingHandlerPtr handler = NULL;
1083*6777b538SAndroid Build Coastguard Worker     int ret;
1084*6777b538SAndroid Build Coastguard Worker 
1085*6777b538SAndroid Build Coastguard Worker     if ((cur == NULL) || (filename == NULL))
1086*6777b538SAndroid Build Coastguard Worker         return(-1);
1087*6777b538SAndroid Build Coastguard Worker 
1088*6777b538SAndroid Build Coastguard Worker     xmlInitParser();
1089*6777b538SAndroid Build Coastguard Worker 
1090*6777b538SAndroid Build Coastguard Worker     handler = htmlFindOutputEncoder(encoding);
1091*6777b538SAndroid Build Coastguard Worker     if (handler != NULL)
1092*6777b538SAndroid Build Coastguard Worker         htmlSetMetaEncoding(cur, (const xmlChar *) handler->name);
1093*6777b538SAndroid Build Coastguard Worker     else
1094*6777b538SAndroid Build Coastguard Worker 	htmlSetMetaEncoding(cur, (const xmlChar *) "UTF-8");
1095*6777b538SAndroid Build Coastguard Worker 
1096*6777b538SAndroid Build Coastguard Worker     /*
1097*6777b538SAndroid Build Coastguard Worker      * save the content to a temp buffer.
1098*6777b538SAndroid Build Coastguard Worker      */
1099*6777b538SAndroid Build Coastguard Worker     buf = xmlOutputBufferCreateFilename(filename, handler, 0);
1100*6777b538SAndroid Build Coastguard Worker     if (buf == NULL) return(0);
1101*6777b538SAndroid Build Coastguard Worker 
1102*6777b538SAndroid Build Coastguard Worker     htmlDocContentDumpFormatOutput(buf, cur, encoding, format);
1103*6777b538SAndroid Build Coastguard Worker 
1104*6777b538SAndroid Build Coastguard Worker     ret = xmlOutputBufferClose(buf);
1105*6777b538SAndroid Build Coastguard Worker     return(ret);
1106*6777b538SAndroid Build Coastguard Worker }
1107*6777b538SAndroid Build Coastguard Worker 
1108*6777b538SAndroid Build Coastguard Worker /**
1109*6777b538SAndroid Build Coastguard Worker  * htmlSaveFileEnc:
1110*6777b538SAndroid Build Coastguard Worker  * @filename:  the filename
1111*6777b538SAndroid Build Coastguard Worker  * @cur:  the document
1112*6777b538SAndroid Build Coastguard Worker  * @encoding: the document encoding
1113*6777b538SAndroid Build Coastguard Worker  *
1114*6777b538SAndroid Build Coastguard Worker  * Dump an HTML document to a file using a given encoding
1115*6777b538SAndroid Build Coastguard Worker  * and formatting returns/spaces are added.
1116*6777b538SAndroid Build Coastguard Worker  *
1117*6777b538SAndroid Build Coastguard Worker  * returns: the number of byte written or -1 in case of failure.
1118*6777b538SAndroid Build Coastguard Worker  */
1119*6777b538SAndroid Build Coastguard Worker int
htmlSaveFileEnc(const char * filename,xmlDocPtr cur,const char * encoding)1120*6777b538SAndroid Build Coastguard Worker htmlSaveFileEnc(const char *filename, xmlDocPtr cur, const char *encoding) {
1121*6777b538SAndroid Build Coastguard Worker     return(htmlSaveFileFormat(filename, cur, encoding, 1));
1122*6777b538SAndroid Build Coastguard Worker }
1123*6777b538SAndroid Build Coastguard Worker 
1124*6777b538SAndroid Build Coastguard Worker #endif /* LIBXML_OUTPUT_ENABLED */
1125*6777b538SAndroid Build Coastguard Worker 
1126*6777b538SAndroid Build Coastguard Worker #endif /* LIBXML_HTML_ENABLED */
1127