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);