]>
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 /////////////////////////////////////////////////////////////////////////////
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"
41 FORCE_LINK_ME(m_image
)
46 WX_DECLARE_OBJARRAY(int, CoordArray
);
47 #include "wx/arrimpl.cpp" // this is a magic incantation which must be done!
48 WX_DEFINE_OBJARRAY(CoordArray
);
51 //--------------------------------------------------------------------------------
52 // wxHtmlImageMapAreaCell
53 // 0-width, 0-height cell that represents single area in imagemap
54 // (it's GetLink is called from wxHtmlImageCell's)
55 //--------------------------------------------------------------------------------
57 class wxHtmlImageMapAreaCell
: public wxHtmlCell
60 enum celltype
{ CIRCLE
, RECT
, POLY
};
66 wxHtmlImageMapAreaCell( celltype t
, wxString
&coords
, double pixel_scale
= 1.0);
67 virtual wxHtmlLinkInfo
*GetLink( int x
= 0, int y
= 0 ) const;
74 wxHtmlImageMapAreaCell::wxHtmlImageMapAreaCell( wxHtmlImageMapAreaCell::celltype t
, wxString
&incoords
, double pixel_scale
)
77 wxString x
= incoords
, y
;
80 while ((i
= x
.Find( ',' )) != -1)
82 coords
.Add( (int)(pixel_scale
* (double)wxAtoi( x
.Left( i
).c_str())) );
85 coords
.Add( (int)(pixel_scale
* (double)wxAtoi( x
.c_str())) );
88 wxHtmlLinkInfo
*wxHtmlImageMapAreaCell::GetLink( int x
, int y
) const
100 if (x
>= l
&& x
<= r
&& y
>= t
&& y
<= b
)
114 d
= sqrt( (double) (((x
- l
) * (x
- l
)) + ((y
- t
) * (y
- t
))) );
123 if (coords
.GetCount() >= 6)
128 int totalv
= coords
.GetCount() / 2;
129 int totalc
= totalv
* 2;
130 int xval
= coords
[totalc
- 2];
131 int yval
= coords
[totalc
- 1];
135 if ((yval
>= wherey
) != (coords
[pointer
] >= wherey
))
137 if ((xval
>= wherex
) == (coords
[0] >= wherex
))
139 intersects
+= (xval
>= wherex
) ? 1 : 0;
143 intersects
+= ((xval
- (yval
- wherey
) *
145 (coords
[pointer
] - yval
)) >= wherex
) ? 1 : 0;
149 while (pointer
< end
)
151 yval
= coords
[pointer
];
155 while ((pointer
< end
) && (coords
[pointer
] >= wherey
))
163 if ((coords
[pointer
- 3] >= wherex
) ==
164 (coords
[pointer
- 1] >= wherex
)) {
165 intersects
+= (coords
[pointer
- 3] >= wherex
) ? 1 : 0;
170 ((coords
[pointer
- 3] - (coords
[pointer
- 2] - wherey
) *
171 (coords
[pointer
- 1] - coords
[pointer
- 3]) /
172 (coords
[pointer
] - coords
[pointer
- 2])) >= wherex
) ? 1 : 0;
177 while ((pointer
< end
) && (coords
[pointer
] < wherey
))
185 if ((coords
[pointer
- 3] >= wherex
) ==
186 (coords
[pointer
- 1] >= wherex
))
188 intersects
+= (coords
[pointer
- 3] >= wherex
) ? 1 : 0;
193 ((coords
[pointer
- 3] - (coords
[pointer
- 2] - wherey
) *
194 (coords
[pointer
- 1] - coords
[pointer
- 3]) /
195 (coords
[pointer
] - coords
[pointer
- 2])) >= wherex
) ? 1 : 0;
199 if ((intersects
& 1) != 0)
210 wxHtmlImageMapAreaCell
*a
= (wxHtmlImageMapAreaCell
*)m_Next
;
211 return a
->GetLink( x
, y
);
223 //--------------------------------------------------------------------------------
224 // wxHtmlImageMapCell
225 // 0-width, 0-height cell that represents map from imagemaps
226 // it is always placed before wxHtmlImageMapAreaCells
227 // It responds to Find(wxHTML_COND_ISIMAGEMAP)
228 //--------------------------------------------------------------------------------
231 class wxHtmlImageMapCell
: public wxHtmlCell
234 wxHtmlImageMapCell( wxString
&name
);
238 virtual wxHtmlLinkInfo
*GetLink( int x
= 0, int y
= 0 ) const;
239 virtual const wxHtmlCell
*Find( int cond
, const void *param
) const;
243 wxHtmlImageMapCell::wxHtmlImageMapCell( wxString
&name
)
248 wxHtmlLinkInfo
*wxHtmlImageMapCell::GetLink( int x
, int y
) const
250 wxHtmlImageMapAreaCell
*a
= (wxHtmlImageMapAreaCell
*)m_Next
;
252 return a
->GetLink( x
, y
);
253 return wxHtmlCell::GetLink( x
, y
);
256 const wxHtmlCell
*wxHtmlImageMapCell::Find( int cond
, const void *param
) const
258 if (cond
== wxHTML_COND_ISIMAGEMAP
)
260 if (m_Name
== *((wxString
*)(param
)))
263 return wxHtmlCell::Find(cond
, param
);
270 //--------------------------------------------------------------------------------
273 //--------------------------------------------------------------------------------
275 class wxHtmlImageCell
: public wxHtmlCell
278 wxHtmlImageCell(wxWindow
*window
,
279 wxFSFile
*input
, int w
= -1, int h
= -1,
280 double scale
= 1.0, int align
= wxHTML_ALIGN_BOTTOM
,
281 const wxString
& mapname
= wxEmptyString
);
283 void Draw(wxDC
& dc
, int x
, int y
, int view_y1
, int view_y2
);
284 virtual wxHtmlLinkInfo
*GetLink(int x
= 0, int y
= 0) const;
286 void SetImage(const wxImage
& img
);
287 #if wxUSE_GIF && wxUSE_TIMER
288 void AdvanceAnimation(wxTimer
*timer
);
289 virtual void Layout(int w
);
296 wxScrolledWindow
*m_window
;
297 #if wxUSE_GIF && wxUSE_TIMER
298 wxGIFDecoder
*m_gifDecoder
;
300 int m_physX
, m_physY
;
303 wxHtmlImageMapCell
*m_imageMap
;
307 #if wxUSE_GIF && wxUSE_TIMER
308 class wxGIFTimer
: public wxTimer
311 wxGIFTimer(wxHtmlImageCell
*cell
) : m_cell(cell
) {}
312 virtual void Notify()
314 m_cell
->AdvanceAnimation(this);
318 wxHtmlImageCell
*m_cell
;
323 //--------------------------------------------------------------------------------
325 //--------------------------------------------------------------------------------
328 static const char * broken_image_xpm
[] = {
337 "..................... ",
338 ".+++++++++++++++++++.. ",
339 ".+++++++++++++++++++.@. ",
340 ".++@@@@@@@@@@@@@@@@@.+@. ",
341 ".++@@@@@@@@@@@@@@@@@.++@. ",
342 ".++@@@@@.@@@@.@@@@@@.+++@. ",
343 ".++@@@@@@@@@@@@@@@@@.++++@. ",
344 ".++@@@@@@@@@@@@@@@@@.+++++@. ",
345 ".++@@.@@@@@@@@@@.@@@######## ",
346 ".++@@@@@@@@@@@@@@@@@@$$$$$$#.",
347 ".######@@@@@@@@@@@@@@@.....#.",
348 " ###@@@@@@@@@@@@@@@++#.",
349 " #####@@@@@@@@@@++#.",
352 ".+.... #@@@@@@++#.",
353 ".++@@@... ####@@++#.",
354 ".++@@@@@@.. #####.",
357 ".++@@@@@@%%%%@@.... ",
358 ".++@@@@@@%%%%@@@@@@.... ",
359 ".++@@@@@@%%%%@@@@@@@@@@.... ",
360 ".++@@@@@@@@@@@@@@@@@@@@@@++#.",
361 ".++@@@@@@@@@@@@@@@@@@@@@@++#.",
362 ".++@@@@@@@@@@@@@@@@@@@@@@++#.",
363 ".++@@@@@@@@@@@@@@@@@@@@@@++#.",
364 ".++@@@@@@@@@@@@@@@@@@@@@@++#.",
365 ".++++++++++++++++++++++++++#.",
366 ".++++++++++++++++++++++++++#.",
367 "############################."};
369 wxHtmlImageCell::wxHtmlImageCell(wxWindow
*window
, wxFSFile
*input
,
370 int w
, int h
, double scale
, int align
,
371 const wxString
& mapname
) : wxHtmlCell()
373 m_window
= window
? wxStaticCast(window
, wxScrolledWindow
) : NULL
;
381 SetCanLiveOnPagebreak(FALSE
);
382 #if wxUSE_GIF && wxUSE_TIMER
385 m_physX
= m_physY
= -1;
390 wxInputStream
*s
= input
->GetStream();
396 #if wxUSE_GIF && wxUSE_TIMER
397 if ( (input
->GetLocation().Matches(wxT("*.gif")) ||
398 input
->GetLocation().Matches(wxT("*.GIF"))) && m_window
)
400 m_gifDecoder
= new wxGIFDecoder(s
, TRUE
);
401 if ( m_gifDecoder
->ReadGIF() == wxGIF_OK
)
404 if ( m_gifDecoder
->ConvertToImage(&img
) )
409 if ( m_gifDecoder
->IsAnimation() )
411 m_gifTimer
= new wxGIFTimer(this);
412 m_gifTimer
->Start(m_gifDecoder
->GetDelay(), TRUE
);
416 wxDELETE(m_gifDecoder
);
421 wxDELETE(m_gifDecoder
);
426 #endif // wxUSE_GIF && wxUSE_TIMER
428 SetImage(wxImage(*s
, wxBITMAP_TYPE_ANY
));
432 else // input==NULL, use "broken image" bitmap
434 if ( m_bmpW
== -1 && m_bmpH
== -1 )
442 if ( m_bmpW
== -1 ) m_bmpW
= 31;
443 if ( m_bmpH
== -1 ) m_bmpH
= 33;
445 m_bitmap
= new wxBitmap(broken_image_xpm
);
448 m_Width
= (int)(scale
* (double)m_bmpW
);
449 m_Height
= (int)(scale
* (double)m_bmpH
);
453 case wxHTML_ALIGN_TOP
:
454 m_Descent
= m_Height
;
456 case wxHTML_ALIGN_CENTER
:
457 m_Descent
= m_Height
/ 2;
459 case wxHTML_ALIGN_BOTTOM
:
466 void wxHtmlImageCell::SetImage(const wxImage
& img
)
474 hh
= img
.GetHeight();
481 if ((m_bmpW
!= ww
) || (m_bmpH
!= hh
))
483 wxImage img2
= img
.Scale(m_bmpW
, m_bmpH
);
484 m_bitmap
= new wxBitmap(img2
);
487 m_bitmap
= new wxBitmap(img
);
491 #if wxUSE_GIF && wxUSE_TIMER
492 void wxHtmlImageCell::AdvanceAnimation(wxTimer
*timer
)
496 m_gifDecoder
->GoNextFrame(TRUE
);
500 m_physX
= m_physY
= 0;
501 for (wxHtmlCell
*cell
= this; cell
; cell
= cell
->GetParent())
503 m_physX
+= cell
->GetPosX();
504 m_physY
+= cell
->GetPosY();
509 m_window
->CalcScrolledPosition(m_physX
, m_physY
, &x
, &y
);
510 wxRect
rect(x
, y
, m_Width
, m_Height
);
512 if ( m_window
->GetClientRect().Intersects(rect
) &&
513 m_gifDecoder
->ConvertToImage(&img
) )
515 if ( (int)m_gifDecoder
->GetWidth() != m_Width
||
516 (int)m_gifDecoder
->GetHeight() != m_Height
||
517 m_gifDecoder
->GetLeft() != 0 || m_gifDecoder
->GetTop() != 0 )
521 dc
.SelectObject(*m_bitmap
);
522 dc
.DrawBitmap(bmp
, m_gifDecoder
->GetLeft(), m_gifDecoder
->GetTop());
526 m_window
->Refresh(img
.HasMask(), &rect
);
529 timer
->Start(m_gifDecoder
->GetDelay(), TRUE
);
532 void wxHtmlImageCell::Layout(int w
)
534 wxHtmlCell::Layout(w
);
535 m_physX
= m_physY
= -1;
540 wxHtmlImageCell::~wxHtmlImageCell()
543 #if wxUSE_GIF && wxUSE_TIMER
550 void wxHtmlImageCell::Draw(wxDC
& dc
, int x
, int y
, int WXUNUSED(view_y1
), int WXUNUSED(view_y2
))
554 dc
.SetBrush(*wxTRANSPARENT_BRUSH
);
555 dc
.SetPen(*wxBLACK_PEN
);
556 dc
.DrawRectangle(x
+ m_PosX
, y
+ m_PosY
, m_Width
, m_Height
);
562 dc
.GetUserScale(&us_x
, &us_y
);
563 dc
.SetUserScale(us_x
* m_scale
, us_y
* m_scale
);
565 dc
.DrawBitmap(*m_bitmap
, (int) ((x
+ m_PosX
) / m_scale
),
566 (int) ((y
+ m_PosY
) / m_scale
), TRUE
);
567 dc
.SetUserScale(us_x
, us_y
);
571 wxHtmlLinkInfo
*wxHtmlImageCell::GetLink( int x
, int y
) const
573 if (m_mapName
.IsEmpty())
574 return wxHtmlCell::GetLink( x
, y
);
577 wxHtmlContainerCell
*p
, *op
;
578 op
= p
= GetParent();
585 wxHtmlCell
*cell
= (wxHtmlCell
*)p
->Find(wxHTML_COND_ISIMAGEMAP
,
586 (const void*)(&m_mapName
));
589 ((wxString
&)m_mapName
).Clear();
590 return wxHtmlCell::GetLink( x
, y
);
592 { // dirty hack, ask Joel why he fills m_ImageMap in this place
593 // THE problem is that we're in const method and we can't modify m_ImageMap
594 wxHtmlImageMapCell
**cx
= (wxHtmlImageMapCell
**)(&m_imageMap
);
595 *cx
= (wxHtmlImageMapCell
*)cell
;
598 return m_imageMap
->GetLink(x
, y
);
603 //--------------------------------------------------------------------------------
605 //--------------------------------------------------------------------------------
607 TAG_HANDLER_BEGIN(IMG
, "IMG,MAP,AREA")
609 TAG_HANDLER_PROC(tag
)
611 if (tag
.GetName() == wxT("IMG"))
613 if (tag
.HasParam(wxT("SRC")))
618 wxString tmp
= tag
.GetParam(wxT("SRC"));
619 wxString mn
= wxEmptyString
;
621 str
= m_WParser
->OpenURL(wxHTML_URL_IMAGE
, tmp
);
623 if (tag
.HasParam(wxT("WIDTH")))
624 tag
.GetParamAsInt(wxT("WIDTH"), &w
);
625 if (tag
.HasParam(wxT("HEIGHT")))
626 tag
.GetParamAsInt(wxT("HEIGHT"), &h
);
627 al
= wxHTML_ALIGN_BOTTOM
;
628 if (tag
.HasParam(wxT("ALIGN")))
630 wxString alstr
= tag
.GetParam(wxT("ALIGN"));
631 alstr
.MakeUpper(); // for the case alignment was in ".."
632 if (alstr
== wxT("TEXTTOP"))
633 al
= wxHTML_ALIGN_TOP
;
634 else if ((alstr
== wxT("CENTER")) || (alstr
== wxT("ABSCENTER")))
635 al
= wxHTML_ALIGN_CENTER
;
637 if (tag
.HasParam(wxT("USEMAP")))
639 mn
= tag
.GetParam( wxT("USEMAP") );
640 if (mn
.GetChar(0) == wxT('#'))
645 wxHtmlImageCell
*cel
= new wxHtmlImageCell(
646 m_WParser
->GetWindow(),
648 m_WParser
->GetPixelScale(),
650 cel
->SetLink(m_WParser
->GetLink());
651 cel
->SetId(tag
.GetParam(wxT("id"))); // may be empty
652 m_WParser
->GetContainer()->InsertCell(cel
);
657 if (tag
.GetName() == wxT("MAP"))
659 m_WParser
->CloseContainer();
660 m_WParser
->OpenContainer();
661 if (tag
.HasParam(wxT("NAME")))
663 wxString tmp
= tag
.GetParam(wxT("NAME"));
664 wxHtmlImageMapCell
*cel
= new wxHtmlImageMapCell( tmp
);
665 m_WParser
->GetContainer()->InsertCell( cel
);
668 m_WParser
->CloseContainer();
669 m_WParser
->OpenContainer();
671 if (tag
.GetName() == wxT("AREA"))
673 if (tag
.HasParam(wxT("SHAPE")))
675 wxString tmp
= tag
.GetParam(wxT("SHAPE"));
676 wxString coords
= wxEmptyString
;
678 wxHtmlImageMapAreaCell
*cel
= NULL
;
679 if (tag
.HasParam(wxT("COORDS")))
681 coords
= tag
.GetParam(wxT("COORDS"));
683 if (tmp
== wxT("POLY"))
685 cel
= new wxHtmlImageMapAreaCell( wxHtmlImageMapAreaCell::POLY
, coords
, m_WParser
->GetPixelScale() );
687 else if (tmp
== wxT("CIRCLE"))
689 cel
= new wxHtmlImageMapAreaCell( wxHtmlImageMapAreaCell::CIRCLE
, coords
, m_WParser
->GetPixelScale() );
691 else if (tmp
== wxT("RECT"))
693 cel
= new wxHtmlImageMapAreaCell( wxHtmlImageMapAreaCell::RECT
, coords
, m_WParser
->GetPixelScale() );
695 if (cel
!= NULL
&& tag
.HasParam(wxT("HREF")))
697 wxString tmp
= tag
.GetParam(wxT("HREF"));
698 wxString target
= wxEmptyString
;
699 if (tag
.HasParam(wxT("TARGET"))) target
= tag
.GetParam(wxT("TARGET"));
700 cel
->SetLink( wxHtmlLinkInfo(tmp
, target
));
702 if (cel
!= NULL
) m_WParser
->GetContainer()->InsertCell( cel
);
713 TAGS_MODULE_BEGIN(Image
)
717 TAGS_MODULE_END(Image
)