]>
git.saurik.com Git - wxWidgets.git/blob - src/html/htmltag.cpp
1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxHtmlTag class (represents single tag)
4 // Author: Vaclav Slavik
6 // Copyright: (c) 1999 Vaclav Slavik
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
11 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
12 #pragma implementation "htmltag.h"
15 #include "wx/wxprec.h"
27 #include "wx/html/htmltag.h"
28 #include "wx/html/htmlpars.h"
29 #include "wx/colour.h"
30 #include <stdio.h> // for vsscanf
34 //-----------------------------------------------------------------------------
36 //-----------------------------------------------------------------------------
38 struct wxHtmlCacheItem
40 // this is "pos" value passed to wxHtmlTag's constructor.
41 // it is position of '<' character of the tag
44 // end positions for the tag:
45 // end1 is '<' of ending tag,
46 // end2 is '>' or both are
47 // -1 if there is no ending tag for this one...
48 // or -2 if this is ending tag </...>
56 IMPLEMENT_CLASS(wxHtmlTagsCache
,wxObject
)
58 #define CACHE_INCREMENT 64
60 bool wxIsCDATAElement(const wxChar
*tag
)
62 return (wxStrcmp(tag
, _T("SCRIPT")) == 0) ||
63 (wxStrcmp(tag
, _T("STYLE")) == 0);
66 wxHtmlTagsCache::wxHtmlTagsCache(const wxString
& source
)
68 const wxChar
*src
= source
.c_str();
69 int lng
= source
.Length();
70 wxChar tagBuffer
[256];
79 if (src
[pos
] == wxT('<')) // tag found:
81 if (m_CacheSize
% CACHE_INCREMENT
== 0)
82 m_Cache
= (wxHtmlCacheItem
*) realloc(m_Cache
, (m_CacheSize
+ CACHE_INCREMENT
) * sizeof(wxHtmlCacheItem
));
83 int tg
= m_CacheSize
++;
85 m_Cache
[tg
].Key
= stpos
;
89 pos
< lng
&& i
< (int)WXSIZEOF(tagBuffer
) - 1 &&
90 src
[pos
] != wxT('>') && !wxIsspace(src
[pos
]);
93 tagBuffer
[i
] = (wxChar
)wxToupper(src
[pos
]);
95 tagBuffer
[i
] = _T('\0');
97 m_Cache
[tg
].Name
= new wxChar
[i
+1];
98 memcpy(m_Cache
[tg
].Name
, tagBuffer
, (i
+1)*sizeof(wxChar
));
100 while (pos
< lng
&& src
[pos
] != wxT('>')) pos
++;
102 if (src
[stpos
+1] == wxT('/')) // ending tag:
104 m_Cache
[tg
].End1
= m_Cache
[tg
].End2
= -2;
105 // find matching begin tag:
106 for (i
= tg
; i
>= 0; i
--)
107 if ((m_Cache
[i
].End1
== -1) && (wxStrcmp(m_Cache
[i
].Name
, tagBuffer
+1) == 0))
109 m_Cache
[i
].End1
= stpos
;
110 m_Cache
[i
].End2
= pos
+ 1;
116 m_Cache
[tg
].End1
= m_Cache
[tg
].End2
= -1;
118 if (wxIsCDATAElement(tagBuffer
))
120 // store the orig pos in case we are missing the closing
122 wxInt32 old_pos
= pos
;
123 bool foundCloseTag
= false;
125 // find next matching tag
126 int tag_len
= wxStrlen(tagBuffer
);
129 // find the ending tag
130 while (pos
+ 1 < lng
&&
131 (src
[pos
] != '<' || src
[pos
+1] != '/'))
138 while (pos
< lng
&& match_pos
< tag_len
&& src
[pos
] != '>' && src
[pos
] != '<') {
139 // cast to wxChar needed to suppress warning in
141 if ((wxChar
)wxToupper(src
[pos
]) == tagBuffer
[match_pos
]) {
144 else if (src
[pos
] == wxT(' ') || src
[pos
] == wxT('\n') ||
145 src
[pos
] == wxT('\r') || src
[pos
] == wxT('\t')) {
146 // need to skip over these
155 if (match_pos
== tag_len
)
157 pos
= pos
- tag_len
- 3;
158 foundCloseTag
= true;
161 else // keep looking for the closing tag
168 // we didn't find closing tag; this means the markup
169 // is incorrect and the best thing we can do is to
170 // ignore the unclosed tag and continue parsing as if
181 // ok, we're done, now we'll free .Name members of cache - we don't need it anymore:
182 for (int i
= 0; i
< m_CacheSize
; i
++)
184 delete[] m_Cache
[i
].Name
;
185 m_Cache
[i
].Name
= NULL
;
189 void wxHtmlTagsCache::QueryTag(int at
, int* end1
, int* end2
)
191 if (m_Cache
== NULL
) return;
192 if (m_Cache
[m_CachePos
].Key
!= at
)
194 int delta
= (at
< m_Cache
[m_CachePos
].Key
) ? -1 : 1;
197 if ( m_CachePos
< 0 || m_CachePos
== m_CacheSize
)
199 // something is very wrong with HTML, give up by returning an
200 // impossibly large value which is going to be ignored by the
209 while (m_Cache
[m_CachePos
].Key
!= at
);
211 *end1
= m_Cache
[m_CachePos
].End1
;
212 *end2
= m_Cache
[m_CachePos
].End2
;
218 //-----------------------------------------------------------------------------
220 //-----------------------------------------------------------------------------
222 IMPLEMENT_CLASS(wxHtmlTag
,wxObject
)
224 wxHtmlTag::wxHtmlTag(wxHtmlTag
*parent
,
225 const wxString
& source
, int pos
, int end_pos
,
226 wxHtmlTagsCache
*cache
,
227 wxHtmlEntitiesParser
*entParser
) : wxObject()
229 /* Setup DOM relations */
232 m_FirstChild
= m_LastChild
= NULL
;
236 m_Prev
= m_Parent
->m_LastChild
;
238 m_Parent
->m_FirstChild
= this;
240 m_Prev
->m_Next
= this;
241 m_Parent
->m_LastChild
= this;
246 /* Find parameters and their values: */
251 // fill-in name, params and begin pos:
254 // find tag's name and convert it to uppercase:
255 while ((i
< end_pos
) &&
256 ((c
= source
[i
++]) != wxT(' ') && c
!= wxT('\r') &&
257 c
!= wxT('\n') && c
!= wxT('\t') &&
260 if ((c
>= wxT('a')) && (c
<= wxT('z')))
261 c
-= (wxT('a') - wxT('A'));
265 // if the tag has parameters, read them and "normalize" them,
266 // i.e. convert to uppercase, replace whitespaces by spaces and
267 // remove whitespaces around '=':
268 if (source
[i
-1] != wxT('>'))
270 #define IS_WHITE(c) (c == wxT(' ') || c == wxT('\r') || \
271 c == wxT('\n') || c == wxT('\t'))
272 wxString pname
, pvalue
;
284 state
= ST_BEFORE_NAME
;
289 if (c
== wxT('>') && !(state
== ST_VALUE
&& quote
!= 0))
291 if (state
== ST_BEFORE_EQ
|| state
== ST_NAME
)
293 m_ParamNames
.Add(pname
);
294 m_ParamValues
.Add(wxEmptyString
);
296 else if (state
== ST_VALUE
&& quote
== 0)
298 m_ParamNames
.Add(pname
);
300 m_ParamValues
.Add(entParser
->Parse(pvalue
));
302 m_ParamValues
.Add(pvalue
);
317 state
= ST_BEFORE_EQ
;
318 else if (c
== wxT('='))
319 state
= ST_BEFORE_VALUE
;
325 state
= ST_BEFORE_VALUE
;
326 else if (!IS_WHITE(c
))
328 m_ParamNames
.Add(pname
);
329 m_ParamValues
.Add(wxEmptyString
);
334 case ST_BEFORE_VALUE
:
337 if (c
== wxT('"') || c
== wxT('\''))
338 quote
= c
, pvalue
= wxEmptyString
;
340 quote
= 0, pvalue
= c
;
345 if ((quote
!= 0 && c
== quote
) ||
346 (quote
== 0 && IS_WHITE(c
)))
348 m_ParamNames
.Add(pname
);
351 // VS: backward compatibility, no real reason,
352 // but wxHTML code relies on this... :(
356 m_ParamValues
.Add(entParser
->Parse(pvalue
));
358 m_ParamValues
.Add(pvalue
);
359 state
= ST_BEFORE_NAME
;
371 cache
->QueryTag(pos
, &m_End1
, &m_End2
);
372 if (m_End1
> end_pos
) m_End1
= end_pos
;
373 if (m_End2
> end_pos
) m_End2
= end_pos
;
376 wxHtmlTag::~wxHtmlTag()
382 t2
= t1
->GetNextSibling();
388 bool wxHtmlTag::HasParam(const wxString
& par
) const
390 return (m_ParamNames
.Index(par
, false) != wxNOT_FOUND
);
393 wxString
wxHtmlTag::GetParam(const wxString
& par
, bool with_commas
) const
395 int index
= m_ParamNames
.Index(par
, false);
396 if (index
== wxNOT_FOUND
)
397 return wxEmptyString
;
400 // VS: backward compatibility, seems to be never used by wxHTML...
402 s
<< wxT('"') << m_ParamValues
[index
] << wxT('"');
406 return m_ParamValues
[index
];
409 int wxHtmlTag::ScanParam(const wxString
& par
,
410 const wxChar
*format
,
413 wxString parval
= GetParam(par
);
414 return wxSscanf(parval
, format
, param
);
417 bool wxHtmlTag::GetParamAsColour(const wxString
& par
, wxColour
*clr
) const
419 wxString str
= GetParam(par
);
421 if (str
.empty()) return false;
422 if (str
.GetChar(0) == wxT('#'))
425 if (ScanParam(par
, wxT("#%lX"), &tmp
) != 1)
427 *clr
= wxColour((unsigned char)((tmp
& 0xFF0000) >> 16),
428 (unsigned char)((tmp
& 0x00FF00) >> 8),
429 (unsigned char)(tmp
& 0x0000FF));
434 // Handle colours defined in HTML 4.0:
435 #define HTML_COLOUR(name,r,g,b) \
436 if (str.IsSameAs(wxT(name), false)) \
437 { *clr = wxColour(r,g,b); return true; }
438 HTML_COLOUR("black", 0x00,0x00,0x00)
439 HTML_COLOUR("silver", 0xC0,0xC0,0xC0)
440 HTML_COLOUR("gray", 0x80,0x80,0x80)
441 HTML_COLOUR("white", 0xFF,0xFF,0xFF)
442 HTML_COLOUR("maroon", 0x80,0x00,0x00)
443 HTML_COLOUR("red", 0xFF,0x00,0x00)
444 HTML_COLOUR("purple", 0x80,0x00,0x80)
445 HTML_COLOUR("fuchsia", 0xFF,0x00,0xFF)
446 HTML_COLOUR("green", 0x00,0x80,0x00)
447 HTML_COLOUR("lime", 0x00,0xFF,0x00)
448 HTML_COLOUR("olive", 0x80,0x80,0x00)
449 HTML_COLOUR("yellow", 0xFF,0xFF,0x00)
450 HTML_COLOUR("navy", 0x00,0x00,0x80)
451 HTML_COLOUR("blue", 0x00,0x00,0xFF)
452 HTML_COLOUR("teal", 0x00,0x80,0x80)
453 HTML_COLOUR("aqua", 0x00,0xFF,0xFF)
460 bool wxHtmlTag::GetParamAsInt(const wxString
& par
, int *clr
) const
462 if (!HasParam(par
)) return false;
464 bool succ
= GetParam(par
).ToLong(&i
);
469 wxString
wxHtmlTag::GetAllParams() const
471 // VS: this function is for backward compatibility only,
472 // never used by wxHTML
474 size_t cnt
= m_ParamNames
.GetCount();
475 for (size_t i
= 0; i
< cnt
; i
++)
477 s
<< m_ParamNames
[i
];
479 if (m_ParamValues
[i
].Find(wxT('"')) != wxNOT_FOUND
)
480 s
<< wxT('\'') << m_ParamValues
[i
] << wxT('\'');
482 s
<< wxT('"') << m_ParamValues
[i
] << wxT('"');
487 wxHtmlTag
*wxHtmlTag::GetFirstSibling() const
490 return m_Parent
->m_FirstChild
;
493 wxHtmlTag
*cur
= (wxHtmlTag
*)this;
500 wxHtmlTag
*wxHtmlTag::GetLastSibling() const
503 return m_Parent
->m_LastChild
;
506 wxHtmlTag
*cur
= (wxHtmlTag
*)this;
513 wxHtmlTag
*wxHtmlTag::GetNextTag() const
515 if (m_FirstChild
) return m_FirstChild
;
516 if (m_Next
) return m_Next
;
517 wxHtmlTag
*cur
= m_Parent
;
518 if (!cur
) return NULL
;
519 while (cur
->m_Parent
&& !cur
->m_Next
)
524 #if WXWIN_COMPATIBILITY_2_2
526 bool wxHtmlTag::IsEnding() const
531 #endif // WXWIN_COMPATIBILITY_2_2