]>
git.saurik.com Git - wxWidgets.git/blob - src/html/m_image.cpp
   1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxHtml module for displaying images 
   4 // Author:      Vaclav Slavik 
   6 // Copyright:   (c) 1999 Vaclav Slavik, Joel Lucsy 
   7 // Licence:     wxWindows Licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  10 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  11 #pragma implementation 
  14 #include "wx/wxprec.h" 
  17 #if wxUSE_HTML && wxUSE_STREAMS 
  25     #include "wx/scrolwin.h" 
  27     #include "wx/dcmemory.h" 
  30 #include "wx/html/forcelnk.h" 
  31 #include "wx/html/m_templ.h" 
  32 #include "wx/html/htmlwin.h" 
  35 #include "wx/gifdecod.h" 
  36 #include "wx/dynarray.h" 
  38 #include "wx/artprov.h" 
  43 FORCE_LINK_ME(m_image
) 
  48 WX_DECLARE_OBJARRAY(int, CoordArray
); 
  49 #include "wx/arrimpl.cpp" // this is a magic incantation which must be done! 
  50 WX_DEFINE_OBJARRAY(CoordArray
); 
  53 // --------------------------------------------------------------------------- 
  54 // wxHtmlImageMapAreaCell 
  55 //                  0-width, 0-height cell that represents single area in 
  56 //                  imagemap (it's GetLink is called from wxHtmlImageCell's) 
  57 // --------------------------------------------------------------------------- 
  59 class wxHtmlImageMapAreaCell 
: public wxHtmlCell
 
  62         enum celltype 
{ CIRCLE
, RECT
, POLY 
}; 
  68         wxHtmlImageMapAreaCell( celltype t
, wxString 
&coords
, double pixel_scale 
= 1.0); 
  69         virtual wxHtmlLinkInfo 
*GetLink( int x 
= 0, int y 
= 0 ) const; 
  70         void Draw(wxDC
& WXUNUSED(dc
), 
  71                   int WXUNUSED(x
), int WXUNUSED(y
), 
  72                   int WXUNUSED(view_y1
), int WXUNUSED(view_y2
), 
  73                   wxHtmlRenderingInfo
& WXUNUSED(info
)) {} 
  76     DECLARE_NO_COPY_CLASS(wxHtmlImageMapAreaCell
) 
  83 wxHtmlImageMapAreaCell::wxHtmlImageMapAreaCell( wxHtmlImageMapAreaCell::celltype t
, wxString 
&incoords
, double pixel_scale 
) 
  86     wxString x 
= incoords
, y
; 
  89     while ((i 
= x
.Find( ',' )) != -1) 
  91         coords
.Add( (int)(pixel_scale 
* (double)wxAtoi( x
.Left( i 
).c_str())) ); 
  94     coords
.Add( (int)(pixel_scale 
* (double)wxAtoi( x
.c_str())) ); 
  97 wxHtmlLinkInfo 
*wxHtmlImageMapAreaCell::GetLink( int x
, int y 
) const 
 109                 if (x 
>= l 
&& x 
<= r 
&& y 
>= t 
&& y 
<= b
) 
 123                 d 
= sqrt( (double) (((x 
- l
) * (x 
- l
)) + ((y 
- t
) * (y 
- t
))) ); 
 132                 if (coords
.GetCount() >= 6) 
 137                     int totalv 
= coords
.GetCount() / 2; 
 138                     int totalc 
= totalv 
* 2; 
 139                     int xval 
= coords
[totalc 
- 2]; 
 140                     int yval 
= coords
[totalc 
- 1]; 
 144                     if ((yval 
>= wherey
) != (coords
[pointer
] >= wherey
)) 
 146                         if ((xval 
>= wherex
) == (coords
[0] >= wherex
)) 
 148                             intersects 
+= (xval 
>= wherex
) ? 1 : 0; 
 152                             intersects 
+= ((xval 
- (yval 
- wherey
) * 
 154                                             (coords
[pointer
] - yval
)) >= wherex
) ? 1 : 0; 
 158                     while (pointer 
< end
) 
 160                         yval 
