system_fonts_test.dart 10.6 KB
Newer Older
Ian Hickson's avatar
Ian Hickson committed
1
// Copyright 2014 The Flutter Authors. All rights reserved.
2 3 4
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

5 6
import 'dart:async';

7
import 'package:flutter/cupertino.dart';
8 9
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
10
import 'package:flutter/rendering.dart';
11 12 13
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';

14 15
Future<void> verifyMarkedNeedsLayoutDuringTransientCallbacksPhase(WidgetTester tester, RenderObject renderObject) async {
  assert(!renderObject.debugNeedsLayout);
16

17 18 19 20 21 22 23 24
  const Map<String, dynamic> data = <String, dynamic>{
    'type': 'fontsChange',
  };
  await ServicesBinding.instance.defaultBinaryMessenger.handlePlatformMessage(
    'flutter/system',
    SystemChannels.system.codec.encodeMessage(data),
    (ByteData? data) { },
  );
25

26 27 28
  final Completer<bool> animation = Completer<bool>();
  tester.binding.scheduleFrameCallback((Duration timeStamp) {
    animation.complete(renderObject.debugNeedsLayout);
29 30
  });

31 32 33 34 35 36 37 38 39
  // The fonts change does not mark the render object as needing layout
  // immediately.
  expect(renderObject.debugNeedsLayout, isFalse);
  await tester.pump();
  expect(await animation.future, isTrue);
}

