mirror of
https://github.com/italicsjenga/vello.git
synced 2025-01-23 18:06:34 +11:00
Let's add images
This commit is contained in:
parent
2f268d4e0f
commit
165b3a083b
22 changed files with 508 additions and 38 deletions
|
@ -46,6 +46,7 @@ bytemuck = { version = "1.12.1", features = ["derive"] }
|
||||||
smallvec = "1.8.0"
|
smallvec = "1.8.0"
|
||||||
moscato = { git = "https://github.com/dfrg/pinot", rev = "59db153" }
|
moscato = { git = "https://github.com/dfrg/pinot", rev = "59db153" }
|
||||||
peniko = { git = "https://github.com/linebender/peniko", rev = "cafdac9a211a0fb2fec5656bd663d1ac770bcc81" }
|
peniko = { git = "https://github.com/linebender/peniko", rev = "cafdac9a211a0fb2fec5656bd663d1ac770bcc81" }
|
||||||
|
guillotiere = "0.6.2"
|
||||||
|
|
||||||
[workspace.dependencies]
|
[workspace.dependencies]
|
||||||
wgpu = "0.15"
|
wgpu = "0.15"
|
||||||
|
|
BIN
examples/assets/piet-logo.png
Normal file
BIN
examples/assets/piet-logo.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 12 KiB |
|
@ -6,7 +6,7 @@ use std::{
|
||||||
|
|
||||||
use anyhow::{anyhow, bail, Context, Result};
|
use anyhow::{anyhow, bail, Context, Result};
|
||||||
use clap::{CommandFactory, Parser};
|
use clap::{CommandFactory, Parser};
|
||||||
use scenes::{SceneParams, SceneSet, SimpleText};
|
use scenes::{ImageCache, SceneParams, SceneSet, SimpleText};
|
||||||
use vello::{
|
use vello::{
|
||||||
block_on_wgpu,
|
block_on_wgpu,
|
||||||
kurbo::{Affine, Vec2},
|
kurbo::{Affine, Vec2},
|
||||||
|
@ -97,9 +97,11 @@ async fn render(mut scenes: SceneSet, index: usize, args: &Args) -> Result<()> {
|
||||||
let mut builder = SceneBuilder::for_fragment(&mut fragment);
|
let mut builder = SceneBuilder::for_fragment(&mut fragment);
|
||||||
let example_scene = &mut scenes.scenes[index];
|
let example_scene = &mut scenes.scenes[index];
|
||||||
let mut text = SimpleText::new();
|
let mut text = SimpleText::new();
|
||||||
|
let mut images = ImageCache::new();
|
||||||
let mut scene_params = SceneParams {
|
let mut scene_params = SceneParams {
|
||||||
time: args.time.unwrap_or(0.),
|
time: args.time.unwrap_or(0.),
|
||||||
text: &mut text,
|
text: &mut text,
|
||||||
|
images: &mut images,
|
||||||
resolution: None,
|
resolution: None,
|
||||||
base_color: None,
|
base_color: None,
|
||||||
};
|
};
|
||||||
|
|
|
@ -15,6 +15,7 @@ vello = { path = "../../" }
|
||||||
vello_svg = { path = "../../integrations/vello_svg" }
|
vello_svg = { path = "../../integrations/vello_svg" }
|
||||||
anyhow = { workspace = true }
|
anyhow = { workspace = true }
|
||||||
clap = { workspace = true, features = ["derive"] }
|
clap = { workspace = true, features = ["derive"] }
|
||||||
|
image = "0.24.5"
|
||||||
|
|
||||||
# Used for the `download` command
|
# Used for the `download` command
|
||||||
byte-unit = "4.0"
|
byte-unit = "4.0"
|
||||||
|
|
53
examples/scenes/src/images.rs
Normal file
53
examples/scenes/src/images.rs
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use std::path::{Path, PathBuf};
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
use vello::peniko::{Blob, Format, Image};
|
||||||
|
|
||||||
|
/// Simple hack to support loading images for examples.
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct ImageCache {
|
||||||
|
files: HashMap<PathBuf, Image>,
|
||||||
|
bytes: HashMap<usize, Image>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ImageCache {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self::default()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_file(&mut self, path: impl AsRef<Path>) -> Option<Image> {
|
||||||
|
let path = path.as_ref();
|
||||||
|
if let Some(image) = self.files.get(path) {
|
||||||
|
Some(image.clone())
|
||||||
|
} else {
|
||||||
|
let data = std::fs::read(path).ok()?;
|
||||||
|
let image = decode_image(&data)?;
|
||||||
|
self.files.insert(path.to_owned(), image.clone());
|
||||||
|
Some(image)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_bytes(&mut self, key: usize, bytes: &[u8]) -> Option<Image> {
|
||||||
|
if let Some(image) = self.bytes.get(&key) {
|
||||||
|
Some(image.clone())
|
||||||
|
} else {
|
||||||
|
let image = decode_image(bytes)?;
|
||||||
|
self.bytes.insert(key, image.clone());
|
||||||
|
Some(image)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn decode_image(data: &[u8]) -> Option<Image> {
|
||||||
|
let image = image::io::Reader::new(std::io::Cursor::new(data))
|
||||||
|
.with_guessed_format()
|
||||||
|
.ok()?
|
||||||
|
.decode()
|
||||||
|
.ok()?;
|
||||||
|
let width = image.width();
|
||||||
|
let height = image.height();
|
||||||
|
let data = Arc::new(image.into_rgba8().into_vec());
|
||||||
|
let blob = Blob::new(data);
|
||||||
|
Some(Image::new(blob, Format::Rgba8, width, height))
|
||||||
|
}
|
|
@ -1,4 +1,5 @@
|
||||||
pub mod download;
|
pub mod download;
|
||||||
|
mod images;
|
||||||
mod simple_text;
|
mod simple_text;
|
||||||
mod svg;
|
mod svg;
|
||||||
mod test_scenes;
|
mod test_scenes;
|
||||||
|
@ -7,6 +8,7 @@ use std::path::PathBuf;
|
||||||
use anyhow::{anyhow, Result};
|
use anyhow::{anyhow, Result};
|
||||||
use clap::{Args, Subcommand};
|
use clap::{Args, Subcommand};
|
||||||
use download::Download;
|
use download::Download;
|
||||||
|
pub use images::ImageCache;
|
||||||
pub use simple_text::SimpleText;
|
pub use simple_text::SimpleText;
|
||||||
pub use svg::{default_scene, scene_from_files};
|
pub use svg::{default_scene, scene_from_files};
|
||||||
pub use test_scenes::test_scenes;
|
pub use test_scenes::test_scenes;
|
||||||
|
@ -16,6 +18,7 @@ use vello::{kurbo::Vec2, peniko::Color, SceneBuilder};
|
||||||
pub struct SceneParams<'a> {
|
pub struct SceneParams<'a> {
|
||||||
pub time: f64,
|
pub time: f64,
|
||||||
pub text: &'a mut SimpleText,
|
pub text: &'a mut SimpleText,
|
||||||
|
pub images: &'a mut ImageCache,
|
||||||
pub resolution: Option<Vec2>,
|
pub resolution: Option<Vec2>,
|
||||||
pub base_color: Option<vello::peniko::Color>,
|
pub base_color: Option<vello::peniko::Color>,
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,12 @@
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
use crate::{ExampleScene, SceneConfig, SceneParams, SceneSet};
|
use crate::{ExampleScene, SceneConfig, SceneParams, SceneSet};
|
||||||
use vello::kurbo::{Affine, BezPath, Ellipse, PathEl, Point, Rect};
|
use vello::kurbo::{Affine, BezPath, Ellipse, PathEl, Point, Rect, Vec2};
|
||||||
use vello::peniko::*;
|
use vello::peniko::*;
|
||||||
use vello::*;
|
use vello::*;
|
||||||
|
|
||||||
|
const PIET_LOGO_IMAGE: &[u8] = include_bytes!("../../assets/piet-logo.png");
|
||||||
|
|
||||||
macro_rules! scene {
|
macro_rules! scene {
|
||||||
($name: ident) => {
|
($name: ident) => {
|
||||||
scene!($name: false)
|
scene!($name: false)
|
||||||
|
@ -97,6 +101,11 @@ fn cardioid_and_friends(sb: &mut SceneBuilder, _: &mut SceneParams) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn animated_text(sb: &mut SceneBuilder, params: &mut SceneParams) {
|
fn animated_text(sb: &mut SceneBuilder, params: &mut SceneParams) {
|
||||||
|
let piet_logo = params
|
||||||
|
.images
|
||||||
|
.from_bytes(PIET_LOGO_IMAGE.as_ptr() as usize, PIET_LOGO_IMAGE)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
use PathEl::*;
|
use PathEl::*;
|
||||||
let rect = Rect::from_origin_size(Point::new(0.0, 0.0), (1000.0, 1000.0));
|
let rect = Rect::from_origin_size(Point::new(0.0, 0.0), (1000.0, 1000.0));
|
||||||
let star = [
|
let star = [
|
||||||
|
@ -184,6 +193,10 @@ fn animated_text(sb: &mut SceneBuilder, params: &mut SceneParams) {
|
||||||
None,
|
None,
|
||||||
&star,
|
&star,
|
||||||
);
|
);
|
||||||
|
sb.draw_image(
|
||||||
|
&piet_logo,
|
||||||
|
Affine::translate((550.0, 250.0)) * Affine::skew(-20f64.to_radians().tan(), 0.0),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn brush_transform(sb: &mut SceneBuilder, params: &mut SceneParams) {
|
fn brush_transform(sb: &mut SceneBuilder, params: &mut SceneParams) {
|
||||||
|
|
|
@ -19,7 +19,7 @@ use std::time::Instant;
|
||||||
|
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use clap::{CommandFactory, Parser};
|
use clap::{CommandFactory, Parser};
|
||||||
use scenes::{SceneParams, SceneSet, SimpleText};
|
use scenes::{ImageCache, SceneParams, SceneSet, SimpleText};
|
||||||
use vello::peniko::Color;
|
use vello::peniko::Color;
|
||||||
use vello::util::RenderSurface;
|
use vello::util::RenderSurface;
|
||||||
use vello::{
|
use vello::{
|
||||||
|
@ -96,6 +96,7 @@ fn run(
|
||||||
let mut scene = Scene::new();
|
let mut scene = Scene::new();
|
||||||
let mut fragment = SceneFragment::new();
|
let mut fragment = SceneFragment::new();
|
||||||
let mut simple_text = SimpleText::new();
|
let mut simple_text = SimpleText::new();
|
||||||
|
let mut images = ImageCache::new();
|
||||||
let start = Instant::now();
|
let start = Instant::now();
|
||||||
|
|
||||||
let mut touch_state = multi_touch::TouchState::new();
|
let mut touch_state = multi_touch::TouchState::new();
|
||||||
|
@ -264,6 +265,7 @@ fn run(
|
||||||
let mut scene_params = SceneParams {
|
let mut scene_params = SceneParams {
|
||||||
time: start.elapsed().as_secs_f64(),
|
time: start.elapsed().as_secs_f64(),
|
||||||
text: &mut simple_text,
|
text: &mut simple_text,
|
||||||
|
images: &mut images,
|
||||||
resolution: None,
|
resolution: None,
|
||||||
base_color: None,
|
base_color: None,
|
||||||
};
|
};
|
||||||
|
|
|
@ -126,6 +126,15 @@ fn write_grad(ty: u32, index: u32, info_offset: u32) {
|
||||||
cmd_offset += 3u;
|
cmd_offset += 3u;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn write_image(xy: u32, width_height: u32, info_offset: u32) {
|
||||||
|
alloc_cmd(4u);
|
||||||
|
ptcl[cmd_offset] = CMD_IMAGE;
|
||||||
|
ptcl[cmd_offset + 1u] = info_offset;
|
||||||
|
ptcl[cmd_offset + 2u] = xy;
|
||||||
|
ptcl[cmd_offset + 3u] = width_height;
|
||||||
|
cmd_offset += 4u;
|
||||||
|
}
|
||||||
|
|
||||||
fn write_begin_clip() {
|
fn write_begin_clip() {
|
||||||
alloc_cmd(1u);
|
alloc_cmd(1u);
|
||||||
ptcl[cmd_offset] = CMD_BEGIN_CLIP;
|
ptcl[cmd_offset] = CMD_BEGIN_CLIP;
|
||||||
|
@ -377,6 +386,15 @@ fn main(
|
||||||
write_grad(CMD_RAD_GRAD, index, info_offset);
|
write_grad(CMD_RAD_GRAD, index, info_offset);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// DRAWTAG_FILL_IMAGE
|
||||||
|
case 0x1c8u: {
|
||||||
|
let linewidth = bitcast<f32>(info_bin_data[di]);
|
||||||
|
if write_path(tile, linewidth) {
|
||||||
|
let xy = scene[dd];
|
||||||
|
let width_height = scene[dd + 1u];
|
||||||
|
write_image(xy, width_height, di + 1u);
|
||||||
|
}
|
||||||
|
}
|
||||||
// DRAWTAG_BEGIN_CLIP
|
// DRAWTAG_BEGIN_CLIP
|
||||||
case 0x9u: {
|
case 0x9u: {
|
||||||
if tile.segments == 0u && tile.backdrop == 0 {
|
if tile.segments == 0u && tile.backdrop == 0 {
|
||||||
|
|
|
@ -113,7 +113,9 @@ fn main(
|
||||||
var matrx: vec4<f32>;
|
var matrx: vec4<f32>;
|
||||||
var translate: vec2<f32>;
|
var translate: vec2<f32>;
|
||||||
var linewidth = bbox.linewidth;
|
var linewidth = bbox.linewidth;
|
||||||
if linewidth >= 0.0 || tag_word == DRAWTAG_FILL_LIN_GRADIENT || tag_word == DRAWTAG_FILL_RAD_GRADIENT {
|
if linewidth >= 0.0 || tag_word == DRAWTAG_FILL_LIN_GRADIENT || tag_word == DRAWTAG_FILL_RAD_GRADIENT ||
|
||||||
|
tag_word == DRAWTAG_FILL_IMAGE
|
||||||
|
{
|
||||||
let transform = read_transform(config.transform_base, bbox.trans_ix);
|
let transform = read_transform(config.transform_base, bbox.trans_ix);
|
||||||
matrx = transform.matrx;
|
matrx = transform.matrx;
|
||||||
translate = transform.translate;
|
translate = transform.translate;
|
||||||
|
@ -124,7 +126,7 @@ fn main(
|
||||||
}
|
}
|
||||||
switch tag_word {
|
switch tag_word {
|
||||||
// DRAWTAG_FILL_COLOR, DRAWTAG_FILL_IMAGE
|
// DRAWTAG_FILL_COLOR, DRAWTAG_FILL_IMAGE
|
||||||
case 0x44u, 0x48u: {
|
case 0x44u: {
|
||||||
info[di] = bitcast<u32>(linewidth);
|
info[di] = bitcast<u32>(linewidth);
|
||||||
}
|
}
|
||||||
// DRAWTAG_FILL_LIN_GRADIENT
|
// DRAWTAG_FILL_LIN_GRADIENT
|
||||||
|
@ -169,6 +171,19 @@ fn main(
|
||||||
info[di + 9u] = bitcast<u32>(ra);
|
info[di + 9u] = bitcast<u32>(ra);
|
||||||
info[di + 10u] = bitcast<u32>(roff);
|
info[di + 10u] = bitcast<u32>(roff);
|
||||||
}
|
}
|
||||||
|
// DRAWTAG_FILL_IMAGE
|
||||||
|
case 0x1c8u: {
|
||||||
|
info[di] = bitcast<u32>(linewidth);
|
||||||
|
let inv_det = 1.0 / (matrx.x * matrx.w - matrx.y * matrx.z);
|
||||||
|
let inv_mat = inv_det * vec4(matrx.w, -matrx.y, -matrx.z, matrx.x);
|
||||||
|
let inv_tr = mat2x2(inv_mat.xy, inv_mat.zw) * -translate;
|
||||||
|
info[di + 1u] = bitcast<u32>(inv_mat.x);
|
||||||
|
info[di + 2u] = bitcast<u32>(inv_mat.y);
|
||||||
|
info[di + 3u] = bitcast<u32>(inv_mat.z);
|
||||||
|
info[di + 4u] = bitcast<u32>(inv_mat.w);
|
||||||
|
info[di + 5u] = bitcast<u32>(inv_tr.x);
|
||||||
|
info[di + 6u] = bitcast<u32>(inv_tr.y);
|
||||||
|
}
|
||||||
default: {}
|
default: {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -40,6 +40,9 @@ var gradients: texture_2d<f32>;
|
||||||
@group(0) @binding(6)
|
@group(0) @binding(6)
|
||||||
var<storage> info: array<u32>;
|
var<storage> info: array<u32>;
|
||||||
|
|
||||||
|
@group(0) @binding(7)
|
||||||
|
var image_atlas: texture_2d<f32>;
|
||||||
|
|
||||||
fn read_fill(cmd_ix: u32) -> CmdFill {
|
fn read_fill(cmd_ix: u32) -> CmdFill {
|
||||||
let tile = ptcl[cmd_ix + 1u];
|
let tile = ptcl[cmd_ix + 1u];
|
||||||
let backdrop = i32(ptcl[cmd_ix + 2u]);
|
let backdrop = i32(ptcl[cmd_ix + 2u]);
|
||||||
|
@ -81,6 +84,24 @@ fn read_rad_grad(cmd_ix: u32) -> CmdRadGrad {
|
||||||
return CmdRadGrad(index, matrx, xlat, c1, ra, roff);
|
return CmdRadGrad(index, matrx, xlat, c1, ra, roff);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn read_image(cmd_ix: u32) -> CmdImage {
|
||||||
|
let info_offset = ptcl[cmd_ix + 1u];
|
||||||
|
let xy = ptcl[cmd_ix + 2u];
|
||||||
|
let width_height = ptcl[cmd_ix + 3u];
|
||||||
|
let m0 = bitcast<f32>(info[info_offset]);
|
||||||
|
let m1 = bitcast<f32>(info[info_offset + 1u]);
|
||||||
|
let m2 = bitcast<f32>(info[info_offset + 2u]);
|
||||||
|
let m3 = bitcast<f32>(info[info_offset + 3u]);
|
||||||
|
let matrx = vec4(m0, m1, m2, m3);
|
||||||
|
let xlat = vec2(bitcast<f32>(info[info_offset + 4u]), bitcast<f32>(info[info_offset + 5u]));
|
||||||
|
// The following are not intended to be bitcasts
|
||||||
|
let x = f32(xy >> 16u);
|
||||||
|
let y = f32(xy & 0xffffu);
|
||||||
|
let width = f32(width_height >> 16u);
|
||||||
|
let height = f32(width_height & 0xffffu);
|
||||||
|
return CmdImage(matrx, xlat, vec2(x, y), vec2(width, height));
|
||||||
|
}
|
||||||
|
|
||||||
fn read_end_clip(cmd_ix: u32) -> CmdEndClip {
|
fn read_end_clip(cmd_ix: u32) -> CmdEndClip {
|
||||||
let blend = ptcl[cmd_ix + 1u];
|
let blend = ptcl[cmd_ix + 1u];
|
||||||
let alpha = bitcast<f32>(ptcl[cmd_ix + 2u]);
|
let alpha = bitcast<f32>(ptcl[cmd_ix + 2u]);
|
||||||
|
@ -265,6 +286,29 @@ fn main(
|
||||||
}
|
}
|
||||||
cmd_ix += 3u;
|
cmd_ix += 3u;
|
||||||
}
|
}
|
||||||
|
// CMD_IMAGE
|
||||||
|
case 8u: {
|
||||||
|
let image = read_image(cmd_ix);
|
||||||
|
let atlas_extents = image.atlas_offset + image.extents;
|
||||||
|
for (var i = 0u; i < PIXELS_PER_THREAD; i += 1u) {
|
||||||
|
let my_xy = vec2(xy.x + f32(i), xy.y);
|
||||||
|
let atlas_uv = image.matrx.xy * my_xy.x + image.matrx.zw * my_xy.y + image.xlat + image.atlas_offset;
|
||||||
|
// This currently clips to the image bounds. TODO: extend modes
|
||||||
|
if all(atlas_uv < atlas_extents) {
|
||||||
|
let uv_quad = vec4(max(floor(atlas_uv), image.atlas_offset), min(ceil(atlas_uv), atlas_extents));
|
||||||
|
let uv_frac = fract(atlas_uv);
|
||||||
|
let a = textureLoad(image_atlas, vec2<i32>(uv_quad.xy), 0);
|
||||||
|
let b = textureLoad(image_atlas, vec2<i32>(uv_quad.xw), 0);
|
||||||
|
let c = textureLoad(image_atlas, vec2<i32>(uv_quad.zy), 0);
|
||||||
|
let d = textureLoad(image_atlas, vec2<i32>(uv_quad.zw), 0);
|
||||||
|
let color = mix(mix(a, b, uv_frac.y), mix(c, d, uv_frac.y), uv_frac.x);
|
||||||
|
let fg_rgba = vec4(color.rgb * color.a, color.a);
|
||||||
|
let fg_i = fg_rgba * area[i];
|
||||||
|
rgba[i] = rgba[i] * (1.0 - fg_i.a) + fg_i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cmd_ix += 4u;
|
||||||
|
}
|
||||||
// CMD_BEGIN_CLIP
|
// CMD_BEGIN_CLIP
|
||||||
case 9u: {
|
case 9u: {
|
||||||
if clip_depth < BLEND_STACK_SPLIT {
|
if clip_depth < BLEND_STACK_SPLIT {
|
||||||
|
|
|
@ -19,7 +19,7 @@ let DRAWTAG_NOP = 0u;
|
||||||
let DRAWTAG_FILL_COLOR = 0x44u;
|
let DRAWTAG_FILL_COLOR = 0x44u;
|
||||||
let DRAWTAG_FILL_LIN_GRADIENT = 0x114u;
|
let DRAWTAG_FILL_LIN_GRADIENT = 0x114u;
|
||||||
let DRAWTAG_FILL_RAD_GRADIENT = 0x2dcu;
|
let DRAWTAG_FILL_RAD_GRADIENT = 0x2dcu;
|
||||||
let DRAWTAG_FILL_IMAGE = 0x48u;
|
let DRAWTAG_FILL_IMAGE = 0x1c8u;
|
||||||
let DRAWTAG_BEGIN_CLIP = 0x9u;
|
let DRAWTAG_BEGIN_CLIP = 0x9u;
|
||||||
let DRAWTAG_END_CLIP = 0x21u;
|
let DRAWTAG_END_CLIP = 0x21u;
|
||||||
|
|
||||||
|
|
|
@ -16,6 +16,7 @@ let CMD_SOLID = 3u;
|
||||||
let CMD_COLOR = 5u;
|
let CMD_COLOR = 5u;
|
||||||
let CMD_LIN_GRAD = 6u;
|
let CMD_LIN_GRAD = 6u;
|
||||||
let CMD_RAD_GRAD = 7u;
|
let CMD_RAD_GRAD = 7u;
|
||||||
|
let CMD_IMAGE = 8u;
|
||||||
let CMD_BEGIN_CLIP = 9u;
|
let CMD_BEGIN_CLIP = 9u;
|
||||||
let CMD_END_CLIP = 10u;
|
let CMD_END_CLIP = 10u;
|
||||||
let CMD_JUMP = 11u;
|
let CMD_JUMP = 11u;
|
||||||
|
@ -57,6 +58,13 @@ struct CmdRadGrad {
|
||||||
roff: f32,
|
roff: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct CmdImage {
|
||||||
|
matrx: vec4<f32>,
|
||||||
|
xlat: vec2<f32>,
|
||||||
|
atlas_offset: vec2<f32>,
|
||||||
|
extents: vec2<f32>,
|
||||||
|
}
|
||||||
|
|
||||||
struct CmdEndClip {
|
struct CmdEndClip {
|
||||||
blend: u32,
|
blend: u32,
|
||||||
alpha: f32,
|
alpha: f32,
|
||||||
|
|
|
@ -20,6 +20,7 @@ mod draw;
|
||||||
mod encoding;
|
mod encoding;
|
||||||
mod glyph;
|
mod glyph;
|
||||||
mod glyph_cache;
|
mod glyph_cache;
|
||||||
|
mod image_cache;
|
||||||
mod math;
|
mod math;
|
||||||
mod monoid;
|
mod monoid;
|
||||||
mod path;
|
mod path;
|
||||||
|
|
|
@ -38,7 +38,7 @@ impl DrawTag {
|
||||||
pub const RADIAL_GRADIENT: Self = Self(0x2dc);
|
pub const RADIAL_GRADIENT: Self = Self(0x2dc);
|
||||||
|
|
||||||
/// Image fill.
|
/// Image fill.
|
||||||
pub const IMAGE: Self = Self(0x48);
|
pub const IMAGE: Self = Self(0x1c8);
|
||||||
|
|
||||||
/// Begin layer/clip.
|
/// Begin layer/clip.
|
||||||
pub const BEGIN_CLIP: Self = Self(0x9);
|
pub const BEGIN_CLIP: Self = Self(0x9);
|
||||||
|
@ -104,10 +104,10 @@ pub struct DrawRadialGradient {
|
||||||
#[derive(Clone, Copy, Debug, Default, Zeroable, Pod)]
|
#[derive(Clone, Copy, Debug, Default, Zeroable, Pod)]
|
||||||
#[repr(C)]
|
#[repr(C)]
|
||||||
pub struct DrawImage {
|
pub struct DrawImage {
|
||||||
/// Image index.
|
/// Packed atlas coordinates.
|
||||||
pub index: u32,
|
pub xy: u32,
|
||||||
/// Packed image offset.
|
/// Packed image dimensions.
|
||||||
pub offset: u32,
|
pub width_height: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Draw data for a clip or layer.
|
/// Draw data for a clip or layer.
|
||||||
|
|
|
@ -14,12 +14,14 @@
|
||||||
//
|
//
|
||||||
// Also licensed under MIT license, at your choice.
|
// Also licensed under MIT license, at your choice.
|
||||||
|
|
||||||
|
use crate::encoding::DrawImage;
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
resolve::Patch, DrawColor, DrawLinearGradient, DrawRadialGradient, DrawTag, Glyph, GlyphRun,
|
resolve::Patch, DrawColor, DrawLinearGradient, DrawRadialGradient, DrawTag, Glyph, GlyphRun,
|
||||||
PathEncoder, PathTag, Transform,
|
PathEncoder, PathTag, Transform,
|
||||||
};
|
};
|
||||||
|
|
||||||
use peniko::{kurbo::Shape, BlendMode, BrushRef, ColorStop, Extend, GradientKind};
|
use peniko::{kurbo::Shape, BlendMode, BrushRef, ColorStop, Extend, GradientKind, Image};
|
||||||
|
|
||||||
/// Encoded data streams for a scene.
|
/// Encoded data streams for a scene.
|
||||||
#[derive(Clone, Default)]
|
#[derive(Clone, Default)]
|
||||||
|
@ -122,16 +124,26 @@ impl Encoding {
|
||||||
self.n_open_clips += other.n_open_clips;
|
self.n_open_clips += other.n_open_clips;
|
||||||
self.patches
|
self.patches
|
||||||
.extend(other.patches.iter().map(|patch| match patch {
|
.extend(other.patches.iter().map(|patch| match patch {
|
||||||
Patch::Ramp { offset, stops } => {
|
Patch::Ramp {
|
||||||
|
draw_data_offset: offset,
|
||||||
|
stops,
|
||||||
|
} => {
|
||||||
let stops = stops.start + stops_base..stops.end + stops_base;
|
let stops = stops.start + stops_base..stops.end + stops_base;
|
||||||
Patch::Ramp {
|
Patch::Ramp {
|
||||||
offset: offset + offsets.draw_data,
|
draw_data_offset: offset + offsets.draw_data,
|
||||||
stops,
|
stops,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Patch::GlyphRun { index } => Patch::GlyphRun {
|
Patch::GlyphRun { index } => Patch::GlyphRun {
|
||||||
index: index + glyph_runs_base,
|
index: index + glyph_runs_base,
|
||||||
},
|
},
|
||||||
|
Patch::Image {
|
||||||
|
image,
|
||||||
|
draw_data_offset,
|
||||||
|
} => Patch::Image {
|
||||||
|
image: image.clone(),
|
||||||
|
draw_data_offset: *draw_data_offset + offsets.draw_data,
|
||||||
|
},
|
||||||
}));
|
}));
|
||||||
self.color_stops.extend_from_slice(&other.color_stops);
|
self.color_stops.extend_from_slice(&other.color_stops);
|
||||||
if let Some(transform) = *transform {
|
if let Some(transform) = *transform {
|
||||||
|
@ -250,8 +262,8 @@ impl Encoding {
|
||||||
todo!("sweep gradients aren't supported yet!")
|
todo!("sweep gradients aren't supported yet!")
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
BrushRef::Image(_) => {
|
BrushRef::Image(image) => {
|
||||||
todo!("images aren't supported yet!")
|
self.encode_image(image, alpha);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -290,6 +302,22 @@ impl Encoding {
|
||||||
.extend_from_slice(bytemuck::bytes_of(&gradient));
|
.extend_from_slice(bytemuck::bytes_of(&gradient));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Encodes an image brush.
|
||||||
|
pub fn encode_image(&mut self, image: &Image, _alpha: f32) {
|
||||||
|
// TODO: feed the alpha multiplier through the full pipeline for consistency
|
||||||
|
// with other brushes?
|
||||||
|
self.patches.push(Patch::Image {
|
||||||
|
image: image.clone(),
|
||||||
|
draw_data_offset: self.draw_data.len(),
|
||||||
|
});
|
||||||
|
self.draw_tags.push(DrawTag::IMAGE);
|
||||||
|
self.draw_data
|
||||||
|
.extend_from_slice(bytemuck::bytes_of(&DrawImage {
|
||||||
|
xy: 0,
|
||||||
|
width_height: (image.width << 16) | (image.height & 0xFFFF),
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
/// Encodes a begin clip command.
|
/// Encodes a begin clip command.
|
||||||
pub fn encode_begin_clip(&mut self, blend_mode: BlendMode, alpha: f32) {
|
pub fn encode_begin_clip(&mut self, blend_mode: BlendMode, alpha: f32) {
|
||||||
use super::DrawBeginClip;
|
use super::DrawBeginClip;
|
||||||
|
@ -329,7 +357,7 @@ impl Encoding {
|
||||||
self.color_stops.extend(color_stops);
|
self.color_stops.extend(color_stops);
|
||||||
}
|
}
|
||||||
self.patches.push(Patch::Ramp {
|
self.patches.push(Patch::Ramp {
|
||||||
offset,
|
draw_data_offset: offset,
|
||||||
stops: stops_start..self.color_stops.len(),
|
stops: stops_start..self.color_stops.len(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
92
src/encoding/image_cache.rs
Normal file
92
src/encoding/image_cache.rs
Normal file
|
@ -0,0 +1,92 @@
|
||||||
|
// Copyright 2022 Google LLC
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
//
|
||||||
|
// Also licensed under MIT license, at your choice.
|
||||||
|
|
||||||
|
use guillotiere::{size2, AtlasAllocator};
|
||||||
|
use peniko::Image;
|
||||||
|
use std::collections::{hash_map::Entry, HashMap};
|
||||||
|
|
||||||
|
const DEFAULT_ATLAS_SIZE: i32 = 1024;
|
||||||
|
const MAX_ATLAS_SIZE: i32 = 8192;
|
||||||
|
|
||||||
|
pub struct Images<'a> {
|
||||||
|
pub width: u32,
|
||||||
|
pub height: u32,
|
||||||
|
pub images: &'a [(Image, u32, u32)],
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ImageCache {
|
||||||
|
atlas: AtlasAllocator,
|
||||||
|
/// Map from image blob id to atlas location.
|
||||||
|
map: HashMap<u64, (u32, u32)>,
|
||||||
|
/// List of all allocated images with associated atlas location.
|
||||||
|
images: Vec<(Image, u32, u32)>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for ImageCache {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::new()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ImageCache {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
atlas: AtlasAllocator::new(size2(DEFAULT_ATLAS_SIZE, DEFAULT_ATLAS_SIZE)),
|
||||||
|
map: Default::default(),
|
||||||
|
images: Default::default(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn images(&self) -> Images {
|
||||||
|
Images {
|
||||||
|
width: self.atlas.size().width as u32,
|
||||||
|
height: self.atlas.size().height as u32,
|
||||||
|
images: &self.images,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn bump_size(&mut self) -> bool {
|
||||||
|
let new_size = self.atlas.size().width * 2;
|
||||||
|
if new_size > MAX_ATLAS_SIZE {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
self.atlas = AtlasAllocator::new(size2(new_size, new_size));
|
||||||
|
self.map.clear();
|
||||||
|
self.images.clear();
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn clear(&mut self) {
|
||||||
|
self.atlas.clear();
|
||||||
|
self.map.clear();
|
||||||
|
self.images.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_or_insert(&mut self, image: &Image) -> Option<(u32, u32)> {
|
||||||
|
match self.map.entry(image.data.id()) {
|
||||||
|
Entry::Occupied(occupied) => Some(*occupied.get()),
|
||||||
|
Entry::Vacant(vacant) => {
|
||||||
|
let alloc = self
|
||||||
|
.atlas
|
||||||
|
.allocate(size2(image.width as _, image.height as _))?;
|
||||||
|
let x = alloc.rectangle.min.x as u32;
|
||||||
|
let y = alloc.rectangle.min.y as u32;
|
||||||
|
self.images.push((image.clone(), x, y));
|
||||||
|
Some(*vacant.insert((x, y)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -18,9 +18,11 @@ use std::ops::Range;
|
||||||
|
|
||||||
use bytemuck::{Pod, Zeroable};
|
use bytemuck::{Pod, Zeroable};
|
||||||
use moscato::pinot::FontRef;
|
use moscato::pinot::FontRef;
|
||||||
|
use peniko::Image;
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
glyph_cache::{CachedRange, GlyphCache, GlyphKey},
|
glyph_cache::{CachedRange, GlyphCache, GlyphKey},
|
||||||
|
image_cache::{ImageCache, Images},
|
||||||
ramp_cache::{RampCache, Ramps},
|
ramp_cache::{RampCache, Ramps},
|
||||||
DrawTag, Encoding, PathTag, StreamOffsets, Transform,
|
DrawTag, Encoding, PathTag, StreamOffsets, Transform,
|
||||||
};
|
};
|
||||||
|
@ -144,6 +146,8 @@ pub struct Resolver {
|
||||||
glyph_ranges: Vec<CachedRange>,
|
glyph_ranges: Vec<CachedRange>,
|
||||||
glyph_cx: GlyphContext,
|
glyph_cx: GlyphContext,
|
||||||
ramp_cache: RampCache,
|
ramp_cache: RampCache,
|
||||||
|
image_cache: ImageCache,
|
||||||
|
pending_images: Vec<PendingImage>,
|
||||||
patches: Vec<ResolvedPatch>,
|
patches: Vec<ResolvedPatch>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -159,8 +163,9 @@ impl Resolver {
|
||||||
&'a mut self,
|
&'a mut self,
|
||||||
encoding: &Encoding,
|
encoding: &Encoding,
|
||||||
packed: &mut Vec<u8>,
|
packed: &mut Vec<u8>,
|
||||||
) -> (Layout, Ramps<'a>) {
|
) -> (Layout, Ramps<'a>, Images<'a>) {
|
||||||
let sizes = self.resolve_patches(encoding);
|
let sizes = self.resolve_patches(encoding);
|
||||||
|
self.resolve_pending_images();
|
||||||
let data = packed;
|
let data = packed;
|
||||||
data.clear();
|
data.clear();
|
||||||
let mut layout = Layout::default();
|
let mut layout = Layout::default();
|
||||||
|
@ -261,6 +266,26 @@ impl Resolver {
|
||||||
pos = *draw_data_offset + 4;
|
pos = *draw_data_offset + 4;
|
||||||
}
|
}
|
||||||
ResolvedPatch::GlyphRun { .. } => {}
|
ResolvedPatch::GlyphRun { .. } => {}
|
||||||
|
ResolvedPatch::Image {
|
||||||
|
index,
|
||||||
|
draw_data_offset,
|
||||||
|
} => {
|
||||||
|
if pos < *draw_data_offset {
|
||||||
|
data.extend_from_slice(&encoding.draw_data[pos..*draw_data_offset]);
|
||||||
|
}
|
||||||
|
if let Some((x, y)) = self.pending_images[*index].xy {
|
||||||
|
let xy = (x << 16) | y;
|
||||||
|
data.extend_from_slice(bytemuck::bytes_of(&xy));
|
||||||
|
pos = *draw_data_offset + 4;
|
||||||
|
} else {
|
||||||
|
// If we get here, we failed to allocate a slot for this image in the atlas.
|
||||||
|
// In this case, let's zero out the dimensions so we don't attempt to render
|
||||||
|
// anything.
|
||||||
|
// TODO: a better strategy: texture array? downsample large images?
|
||||||
|
data.extend_from_slice(&[0u8; 8]);
|
||||||
|
pos = *draw_data_offset + 8;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if pos < stream.len() {
|
if pos < stream.len() {
|
||||||
|
@ -336,21 +361,26 @@ impl Resolver {
|
||||||
}
|
}
|
||||||
layout.n_draw_objects = layout.n_paths;
|
layout.n_draw_objects = layout.n_paths;
|
||||||
assert_eq!(capacity, data.len());
|
assert_eq!(capacity, data.len());
|
||||||
(layout, self.ramp_cache.ramps())
|
(layout, self.ramp_cache.ramps(), self.image_cache.images())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn resolve_patches(&mut self, encoding: &Encoding) -> StreamOffsets {
|
fn resolve_patches(&mut self, encoding: &Encoding) -> StreamOffsets {
|
||||||
self.ramp_cache.advance();
|
self.ramp_cache.advance();
|
||||||
self.glyph_cache.clear();
|
self.glyph_cache.clear();
|
||||||
self.glyph_ranges.clear();
|
self.glyph_ranges.clear();
|
||||||
|
self.image_cache.clear();
|
||||||
|
self.pending_images.clear();
|
||||||
self.patches.clear();
|
self.patches.clear();
|
||||||
let mut sizes = StreamOffsets::default();
|
let mut sizes = StreamOffsets::default();
|
||||||
for patch in &encoding.patches {
|
for patch in &encoding.patches {
|
||||||
match patch {
|
match patch {
|
||||||
Patch::Ramp { offset, stops } => {
|
Patch::Ramp {
|
||||||
|
draw_data_offset,
|
||||||
|
stops,
|
||||||
|
} => {
|
||||||
let ramp_id = self.ramp_cache.add(&encoding.color_stops[stops.clone()]);
|
let ramp_id = self.ramp_cache.add(&encoding.color_stops[stops.clone()]);
|
||||||
self.patches.push(ResolvedPatch::Ramp {
|
self.patches.push(ResolvedPatch::Ramp {
|
||||||
draw_data_offset: *offset + sizes.draw_data,
|
draw_data_offset: *draw_data_offset + sizes.draw_data,
|
||||||
ramp_id,
|
ramp_id,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -414,10 +444,50 @@ impl Resolver {
|
||||||
transform,
|
transform,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
Patch::Image {
|
||||||
|
draw_data_offset,
|
||||||
|
image,
|
||||||
|
} => {
|
||||||
|
let index = self.pending_images.len();
|
||||||
|
self.pending_images.push(PendingImage {
|
||||||
|
image: image.clone(),
|
||||||
|
xy: None,
|
||||||
|
});
|
||||||
|
self.patches.push(ResolvedPatch::Image {
|
||||||
|
index,
|
||||||
|
draw_data_offset: *draw_data_offset + sizes.draw_data,
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
sizes
|
sizes
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn resolve_pending_images(&mut self) {
|
||||||
|
self.image_cache.clear();
|
||||||
|
'outer: loop {
|
||||||
|
// Loop over the images, attempting to allocate them all into the atlas.
|
||||||
|
for pending_image in &mut self.pending_images {
|
||||||
|
if let Some(xy) = self.image_cache.get_or_insert(&pending_image.image) {
|
||||||
|
pending_image.xy = Some(xy);
|
||||||
|
} else {
|
||||||
|
// We failed to allocate. Try to bump the atlas size.
|
||||||
|
if self.image_cache.bump_size() {
|
||||||
|
// We were able to increase the atlas size. Restart the outer loop.
|
||||||
|
continue 'outer;
|
||||||
|
} else {
|
||||||
|
// If the atlas is already maximum size, there's nothing we can do. Set
|
||||||
|
// the xy field to None so this image isn't rendered and then carry on--
|
||||||
|
// other images might still fit.
|
||||||
|
pending_image.xy = None;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// If we made it here, we've either successfully allocated all images or we reached
|
||||||
|
// the maximum atlas size.
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
|
@ -426,7 +496,7 @@ pub enum Patch {
|
||||||
/// Gradient ramp resource.
|
/// Gradient ramp resource.
|
||||||
Ramp {
|
Ramp {
|
||||||
/// Byte offset to the ramp id in the draw data stream.
|
/// Byte offset to the ramp id in the draw data stream.
|
||||||
offset: usize,
|
draw_data_offset: usize,
|
||||||
/// Range of the gradient stops in the resource set.
|
/// Range of the gradient stops in the resource set.
|
||||||
stops: Range<usize>,
|
stops: Range<usize>,
|
||||||
},
|
},
|
||||||
|
@ -435,6 +505,20 @@ pub enum Patch {
|
||||||
/// Index in the glyph run buffer.
|
/// Index in the glyph run buffer.
|
||||||
index: usize,
|
index: usize,
|
||||||
},
|
},
|
||||||
|
/// Image resource.
|
||||||
|
Image {
|
||||||
|
/// Offset to the atlas coordinates in the draw data stream.
|
||||||
|
draw_data_offset: usize,
|
||||||
|
/// Underlying image data.
|
||||||
|
image: Image,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Image to be allocated in the atlas.
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
struct PendingImage {
|
||||||
|
image: Image,
|
||||||
|
xy: Option<(u32, u32)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
|
@ -453,6 +537,12 @@ enum ResolvedPatch {
|
||||||
/// Global transform.
|
/// Global transform.
|
||||||
transform: Transform,
|
transform: Transform,
|
||||||
},
|
},
|
||||||
|
Image {
|
||||||
|
/// Index of pending image element.
|
||||||
|
index: usize,
|
||||||
|
/// Offset to the atlas location in the draw data stream.
|
||||||
|
draw_data_offset: usize,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
fn slice_size_in_bytes<T: Sized>(slice: &[T], extra: usize) -> usize {
|
fn slice_size_in_bytes<T: Sized>(slice: &[T], extra: usize) -> usize {
|
||||||
|
|
|
@ -89,6 +89,7 @@ pub enum Command {
|
||||||
Upload(BufProxy, Vec<u8>),
|
Upload(BufProxy, Vec<u8>),
|
||||||
UploadUniform(BufProxy, Vec<u8>),
|
UploadUniform(BufProxy, Vec<u8>),
|
||||||
UploadImage(ImageProxy, Vec<u8>),
|
UploadImage(ImageProxy, Vec<u8>),
|
||||||
|
WriteImage(ImageProxy, [u32; 4], Vec<u8>),
|
||||||
// Discussion question: third argument is vec of resources?
|
// Discussion question: third argument is vec of resources?
|
||||||
// Maybe use tricks to make more ergonomic?
|
// Maybe use tricks to make more ergonomic?
|
||||||
// Alternative: provide bufs & images as separate sequences
|
// Alternative: provide bufs & images as separate sequences
|
||||||
|
@ -275,11 +276,6 @@ impl Engine {
|
||||||
self.bind_map.insert_buf(buf_proxy, buf);
|
self.bind_map.insert_buf(buf_proxy, buf);
|
||||||
}
|
}
|
||||||
Command::UploadImage(image_proxy, bytes) => {
|
Command::UploadImage(image_proxy, bytes) => {
|
||||||
let buf = device.create_buffer_init(&wgpu::util::BufferInitDescriptor {
|
|
||||||
label: None,
|
|
||||||
contents: bytes,
|
|
||||||
usage: wgpu::BufferUsages::COPY_SRC,
|
|
||||||
});
|
|
||||||
let format = image_proxy.format.to_wgpu();
|
let format = image_proxy.format.to_wgpu();
|
||||||
let texture = device.create_texture(&wgpu::TextureDescriptor {
|
let texture = device.create_texture(&wgpu::TextureDescriptor {
|
||||||
label: None,
|
label: None,
|
||||||
|
@ -305,21 +301,19 @@ impl Engine {
|
||||||
array_layer_count: None,
|
array_layer_count: None,
|
||||||
format: Some(TextureFormat::Rgba8Unorm),
|
format: Some(TextureFormat::Rgba8Unorm),
|
||||||
});
|
});
|
||||||
encoder.copy_buffer_to_texture(
|
queue.write_texture(
|
||||||
wgpu::ImageCopyBuffer {
|
|
||||||
buffer: &buf,
|
|
||||||
layout: wgpu::ImageDataLayout {
|
|
||||||
offset: 0,
|
|
||||||
bytes_per_row: NonZeroU32::new(image_proxy.width * 4),
|
|
||||||
rows_per_image: None,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
wgpu::ImageCopyTexture {
|
wgpu::ImageCopyTexture {
|
||||||
texture: &texture,
|
texture: &texture,
|
||||||
mip_level: 0,
|
mip_level: 0,
|
||||||
origin: wgpu::Origin3d { x: 0, y: 0, z: 0 },
|
origin: wgpu::Origin3d { x: 0, y: 0, z: 0 },
|
||||||
aspect: TextureAspect::All,
|
aspect: TextureAspect::All,
|
||||||
},
|
},
|
||||||
|
bytes,
|
||||||
|
wgpu::ImageDataLayout {
|
||||||
|
offset: 0,
|
||||||
|
bytes_per_row: NonZeroU32::new(image_proxy.width * 4),
|
||||||
|
rows_per_image: None,
|
||||||
|
},
|
||||||
wgpu::Extent3d {
|
wgpu::Extent3d {
|
||||||
width: image_proxy.width,
|
width: image_proxy.width,
|
||||||
height: image_proxy.height,
|
height: image_proxy.height,
|
||||||
|
@ -329,6 +323,29 @@ impl Engine {
|
||||||
self.bind_map
|
self.bind_map
|
||||||
.insert_image(image_proxy.id, texture, texture_view)
|
.insert_image(image_proxy.id, texture, texture_view)
|
||||||
}
|
}
|
||||||
|
Command::WriteImage(proxy, [x, y, width, height], data) => {
|
||||||
|
if let Ok((texture, _)) = self.bind_map.get_or_create_image(*proxy, device) {
|
||||||
|
queue.write_texture(
|
||||||
|
wgpu::ImageCopyTexture {
|
||||||
|
texture: &texture,
|
||||||
|
mip_level: 0,
|
||||||
|
origin: wgpu::Origin3d { x: *x, y: *y, z: 0 },
|
||||||
|
aspect: TextureAspect::All,
|
||||||
|
},
|
||||||
|
&data[..],
|
||||||
|
wgpu::ImageDataLayout {
|
||||||
|
offset: 0,
|
||||||
|
bytes_per_row: NonZeroU32::new(*width * 4),
|
||||||
|
rows_per_image: None,
|
||||||
|
},
|
||||||
|
wgpu::Extent3d {
|
||||||
|
width: *width,
|
||||||
|
height: *height,
|
||||||
|
depth_or_array_layers: 1,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
Command::Dispatch(shader_id, wg_size, bindings) => {
|
Command::Dispatch(shader_id, wg_size, bindings) => {
|
||||||
// println!("dispatching {:?} with {} bindings", wg_size, bindings.len());
|
// println!("dispatching {:?} with {} bindings", wg_size, bindings.len());
|
||||||
let shader = &self.shaders[shader_id.0];
|
let shader = &self.shaders[shader_id.0];
|
||||||
|
@ -444,6 +461,19 @@ impl Recording {
|
||||||
image_proxy
|
image_proxy
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn write_image(
|
||||||
|
&mut self,
|
||||||
|
image: ImageProxy,
|
||||||
|
x: u32,
|
||||||
|
y: u32,
|
||||||
|
width: u32,
|
||||||
|
height: u32,
|
||||||
|
data: impl Into<Vec<u8>>,
|
||||||
|
) {
|
||||||
|
let data = data.into();
|
||||||
|
self.push(Command::WriteImage(image, [x, y, width, height], data));
|
||||||
|
}
|
||||||
|
|
||||||
pub fn dispatch<R>(&mut self, shader: ShaderId, wg_size: (u32, u32, u32), resources: R)
|
pub fn dispatch<R>(&mut self, shader: ShaderId, wg_size: (u32, u32, u32), resources: R)
|
||||||
where
|
where
|
||||||
R: IntoIterator,
|
R: IntoIterator,
|
||||||
|
@ -716,6 +746,44 @@ impl BindMap {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn get_or_create_image(
|
||||||
|
&mut self,
|
||||||
|
proxy: ImageProxy,
|
||||||
|
device: &Device,
|
||||||
|
) -> Result<&(Texture, TextureView), Error> {
|
||||||
|
match self.image_map.entry(proxy.id) {
|
||||||
|
Entry::Occupied(occupied) => Ok(occupied.into_mut()),
|
||||||
|
Entry::Vacant(vacant) => {
|
||||||
|
let format = proxy.format.to_wgpu();
|
||||||
|
let texture = device.create_texture(&wgpu::TextureDescriptor {
|
||||||
|
label: None,
|
||||||
|
size: wgpu::Extent3d {
|
||||||
|
width: proxy.width,
|
||||||
|
height: proxy.height,
|
||||||
|
depth_or_array_layers: 1,
|
||||||
|
},
|
||||||
|
mip_level_count: 1,
|
||||||
|
sample_count: 1,
|
||||||
|
dimension: wgpu::TextureDimension::D2,
|
||||||
|
usage: TextureUsages::TEXTURE_BINDING | TextureUsages::COPY_DST,
|
||||||
|
format,
|
||||||
|
view_formats: &[],
|
||||||
|
});
|
||||||
|
let texture_view = texture.create_view(&wgpu::TextureViewDescriptor {
|
||||||
|
label: None,
|
||||||
|
dimension: Some(TextureViewDimension::D2),
|
||||||
|
aspect: TextureAspect::All,
|
||||||
|
mip_level_count: None,
|
||||||
|
base_mip_level: 0,
|
||||||
|
base_array_layer: 0,
|
||||||
|
array_layer_count: None,
|
||||||
|
format: Some(TextureFormat::Rgba8Unorm),
|
||||||
|
});
|
||||||
|
Ok(vacant.insert((texture, texture_view)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const SIZE_CLASS_BITS: u32 = 1;
|
const SIZE_CLASS_BITS: u32 = 1;
|
||||||
|
|
|
@ -33,6 +33,7 @@ struct FineResources {
|
||||||
ptcl_buf: ResourceProxy,
|
ptcl_buf: ResourceProxy,
|
||||||
gradient_image: ResourceProxy,
|
gradient_image: ResourceProxy,
|
||||||
info_bin_data_buf: ResourceProxy,
|
info_bin_data_buf: ResourceProxy,
|
||||||
|
image_atlas: ResourceProxy,
|
||||||
|
|
||||||
out_image: ImageProxy,
|
out_image: ImageProxy,
|
||||||
}
|
}
|
||||||
|
@ -216,7 +217,7 @@ impl Render {
|
||||||
let mut recording = Recording::default();
|
let mut recording = Recording::default();
|
||||||
let mut resolver = Resolver::new();
|
let mut resolver = Resolver::new();
|
||||||
let mut packed = vec![];
|
let mut packed = vec![];
|
||||||
let (layout, ramps) = resolver.resolve(encoding, &mut packed);
|
let (layout, ramps, images) = resolver.resolve(encoding, &mut packed);
|
||||||
let gradient_image = if ramps.height == 0 {
|
let gradient_image = if ramps.height == 0 {
|
||||||
ResourceProxy::new_image(1, 1, ImageFormat::Rgba8)
|
ResourceProxy::new_image(1, 1, ImageFormat::Rgba8)
|
||||||
} else {
|
} else {
|
||||||
|
@ -228,6 +229,11 @@ impl Render {
|
||||||
data,
|
data,
|
||||||
))
|
))
|
||||||
};
|
};
|
||||||
|
let image_atlas = if images.images.is_empty() {
|
||||||
|
ImageProxy::new(1, 1, ImageFormat::Rgba8)
|
||||||
|
} else {
|
||||||
|
ImageProxy::new(images.width, images.height, ImageFormat::Rgba8)
|
||||||
|
};
|
||||||
// TODO: calculate for real when we do rectangles
|
// TODO: calculate for real when we do rectangles
|
||||||
let n_pathtag = layout.path_tags(&packed).len();
|
let n_pathtag = layout.path_tags(&packed).len();
|
||||||
let pathtag_padded = align_up(n_pathtag, 4 * shaders::PATHTAG_REDUCE_WG);
|
let pathtag_padded = align_up(n_pathtag, 4 * shaders::PATHTAG_REDUCE_WG);
|
||||||
|
@ -251,6 +257,16 @@ impl Render {
|
||||||
ptcl_size: self.ptcl_size,
|
ptcl_size: self.ptcl_size,
|
||||||
layout: layout,
|
layout: layout,
|
||||||
};
|
};
|
||||||
|
for image in images.images {
|
||||||
|
recording.write_image(
|
||||||
|
image_atlas,
|
||||||
|
image.1,
|
||||||
|
image.2,
|
||||||
|
image.0.width,
|
||||||
|
image.0.height,
|
||||||
|
image.0.data.data(),
|
||||||
|
);
|
||||||
|
}
|
||||||
// println!("{:?}", config);
|
// println!("{:?}", config);
|
||||||
let scene_buf = ResourceProxy::Buf(recording.upload("scene", packed));
|
let scene_buf = ResourceProxy::Buf(recording.upload("scene", packed));
|
||||||
let config_buf =
|
let config_buf =
|
||||||
|
@ -504,6 +520,7 @@ impl Render {
|
||||||
ptcl_buf,
|
ptcl_buf,
|
||||||
gradient_image,
|
gradient_image,
|
||||||
info_bin_data_buf,
|
info_bin_data_buf,
|
||||||
|
image_atlas: ResourceProxy::Image(image_atlas),
|
||||||
out_image,
|
out_image,
|
||||||
});
|
});
|
||||||
if robust {
|
if robust {
|
||||||
|
@ -527,6 +544,7 @@ impl Render {
|
||||||
fine.ptcl_buf,
|
fine.ptcl_buf,
|
||||||
fine.gradient_image,
|
fine.gradient_image,
|
||||||
fine.info_bin_data_buf,
|
fine.info_bin_data_buf,
|
||||||
|
fine.image_atlas,
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
recording.free_resource(fine.config_buf);
|
recording.free_resource(fine.config_buf);
|
||||||
|
@ -534,6 +552,7 @@ impl Render {
|
||||||
recording.free_resource(fine.segments_buf);
|
recording.free_resource(fine.segments_buf);
|
||||||
recording.free_resource(fine.ptcl_buf);
|
recording.free_resource(fine.ptcl_buf);
|
||||||
recording.free_resource(fine.gradient_image);
|
recording.free_resource(fine.gradient_image);
|
||||||
|
recording.free_resource(fine.image_atlas);
|
||||||
recording.free_resource(fine.info_bin_data_buf);
|
recording.free_resource(fine.info_bin_data_buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
13
src/scene.rs
13
src/scene.rs
|
@ -15,7 +15,7 @@
|
||||||
// Also licensed under MIT license, at your choice.
|
// Also licensed under MIT license, at your choice.
|
||||||
|
|
||||||
use peniko::kurbo::{Affine, Rect, Shape};
|
use peniko::kurbo::{Affine, Rect, Shape};
|
||||||
use peniko::{BlendMode, BrushRef, Color, Fill, Font, Stroke, StyleRef};
|
use peniko::{BlendMode, BrushRef, Color, Fill, Font, Image, Stroke, StyleRef};
|
||||||
|
|
||||||
use crate::encoding::{Encoding, Glyph, GlyphRun, Patch, Transform};
|
use crate::encoding::{Encoding, Glyph, GlyphRun, Patch, Transform};
|
||||||
|
|
||||||
|
@ -168,6 +168,17 @@ impl<'a> SceneBuilder<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Draws an image at its natural size with the given transform.
|
||||||
|
pub fn draw_image(&mut self, image: &Image, transform: Affine) {
|
||||||
|
self.fill(
|
||||||
|
Fill::NonZero,
|
||||||
|
transform,
|
||||||
|
image,
|
||||||
|
None,
|
||||||
|
&Rect::new(0.0, 0.0, image.width as f64, image.height as f64),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns a builder for encoding a glyph run.
|
/// Returns a builder for encoding a glyph run.
|
||||||
pub fn draw_glyphs(&mut self, font: &Font) -> DrawGlyphs {
|
pub fn draw_glyphs(&mut self, font: &Font) -> DrawGlyphs {
|
||||||
DrawGlyphs::new(self.scene, font)
|
DrawGlyphs::new(self.scene, font)
|
||||||
|
|
|
@ -351,6 +351,7 @@ pub fn full_shaders(device: &Device, engine: &mut Engine) -> Result<FullShaders,
|
||||||
BindType::BufReadOnly,
|
BindType::BufReadOnly,
|
||||||
BindType::ImageRead(ImageFormat::Rgba8),
|
BindType::ImageRead(ImageFormat::Rgba8),
|
||||||
BindType::BufReadOnly,
|
BindType::BufReadOnly,
|
||||||
|
BindType::ImageRead(ImageFormat::Rgba8),
|
||||||
],
|
],
|
||||||
)?;
|
)?;
|
||||||
Ok(FullShaders {
|
Ok(FullShaders {
|
||||||
|
|
Loading…
Add table
Reference in a new issue