Commit 0c2546c6 authored by Ian Hickson's avatar Ian Hickson Committed by GitHub

Factor out some common code in PaintingContext (#10607)

...and remove some highly specialised methods now that they can just
be implemented directly by the previous callers.
parent 09eba82a
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
// found in the LICENSE file. // found in the LICENSE file.
import 'dart:developer'; import 'dart:developer';
import 'dart:ui' as ui show ImageFilter, PictureRecorder; import 'dart:ui' as ui show PictureRecorder;
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
import 'package:flutter/gestures.dart'; import 'package:flutter/gestures.dart';
...@@ -46,8 +46,8 @@ typedef void PaintingContextCallback(PaintingContext context, Offset offset); ...@@ -46,8 +46,8 @@ typedef void PaintingContextCallback(PaintingContext context, Offset offset);
/// A place to paint. /// A place to paint.
/// ///
/// Rather than holding a canvas directly, render objects paint using a painting /// Rather than holding a canvas directly, [RenderObject]s paint using a painting
/// context. The painting context has a canvas, which receives the /// context. The painting context has a [Canvas], which receives the
/// individual draw operations, and also has functions for painting child /// individual draw operations, and also has functions for painting child
/// render objects. /// render objects.
/// ///
...@@ -242,7 +242,7 @@ class PaintingContext { ...@@ -242,7 +242,7 @@ class PaintingContext {
_currentLayer?.willChangeHint = true; _currentLayer?.willChangeHint = true;
} }
/// Adds a composited layer to the recording. /// Adds a composited leaf layer to the recording.
/// ///
/// After calling this function, the [canvas] property will change to refer to /// After calling this function, the [canvas] property will change to refer to
/// a new [Canvas] that draws on top of the given layer. /// a new [Canvas] that draws on top of the given layer.
...@@ -250,12 +250,46 @@ class PaintingContext { ...@@ -250,12 +250,46 @@ class PaintingContext {
/// A [RenderObject] that uses this function is very likely to require its /// A [RenderObject] that uses this function is very likely to require its
/// [RenderObject.alwaysNeedsCompositing] property to return true. That informs /// [RenderObject.alwaysNeedsCompositing] property to return true. That informs
/// ancestor render objects that this render object will include a composited /// ancestor render objects that this render object will include a composited
/// layer, which causes them to use composited clips, for example. /// layer, which, for example, causes them to use composited clips.
///
/// See also:
///
/// * [pushLayer], for adding a layer and using its canvas to paint with that
/// layer.
void addLayer(Layer layer) { void addLayer(Layer layer) {
_stopRecordingIfNeeded(); _stopRecordingIfNeeded();
_appendLayer(layer); _appendLayer(layer);
} }
/// Appends the given layer to the recording, and calls the `painter` callback
/// with that layer, providing the [childPaintBounds] as the paint bounds of
/// the child. Canvas recording commands are not guaranteed to be stored
/// outside of the paint bounds.
///
/// The given layer must be an unattached orphan. (Providing a newly created
/// object, rather than reusing an existing layer, satisfies that
/// requirement.)
///
/// The `offset` is the offset to pass to the `painter`.
///
/// If the `childPaintBounds` are not specified then the current layer's
/// bounds are used. This is appropriate if the child layer does not apply any
/// transformation or clipping to its contents.
///
/// See also:
///
/// * [addLayer], for pushing a leaf layer whose canvas is not used.
void pushLayer(Layer childLayer, PaintingContextCallback painter, Offset offset, { Rect childPaintBounds }) {
assert(!childLayer.attached);
assert(childLayer.parent == null);
assert(painter != null);
_stopRecordingIfNeeded();
_appendLayer(childLayer);
final PaintingContext childContext = new PaintingContext._(childLayer, childPaintBounds ?? _paintBounds);
painter(childContext, offset);
childContext._stopRecordingIfNeeded();
}
/// Clip further painting using a rectangle. /// Clip further painting using a rectangle.
/// ///
/// * `needsCompositing` is whether the child needs compositing. Typically /// * `needsCompositing` is whether the child needs compositing. Typically
...@@ -269,12 +303,7 @@ class PaintingContext { ...@@ -269,12 +303,7 @@ class PaintingContext {
void pushClipRect(bool needsCompositing, Offset offset, Rect clipRect, PaintingContextCallback painter) { void pushClipRect(bool needsCompositing, Offset offset, Rect clipRect, PaintingContextCallback painter) {
final Rect offsetClipRect = clipRect.shift(offset); final Rect offsetClipRect = clipRect.shift(offset);
if (needsCompositing) { if (needsCompositing) {
_stopRecordingIfNeeded(); pushLayer(new ClipRectLayer(clipRect: offsetClipRect), painter, offset, childPaintBounds: offsetClipRect);
final ClipRectLayer clipLayer = new ClipRectLayer(clipRect: offsetClipRect);
_appendLayer(clipLayer);
final PaintingContext childContext = new PaintingContext._(clipLayer, offsetClipRect);
painter(childContext, offset);
childContext._stopRecordingIfNeeded();
} else { } else {
canvas.save(); canvas.save();
canvas.clipRect(offsetClipRect); canvas.clipRect(offsetClipRect);
...@@ -299,12 +328,7 @@ class PaintingContext { ...@@ -299,12 +328,7 @@ class PaintingContext {
final Rect offsetBounds = bounds.shift(offset); final Rect offsetBounds = bounds.shift(offset);
final RRect offsetClipRRect = clipRRect.shift(offset); final RRect offsetClipRRect = clipRRect.shift(offset);
if (needsCompositing) { if (needsCompositing) {
_stopRecordingIfNeeded(); pushLayer(new ClipRRectLayer(clipRRect: offsetClipRRect), painter, offset, childPaintBounds: offsetBounds);
final ClipRRectLayer clipLayer = new ClipRRectLayer(clipRRect: offsetClipRRect);
_appendLayer(clipLayer);
final PaintingContext childContext = new PaintingContext._(clipLayer, offsetBounds);
painter(childContext, offset);
childContext._stopRecordingIfNeeded();
} else { } else {
canvas.saveLayer(offsetBounds, _defaultPaint); canvas.saveLayer(offsetBounds, _defaultPaint);
canvas.clipRRect(offsetClipRRect); canvas.clipRRect(offsetClipRRect);
...@@ -329,12 +353,7 @@ class PaintingContext { ...@@ -329,12 +353,7 @@ class PaintingContext {
final Rect offsetBounds = bounds.shift(offset); final Rect offsetBounds = bounds.shift(offset);
final Path offsetClipPath = clipPath.shift(offset); final Path offsetClipPath = clipPath.shift(offset);
if (needsCompositing) { if (needsCompositing) {
_stopRecordingIfNeeded(); pushLayer(new ClipPathLayer(clipPath: offsetClipPath), painter, offset, childPaintBounds: offsetBounds);
final ClipPathLayer clipLayer = new ClipPathLayer(clipPath: offsetClipPath);
_appendLayer(clipLayer);
final PaintingContext childContext = new PaintingContext._(clipLayer, offsetBounds);
painter(childContext, offset);
childContext._stopRecordingIfNeeded();
} else { } else {
canvas.saveLayer(bounds.shift(offset), _defaultPaint); canvas.saveLayer(bounds.shift(offset), _defaultPaint);
canvas.clipPath(clipPath.shift(offset)); canvas.clipPath(clipPath.shift(offset));
...@@ -356,13 +375,12 @@ class PaintingContext { ...@@ -356,13 +375,12 @@ class PaintingContext {
final Matrix4 effectiveTransform = new Matrix4.translationValues(offset.dx, offset.dy, 0.0) final Matrix4 effectiveTransform = new Matrix4.translationValues(offset.dx, offset.dy, 0.0)
..multiply(transform)..translate(-offset.dx, -offset.dy); ..multiply(transform)..translate(-offset.dx, -offset.dy);
if (needsCompositing) { if (needsCompositing) {
_stopRecordingIfNeeded(); pushLayer(
final TransformLayer transformLayer = new TransformLayer(transform: effectiveTransform); new TransformLayer(transform: effectiveTransform),
_appendLayer(transformLayer); painter,
final Rect transformedPaintBounds = MatrixUtils.inverseTransformRect(effectiveTransform, _paintBounds); offset,
final PaintingContext childContext = new PaintingContext._(transformLayer, transformedPaintBounds); childPaintBounds: MatrixUtils.inverseTransformRect(effectiveTransform, _paintBounds),
painter(childContext, offset); );
childContext._stopRecordingIfNeeded();
} else { } else {
canvas.save(); canvas.save();
canvas.transform(effectiveTransform.storage); canvas.transform(effectiveTransform.storage);
...@@ -384,112 +402,9 @@ class PaintingContext { ...@@ -384,112 +402,9 @@ class PaintingContext {
/// A [RenderObject] that uses this function is very likely to require its /// A [RenderObject] that uses this function is very likely to require its
/// [RenderObject.alwaysNeedsCompositing] property to return true. That informs /// [RenderObject.alwaysNeedsCompositing] property to return true. That informs
/// ancestor render objects that this render object will include a composited /// ancestor render objects that this render object will include a composited
/// layer, which causes them to use composited clips, for example. /// layer, which, for example, causes them to use composited clips.
void pushOpacity(Offset offset, int alpha, PaintingContextCallback painter) { void pushOpacity(Offset offset, int alpha, PaintingContextCallback painter) {
_stopRecordingIfNeeded(); pushLayer(new OpacityLayer(alpha: alpha), painter, offset);
final OpacityLayer opacityLayer = new OpacityLayer(alpha: alpha);
_appendLayer(opacityLayer);
final PaintingContext childContext = new PaintingContext._(opacityLayer, _paintBounds);
painter(childContext, offset);
childContext._stopRecordingIfNeeded();
}
/// Apply a mask derived from a shader to further painting.
///
/// * `offset` is the offset from the origin of the canvas' coordinate system
/// to the origin of the caller's coordinate system.
/// * `shader` is the shader that will generate the mask. The shader operates
/// in the coordinate system of the caller.
/// * `maskRect` is the region of the canvas (in the coodinate system of the
/// caller) in which to apply the mask.
/// * `blendMode` is the [BlendMode] to use when applying the shader to
/// the painting done by `painter`.
/// * `painter` is a callback that will paint with the mask applied. This
/// function calls the `painter` synchronously.
///
/// A [RenderObject] that uses this function is very likely to require its
/// [RenderObject.alwaysNeedsCompositing] property to return true. That informs
/// ancestor render objects that this render object will include a composited
/// layer, which causes them to use composited clips, for example.
void pushShaderMask(Offset offset, Shader shader, Rect maskRect, BlendMode blendMode, PaintingContextCallback painter) {
_stopRecordingIfNeeded();
final ShaderMaskLayer shaderLayer = new ShaderMaskLayer(
shader: shader,
maskRect: maskRect.shift(offset),
blendMode: blendMode,
);
_appendLayer(shaderLayer);
final PaintingContext childContext = new PaintingContext._(shaderLayer, _paintBounds);
painter(childContext, offset);
childContext._stopRecordingIfNeeded();
}
/// Push a backdrop filter.
///
/// This function applies a filter to the existing painted content and then
/// synchronously calls the painter to paint on top of the filtered backdrop.
///
/// A [RenderObject] that uses this function is very likely to require its
/// [RenderObject.alwaysNeedsCompositing] property to return true. That informs
/// ancestor render objects that this render object will include a composited
/// layer, which causes them to use composited clips, for example.
// TODO(abarth): I don't quite understand how this API is supposed to work.
void pushBackdropFilter(Offset offset, ui.ImageFilter filter, PaintingContextCallback painter) {
_stopRecordingIfNeeded();
final BackdropFilterLayer backdropFilterLayer = new BackdropFilterLayer(filter: filter);
_appendLayer(backdropFilterLayer);
final PaintingContext childContext = new PaintingContext._(backdropFilterLayer, _paintBounds);
painter(childContext, offset);
childContext._stopRecordingIfNeeded();
}
/// Clip using a physical model layer.
///
/// * `offset` is the offset from the origin of the canvas' coordinate system
/// to the origin of the caller's coordinate system.
/// * `bounds` is the region of the canvas (in the caller's coodinate system)
/// into which `painter` will paint in.
/// * `clipRRect` is the rounded-rectangle (in the caller's coodinate system)
/// to use to clip the painting done by `painter`.
/// * `elevation` is the z-coordinate at which to place this material.
/// * `color` is the background color.
/// * `painter` is a callback that will paint with the `clipRRect` applied. This
/// function calls the `painter` synchronously.
void pushPhysicalModel(bool needsCompositing, Offset offset, Rect bounds, RRect clipRRect, double elevation, Color color, PaintingContextCallback painter) {
final Rect offsetBounds = bounds.shift(offset);
final RRect offsetClipRRect = clipRRect.shift(offset);
if (needsCompositing) {
_stopRecordingIfNeeded();
final PhysicalModelLayer physicalModel = new PhysicalModelLayer(
clipRRect: offsetClipRRect,
elevation: elevation,
color: color,
);
_appendLayer(physicalModel);
final PaintingContext childContext = new PaintingContext._(physicalModel, offsetBounds);
painter(childContext, offset);
childContext._stopRecordingIfNeeded();
} else {
if (elevation != 0) {
// The drawShadow call doesn't add the region of the shadow to the
// picture's bounds, so we draw a hardcoded amount of extra space to
// account for the maximum potential area of the shadow.
// TODO(jsimmons): remove this when Skia does it for us.
canvas.drawRect(offsetBounds.inflate(20.0),
new Paint()..color=const Color(0));
canvas.drawShadow(
new Path()..addRRect(offsetClipRRect),
const Color(0xFF000000),
elevation.toDouble(),
color.alpha != 0xFF,
);
}
canvas.drawRRect(offsetClipRRect, new Paint()..color=color);
canvas.saveLayer(offsetBounds, _defaultPaint);
canvas.clipRRect(offsetClipRRect);
painter(this, offset);
canvas.restore();
}
} }
} }
...@@ -2082,8 +1997,8 @@ abstract class RenderObject extends AbstractNode implements HitTestTarget { ...@@ -2082,8 +1997,8 @@ abstract class RenderObject extends AbstractNode implements HitTestTarget {
/// creates at least one composited layer. For example, videos should return /// creates at least one composited layer. For example, videos should return
/// true if they use hardware decoders. /// true if they use hardware decoders.
/// ///
/// You must call markNeedsCompositingBitsUpdate() if the value of this /// You must call [markNeedsCompositingBitsUpdate] if the value of this getter
/// getter changes. /// changes. (This is implied when [adoptChild] or [dropChild] are called.)
@protected @protected
bool get alwaysNeedsCompositing => false; bool get alwaysNeedsCompositing => false;
......
...@@ -12,6 +12,7 @@ import 'package:vector_math/vector_math_64.dart'; ...@@ -12,6 +12,7 @@ import 'package:vector_math/vector_math_64.dart';
import 'box.dart'; import 'box.dart';
import 'debug.dart'; import 'debug.dart';
import 'layer.dart';
import 'object.dart'; import 'object.dart';
import 'semantics.dart'; import 'semantics.dart';
...@@ -837,8 +838,15 @@ class RenderShaderMask extends RenderProxyBox { ...@@ -837,8 +838,15 @@ class RenderShaderMask extends RenderProxyBox {
void paint(PaintingContext context, Offset offset) { void paint(PaintingContext context, Offset offset) {
if (child != null) { if (child != null) {
assert(needsCompositing); assert(needsCompositing);
final Shader shader = _shaderCallback(offset & size); context.pushLayer(
context.pushShaderMask(offset, shader, Offset.zero & size, _blendMode, super.paint); new ShaderMaskLayer(
shader: _shaderCallback(offset & size),
maskRect: offset & size,
blendMode: _blendMode,
),
super.paint,
offset,
);
} }
} }
} }
...@@ -878,7 +886,7 @@ class RenderBackdropFilter extends RenderProxyBox { ...@@ -878,7 +886,7 @@ class RenderBackdropFilter extends RenderProxyBox {
void paint(PaintingContext context, Offset offset) { void paint(PaintingContext context, Offset offset) {
if (child != null) { if (child != null) {
assert(needsCompositing); assert(needsCompositing);
context.pushBackdropFilter(offset, _filter, super.paint); context.pushLayer(new BackdropFilterLayer(filter: _filter), super.paint, offset);
} }
} }
} }
...@@ -1308,11 +1316,47 @@ class RenderPhysicalModel extends _RenderCustomClip<RRect> { ...@@ -1308,11 +1316,47 @@ class RenderPhysicalModel extends _RenderCustomClip<RRect> {
return super.hitTest(result, position: position); return super.hitTest(result, position: position);
} }
static final Paint _defaultPaint = new Paint();
static final Paint _transparentPaint = new Paint()..color = const Color(0x00000000);
@override @override
void paint(PaintingContext context, Offset offset) { void paint(PaintingContext context, Offset offset) {
if (child != null) { if (child != null) {
_updateClip(); _updateClip();
context.pushPhysicalModel(needsCompositing, offset, _clip.outerRect, _clip, elevation, color, super.paint); final RRect offsetClipRRect = _clip.shift(offset);
final Rect offsetBounds = offsetClipRRect.outerRect;
if (needsCompositing) {
final PhysicalModelLayer physicalModel = new PhysicalModelLayer(
clipRRect: offsetClipRRect,
elevation: elevation,
color: color,
);
context.pushLayer(physicalModel, super.paint, offset, childPaintBounds: offsetBounds);
} else {
final Canvas canvas = context.canvas;
if (elevation != 0.0) {
// The drawShadow call doesn't add the region of the shadow to the
// picture's bounds, so we draw a hardcoded amount of extra space to
// account for the maximum potential area of the shadow.
// TODO(jsimmons): remove this when Skia does it for us.
canvas.drawRect(
offsetBounds.inflate(20.0),
_transparentPaint,
);
canvas.drawShadow(
new Path()..addRRect(offsetClipRRect),
const Color(0xFF000000),
elevation,
color.alpha != 0xFF,
);
}
canvas.drawRRect(offsetClipRRect, new Paint()..color = color);
canvas.saveLayer(offsetBounds, _defaultPaint);
canvas.clipRRect(offsetClipRRect);
super.paint(context, offset);
canvas.restore();
assert(context.canvas == canvas, 'canvas changed even though needsCompositing was false');
}
} }
} }
......
...@@ -38,6 +38,12 @@ class TestRecordingCanvas implements Canvas { ...@@ -38,6 +38,12 @@ class TestRecordingCanvas implements Canvas {
invocations.add(new _MethodCall(#save)); invocations.add(new _MethodCall(#save));
} }
@override
void saveLayer(Rect bounds, Paint paint) {
_saveCount += 1;
invocations.add(new _MethodCall(#saveLayer, <dynamic>[bounds, paint]));
}
@override @override
void restore() { void restore() {
_saveCount -= 1; _saveCount -= 1;
...@@ -77,8 +83,9 @@ class TestRecordingPaintingContext implements PaintingContext { ...@@ -77,8 +83,9 @@ class TestRecordingPaintingContext implements PaintingContext {
} }
class _MethodCall implements Invocation { class _MethodCall implements Invocation {
_MethodCall(this._name); _MethodCall(this._name, [ this._arguments = const <dynamic>[] ]);
final Symbol _name; final Symbol _name;
final List<dynamic> _arguments;
@override @override
bool get isAccessor => false; bool get isAccessor => false;
@override @override
...@@ -92,5 +99,5 @@ class _MethodCall implements Invocation { ...@@ -92,5 +99,5 @@ class _MethodCall implements Invocation {
@override @override
Map<Symbol, dynamic> get namedArguments => <Symbol, dynamic>{}; Map<Symbol, dynamic> get namedArguments => <Symbol, dynamic>{};
@override @override
List<dynamic> get positionalArguments => <dynamic>[]; List<dynamic> get positionalArguments => _arguments;
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment