1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxHtmlParser class (generic parser) 
   4 // Author:      Vaclav Slavik 
   6 // Copyright:   (c) 1999 Vaclav Slavik 
   7 // Licence:     wxWindows Licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  12 #pragma implementation "winpars.h" 
  15 #include "wx/wxprec.h" 
  18 #if wxUSE_HTML && wxUSE_STREAMS 
  29 #include "wx/html/htmldefs.h" 
  30 #include "wx/html/winpars.h" 
  31 #include "wx/html/htmlwin.h" 
  32 #include "wx/fontmap.h" 
  36 //----------------------------------------------------------------------------- 
  38 //----------------------------------------------------------------------------- 
  41 wxList 
wxHtmlWinParser::m_Modules
; 
  43 wxHtmlWinParser::wxHtmlWinParser(wxHtmlWindow 
*wnd
) : wxHtmlParser() 
  50     m_CharHeight 
= m_CharWidth 
= 0; 
  53     m_InputEnc 
= m_OutputEnc 
= wxFONTENCODING_DEFAULT
; 
  57         for (i 
= 0; i 
< 2; i
++) 
  58             for (j 
= 0; j 
< 2; j
++) 
  59                 for (k 
= 0; k 
< 2; k
++) 
  60                     for (l 
= 0; l 
< 2; l
++) 
  61                         for (m 
= 0; m 
< 7; m
++) 
  63                             m_FontsTable
[i
][j
][k
][l
][m
] = NULL
; 
  64                             m_FontsFacesTable
[i
][j
][k
][l
][m
] = wxEmptyString
; 
  65                             m_FontsEncTable
[i
][j
][k
][l
][m
] = wxFONTENCODING_DEFAULT
; 
  68         static int default_sizes
[7] = {7, 8, 10, 12, 16, 22, 30}; 
  69 #elif defined(__WXMAC__) 
  70         static int default_sizes
[7] = {9, 12, 14, 18, 24, 30, 36}; 
  72         static int default_sizes
[7] = {10, 12, 14, 16, 19, 24, 32}; 
  74         SetFonts("", "", default_sizes
); 
  77     // fill in wxHtmlParser's tables: 
  78     wxNode 
*node 
= m_Modules
.GetFirst(); 
  81         wxHtmlTagsModule 
*mod 
= (wxHtmlTagsModule
*) node
->GetData(); 
  82         mod
->FillHandlersTable(this); 
  83         node 
= node
->GetNext(); 
  87 wxHtmlWinParser::~wxHtmlWinParser() 
  91     for (i 
= 0; i 
< 2; i
++) 
  92         for (j 
= 0; j 
< 2; j
++) 
  93             for (k 
= 0; k 
< 2; k
++) 
  94                 for (l 
= 0; l 
< 2; l
++) 
  95                     for (m 
= 0; m 
< 7; m
++) 
  97                         if (m_FontsTable
[i
][j
][k
][l
][m
] != NULL
) 
  98                             delete m_FontsTable
[i
][j
][k
][l
][m
]; 
 101     delete[] m_tmpStrBuf
; 
 104 void wxHtmlWinParser::AddModule(wxHtmlTagsModule 
*module) 
 106     m_Modules
.Append(module); 
 109 void wxHtmlWinParser::RemoveModule(wxHtmlTagsModule 
*module) 
 111     m_Modules
.DeleteObject(module); 
 114 void wxHtmlWinParser::SetFonts(wxString normal_face
, wxString fixed_face
, const int *sizes
) 
 118     for (i 
= 0; i 
< 7; i
++) m_FontsSizes
[i
] = sizes
[i
]; 
 119     m_FontFaceFixed 
= fixed_face
; 
 120     m_FontFaceNormal 
