2023-10-03 04:14:45 +00:00
|
|
|
use std::{iter::Peekable, mem};
|
2023-09-26 17:29:44 +00:00
|
|
|
|
2023-09-19 19:19:22 +00:00
|
|
|
use super::{Bounds, Hsla, Pixels, Point};
|
2023-10-03 15:36:12 +00:00
|
|
|
use crate::{AtlasTextureId, AtlasTile, Corners, Edges};
|
2023-09-19 19:19:22 +00:00
|
|
|
use bytemuck::{Pod, Zeroable};
|
2023-10-03 15:36:12 +00:00
|
|
|
use collections::BTreeMap;
|
|
|
|
use smallvec::SmallVec;
|
2023-09-19 19:19:22 +00:00
|
|
|
|
2023-09-23 17:06:10 +00:00
|
|
|
// Exported to metal
|
|
|
|
pub type PointF = Point<f32>;
|
2023-10-03 04:14:45 +00:00
|
|
|
pub type StackingOrder = SmallVec<[u32; 16]>;
|
2023-09-23 17:06:10 +00:00
|
|
|
|
2023-09-26 17:34:41 +00:00
|
|
|
#[derive(Debug)]
|
2023-09-19 19:19:22 +00:00
|
|
|
pub struct Scene {
|
2023-10-03 14:53:49 +00:00
|
|
|
pub(crate) scale_factor: f32,
|
2023-10-03 04:14:45 +00:00
|
|
|
pub(crate) layers: BTreeMap<StackingOrder, SceneLayer>,
|
2023-09-24 22:52:33 +00:00
|
|
|
}
|
|
|
|
|
2023-09-19 19:19:22 +00:00
|
|
|
impl Scene {
|
2023-09-23 20:20:07 +00:00
|
|
|
pub fn new(scale_factor: f32) -> Scene {
|
2023-09-19 19:19:22 +00:00
|
|
|
Scene {
|
2023-09-23 20:20:07 +00:00
|
|
|
scale_factor,
|
2023-10-03 04:14:45 +00:00
|
|
|
layers: BTreeMap::new(),
|
2023-09-19 19:19:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-26 17:29:44 +00:00
|
|
|
pub fn take(&mut self) -> Scene {
|
|
|
|
Scene {
|
|
|
|
scale_factor: self.scale_factor,
|
2023-10-03 04:14:45 +00:00
|
|
|
layers: mem::take(&mut self.layers),
|
2023-09-26 17:29:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-03 14:53:49 +00:00
|
|
|
pub fn insert(&mut self, stacking_order: StackingOrder, primitive: impl Into<Primitive>) {
|
|
|
|
let layer = self.layers.entry(stacking_order).or_default();
|
2023-10-03 04:14:45 +00:00
|
|
|
|
|
|
|
let primitive = primitive.into();
|
2023-09-23 18:21:52 +00:00
|
|
|
match primitive {
|
2023-10-03 04:14:45 +00:00
|
|
|
Primitive::Quad(mut quad) => {
|
|
|
|
quad.scale(self.scale_factor);
|
|
|
|
layer.quads.push(quad);
|
|
|
|
}
|
2023-10-03 15:36:12 +00:00
|
|
|
Primitive::Sprite(sprite) => {
|
2023-10-03 04:14:45 +00:00
|
|
|
layer.sprites.push(sprite);
|
|
|
|
}
|
2023-09-19 19:19:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-03 04:14:45 +00:00
|
|
|
pub(crate) fn layers(&mut self) -> impl Iterator<Item = &mut SceneLayer> {
|
|
|
|
self.layers.values_mut()
|
2023-09-23 17:06:10 +00:00
|
|
|
}
|
2023-09-19 19:19:22 +00:00
|
|
|
}
|
|
|
|
|
2023-10-03 04:14:45 +00:00
|
|
|
#[derive(Debug, Default)]
|
|
|
|
pub(crate) struct SceneLayer {
|
|
|
|
pub quads: Vec<Quad>,
|
|
|
|
pub sprites: Vec<MonochromeSprite>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SceneLayer {
|
|
|
|
pub fn batches(&mut self) -> impl Iterator<Item = PrimitiveBatch> {
|
2023-10-03 15:36:12 +00:00
|
|
|
self.quads.sort_unstable();
|
|
|
|
self.sprites.sort_unstable();
|
2023-10-03 04:14:45 +00:00
|
|
|
|
|
|
|
BatchIterator::new(
|
|
|
|
&self.quads,
|
|
|
|
self.quads.iter().peekable(),
|
|
|
|
&self.sprites,
|
|
|
|
self.sprites.iter().peekable(),
|
|
|
|
)
|
|
|
|
}
|
2023-09-19 19:19:22 +00:00
|
|
|
}
|
|
|
|
|
2023-10-03 04:14:45 +00:00
|
|
|
struct BatchIterator<'a, Q, S>
|
|
|
|
where
|
|
|
|
Q: Iterator<Item = &'a Quad>,
|
|
|
|
S: Iterator<Item = &'a MonochromeSprite>,
|
|
|
|
{
|
|
|
|
quads: &'a [Quad],
|
|
|
|
sprites: &'a [MonochromeSprite],
|
|
|
|
quads_start: usize,
|
|
|
|
sprites_start: usize,
|
|
|
|
quads_iter: Peekable<Q>,
|
|
|
|
sprites_iter: Peekable<S>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, Q: 'a, S: 'a> Iterator for BatchIterator<'a, Q, S>
|
|
|
|
where
|
|
|
|
Q: Iterator<Item = &'a Quad>,
|
|
|
|
S: Iterator<Item = &'a MonochromeSprite>,
|
|
|
|
{
|
|
|
|
type Item = PrimitiveBatch<'a>;
|
|
|
|
|
|
|
|
fn next(&mut self) -> Option<Self::Item> {
|
2023-10-03 15:36:12 +00:00
|
|
|
let mut kinds_and_orders = [
|
|
|
|
(PrimitiveKind::Quad, self.quads_iter.peek().map(|q| q.order)),
|
|
|
|
(
|
|
|
|
PrimitiveKind::Sprite,
|
|
|
|
self.sprites_iter.peek().map(|s| s.order),
|
|
|
|
),
|
|
|
|
];
|
|
|
|
kinds_and_orders.sort_by_key(|(_, order)| order.unwrap_or(u32::MAX));
|
|
|
|
|
|
|
|
let first = kinds_and_orders[0];
|
|
|
|
let second = kinds_and_orders[1];
|
|
|
|
let (batch_kind, max_order) = if first.1.is_some() {
|
|
|
|
(first.0, second.1.unwrap_or(u32::MAX))
|
2023-10-03 04:14:45 +00:00
|
|
|
} else {
|
2023-10-03 15:36:12 +00:00
|
|
|
return None;
|
|
|
|
};
|
|
|
|
|
|
|
|
match batch_kind {
|
|
|
|
PrimitiveKind::Quad => {
|
|
|
|
let quads_start = self.quads_start;
|
|
|
|
let quads_end = quads_start
|
|
|
|
+ self
|
|
|
|
.quads_iter
|
|
|
|
.by_ref()
|
|
|
|
.take_while(|quad| quad.order <= max_order)
|
|
|
|
.count();
|
|
|
|
self.quads_start = quads_end;
|
|
|
|
Some(PrimitiveBatch::Quads(&self.quads[quads_start..quads_end]))
|
|
|
|
}
|
|
|
|
PrimitiveKind::Sprite => {
|
|
|
|
let texture_id = self.sprites_iter.peek().unwrap().tile.texture_id;
|
|
|
|
let sprites_start = self.sprites_start;
|
|
|
|
let sprites_end = sprites_start
|
|
|
|
+ self
|
|
|
|
.sprites_iter
|
|
|
|
.by_ref()
|
|
|
|
.take_while(|sprite| {
|
|
|
|
sprite.order <= max_order && sprite.tile.texture_id == texture_id
|
|
|
|
})
|
|
|
|
.count();
|
|
|
|
self.sprites_start = sprites_end;
|
|
|
|
Some(PrimitiveBatch::Sprites {
|
|
|
|
texture_id,
|
|
|
|
sprites: &self.sprites[sprites_start..sprites_end],
|
|
|
|
})
|
|
|
|
}
|
2023-09-23 20:20:07 +00:00
|
|
|
}
|
|
|
|
}
|
2023-10-03 04:14:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, Q: 'a, S: 'a> BatchIterator<'a, Q, S>
|
|
|
|
where
|
|
|
|
Q: Iterator<Item = &'a Quad>,
|
|
|
|
S: Iterator<Item = &'a MonochromeSprite>,
|
|
|
|
{
|
|
|
|
fn new(
|
|
|
|
quads: &'a [Quad],
|
|
|
|
quads_iter: Peekable<Q>,
|
|
|
|
sprites: &'a [MonochromeSprite],
|
|
|
|
sprites_iter: Peekable<S>,
|
|
|
|
) -> Self {
|
2023-10-03 15:36:12 +00:00
|
|
|
Self {
|
2023-10-03 04:14:45 +00:00
|
|
|
quads,
|
|
|
|
quads_start: 0,
|
|
|
|
quads_iter,
|
|
|
|
sprites,
|
|
|
|
sprites_start: 0,
|
|
|
|
sprites_iter,
|
|
|
|
}
|
2023-09-19 19:19:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-03 04:14:45 +00:00
|
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
|
|
pub enum PrimitiveKind {
|
|
|
|
Quad,
|
|
|
|
Sprite,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub enum Primitive {
|
|
|
|
Quad(Quad),
|
|
|
|
Sprite(MonochromeSprite),
|
|
|
|
}
|
|
|
|
|
2023-10-03 15:36:12 +00:00
|
|
|
pub(crate) enum PrimitiveBatch<'a> {
|
2023-10-03 04:14:45 +00:00
|
|
|
Quads(&'a [Quad]),
|
2023-10-03 15:36:12 +00:00
|
|
|
Sprites {
|
|
|
|
texture_id: AtlasTextureId,
|
|
|
|
sprites: &'a [MonochromeSprite],
|
|
|
|
},
|
2023-10-03 04:14:45 +00:00
|
|
|
}
|
|
|
|
|
2023-10-03 15:36:12 +00:00
|
|
|
#[derive(Debug, Copy, Clone, Zeroable, Pod, Eq, PartialEq)]
|
2023-09-19 19:19:22 +00:00
|
|
|
#[repr(C)]
|
|
|
|
pub struct Quad {
|
2023-09-23 17:06:10 +00:00
|
|
|
pub order: u32,
|
2023-09-19 19:19:22 +00:00
|
|
|
pub bounds: Bounds<Pixels>,
|
2023-09-22 22:31:26 +00:00
|
|
|
pub clip_bounds: Bounds<Pixels>,
|
|
|
|
pub clip_corner_radii: Corners<Pixels>,
|
2023-09-19 19:19:22 +00:00
|
|
|
pub background: Hsla,
|
|
|
|
pub border_color: Hsla,
|
2023-09-22 22:31:26 +00:00
|
|
|
pub corner_radii: Corners<Pixels>,
|
|
|
|
pub border_widths: Edges<Pixels>,
|
2023-09-19 19:19:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Quad {
|
|
|
|
pub fn vertices(&self) -> impl Iterator<Item = Point<Pixels>> {
|
|
|
|
let x1 = self.bounds.origin.x;
|
|
|
|
let y1 = self.bounds.origin.y;
|
|
|
|
let x2 = x1 + self.bounds.size.width;
|
|
|
|
let y2 = y1 + self.bounds.size.height;
|
|
|
|
[
|
|
|
|
Point::new(x1, y1),
|
|
|
|
Point::new(x2, y1),
|
|
|
|
Point::new(x2, y2),
|
|
|
|
Point::new(x1, y2),
|
|
|
|
]
|
|
|
|
.into_iter()
|
|
|
|
}
|
|
|
|
|
2023-09-23 20:20:07 +00:00
|
|
|
pub fn scale(&mut self, factor: f32) {
|
2023-09-24 23:12:59 +00:00
|
|
|
self.bounds *= factor;
|
|
|
|
self.clip_bounds *= factor;
|
2023-09-23 20:20:07 +00:00
|
|
|
self.clip_corner_radii *= factor;
|
|
|
|
self.corner_radii *= factor;
|
|
|
|
self.border_widths *= factor;
|
2023-09-19 19:19:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-03 15:36:12 +00:00
|
|
|
impl Ord for Quad {
|
|
|
|
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
|
|
|
|
self.order.cmp(&other.order)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialOrd for Quad {
|
|
|
|
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
|
|
|
|
Some(self.cmp(other))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-23 20:20:07 +00:00
|
|
|
impl From<Quad> for Primitive {
|
|
|
|
fn from(quad: Quad) -> Self {
|
|
|
|
Primitive::Quad(quad)
|
2023-09-19 19:19:22 +00:00
|
|
|
}
|
|
|
|
}
|
2023-10-02 21:43:03 +00:00
|
|
|
|
2023-10-03 15:36:12 +00:00
|
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
2023-10-02 21:43:03 +00:00
|
|
|
#[repr(C)]
|
2023-10-03 04:14:45 +00:00
|
|
|
pub struct MonochromeSprite {
|
2023-10-02 21:43:03 +00:00
|
|
|
pub order: u32,
|
2023-10-03 04:14:45 +00:00
|
|
|
pub bounds: Bounds<Pixels>,
|
2023-10-03 12:48:08 +00:00
|
|
|
pub color: Hsla,
|
|
|
|
pub tile: AtlasTile,
|
2023-10-02 21:43:03 +00:00
|
|
|
}
|
|
|
|
|
2023-10-03 15:36:12 +00:00
|
|
|
impl Ord for MonochromeSprite {
|
|
|
|
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
|
|
|
|
match self.order.cmp(&other.order) {
|
|
|
|
std::cmp::Ordering::Equal => self.tile.tile_id.cmp(&other.tile.tile_id),
|
|
|
|
order => order,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialOrd for MonochromeSprite {
|
|
|
|
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
|
|
|
|
Some(self.cmp(other))
|
2023-10-02 21:43:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-03 04:14:45 +00:00
|
|
|
impl From<MonochromeSprite> for Primitive {
|
|
|
|
fn from(sprite: MonochromeSprite) -> Self {
|
|
|
|
Primitive::Sprite(sprite)
|
2023-10-02 21:43:03 +00:00
|
|
|
}
|
|
|
|
}
|
2023-10-03 04:14:45 +00:00
|
|
|
|
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
|
|
pub struct AtlasId(pub(crate) usize);
|