Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Transform viewport rectangles if the current matrix can losslessly transform them. #278

Merged
merged 1 commit into from May 26, 2016
Merged
Changes from all commits
Commits
File filter...
Filter file types
Jump to…
Jump to file
Failed to load files.

Always

Just for now

Transform viewport rectangles if the current matrix can losslessly

transform them.

This allows our current code to support a subset of nested clips and
transforms without actually having a full-blown clip/transform stack.

Closes servo/servo#11358.
  • Loading branch information
pcwalton committed May 26, 2016
commit ce3ddd70584889714ffabde9970d18f9be2487f1
@@ -1062,29 +1062,37 @@ impl Frame {
}
(ScrollPolicy::Scrollable, Some(scroll_layer_id)) => {
debug_assert!(!self.layers.contains_key(&scroll_layer_id));
let viewport_rect = match scroll_layer_id.info {
let (viewport_rect, viewport_transform) = match scroll_layer_id.info {
ScrollLayerInfo::Scrollable(index) if index > 0 => {
let mut stacking_context_rect =
Rect::new(parent_info.offset_from_origin,
stacking_context.bounds.size);

transform.transform_rect(&stacking_context_rect)
.intersection(&parent_info.viewport_rect)
.unwrap_or(Rect::new(Point2D::new(0.0, 0.0),
Size2D::new(0.0, 0.0)))
(parent_info.viewport_rect
.intersection(&stacking_context_rect)
.unwrap_or(Rect::new(Point2D::new(0.0, 0.0),
Size2D::new(0.0, 0.0))),
Matrix4D::identity())
}
_ if transform.can_losslessly_transform_a_2d_rect() => {
// FIXME(pcwalton): This is pretty much just a hack for
// browser.html to stave off `viewport_rect` becoming a full stack
// of matrices and clipping regions as long as we can.
(transform.transform_rect(&parent_info.viewport_rect),
Matrix4D::identity())
}
_ => parent_info.viewport_rect,
_ => (parent_info.viewport_rect, transform),
};
let layer = Layer::new(parent_info.offset_from_origin,
stacking_context.overflow.size,
&viewport_rect,
&info.viewport_transform,
&viewport_transform,
transform,
parent_info.pipeline_id);
if parent_info.actual_scroll_layer_id != scroll_layer_id {
self.layers.get_mut(&parent_info.actual_scroll_layer_id).unwrap().add_child(scroll_layer_id);
}
self.layers.insert(scroll_layer_id, layer);
info.viewport_rect = viewport_rect;
info.default_scroll_layer_id = scroll_layer_id;
info.actual_scroll_layer_id = scroll_layer_id;
info.offset_from_current_layer = Point2D::zero();
@@ -1418,3 +1426,19 @@ impl Frame {
}
}
}

trait Matrix4DUtils {
/// Returns true if this matrix transforms an axis-aligned 2D rectangle to another axis-aligned
/// 2D rectangle.
///
/// This is used as part of `flatten()` above, as essentially a hack for browser.html.
fn can_losslessly_transform_a_2d_rect(&self) -> bool;
}

impl Matrix4DUtils for Matrix4D<f32> {
fn can_losslessly_transform_a_2d_rect(&self) -> bool {
self.m31 == 0.0 && self.m32 == 0.0 && self.m33 == 1.0 && self.m34 == 0.0 &&
self.m43 == 0.0 && self.m44 == 1.0
}
}

@@ -0,0 +1,12 @@
<style>
body {
background: green;
}
iframe {
width: 100vw;
height: 100vh;
border: 10px solid red;
transform: scale(.5);
}
</style>
<iframe src="data:text/html,<body style='overflow: hidden; overflow-y: auto;'><div style='background: blue; height: 2000px'>"></iframe>
ProTip! Use n and p to navigate between commits in a pull request.
You can’t perform that action at this time.