177 lines
7.2 KiB
Rust
177 lines
7.2 KiB
Rust
use std::usize;
|
|
|
|
use crate::{
|
|
Background, Gsa, Rgb, BACKGROUND_MAX_SIZE, EMPTY_TILE, HALF_TILE_SIZE, MAX_BACKGROUNDS,
|
|
SCREEN_HEIGHT, SCREEN_WIDTH, TILESET_SIZE, TILE_SIZE, TRANSPARENT,
|
|
};
|
|
use glam::IVec2;
|
|
use softbuffer::Buffer;
|
|
|
|
fn draw_tile(
|
|
target: &mut [u8],
|
|
pos: IVec2,
|
|
tile: u16,
|
|
tileset: &[u8],
|
|
half: bool,
|
|
screen_size: IVec2,
|
|
) {
|
|
let tilesize = if half { HALF_TILE_SIZE } else { TILE_SIZE };
|
|
let tx = tile as usize % 0x100 * tilesize;
|
|
let ty = tile as usize / 0x100 * tilesize;
|
|
let startx = pos.x.max(0).min(screen_size.x);
|
|
let starty = pos.y.max(0).min(screen_size.y);
|
|
let endx = (pos.x + tilesize as i32).max(0).min(screen_size.x);
|
|
let endy = (pos.y + tilesize as i32).max(0).min(screen_size.y);
|
|
for y in (starty - pos.y)..(endy - pos.y) {
|
|
for x in (startx - pos.x)..(endx - pos.x) {
|
|
let p = tileset[(x as usize + tx) + (y as usize + ty) * (TILESET_SIZE * TILE_SIZE)];
|
|
if p != TRANSPARENT {
|
|
target[(x + pos.x) as usize + (y + pos.y) as usize * screen_size.x as usize] = p;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn render_map(target: &mut [u8], map: &Background, tileset: &[u8], screen_size: IVec2) {
|
|
let tcmult = if map.half_tile { 2 } else { 1 };
|
|
let tilesize = if map.half_tile {
|
|
HALF_TILE_SIZE
|
|
} else {
|
|
TILE_SIZE
|
|
} as i32;
|
|
let mut startx = map.scroll.x / tilesize;
|
|
let mut starty = map.scroll.y / tilesize;
|
|
let endx = map
|
|
.size
|
|
.x
|
|
.min(startx + (screen_size.x / TILE_SIZE as i32 + 1) * tcmult);
|
|
let endy = map
|
|
.size
|
|
.y
|
|
.min(starty + (screen_size.y / TILE_SIZE as i32 + 1) * tcmult);
|
|
startx = 0.max(startx);
|
|
starty = 0.max(starty);
|
|
for x in startx..endx {
|
|
for y in starty..endy {
|
|
let tile = map.tiles[x as usize][y as usize];
|
|
if tile != EMPTY_TILE {
|
|
draw_tile(
|
|
target,
|
|
IVec2 {
|
|
x: x * tilesize - map.scroll.x,
|
|
y: y * tilesize - map.scroll.y,
|
|
},
|
|
tile,
|
|
tileset,
|
|
map.half_tile,
|
|
screen_size,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub(crate) fn render_to_screen(target: &mut [u8], gsa: &Gsa, tileset: &[u8], screen_size: IVec2) {
|
|
for i in 0..MAX_BACKGROUNDS {
|
|
if gsa.bgs[i].active {
|
|
render_map(target, &gsa.bgs[i], tileset, screen_size);
|
|
}
|
|
for sprite in &gsa.sprites {
|
|
if sprite.tile != EMPTY_TILE && sprite.priority == i as u8 {
|
|
draw_tile(target, sprite.pos, sprite.tile, tileset, false, screen_size);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub(crate) fn render_to_window(
|
|
target: &mut Buffer,
|
|
src: &mut [u8],
|
|
palette: &[Rgb; 256],
|
|
window_size: IVec2,
|
|
screen_size: IVec2,
|
|
scale: usize,
|
|
off_x: usize,
|
|
off_y: usize,
|
|
) {
|
|
let start = window_size.x as usize * off_y;
|
|
let end = start + window_size.x as usize * screen_size.y as usize * scale;
|
|
for (y, row) in target[start..end]
|
|
.chunks_exact_mut(window_size.x as usize * scale)
|
|
.enumerate()
|
|
{
|
|
let y = y as i32;
|
|
for x in 0..screen_size.x as usize {
|
|
let p = src[x + y as usize * screen_size.x as usize];
|
|
|
|
for scaley in 0..scale {
|
|
for scalex in 0..scale {
|
|
let sx = x * scale + scalex;
|
|
row[sx + off_x + scaley * window_size.x as usize] =
|
|
palette[p as usize].to_u32();
|
|
}
|
|
}
|
|
|
|
//unrolled is faster benchmarked... <_<
|
|
/*
|
|
let ww = window_size.x as usize;
|
|
match scale {
|
|
1 => {
|
|
row[x + off_x] = palette[p as usize].to_u32();
|
|
}
|
|
2 => {
|
|
row[x * 2 + off_x] = palette[p as usize].to_u32();
|
|
row[x * 2 + off_x + 1] = palette[p as usize].to_u32();
|
|
row[x * 2 + off_x + ww] = palette[p as usize].to_u32();
|
|
row[x * 2 + off_x + ww + 1] = palette[p as usize].to_u32();
|
|
}
|
|
6 => {
|
|
row[x * 6 + off_x + ww * 0] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 0 + 1] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 0 + 2] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 0 + 3] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 0 + 4] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 0 + 5] = palette[p as usize].to_u32();
|
|
|
|
row[x * 6 + off_x + ww * 1] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 1 + 1] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 1 + 2] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 1 + 3] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 1 + 4] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 1 + 5] = palette[p as usize].to_u32();
|
|
|
|
row[x * 6 + off_x + ww * 2] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 2 + 1] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 2 + 2] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 2 + 3] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 2 + 4] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 2 + 5] = palette[p as usize].to_u32();
|
|
|
|
row[x * 6 + off_x + ww * 3] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 3 + 1] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 3 + 2] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 3 + 3] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 3 + 4] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 3 + 5] = palette[p as usize].to_u32();
|
|
|
|
row[x * 6 + off_x + ww * 4] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 4 + 1] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 4 + 2] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 4 + 3] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 4 + 4] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 4 + 5] = palette[p as usize].to_u32();
|
|
|
|
row[x * 6 + off_x + ww * 5] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 5 + 1] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 5 + 2] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 5 + 3] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 5 + 4] = palette[p as usize].to_u32();
|
|
row[x * 6 + off_x + ww * 5 + 5] = palette[p as usize].to_u32();
|
|
}
|
|
_ => {}
|
|
}
|
|
*/
|
|
}
|
|
}
|
|
}
|