= normal_face
; 
 122     SetInputEncoding(m_InputEnc
); 
 124     for (i 
= 0; i 
< 2; i
++) 
 125         for (j 
= 0; j 
< 2; j
++) 
 126             for (k 
= 0; k 
< 2; k
++) 
 127                 for (l 
= 0; l 
< 2; l
++) 
 128                     for (m 
= 0; m 
< 7; m
++) { 
 129                         if (m_FontsTable
[i
][j
][k
][l
][m
] != NULL
) 
 131                             delete m_FontsTable
[i
][j
][k
][l
][m
]; 
 132                             m_FontsTable
[i
][j
][k
][l
][m
] = NULL
; 
 137 void wxHtmlWinParser::InitParser(const wxString
& source
) 
 139     wxHtmlParser::InitParser(source
); 
 140     wxASSERT_MSG(m_DC 
!= NULL
, _T("no DC assigned to wxHtmlWinParser!!")); 
 142     m_FontBold 
= m_FontItalic 
= m_FontUnderlined 
= m_FontFixed 
= FALSE
; 
 143     m_FontSize 
= 3; //default one 
 144     CreateCurrentFont();           // we're selecting default font into 
 145     m_DC
->GetTextExtent("H", &m_CharWidth
, &m_CharHeight
); 
 146                 /* NOTE : we're not using GetCharWidth/Height() because 
 147                    of differences under X and win 
 151     m_Link 
= wxHtmlLinkInfo("", ""); 
 152     m_LinkColor
.Set(0, 0, 0xFF); 
 153     m_ActualColor
.Set(0, 0, 0); 
 154     m_Align 
= wxHTML_ALIGN_LEFT
; 
 155     m_tmpLastWasSpace 
= FALSE
; 
 160     m_Container
->InsertCell(new wxHtmlColourCell(m_ActualColor
)); 
 161     m_Container
->InsertCell(new wxHtmlFontCell(CreateCurrentFont())); 
 164 void wxHtmlWinParser::DoneParser() 
 167     SetInputEncoding(wxFONTENCODING_DEFAULT
); // for next call 
 168     wxHtmlParser::DoneParser(); 
 171 wxObject
* wxHtmlWinParser::GetProduct() 
 173     wxHtmlContainerCell 
*top
; 
 179     while (top
->GetParent()) top 
= top
->GetParent(); 
 183 wxFSFile 
*wxHtmlWinParser::OpenURL(wxHtmlURLType type
, 
 184                                    const wxString
& url
) const 
 186     // FIXME - normalize the URL to full path before passing to 
 191         wxHtmlOpeningStatus status
; 
 195             status 
= m_Window
->OnOpeningURL(type
, myurl
, &redirect
); 
 196             if ( status 
!= wxHTML_REDIRECT 
) 
 202         if ( status 
== wxHTML_BLOCK 
) 
 205         return GetFS()->OpenFile(myurl
); 
 208     return wxHtmlParser::OpenURL(type
, url
); 
 211 void wxHtmlWinParser::AddText(const wxChar
* txt
) 
 219     wxChar nbsp 
= GetEntitiesParser()->GetCharForCode(160 /* nbsp */); 
 221     if (lng
+1 > m_tmpStrBufSize
) 
 223         delete[] m_tmpStrBuf
; 
 224         m_tmpStrBuf 
= new wxChar
[lng
+1]; 
 225         m_tmpStrBufSize 
= lng
+1; 
 227     wxChar 
*temp 
= m_tmpStrBuf
; 
 229     if (m_tmpLastWasSpace
) 
 232                ((txt
[i
] == wxT('\n')) || (txt
[i
] == wxT('\r')) || (txt
[i
] == wxT(' ')) || 
 233                 (txt
[i
] == wxT('\t')))) i
++; 
 239         d 
= temp
[templen
++] = txt
[i
]; 
 240         if ((d 
== wxT('\n')) || (d 
== wxT('\r')) || (d 
== wxT(' ')) || (d 
== wxT('\t'))) 
 243             while ((i 
< lng
) && ((txt
[i
] == wxT('\n')) || (txt
[i
] == wxT('\r')) || 
 244                                  (txt
[i
] == wxT(' ')) || (txt
[i
] == wxT('\t')))) i
++, x
++; 
 250             temp
