25 #include <boost/math/constants/constants.hpp>
26 #include <SDL2/SDL_rect.h>
27 #include <SDL2/SDL_render.h>
31 #define DBG_D LOG_STREAM(debug, log_draw)
32 #define WRN_D LOG_STREAM(warn, log_draw)
47 SDL_GetRenderDrawBlendMode(
renderer(), &
b);
48 SDL_SetRenderDrawBlendMode(
renderer(), SDL_BLENDMODE_NONE);
50 SDL_SetRenderDrawBlendMode(
renderer(),
b);
55 uint8_t r, uint8_t
g, uint8_t
b, uint8_t a)
58 SDL_SetRenderDrawColor(
renderer(), r,
g,
b, a);
59 SDL_RenderFillRect(
renderer(), &area);
64 uint8_t r, uint8_t
g, uint8_t
b)
77 SDL_SetRenderDrawColor(
renderer(), r,
g,
b, a);
78 SDL_RenderFillRect(
renderer(),
nullptr);
93 DBG_D <<
"sub-pixel fill";
100 DBG_D <<
"fill " << area;
101 SDL_RenderFillRect(
renderer(), &area);
107 SDL_RenderFillRect(
renderer(),
nullptr);
113 SDL_SetRenderDrawColor(
renderer(), r,
g,
b, a);
119 SDL_SetRenderDrawColor(
renderer(), r,
g,
b, SDL_ALPHA_OPAQUE);
124 DBG_D <<
"set color " <<
c;
130 SDL_SetRenderDrawBlendMode(
renderer(),
b);
170 uint8_t r, uint8_t
g, uint8_t
b, uint8_t a)
173 SDL_SetRenderDrawColor(
renderer(), r,
g,
b, a);
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);
199 DBG_D <<
"line from (" << from_x <<
',' << from_y
200 <<
") to (" << to_x <<
',' << to_y
201 <<
") with colour " <<
c;
203 SDL_RenderDrawLine(
renderer(), from_x, from_y, to_x, to_y);
214 DBG_D <<
"point (" << x <<
',' << y <<
')';
215 SDL_RenderDrawPoint(
renderer(), x, y);
226 DBG_D <<
"circle (" << cx <<
',' << cy
227 <<
") -> " << r <<
", oct " << int(octants);
236 std::vector<SDL_Point>
points;
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});
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});
268 DBG_D <<
"disc (" << cx <<
',' << cy
269 <<
") -> " << r <<
", oct " << int(octants);
281 draw::line(cx + x, cy + y + 1, cx + y + 1, cy + y + 1);
284 draw::line(cx + x, cy - y, cx + y + 1, cy - y);
287 draw::line(cx - x - 1, cy + y + 1, cx - y - 2, cy + y + 1);
290 draw::line(cx - x - 1, cy - y, cx - y - 2, cy - y);
294 draw::line(cx + y, cy + x + 1, cx + y, cy + y + 1);
300 draw::line(cx - y - 1, cy + x + 1, cx - y - 1, cy + y + 1);
303 draw::line(cx - y - 1, cy - x, cx - y - 1, cy - y);
327 reinterpret_cast<uint8_t*
>(sdl_surf->pixels), ::
point(sdl_surf->w, sdl_surf->h));
329 cairo_t* ctx = cairo_context.get();
331 cairo_set_antialias(ctx, CAIRO_ANTIALIAS_BEST);
333 cairo_set_source_rgba(ctx, 0.0, 0.0, 0.0, 0.0);
336 cairo_set_line_width(ctx, thickness);
337 cairo_set_source_rgba(ctx,
343 cairo_arc(ctx, r, r, r-thickness, 0, 2*boost::math::constants::pi<double>());
358 reinterpret_cast<uint8_t*
>(sdl_surf->pixels), ::
point(sdl_surf->w, sdl_surf->h));
360 cairo_t* ctx = cairo_context.get();
362 cairo_set_antialias(ctx, CAIRO_ANTIALIAS_BEST);
364 cairo_set_source_rgba(ctx,
370 cairo_arc(ctx, r, r, r, 0, 2*2*boost::math::constants::pi<double>());
387 if (!tex) {
DBG_D <<
"null blit";
return; }
395 if (!tex) {
DBG_D <<
"null blit";
return; }
398 SDL_RenderCopy(
renderer(), tex, tex.
src(),
nullptr);
402 static SDL_RendererFlip
get_flip(
bool flip_h,
bool flip_v)
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)
421 if (!tex) {
DBG_D <<
"null flipped";
return; }
422 DBG_D <<
"flipped (" << flip_h <<
'|' << flip_v
425 SDL_RendererFlip flip =
get_flip(flip_h, flip_v);
426 SDL_RenderCopyEx(
renderer(), tex, tex.
src(), &
dst, 0.0,
nullptr, flip);
431 if (!tex) {
DBG_D <<
"null flipped";
return; }
432 DBG_D <<
"flipped (" << flip_h <<
'|' << flip_v <<
')';
434 SDL_RendererFlip flip =
get_flip(flip_h, flip_v);
435 SDL_RenderCopyEx(
renderer(), tex, tex.
src(),
nullptr, 0.0,
nullptr, flip);
443 if (!tex) {
DBG_D <<
"null tiled";
return; }
444 DBG_D <<
"tiled (" << centered <<
'|' << mirrored
450 const int xoff = centered ? (
dst.w - tex.
w()) / 2 : 0;
451 const int yoff = centered ? (
dst.h - tex.
h()) / 2 : 0;
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) {
458 for (
t.x =
dst.x - xoff;
t.x <
dst.x +
dst.w;
t.x +=
t.w, hf = !hf) {
469 bool centered,
bool mirrored)
471 if (!tex) {
DBG_D <<
"null tiled_highres";
return; }
472 DBG_D <<
"tiled_highres (" << centered <<
'|' << mirrored
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;
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) {
491 for (
t.x =
dst.x - xoff;
t.x <
dst.x +
dst.w;
t.x +=
t.w, hf = !hf) {
493 SDL_RendererFlip flip =
get_flip(hf, vf);
494 SDL_RenderCopyExF(
renderer(), tex,
nullptr, &
t, 0.0,
nullptr, flip);
496 SDL_RenderCopyF(
renderer(), tex,
nullptr, &
t);
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)
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},
522 const SDL_Color& cTL,
const SDL_Color& cTR,
523 const SDL_Color& cBL,
const SDL_Color& cBR)
532 uv[0], uv[1], uv[2], uv[3]);
536 const std::array<SDL_Vertex, 4>& verts)
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
544 int indices[6] = {0, 1, 2, 2, 1, 3};
545 SDL_RenderGeometry(
renderer(), tex, &verts[0], 4, indices, 6);
585 WRN_D <<
"trying to force clip will null renderer";
588 DBG_D <<
"forcing clip to " << clip;
590 SDL_RenderSetClipRect(
renderer(), &clip);
600 if (!SDL_RenderIsClipEnabled(
renderer())) {
605 SDL_RenderGetClipRect(
renderer(), &clip);
614 return SDL_RenderIsClipEnabled(
renderer());
622 SDL_RenderSetClipRect(
renderer(),
nullptr);
623 DBG_D <<
"clip disabled";
631 if (!SDL_RenderIsClipEnabled(
renderer())) {
635 SDL_RenderGetClipRect(
renderer(), &clip);
636 return clip.w <= 0 || clip.h <= 0;
649 c_.x +
v_.x - view.x,
650 c_.y +
v_.y - view.y,
675 WRN_D <<
"trying to force viewport will null renderer";
678 DBG_D <<
"forcing viewport to " << viewport;
680 SDL_RenderSetViewport(
renderer(), &viewport);
686 WRN_D <<
"no renderer available to get viewport";
691 SDL_RenderGetViewport(
renderer(), &viewport);
705 assert(!
t ||
t.get_access() == SDL_TEXTUREACCESS_TARGET);
708 WRN_D <<
"can't set render target with null renderer";
726 WRN_D <<
"can't reset render target with null renderer";
730 SDL_RenderSetViewport(
renderer(), &viewport_);
732 SDL_RenderSetClipRect(
renderer(), &clip_);
738 DBG_D <<
"setting render target to "
739 <<
t.w() <<
'x' <<
t.h() <<
" texture";
741 DBG_D <<
"setting render target to main render buffer";
A class to manage automatic restoration of the clipping region.
clip_setter(const SDL_Rect &clip)
A class to manage automatic restoration of the render target.
render_target_setter(const texture &t)
A class to manage automatic restoration of the viewport region.
viewport_setter(const SDL_Rect &viewport)
Wrapper class to encapsulate creation and management of an SDL_Texture.
int w() const
The draw-space width of the texture, in pixels.
point get_raw_size() const
The raw internal texture size.
const rect * src() const
A pointer to a rect indicating the source region of the underlying SDL_Texture to be used when drawin...
int h() const
The draw-space height of the texture, in pixels.
static lg::log_domain log_draw("draw")
static bool sdl_bad_at_rects()
Some versions of SDL have a bad rectangle drawing implementation.
static SDL_Renderer * renderer()
static void draw_rect_as_lines(const SDL_Rect &rect)
For some SDL versions, draw rectangles as lines.
static SDL_RendererFlip get_flip(bool flip_h, bool flip_v)
Drawing functions, for drawing things on the screen.
const gui2::tracked_drawable & target_
The drawable whose render calls we are tracking.
Standard logging facilities (interface).
context_ptr create_context(const surface_ptr &surf)
surface_ptr create_surface(uint8_t *buffer, const point &size)
viewport_setter set_viewport(const SDL_Rect &viewport)
Set the viewport.
void force_viewport(const SDL_Rect &viewport)
Set the viewport, without any provided way of setting it back.
render_target_setter set_render_target(const texture &t)
Set the given texture as the active render target.
clip_setter override_clip(const SDL_Rect &clip)
Override the clipping area.
void circle(int x, int y, int r, const color_t &c, uint8_t octants=0xff)
Draw a circle of the given colour.
void points(const std::vector< SDL_Point > &points)
Draw a set of points.
void force_clip(const SDL_Rect &clip)
Set the clipping area, without any provided way of setting it back.
SDL_Rect get_viewport()
Get the current viewport.
void cairo_disc(int cx, int cy, int r, const color_t &c)
Draw filled circle using Cairo.
bool null_clip()
Whether the current clipping region will disallow drawing.
clip_setter reduce_clip(const SDL_Rect &clip)
Set the clipping area to the intersection of the current clipping area and the given rectangle.
bool clip_enabled()
Whether clipping is enabled.
void cairo_circle(int cx, int cy, int r, const color_t &c, int thickness)
Draw outline of circle using Cairo.
void tiled(const texture &tex, const SDL_Rect &dst, bool centered=false, bool mirrored=false)
Tile a texture to fill a region.
void disc(int x, int y, int r, const color_t &c, uint8_t octants=0xff)
Draw a solid disc of the given colour.
void set_color(uint8_t r, uint8_t g, uint8_t b, uint8_t a)
Set the drawing colour.
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...
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.
void point(int x, int y)
Draw a single point.
void clear()
Clear the current render target.
void set_blend_mode(SDL_BlendMode b)
Set the blend mode used for drawing operations such as fill() and line().
void blit(const texture &tex, const SDL_Rect &dst)
Draws a texture, or part of a texture, at the given location.
void tiled_highres(const texture &tex, const SDL_Rect &dst, bool centered=false, bool mirrored=false)
Tile a texture to fill a region.
::rect get_clip()
Get the current clipping area, in draw coordinates.
void rect(const SDL_Rect &rect)
Draw a rectangle.
void disable_clip()
Disable clipping.
void line(int from_x, int from_y, int to_x, int to_y)
Draw a line.
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 ...
constexpr const SDL_Rect empty_rect
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,...
std::size_t size(std::string_view str)
Length in characters of a UTF-8 string.
rect draw_area()
The current drawable area.
void reset_render_target()
Reset the render target to the primary render buffer.
void force_render_target(const texture &t)
Set the render target, without any provided way of setting it back.
int get_pixel_scale()
Get the current active pixel scale multiplier.
SDL_Renderer * get_renderer()
texture get_render_target()
Get the current render target.
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.
An abstract description of a rectangle with integer coordinates.