The Battle for Wesnoth  1.19.10+dev
draw.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2022 - 2025
3  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
4 
5  This program is free software; you can redistribute it and/or modify
6  it under the terms of the GNU General Public License as published by
7  the Free Software Foundation; either version 2 of the License, or
8  (at your option) any later version.
9  This program is distributed in the hope that it will be useful,
10  but WITHOUT ANY WARRANTY.
11 
12  See the COPYING file for more details.
13 */
14 
15 #include "draw.hpp"
16 
17 #include "color.hpp"
18 #include "font/cairo.hpp"
19 #include "log.hpp"
20 #include "sdl/rect.hpp"
21 #include "sdl/texture.hpp"
22 #include "sdl/utils.hpp" // sdl::runtime_at_least
23 #include "video.hpp"
24 
25 #include <boost/math/constants/constants.hpp>
26 #include <SDL2/SDL_rect.h>
27 #include <SDL2/SDL_render.h>
28 #include <cairo.h>
29 
30 static lg::log_domain log_draw("draw");
31 #define DBG_D LOG_STREAM(debug, log_draw)
32 #define WRN_D LOG_STREAM(warn, log_draw)
33 
34 static SDL_Renderer* renderer()
35 {
36  return video::get_renderer();
37 }
38 
39 /**************************************/
40 /* basic drawing and pixel primatives */
41 /**************************************/
42 
44 {
45  DBG_D << "clear";
46  SDL_BlendMode b;
47  SDL_GetRenderDrawBlendMode(renderer(), &b);
48  SDL_SetRenderDrawBlendMode(renderer(), SDL_BLENDMODE_NONE);
49  fill(0, 0, 0, 0);
50  SDL_SetRenderDrawBlendMode(renderer(), b);
51 }
52 
54  const SDL_Rect& area,
55  uint8_t r, uint8_t g, uint8_t b, uint8_t a)
56 {
57  DBG_D << "fill " << area << ' ' << color_t{r,g,b,a};
58  SDL_SetRenderDrawColor(renderer(), r, g, b, a);
59  SDL_RenderFillRect(renderer(), &area);
60 }
61 
63  const SDL_Rect& area,
64  uint8_t r, uint8_t g, uint8_t b)
65 {
66  draw::fill(area, r, g, b, SDL_ALPHA_OPAQUE);
67 }
68 
69 void draw::fill(const SDL_Rect& area, const color_t& c)
70 {
71  draw::fill(area, c.r, c.g, c.b, c.a);
72 }
73 
74 void draw::fill(uint8_t r, uint8_t g, uint8_t b, uint8_t a)
75 {
76  DBG_D << "fill " << color_t{r,g,b,a};
77  SDL_SetRenderDrawColor(renderer(), r, g, b, a);
78  SDL_RenderFillRect(renderer(), nullptr);
79 }
80 
81 void draw::fill(uint8_t r, uint8_t g, uint8_t b)
82 {
83  draw::fill(r, g, b, SDL_ALPHA_OPAQUE);
84 }
85 
86 void draw::fill(const color_t& c)
87 {
88  draw::fill(c.r, c.g, c.b, c.a);
89 }
90 
91 void draw::fill(const SDL_FRect& rect, const color_t& c)
92 {
93  DBG_D << "sub-pixel fill";
94  SDL_SetRenderDrawColor(renderer(), c.r, c.g, c.b, c.a);
95  SDL_RenderFillRectF(renderer(), &rect);
96 }
97 
98 void draw::fill(const SDL_Rect& area)
99 {
100  DBG_D << "fill " << area;
101  SDL_RenderFillRect(renderer(), &area);
102 }
103 
105 {
106  DBG_D << "fill";
107  SDL_RenderFillRect(renderer(), nullptr);
108 }
109 
110 void draw::set_color(uint8_t r, uint8_t g, uint8_t b, uint8_t a)
111 {
112  DBG_D << "set color " << color_t{r,g,b,a};
113  SDL_SetRenderDrawColor(renderer(), r, g, b, a);
114 }
115 
116 void draw::set_color(uint8_t r, uint8_t g, uint8_t b)
117 {
118  DBG_D << "set color " << color_t{r,g,b};
119  SDL_SetRenderDrawColor(renderer(), r, g, b, SDL_ALPHA_OPAQUE);
120 }
121 
123 {
124  DBG_D << "set color " << c;
125  SDL_SetRenderDrawColor(renderer(), c.r, c.g, c.b, c.a);
126 }
127 
128 void draw::set_blend_mode(SDL_BlendMode b)
129 {
130  SDL_SetRenderDrawBlendMode(renderer(), b);
131 }
132 
133 /** Some versions of SDL have a bad rectangle drawing implementation. */
134 static bool sdl_bad_at_rects()
135 {
136  // This could be done once at program start and cached,
137  // but it isn't all that heavy.
138  if (sdl::runtime_at_least(2,0,15) && !sdl::runtime_at_least(2,0,18)) {
139  return true;
140  }
141  return false;
142 }
143 
144 /** For some SDL versions, draw rectangles as lines. */
145 static void draw_rect_as_lines(const SDL_Rect& rect)
146 {
147  // w and h indicate the final pixel width/height of the box.
148  // This is 1 greater than the difference in corner coordinates.
149  if (rect.w <= 0 || rect.h <= 0) {
150  return;
151  }
152  int x2 = rect.x + rect.w - 1;
153  int y2 = rect.y + rect.h - 1;
154  draw::line(rect.x, rect.y, x2, rect.y);
155  draw::line(rect.x, rect.y, rect.x, y2);
156  draw::line(x2, rect.y, x2, y2);
157  draw::line(rect.x, y2, x2, y2);
158 }
159 
160 void draw::rect(const SDL_Rect& rect)
161 {
162  DBG_D << "rect " << rect;
163  if (sdl_bad_at_rects()) {
164  return draw_rect_as_lines(rect);
165  }
166  SDL_RenderDrawRect(renderer(), &rect);
167 }
168 
169 void draw::rect(const SDL_Rect& rect,
170  uint8_t r, uint8_t g, uint8_t b, uint8_t a)
171 {
172  DBG_D << "rect " << rect << ' ' << color_t{r,g,b,a};
173  SDL_SetRenderDrawColor(renderer(), r, g, b, a);
174  if (sdl_bad_at_rects()) {
175  return draw_rect_as_lines(rect);
176  }
177  SDL_RenderDrawRect(renderer(), &rect);
178 }
179 
180 void draw::rect(const SDL_Rect& rect, uint8_t r, uint8_t g, uint8_t b)
181 {
182  draw::rect(rect, r, g, b, SDL_ALPHA_OPAQUE);
183 }
184 
185 void draw::rect(const SDL_Rect& rect, const color_t& c)
186 {
187  draw::rect(rect, c.r, c.g, c.b, c.a);
188 }
189 
190 void draw::line(int from_x, int from_y, int to_x, int to_y)
191 {
192  DBG_D << "line from (" << from_x << ',' << from_y
193  << ") to (" << to_x << ',' << to_y << ')';
194  SDL_RenderDrawLine(renderer(), from_x, from_y, to_x, to_y);
195 }
196 
197 void draw::line(int from_x, int from_y, int to_x, int to_y, const color_t& c)
198 {
199  DBG_D << "line from (" << from_x << ',' << from_y
200  << ") to (" << to_x << ',' << to_y
201  << ") with colour " << c;
202  SDL_SetRenderDrawColor(renderer(), c.r, c.g, c.b, c.a);
203  SDL_RenderDrawLine(renderer(), from_x, from_y, to_x, to_y);
204 }
205 
206 void draw::points(const std::vector<SDL_Point>& points)
207 {
208  DBG_D << points.size() << " points";
209  SDL_RenderDrawPoints(renderer(), points.data(), points.size());
210 }
211 
212 void draw::point(int x, int y)
213 {
214  DBG_D << "point (" << x << ',' << y << ')';
215  SDL_RenderDrawPoint(renderer(), x, y);
216 }
217 
218 void draw::circle(int cx, int cy, int r, const color_t& c, uint8_t octants)
219 {
221  draw::circle(cx, cy, r, octants);
222 }
223 
224 void draw::circle(int cx, int cy, int r, uint8_t octants)
225 {
226  DBG_D << "circle (" << cx << ',' << cy
227  << ") -> " << r << ", oct " << int(octants);
228 
229  // Algorithm based on
230  // http://de.wikipedia.org/wiki/Rasterung_von_Kreisen#Methode_von_Horn
231  // version of 2011.02.07.
232  int d = -r;
233  int x = r;
234  int y = 0;
235 
236  std::vector<SDL_Point> points;
237 
238  while(!(y > x)) {
239  if(octants & 0x04) points.push_back({cx + x, cy + y});
240  if(octants & 0x02) points.push_back({cx + x, cy - y});
241  if(octants & 0x20) points.push_back({cx - x, cy + y});
242  if(octants & 0x40) points.push_back({cx - x, cy - y});
243 
244  if(octants & 0x08) points.push_back({cx + y, cy + x});
245  if(octants & 0x01) points.push_back({cx + y, cy - x});
246  if(octants & 0x10) points.push_back({cx - y, cy + x});
247  if(octants & 0x80) points.push_back({cx - y, cy - x});
248 
249  d += 2 * y + 1;
250  ++y;
251  if(d > 0) {
252  d += -2 * x + 2;
253  --x;
254  }
255  }
256 
258 }
259 
260 void draw::disc(int cx, int cy, int r, const color_t& c, uint8_t octants)
261 {
263  draw::disc(cx, cy, r, octants);
264 }
265 
266 void draw::disc(int cx, int cy, int r, uint8_t octants)
267 {
268  DBG_D << "disc (" << cx << ',' << cy
269  << ") -> " << r << ", oct " << int(octants);
270 
271  int d = -r;
272  int x = r;
273  int y = 0;
274 
275  while(!(y > x)) {
276  // I use the formula of Bresenham's line algorithm
277  // to determine the boundaries of a segment.
278  // The slope of the line is always 1 or -1 in this case.
279  if(octants & 0x04)
280  // x2 - 1 = y2 - (cy + 1) + cx
281  draw::line(cx + x, cy + y + 1, cx + y + 1, cy + y + 1);
282  if(octants & 0x02)
283  // x2 - 1 = cy - y2 + cx
284  draw::line(cx + x, cy - y, cx + y + 1, cy - y);
285  if(octants & 0x20)
286  // x2 + 1 = (cy + 1) - y2 + (cx - 1)
287  draw::line(cx - x - 1, cy + y + 1, cx - y - 2, cy + y + 1);
288  if(octants & 0x40)
289  // x2 + 1 = y2 - cy + (cx - 1)
290  draw::line(cx - x - 1, cy - y, cx - y - 2, cy - y);
291 
292  if(octants & 0x08)
293  // y2 = x2 - cx + (cy + 1)
294  draw::line(cx + y, cy + x + 1, cx + y, cy + y + 1);
295  if(octants & 0x01)
296  // y2 = cx - x2 + cy
297  draw::line(cx + y, cy - x, cx + y, cy - y);
298  if(octants & 0x10)
299  // y2 = (cx - 1) - x2 + (cy + 1)
300  draw::line(cx - y - 1, cy + x + 1, cx - y - 1, cy + y + 1);
301  if(octants & 0x80)
302  // y2 = x2 - (cx - 1) + cy
303  draw::line(cx - y - 1, cy - x, cx - y - 1, cy - y);
304 
305  d += 2 * y + 1;
306  ++y;
307  if(d > 0) {
308  d += -2 * x + 2;
309  --x;
310  }
311  }
312 }
313 
314 /********************/
315 /* Cairo primitives */
316 /********************/
317 
318 void draw::cairo_circle(int cx, int cy, int r, const color_t& c, int thickness)
319 {
320  if (r <= 0) {
321  return;
322  }
323 
324  int size = 2*r;
325  surface sdl_surf(size, size);
326  auto cairo_surface = cairo::create_surface(
327  reinterpret_cast<uint8_t*>(sdl_surf->pixels), ::point(sdl_surf->w, sdl_surf->h));
328  auto cairo_context = cairo::create_context(cairo_surface);
329  cairo_t* ctx = cairo_context.get();
330 
331  cairo_set_antialias(ctx, CAIRO_ANTIALIAS_BEST);
332 
333  cairo_set_source_rgba(ctx, 0.0, 0.0, 0.0, 0.0);
334  cairo_paint(ctx);
335 
336  cairo_set_line_width(ctx, thickness);
337  cairo_set_source_rgba(ctx,
338  c.r / 255.0,
339  c.g / 255.0,
340  c.b / 255.0,
341  c.a / 255.0
342  );
343  cairo_arc(ctx, r, r, r-thickness, 0, 2*boost::math::constants::pi<double>());
344  cairo_stroke(ctx);
345 
346  draw::blit(texture(sdl_surf), ::rect(cx-r, cy-r, size, size));
347 }
348 
349 void draw::cairo_disc(int cx, int cy, int r, const color_t& c)
350 {
351  if (r <= 0) {
352  return;
353  }
354 
355  int size = 2*r;
356  surface sdl_surf(size, size);
357  auto cairo_surface = cairo::create_surface(
358  reinterpret_cast<uint8_t*>(sdl_surf->pixels), ::point(sdl_surf->w, sdl_surf->h));
359  auto cairo_context = cairo::create_context(cairo_surface);
360  cairo_t* ctx = cairo_context.get();
361 
362  cairo_set_antialias(ctx, CAIRO_ANTIALIAS_BEST);
363 
364  cairo_set_source_rgba(ctx,
365  c.r / 255.0,
366  c.g / 255.0,
367  c.b / 255.0,
368  c.a / 255.0
369  );
370  cairo_arc(ctx, r, r, r, 0, 2*2*boost::math::constants::pi<double>());
371  cairo_fill(ctx);
372 
373  draw::blit(texture(sdl_surf), ::rect(cx-r, cy-r, size, size));
374 }
375 
376 /*******************/
377 /* texture drawing */
378 /*******************/
379 
380 
381 void draw::blit(const texture& tex, const SDL_Rect& dst)
382 {
383  if (dst == sdl::empty_rect) {
384  return draw::blit(tex);
385  }
386 
387  if (!tex) { DBG_D << "null blit"; return; }
388  DBG_D << "blit " << dst;
389 
390  SDL_RenderCopy(renderer(), tex, tex.src(), &dst);
391 }
392 
393 void draw::blit(const texture& tex)
394 {
395  if (!tex) { DBG_D << "null blit"; return; }
396  DBG_D << "blit";
397 
398  SDL_RenderCopy(renderer(), tex, tex.src(), nullptr);
399 }
400 
401 
402 static SDL_RendererFlip get_flip(bool flip_h, bool flip_v)
403 {
404  // This should be easier than it is.
405  return static_cast<SDL_RendererFlip>(
406  static_cast<int>(flip_h ? SDL_FLIP_HORIZONTAL : SDL_FLIP_NONE)
407  | static_cast<int>(flip_v ? SDL_FLIP_VERTICAL : SDL_FLIP_NONE)
408  );
409 }
410 
412  const texture& tex,
413  const SDL_Rect& dst,
414  bool flip_h,
415  bool flip_v)
416 {
417  if (dst == sdl::empty_rect) {
418  return draw::flipped(tex, flip_h, flip_v);
419  }
420 
421  if (!tex) { DBG_D << "null flipped"; return; }
422  DBG_D << "flipped (" << flip_h << '|' << flip_v
423  << ") to " << dst;
424 
425  SDL_RendererFlip flip = get_flip(flip_h, flip_v);
426  SDL_RenderCopyEx(renderer(), tex, tex.src(), &dst, 0.0, nullptr, flip);
427 }
428 
429 void draw::flipped(const texture& tex, bool flip_h, bool flip_v)
430 {
431  if (!tex) { DBG_D << "null flipped"; return; }
432  DBG_D << "flipped (" << flip_h << '|' << flip_v << ')';
433 
434  SDL_RendererFlip flip = get_flip(flip_h, flip_v);
435  SDL_RenderCopyEx(renderer(), tex, tex.src(), nullptr, 0.0, nullptr, flip);
436 }
437 
438 
439 // TODO: highdpi - maybe expose this mirrored mode to WML somehow
440 void draw::tiled(const texture& tex, const SDL_Rect& dst, bool centered,
441  bool mirrored)
442 {
443  if (!tex) { DBG_D << "null tiled"; return; }
444  DBG_D << "tiled (" << centered << '|' << mirrored
445  << ") " << dst;
446 
447  // Reduce clip to dst.
448  auto clipper = draw::reduce_clip(dst);
449 
450  const int xoff = centered ? (dst.w - tex.w()) / 2 : 0;
451  const int yoff = centered ? (dst.h - tex.h()) / 2 : 0;
452 
453  // Just blit the image however many times is necessary.
454  bool vf = false;
455  SDL_Rect t{dst.x - xoff, dst.y - yoff, tex.w(), tex.h()};
456  for (; t.y < dst.y + dst.h; t.y += t.h, vf = !vf) {
457  bool hf = false;
458  for (t.x = dst.x - xoff; t.x < dst.x + dst.w; t.x += t.w, hf = !hf) {
459  if (mirrored) {
460  draw::flipped(tex, t, hf, vf);
461  } else {
462  draw::blit(tex, t);
463  }
464  }
465  }
466 }
467 
468 void draw::tiled_highres(const texture& tex, const SDL_Rect& dst,
469  bool centered, bool mirrored)
470 {
471  if (!tex) { DBG_D << "null tiled_highres"; return; }
472  DBG_D << "tiled_highres (" << centered << '|' << mirrored
473  << ") " << dst;
474 
475  const int pixel_scale = video::get_pixel_scale();
476 
477  // Reduce clip to dst.
478  auto clipper = draw::reduce_clip(dst);
479 
481  const float w = float(size.x) / float(pixel_scale);
482  const float h = float(size.y) / float(pixel_scale);
483  const float xoff = centered ? (dst.w - w) / 2 : 0.0f;
484  const float yoff = centered ? (dst.h - h) / 2 : 0.0f;
485 
486  // Just blit the image however many times is necessary.
487  bool vf = false;
488  SDL_FRect t{dst.x - xoff, dst.y - yoff, w, h};
489  for (; t.y < dst.y + dst.h; t.y += t.h, vf = !vf) {
490  bool hf = false;
491  for (t.x = dst.x - xoff; t.x < dst.x + dst.w; t.x += t.w, hf = !hf) {
492  if (mirrored) {
493  SDL_RendererFlip flip = get_flip(hf, vf);
494  SDL_RenderCopyExF(renderer(), tex, nullptr, &t, 0.0, nullptr, flip);
495  } else {
496  SDL_RenderCopyF(renderer(), tex, nullptr, &t);
497  }
498  }
499  }
500 }
501 
502 void draw::smooth_shaded(const texture& tex, const SDL_Rect& dst,
503  const SDL_Color& cTL, const SDL_Color& cTR,
504  const SDL_Color& cBL, const SDL_Color& cBR,
505  const SDL_FPoint& uvTL, const SDL_FPoint& uvTR,
506  const SDL_FPoint& uvBL, const SDL_FPoint& uvBR)
507 {
508  const SDL_FPoint pTL{float(dst.x), float(dst.y)};
509  const SDL_FPoint pTR{float(dst.x + dst.w), float(dst.y)};
510  const SDL_FPoint pBL{float(dst.x), float(dst.y + dst.h)};
511  const SDL_FPoint pBR{float(dst.x + dst.w), float(dst.y + dst.h)};
512  std::array<SDL_Vertex,4> verts {
513  SDL_Vertex{pTL, cTL, uvTL},
514  SDL_Vertex{pTR, cTR, uvTR},
515  SDL_Vertex{pBL, cBL, uvBL},
516  SDL_Vertex{pBR, cBR, uvBR},
517  };
518  draw::smooth_shaded(tex, verts);
519 }
520 
521 void draw::smooth_shaded(const texture& tex, const SDL_Rect& dst,
522  const SDL_Color& cTL, const SDL_Color& cTR,
523  const SDL_Color& cBL, const SDL_Color& cBR)
524 {
525  SDL_FPoint uv[4] = {
526  {0.f, 0.f}, // top left
527  {1.f, 0.f}, // top right
528  {0.f, 1.f}, // bottom left
529  {1.f, 1.f}, // bottom right
530  };
531  draw::smooth_shaded(tex, dst, cTL, cTR, cBL, cBR,
532  uv[0], uv[1], uv[2], uv[3]);
533 }
534 
535 void draw::smooth_shaded(const texture& tex,
536  const std::array<SDL_Vertex, 4>& verts)
537 {
538  DBG_D << "smooth shade, verts:";
539  for (const SDL_Vertex& v : verts) {
540  DBG_D << " {(" << v.position.x << ',' << v.position.y << ") "
541  << v.color << " (" << v.tex_coord.x << ',' << v.tex_coord.y
542  << ")}";
543  }
544  int indices[6] = {0, 1, 2, 2, 1, 3};
545  SDL_RenderGeometry(renderer(), tex, &verts[0], 4, indices, 6);
546 }
547 
548 /***************************/
549 /* RAII state manipulation */
550 /***************************/
551 
552 
553 draw::clip_setter::clip_setter(const SDL_Rect& clip)
554  : c_(draw::get_clip()), clip_enabled_(draw::clip_enabled())
555 {
556  draw::force_clip(clip);
557 }
558 
560 {
561  if (clip_enabled_) {
562  draw::force_clip(c_);
563  } else {
565  }
566 }
567 
569 {
570  return draw::clip_setter(clip);
571 }
572 
574 {
575  if (!draw::clip_enabled()) {
576  return draw::clip_setter(clip);
577  }
578  return draw::clip_setter(draw::get_clip().intersect(clip));
579 }
580 
581 void draw::force_clip(const SDL_Rect& clip)
582 {
583  // TODO: highdpi - fix whatever reason there is for this guard (CI fail)
584  if (!renderer()) {
585  WRN_D << "trying to force clip will null renderer";
586  return;
587  }
588  DBG_D << "forcing clip to " << clip;
589 
590  SDL_RenderSetClipRect(renderer(), &clip);
591 }
592 
594 {
595  // TODO: highdpi - fix whatever reason there is for this guard (CI fail)
596  if (!renderer()) {
597  return sdl::empty_rect;
598  }
599 
600  if (!SDL_RenderIsClipEnabled(renderer())) {
601  return draw::get_viewport();
602  }
603 
604  ::rect clip;
605  SDL_RenderGetClipRect(renderer(), &clip);
606  return clip;
607 }
608 
610 {
611  if (!renderer()) {
612  return false;
613  }
614  return SDL_RenderIsClipEnabled(renderer());
615 }
616 
618 {
619  if (!renderer()) {
620  return;
621  }
622  SDL_RenderSetClipRect(renderer(), nullptr);
623  DBG_D << "clip disabled";
624 }
625 
627 {
628  if (!renderer()) {
629  return true;
630  }
631  if (!SDL_RenderIsClipEnabled(renderer())) {
632  return false;
633  }
634  SDL_Rect clip;
635  SDL_RenderGetClipRect(renderer(), &clip);
636  return clip.w <= 0 || clip.h <= 0;
637 }
638 
639 
641  : v_(), c_(), clip_enabled_(draw::clip_enabled())
642 {
644  draw::force_viewport(view);
645  if (clip_enabled_) {
646  c_ = draw::get_clip();
647  // adjust clip for difference in viewport position
648  SDL_Rect c_view = {
649  c_.x + v_.x - view.x,
650  c_.y + v_.y - view.y,
651  c_.w, c_.h
652  };
653  draw::force_clip(c_view);
654  }
655 }
656 
658 {
660  if (clip_enabled_) {
661  draw::force_clip(c_);
662  } else {
664  }
665 }
666 
667 draw::viewport_setter draw::set_viewport(const SDL_Rect& viewport)
668 {
669  return draw::viewport_setter(viewport);
670 }
671 
672 void draw::force_viewport(const SDL_Rect& viewport)
673 {
674  if (!renderer()) {
675  WRN_D << "trying to force viewport will null renderer";
676  return;
677  }
678  DBG_D << "forcing viewport to " << viewport;
679 
680  SDL_RenderSetViewport(renderer(), &viewport);
681 }
682 
684 {
685  if (!renderer()) {
686  WRN_D << "no renderer available to get viewport";
687  return sdl::empty_rect;
688  }
689 
690  SDL_Rect viewport;
691  SDL_RenderGetViewport(renderer(), &viewport);
692 
693  if (viewport == sdl::empty_rect) {
694  return video::draw_area();
695  }
696  return viewport;
697 }
698 
700  : target_()
701  , viewport_()
702  , clip_()
703 {
704  // Validate we can render to this texture.
705  assert(!t || t.get_access() == SDL_TEXTUREACCESS_TARGET);
706 
707  if (!renderer()) {
708  WRN_D << "can't set render target with null renderer";
709  return;
710  }
711 
713  SDL_RenderGetViewport(renderer(), &viewport_);
714  SDL_RenderGetClipRect(renderer(), &clip_);
715 
716  if (t) {
718  } else {
720  }
721 }
722 
724 {
725  if (!renderer()) {
726  WRN_D << "can't reset render target with null renderer";
727  return;
728  }
730  SDL_RenderSetViewport(renderer(), &viewport_);
731  if(clip_ == sdl::empty_rect) return;
732  SDL_RenderSetClipRect(renderer(), &clip_);
733 }
734 
736 {
737  if (t) {
738  DBG_D << "setting render target to "
739  << t.w() << 'x' << t.h() << " texture";
740  } else {
741  DBG_D << "setting render target to main render buffer";
742  }
744 }
double t
Definition: astarsearch.cpp:63
double g
Definition: astarsearch.cpp:63
A class to manage automatic restoration of the clipping region.
Definition: draw.hpp:346
clip_setter(const SDL_Rect &clip)
Definition: draw.cpp:553
A class to manage automatic restoration of the render target.
Definition: draw.hpp:478
render_target_setter(const texture &t)
Definition: draw.cpp:699
A class to manage automatic restoration of the viewport region.
Definition: draw.hpp:418
viewport_setter(const SDL_Rect &viewport)
Definition: draw.cpp:640
Wrapper class to encapsulate creation and management of an SDL_Texture.
Definition: texture.hpp:33
int w() const
The draw-space width of the texture, in pixels.
Definition: texture.hpp:103
point get_raw_size() const
The raw internal texture size.
Definition: texture.cpp:99
const rect * src() const
A pointer to a rect indicating the source region of the underlying SDL_Texture to be used when drawin...
Definition: texture.hpp:146
int h() const
The draw-space height of the texture, in pixels.
Definition: texture.hpp:112
static lg::log_domain log_draw("draw")
static bool sdl_bad_at_rects()
Some versions of SDL have a bad rectangle drawing implementation.
Definition: draw.cpp:134
static SDL_Renderer * renderer()
Definition: draw.cpp:34
static void draw_rect_as_lines(const SDL_Rect &rect)
For some SDL versions, draw rectangles as lines.
Definition: draw.cpp:145
#define WRN_D
Definition: draw.cpp:32
static SDL_RendererFlip get_flip(bool flip_h, bool flip_v)
Definition: draw.cpp:402
#define DBG_D
Definition: draw.cpp:31
Drawing functions, for drawing things on the screen.
const gui2::tracked_drawable & target_
The drawable whose render calls we are tracking.
Definition: fps_report.cpp:53
int w
Standard logging facilities (interface).
context_ptr create_context(const surface_ptr &surf)
Definition: cairo.hpp:41
surface_ptr create_surface(uint8_t *buffer, const point &size)
Definition: cairo.hpp:30
Definition: draw.hpp:43
viewport_setter set_viewport(const SDL_Rect &viewport)
Set the viewport.
Definition: draw.cpp:667
void force_viewport(const SDL_Rect &viewport)
Set the viewport, without any provided way of setting it back.
Definition: draw.cpp:672
render_target_setter set_render_target(const texture &t)
Set the given texture as the active render target.
Definition: draw.cpp:735
clip_setter override_clip(const SDL_Rect &clip)
Override the clipping area.
Definition: draw.cpp:568
void circle(int x, int y, int r, const color_t &c, uint8_t octants=0xff)
Draw a circle of the given colour.
Definition: draw.cpp:218
void points(const std::vector< SDL_Point > &points)
Draw a set of points.
Definition: draw.cpp:206
void force_clip(const SDL_Rect &clip)
Set the clipping area, without any provided way of setting it back.
Definition: draw.cpp:581
SDL_Rect get_viewport()
Get the current viewport.
Definition: draw.cpp:683
void cairo_disc(int cx, int cy, int r, const color_t &c)
Draw filled circle using Cairo.
Definition: draw.cpp:349
bool null_clip()
Whether the current clipping region will disallow drawing.
Definition: draw.cpp:626
clip_setter reduce_clip(const SDL_Rect &clip)
Set the clipping area to the intersection of the current clipping area and the given rectangle.
Definition: draw.cpp:573
bool clip_enabled()
Whether clipping is enabled.
Definition: draw.cpp:609
void cairo_circle(int cx, int cy, int r, const color_t &c, int thickness)
Draw outline of circle using Cairo.
Definition: draw.cpp:318
void tiled(const texture &tex, const SDL_Rect &dst, bool centered=false, bool mirrored=false)
Tile a texture to fill a region.
Definition: draw.cpp:440
void disc(int x, int y, int r, const color_t &c, uint8_t octants=0xff)
Draw a solid disc of the given colour.
Definition: draw.cpp:260
void set_color(uint8_t r, uint8_t g, uint8_t b, uint8_t a)
Set the drawing colour.
Definition: draw.cpp:110
void flipped(const texture &tex, const SDL_Rect &dst, bool flip_h=true, bool flip_v=false)
Draws a texture, or part of a texture, at the given location, also mirroring/flipping the texture hor...
Definition: draw.cpp:411
void fill(const SDL_Rect &rect, uint8_t r, uint8_t g, uint8_t b, uint8_t a)
Fill an area with the given colour.
Definition: draw.cpp:53
void point(int x, int y)
Draw a single point.
Definition: draw.cpp:212
void clear()
Clear the current render target.
Definition: draw.cpp:43
void set_blend_mode(SDL_BlendMode b)
Set the blend mode used for drawing operations such as fill() and line().
Definition: draw.cpp:128
void blit(const texture &tex, const SDL_Rect &dst)
Draws a texture, or part of a texture, at the given location.
Definition: draw.cpp:381
void tiled_highres(const texture &tex, const SDL_Rect &dst, bool centered=false, bool mirrored=false)
Tile a texture to fill a region.
Definition: draw.cpp:468
::rect get_clip()
Get the current clipping area, in draw coordinates.
Definition: draw.cpp:593
void rect(const SDL_Rect &rect)
Draw a rectangle.
Definition: draw.cpp:160
void disable_clip()
Disable clipping.
Definition: draw.cpp:617
void line(int from_x, int from_y, int to_x, int to_y)
Draw a line.
Definition: draw.cpp:190
void smooth_shaded(const texture &tex, const SDL_Rect &dst, const SDL_Color &cTL, const SDL_Color &cTR, const SDL_Color &cBL, const SDL_Color &cBR, const SDL_FPoint &uvTL, const SDL_FPoint &uvTR, const SDL_FPoint &uvBL, const SDL_FPoint &uvBR)
Draw a texture with smoothly varying colour and alpha modification, specified at the four corners of ...
Definition: draw.cpp:502
constexpr const SDL_Rect empty_rect
Definition: rect.hpp:32
bool runtime_at_least(uint8_t major, uint8_t minor=0, uint8_t patch=0)
Returns true if the runtime SDL version is at or greater than the specified version,...
Definition: utils.cpp:43
std::size_t size(std::string_view str)
Length in characters of a UTF-8 string.
Definition: unicode.cpp:85
rect draw_area()
The current drawable area.
Definition: video.cpp:441
void reset_render_target()
Reset the render target to the primary render buffer.
Definition: video.cpp:536
void force_render_target(const texture &t)
Set the render target, without any provided way of setting it back.
Definition: video.cpp:500
int get_pixel_scale()
Get the current active pixel scale multiplier.
Definition: video.cpp:480
SDL_Renderer * get_renderer()
Definition: video.cpp:658
texture get_render_target()
Get the current render target.
Definition: video.cpp:541
Contains the SDL_Rect helper code.
rect dst
Location on the final composed sheet.
The basic class for representing 8-bit RGB or RGBA colour values.
Definition: color.hpp:59
Holds a 2D point.
Definition: point.hpp:25
An abstract description of a rectangle with integer coordinates.
Definition: rect.hpp:49
mock_char c
#define d
#define h
#define b