debug.dart 8.46 KB
Newer Older
1 2 3 4
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

5
import 'package:flutter/foundation.dart';
6
import 'package:flutter/painting.dart';
7
import 'package:vector_math/vector_math_64.dart';
8

9
export 'package:flutter/foundation.dart' show debugPrint;
10

11 12 13
// Any changes to this file should be reflected in the debugAssertAllRenderVarsUnset()
// function below.

14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
/// Used by [debugDumpSemanticsTree] to specify the order in which child nodes
/// are printed.
enum DebugSemanticsDumpOrder {
  /// Print nodes in inverse hit test order.
  ///
  /// In inverse hit test order, the last child of a [SemanticsNode] will be
  /// asked first if it wants to respond to a user's interaction, followed by
  /// the second last, etc. until a taker is found.
  inverseHitTest,

  /// Print nodes in traversal order.
  ///
  /// Traversal order defines how the user can move the accessibility focus from
  /// one node to another.
  traversal,
}

31
const HSVColor _kDebugDefaultRepaintColor = const HSVColor.fromAHSV(0.4, 60.0, 1.0, 1.0);
32

33
/// Causes each RenderBox to paint a box around its bounds, and some extra
Ian Hickson's avatar
Ian Hickson committed
34
/// boxes, such as [RenderPadding], to draw construction lines.
35
///
36
/// The edges of boies are painted as a one-pixel-thick `const Color(0xFF00FFFF)` outline.
37
///
38
/// Spacing is painted as a solid `const Color(0x90909090)` area.
39
///
40 41 42
/// Padding is filled in solid `const Color(0x900090FF)`, with the inner edge
/// outlined in `const Color(0xFF0090FF)`, using [debugPaintPadding].
bool debugPaintSizeEnabled = false;
43

Adam Barth's avatar
Adam Barth committed
44
/// Causes each RenderBox to paint a line at each of its baselines.
45
bool debugPaintBaselinesEnabled = false;
Adam Barth's avatar
Adam Barth committed
46 47

/// Causes each Layer to paint a box around its bounds.
48
bool debugPaintLayerBordersEnabled = false;
Adam Barth's avatar
Adam Barth committed
49

50 51 52 53 54 55
/// Causes objects like [RenderPointerListener] to flash while they are being
/// tapped. This can be useful to see how large the hit box is, e.g. when
/// debugging buttons that are harder to hit than expected.
///
/// For details on how to support this in your [RenderBox] subclass, see
/// [RenderBox.debugHandleEvent].
56
bool debugPaintPointersEnabled = false;
57

58
/// Overlay a rotating set of colors when repainting layers in checked mode.
59
bool debugRepaintRainbowEnabled = false;
60

61 62 63
/// Overlay a rotating set of colors when repainting text in checked mode.
bool debugRepaintTextRainbowEnabled = false;

64
/// The current color to overlay when repainting a layer.
65 66 67 68 69 70 71
///
/// This is used by painting debug code that implements
/// [debugRepaintRainbowEnabled] or [debugRepaintTextRainbowEnabled].
///
/// The value is incremented by [RenderView.compositeFrame] if either of those
/// flags is enabled.
HSVColor debugCurrentRepaintColor = _kDebugDefaultRepaintColor;
72

73
/// Log the call stacks that mark render objects as needing layout.
74 75 76 77 78
///
/// For sanity, this only logs the stack traces of cases where an object is
/// added to the list of nodes needing layout. This avoids printing multiple
/// redundant stack traces as a single [RenderObject.markNeedsLayout] call walks
/// up the tree.
79 80
bool debugPrintMarkNeedsLayoutStacks = false;

81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
/// Log the call stacks that mark render objects as needing paint.
bool debugPrintMarkNeedsPaintStacks = false;

/// Log the dirty render objects that are laid out each frame.
///
/// Combined with [debugPrintBeginFrameBanner], this allows you to distinguish
/// layouts triggered by the initial mounting of a render tree (e.g. in a call
/// to [runApp]) from the regular layouts triggered by the pipeline.
///
/// Combined with [debugPrintMarkNeedsLayoutStacks], this lets you watch a
/// render object's dirty/clean lifecycle.
///
/// See also:
///
///  * [debugProfilePaintsEnabled], which does something similar for
///    painting but using the timeline view.
///
///  * [debugPrintRebuildDirtyWidgets], which does something similar for widgets
///    being rebuilt.
///
///  * The discussion at [RendererBinding.drawFrame].
bool debugPrintLayouts = false;

104
/// Check the intrinsic sizes of each [RenderBox] during layout.
105 106 107
///
/// By default this is turned off since these checks are expensive, but it is
/// enabled by the test framework.
108 109
bool debugCheckIntrinsicSizes = false;