[templen
-1] = wxT(' '); 
 252 #if 0 // VS - WHY was this here?! 
 253             if (templen 
== 1) continue; 
 257                 m_EncConv
->Convert(temp
); 
 258             size_t len 
= wxStrlen(temp
); 
 259             for (size_t j 
= 0; j 
< len
; j
++) 
 262             c 
= new wxHtmlWordCell(temp
, *(GetDC())); 
 265             m_Container
->InsertCell(c
); 
 266             m_tmpLastWasSpace 
= TRUE
; 
 270     if (templen 
&& (templen 
> 1 || temp
[0] != wxT(' '))) 
 274             m_EncConv
->Convert(temp
); 
 275         size_t len 
= wxStrlen(temp
); 
 276         for (size_t j 
= 0; j 
< len
; j
++) 
 279         c 
= new wxHtmlWordCell(temp
, *(GetDC())); 
 282         m_Container
->InsertCell(c
); 
 283         m_tmpLastWasSpace 
= FALSE
; 
 289 wxHtmlContainerCell
* wxHtmlWinParser::OpenContainer() 
 291     m_Container 
= new wxHtmlContainerCell(m_Container
); 
 292     m_Container
->SetAlignHor(m_Align
); 
 293     m_tmpLastWasSpace 
= TRUE
; 
 294         /* to avoid space being first character in paragraph */ 
 300 wxHtmlContainerCell
* wxHtmlWinParser::SetContainer(wxHtmlContainerCell 
*c
) 
 302     m_tmpLastWasSpace 
= TRUE
; 
 303         /* to avoid space being first character in paragraph */ 
 304     return m_Container 
= c
; 
 309 wxHtmlContainerCell
* wxHtmlWinParser::CloseContainer() 
 311     m_Container 
= m_Container
->GetParent(); 
 316 void wxHtmlWinParser::SetFontSize(int s
) 
 319     else if (s 
> 7) s 
= 7; 
 325 wxFont
* wxHtmlWinParser::CreateCurrentFont() 
 327     int fb 
= GetFontBold(), 
 328         fi 
= GetFontItalic(), 
 329         fu 
= GetFontUnderlined(), 
 331         fs 
= GetFontSize() - 1 /*remap from <1;7> to <0;6>*/ ; 
 333     wxString face 
= ff 
? m_FontFaceFixed 
: m_FontFaceNormal
; 
 334     wxString 
*faceptr 
= &(m_FontsFacesTable
[fb
][fi
][fu
][ff
][fs
]); 
 335     wxFont 
**fontptr 
= &(m_FontsTable
[fb
][fi
][fu
][ff
][fs
]); 
 336     wxFontEncoding 
*encptr 
= &(m_FontsEncTable
[fb
][fi
][fu
][ff
][fs
]); 
 338     if (*fontptr 
!= NULL 
&& (*faceptr 
!= face 
|| *encptr 
!= m_OutputEnc
)) 
 344     if (*fontptr 
== NULL
) 
 347         *encptr 
= m_OutputEnc
; 
 348         *fontptr 
