1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
|
use markup5ever::{expanded_name, local_name, namespace_url, ns};
use crate::document::AcquiredNodes;
use crate::drawing_ctx::DrawingCtx;
use crate::element::{ElementResult, SetAttributes};
use crate::node::Node;
use crate::parsers::ParseValue;
use crate::properties::ColorInterpolationFilters;
use crate::rect::IRect;
use crate::session::Session;
use crate::xml::Attributes;
use super::bounds::BoundsBuilder;
use super::context::{FilterContext, FilterOutput};
use super::{
FilterEffect, FilterError, FilterResolveError, Input, Primitive, PrimitiveParams,
ResolvedPrimitive,
};
/// The `feOffset` filter primitive.
#[derive(Default)]
pub struct FeOffset {
base: Primitive,
params: Offset,
}
/// Resolved `feOffset` primitive for rendering.
#[derive(Clone, Default)]
pub struct Offset {
pub in1: Input,
pub dx: f64,
pub dy: f64,
}
impl SetAttributes for FeOffset {
fn set_attributes(&mut self, attrs: &Attributes, session: &Session) -> ElementResult {
self.params.in1 = self.base.parse_one_input(attrs, session)?;
for (attr, value) in attrs.iter() {
match attr.expanded() {
expanded_name!("", "dx") => self.params.dx = attr.parse(value)?,
expanded_name!("", "dy") => self.params.dy = attr.parse(value)?,
_ => (),
}
}
Ok(())
}
}
impl Offset {
pub fn render(
&self,
bounds_builder: BoundsBuilder,
ctx: &FilterContext,
acquired_nodes: &mut AcquiredNodes<'_>,
draw_ctx: &mut DrawingCtx,
) -> Result<FilterOutput, FilterError> {
// https://www.w3.org/TR/filter-effects/#ColorInterpolationFiltersProperty
//
// "Note: The color-interpolation-filters property just has an
// effect on filter operations. Therefore, it has no effect on
// filter primitives like feOffset"
//
// This is why we pass Auto here.
let input_1 = ctx.get_input(
acquired_nodes,
draw_ctx,
&self.in1,
ColorInterpolationFilters::Auto,
)?;
let bounds: IRect = bounds_builder
.add_input(&input_1)
.compute(ctx)
.clipped
.into();
let (dx, dy) = ctx.paffine().transform_distance(self.dx, self.dy);
let surface = input_1.surface().offset(bounds, dx, dy)?;
Ok(FilterOutput { surface, bounds })
}
}
impl FilterEffect for FeOffset {
fn resolve(
&self,
_acquired_nodes: &mut AcquiredNodes<'_>,
_node: &Node,
) -> Result<ResolvedPrimitive, FilterResolveError> {
Ok(ResolvedPrimitive {
primitive: self.base.clone(),
params: PrimitiveParams::Offset(self.params.clone()),
})
}
}
|