110 111 112 113 114
/// Adds [dart:developer.Timeline] events for every [RenderObject] painted.
///
/// This is only enabled in debug builds. The timing information this exposes is
/// not representative of actual paints. However, it can expose unexpected
/// painting in the timeline.
115
///
Ian Hickson's avatar
Ian Hickson committed
116 117 118
/// For details on how to use [dart:developer.Timeline] events in the Dart
/// Observatory to optimize your app, see:
/// <https://fuchsia.googlesource.com/sysui/+/master/docs/performance.md>
119 120 121 122 123 124
///
/// See also:
///
///  * [debugPrintLayouts], which does something similar for layout but using
///    console output.
///
125 126 127
///  * [debugProfileBuildsEnabled], which does something similar for widgets
///    being rebuilt, and [debugPrintRebuildDirtyWidgets], its console
///    equivalent.
128 129
///
///  * The discussion at [RendererBinding.drawFrame].
130 131 132
bool debugProfilePaintsEnabled = false;


133
/// Returns a list of strings representing the given transform in a format
134
/// useful for [TransformProperty].
135 136
///
/// If the argument is null, returns a list with the single string "null".
137
List<String> debugDescribeTransform(Matrix4 transform) {
138 139
  if (transform == null)
    return const <String>['null'];
140
  final List<String> matrix = transform.toString().split('\n').map((String s) => '  $s').toList();
141 142 143
  matrix.removeLast();
  return matrix;
}
144

145 146
/// Property which handles [Matrix4] that represent transforms.
class TransformProperty extends DiagnosticsProperty<Matrix4> {
147
  /// Create a diagnostics property for [Matrix4] objects.
148 149 150 151 152 153 154 155 156
  TransformProperty(String name, Matrix4 value, {
    Object defaultValue: kNoDefaultValue,
  }) : super(
    name,
    value,
    defaultValue: defaultValue,
  );

  @override
157 158 159 160 161 162 163 164 165 166 167 168 169 170
  String valueToString({ TextTreeConfiguration parentConfiguration }) {
    if (parentConfiguration != null && !parentConfiguration.lineBreakProperties) {
      // Format the value on a single line to be compatible with the parent's
      // style.
      final List<Vector4> rows = <Vector4>[
        value.getRow(0),
        value.getRow(1),
        value.getRow(2),
        value.getRow(3),
      ];
      return '[${rows.join("; ")}]';
    }
    return debugDescribeTransform(value).join('\n');
  }
171 172
}

173 174 175 176 177 178 179 180 181 182
void _debugDrawDoubleRect(Canvas canvas, Rect outerRect, Rect innerRect, Color color) {
  final Path path = new Path()
    ..fillType = PathFillType.evenOdd
    ..addRect(outerRect)
    ..addRect(innerRect);
  final Paint paint = new Paint()
    ..color = color;
  canvas.drawPath(path, paint);
}

183
/// Paint a diagram showing the given area as padding.
184 185 186 187 188 189
///
/// Called by [RenderPadding.debugPaintSize] when [debugPaintSizeEnabled] is
/// true.
void debugPaintPadding(Canvas canvas, Rect outerRect, Rect innerRect, { double outlineWidth: 2.0 }) {
  assert(() {
    if (innerRect != null && !innerRect.isEmpty) {
190 191
      _debugDrawDoubleRect(canvas, outerRect, innerRect, const Color(0x900090FF));
      _debugDrawDoubleRect(canvas, innerRect.inflate(outlineWidth).intersect(outerRect), innerRect, const Color(0xFF0090FF));
192 193
    } else {
      final Paint paint = new Paint()
194
        ..color = const Color(0x90909090);
195 196 197 198 199 200
      canvas.drawRect(outerRect, paint);
    }
    return true;
  });
}

201 202 203 204 205
/// Returns true if none of the rendering library debug variables have been changed.
///
/// This function is used by the test framework to ensure that debug variables
/// haven't been inadvertently changed.
///
206
/// See <https://docs.flutter.io/flutter/rendering/rendering-library.html> for
207
/// a complete list.
208 209 210 211 212
///
/// The `debugCheckIntrinsicSizesOverride` argument can be provided to override
/// the expected value for [debugCheckIntrinsicSizes]. (This exists because the
/// test framework itself overrides this value in some cases.)
bool debugAssertAllRenderVarsUnset(String reason, { bool debugCheckIntrinsicSizesOverride: false }) {
213 214 215 216 217 218
  assert(() {
    if (debugPaintSizeEnabled ||
        debugPaintBaselinesEnabled ||
        debugPaintLayerBordersEnabled ||
        debugPaintPointersEnabled ||
        debugRepaintRainbowEnabled ||
219
        debugRepaintTextRainbowEnabled ||
220
        debugCurrentRepaintColor != _kDebugDefaultRepaintColor ||
221
        debugPrintMarkNeedsLayoutStacks ||
222 223
        debugPrintMarkNeedsPaintStacks ||
        debugPrintLayouts ||
224
        debugCheckIntrinsicSizes != debugCheckIntrinsicSizesOverride ||
225
        debugProfilePaintsEnabled) {
226 227 228 229 230 231
      throw new FlutterError(reason);
    }
    return true;
  });
  return true;
}