= new wxFont( 
 349                        (int) (m_FontsSizes
[fs
] * m_PixelScale
), 
 350                        ff 
? wxMODERN 
: wxSWISS
, 
 351                        fi 
? wxITALIC 
: wxNORMAL
, 
 352                        fb 
? wxBOLD 
: wxNORMAL
, 
 353                        fu 
? TRUE 
: FALSE
, face
, 
 356     m_DC
->SetFont(**fontptr
); 
 362 void wxHtmlWinParser::SetLink(const wxHtmlLinkInfo
& link
) 
 365     m_UseLink 
= (link
.GetHref() != wxEmptyString
); 
 369 void wxHtmlWinParser::SetFontFace(const wxString
& face
) 
 371     if (GetFontFixed()) m_FontFaceFixed 
= face
; 
 372     else m_FontFaceNormal 
= face
; 
 374     if (m_InputEnc 
!= wxFONTENCODING_DEFAULT
) 
 375         SetInputEncoding(m_InputEnc
); 
 380 void wxHtmlWinParser::SetInputEncoding(wxFontEncoding enc
) 
 382     m_InputEnc 
= m_OutputEnc 
= wxFONTENCODING_DEFAULT
; 
 389     if (enc 
== wxFONTENCODING_DEFAULT
) return; 
 391     wxFontEncoding altfix
, altnorm
; 
 392     bool availfix
, availnorm
; 
 395     availnorm 
= wxFontMapper::Get()->IsEncodingAvailable(enc
, m_FontFaceNormal
); 
 396     availfix 
= wxFontMapper::Get()->IsEncodingAvailable(enc
, m_FontFaceFixed
); 
 397     if (availnorm 
&& availfix
) 
 401     else if (wxFontMapper::Get()->GetAltForEncoding(enc
, &altnorm
, m_FontFaceNormal
, FALSE
) && 
 402              wxFontMapper::Get()->GetAltForEncoding(enc
, &altfix
, m_FontFaceFixed
, FALSE
) && 
 404         m_OutputEnc 
= altnorm
; 
 406     // at least normal face? 
 409     else if (wxFontMapper::Get()->GetAltForEncoding(enc
, &altnorm
, m_FontFaceNormal
, FALSE
)) 
 410         m_OutputEnc 
= altnorm
; 
 412     // okay, let convert to ISO_8859-1, available always 
 414         m_OutputEnc 
= wxFONTENCODING_DEFAULT
; 
 417     if (m_OutputEnc 
== wxFONTENCODING_DEFAULT
) 
 418         GetEntitiesParser()->SetEncoding(wxFONTENCODING_SYSTEM
); 
 420         GetEntitiesParser()->SetEncoding(m_OutputEnc
); 
 422     if (m_InputEnc 
== m_OutputEnc
) return; 
 424     m_EncConv 
= new wxEncodingConverter(); 
 425     if (!m_EncConv
->Init(m_InputEnc
, 
 426                            (m_OutputEnc 
== wxFONTENCODING_DEFAULT
) ? 
 427                                       wxFONTENCODING_ISO8859_1 
: m_OutputEnc
, 
 428                            wxCONVERT_SUBSTITUTE
)) 
 429     { // total failture :-( 
 430         wxLogError(_("Failed to display HTML document in %s encoding"), 
 431                    wxFontMapper::GetEncodingName(enc
).c_str()); 
 432         m_InputEnc 
= m_OutputEnc 
= wxFONTENCODING_DEFAULT
; 
 442 //----------------------------------------------------------------------------- 
 443 // wxHtmlWinTagHandler 
 444 //----------------------------------------------------------------------------- 
 446 IMPLEMENT_ABSTRACT_CLASS(wxHtmlWinTagHandler
, wxHtmlTagHandler
) 
 448 //----------------------------------------------------------------------------- 
 450 //----------------------------------------------------------------------------- 
 452 // NB: This is *NOT* winpars.cpp's initialization and shutdown code!! 
 453 //     This module is an ancestor for tag handlers modules defined 
 454 //     in m_*.cpp files with TAGS_MODULE_BEGIN...TAGS_MODULE_END construct. 
 456 //     Do not add any winpars.cpp shutdown or initialization code to it, 
 457 //     create a new module instead! 
 459 IMPLEMENT_DYNAMIC_CLASS(wxHtmlTagsModule
, wxModule
) 
 461 bool wxHtmlTagsModule::OnInit() 
 463     wxHtmlWinParser::AddModule(this); 
 467 void wxHtmlTagsModule::OnExit() 
 469     wxHtmlWinParser::RemoveModule(this);