= coords
[pointer
]; 
 164                             while ((pointer 
< end
) && (coords
[pointer
] >= wherey
)) 
 172                             if ((coords
[pointer 
- 3] >= wherex
) == 
 173                                     (coords
[pointer 
- 1] >= wherex
)) { 
 174                                 intersects 
+= (coords
[pointer 
- 3] >= wherex
) ? 1 : 0; 
 179                                     ((coords
[pointer 
- 3] - (coords
[pointer 
- 2] - wherey
) * 
 180                                       (coords
[pointer 
- 1] - coords
[pointer 
- 3]) / 
 181                                       (coords
[pointer
] - coords
[pointer 
- 2])) >= wherex
) ? 1 : 0; 
 186                             while ((pointer 
< end
) && (coords
[pointer
] < wherey
)) 
 194                             if ((coords
[pointer 
- 3] >= wherex
) == 
 195                                     (coords
[pointer 
- 1] >= wherex
)) 
 197                                 intersects 
+= (coords
[pointer 
- 3] >= wherex
) ? 1 : 0; 
 202                                     ((coords
[pointer 
- 3] - (coords
[pointer 
- 2] - wherey
) * 
 203                                       (coords
[pointer 
- 1] - coords
[pointer 
- 3]) / 
 204                                       (coords
[pointer
] - coords
[pointer 
- 2])) >= wherex
) ? 1 : 0; 
 208                     if ((intersects 
& 1) != 0) 
 219         wxHtmlImageMapAreaCell  
*a 
= (wxHtmlImageMapAreaCell
*)m_Next
; 
 220         return a
->GetLink( x
, y 
); 
 232 //-------------------------------------------------------------------------------- 
 233 // wxHtmlImageMapCell 
 234 //                  0-width, 0-height cell that represents map from imagemaps 
 235 //                  it is always placed before wxHtmlImageMapAreaCells 
 236 //                  It responds to Find(wxHTML_COND_ISIMAGEMAP) 
 237 //-------------------------------------------------------------------------------- 
 240 class wxHtmlImageMapCell 
: public wxHtmlCell
 
 243         wxHtmlImageMapCell( wxString 
&name 
); 
 247         virtual wxHtmlLinkInfo 
*GetLink( int x 
= 0, int y 
= 0 ) const; 
 248         virtual const wxHtmlCell 
*Find( int cond
, const void *param 
) const; 
 249         void Draw(wxDC
& WXUNUSED(dc
), 
 250                   int WXUNUSED(x
), int WXUNUSED(y
), 
 251                   int WXUNUSED(view_y1
), int WXUNUSED(view_y2
), 
 252                   wxHtmlRenderingInfo
& WXUNUSED(info
)) {} 
 254     DECLARE_NO_COPY_CLASS(wxHtmlImageMapCell
) 
 258 wxHtmlImageMapCell::wxHtmlImageMapCell( wxString 
&name 
) 
 263 wxHtmlLinkInfo 
*wxHtmlImageMapCell::GetLink( int x
, int y 
) const 
 265     wxHtmlImageMapAreaCell  
*a 
= (wxHtmlImageMapAreaCell
*)m_Next
; 
 267         return a
->GetLink( x
, y 
); 
 268     return wxHtmlCell::GetLink( x
, y 
); 
 271 const wxHtmlCell 
*wxHtmlImageMapCell::Find( int cond
, const void *param 
) const 
 273     if (cond 
== wxHTML_COND_ISIMAGEMAP
) 
 275         if (m_Name 
== *((wxString
*)(param
))) 
 278     return wxHtmlCell::Find(cond
, param
); 
 285 //-------------------------------------------------------------------------------- 
 288 //-------------------------------------------------------------------------------- 
 290 class wxHtmlImageCell 
: public wxHtmlCell
 
 293     wxHtmlImageCell(wxWindow 
*window
, 
 294                     wxFSFile 
*input
, int w 
= -1, int h 
= -1, 
 295                     double scale 
= 1.0, int align 
= wxHTML_ALIGN_BOTTOM
, 
 296                     const wxString
& mapname 
= wxEmptyString
); 
 298     void Draw(wxDC
& dc
, int x
, int y
, int view_y1
, int view_y2
, 
 299               wxHtmlRenderingInfo
& info
); 
 300     virtual wxHtmlLinkInfo 
*GetLink(int x 
= 0, int y 
= 0) const; 
 302     void SetImage(const wxImage
& img
); 
 303 #if wxUSE_GIF && wxUSE_TIMER 
 304     void AdvanceAnimation(wxTimer 
*timer
); 
 305     virtual void Layout(int w
); 
 312     wxScrolledWindow   
*m_window
; 
 313 #if wxUSE_GIF && wxUSE_TIMER 
 314     wxGIFDecoder       
*m_gifDecoder
; 
 316     int                 m_physX
, m_physY
; 
 319     wxHtmlImageMapCell 
*m_imageMap
; 
 322     DECLARE_NO_COPY_CLASS(wxHtmlImageCell
) 
 325 #if wxUSE_GIF && wxUSE_TIMER 
 326 class wxGIFTimer 
: public wxTimer
 
 329         wxGIFTimer(wxHtmlImageCell 
*cell
) : m_cell(cell
) {} 
 330         virtual void Notify() 
 332             m_cell
->AdvanceAnimation(this); 
 336         wxHtmlImageCell 
*m_cell
; 
 338     DECLARE_NO_COPY_CLASS(wxGIFTimer
) 
 343 //---------------------------------------------------------------------------- 
 345 //---------------------------------------------------------------------------- 
 348 wxHtmlImageCell::wxHtmlImageCell(wxWindow 
*window
, wxFSFile 
*input
, 
 349                                  int w
, int h
, double scale
, int align
, 
 350                                  const wxString
& mapname
) : wxHtmlCell() 
 352     m_window 
= window 
? wxStaticCast(window
, wxScrolledWindow
) : NULL
; 
 360     SetCanLiveOnPagebreak(FALSE
); 
 361 #if wxUSE_GIF && wxUSE_TIMER 
 364     m_physX 
= m_physY 
= -1; 
 367     if ( m_bmpW 
&& m_bmpH 
) 
 371             wxInputStream 
*s 
= input
->GetStream(); 
 377 #if wxUSE_GIF && wxUSE_TIMER 
 378                 if ( (input
->GetLocation().Matches(wxT("*.gif")) || 
 379                       input
->GetLocation().Matches(wxT("*.GIF"))) && m_window 
) 
 381                     m_gifDecoder 
= new wxGIFDecoder(s
, TRUE
); 
 382                     if ( m_gifDecoder
->ReadGIF() == wxGIF_OK 
) 
 385                         if ( m_gifDecoder
->ConvertToImage(&img
) ) 
 390                         if ( m_gifDecoder
->IsAnimation() ) 
 392                             m_gifTimer 
= new wxGIFTimer(this); 
 393                             m_gifTimer
->Start(m_gifDecoder
->GetDelay(), TRUE
); 
 397                             wxDELETE(m_gifDecoder
); 
 402                         wxDELETE(m_gifDecoder
); 
 407 #endif // wxUSE_GIF && wxUSE_TIMER 
 409                     wxImage 
image(*s
, wxBITMAP_TYPE_ANY
); 
 415         else // input==NULL, use "broken image" bitmap 
 417             if ( m_bmpW 
== -1 && m_bmpH 
== -1 ) 
 425                 if ( m_bmpW 
== -1 ) m_bmpW 
= 31; 
 426                 if ( m_bmpH 
== -1 ) m_bmpH 
= 33; 
 429                 new wxBitmap(wxArtProvider::GetBitmap(wxART_MISSING_IMAGE
)); 
 432     //else: ignore the 0-sized images used sometimes on the Web pages 
 434     m_Width 
= (int)(scale 
* (double)m_bmpW
); 
 435     m_Height 
= (int)(scale 
* (double)m_bmpH
); 
 439         case wxHTML_ALIGN_TOP 
: 
 440             m_Descent 
= m_Height
; 
 442         case wxHTML_ALIGN_CENTER 
: 
 443             m_Descent 
= m_Height 
/ 2; 
 445         case wxHTML_ALIGN_BOTTOM 
: 
 452 void wxHtmlImageCell::SetImage(const wxImage
& img
) 
 460         hh 
= img
.GetHeight(); 
 467         if ((m_bmpW 
!= ww
) || (m_bmpH 
!= hh
)) 
 469             wxImage img2 
= img
.Scale(m_bmpW
, m_bmpH
); 
 470             m_bitmap 
= new wxBitmap(img2
); 
 473             m_bitmap 
= new wxBitmap(img
); 
 477 #if wxUSE_GIF && wxUSE_TIMER 
 478 void wxHtmlImageCell::AdvanceAnimation(wxTimer 
*timer
) 
 482     m_gifDecoder
->GoNextFrame(TRUE
); 
 486         m_physX 
= m_physY 
= 0; 
 487         for (wxHtmlCell 
*cell 
= this; cell
; cell 
= cell
->GetParent()) 
 489             m_physX 
+= cell
->GetPosX(); 
 490             m_physY 
+= cell
->GetPosY(); 
 495     m_window
->CalcScrolledPosition(m_physX
, m_physY
, &x
, &y
); 
 496     wxRect 
rect(x
, y
, m_Width
, m_Height
); 
 498     if ( m_window
->GetClientRect().Intersects(rect
) && 
 499          m_gifDecoder
->ConvertToImage(&img
) ) 
 501         if ( (int)m_gifDecoder
->GetWidth() != m_Width 
|| 
 502              (int)m_gifDecoder
->GetHeight() != m_Height 
|| 
 503              m_gifDecoder
->GetLeft() != 0 || m_gifDecoder
->GetTop() != 0 ) 
 507             dc
.SelectObject(*m_bitmap
); 
 508             dc
.DrawBitmap(bmp
, m_gifDecoder
->GetLeft(), m_gifDecoder
->GetTop(), 
 509                           TRUE 
/* use mask */); 
 513         m_window
->Refresh(img
.HasMask(), &rect
); 
 516     timer
->Start(m_gifDecoder
->GetDelay(), TRUE
); 
 519 void wxHtmlImageCell::Layout(int w
) 
 521     wxHtmlCell::Layout(w
); 
 522     m_physX 
= m_physY 
= -1; 
 527 wxHtmlImageCell::~wxHtmlImageCell() 
 530 #if wxUSE_GIF && wxUSE_TIMER 
 537 void wxHtmlImageCell::Draw(wxDC
& dc
, int x
, int y
, 
 538                            int WXUNUSED(view_y1
), int WXUNUSED(view_y2
), 
 539                            wxHtmlRenderingInfo
& WXUNUSED(info
)) 
 543         dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 544         dc
.SetPen(*wxBLACK_PEN
); 
 545         dc
.DrawRectangle(x 
+ m_PosX
, y 
+ m_PosY
, m_Width
, m_Height
); 
 551         dc
.GetUserScale(&us_x
, &us_y
); 
 552         dc
.SetUserScale(us_x 
* m_scale
, us_y 
* m_scale
); 
 554         dc
.DrawBitmap(*m_bitmap
, (int) ((x 
+ m_PosX
) / m_scale
), 
 555                                  (int) ((y 
+ m_PosY
) / m_scale
), TRUE
); 
 556         dc
.SetUserScale(us_x
, us_y
); 
 560 wxHtmlLinkInfo 
*wxHtmlImageCell::GetLink( int x
, int y 
) const 
 562     if (m_mapName
.IsEmpty()) 
 563         return wxHtmlCell::GetLink( x
, y 
); 
 566         wxHtmlContainerCell 
*p
, *op
; 
 567         op 
= p 
= GetParent(); 
 574         wxHtmlCell 
*cell 
= (wxHtmlCell
*)p
->Find(wxHTML_COND_ISIMAGEMAP
, 
 575                                                 (const void*)(&m_mapName
)); 
 578             ((wxString
&)m_mapName
).Clear(); 
 579             return wxHtmlCell::GetLink( x
, y 
); 
 581         {   // dirty hack, ask Joel why he fills m_ImageMap in this place 
 582             // THE problem is that we're in const method and we can't modify m_ImageMap 
 583             wxHtmlImageMapCell 
**cx 
= (wxHtmlImageMapCell
**)(&m_imageMap
); 
 584             *cx 
= (wxHtmlImageMapCell
*)cell
; 
 587     return m_imageMap
->GetLink(x
, y
); 
 592 //-------------------------------------------------------------------------------- 
 594 //-------------------------------------------------------------------------------- 
 596 TAG_HANDLER_BEGIN(IMG
, "IMG,MAP,AREA") 
 597     TAG_HANDLER_CONSTR(IMG
) { } 
 599     TAG_HANDLER_PROC(tag
) 
 601         if (tag
.GetName() == wxT("IMG")) 
 603             if (tag
.HasParam(wxT("SRC"))) 
 608                 wxString tmp 
= tag
.GetParam(wxT("SRC")); 
 609                 wxString mn 
= wxEmptyString
; 
 611                 str 
= m_WParser
->OpenURL(wxHTML_URL_IMAGE
, tmp
); 
 613                 if (tag
.HasParam(wxT("WIDTH"))) 
 614                     tag
.GetParamAsInt(wxT("WIDTH"), &w
); 
 615                 if (tag
.HasParam(wxT("HEIGHT"))) 
 616                     tag
.GetParamAsInt(wxT("HEIGHT"), &h
); 
 617                 al 
= wxHTML_ALIGN_BOTTOM
; 
 618                 if (tag
.HasParam(wxT("ALIGN"))) 
 620                     wxString alstr 
= tag
.GetParam(wxT("ALIGN")); 
 621                     alstr
.MakeUpper();  // for the case alignment was in ".." 
 622                     if (alstr 
== wxT("TEXTTOP")) 
 623                         al 
= wxHTML_ALIGN_TOP
; 
 624                     else if ((alstr 
== wxT("CENTER")) || (alstr 
== wxT("ABSCENTER"))) 
 625                         al 
= wxHTML_ALIGN_CENTER
; 
 627                 if (tag
.HasParam(wxT("USEMAP"))) 
 629                     mn 
= tag
.GetParam( wxT("USEMAP") ); 
 630                     if (mn
.GetChar(0) == wxT('#')) 
 635                 wxHtmlImageCell 
*cel 
= new wxHtmlImageCell( 
 636                                           m_WParser
->GetWindow(), 
 638                                           m_WParser
->GetPixelScale(), 
 640                 cel
->SetLink(m_WParser
->GetLink()); 
 641                 cel
->SetId(tag
.GetParam(wxT("id"))); // may be empty 
 642                 m_WParser
->GetContainer()->InsertCell(cel
); 
 647         if (tag
.GetName() == wxT("MAP")) 
 649             m_WParser
->CloseContainer(); 
 650             m_WParser
->OpenContainer(); 
 651             if (tag
.HasParam(wxT("NAME"))) 
 653                 wxString tmp 
= tag
.GetParam(wxT("NAME")); 
 654                 wxHtmlImageMapCell 
*cel 
= new wxHtmlImageMapCell( tmp 
); 
 655                 m_WParser
->GetContainer()->InsertCell( cel 
); 
 658             m_WParser
->CloseContainer(); 
 659             m_WParser
->OpenContainer(); 
 661         if (tag
.GetName() == wxT("AREA")) 
 663             if (tag
.HasParam(wxT("SHAPE"))) 
 665                 wxString tmp 
= tag
.GetParam(wxT("SHAPE")); 
 666                 wxString coords 
= wxEmptyString
; 
 668                 wxHtmlImageMapAreaCell 
*cel 
= NULL
; 
 669                 if (tag
.HasParam(wxT("COORDS"))) 
 671                     coords 
= tag
.GetParam(wxT("COORDS")); 
 673                 if (tmp 
== wxT("POLY")) 
 675                     cel 
= new wxHtmlImageMapAreaCell( wxHtmlImageMapAreaCell::POLY
, coords
, m_WParser
->GetPixelScale() ); 
 677                 else if (tmp 
== wxT("CIRCLE")) 
 679                     cel 
= new wxHtmlImageMapAreaCell( wxHtmlImageMapAreaCell::CIRCLE
, coords
, m_WParser
->GetPixelScale() ); 
 681                 else if (tmp 
== wxT("RECT")) 
 683                     cel 
= new wxHtmlImageMapAreaCell( wxHtmlImageMapAreaCell::RECT
, coords
, m_WParser
->GetPixelScale() ); 
 685                 if (cel 
!= NULL 
&& tag
.HasParam(wxT("HREF"))) 
 687                     wxString tmp 
= tag
.GetParam(wxT("HREF")); 
 688                     wxString target 
= wxEmptyString
; 
 689                     if (tag
.HasParam(wxT("TARGET"))) target 
= tag
.GetParam(wxT("TARGET")); 
 690                     cel
->SetLink( wxHtmlLinkInfo(tmp
, target
)); 
 692                 if (cel 
!= NULL
) m_WParser
->GetContainer()->InsertCell( cel 
); 
 703 TAGS_MODULE_BEGIN(Image
) 
 707 TAGS_MODULE_END(Image
)