mirror of
https://github.com/italicsjenga/vello.git
synced 2025-01-25 18:56:35 +11:00
Fix rendering artifacts with strokes
A number of things weren't being computed correctly for strokes, including bounding boxes and also incorrectly applying y_edge logic (which is only correct for fills).
This commit is contained in:
parent
cf9b5c5805
commit
3c0752f6ae
4 changed files with 32 additions and 25 deletions
|
@ -113,6 +113,7 @@ fn main(
|
||||||
// classic memory vs ALU tradeoff.
|
// classic memory vs ALU tradeoff.
|
||||||
let cubic = cubics[global_id.x];
|
let cubic = cubics[global_id.x];
|
||||||
let path = paths[cubic.path_ix];
|
let path = paths[cubic.path_ix];
|
||||||
|
let is_stroke = (cubic.flags & CUBIC_IS_STROKE) != 0u;
|
||||||
let bbox = vec4<i32>(path.bbox);
|
let bbox = vec4<i32>(path.bbox);
|
||||||
let p0 = cubic.p0;
|
let p0 = cubic.p0;
|
||||||
let p1 = cubic.p1;
|
let p1 = cubic.p1;
|
||||||
|
@ -169,15 +170,15 @@ fn main(
|
||||||
}
|
}
|
||||||
|
|
||||||
// Output line segment lp0..lp1
|
// Output line segment lp0..lp1
|
||||||
let xymin = min(lp0, lp1);
|
let xymin = min(lp0, lp1) - cubic.stroke;
|
||||||
let xymax = max(lp0, lp1);
|
let xymax = max(lp0, lp1) + cubic.stroke;
|
||||||
let dp = lp1 - lp0;
|
let dp = lp1 - lp0;
|
||||||
let recip_dx = 1.0 / dp.x;
|
let recip_dx = 1.0 / dp.x;
|
||||||
let invslope = select(dp.x / dp.y, 1.0e9, abs(dp.y) < 1.0e-9);
|
let invslope = select(dp.x / dp.y, 1.0e9, abs(dp.y) < 1.0e-9);
|
||||||
let c = 0.5 * abs(invslope);
|
|
||||||
let b = invslope;
|
|
||||||
let SX = 1.0 / f32(TILE_WIDTH);
|
let SX = 1.0 / f32(TILE_WIDTH);
|
||||||
let SY = 1.0 / f32(TILE_HEIGHT);
|
let SY = 1.0 / f32(TILE_HEIGHT);
|
||||||
|
let c = (cubic.stroke.x + abs(invslope) * (0.5 * f32(TILE_HEIGHT) + cubic.stroke.y)) * SX;
|
||||||
|
let b = invslope;
|
||||||
let a = (lp0.x - (lp0.y - 0.5 * f32(TILE_HEIGHT)) * b) * SX;
|
let a = (lp0.x - (lp0.y - 0.5 * f32(TILE_HEIGHT)) * b) * SX;
|
||||||
var x0 = i32(floor(xymin.x * SX));
|
var x0 = i32(floor(xymin.x * SX));
|
||||||
var x1 = i32(floor(xymax.x * SX) + 1.0);
|
var x1 = i32(floor(xymax.x * SX) + 1.0);
|
||||||
|
@ -200,7 +201,7 @@ fn main(
|
||||||
for (var y = y0; y < y1; y += 1) {
|
for (var y = y0; y < y1; y += 1) {
|
||||||
let tile_y0 = f32(y) * f32(TILE_HEIGHT);
|
let tile_y0 = f32(y) * f32(TILE_HEIGHT);
|
||||||
let xbackdrop = max(xray + 1, bbox.x);
|
let xbackdrop = max(xray + 1, bbox.x);
|
||||||
if xymin.y < tile_y0 && xbackdrop < bbox.z {
|
if !is_stroke && xymin.y < tile_y0 && xbackdrop < bbox.z {
|
||||||
let backdrop = select(-1, 1, dp.y < 0.0);
|
let backdrop = select(-1, 1, dp.y < 0.0);
|
||||||
let tile_ix = base + xbackdrop;
|
let tile_ix = base + xbackdrop;
|
||||||
atomicAdd(&tiles[tile_ix].backdrop, backdrop);
|
atomicAdd(&tiles[tile_ix].backdrop, backdrop);
|
||||||
|
@ -226,7 +227,9 @@ fn main(
|
||||||
let old = atomicExchange(&tiles[tile_ix].segments, seg_ix);
|
let old = atomicExchange(&tiles[tile_ix].segments, seg_ix);
|
||||||
tile_seg.origin = lp0;
|
tile_seg.origin = lp0;
|
||||||
tile_seg.delta = dp;
|
tile_seg.delta = dp;
|
||||||
var y_edge = mix(lp0.y, lp1.y, (tile_x0 - lp0.x) * recip_dx);
|
var y_edge = 0.0;
|
||||||
|
if !is_stroke {
|
||||||
|
y_edge = mix(lp0.y, lp1.y, (tile_x0 - lp0.x) * recip_dx);
|
||||||
if xymin.x < tile_x0 {
|
if xymin.x < tile_x0 {
|
||||||
let p = vec2(tile_x0, y_edge);
|
let p = vec2(tile_x0, y_edge);
|
||||||
if dp.x < 0.0 {
|
if dp.x < 0.0 {
|
||||||
|
@ -242,6 +245,7 @@ fn main(
|
||||||
if x <= min_xray || max_xray < x {
|
if x <= min_xray || max_xray < x {
|
||||||
y_edge = 1e9;
|
y_edge = 1e9;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
tile_seg.y_edge = y_edge;
|
tile_seg.y_edge = y_edge;
|
||||||
tile_seg.next = old;
|
tile_seg.next = old;
|
||||||
segments[seg_ix] = tile_seg;
|
segments[seg_ix] = tile_seg;
|
||||||
|
|
|
@ -129,9 +129,8 @@ fn main(
|
||||||
var tag_byte = (tag_word >> shift) & 0xffu;
|
var tag_byte = (tag_word >> shift) & 0xffu;
|
||||||
|
|
||||||
let out = &path_bboxes[tm.path_ix];
|
let out = &path_bboxes[tm.path_ix];
|
||||||
var linewidth: f32;
|
let linewidth = bitcast<f32>(scene[config.linewidth_base + tm.linewidth_ix]);
|
||||||
if (tag_byte & PATH_TAG_PATH) != 0u {
|
if (tag_byte & PATH_TAG_PATH) != 0u {
|
||||||
linewidth = bitcast<f32>(scene[config.linewidth_base + tm.linewidth_ix]);
|
|
||||||
(*out).linewidth = linewidth;
|
(*out).linewidth = linewidth;
|
||||||
(*out).trans_ix = tm.trans_ix;
|
(*out).trans_ix = tm.trans_ix;
|
||||||
}
|
}
|
||||||
|
@ -182,14 +181,16 @@ fn main(
|
||||||
p1 = mix(p1, p0, 1.0 / 3.0);
|
p1 = mix(p1, p0, 1.0 / 3.0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
var stroke = vec2<f32>(0.0, 0.0);
|
||||||
if linewidth >= 0.0 {
|
if linewidth >= 0.0 {
|
||||||
// See https://www.iquilezles.org/www/articles/ellipses/ellipses.htm
|
// See https://www.iquilezles.org/www/articles/ellipses/ellipses.htm
|
||||||
// This is the correct bounding box, but we're not handling rendering
|
// This is the correct bounding box, but we're not handling rendering
|
||||||
// in the isotropic case, so it may mismatch.
|
// in the isotropic case, so it may mismatch.
|
||||||
let stroke = 0.5 * linewidth * vec2(length(transform.matrx.xz), length(transform.matrx.yw));
|
stroke = 0.5 * linewidth * vec2(length(transform.matrx.xz), length(transform.matrx.yw));
|
||||||
bbox += vec4(-stroke, stroke);
|
bbox += vec4(-stroke, stroke);
|
||||||
}
|
}
|
||||||
cubics[global_id.x] = Cubic(p0, p1, p2, p3, tm.path_ix, 0u);
|
let flags = u32(linewidth >= 0.0);
|
||||||
|
cubics[global_id.x] = Cubic(p0, p1, p2, p3, stroke, tm.path_ix, flags);
|
||||||
// Update bounding box using atomics only. Computing a monoid is a
|
// Update bounding box using atomics only. Computing a monoid is a
|
||||||
// potential future optimization.
|
// potential future optimization.
|
||||||
if bbox.z > bbox.x || bbox.w > bbox.y {
|
if bbox.z > bbox.x || bbox.w > bbox.y {
|
||||||
|
|
|
@ -5,7 +5,9 @@ struct Cubic {
|
||||||
p1: vec2<f32>,
|
p1: vec2<f32>,
|
||||||
p2: vec2<f32>,
|
p2: vec2<f32>,
|
||||||
p3: vec2<f32>,
|
p3: vec2<f32>,
|
||||||
|
stroke: vec2<f32>,
|
||||||
path_ix: u32,
|
path_ix: u32,
|
||||||
// Needed?
|
flags: u32,
|
||||||
padding: u32,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let CUBIC_IS_STROKE = 1u;
|
||||||
|
|
|
@ -11,7 +11,7 @@ use crate::{
|
||||||
const TAG_MONOID_SIZE: u64 = 12;
|
const TAG_MONOID_SIZE: u64 = 12;
|
||||||
const TAG_MONOID_FULL_SIZE: u64 = 20;
|
const TAG_MONOID_FULL_SIZE: u64 = 20;
|
||||||
const PATH_BBOX_SIZE: u64 = 24;
|
const PATH_BBOX_SIZE: u64 = 24;
|
||||||
const CUBIC_SIZE: u64 = 40;
|
const CUBIC_SIZE: u64 = 48;
|
||||||
const DRAWMONOID_SIZE: u64 = 16;
|
const DRAWMONOID_SIZE: u64 = 16;
|
||||||
const MAX_DRAWINFO_SIZE: u64 = 44;
|
const MAX_DRAWINFO_SIZE: u64 = 44;
|
||||||
const CLIP_BIC_SIZE: u64 = 8;
|
const CLIP_BIC_SIZE: u64 = 8;
|
||||||
|
|
Loading…
Add table
Reference in a new issue