added support for animated GIFs to wxHTML

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@11734 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Václav Slavík
2001-09-29 16:23:09 +00:00
parent e4b7b2b052
commit ea5c1679ad
2 changed files with 186 additions and 46 deletions

View File

@@ -25,6 +25,7 @@ wxHTML:
- new HTML parser with correct parsing of character entities and fixes - new HTML parser with correct parsing of character entities and fixes
to tags parsing to tags parsing
- added support for animated GIFs
2.3.1 2.3.1
----- -----

View File

@@ -22,12 +22,15 @@
#ifndef WXPRECOMP #ifndef WXPRECOMP
#include "wx/dc.h" #include "wx/dc.h"
#include "wx/scrolwin.h"
#include "wx/timer.h"
#endif #endif
#include "wx/html/forcelnk.h" #include "wx/html/forcelnk.h"
#include "wx/html/m_templ.h" #include "wx/html/m_templ.h"
#include "wx/image.h" #include "wx/image.h"
#include "wx/gifdecod.h"
#include "wx/dynarray.h" #include "wx/dynarray.h"
#include <math.h> #include <math.h>
@@ -269,53 +272,118 @@ const wxHtmlCell *wxHtmlImageMapCell::Find( int cond, const void *param ) const
class wxHtmlImageCell : public wxHtmlCell class wxHtmlImageCell : public wxHtmlCell
{ {
public: public:
wxBitmap *m_Image; wxHtmlImageCell(wxWindow *window,
double m_Scale; wxFSFile *input, int w = -1, int h = -1,
wxHtmlImageMapCell *m_ImageMap; double scale = 1.0, int align = wxHTML_ALIGN_BOTTOM,
wxString m_MapName; const wxString& mapname = wxEmptyString);
~wxHtmlImageCell();
void Draw(wxDC& dc, int x, int y, int view_y1, int view_y2);
virtual wxHtmlLinkInfo *GetLink(int x = 0, int y = 0) const;
wxHtmlImageCell(wxFSFile *input, int w = -1, int h = -1, double scale = 1.0, int align = wxHTML_ALIGN_BOTTOM, wxString mapname = wxEmptyString); void SetImage(const wxImage& img);
~wxHtmlImageCell() {if (m_Image) delete m_Image; } #if wxUSE_GIF && wxUSE_TIMER
void Draw(wxDC& dc, int x, int y, int view_y1, int view_y2); void AdvanceAnimation(wxTimer *timer);
virtual wxHtmlLinkInfo *GetLink( int x = 0, int y = 0 ) const; virtual void Layout(int w);
#endif
private:
wxBitmap *m_bitmap;
int m_bmpW, m_bmpH;
wxScrolledWindow *m_window;
#if wxUSE_GIF && wxUSE_TIMER
wxGIFDecoder *m_gifDecoder;
wxTimer *m_gifTimer;
int m_physX, m_physY;
#endif
double m_scale;
wxHtmlImageMapCell *m_imageMap;
wxString m_mapName;
}; };
#if wxUSE_GIF && wxUSE_TIMER
class wxGIFTimer : public wxTimer
{
public:
wxGIFTimer(wxHtmlImageCell *cell) : m_cell(cell) {}
virtual void Notify()
{
m_cell->AdvanceAnimation(this);
}
private:
wxHtmlImageCell *m_cell;
};
#endif
//-------------------------------------------------------------------------------- //--------------------------------------------------------------------------------
// wxHtmlImageCell // wxHtmlImageCell
//-------------------------------------------------------------------------------- //--------------------------------------------------------------------------------
wxHtmlImageCell::wxHtmlImageCell(wxFSFile *input, int w, int h, double scale, int align, wxString mapname) : wxHtmlCell() wxHtmlImageCell::wxHtmlImageCell(wxWindow *window, wxFSFile *input,
int w, int h, double scale, int align,
const wxString& mapname) : wxHtmlCell()
{ {
wxImage *img; m_window = wxStaticCast(window, wxScrolledWindow);
int ww, hh, bw, bh; m_scale = scale;
m_bitmap = NULL;
m_bmpW = w;
m_bmpH = h;
m_imageMap = NULL;
m_mapName = mapname;
SetCanLiveOnPagebreak(FALSE);
#if wxUSE_GIF && wxUSE_TIMER
m_gifDecoder = NULL;
m_gifTimer = NULL;
m_physX = m_physY = -1;
#endif
wxInputStream *s = input->GetStream(); wxInputStream *s = input->GetStream();
m_Scale = scale; if ( s )
img = new wxImage(*s, wxBITMAP_TYPE_ANY);
m_Image = NULL;
if (img && (img->Ok()))
{ {
ww = img->GetWidth(); bool readImg = TRUE;
hh = img->GetHeight();
if (w != -1) bw = w; else bw = ww; #if wxUSE_GIF && wxUSE_TIMER
if (h != -1) bh = h; else bh = hh; if ( (input->GetLocation().Matches(wxT("*.gif")) ||
input->GetLocation().Matches(wxT("*.GIF"))) && m_window )
m_Width = (int)(scale * (double)bw);
m_Height = (int)(scale * (double)bh);
if ((bw != ww) || (bh != hh))
{ {
wxImage img2 = img->Scale(bw, bh); m_gifDecoder = new wxGIFDecoder(s, TRUE);
m_Image = new wxBitmap(img2.ConvertToBitmap()); if ( m_gifDecoder->ReadGIF() == wxGIF_OK )
{
wxImage img;
if ( m_gifDecoder->ConvertToImage(&img) )
SetImage(img);
readImg = FALSE;
if ( m_gifDecoder->IsAnimation() )
{
m_gifTimer = new wxGIFTimer(this);
m_gifTimer->Start(m_gifDecoder->GetDelay(), TRUE);
}
else
{
wxDELETE(m_gifDecoder);
}
}
else
{
wxDELETE(m_gifDecoder);
}
}
if ( readImg )
#endif
{
SetImage(wxImage(*s, wxBITMAP_TYPE_ANY));
} }
else
m_Image = new wxBitmap(img->ConvertToBitmap());
delete img;
} }
m_Width = (int)(scale * (double)m_bmpW);
m_Height = (int)(scale * (double)m_bmpH);
switch (align) switch (align)
{ {
case wxHTML_ALIGN_TOP : case wxHTML_ALIGN_TOP :
@@ -329,34 +397,101 @@ wxHtmlImageCell::wxHtmlImageCell(wxFSFile *input, int w, int h, double scale, in
m_Descent = 0; m_Descent = 0;
break; break;
} }
}
m_ImageMap = NULL; void wxHtmlImageCell::SetImage(const wxImage& img)
m_MapName = mapname; {
SetCanLiveOnPagebreak(FALSE); if ( img.Ok() )
{
delete m_bitmap;
int ww, hh;
ww = img.GetWidth();
hh = img.GetHeight();
if ( m_bmpW == -1 )
m_bmpW = ww;
if ( m_bmpH == -1 )
m_bmpH = hh;
if ((m_bmpW != ww) || (m_bmpH != hh))
{
wxImage img2 = img.Scale(m_bmpW, m_bmpH);
m_bitmap = new wxBitmap(img2);
}
else
m_bitmap = new wxBitmap(img);
}
} }
#if wxUSE_GIF && wxUSE_TIMER
void wxHtmlImageCell::AdvanceAnimation(wxTimer *timer)
{
wxImage img;
m_gifDecoder->GoNextFrame(TRUE);
if ( m_physX == -1 )
{
m_physX = m_physY = 0;
for (wxHtmlCell *cell = this; cell; cell = cell->GetParent())
{
m_physX += cell->GetPosX();
m_physY += cell->GetPosY();
}
}
int x, y;
m_window->CalcScrolledPosition(m_physX, m_physY, &x, &y);
wxRect rect(x, y, m_Width, m_Height);
if ( m_window->GetClientRect().Intersects(rect) &&
m_gifDecoder->ConvertToImage(&img) )
{
SetImage(img);
m_window->Refresh(img.HasMask(), &rect);
}
timer->Start(m_gifDecoder->GetDelay(), TRUE);
}
void wxHtmlImageCell::Layout(int w)
{
wxHtmlCell::Layout(w);
m_physX = m_physY = -1;
}
#endif
wxHtmlImageCell::~wxHtmlImageCell()
{
delete m_bitmap;
#if wxUSE_GIF && wxUSE_TIMER
delete m_gifTimer;
delete m_gifDecoder;
#endif
}
void wxHtmlImageCell::Draw(wxDC& dc, int x, int y, int WXUNUSED(view_y1), int WXUNUSED(view_y2)) void wxHtmlImageCell::Draw(wxDC& dc, int x, int y, int WXUNUSED(view_y1), int WXUNUSED(view_y2))
{ {
if (m_Image) if (m_bitmap)
{ {
double us_x, us_y; double us_x, us_y;
dc.GetUserScale(&us_x, &us_y); dc.GetUserScale(&us_x, &us_y);
dc.SetUserScale(us_x * m_Scale, us_y * m_Scale); dc.SetUserScale(us_x * m_scale, us_y * m_scale);
// dc.DrawBitmap(*m_Image, x + m_PosX, y + m_PosY, (m_Image->GetMask() != (wxMask*) 0)); dc.DrawBitmap(*m_bitmap, (int) ((x + m_PosX) / m_scale),
dc.DrawBitmap(*m_Image, (int) ((x + m_PosX) / m_Scale), (int) ((y + m_PosY) / m_scale), TRUE);
(int) ((y + m_PosY) / m_Scale), TRUE);
dc.SetUserScale(us_x, us_y); dc.SetUserScale(us_x, us_y);
} }
} }
wxHtmlLinkInfo *wxHtmlImageCell::GetLink( int x, int y ) const wxHtmlLinkInfo *wxHtmlImageCell::GetLink( int x, int y ) const
{ {
if (m_MapName.IsEmpty()) if (m_mapName.IsEmpty())
return wxHtmlCell::GetLink( x, y ); return wxHtmlCell::GetLink( x, y );
if (!m_ImageMap) if (!m_imageMap)
{ {
wxHtmlContainerCell *p, *op; wxHtmlContainerCell *p, *op;
op = p = GetParent(); op = p = GetParent();
@@ -366,19 +501,20 @@ wxHtmlLinkInfo *wxHtmlImageCell::GetLink( int x, int y ) const
p = p->GetParent(); p = p->GetParent();
} }
p = op; p = op;
wxHtmlCell *cell = (wxHtmlCell*)p->Find( wxHTML_COND_ISIMAGEMAP, (const void*)(&m_MapName)); wxHtmlCell *cell = (wxHtmlCell*)p->Find(wxHTML_COND_ISIMAGEMAP,
(const void*)(&m_mapName));
if (!cell) if (!cell)
{ {
((wxString&)m_MapName).Clear(); ((wxString&)m_mapName).Clear();
return wxHtmlCell::GetLink( x, y ); return wxHtmlCell::GetLink( x, y );
} }
{ // dirty hack, ask Joel why he fills m_ImageMap in this place { // dirty hack, ask Joel why he fills m_ImageMap in this place
// THE problem is that we're in const method and we can't modify m_ImageMap // THE problem is that we're in const method and we can't modify m_ImageMap
wxHtmlImageMapCell **cx = (wxHtmlImageMapCell**)(&m_ImageMap); wxHtmlImageMapCell **cx = (wxHtmlImageMapCell**)(&m_imageMap);
*cx = (wxHtmlImageMapCell*)cell; *cx = (wxHtmlImageMapCell*)cell;
} }
} }
return m_ImageMap->GetLink( x, y ); return m_imageMap->GetLink(x, y);
} }
@@ -427,7 +563,10 @@ TAG_HANDLER_BEGIN(IMG, "IMG,MAP,AREA")
wxHtmlImageCell *cel = NULL; wxHtmlImageCell *cel = NULL;
if (str) if (str)
{ {
cel = new wxHtmlImageCell(str, w, h, m_WParser->GetPixelScale(), al, mn); cel = new wxHtmlImageCell(m_WParser->GetWindow(),
str, w, h,
m_WParser->GetPixelScale(),
al, mn);
cel->SetLink(m_WParser->GetLink()); cel->SetLink(m_WParser->GetLink());
cel->SetId(tag.GetParam(wxT("id"))); // may be empty cel->SetId(tag.GetParam(wxT("id"))); // may be empty
m_WParser->GetContainer()->InsertCell(cel); m_WParser->GetContainer()->InsertCell(cel);