]>
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                     // find next matching tag 
 121                     int tag_len 
= wxStrlen(tagBuffer
); 
 124                         // find the ending tag 
 125                         while (pos 
+ 1 < lng 
&& 
 126                                (src
[pos
] != '<' || src
[pos
+1] != '/')) 
 133                         while (pos 
< lng 
&& match_pos 
< tag_len 
&& src
[pos
] != '>' && src
[pos
] != '<') { 
 134                             // cast to wxChar needed to suppress warning in 
 136                             if ((wxChar
)wxToupper(src
[pos
]) == tagBuffer
[match_pos
]) { 
 139                             else if (src
[pos
] == wxT(' ') || src
[pos
] == wxT('\n') || 
 140                                 src
[pos
] == wxT('\r') || src
[pos
] == wxT('\t')) { 
 141                                 // need to skip over these 
 150                         if (match_pos 
== tag_len
) { 
 151                             pos 
= pos 
- tag_len 
- 3; 
 165     // ok, we're done, now we'll free .Name members of cache - we don't need it anymore: 
 166     for (int i 
= 0; i 
< m_CacheSize
; i
++) 
 168         delete[] m_Cache
[i
].Name
; 
 169         m_Cache
[i
].Name 
= NULL
; 
 173 void wxHtmlTagsCache::QueryTag(int at
, int* end1
, int* end2
) 
 175     if (m_Cache 
== NULL
) return; 
 176     if (m_Cache
[m_CachePos
].Key 
!= at
) 
 178         int delta 
= (at 
< m_Cache
[m_CachePos
].Key
) ? -1 : 1; 
 183         while (m_Cache
[m_CachePos
].Key 
!= at
); 
 185     *end1 
= m_Cache
[m_CachePos
].End1
; 
 186     *end2 
= m_Cache
[m_CachePos
].End2
; 
 192 //----------------------------------------------------------------------------- 
 194 //----------------------------------------------------------------------------- 
 196 IMPLEMENT_CLASS(wxHtmlTag
,wxObject
) 
 198 wxHtmlTag::wxHtmlTag(wxHtmlTag 
*parent
, 
 199                      const wxString
& source
, int pos
, int end_pos
, 
 200                      wxHtmlTagsCache 
*cache
, 
 201                      wxHtmlEntitiesParser 
*entParser
) : wxObject() 
 203     /* Setup DOM relations */ 
 206     m_FirstChild 
= m_LastChild 
= NULL
; 
 210         m_Prev 
= m_Parent
->m_LastChild
; 
 212             m_Parent
->m_FirstChild 
= this; 
 214             m_Prev
->m_Next 
= this; 
 215         m_Parent
->m_LastChild 
= this; 
 220     /* Find parameters and their values: */ 
 225     // fill-in name, params and begin pos: 
 228     // find tag's name and convert it to uppercase: 
 229     while ((i 
< end_pos
) && 
 230            ((c 
= source
[i
++]) != wxT(' ') && c 
!= wxT('\r') && 
 231              c 
!= wxT('\n') && c 
!= wxT('\t') && 
 234         if ((c 
>= wxT('a')) && (c 
<= wxT('z'))) 
 235             c 
-= (wxT('a') - wxT('A')); 
 239     // if the tag has parameters, read them and "normalize" them, 
 240     // i.e. convert to uppercase, replace whitespaces by spaces and 
 241     // remove whitespaces around '=': 
 242     if (source
[i
-1] != wxT('>')) 
 244         #define IS_WHITE(c) (c == wxT(' ') || c == wxT('\r') || \ 
 245                              c == wxT('\n') || c == wxT('\t')) 
 246         wxString pname
, pvalue
; 
 258         state 
= ST_BEFORE_NAME
; 
 263             if (c 
== wxT('>') && !(state 
== ST_VALUE 
&& quote 
!= 0)) 
 265                 if (state 
== ST_BEFORE_EQ 
|| state 
== ST_NAME
) 
 267                     m_ParamNames
.Add(pname
); 
 268                     m_ParamValues
.Add(wxEmptyString
); 
 270                 else if (state 
== ST_VALUE 
&& quote 
== 0) 
 272                     m_ParamNames
.Add(pname
); 
 274                         m_ParamValues
.Add(entParser
->Parse(pvalue
)); 
 276                         m_ParamValues
.Add(pvalue
); 
 291                         state 
= ST_BEFORE_EQ
; 
 292                     else if (c 
== wxT('=')) 
 293                         state 
= ST_BEFORE_VALUE
; 
 299                         state 
= ST_BEFORE_VALUE
; 
 300                     else if (!IS_WHITE(c
)) 
 302                         m_ParamNames
.Add(pname
); 
 303                         m_ParamValues
.Add(wxEmptyString
); 
 308                 case ST_BEFORE_VALUE
: 
 311                         if (c 
== wxT('"') || c 
== wxT('\'')) 
 312                             quote 
= c
, pvalue 
= wxEmptyString
; 
 314                             quote 
= 0, pvalue 
= c
; 
 319                     if ((quote 
!= 0 && c 
== quote
) || 
 320                         (quote 
== 0 && IS_WHITE(c
))) 
 322                         m_ParamNames
.Add(pname
); 
 325                             // VS: backward compatibility, no real reason, 
 326                             //     but wxHTML code relies on this... :( 
 330                             m_ParamValues
.Add(entParser
->Parse(pvalue
)); 
 332                             m_ParamValues
.Add(pvalue
); 
 333                         state 
= ST_BEFORE_NAME
; 
 345    cache
->QueryTag(pos
, &m_End1
, &m_End2
); 
 346    if (m_End1 
> end_pos
) m_End1 
= end_pos
; 
 347    if (m_End2 
> end_pos
) m_End2 
= end_pos
; 
 350 wxHtmlTag::~wxHtmlTag() 
 356         t2 
= t1
->GetNextSibling(); 
 362 bool wxHtmlTag::HasParam(const wxString
& par
) const 
 364     return (m_ParamNames
.Index(par
, false) != wxNOT_FOUND
); 
 367 wxString 
wxHtmlTag::GetParam(const wxString
& par
, bool with_commas
) const 
 369     int index 
= m_ParamNames
.Index(par
, false); 
 370     if (index 
== wxNOT_FOUND
) 
 371         return wxEmptyString
; 
 374         // VS: backward compatibility, seems to be never used by wxHTML... 
 376         s 
<< wxT('"') << m_ParamValues
[index
] << wxT('"'); 
 380         return m_ParamValues
[index
]; 
 383 int wxHtmlTag::ScanParam(const wxString
& par
, 
 384                          const wxChar 
*format
, 
 387     wxString parval 
= GetParam(par
); 
 388     return wxSscanf(parval
, format
, param
); 
 391 bool wxHtmlTag::GetParamAsColour(const wxString
& par
, wxColour 
*clr
) const 
 393     wxString str 
= GetParam(par
); 
 395     if (str
.empty()) return false; 
 396     if (str
.GetChar(0) == wxT('#')) 
 399         if (ScanParam(par
, wxT("#%lX"), &tmp
) != 1) 
 401         *clr 
= wxColour((unsigned char)((tmp 
& 0xFF0000) >> 16), 
 402                         (unsigned char)((tmp 
& 0x00FF00) >> 8), 
 403                         (unsigned char)(tmp 
& 0x0000FF)); 
 408         // Handle colours defined in HTML 4.0: 
 409         #define HTML_COLOUR(name,r,g,b)                 \ 
 410             if (str.IsSameAs(wxT(name), false))         \ 
 411                 { *clr = wxColour(r,g,b); return true; } 
 412         HTML_COLOUR("black",   0x00,0x00,0x00) 
 413         HTML_COLOUR("silver",  0xC0,0xC0,0xC0) 
 414         HTML_COLOUR("gray",    0x80,0x80,0x80) 
 415         HTML_COLOUR("white",   0xFF,0xFF,0xFF) 
 416         HTML_COLOUR("maroon",  0x80,0x00,0x00) 
 417         HTML_COLOUR("red",     0xFF,0x00,0x00) 
 418         HTML_COLOUR("purple",  0x80,0x00,0x80) 
 419         HTML_COLOUR("fuchsia", 0xFF,0x00,0xFF) 
 420         HTML_COLOUR("green",   0x00,0x80,0x00) 
 421         HTML_COLOUR("lime",    0x00,0xFF,0x00) 
 422         HTML_COLOUR("olive",   0x80,0x80,0x00) 
 423         HTML_COLOUR("yellow",  0xFF,0xFF,0x00) 
 424         HTML_COLOUR("navy",    0x00,0x00,0x80) 
 425         HTML_COLOUR("blue",    0x00,0x00,0xFF) 
 426         HTML_COLOUR("teal",    0x00,0x80,0x80) 
 427         HTML_COLOUR("aqua",    0x00,0xFF,0xFF) 
 434 bool wxHtmlTag::GetParamAsInt(const wxString
& par
, int *clr
) const 
 436     if (!HasParam(par
)) return false; 
 438     bool succ 
= GetParam(par
).ToLong(&i
); 
 443 wxString 
wxHtmlTag::GetAllParams() const 
 445     // VS: this function is for backward compatiblity only, 
 446     //     never used by wxHTML 
 448     size_t cnt 
= m_ParamNames
.GetCount(); 
 449     for (size_t i 
= 0; i 
< cnt
; i
++) 
 451         s 
<< m_ParamNames
[i
]; 
 453         if (m_ParamValues
[i
].Find(wxT('"')) != wxNOT_FOUND
) 
 454             s 
<< wxT('\'') << m_ParamValues
[i
] << wxT('\''); 
 456             s 
<< wxT('"') << m_ParamValues
[i
] << wxT('"'); 
 461 wxHtmlTag 
*wxHtmlTag::GetFirstSibling() const 
 464         return m_Parent
->m_FirstChild
; 
 467         wxHtmlTag 
*cur 
= (wxHtmlTag
*)this; 
 474 wxHtmlTag 
*wxHtmlTag::GetLastSibling() const 
 477         return m_Parent
->m_LastChild
; 
 480         wxHtmlTag 
*cur 
= (wxHtmlTag
*)this; 
 487 wxHtmlTag 
*wxHtmlTag::GetNextTag() const 
 489     if (m_FirstChild
) return m_FirstChild
; 
 490     if (m_Next
) return m_Next
; 
 491     wxHtmlTag 
*cur 
= m_Parent
; 
 492     if (!cur
) return NULL
; 
 493     while (cur
->m_Parent 
&& !cur
->m_Next
) 
 498 #if WXWIN_COMPATIBILITY_2_2 
 500 bool wxHtmlTag::IsEnding() const 
 505 #endif // WXWIN_COMPATIBILITY_2_2