use api::BorderRadius;
use api::units::*;
use euclid::{Point2D, Rect, Box2D, Size2D, Vector2D, point2};
use euclid::{default, Transform2D, Transform3D, Scale};
use malloc_size_of::{MallocShallowSizeOf, MallocSizeOf, MallocSizeOfOps};
use plane_split::{Clipper, Polygon};
use std::{i32, f32, fmt, ptr};
use std::borrow::Cow;
use std::num::NonZeroUsize;
use std::os::raw::c_void;
use std::sync::Arc;
use std::mem::replace;
const NEARLY_ZERO: f32 = 1.0 / 4096.0;
pub struct Allocation<'a, T: 'a> {
vec: &'a mut Vec<T>,
index: usize,
}
impl<'a, T> Allocation<'a, T> {
#[inline(always)]
pub fn init(self, value: T) -> usize {
unsafe {
ptr::write(self.vec.as_mut_ptr().add(self.index), value);
self.vec.set_len(self.index + 1);
}
self.index
}
}
pub enum VecEntry<'a, T: 'a> {
Vacant(Allocation<'a, T>),
Occupied(&'a mut T),
}
impl<'a, T> VecEntry<'a, T> {
#[inline(always)]
pub fn set(self, value: T) {
match self {
VecEntry::Vacant(alloc) => { alloc.init(value); }
VecEntry::Occupied(slot) => { *slot = value; }
}
}
}
pub trait VecHelper<T> {
fn alloc(&mut self) -> Allocation<T>;
fn entry(&mut self, index: usize) -> VecEntry<T>;
fn take(&mut self) -> Self;
fn cleared(self) -> Self;
fn take_and_preallocate(&mut self) -> Self;
}
impl<T> VecHelper<T> for Vec<T> {
fn alloc(&mut self) -> Allocation<T> {
let index = self.len();
if self.capacity() == index {
self.reserve(1);
}
Allocation {
vec: self,
index,
}
}
fn entry(&mut self, index: usize) -> VecEntry<T> {
if index < self.len() {
VecEntry::Occupied(unsafe {
self.get_unchecked_mut(index)
})
} else {
assert_eq!(index, self.len());
VecEntry::Vacant(self.alloc())
}
}
fn take(&mut self) -> Self {
replace(self, Vec::new())
}
fn cleared(mut self) -> Self {
self.clear();
self
}
fn take_and_preallocate(&mut self) -> Self {
let len = self.len();
if len == 0 {
self.clear();
return Vec::new();
}
replace(self, Vec::with_capacity(len + 8))
}
}
#[derive(Debug, Clone, Copy, MallocSizeOf)]
#[cfg_attr(feature = "capture", derive(Serialize))]
#[cfg_attr(feature = "replay", derive(Deserialize))]
pub struct ScaleOffset {
pub scale: default::Vector2D<f32>,
pub offset: default::Vector2D<f32>,
}
impl ScaleOffset {
pub fn identity() -> Self {
ScaleOffset {
scale: Vector2D::new(1.0, 1.0),
offset: Vector2D::zero(),
}
}
pub fn from_transform<F, T>(
m: &Transform3D<f32, F, T>,
) -> Option<ScaleOffset> {
if m.m12.abs() > NEARLY_ZERO ||
m.m13.abs() > NEARLY_ZERO ||
m.m14.abs() > NEARLY_ZERO ||
m.m21.abs() > NEARLY_ZERO ||
m.m23.abs() > NEARLY_ZERO ||
m.m24.abs() > NEARLY_ZERO ||
m.m31.abs() > NEARLY_ZERO ||
m.m32.abs() > NEARLY_ZERO ||
(m.m33 - 1.0).abs() > NEARLY_ZERO ||
m.m34.abs() > NEARLY_ZERO ||
m.m43.abs() > NEARLY_ZERO ||
(m.m44 - 1.0).abs() > NEARLY_ZERO {
return None;
}
Some(ScaleOffset {
scale: Vector2D::new(m.m11, m.m22),
offset: Vector2D::new(m.m41, m.m42),
})
}
pub fn from_offset(offset: default::Vector2D<f32>) -> Self {
ScaleOffset {
scale: Vector2D::new(1.0, 1.0),
offset,
}
}
pub fn from_scale(scale: default::Vector2D<f32>) -> Self {
ScaleOffset {
scale,
offset: Vector2D::new(0.0, 0.0),
}
}
pub fn inverse(&self) -> Self {
ScaleOffset {
scale: Vector2D::new(
1.0 / self.scale.x,
1.0 / self.scale.y,
),
offset: Vector2D::new(
-self.offset.x / self.scale.x,
-self.offset.y / self.scale.y,
),
}
}
pub fn offset(&self, offset: default::Vector2D<f32>) -> Self {
self.accumulate(
&ScaleOffset {
scale: Vector2D::new(1.0, 1.0),
offset,
}
)
}
pub fn scale(&self, scale: f32) -> Self {
self.accumulate(
&ScaleOffset {
scale: Vector2D::new(scale, scale),
offset: Vector2D::zero(),
}
)
}
pub fn accumulate(&self, other: &ScaleOffset) -> Self {
ScaleOffset {
scale: Vector2D::new(
self.scale.x * other.scale.x,
self.scale.y * other.scale.y,
),
offset: Vector2D::new(
self.offset.x + self.scale.x * other.offset.x,
self.offset.y + self.scale.y * other.offset.y,
),
}
}
pub fn map_rect<F, T>(&self, rect: &Box2D<f32, F>) -> Box2D<f32, T> {
let w = rect.width().max(0.0);
let h = rect.height().max(0.0);
let mut x0 = rect.min.x * self.scale.x + self.offset.x;
let mut y0 = rect.min.y * self.scale.y + self.offset.y;
let mut sx = w * self.scale.x;
let mut sy = h * self.scale.y;
if self.scale.x < 0.0 {
x0 += sx;
sx = -sx;
}
if self.scale.y < 0.0 {
y0 += sy;
sy = -sy;
}
Box2D::from_origin_and_size(
Point2D::new(x0, y0),
Size2D::new(sx, sy),
)
}
pub fn unmap_rect<F, T>(&self, rect: &Box2D<f32, F>) -> Box2D<f32, T> {
let w = rect.width().max(0.0);
let h = rect.height().max(0.0);
let mut x0 = (rect.min.x - self.offset.x) / self.scale.x;
let mut y0 = (rect.min.y - self.offset.y) / self.scale.y;
let mut sx = w / self.scale.x;
let mut sy = h / self.scale.y;
if self.scale.x < 0.0 {
x0 += sx;
sx = -sx;
}
if self.scale.y < 0.0 {
y0 += sy;
sy = -sy;
}
Box2D::from_origin_and_size(
Point2D::new(x0, y0),
Size2D::new(sx, sy),
)
}
pub fn map_vector<F, T>(&self, vector: &Vector2D<f32, F>) -> Vector2D<f32, T> {
Vector2D::new(
vector.x * self.scale.x,
vector.y * self.scale.y,
)
}
pub fn unmap_vector<F, T>(&self, vector: &Vector2D<f32, F>) -> Vector2D<f32, T> {
Vector2D::new(
vector.x / self.scale.x,
vector.y / self.scale.y,
)
}
pub fn map_point<F, T>(&self, point: &Point2D<f32, F>) -> Point2D<f32, T> {
Point2D::new(
point.x * self.scale.x + self.offset.x,
point.y * self.scale.y + self.offset.y,
)
}
pub fn unmap_point<F, T>(&self, point: &Point2D<f32, F>) -> Point2D<f32, T> {
Point2D::new(
(point.x - self.offset.x) / self.scale.x,
(point.y - self.offset.y) / self.scale.y,
)
}
pub fn to_transform<F, T>(&self) -> Transform3D<f32, F, T> {
Transform3D::new(
self.scale.x,
0.0,
0.0,
0.0,
0.0,
self.scale.y,
0.0,
0.0,
0.0,
0.0,
1.0,
0.0,
self.offset.x,
self.offset.y,
0.0,
1.0,
)
}
}
pub trait MatrixHelpers<Src, Dst> {
fn preserves_2d_axis_alignment(&self) -> bool;
fn has_perspective_component(&self) -> bool;
fn has_2d_inverse(&self) -> bool;
fn exceeds_2d_scale(&self, limit: f64) -> bool;
fn inverse_project(&self, target: &Point2D<f32, Dst>) -> Option<Point2D<f32, Src>>;
fn inverse_rect_footprint(&self, rect: &Box2D<f32, Dst>) -> Option<Box2D<f32, Src>>;
fn transform_kind(&self) -> TransformedRectKind;
fn is_simple_translation(&self) -> bool;
fn is_simple_2d_translation(&self) -> bool;
fn is_2d_scale_translation(&self) -> bool;
fn determinant_2d(&self) -> f32;
fn inverse_project_2d_origin(&self) -> Option<Point2D<f32, Src>>;
fn flatten_z_output(&mut self);
fn cast_unit<NewSrc, NewDst>(&self) -> Transform3D<f32, NewSrc, NewDst>;
}
impl<Src, Dst> MatrixHelpers<Src, Dst> for Transform3D<f32, Src, Dst> {
fn preserves_2d_axis_alignment(&self) -> bool {
if self.m14 != 0.0 || self.m24 != 0.0 {
return false;
}
let mut col0 = 0;
let mut col1 = 0;
let mut row0 = 0;
let mut row1 = 0;
if self.m11.abs() > NEARLY_ZERO {
col0 += 1;
row0 += 1;
}
if self.m12.abs() > NEARLY_ZERO {
col1 += 1;
row0 += 1;
}
if self.m21.abs() > NEARLY_ZERO {
col0 += 1;
row1 += 1;
}
if self.m22.abs() > NEARLY_ZERO {
col1 += 1;
row1 += 1;
}
col0 < 2 && col1 < 2 && row0 < 2 && row1 < 2
}
fn has_perspective_component(&self) -> bool {
self.m14.abs() > NEARLY_ZERO ||
self.m24.abs() > NEARLY_ZERO ||
self.m34.abs() > NEARLY_ZERO ||
(self.m44 - 1.0).abs() > NEARLY_ZERO
}
fn has_2d_inverse(&self) -> bool {
self.determinant_2d() != 0.0
}
fn exceeds_2d_scale(&self, limit: f64) -> bool {
let limit2 = (limit * limit) as f32;
self.m11 * self.m11 + self.m12 * self.m12 > limit2 ||
self.m21 * self.m21 + self.m22 * self.m22 > limit2
}
fn inverse_project(&self, target: &Point2D<f32, Dst>) -> Option<Point2D<f32, Src>> {
let m = Transform2D::<f32, Src, Dst>::new(
self.m11 - target.x * self.m14, self.m12 - target.y * self.m14,
self.m21 - target.x * self.m24, self.m22 - target.y * self.m24,
self.m41 - target.x * self.m44, self.m42 - target.y * self.m44,
);
let inv = m.inverse()?;
if inv.m31 * self.m14 + inv.m32 * self.m24 + self.m44 > 0.0 {
Some(Point2D::new(inv.m31, inv.m32))
} else {
None
}
}
fn inverse_rect_footprint(&self, rect: &Box2D<f32, Dst>) -> Option<Box2D<f32, Src>> {
Some(Box2D::from_points(&[
self.inverse_project(&rect.top_left())?,
self.inverse_project(&rect.top_right())?,
self.inverse_project(&rect.bottom_left())?,
self.inverse_project(&rect.bottom_right())?,
]))
}
fn transform_kind(&self) -> TransformedRectKind {
if self.preserves_2d_axis_alignment() {
TransformedRectKind::AxisAligned
} else {
TransformedRectKind::Complex
}
}
fn is_simple_translation(&self) -> bool {
if (self.m11 - 1.0).abs() > NEARLY_ZERO ||
(self.m22 - 1.0).abs() > NEARLY_ZERO ||
(self.m33 - 1.0).abs() > NEARLY_ZERO ||
(self.m44 - 1.0).abs() > NEARLY_ZERO {
return false;
}
self.m12.abs() < NEARLY_ZERO && self.m13.abs() < NEARLY_ZERO &&
self.m14.abs() < NEARLY_ZERO && self.m21.abs() < NEARLY_ZERO &&
self.m23.abs() < NEARLY_ZERO && self.m24.abs() < NEARLY_ZERO &&
self.m31.abs() < NEARLY_ZERO && self.m32.abs() < NEARLY_ZERO &&
self.m34.abs() < NEARLY_ZERO
}
fn is_simple_2d_translation(&self) -> bool {
if !self.is_simple_translation() {
return false;
}
self.m43.abs() < NEARLY_ZERO
}
fn is_2d_scale_translation(&self) -> bool {
(self.m33 - 1.0).abs() < NEARLY_ZERO &&
(self.m44 - 1.0).abs() < NEARLY_ZERO &&
self.m12.abs() < NEARLY_ZERO && self.m13.abs() < NEARLY_ZERO && self.m14.abs() < NEARLY_ZERO &&
self.m21.abs() < NEARLY_ZERO && self.m23.abs() < NEARLY_ZERO && self.m24.abs() < NEARLY_ZERO &&
self.m31.abs() < NEARLY_ZERO && self.m32.abs() < NEARLY_ZERO && self.m34.abs() < NEARLY_ZERO &&
self.m43.abs() < NEARLY_ZERO
}
fn determinant_2d(&self) -> f32 {
self.m11 * self.m22 - self.m12 * self.m21
}
fn inverse_project_2d_origin(&self) -> Option<Point2D<f32, Src>> {
let det = self.determinant_2d();
if det != 0.0 {
let x = (self.m21 * self.m42 - self.m41 * self.m22) / det;
let y = (self.m12 * self.m41 - self.m11 * self.m42) / det;
Some(Point2D::new(x, y))
} else {
None
}
}
fn flatten_z_output(&mut self) {
self.m13 = 0.0;
self.m23 = 0.0;
self.m33 = 1.0;
self.m43 = 0.0;
}
fn cast_unit<NewSrc, NewDst>(&self) -> Transform3D<f32, NewSrc, NewDst> {
Transform3D::new(
self.m11, self.m12, self.m13, self.m14,
self.m21, self.m22, self.m23, self.m24,
self.m31, self.m32, self.m33, self.m34,
self.m41, self.m42, self.m43, self.m44,
)
}
}
pub trait PointHelpers<U>
where
Self: Sized,
{
fn snap(&self) -> Self;
}
impl<U> PointHelpers<U> for Point2D<f32, U> {
fn snap(&self) -> Self {
Point2D::new(
(self.x + 0.5).floor(),
(self.y + 0.5).floor(),
)
}
}
pub trait RectHelpers<U>
where
Self: Sized,
{
fn from_floats(x0: f32, y0: f32, x1: f32, y1: f32) -> Self;
fn snap(&self) -> Self;
}
impl<U> RectHelpers<U> for Rect<f32, U> {
fn from_floats(x0: f32, y0: f32, x1: f32, y1: f32) -> Self {
Rect::new(
Point2D::new(x0, y0),
Size2D::new(x1 - x0, y1 - y0),
)
}
fn snap(&self) -> Self {
let origin = Point2D::new(
(self.origin.x + 0.5).floor(),
(self.origin.y + 0.5).floor(),
);
Rect::new(
origin,
Size2D::new(
(self.origin.x + self.size.width + 0.5).floor() - origin.x,
(self.origin.y + self.size.height + 0.5).floor() - origin.y,
),
)
}
}
impl<U> RectHelpers<U> for Box2D<f32, U> {
fn from_floats(x0: f32, y0: f32, x1: f32, y1: f32) -> Self {
Box2D {
min: Point2D::new(x0, y0),
max: Point2D::new(x1, y1),
}
}
fn snap(&self) -> Self {
self.round()
}
}
pub trait VectorHelpers<U>
where
Self: Sized,
{
fn snap(&self) -> Self;
}
impl<U> VectorHelpers<U> for Vector2D<f32, U> {
fn snap(&self) -> Self {
Vector2D::new(
(self.x + 0.5).floor(),
(self.y + 0.5).floor(),
)
}
}
pub fn lerp(a: f32, b: f32, t: f32) -> f32 {
(b - a) * t + a
}
#[repr(u32)]
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)]
#[cfg_attr(feature = "capture", derive(Serialize))]
#[cfg_attr(feature = "replay", derive(Deserialize))]
pub enum TransformedRectKind {
AxisAligned = 0,
Complex = 1,
}
#[inline(always)]
pub fn pack_as_float(value: u32) -> f32 {
value as f32 + 0.5
}
#[inline]
fn extract_inner_rect_impl<U>(
rect: &Box2D<f32, U>,
radii: &BorderRadius,
k: f32,
) -> Option<Box2D<f32, U>> {
let xl = (k * radii.top_left.width.max(radii.bottom_left.width)).ceil();
let xr = (rect.width() - k * radii.top_right.width.max(radii.bottom_right.width)).floor();
let yt = (k * radii.top_left.height.max(radii.top_right.height)).ceil();
let yb =
(rect.height() - k * radii.bottom_left.height.max(radii.bottom_right.height)).floor();
if xl <= xr && yt <= yb {
Some(Box2D::from_origin_and_size(
Point2D::new(rect.min.x + xl, rect.min.y + yt),
Size2D::new(xr - xl, yb - yt),
))
} else {
None
}
}
pub fn extract_inner_rect_safe<U>(
rect: &Box2D<f32, U>,
radii: &BorderRadius,
) -> Option<Box2D<f32, U>> {
extract_inner_rect_impl(rect, radii, 1.0)
}
#[cfg(test)]
use euclid::vec3;
#[cfg(test)]
pub mod test {
use super::*;
use euclid::default::{Point2D, Size2D, Transform3D};
use euclid::{Angle, approxeq::ApproxEq};
use std::f32::consts::PI;
use crate::clip::{is_left_of_line, polygon_contains_point};
use crate::prim_store::PolygonKey;
use api::FillRule;
#[test]
fn inverse_project() {
let m0 = Transform3D::identity();
let p0 = Point2D::new(1.0, 2.0);
assert_eq!(m0.inverse_project(&p0), Some(p0));
let m1 = Transform3D::rotation(0.0, 1.0, 0.0, Angle::radians(-PI / 3.0));
assert_eq!(m1.inverse_project(&p0), Some(Point2D::new(2.0, 2.0)));
}
#[test]
fn inverse_project_footprint() {
let m = Transform3D::new(
0.477499992, 0.135000005, -1.0, 0.000624999986,
-0.642787635, 0.766044438, 0.0, 0.0,
0.766044438, 0.642787635, 0.0, 0.0,
1137.10986, 113.71286, 402.0, 0.748749971,
);
let r = Box2D::from_size(Size2D::new(804.0, 804.0));
{
let points = &[
r.top_left(),
r.top_right(),
r.bottom_left(),
r.bottom_right(),
];
let mi = m.inverse().unwrap();
println!("Points:");
for p in points {
let pp = m.transform_point2d_homogeneous(*p);
let p3 = pp.to_point3d().unwrap();
let pi = mi.transform_point3d_homogeneous(p3);
let px = pi.to_point2d().unwrap();
let py = m.inverse_project(&pp.to_point2d().unwrap()).unwrap();
println!("\t{:?} -> {:?} -> {:?} -> ({:?} -> {:?}, {:?})", p, pp, p3, pi, px, py);
assert!(px.approx_eq_eps(p, &Point2D::new(0.001, 0.001)));
assert!(py.approx_eq_eps(p, &Point2D::new(0.001, 0.001)));
}
}
let rp = project_rect(&m, &r, &Box2D::from_size(Size2D::new(1000.0, 1000.0))).unwrap();
println!("Projected {:?}", rp);
assert_eq!(m.inverse_project(&rp.min), None);
if let Some(ri) = m.inverse_rect_footprint(&rp) {
assert!(ri.contains_box(&r), "Inverse {:?}", ri);
}
}
fn validate_convert(xref: &LayoutTransform) {
let so = ScaleOffset::from_transform(xref).unwrap();
let xf = so.to_transform();
assert!(xref.approx_eq(&xf));
}
#[test]
fn negative_scale_map_unmap() {
let xref = LayoutTransform::scale(1.0, -1.0, 1.0)
.pre_translate(LayoutVector3D::new(124.0, 38.0, 0.0));
let so = ScaleOffset::from_transform(&xref).unwrap();
let local_rect = Box2D {
min: LayoutPoint::new(50.0, -100.0),
max: LayoutPoint::new(250.0, 300.0),
};
let mapped_rect = so.map_rect::<LayoutPixel, DevicePixel>(&local_rect);
let xf_rect = project_rect(
&xref,
&local_rect,
&LayoutRect::max_rect(),
).unwrap();
assert!(mapped_rect.min.x.approx_eq(&xf_rect.min.x));
assert!(mapped_rect.min.y.approx_eq(&xf_rect.min.y));
assert!(mapped_rect.max.x.approx_eq(&xf_rect.max.x));
assert!(mapped_rect.max.y.approx_eq(&xf_rect.max.y));
let unmapped_rect = so.unmap_rect::<DevicePixel, LayoutPixel>(&mapped_rect);
assert!(unmapped_rect.min.x.approx_eq(&local_rect.min.x));
assert!(unmapped_rect.min.y.approx_eq(&local_rect.min.y));
assert!(unmapped_rect.max.x.approx_eq(&local_rect.max.x));
assert!(unmapped_rect.max.y.approx_eq(&local_rect.max.y));
}
#[test]
fn scale_offset_convert() {
let xref = LayoutTransform::translation(130.0, 200.0, 0.0);
validate_convert(&xref);
let xref = LayoutTransform::scale(13.0, 8.0, 1.0);
validate_convert(&xref);
let xref = LayoutTransform::scale(0.5, 0.5, 1.0)
.pre_translate(LayoutVector3D::new(124.0, 38.0, 0.0));
validate_convert(&xref);
let xref = LayoutTransform::scale(30.0, 11.0, 1.0)
.then_translate(vec3(50.0, 240.0, 0.0));
validate_convert(&xref);
}
fn validate_inverse(xref: &LayoutTransform) {
let s0 = ScaleOffset::from_transform(xref).unwrap();
let s1 = s0.inverse().accumulate(&s0);
assert!((s1.scale.x - 1.0).abs() < NEARLY_ZERO &&
(s1.scale.y - 1.0).abs() < NEARLY_ZERO &&
s1.offset.x.abs() < NEARLY_ZERO &&
s1.offset.y.abs() < NEARLY_ZERO,
"{:?}",
s1);
}
#[test]
fn scale_offset_inverse() {
let xref = LayoutTransform::translation(130.0, 200.0, 0.0);
validate_inverse(&xref);
let xref = LayoutTransform::scale(13.0, 8.0, 1.0);
validate_inverse(&xref);
let xref = LayoutTransform::translation(124.0, 38.0, 0.0).
then_scale(0.5, 0.5, 1.0);
validate_inverse(&xref);
let xref = LayoutTransform::scale(30.0, 11.0, 1.0)
.then_translate(vec3(50.0, 240.0, 0.0));
validate_inverse(&xref);
}
fn validate_accumulate(x0: &LayoutTransform, x1: &LayoutTransform) {
let x = x1.then(&x0);
let s0 = ScaleOffset::from_transform(x0).unwrap();
let s1 = ScaleOffset::from_transform(x1).unwrap();
let s = s0.accumulate(&s1).to_transform();
assert!(x.approx_eq(&s), "{:?}\n{:?}", x, s);
}
#[test]
fn scale_offset_accumulate() {
let x0 = LayoutTransform::translation(130.0, 200.0, 0.0);
let x1 = LayoutTransform::scale(7.0, 3.0, 1.0);
validate_accumulate(&x0, &x1);
}
#[test]
fn inverse_project_2d_origin() {
let mut m = Transform3D::identity();
assert_eq!(m.inverse_project_2d_origin(), Some(Point2D::zero()));
m.m11 = 0.0;
assert_eq!(m.inverse_project_2d_origin(), None);
m.m21 = -2.0;
m.m22 = 0.0;
m.m12 = -0.5;
m.m41 = 1.0;
m.m42 = 0.5;
let origin = m.inverse_project_2d_origin().unwrap();
assert_eq!(origin, Point2D::new(1.0, 0.5));
assert_eq!(m.transform_point2d(origin), Some(Point2D::zero()));
}
#[test]
fn polygon_clip_is_left_of_point() {
let p0_x = 1.0;
let p0_y = -3.0;
let p1_x = -2.0;
let p1_y = 6.0;
assert!(is_left_of_line(-9.0, 0.0, p0_x, p0_y, p1_x, p1_y) > 0.0);
assert!(is_left_of_line(-1.0, 1.0, p0_x, p0_y, p1_x, p1_y) > 0.0);
assert!(is_left_of_line(1.0, -4.0, p0_x, p0_y, p1_x, p1_y) > 0.0);
assert!(is_left_of_line(-3.0, 9.0, p0_x, p0_y, p1_x, p1_y) == 0.0);
assert!(is_left_of_line(0.0, 0.0, p0_x, p0_y, p1_x, p1_y) == 0.0);
assert!(is_left_of_line(100.0, -300.0, p0_x, p0_y, p1_x, p1_y) == 0.0);
assert!(is_left_of_line(0.0, 1.0, p0_x, p0_y, p1_x, p1_y) < 0.0);
assert!(is_left_of_line(-4.0, 13.0, p0_x, p0_y, p1_x, p1_y) < 0.0);
assert!(is_left_of_line(5.0, -12.0, p0_x, p0_y, p1_x, p1_y) < 0.0);
}
#[test]
fn polygon_clip_contains_point() {
let p0 = LayoutPoint::new(4.0, 4.0);
let p1 = LayoutPoint::new(6.0, 4.0);
let p2 = LayoutPoint::new(4.0, 7.0);
let p3 = LayoutPoint::new(2.0, 1.0);
let p4 = LayoutPoint::new(8.0, 1.0);
let p5 = LayoutPoint::new(6.0, 7.0);
let poly_clockwise_nonzero = PolygonKey::new(
&[p5, p4, p3, p2, p1, p0].to_vec(), FillRule::Nonzero
);
let poly_clockwise_evenodd = PolygonKey::new(
&[p5, p4, p3, p2, p1, p0].to_vec(), FillRule::Evenodd
);
let poly_counter_clockwise_nonzero = PolygonKey::new(
&[p0, p1, p2, p3, p4, p5].to_vec(), FillRule::Nonzero
);
let poly_counter_clockwise_evenodd = PolygonKey::new(
&[p0, p1, p2, p3, p4, p5].to_vec(), FillRule::Evenodd
);
let rect = LayoutRect::from_size(LayoutSize::new(10.0, 10.0));
let p_inside_once = LayoutPoint::new(5.0, 3.0);
let p_inside_twice = LayoutPoint::new(5.0, 5.0);
let p_outside = LayoutPoint::new(9.0, 9.0);
for poly_nonzero in vec![poly_clockwise_nonzero, poly_counter_clockwise_nonzero].iter() {
assert_eq!(polygon_contains_point(&p_inside_once, &rect, &poly_nonzero), true);
assert_eq!(polygon_contains_point(&p_inside_twice, &rect, &poly_nonzero), true);
assert_eq!(polygon_contains_point(&p_outside, &rect, &poly_nonzero), false);
}
for poly_evenodd in vec![poly_clockwise_evenodd, poly_counter_clockwise_evenodd].iter() {
assert_eq!(polygon_contains_point(&p_inside_once, &rect, &poly_evenodd), true);
assert_eq!(polygon_contains_point(&p_inside_twice, &rect, &poly_evenodd), false);
assert_eq!(polygon_contains_point(&p_outside, &rect, &poly_evenodd), false);
}
}
}
pub trait MaxRect {
fn max_rect() -> Self;
}
impl MaxRect for DeviceIntRect {
fn max_rect() -> Self {
DeviceIntRect::from_origin_and_size(
DeviceIntPoint::new(i32::MIN / 2, i32::MIN / 2),
DeviceIntSize::new(i32::MAX, i32::MAX),
)
}
}
impl<U> MaxRect for Rect<f32, U> {
fn max_rect() -> Self {
const MAX_COORD: f32 = 1.0e9;
Rect::new(
Point2D::new(-MAX_COORD, -MAX_COORD),
Size2D::new(2.0 * MAX_COORD, 2.0 * MAX_COORD),
)
}
}
impl<U> MaxRect for Box2D<f32, U> {
fn max_rect() -> Self {
const MAX_COORD: f32 = 1.0e9;
Box2D::new(
Point2D::new(-MAX_COORD, -MAX_COORD),
Point2D::new(MAX_COORD, MAX_COORD),
)
}
}
#[derive(Debug, MallocSizeOf)]
pub enum FastTransform<Src, Dst> {
Offset(Vector2D<f32, Src>),
Transform {
transform: Transform3D<f32, Src, Dst>,
inverse: Option<Transform3D<f32, Dst, Src>>,
is_2d: bool,
},
}
impl<Src, Dst> Clone for FastTransform<Src, Dst> {
fn clone(&self) -> Self {
*self
}
}
impl<Src, Dst> Copy for FastTransform<Src, Dst> { }
impl<Src, Dst> FastTransform<Src, Dst> {
pub fn identity() -> Self {
FastTransform::Offset(Vector2D::zero())
}
pub fn with_vector(offset: Vector2D<f32, Src>) -> Self {
FastTransform::Offset(offset)
}
pub fn with_scale_offset(scale_offset: ScaleOffset) -> Self {
if scale_offset.scale == Vector2D::new(1.0, 1.0) {
FastTransform::Offset(Vector2D::from_untyped(scale_offset.offset))
} else {
FastTransform::Transform {
transform: scale_offset.to_transform(),
inverse: Some(scale_offset.inverse().to_transform()),
is_2d: true,
}
}
}
#[inline(always)]
pub fn with_transform(transform: Transform3D<f32, Src, Dst>) -> Self {
if transform.is_simple_2d_translation() {
return FastTransform::Offset(Vector2D::new(transform.m41, transform.m42));
}
let inverse = transform.inverse();
let is_2d = transform.is_2d();
FastTransform::Transform { transform, inverse, is_2d}
}
pub fn to_transform(&self) -> Cow<Transform3D<f32, Src, Dst>> {
match *self {
FastTransform::Offset(offset) => Cow::Owned(
Transform3D::translation(offset.x, offset.y, 0.0)
),
FastTransform::Transform { ref transform, .. } => Cow::Borrowed(transform),
}
}
#[allow(unused)]
pub fn is_identity(&self)-> bool {
match *self {
FastTransform::Offset(offset) => {
offset == Vector2D::zero()
}
FastTransform::Transform { ref transform, .. } => {
*transform == Transform3D::identity()
}
}
}
pub fn then<NewDst>(&self, other: &FastTransform<Dst, NewDst>) -> FastTransform<Src, NewDst> {
match *self {
FastTransform::Offset(offset) => match *other {
FastTransform::Offset(other_offset) => {
FastTransform::Offset(offset + other_offset * Scale::<_, _, Src>::new(1.0))
}
FastTransform::Transform { transform: ref other_transform, .. } => {
FastTransform::with_transform(
other_transform
.with_source::<Src>()
.pre_translate(offset.to_3d())
)
}
}
FastTransform::Transform { ref transform, ref inverse, is_2d } => match *other {
FastTransform::Offset(other_offset) => {
FastTransform::with_transform(
transform
.then_translate(other_offset.to_3d())
.with_destination::<NewDst>()
)
}
FastTransform::Transform { transform: ref other_transform, inverse: ref other_inverse, is_2d: other_is_2d } => {
FastTransform::Transform {
transform: transform.then(other_transform),
inverse: inverse.as_ref().and_then(|self_inv|
other_inverse.as_ref().map(|other_inv| other_inv.then(self_inv))
),
is_2d: is_2d & other_is_2d,
}
}
}
}
}
pub fn pre_transform<NewSrc>(
&self,
other: &FastTransform<NewSrc, Src>
) -> FastTransform<NewSrc, Dst> {
other.then(self)
}
pub fn pre_translate(&self, other_offset: Vector2D<f32, Src>) -> Self {
match *self {
FastTransform::Offset(offset) =>
FastTransform::Offset(offset + other_offset),
FastTransform::Transform { transform, .. } =>
FastTransform::with_transform(transform.pre_translate(other_offset.to_3d()))
}
}
pub fn then_translate(&self, other_offset: Vector2D<f32, Dst>) -> Self {
match *self {
FastTransform::Offset(offset) => {
FastTransform::Offset(offset + other_offset * Scale::<_, _, Src>::new(1.0))
}
FastTransform::Transform { ref transform, .. } => {
let transform = transform.then_translate(other_offset.to_3d());
FastTransform::with_transform(transform)
}
}
}
#[inline(always)]
pub fn is_backface_visible(&self) -> bool {
match *self {
FastTransform::Offset(..) => false,
FastTransform::Transform { inverse: None, .. } => false,
FastTransform::Transform { inverse: Some(ref inverse), .. } => inverse.m33 < 0.0,
}
}
#[inline(always)]
pub fn transform_point2d(&self, point: Point2D<f32, Src>) -> Option<Point2D<f32, Dst>> {
match *self {
FastTransform::Offset(offset) => {
let new_point = point + offset;
Some(Point2D::from_untyped(new_point.to_untyped()))
}
FastTransform::Transform { ref transform, .. } => transform.transform_point2d(point),
}
}
#[inline(always)]
pub fn inverse(&self) -> Option<FastTransform<Dst, Src>> {
match *self {
FastTransform::Offset(offset) =>
Some(FastTransform::Offset(Vector2D::new(-offset.x, -offset.y))),
FastTransform::Transform { transform, inverse: Some(inverse), is_2d, } =>
Some(FastTransform::Transform {
transform: inverse,
inverse: Some(transform),
is_2d
}),
FastTransform::Transform { inverse: None, .. } => None,
}
}
}
impl<Src, Dst> From<Transform3D<f32, Src, Dst>> for FastTransform<Src, Dst> {
fn from(transform: Transform3D<f32, Src, Dst>) -> Self {
FastTransform::with_transform(transform)
}
}
impl<Src, Dst> From<Vector2D<f32, Src>> for FastTransform<Src, Dst> {
fn from(vector: Vector2D<f32, Src>) -> Self {
FastTransform::with_vector(vector)
}
}
pub type LayoutFastTransform = FastTransform<LayoutPixel, LayoutPixel>;
pub type LayoutToWorldFastTransform = FastTransform<LayoutPixel, WorldPixel>;
pub fn project_rect<F, T>(
transform: &Transform3D<f32, F, T>,
rect: &Box2D<f32, F>,
bounds: &Box2D<f32, T>,
) -> Option<Box2D<f32, T>>
where F: fmt::Debug
{
let homogens = [
transform.transform_point2d_homogeneous(rect.top_left()),
transform.transform_point2d_homogeneous(rect.top_right()),
transform.transform_point2d_homogeneous(rect.bottom_left()),
transform.transform_point2d_homogeneous(rect.bottom_right()),
];
if homogens.iter().any(|h| h.w <= 0.0 || h.w.is_nan()) {
let mut clipper = Clipper::new();
let polygon = Polygon::from_rect(rect.to_rect(), 1);
let planes = match Clipper::<_, _, usize>::frustum_planes(
transform,
Some(bounds.to_rect()),
) {
Ok(planes) => planes,
Err(..) => return None,
};
for plane in planes {
clipper.add(plane);
}
let results = clipper.clip(polygon);
if results.is_empty() {
return None
}
Some(Box2D::from_points(results
.into_iter()
.flat_map(|poly| &poly.points)
.map(|p| {
let mut homo = transform.transform_point2d_homogeneous(p.to_2d());
homo.w = homo.w.max(0.00000001); homo.to_point2d().unwrap()
})
))
} else {
Some(Box2D::from_points(&[
homogens[0].to_point2d().unwrap(),
homogens[1].to_point2d().unwrap(),
homogens[2].to_point2d().unwrap(),
homogens[3].to_point2d().unwrap(),
]))
}
}
pub fn raster_rect_to_device_pixels(
rect: RasterRect,
device_pixel_scale: DevicePixelScale,
) -> DeviceRect {
let world_rect = rect * Scale::new(1.0);
let device_rect = world_rect * device_pixel_scale;
device_rect.round_out()
}
pub fn drain_filter<T, Filter, Action>(
vec: &mut Vec<T>,
mut filter: Filter,
mut action: Action,
)
where
Filter: FnMut(&mut T) -> bool,
Action: FnMut(T)
{
let mut i = 0;
while i != vec.len() {
if filter(&mut vec[i]) {
action(vec.remove(i));
} else {
i += 1;
}
}
}
#[derive(Debug)]
pub struct Recycler {
pub num_allocations: usize,
}
impl Recycler {
const MAX_EXTRA_CAPACITY_PERCENT: usize = 200;
const MIN_EXTRA_CAPACITY_PERCENT: usize = 20;
const MIN_VECTOR_LENGTH: usize = 16;
pub fn new() -> Self {
Recycler {
num_allocations: 0,
}
}
pub fn recycle_vec<T>(&mut self, vec: &mut Vec<T>) {
let extra_capacity = (vec.capacity() - vec.len()) * 100 / vec.len().max(Self::MIN_VECTOR_LENGTH);
if extra_capacity > Self::MAX_EXTRA_CAPACITY_PERCENT {
*vec = Vec::with_capacity(vec.len() + vec.len() * Self::MIN_EXTRA_CAPACITY_PERCENT / 100);
self.num_allocations += 1;
} else {
vec.clear();
}
}
}
#[derive(Copy, Clone, Debug)]
pub struct Preallocator {
size: usize,
}
impl Preallocator {
pub fn new(initial_size: usize) -> Self {
Preallocator {
size: initial_size,
}
}
pub fn record_vec<T>(&mut self, vec: &Vec<T>) {
let len = vec.len();
if len > self.size {
self.size = len;
} else {
self.size = (self.size + len) / 2;
}
}
pub fn preallocation_size(&self) -> usize {
(self.size + 15) & !15
}
pub fn preallocate_vec<T>(&self, vec: &mut Vec<T>) {
let len = vec.len();
let cap = self.preallocation_size();
if len < cap {
vec.reserve(cap - len);
}
}
}
impl Default for Preallocator {
fn default() -> Self {
Self::new(0)
}
}
#[cfg_attr(feature = "capture", derive(Serialize))]
#[cfg_attr(feature = "replay", derive(Deserialize))]
#[derive(Clone, Debug, Hash, PartialEq, Eq)]
pub struct PrimaryArc<T>(pub Arc<T>);
impl<T> ::std::ops::Deref for PrimaryArc<T> {
type Target = Arc<T>;
#[inline]
fn deref(&self) -> &Arc<T> {
&self.0
}
}
impl<T> MallocShallowSizeOf for PrimaryArc<T> {
fn shallow_size_of(&self, ops: &mut MallocSizeOfOps) -> usize {
unsafe {
let raw_arc_ptr: *const Arc<T> = &self.0;
let raw_ptr_ptr: *const *const c_void = raw_arc_ptr as _;
let raw_ptr = *raw_ptr_ptr;
(ops.size_of_op)(raw_ptr)
}
}
}
impl<T: MallocSizeOf> MallocSizeOf for PrimaryArc<T> {
fn size_of(&self, ops: &mut MallocSizeOfOps) -> usize {
self.shallow_size_of(ops) + (**self).size_of(ops)
}
}
pub fn scale_factors<Src, Dst>(
mat: &Transform3D<f32, Src, Dst>
) -> (f32, f32) {
let det = mat.m11 * mat.m22 - mat.m12 * mat.m21;
if det == 0.0 {
return (0.0, 0.0);
}
let det = det.abs();
let major = (mat.m11 * mat.m11 + mat.m12 * mat.m12).sqrt();
let minor = if major != 0.0 { det / major } else { 0.0 };
(major, minor)
}
pub fn clamp_to_scale_factor(val: f32, round_down: bool) -> f32 {
const SCALE_RESOLUTION: f32 = 2.0;
let val = val.abs();
let (val, inverse) = if val < 1.0 {
(1.0 / val, true)
} else {
(val, false)
};
let power = val.log2() / SCALE_RESOLUTION.log2();
let power = if (power - power.round()).abs() < 1e-5 {
power.round()
} else if inverse != round_down {
power.floor()
} else {
power.ceil()
};
let scale = SCALE_RESOLUTION.powf(power);
if inverse {
1.0 / scale
} else {
scale
}
}
pub fn round_up_to_multiple(val: usize, mul: NonZeroUsize) -> usize {
match val % mul.get() {
0 => val,
rem => val - rem + mul.get(),
}
}
#[macro_export]
macro_rules! c_str {
($lit:expr) => {
unsafe {
std::ffi::CStr::from_ptr(concat!($lit, "\0").as_ptr()
as *const std::os::raw::c_char)
}
}
}
pub fn conservative_union_rect<U>(r1: &Box2D<f32, U>, r2: &Box2D<f32, U>) -> Box2D<f32, U> {
if r1.min.y == r2.min.y && r1.max.y == r2.max.y {
if r2.min.x <= r1.max.x && r2.max.x >= r1.min.x {
let min_x = f32::min(r1.min.x, r2.min.x);
let max_x = f32::max(r1.max.x, r2.max.x);
return Box2D {
min: point2(min_x, r1.min.y),
max: point2(max_x, r1.max.y),
}
}
}
if r1.min.x == r2.min.x && r1.max.x == r2.max.x {
if r2.min.y <= r1.max.y && r2.max.y >= r1.min.y {
let min_y = f32::min(r1.min.y, r2.min.y);
let max_y = f32::max(r1.max.y, r2.max.y);
return Box2D {
min: point2(r1.min.x, min_y),
max: point2(r1.max.x, max_y),
}
}
}
if r1.area() >= r2.area() { *r1 } else {*r2 }
}
#[test]
fn test_conservative_union_rect() {
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
&LayoutRect { min: point2(4.0, 2.0), max: point2(9.0, 6.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 2.0), max: point2(9.0, 6.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(4.0, 2.0), max: point2(9.0, 6.0) },
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 2.0), max: point2(9.0, 6.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
&LayoutRect { min: point2(3.0, 2.0), max: point2(8.0, 6.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 2.0), max: point2(8.0, 6.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(5.0, 2.0), max: point2(8.0, 6.0) },
&LayoutRect { min: point2(1.0, 2.0), max: point2(6.0, 6.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 2.0), max: point2(8.0, 6.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
&LayoutRect { min: point2(6.0, 2.0), max: point2(11.0, 6.0) },
);
assert_eq!(r, LayoutRect { min: point2(6.0, 2.0), max: point2(11.0, 6.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
&LayoutRect { min: point2(-6.0, 2.0), max: point2(-5.0, 6.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
&LayoutRect { min: point2(1.0, 6.0), max: point2(4.0, 10.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 10.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 5.0), max: point2(4.0, 9.0) },
&LayoutRect { min: point2(1.0, 1.0), max: point2(4.0, 5.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 1.0), max: point2(4.0, 9.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
&LayoutRect { min: point2(1.0, 3.0), max: point2(4.0, 7.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 7.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 4.0), max: point2(4.0, 8.0) },
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 8.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
&LayoutRect { min: point2(1.0, 10.0), max: point2(4.0, 15.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 10.0), max: point2(4.0, 15.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 5.0), max: point2(4.0, 9.0) },
&LayoutRect { min: point2(1.0, 0.0), max: point2(4.0, 3.0) },
);
assert_eq!(r, LayoutRect { min: point2(1.0, 5.0), max: point2(4.0, 9.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
&LayoutRect { min: point2(0.0, 1.0), max: point2(10.0, 12.0) },
);
assert_eq!(r, LayoutRect { min: point2(0.0, 1.0), max: point2(10.0, 12.0) });
let r = conservative_union_rect(
&LayoutRect { min: point2(0.0, 1.0), max: point2(10.0, 12.0) },
&LayoutRect { min: point2(1.0, 2.0), max: point2(4.0, 6.0) },
);
assert_eq!(r, LayoutRect { min: point2(0.0, 1.0), max: point2(10.0, 12.0) });
}
pub struct WeakTable {
inner: Vec<std::sync::Weak<Vec<u8>>>
}
impl WeakTable {
pub fn new() -> WeakTable {
WeakTable { inner: Vec::new() }
}
pub fn insert(&mut self, x: std::sync::Weak<Vec<u8>>) {
if self.inner.len() == self.inner.capacity() {
self.remove_expired();
if self.inner.len() * 3 < self.inner.capacity() {
self.inner.shrink_to_fit();
} else {
self.inner.reserve(self.inner.len())
}
}
self.inner.push(x);
}
fn remove_expired(&mut self) {
self.inner.retain(|x| x.strong_count() > 0)
}
pub fn iter(&self) -> impl Iterator<Item = Arc<Vec<u8>>> + '_ {
self.inner.iter().filter_map(|x| x.upgrade())
}
}
#[test]
fn weak_table() {
let mut tbl = WeakTable::new();
let mut things = Vec::new();
let target_count = 50;
for _ in 0..target_count {
things.push(Arc::new(vec![4]));
}
for i in &things {
tbl.insert(Arc::downgrade(i))
}
assert_eq!(tbl.inner.len(), target_count);
drop(things);
assert_eq!(tbl.iter().count(), 0);
for _ in 0..target_count*2 {
tbl.insert(Arc::downgrade(&Arc::new(vec![5])))
}
assert!(tbl.inner.capacity() <= 4);
}