]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/dfb/dc.cpp
Added wxRichTextTableBlock class to help with table UI operations
[wxWidgets.git] / src / dfb / dc.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/dfb/dc.cpp
3// Purpose: wxDFBDCImpl class
4// Author: Vaclav Slavik
5// Created: 2006-08-07
6// Copyright: (c) 2006 REA Elektronik GmbH
7// Licence: wxWindows licence
8/////////////////////////////////////////////////////////////////////////////
9
10// ===========================================================================
11// declarations
12// ===========================================================================
13
14// ---------------------------------------------------------------------------
15// headers
16// ---------------------------------------------------------------------------
17
18// For compilers that support precompilation, includes "wx.h".
19#include "wx/wxprec.h"
20
21#ifdef __BORLANDC__
22 #pragma hdrstop
23#endif
24
25#ifndef WX_PRECOMP
26 #include "wx/dcmemory.h"
27 #include "wx/log.h"
28#endif
29
30#include "wx/dfb/dc.h"
31#include "wx/dfb/private.h"
32
33// these values are used to initialize newly created DC
34#define DEFAULT_FONT (*wxNORMAL_FONT)
35#define DEFAULT_PEN (*wxBLACK_PEN)
36#define DEFAULT_BRUSH (*wxWHITE_BRUSH)
37
38// ===========================================================================
39// implementation
40// ===========================================================================
41
42//-----------------------------------------------------------------------------
43// wxDFBDCImpl
44//-----------------------------------------------------------------------------
45
46IMPLEMENT_ABSTRACT_CLASS(wxDFBDCImpl, wxDCImpl)
47
48void wxDFBDCImpl::DFBInit(const wxIDirectFBSurfacePtr& surface)
49{
50 m_surface = surface;
51
52 wxCHECK_RET( surface != NULL, "invalid surface" );
53
54 m_mm_to_pix_x = (double)wxGetDisplaySize().GetWidth() /
55 (double)wxGetDisplaySizeMM().GetWidth();
56 m_mm_to_pix_y = (double)wxGetDisplaySize().GetHeight() /
57 (double)wxGetDisplaySizeMM().GetHeight();
58
59 SetFont(DEFAULT_FONT);
60 SetPen(DEFAULT_PEN);
61 SetBrush(DEFAULT_BRUSH);
62}
63
64
65// ---------------------------------------------------------------------------
66// clipping
67// ---------------------------------------------------------------------------
68
69void wxDFBDCImpl::DoSetClippingRegion(wxCoord cx, wxCoord cy, wxCoord cw, wxCoord ch)
70{
71 wxCHECK_RET( IsOk(), wxT("invalid dc") );
72
73 wxSize size(GetSize());
74
75 wxASSERT_MSG( !m_clipping,
76 "narrowing clipping region not implemented yet" );
77
78 // NB: We intersect the clipping rectangle with surface's area here because
79 // DirectFB will return an error if you try to set clipping rectangle
80 // that is partially outside of the surface.
81 DFBRegion r;
82 r.x1 = wxMax(0, XLOG2DEV(cx));
83 r.y1 = wxMax(0, YLOG2DEV(cy));
84 r.x2 = wxMin(r.x1 + XLOG2DEVREL(cw), size.x) - 1;
85 r.y2 = wxMin(r.y1 + YLOG2DEVREL(ch), size.y) - 1;
86
87 if ( !m_surface->SetClip(&r) )
88 return;
89
90 m_clipX1 = cx;
91 m_clipY1 = cy;
92 m_clipX2 = cx + cw - 1;
93 m_clipY2 = cy + ch -1;
94 m_clipping = true;
95}
96
97void wxDFBDCImpl::DoSetDeviceClippingRegion(const wxRegion& region)
98{
99 // NB: this can be done because wxDFB only supports rectangular regions
100 wxRect rect = region.AsRect();
101
102 // our parameter is in physical coordinates while DoSetClippingRegion()
103 // takes logical ones
104 rect.x = XDEV2LOG(rect.x);
105 rect.y = YDEV2LOG(rect.y);
106 rect.width = XDEV2LOG(rect.width);
107 rect.height = YDEV2LOG(rect.height);
108
109 DoSetClippingRegion(rect.x, rect.y, rect.width, rect.height);
110}
111
112void wxDFBDCImpl::DestroyClippingRegion()
113{
114 wxCHECK_RET( IsOk(), wxT("invalid dc") );
115
116 m_surface->SetClip(NULL);
117
118 ResetClipping();
119}
120
121// ---------------------------------------------------------------------------
122// query capabilities
123// ---------------------------------------------------------------------------
124
125int wxDFBDCImpl::GetDepth() const
126{
127 return m_surface->GetDepth();
128}
129
130// ---------------------------------------------------------------------------
131// drawing
132// ---------------------------------------------------------------------------
133
134void wxDFBDCImpl::Clear()
135{
136 wxCHECK_RET( IsOk(), wxT("invalid dc") );
137
138 if ( m_backgroundBrush.GetStyle() == wxTRANSPARENT )
139 return;
140
141 wxColour clr = m_backgroundBrush.GetColour();
142 m_surface->Clear(clr.Red(), clr.Green(), clr.Blue(), clr.Alpha());
143
144 wxSize size(GetSize());
145 CalcBoundingBox(XDEV2LOG(0), YDEV2LOG(0));
146 CalcBoundingBox(XDEV2LOG(size.x), YDEV2LOG(size.y));
147}
148
149extern bool wxDoFloodFill(wxDC *dc, wxCoord x, wxCoord y,
150 const wxColour & col, wxFloodFillStyle style);
151
152bool wxDFBDCImpl::DoFloodFill(wxCoord x, wxCoord y,
153 const wxColour& col, wxFloodFillStyle style)
154{
155 return wxDoFloodFill(GetOwner(), x, y, col, style);
156}
157
158bool wxDFBDCImpl::DoGetPixel(wxCoord x, wxCoord y, wxColour *col) const
159{
160 wxCHECK_MSG( col, false, "NULL colour parameter in wxDFBDCImpl::GetPixel");
161
162 wxFAIL_MSG( "GetPixel not implemented" );
163
164 wxUnusedVar(x);
165 wxUnusedVar(y);
166
167 return false;
168}
169
170void wxDFBDCImpl::DoCrossHair(wxCoord x, wxCoord y)
171{
172 wxCHECK_RET( IsOk(), wxT("invalid dc") );
173
174 wxFAIL_MSG( "CrossHair not implemented" );
175
176 wxUnusedVar(x);
177 wxUnusedVar(y);
178}
179
180void wxDFBDCImpl::DoDrawLine(wxCoord x1, wxCoord y1, wxCoord x2, wxCoord y2)
181{
182 wxCHECK_RET( IsOk(), wxT("invalid dc") );
183
184 if ( m_pen.GetStyle() == wxTRANSPARENT )
185 return;
186
187 wxCoord xx1 = XLOG2DEV(x1);
188 wxCoord yy1 = YLOG2DEV(y1);
189 wxCoord xx2 = XLOG2DEV(x2);
190 wxCoord yy2 = YLOG2DEV(y2);
191
192 // FIXME: DrawLine() shouldn't draw the last pixel, but DFB's DrawLine()
193 // does draw it. We should undo any change to the last pixel by
194 // using GetPixel() and PutPixel(), but until they are implemented,
195 // handle at least the special case of vertical and horizontal
196 // lines correctly:
197 if ( xx1 == xx2 )
198 {
199 if ( yy1 < yy2 )
200 yy2--;
201 else if ( yy1 > yy2 )
202 yy2++;
203 }
204 if ( yy1 == yy2 )
205 {
206 if ( xx1 < xx2 )
207 xx2--;
208 else if ( xx1 > xx2 )
209 xx2++;
210 }
211
212 m_surface->DrawLine(xx1, yy1, xx2, yy2);
213
214 CalcBoundingBox(x1, y1);
215 CalcBoundingBox(x2, y2);
216}
217
218// Draws an arc of a circle, centred on (xc, yc), with starting point (x1, y1)
219// and ending at (x2, y2)
220void wxDFBDCImpl::DoDrawArc(wxCoord WXUNUSED(x1), wxCoord WXUNUSED(y1),
221 wxCoord WXUNUSED(x2), wxCoord WXUNUSED(y2),
222 wxCoord WXUNUSED(xc), wxCoord WXUNUSED(yc))
223{
224 wxCHECK_RET( IsOk(), wxT("invalid dc") );
225
226 wxFAIL_MSG( "DrawArc not implemented" );
227}
228
229void wxDFBDCImpl::DoDrawPoint(wxCoord x, wxCoord y)
230{
231 wxCHECK_RET( IsOk(), wxT("invalid dc") );
232
233 // NB: DirectFB API doesn't provide a function for drawing points, so
234 // implement it as 1px long line. This is inefficient, but then, so is
235 // using DrawPoint() for drawing more than a few points.
236 DoDrawLine(x, y, x, y);
237
238 // FIXME_DFB: implement special cases for common formats (RGB24,RGBA/RGB32)
239}
240
241void wxDFBDCImpl::DoDrawPolygon(int WXUNUSED(n), const wxPoint WXUNUSED(points)[],
242 wxCoord WXUNUSED(xoffset), wxCoord WXUNUSED(yoffset),
243 wxPolygonFillMode WXUNUSED(fillStyle))
244{
245 wxCHECK_RET( IsOk(), wxT("invalid dc") );
246
247 wxFAIL_MSG( "DrawPolygon not implemented" );
248}
249
250void wxDFBDCImpl::DoDrawLines(int WXUNUSED(n), const wxPoint WXUNUSED(points)[],
251 wxCoord WXUNUSED(xoffset), wxCoord WXUNUSED(yoffset))
252{
253 wxCHECK_RET( IsOk(), wxT("invalid dc") );
254
255 // TODO: impl. using DirectDB's DrawLines
256 wxFAIL_MSG( "DrawLines not implemented" );
257}
258
259void wxDFBDCImpl::DoDrawRectangle(wxCoord x, wxCoord y, wxCoord width, wxCoord height)
260{
261 wxCHECK_RET( IsOk(), wxT("invalid dc") );
262
263 wxCoord xx = XLOG2DEV(x);
264 wxCoord yy = YLOG2DEV(y);
265 wxCoord ww = m_signX * XLOG2DEVREL(width);
266 wxCoord hh = m_signY * YLOG2DEVREL(height);
267
268 if ( ww == 0 || hh == 0 ) return;
269
270 if ( ww < 0 )
271 {
272 ww = -ww;
273 xx = xx - ww;
274 }
275 if ( hh < 0 )
276 {
277 hh = -hh;
278 yy = yy - hh;
279 }
280
281 if ( m_brush.GetStyle() != wxTRANSPARENT )
282 {
283 SelectColour(m_brush.GetColour());
284 m_surface->FillRectangle(xx, yy, ww, hh);
285 // restore pen's colour, because other drawing functions expect the
286 // colour to be set to the pen:
287 SelectColour(m_pen.GetColour());
288 }
289
290 if ( m_pen.GetStyle() != wxTRANSPARENT )
291 {
292 m_surface->DrawRectangle(xx, yy, ww, hh);
293 }
294
295 CalcBoundingBox(x, y);
296 CalcBoundingBox(x + width, y + height);
297}
298
299void wxDFBDCImpl::DoDrawRoundedRectangle(wxCoord WXUNUSED(x),
300 wxCoord WXUNUSED(y),
301 wxCoord WXUNUSED(width),
302 wxCoord WXUNUSED(height),
303 double WXUNUSED(radius))
304{
305 wxCHECK_RET( IsOk(), wxT("invalid dc") );
306
307 wxFAIL_MSG( "DrawRoundedRectangle not implemented" );
308}
309
310void wxDFBDCImpl::DoDrawEllipse(wxCoord WXUNUSED(x),
311 wxCoord WXUNUSED(y),
312 wxCoord WXUNUSED(width),
313 wxCoord WXUNUSED(height))
314{
315 wxCHECK_RET( IsOk(), wxT("invalid dc") );
316
317 wxFAIL_MSG( "DrawElipse not implemented" );
318}
319
320void wxDFBDCImpl::DoDrawEllipticArc(wxCoord WXUNUSED(x),
321 wxCoord WXUNUSED(y),
322 wxCoord WXUNUSED(w),
323 wxCoord WXUNUSED(h),
324 double WXUNUSED(sa),
325 double WXUNUSED(ea))
326{
327 wxCHECK_RET( IsOk(), wxT("invalid dc") );
328
329 wxFAIL_MSG( "DrawElipticArc not implemented" );
330}
331
332void wxDFBDCImpl::DoDrawText(const wxString& text, wxCoord x, wxCoord y)
333{
334 wxCHECK_RET( IsOk(), wxT("invalid dc") );
335
336 wxCoord xx = XLOG2DEV(x);
337 wxCoord yy = YLOG2DEV(y);
338
339 // update the bounding box
340 wxCoord w, h;
341 CalcBoundingBox(x, y);
342 DoGetTextExtent(text, &w, &h);
343 CalcBoundingBox(x + w, y + h);
344
345 // if background mode is solid, DrawText must paint text's background:
346 if ( m_backgroundMode == wxSOLID )
347 {
348 wxCHECK_RET( m_textBackgroundColour.IsOk(),
349 wxT("invalid background color") );
350
351 SelectColour(m_textBackgroundColour);
352 m_surface->FillRectangle(xx, yy, XLOG2DEVREL(w), YLOG2DEVREL(h));
353 }
354
355 // finally draw the text itself:
356 wxCHECK_RET( m_textForegroundColour.IsOk(),
357 wxT("invalid foreground color") );
358 SelectColour(m_textForegroundColour);
359 m_surface->DrawString(text.utf8_str(), -1, xx, yy, DSTF_LEFT | DSTF_TOP);
360
361 // restore pen's colour, because other drawing functions expect the colour
362 // to be set to the pen:
363 SelectColour(m_pen.GetColour());
364}
365
366void wxDFBDCImpl::DoDrawRotatedText(const wxString& WXUNUSED(text),
367 wxCoord WXUNUSED(x), wxCoord WXUNUSED(y),
368 double WXUNUSED(angle))
369{
370 wxCHECK_RET( IsOk(), wxT("invalid dc") );
371
372 wxFAIL_MSG( "DrawRotatedText not implemented" );
373}
374
375// ---------------------------------------------------------------------------
376// set GDI objects
377// ---------------------------------------------------------------------------
378
379void wxDFBDCImpl::SetPen(const wxPen& pen)
380{
381 m_pen = pen.IsOk() ? pen : DEFAULT_PEN;
382
383 SelectColour(m_pen.GetColour());
384}
385
386void wxDFBDCImpl::SetBrush(const wxBrush& brush)
387{
388 m_brush = brush.IsOk() ? brush : DEFAULT_BRUSH;
389}
390
391void wxDFBDCImpl::SelectColour(const wxColour& clr)
392{
393 m_surface->SetColor(clr.Red(), clr.Green(), clr.Blue(), clr.Alpha());
394 #warning "use SetColorIndex?"
395}
396
397#if wxUSE_PALETTE
398void wxDFBDCImpl::SetPalette(const wxPalette& WXUNUSED(palette))
399{
400 wxCHECK_RET( IsOk(), wxT("invalid dc") );
401
402 wxFAIL_MSG( "SetPalette not implemented" );
403}
404#endif // wxUSE_PALETTE
405
406void wxDFBDCImpl::SetFont(const wxFont& font)
407{
408 wxCHECK_RET( IsOk(), wxT("invalid dc") );
409
410 wxFont f(font.IsOk() ? font : DEFAULT_FONT);
411
412 wxFont oldfont(m_font);
413
414 m_font = f;
415
416 if ( !m_surface->SetFont(GetCurrentFont()) )
417 {
418 m_font = oldfont;
419 return;
420 }
421}
422
423wxIDirectFBFontPtr wxDFBDCImpl::GetCurrentFont() const
424{
425 bool aa = (GetDepth() > 8);
426 return m_font.GetDirectFBFont(aa);
427}
428
429void wxDFBDCImpl::SetBackground(const wxBrush& brush)
430{
431 wxCHECK_RET( IsOk(), wxT("invalid dc") );
432
433 if (!brush.IsOk()) return;
434
435 m_backgroundBrush = brush;
436}
437
438void wxDFBDCImpl::SetBackgroundMode(int mode)
439{
440 m_backgroundMode = mode;
441}
442
443void wxDFBDCImpl::SetLogicalFunction(wxRasterOperationMode function)
444{
445 wxCHECK_RET( IsOk(), wxT("invalid dc") );
446
447 // NB: we could also support XOR, but for blitting only (via DSBLIT_XOR);
448 // and possibly others via SetSrc/DstBlendFunction()
449 wxASSERT_MSG( function == wxCOPY,
450 "only wxCOPY logical function supported" );
451
452 m_logicalFunction = function;
453}
454
455bool wxDFBDCImpl::StartDoc(const wxString& WXUNUSED(message))
456{
457 // We might be previewing, so return true to let it continue.
458 return true;
459}
460
461void wxDFBDCImpl::EndDoc()
462{
463}
464
465void wxDFBDCImpl::StartPage()
466{
467}
468
469void wxDFBDCImpl::EndPage()
470{
471}
472
473// ---------------------------------------------------------------------------
474// text metrics
475// ---------------------------------------------------------------------------
476
477wxCoord wxDFBDCImpl::GetCharHeight() const
478{
479 wxCHECK_MSG( IsOk(), -1, wxT("invalid dc") );
480 wxCHECK_MSG( m_font.IsOk(), -1, wxT("no font selected") );
481
482 int h = -1;
483 GetCurrentFont()->GetHeight(&h);
484 return YDEV2LOGREL(h);
485}
486
487wxCoord wxDFBDCImpl::GetCharWidth() const
488{
489 wxCHECK_MSG( IsOk(), -1, wxT("invalid dc") );
490 wxCHECK_MSG( m_font.IsOk(), -1, wxT("no font selected") );
491
492 int w = -1;
493 GetCurrentFont()->GetStringWidth("H", 1, &w);
494 // VS: YDEV is corrent, it should *not* be XDEV, because font's are only
495 // scaled according to m_scaleY
496 return YDEV2LOGREL(w);
497}
498
499void wxDFBDCImpl::DoGetTextExtent(const wxString& string, wxCoord *x, wxCoord *y,
500 wxCoord *descent, wxCoord *externalLeading,
501 const wxFont *theFont) const
502{
503 wxCHECK_RET( IsOk(), wxT("invalid dc") );
504 wxCHECK_RET( m_font.IsOk(), wxT("no font selected") );
505 wxCHECK_RET( !theFont || theFont->IsOk(), wxT("invalid font") );
506
507 wxFont oldFont;
508 if ( theFont != NULL )
509 {
510 oldFont = m_font;
511 wxConstCast(this, wxDFBDCImpl)->SetFont(*theFont);
512 }
513
514 wxCoord xx = 0, yy = 0;
515 DFBRectangle rect;
516 wxIDirectFBFontPtr f = GetCurrentFont();
517
518 if ( f->GetStringExtents(string.utf8_str(), -1, &rect, NULL) )
519 {
520 // VS: YDEV is corrent, it should *not* be XDEV, because font's are
521 // only scaled according to m_scaleY
522 xx = YDEV2LOGREL(rect.w);
523 yy = YDEV2LOGREL(rect.h);
524
525 if ( descent )
526 {
527 int d;
528 if ( f->GetDescender(&d) )
529 *descent = YDEV2LOGREL(-d);
530 else
531 *descent = 0;
532 }
533 }
534
535 if ( x ) *x = xx;
536 if ( y ) *y = yy;
537 if ( externalLeading ) *externalLeading = 0;
538
539 if ( theFont != NULL )
540 wxConstCast(this, wxDFBDCImpl)->SetFont(oldFont);
541}
542
543
544
545// ---------------------------------------------------------------------------
546// mapping modes
547// ---------------------------------------------------------------------------
548
549// FIXME_DFB: scaling affects pixel size of font, pens, brushes, which
550// is not currently implemented here; probably makes sense to
551// switch to Cairo instead of implementing everything for DFB
552
553void wxDFBDCImpl::DoGetSize(int *w, int *h) const
554{
555 wxCHECK_RET( IsOk(), wxT("invalid dc") );
556
557 m_surface->GetSize(w, h);
558}
559
560void wxDFBDCImpl::DoGetSizeMM(int *width, int *height) const
561{
562 #warning "move this to common code?"
563 int w = 0;
564 int h = 0;
565 GetSize(&w, &h);
566 if ( width ) *width = int(double(w) / (m_userScaleX*m_mm_to_pix_x));
567 if ( height ) *height = int(double(h) / (m_userScaleY*m_mm_to_pix_y));
568}
569
570wxSize wxDFBDCImpl::GetPPI() const
571{
572 #warning "move this to common code?"
573 return wxSize(int(double(m_mm_to_pix_x) * inches2mm),
574 int(double(m_mm_to_pix_y) * inches2mm));
575}
576
577
578// ---------------------------------------------------------------------------
579// Blitting
580// ---------------------------------------------------------------------------
581
582bool wxDFBDCImpl::DoBlit(wxCoord xdest, wxCoord ydest,
583 wxCoord width, wxCoord height,
584 wxDC *source, wxCoord xsrc, wxCoord ysrc,
585 wxRasterOperationMode rop, bool useMask,
586 wxCoord xsrcMask, wxCoord ysrcMask)
587{
588 wxCHECK_MSG( IsOk(), false, "invalid dc" );
589 wxCHECK_MSG( source, false, "invalid source dc" );
590
591 // NB: we could also support XOR here (via DSBLIT_XOR)
592 // and possibly others via SetSrc/DstBlendFunction()
593 wxCHECK_MSG( rop == wxCOPY, false, "only wxCOPY function supported" );
594
595 // transform the source DC coords to the device ones
596 xsrc = source->LogicalToDeviceX(xsrc);
597 ysrc = source->LogicalToDeviceY(ysrc);
598
599 // FIXME_DFB: use the mask origin when drawing transparently
600 wxASSERT_MSG( xsrcMask == -1 && ysrcMask == -1,
601 "non-default source mask offset not implemented" );
602#if 0
603 if (xsrcMask == -1 && ysrcMask == -1)
604 {
605 xsrcMask = xsrc; ysrcMask = ysrc;
606 }
607 else
608 {
609 xsrcMask = source->LogicalToDeviceX(xsrcMask);
610 ysrcMask = source->LogicalToDeviceY(ysrcMask);
611 }
612#endif
613
614 wxMemoryDC *sourceAsMemDC = wxDynamicCast(source, wxMemoryDC);
615 if ( sourceAsMemDC )
616 {
617 DoDrawSubBitmap(sourceAsMemDC->GetSelectedBitmap(),
618 xsrc, ysrc,
619 width, height,
620 xdest, ydest,
621 rop,
622 useMask);
623 }
624 else
625 {
626 return DoBlitFromSurface
627 (
628 static_cast<wxDFBDCImpl *>(source->GetImpl())
629 ->GetDirectFBSurface(),
630 xsrc, ysrc,
631 width, height,
632 xdest, ydest
633 );
634 }
635
636 return true;
637}
638
639void wxDFBDCImpl::DoDrawBitmap(const wxBitmap &bmp, wxCoord x, wxCoord y, bool useMask)
640{
641 wxCHECK_RET( IsOk(), wxT("invalid dc") );
642 wxCHECK_RET( bmp.IsOk(), wxT("invalid bitmap") );
643
644 DoDrawSubBitmap(bmp,
645 0, 0, bmp.GetWidth(), bmp.GetHeight(),
646 x, y,
647 m_logicalFunction, useMask);
648}
649
650void wxDFBDCImpl::DoDrawIcon(const wxIcon& icon, wxCoord x, wxCoord y)
651{
652 // VZ: egcs 1.0.3 refuses to compile this without cast, no idea why
653 DoDrawBitmap((const wxBitmap&)icon, x, y, true);
654}
655
656void wxDFBDCImpl::DoDrawSubBitmap(const wxBitmap &bmp,
657 wxCoord x, wxCoord y, wxCoord w, wxCoord h,
658 wxCoord destx, wxCoord desty, int rop, bool useMask)
659{
660 wxCHECK_RET( IsOk(), wxT("invalid dc") );
661 wxCHECK_RET( bmp.IsOk(), wxT("invalid bitmap") );
662
663 // NB: we could also support XOR here (via DSBLIT_XOR)
664 // and possibly others via SetSrc/DstBlendFunction()
665 wxCHECK_RET( rop == wxCOPY, "only wxCOPY function supported" );
666
667 if ( bmp.GetDepth() == 1 )
668 {
669 // Mono bitmaps are handled in special way -- all 1s are drawn in
670 // foreground colours, all 0s in background colour.
671 wxFAIL_MSG( "drawing mono bitmaps not implemented" );
672 return;
673 }
674
675 if ( useMask && bmp.GetMask() )
676 {
677 // FIXME_DFB: Could use blitting modes for this; also see
678 // DFB's SetSrcBlendFunction() and SetSrcColorKey()
679 wxFAIL_MSG( "drawing bitmaps with masks not implemented" );
680 return;
681 }
682
683 DoBlitFromSurface(bmp.GetDirectFBSurface(),
684 x, y,
685 w, h,
686 destx, desty);
687}
688
689bool wxDFBDCImpl::DoBlitFromSurface(const wxIDirectFBSurfacePtr& src,
690 wxCoord srcx, wxCoord srcy,
691 wxCoord w, wxCoord h,
692 wxCoord dstx, wxCoord dsty)
693{
694 // don't do anything if the source rectangle is outside of source surface,
695 // DirectFB would assert in that case:
696 wxSize srcsize;
697 src->GetSize(&srcsize.x, &srcsize.y);
698 if ( !wxRect(srcx, srcy, w, h).Intersects(wxRect(srcsize)) )
699 {
700 wxLogDebug("Blitting from area outside of the source surface, caller code needs fixing.");
701 return false;
702 }
703
704 CalcBoundingBox(dstx, dsty);
705 CalcBoundingBox(dstx + w, dsty + h);
706
707 DFBRectangle srcRect = { srcx, srcy, w, h };
708 DFBRectangle dstRect = { XLOG2DEV(dstx), YLOG2DEV(dsty),
709 XLOG2DEVREL(w), YLOG2DEVREL(h) };
710
711 wxIDirectFBSurfacePtr dst(m_surface);
712
713 // FIXME: this will have to be different in useMask case, see above
714 DFBSurfaceBlittingFlags blitFlag = (src->GetPixelFormat() == DSPF_ARGB)
715 ? DSBLIT_BLEND_ALPHACHANNEL
716 : DSBLIT_NOFX;
717 if ( !dst->SetBlittingFlags(blitFlag) )
718 return false;
719
720 if ( srcRect.w != dstRect.w || srcRect.h != dstRect.h )
721 {
722 // the bitmap is drawn stretched:
723 dst->StretchBlit(src, &srcRect, &dstRect);
724 }
725 else
726 {
727 // no stretching, size is preserved:
728 dst->Blit(src, &srcRect, dstRect.x, dstRect.y);
729 }
730
731 return true;
732}