void main() {
  testWidgets('RenderParagraph relayout upon system fonts changes', (WidgetTester tester) async {
40 41 42 43 44
    await tester.pumpWidget(
      const MaterialApp(
        home: Text('text widget'),
      ),
    );
45 46
    final RenderObject renderObject = tester.renderObject(find.text('text widget'));
    await verifyMarkedNeedsLayoutDuringTransientCallbacksPhase(tester, renderObject);
47 48
  });

49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70
  testWidgets(
    'Safe to query a RelayoutWhenSystemFontsChangeMixin for text layout after system fonts changes',
    (WidgetTester tester) async {
      final _RenderCustomRelayoutWhenSystemFontsChange child = _RenderCustomRelayoutWhenSystemFontsChange();
      await tester.pumpWidget(
        Directionality(
          textDirection: TextDirection.ltr,
          child: WidgetToRenderBoxAdapter(renderBox: child),
        ),
      );
      const Map<String, dynamic> data = <String, dynamic>{
        'type': 'fontsChange',
      };
      await ServicesBinding.instance.defaultBinaryMessenger.handlePlatformMessage(
        'flutter/system',
        SystemChannels.system.codec.encodeMessage(data),
        (ByteData? data) { },
      );
      expect(child.hasValidTextLayout, isTrue);
    },
  );

71 72 73 74
  testWidgets('RenderEditable relayout upon system fonts changes', (WidgetTester tester) async {
    await tester.pumpWidget(
      const MaterialApp(
        home: SelectableText('text widget'),
75
      ),
76
    );
77

78
    final EditableTextState state = tester.state(find.byType(EditableText));
79
    await verifyMarkedNeedsLayoutDuringTransientCallbacksPhase(tester, state.renderEditable);
80 81 82 83 84 85 86 87 88
  });

  testWidgets('Banner repaint upon system fonts changes', (WidgetTester tester) async {
    await tester.pumpWidget(
      const Banner(
        message: 'message',
        location: BannerLocation.topStart,
        textDirection: TextDirection.ltr,
        layoutDirection: TextDirection.ltr,
89
      ),
90 91 92 93
    );
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
94
    await ServicesBinding.instance.defaultBinaryMessenger.handlePlatformMessage(
95 96
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
97
        (ByteData? data) { },
98 99 100 101 102 103 104 105 106 107
    );
    final RenderObject renderObject = tester.renderObject(find.byType(Banner));
    expect(renderObject.debugNeedsPaint, isTrue);
  });

  testWidgets('CupertinoDatePicker reset cache upon system fonts change - date time mode', (WidgetTester tester) async {
    await tester.pumpWidget(
      CupertinoApp(
        home: CupertinoDatePicker(
          onDateTimeChanged: (DateTime dateTime) { },
108 109
        ),
      ),
110
    );
111
    final dynamic state = tester.state(find.byType(CupertinoDatePicker));
112
    // ignore: avoid_dynamic_calls
113
    final Map<int, double> cache = state.estimatedColumnWidths as Map<int, double>;
114 115 116 117
    expect(cache.isNotEmpty, isTrue);
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
118
    await ServicesBinding.instance.defaultBinaryMessenger.handlePlatformMessage(
119 120
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
121
        (ByteData? data) { },
122 123 124 125 126
    );
    // Cache should be cleaned.
    expect(cache.isEmpty, isTrue);
    final Element element = tester.element(find.byType(CupertinoDatePicker));
    expect(element.dirty, isTrue);
127
  }, skip: isBrowser);  // TODO(yjbanov): cupertino does not work on the Web yet: https://github.com/flutter/flutter/issues/41920
128 129 130 131 132 133 134

  testWidgets('CupertinoDatePicker reset cache upon system fonts change - date mode', (WidgetTester tester) async {
    await tester.pumpWidget(
      CupertinoApp(
        home: CupertinoDatePicker(
          mode: CupertinoDatePickerMode.date,
          onDateTimeChanged: (DateTime dateTime) { },
135 136
        ),
      ),
137
    );
138
    final dynamic state = tester.state(find.byType(CupertinoDatePicker));
139
    // ignore: avoid_dynamic_calls
140
    final Map<int, double> cache = state.estimatedColumnWidths as Map<int, double>;
141 142 143 144 145
    // Simulates font missing.
    cache.clear();
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
146
    await ServicesBinding.instance.defaultBinaryMessenger.handlePlatformMessage(
147 148
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
149
        (ByteData? data) { },
150 151 152 153 154
    );
    // Cache should be replenished
    expect(cache.isNotEmpty, isTrue);
    final Element element = tester.element(find.byType(CupertinoDatePicker));
    expect(element.dirty, isTrue);
155
  }, skip: isBrowser);  // TODO(yjbanov): cupertino does not work on the Web yet: https://github.com/flutter/flutter/issues/41920
156 157 158 159 160 161

  testWidgets('CupertinoDatePicker reset cache upon system fonts change - time mode', (WidgetTester tester) async {
    await tester.pumpWidget(
      CupertinoApp(
        home: CupertinoTimerPicker(
          onTimerDurationChanged: (Duration d) { },
162 163
        ),
      ),
164
    );
165
    final dynamic state = tester.state(find.byType(CupertinoTimerPicker));
166
    // Simulates wrong metrics due to font missing.
167
    // ignore: avoid_dynamic_calls
168
    state.numberLabelWidth = 0.0;
169
    // ignore: avoid_dynamic_calls
170
    state.numberLabelHeight = 0.0;
171
    // ignore: avoid_dynamic_calls
172 173 174 175
    state.numberLabelBaseline = 0.0;
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
176
    await ServicesBinding.instance.defaultBinaryMessenger.handlePlatformMessage(
177 178
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
179
        (ByteData? data) { },
180 181
    );
    // Metrics should be refreshed
182
    // ignore: avoid_dynamic_calls
183
    expect(state.numberLabelWidth - 46.0 < precisionErrorTolerance, isTrue);
184
    // ignore: avoid_dynamic_calls
185
    expect(state.numberLabelHeight - 23.0 < precisionErrorTolerance, isTrue);
186
    // ignore: avoid_dynamic_calls
187 188 189
    expect(state.numberLabelBaseline - 18.400070190429688 < precisionErrorTolerance, isTrue);
    final Element element = tester.element(find.byType(CupertinoTimerPicker));
    expect(element.dirty, isTrue);
190
  }, skip: isBrowser);  // TODO(yjbanov): cupertino does not work on the Web yet: https://github.com/flutter/flutter/issues/41920
191 192 193 194 195 196 197 198 199 200

  testWidgets('RangeSlider relayout upon system fonts changes', (WidgetTester tester) async {
    await tester.pumpWidget(
      MaterialApp(
        home: Material(
          child: RangeSlider(
            values: const RangeValues(0.0, 1.0),
            onChanged: (RangeValues values) { },
          ),
        ),
201
      ),
202 203 204 205
    );
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
206
    await ServicesBinding.instance.defaultBinaryMessenger.handlePlatformMessage(
207 208
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
209
        (ByteData? data) { },
210
    );
211 212
    final RenderObject renderObject = tester.renderObject(find.byWidgetPredicate((Widget widget) => widget.runtimeType.toString() == '_RangeSliderRenderObjectWidget'));
    await verifyMarkedNeedsLayoutDuringTransientCallbacksPhase(tester, renderObject);
213 214 215 216 217 218 219 220 221 222 223
  });

  testWidgets('Slider relayout upon system fonts changes', (WidgetTester tester) async {
    await tester.pumpWidget(
      MaterialApp(
        home: Material(
          child: Slider(
            value: 0.0,
            onChanged: (double value) { },
          ),
        ),
224
      ),
225
    );
226 227
    // _RenderSlider is the last render object in the tree.
    final RenderObject renderObject = tester.allRenderObjects.last;
228
    await verifyMarkedNeedsLayoutDuringTransientCallbacksPhase(tester, renderObject);
229 230 231 232 233 234 235 236 237
  });

  testWidgets('TimePicker relayout upon system fonts changes', (WidgetTester tester) async {
    await tester.pumpWidget(
      MaterialApp(
        home: Material(
          child: Center(
            child: Builder(
              builder: (BuildContext context) {
238
                return ElevatedButton(
239 240 241 242 243
                  child: const Text('X'),
                  onPressed: () {
                    showTimePicker(
                      context: context,
                      initialTime: const TimeOfDay(hour: 7, minute: 0),
244
                      builder: (BuildContext context, Widget? child) {
245 246 247
                        return Directionality(
                          key: const Key('parent'),
                          textDirection: TextDirection.ltr,
248
                          child: child!,
249 250 251 252 253 254 255 256 257
                        );
                      },
                    );
                  },
                );
              },
            ),
          ),
        ),
258
      ),
259 260 261 262 263 264
    );
    await tester.tap(find.text('X'));
    await tester.pumpAndSettle();
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
265
    await ServicesBinding.instance.defaultBinaryMessenger.handlePlatformMessage(
266 267
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
268
        (ByteData? data) { },
269 270 271 272
    );
    final RenderObject renderObject = tester.renderObject(
      find.descendant(
        of: find.byKey(const Key('parent')),
273 274
        matching: find.byKey(const ValueKey<String>('time-picker-dial')),
      ),
275 276 277 278
    );
    expect(renderObject.debugNeedsPaint, isTrue);
  });
}
279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294

class _RenderCustomRelayoutWhenSystemFontsChange extends RenderBox with RelayoutWhenSystemFontsChangeMixin {
  bool hasValidTextLayout = false;

  @override
  void systemFontsDidChange() {
    super.systemFontsDidChange();
    hasValidTextLayout = false;
  }

  @override
  void performLayout() {
    size = constraints.biggest;
    hasValidTextLayout = true;
  }
}