use crate::types::{Color, Fixed, Point, Rect, Transform, Viewport};
use super::canvas::Canvas;
use super::command::DrawCommand;
use super::path::Path;
use super::renderer::Renderer;
use super::texture::Texture;
#[cfg(feature = "perf")]
pub mod perf;
#[cfg(feature = "perf")]
pub use perf::{PerfCtx, quad_perf};
mod blit_dispatch;
mod blit_fast;
mod label;
mod path;
mod quad;
mod quad_aa;
mod rect_fill;
mod rect_stroke;
mod transformed;
use quad::{blit_quad, fill_rect_quad, stroke_rect_quad};
use transformed::{blit_transformed, fill_rect_transformed, offset_point, offset_rect};
pub struct SwRenderer<'a> {
pub target: Texture<'a>,
pub viewport: Viewport,
#[cfg(feature = "perf")]
pub perf: Option<PerfCtx>,
}
impl<'a> SwRenderer<'a> {
pub fn new(target: Texture<'a>) -> Self {
let w = target.width;
let h = target.height;
Self {
target,
viewport: Viewport::new(w, h, Fixed::ONE),
#[cfg(feature = "perf")]
perf: None,
}
}
}
impl<'a> SwRenderer<'a> {
fn draw_transformed(&mut self, cmd: &DrawCommand, clip: &Rect, tf: &Transform) {
let vp = self.viewport.as_transform();
let phys_tf = vp.compose(tf);
let phys_clip = self.viewport.rect_to_physical(*clip);
match cmd {
DrawCommand::Fill {
area, color, opa, ..
} => {
let phys_area = self.viewport.rect_to_physical(*area);
fill_rect_transformed(
&mut self.target,
phys_area,
phys_clip,
&phys_tf,
color,
*opa,
);
}
DrawCommand::Blit {
pos, size, texture, ..
} => {
let phys_pos = self.viewport.point_to_physical(*pos);
let phys_size = self.viewport.point_to_physical(*size);
let src_rect = Rect::new(0, 0, texture.width, texture.height);
let phys_dst = Rect {
x: phys_pos.x,
y: phys_pos.y,
w: phys_size.x,
h: phys_size.y,
};
blit_transformed(
&mut self.target,
texture,
&src_rect,
phys_dst,
phys_clip,
&phys_tf,
);
}
_ => unimplemented!(
"sw backend: {:?} under non-axis-aligned transform not yet supported",
core::mem::discriminant(cmd)
),
}
}
}
impl<'a> Canvas for SwRenderer<'a> {
fn fill_path(&mut self, path: &Path, clip: &Rect, color: &Color, opa: u8) {
self.fill_path_inner(path, clip, color, opa);
}
fn stroke_path(&mut self, path: &Path, clip: &Rect, width: Fixed, color: &Color, opa: u8) {
self.stroke_path_inner(path, clip, width, color, opa);
}
fn fill_rect(&mut self, area: &Rect, clip: &Rect, color: &Color, radius: Fixed, opa: u8) {
self.fill_rect_inner(area, clip, color, radius, opa);
}
fn stroke_rect(
&mut self,
area: &Rect,
clip: &Rect,
width: Fixed,
color: &Color,
radius: Fixed,
opa: u8,
) {
self.stroke_rect_inner(area, clip, width, color, radius, opa);
}
fn blit(&mut self, src: &Texture, src_rect: &Rect, dst: Point, dst_size: Point, clip: &Rect) {
self.blit_inner(src, src_rect, dst, dst_size, clip);
}
fn clear(&mut self, area: &Rect, color: &Color) {
let phys_area = self.viewport.rect_to_physical(*area);
let screen = Rect::new(0, 0, self.target.width, self.target.height);
let Some(draw_area) = phys_area.intersect(&screen) else {
return;
};
let (px_x0, px_y0, px_x1, px_y1) = draw_area.pixel_bounds();
for py in px_y0..px_y1 {
for px in px_x0..px_x1 {
self.target.set_pixel(px, py, color);
}
}
}
fn draw_label(&mut self, pos: &Point, text: &[u8], clip: &Rect, color: &Color, opa: u8) {
self.draw_label_inner(pos, text, clip, color, opa);
}
fn flush(&mut self) {}
}
impl SwRenderer<'_> {
fn dispatch_fill_quad(
&mut self,
q: &[Point; 4],
area: &Rect,
color: &Color,
radius: Fixed,
opa: u8,
clip: &Rect,
) {
#[cfg(feature = "perf")]
let t0 = quad_perf::now();
let phys_clip = self.viewport.rect_to_physical(*clip);
let phys_q = [
self.viewport.point_to_physical(q[0]),
self.viewport.point_to_physical(q[1]),
self.viewport.point_to_physical(q[2]),
self.viewport.point_to_physical(q[3]),
];
let s = self.viewport.scale();
fill_rect_quad(
&mut self.target,
&phys_q,
phys_clip,
color,
radius * s,
area.w * s,
area.h * s,
opa,
);
#[cfg(feature = "perf")]
quad_perf::add_fill(quad_perf::now().wrapping_sub(t0));
}
fn dispatch_blit_quad(&mut self, q: &[Point; 4], texture: &Texture, clip: &Rect) {
#[cfg(feature = "perf")]
let t0 = quad_perf::now();
let phys_clip = self.viewport.rect_to_physical(*clip);
let phys_q = [
self.viewport.point_to_physical(q[0]),
self.viewport.point_to_physical(q[1]),
self.viewport.point_to_physical(q[2]),
self.viewport.point_to_physical(q[3]),
];
blit_quad(&mut self.target, texture, &phys_q, phys_clip);
#[cfg(feature = "perf")]
quad_perf::add_blit(quad_perf::now().wrapping_sub(t0));
}
fn dispatch_border_quad(
&mut self,
q: &[Point; 4],
color: &Color,
width: Fixed,
radius: Fixed,
opa: u8,
clip: &Rect,
) {
let phys_clip = self.viewport.rect_to_physical(*clip);
let phys_q = [
self.viewport.point_to_physical(q[0]),
self.viewport.point_to_physical(q[1]),
self.viewport.point_to_physical(q[2]),
self.viewport.point_to_physical(q[3]),
];
let s = self.viewport.scale();
stroke_rect_quad(
&mut self.target,
&phys_q,
phys_clip,
color,
width * s,
radius * s,
opa,
);
}
#[allow(clippy::too_many_arguments)]
fn dispatch_fill(
&mut self,
area: &Rect,
color: &Color,
radius: Fixed,
opa: u8,
tx: Fixed,
ty: Fixed,
clip: &Rect,
) {
#[cfg(feature = "perf")]
let t0 = self.perf.as_ref().map(|p| (p.clock)());
let area = offset_rect(area, tx, ty);
self.fill_rect(&area, clip, color, radius, opa);
#[cfg(feature = "perf")]
if let (Some(t0), Some(p)) = (t0, self.perf.as_mut()) {
p.fill += (p.clock)() - t0;
p.count_fill += 1;
}
}
#[allow(clippy::too_many_arguments)]
fn dispatch_border(
&mut self,
area: &Rect,
color: &Color,
width: Fixed,
radius: Fixed,
opa: u8,
tx: Fixed,
ty: Fixed,
clip: &Rect,
) {
#[cfg(feature = "perf")]
let t0 = self.perf.as_ref().map(|p| (p.clock)());
let area = offset_rect(area, tx, ty);
self.stroke_rect(&area, clip, width, color, radius, opa);
#[cfg(feature = "perf")]
if let (Some(t0), Some(p)) = (t0, self.perf.as_mut()) {
p.stroke += (p.clock)() - t0;
p.count_stroke += 1;
}
}
fn dispatch_blit(
&mut self,
pos: &Point,
size: Point,
texture: &Texture,
tx: Fixed,
ty: Fixed,
clip: &Rect,
) {
#[cfg(feature = "perf")]
let t0 = self.perf.as_ref().map(|p| (p.clock)());
let src_rect = Rect::new(0, 0, texture.width, texture.height);
let pos = offset_point(pos, tx, ty);
self.blit(texture, &src_rect, pos, size, clip);
#[cfg(feature = "perf")]
if let (Some(t0), Some(p)) = (t0, self.perf.as_mut()) {
p.blit += (p.clock)() - t0;
p.count_blit += 1;
}
}
#[allow(clippy::too_many_arguments)]
fn dispatch_label(
&mut self,
pos: &Point,
text: &[u8],
color: &Color,
opa: u8,
tx: Fixed,
ty: Fixed,
clip: &Rect,
) {
#[cfg(feature = "perf")]
let t0 = self.perf.as_ref().map(|p| (p.clock)());
let pos = offset_point(pos, tx, ty);
self.draw_label(&pos, text, clip, color, opa);
#[cfg(feature = "perf")]
if let (Some(t0), Some(p)) = (t0, self.perf.as_mut()) {
p.label += (p.clock)() - t0;
p.count_label += 1;
}
}
#[allow(clippy::too_many_arguments)]
fn dispatch_line(
&mut self,
p1: &Point,
p2: &Point,
color: &Color,
width: Fixed,
opa: u8,
tx: Fixed,
ty: Fixed,
clip: &Rect,
) {
#[cfg(feature = "perf")]
let t0 = self.perf.as_ref().map(|p| (p.clock)());
let p1 = offset_point(p1, tx, ty);
let p2 = offset_point(p2, tx, ty);
self.draw_line(p1, p2, clip, width, color, opa);
#[cfg(feature = "perf")]
if let (Some(t0), Some(p)) = (t0, self.perf.as_mut()) {
p.stroke += (p.clock)() - t0;
p.count_stroke += 1;
}
}
#[allow(clippy::too_many_arguments)]
fn dispatch_arc(
&mut self,
center: &Point,
radius: Fixed,
start_angle: Fixed,
end_angle: Fixed,
color: &Color,
width: Fixed,
opa: u8,
tx: Fixed,
ty: Fixed,
clip: &Rect,
) {
#[cfg(feature = "perf")]
let t0 = self.perf.as_ref().map(|p| (p.clock)());
let center = offset_point(center, tx, ty);
self.draw_arc(
center,
radius,
start_angle,
end_angle,
clip,
width,
color,
opa,
);
#[cfg(feature = "perf")]
if let (Some(t0), Some(p)) = (t0, self.perf.as_mut()) {
p.stroke += (p.clock)() - t0;
p.count_stroke += 1;
}
}
}
impl Renderer for SwRenderer<'_> {
fn draw(&mut self, cmd: &DrawCommand, clip: &Rect) {
use crate::types::TransformClass;
if let DrawCommand::Fill {
area,
quad: Some(q),
color,
opa,
radius,
..
} = cmd
{
self.dispatch_fill_quad(q, area, color, *radius, *opa, clip);
return;
}
if let DrawCommand::Blit {
quad: Some(q),
texture,
..
} = cmd
{
self.dispatch_blit_quad(q, texture, clip);
return;
}
if let DrawCommand::Border {
quad: Some(q),
color,
width,
radius,
opa,
..
} = cmd
{
self.dispatch_border_quad(q, color, *width, *radius, *opa, clip);
return;
}
let tf = cmd.transform();
let class = tf.classify();
if !matches!(class, TransformClass::Identity | TransformClass::Translate) {
self.draw_transformed(cmd, clip, &tf);
return;
}
let (tx, ty) = match class {
TransformClass::Identity => (Fixed::ZERO, Fixed::ZERO),
TransformClass::Translate => (tf.tx, tf.ty),
_ => unreachable!(),
};
match cmd {
DrawCommand::Fill {
area,
color,
radius,
opa,
..
} => self.dispatch_fill(area, color, *radius, *opa, tx, ty, clip),
DrawCommand::Border {
area,
color,
width,
radius,
opa,
..
} => self.dispatch_border(area, color, *width, *radius, *opa, tx, ty, clip),
DrawCommand::Blit {
pos, size, texture, ..
} => self.dispatch_blit(pos, *size, texture, tx, ty, clip),
DrawCommand::Label {
pos,
text,
color,
opa,
..
} => self.dispatch_label(pos, text, color, *opa, tx, ty, clip),
DrawCommand::Line {
p1,
p2,
color,
width,
opa,
..
} => self.dispatch_line(p1, p2, color, *width, *opa, tx, ty, clip),
DrawCommand::Arc {
center,
radius,
start_angle,
end_angle,
color,
width,
opa,
..
} => self.dispatch_arc(
center,
*radius,
*start_angle,
*end_angle,
color,
*width,
*opa,
tx,
ty,
clip,
),
}
}
fn flush(&mut self) {
Canvas::flush(self);
}
}
#[cfg(test)]
mod tests {
use super::blit_fast::{blit_1to1_fast, blit_2to2_fast, blit_dda, blit_generic_slow};
use super::*;
use crate::draw::texture::ColorFormat;
use alloc::vec;
#[test]
fn fill_rect_basic() {
let mut buf = vec![0u8; 16 * 16 * 4];
let tex = Texture::new(&mut buf, 16, 16, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let rect = Rect::new(2, 2, 4, 4);
let clip = Rect::new(0, 0, 16, 16);
backend.fill_rect(&rect, &clip, &Color::rgb(255, 0, 0), Fixed::ZERO, 255);
let c = backend.target.get_pixel(3, 3);
assert_eq!(c.r, 255);
assert_eq!(c.g, 0);
assert_eq!(c.b, 0);
let c = backend.target.get_pixel(0, 0);
assert_eq!(c.r, 0);
}
#[test]
fn clear_fills_area() {
let mut buf = vec![0u8; 8 * 8 * 4];
let tex = Texture::new(&mut buf, 8, 8, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
backend.clear(&Rect::new(0, 0, 8, 8), &Color::rgb(50, 100, 150));
let c = backend.target.get_pixel(4, 4);
assert_eq!(c.r, 50);
assert_eq!(c.g, 100);
assert_eq!(c.b, 150);
}
#[test]
fn fill_path_rect_matches_fill_rect() {
let mut buf = vec![0u8; 16 * 16 * 4];
let tex = Texture::new(&mut buf, 16, 16, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let path = super::super::path::Path::rect(
Fixed::from_int(2),
Fixed::from_int(2),
Fixed::from_int(8),
Fixed::from_int(8),
);
let clip = Rect::new(0, 0, 16, 16);
backend.fill_path(&path, &clip, &Color::rgb(0, 0, 255), 255);
let c = backend.target.get_pixel(5, 5);
assert_eq!(c.b, 255);
assert_eq!(c.r, 0);
let c = backend.target.get_pixel(0, 0);
assert_eq!(c.b, 0);
}
#[test]
fn fill_path_empty_is_noop() {
let mut buf = vec![0u8; 4 * 4 * 4];
let tex = Texture::new(&mut buf, 4, 4, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let path = super::super::path::Path::new();
let clip = Rect::new(0, 0, 4, 4);
backend.fill_path(&path, &clip, &Color::rgb(255, 255, 255), 255);
for y in 0..4 {
for x in 0..4 {
assert_eq!(backend.target.get_pixel(x, y).r, 0);
}
}
}
#[test]
fn fill_path_zero_opa_is_noop() {
let mut buf = vec![0u8; 4 * 4 * 4];
let tex = Texture::new(&mut buf, 4, 4, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let path = super::super::path::Path::rect(
Fixed::ZERO,
Fixed::ZERO,
Fixed::from_int(4),
Fixed::from_int(4),
);
let clip = Rect::new(0, 0, 4, 4);
backend.fill_path(&path, &clip, &Color::rgb(255, 0, 0), 0);
assert_eq!(backend.target.get_pixel(2, 2).r, 0);
}
#[test]
fn fill_path_triangle_interior_vs_exterior() {
let mut buf = vec![0u8; 16 * 16 * 4];
let tex = Texture::new(&mut buf, 16, 16, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let mut path = super::super::path::Path::new();
path.move_to(Point {
x: Fixed::ZERO,
y: Fixed::ZERO,
})
.line_to(Point {
x: Fixed::from_int(10),
y: Fixed::ZERO,
})
.line_to(Point {
x: Fixed::ZERO,
y: Fixed::from_int(10),
})
.close();
let clip = Rect::new(0, 0, 16, 16);
backend.fill_path(&path, &clip, &Color::rgb(0, 200, 0), 255);
assert_eq!(backend.target.get_pixel(2, 2).g, 200);
assert_eq!(backend.target.get_pixel(8, 8).g, 0);
}
#[test]
fn draw_label_is_reachable_via_trait() {
let mut buf = vec![0u8; 32 * 16 * 4];
let tex = Texture::new(&mut buf, 32, 16, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let pos = Point {
x: Fixed::from_int(1),
y: Fixed::from_int(1),
};
let clip = Rect::new(0, 0, 32, 16);
Canvas::draw_label(&mut backend, &pos, b"A", &clip, &Color::rgb(255, 0, 0), 255);
let mut found = false;
for y in 0..16 {
for x in 0..32 {
if backend.target.get_pixel(x, y).r > 0 {
found = true;
break;
}
}
}
assert!(found, "expected at least one red pixel from glyph");
}
#[test]
fn stroke_path_line_colors_interior_and_skips_far_pixels() {
let mut buf = vec![0u8; 16 * 16 * 4];
let tex = Texture::new(&mut buf, 16, 16, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let mut path = super::super::path::Path::new();
path.move_to(Point {
x: Fixed::from_int(2),
y: Fixed::from_int(8),
})
.line_to(Point {
x: Fixed::from_int(14),
y: Fixed::from_int(8),
});
let clip = Rect::new(0, 0, 16, 16);
backend.stroke_path(
&path,
&clip,
Fixed::from_int(2),
&Color::rgb(255, 0, 0),
255,
);
assert!(backend.target.get_pixel(8, 8).r > 0);
assert_eq!(backend.target.get_pixel(8, 0).r, 0);
assert_eq!(backend.target.get_pixel(8, 15).r, 0);
}
#[test]
fn renderer_dispatches_line_command() {
use crate::draw::renderer::Renderer;
let mut buf = vec![0u8; 16 * 16 * 4];
let tex = Texture::new(&mut buf, 16, 16, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let cmd = DrawCommand::Line {
p1: Point {
x: Fixed::from_int(2),
y: Fixed::from_int(8),
},
p2: Point {
x: Fixed::from_int(14),
y: Fixed::from_int(8),
},
transform: crate::types::Transform::IDENTITY,
color: Color::rgb(255, 0, 0),
width: Fixed::from_int(2),
opa: 255,
};
let clip = Rect::new(0, 0, 16, 16);
Renderer::draw(&mut backend, &cmd, &clip);
assert!(backend.target.get_pixel(8, 8).r > 0);
}
#[test]
fn renderer_dispatches_arc_command() {
use crate::draw::renderer::Renderer;
let mut buf = vec![0u8; 32 * 32 * 4];
let tex = Texture::new(&mut buf, 32, 32, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let cmd = DrawCommand::Arc {
center: Point {
x: Fixed::from_int(16),
y: Fixed::from_int(16),
},
transform: crate::types::Transform::IDENTITY,
radius: Fixed::from_int(10),
start_angle: Fixed::from_int(0),
end_angle: Fixed::from_int(90),
color: Color::rgb(0, 255, 0),
width: Fixed::from_int(2),
opa: 255,
};
let clip = Rect::new(0, 0, 32, 32);
Renderer::draw(&mut backend, &cmd, &clip);
let hit = backend.target.get_pixel(26, 16).g > 0 || backend.target.get_pixel(25, 16).g > 0;
assert!(hit);
}
#[test]
fn draw_line_default_impl_strokes_pixels() {
let mut buf = vec![0u8; 16 * 16 * 4];
let tex = Texture::new(&mut buf, 16, 16, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let p1 = Point {
x: Fixed::from_int(2),
y: Fixed::from_int(8),
};
let p2 = Point {
x: Fixed::from_int(14),
y: Fixed::from_int(8),
};
let clip = Rect::new(0, 0, 16, 16);
backend.draw_line(
p1,
p2,
&clip,
Fixed::from_int(2),
&Color::rgb(255, 0, 0),
255,
);
assert!(backend.target.get_pixel(8, 8).r > 0);
}
#[test]
fn draw_arc_default_impl_strokes_pixels() {
let mut buf = vec![0u8; 32 * 32 * 4];
let tex = Texture::new(&mut buf, 32, 32, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let center = Point {
x: Fixed::from_int(16),
y: Fixed::from_int(16),
};
let clip = Rect::new(0, 0, 32, 32);
backend.draw_arc(
center,
Fixed::from_int(10),
Fixed::from_int(0),
Fixed::from_int(90),
&clip,
Fixed::from_int(2),
&Color::rgb(0, 255, 0),
255,
);
assert!(backend.target.get_pixel(26, 16).g > 0 || backend.target.get_pixel(25, 16).g > 0);
}
#[test]
fn stroke_path_zero_width_is_noop() {
let mut buf = vec![0u8; 8 * 8 * 4];
let tex = Texture::new(&mut buf, 8, 8, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let mut path = super::super::path::Path::new();
path.move_to(Point {
x: Fixed::ZERO,
y: Fixed::ZERO,
})
.line_to(Point {
x: Fixed::from_int(8),
y: Fixed::ZERO,
});
let clip = Rect::new(0, 0, 8, 8);
backend.stroke_path(&path, &clip, Fixed::ZERO, &Color::rgb(255, 0, 0), 255);
for y in 0..8 {
for x in 0..8 {
assert_eq!(backend.target.get_pixel(x, y).r, 0);
}
}
}
#[test]
fn painter_fill_rect_with_backend() {
use crate::draw::painter::Painter;
let mut buf = vec![0u8; 16 * 16 * 4];
let tex = Texture::new(&mut buf, 16, 16, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
{
let mut painter = Painter::new(&mut backend);
let rect = Rect::new(1, 1, 6, 6);
let clip = Rect::new(0, 0, 16, 16);
painter.fill_rect(&rect, &clip, &Color::rgb(0, 255, 0), Fixed::ZERO, 255);
}
let c = backend.target.get_pixel(3, 3);
assert_eq!(c.r, 0);
assert_eq!(c.g, 255);
assert_eq!(c.b, 0);
}
#[test]
fn painter_forwards_path_and_stroke_methods() {
use crate::draw::painter::Painter;
use crate::draw::path::Path;
let mut buf = vec![0u8; 32 * 32 * 4];
let tex = Texture::new(&mut buf, 32, 32, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let clip = Rect::new(0, 0, 32, 32);
{
let mut painter = Painter::new(&mut backend);
let path = Path::rect(
Fixed::from_int(4),
Fixed::from_int(4),
Fixed::from_int(10),
Fixed::from_int(10),
);
painter.fill_path(&path, &clip, &Color::rgb(255, 0, 0), 255);
painter.draw_line(
Point {
x: Fixed::from_int(20),
y: Fixed::from_int(20),
},
Point {
x: Fixed::from_int(28),
y: Fixed::from_int(28),
},
&clip,
Fixed::from_int(2),
&Color::rgb(0, 255, 0),
255,
);
painter.draw_arc(
Point {
x: Fixed::from_int(24),
y: Fixed::from_int(8),
},
Fixed::from_int(4),
Fixed::from_int(0),
Fixed::from_int(90),
&clip,
Fixed::from_int(2),
&Color::rgb(0, 0, 255),
255,
);
}
assert_eq!(backend.target.get_pixel(8, 8).r, 255);
assert!(backend.target.get_pixel(24, 24).g > 0);
assert!(
backend.target.get_pixel(28, 8).b > 0
|| backend.target.get_pixel(27, 8).b > 0
|| backend.target.get_pixel(28, 9).b > 0,
);
}
#[test]
fn painter_draw_text_forwards_to_backend() {
use crate::draw::painter::Painter;
let mut buf = vec![0u8; 32 * 16 * 4];
let tex = Texture::new(&mut buf, 32, 16, ColorFormat::RGBA8888);
let mut backend = SwRenderer::new(tex);
let clip = Rect::new(0, 0, 32, 16);
{
let mut painter = Painter::new(&mut backend);
painter.draw_text(
&Point {
x: Fixed::from_int(1),
y: Fixed::from_int(1),
},
b"B",
&clip,
&Color::rgb(200, 100, 50),
255,
);
}
let mut found = false;
for y in 0..16 {
for x in 0..32 {
if backend.target.get_pixel(x, y).r > 0 {
found = true;
break;
}
}
}
assert!(found);
}
#[test]
fn blit_dda_matches_generic_slow() {
let mut src_buf = vec![0u8; 4 * 4 * 4];
for y in 0..4 {
for x in 0..4 {
let i = (y * 4 + x) * 4;
src_buf[i] = (y * 4 + x) as u8 * 16 + 1;
src_buf[i + 3] = 255;
}
}
let src = Texture::new(&mut src_buf, 4, 4, ColorFormat::RGBA8888);
let mut dst_a = vec![0u8; 8 * 6 * 4];
let mut dst_b = vec![0u8; 8 * 6 * 4];
{
let mut tex_a = Texture::new(&mut dst_a, 8, 6, ColorFormat::RGBA8888);
blit_generic_slow(&mut tex_a, &src, 0, 0, 4, 4, 0, 0, 7, 5, 0, 0, 8, 6);
}
{
let mut tex_b = Texture::new(&mut dst_b, 8, 6, ColorFormat::RGBA8888);
blit_dda(&mut tex_b, &src, 0, 0, 4, 4, 0, 0, 7, 5, 0, 0, 8, 6);
}
assert_eq!(dst_a, dst_b, "dda sampling diverged from divide path");
}
#[test]
fn blit_1to1_matches_generic_for_argb_to_argb() {
let mut src_buf = vec![0u8; 4 * 4 * 4];
for i in 0..16 {
src_buf[i * 4] = 30 + i as u8;
src_buf[i * 4 + 1] = 60 + i as u8;
src_buf[i * 4 + 2] = 90 + i as u8;
src_buf[i * 4 + 3] = match i {
0 => 0,
3 | 7 => 128,
_ => 255,
};
}
let src = Texture::new(&mut src_buf, 4, 4, ColorFormat::RGBA8888);
let mut dst_a = vec![0u8; 6 * 6 * 4];
for (i, byte) in dst_a.iter_mut().enumerate() {
*byte = (i * 3) as u8;
}
let mut dst_b = dst_a.clone();
{
let mut tex = Texture::new(&mut dst_a, 6, 6, ColorFormat::RGBA8888);
blit_generic_slow(&mut tex, &src, 0, 0, 4, 4, 1, 1, 4, 4, 0, 0, 6, 6);
}
{
let mut tex = Texture::new(&mut dst_b, 6, 6, ColorFormat::RGBA8888);
blit_1to1_fast(&mut tex, &src, 0, 0, 4, 4, 1, 1, 0, 0, 6, 6);
}
for (i, (&a, &b)) in dst_a.iter().zip(dst_b.iter()).enumerate() {
assert!(
(a as i32 - b as i32).abs() <= 1,
"byte {} diverged by more than 1: slow={} fast={}",
i,
a,
b
);
}
}
#[test]
fn blit_1to1_matches_generic_for_argb_to_565sw() {
let mut src_buf = vec![0u8; 4 * 4 * 4];
for i in 0..16 {
src_buf[i * 4] = 30 + i as u8 * 5;
src_buf[i * 4 + 1] = 40 + i as u8 * 3;
src_buf[i * 4 + 2] = 50 + i as u8 * 7;
src_buf[i * 4 + 3] = if i == 0 { 0 } else { 255 };
}
let src = Texture::new(&mut src_buf, 4, 4, ColorFormat::RGBA8888);
let mut dst_a = vec![0u8; 6 * 6 * 2];
for i in 0..dst_a.len() {
dst_a[i] = (i * 5) as u8;
}
let mut dst_b = dst_a.clone();
{
let mut tex = Texture::new(&mut dst_a, 6, 6, ColorFormat::RGB565Swapped);
blit_generic_slow(&mut tex, &src, 0, 0, 4, 4, 1, 1, 4, 4, 0, 0, 6, 6);
}
{
let mut tex = Texture::new(&mut dst_b, 6, 6, ColorFormat::RGB565Swapped);
blit_1to1_fast(&mut tex, &src, 0, 0, 4, 4, 1, 1, 0, 0, 6, 6);
}
assert_eq!(dst_a, dst_b);
}
#[test]
fn blit_2to2_565sw_matches_dda() {
let mut src_buf = vec![0u8; 3 * 3 * 2];
for i in 0..9 {
src_buf[i * 2] = 0x12 + i as u8;
src_buf[i * 2 + 1] = 0x34 + i as u8;
}
let src = Texture::new(&mut src_buf, 3, 3, ColorFormat::RGB565Swapped);
let mut dst_a = vec![0u8; 10 * 10 * 2];
let mut dst_b = vec![0u8; 10 * 10 * 2];
{
let mut tex = Texture::new(&mut dst_a, 10, 10, ColorFormat::RGB565Swapped);
blit_dda(&mut tex, &src, 0, 0, 3, 3, 1, 1, 6, 6, 0, 0, 10, 10);
}
{
let mut tex = Texture::new(&mut dst_b, 10, 10, ColorFormat::RGB565Swapped);
blit_2to2_fast(&mut tex, &src, 0, 0, 3, 3, 1, 1, 0, 0, 10, 10);
}
assert_eq!(dst_a, dst_b);
}
#[test]
fn blit_2to2_odd_clip_falls_back_cleanly() {
let mut src_buf = vec![0u8; 2 * 2 * 2];
src_buf[0] = 0xAA;
src_buf[1] = 0xBB;
let src = Texture::new(&mut src_buf, 2, 2, ColorFormat::RGB565Swapped);
let mut dst = vec![0u8; 6 * 6 * 2];
let mut tex = Texture::new(&mut dst, 6, 6, ColorFormat::RGB565Swapped);
blit_2to2_fast(&mut tex, &src, 0, 0, 2, 2, 0, 0, 1, 0, 6, 6);
assert_eq!(dst[0], 0);
assert_eq!(dst[1], 0);
assert_ne!(dst[2], 0);
}
#[test]
fn fill_rect_transformed_90deg_rotation() {
let mut buf = vec![0u8; 16 * 16 * 4];
let mut dst = Texture::new(&mut buf, 16, 16, ColorFormat::RGBA8888);
let rect = Rect::new(6, 6, 4, 4);
let cx = rect.x + rect.w / Fixed::from_int(2);
let cy = rect.y + rect.h / Fixed::from_int(2);
let tf = Transform::translate(cx, cy)
.compose(&Transform::rotate_deg(Fixed::from_int(90)))
.compose(&Transform::translate(Fixed::ZERO - cx, Fixed::ZERO - cy));
let red = Color::rgb(255, 0, 0);
fill_rect_transformed(&mut dst, rect, Rect::new(0, 0, 16, 16), &tf, &red, 255);
let mut painted = 0;
for y in 0..16 {
for x in 0..16 {
if dst.get_pixel(x, y).r == 255 {
painted += 1;
}
}
}
assert!(
(12..=20).contains(&painted),
"expected ~16 painted pixels, got {}",
painted
);
}
#[test]
fn blit_1to1_with_clip_restricted() {
let mut src_buf = vec![0u8; 4 * 4 * 4];
for i in 0..16 {
src_buf[i * 4] = 100 + i as u8;
src_buf[i * 4 + 3] = 255;
}
let src = Texture::new(&mut src_buf, 4, 4, ColorFormat::RGBA8888);
let mut dst_a = vec![0u8; 8 * 8 * 4];
let mut dst_b = vec![0u8; 8 * 8 * 4];
{
let mut tex = Texture::new(&mut dst_a, 8, 8, ColorFormat::RGBA8888);
blit_generic_slow(&mut tex, &src, 0, 0, 4, 4, 1, 1, 4, 4, 3, 0, 8, 8);
}
{
let mut tex = Texture::new(&mut dst_b, 8, 8, ColorFormat::RGBA8888);
blit_1to1_fast(&mut tex, &src, 0, 0, 4, 4, 1, 1, 3, 0, 8, 8);
}
assert_eq!(dst_a, dst_b);
}
#[test]
fn blit_dda_with_partial_clip() {
let mut src_buf = vec![0u8; 4 * 4 * 4];
for i in 0..16 {
src_buf[i * 4] = (i * 16 + 1) as u8;
src_buf[i * 4 + 3] = 255;
}
let src = Texture::new(&mut src_buf, 4, 4, ColorFormat::RGBA8888);
let mut dst_a = vec![0u8; 10 * 10 * 4];
let mut dst_b = vec![0u8; 10 * 10 * 4];
{
let mut tex_a = Texture::new(&mut dst_a, 10, 10, ColorFormat::RGBA8888);
blit_generic_slow(&mut tex_a, &src, 0, 0, 4, 4, 1, 1, 8, 8, 2, 0, 7, 10);
}
{
let mut tex_b = Texture::new(&mut dst_b, 10, 10, ColorFormat::RGBA8888);
blit_dda(&mut tex_b, &src, 0, 0, 4, 4, 1, 1, 8, 8, 2, 0, 7, 10);
}
assert_eq!(dst_a, dst_b);
}
}