system_fonts_test.dart 9.3 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
import 'package:flutter/cupertino.dart';
6 7 8 9 10 11 12 13 14 15
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';

void main() {
  testWidgets('RenderParagraph relayout upon system fonts changes', (WidgetTester tester) async {
    await tester.pumpWidget(
      const MaterialApp(
        home: Text('text widget'),
16
      ),
17 18 19 20
    );
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
21
    await ServicesBinding.instance!.defaultBinaryMessenger.handlePlatformMessage(
22 23
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
24
      (ByteData? data) { },
25 26 27 28 29 30 31 32 33
    );
    final RenderObject renderObject = tester.renderObject(find.text('text widget'));
    expect(renderObject.debugNeedsLayout, isTrue);
  });

  testWidgets('RenderEditable relayout upon system fonts changes', (WidgetTester tester) async {
    await tester.pumpWidget(
      const MaterialApp(
        home: SelectableText('text widget'),
34
      ),
35 36 37 38
    );
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
39
    await ServicesBinding.instance!.defaultBinaryMessenger.handlePlatformMessage(
40 41
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
42
        (ByteData? data) { },
43 44 45 46 47 48 49 50 51 52 53 54
    );
    final EditableTextState state = tester.state(find.byType(EditableText));
    expect(state.renderEditable.debugNeedsLayout, isTrue);
  });

  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,
55
      ),
56 57 58 59
    );
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
60
    await ServicesBinding.instance!.defaultBinaryMessenger.handlePlatformMessage(
61 62
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
63
        (ByteData? data) { },
64 65 66 67 68 69 70 71 72 73
    );
    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) { },
74 75
        ),
      ),
76
    );
77
    final dynamic state = tester.state(find.byType(CupertinoDatePicker));
78
    // ignore: avoid_dynamic_calls
79
    final Map<int, double> cache = state.estimatedColumnWidths as Map<int, double>;
80 81 82 83
    expect(cache.isNotEmpty, isTrue);
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
84
    await ServicesBinding.instance!.defaultBinaryMessenger.handlePlatformMessage(
85 86
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
87
        (ByteData? data) { },
88 89 90 91 92
    );
    // Cache should be cleaned.
    expect(cache.isEmpty, isTrue);
    final Element element = tester.element(find.byType(CupertinoDatePicker));
    expect(element.dirty, isTrue);
93
  }, skip: isBrowser);  // TODO(yjbanov): cupertino does not work on the Web yet: https://github.com/flutter/flutter/issues/41920
94 95 96 97 98 99 100

  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) { },
101 102
        ),
      ),
103
    );
104
    final dynamic state = tester.state(find.byType(CupertinoDatePicker));
105
    // ignore: avoid_dynamic_calls
106
    final Map<int, double> cache = state.estimatedColumnWidths as Map<int, double>;
107 108 109 110 111
    // Simulates font missing.
    cache.clear();
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
112
    await ServicesBinding.instance!.defaultBinaryMessenger.handlePlatformMessage(
113 114
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
115
        (ByteData? data) { },
116 117 118 119 120
    );
    // Cache should be replenished
    expect(cache.isNotEmpty, isTrue);
    final Element element = tester.element(find.byType(CupertinoDatePicker));
    expect(element.dirty, isTrue);
121
  }, skip: isBrowser);  // TODO(yjbanov): cupertino does not work on the Web yet: https://github.com/flutter/flutter/issues/41920
122 123 124 125 126 127

  testWidgets('CupertinoDatePicker reset cache upon system fonts change - time mode', (WidgetTester tester) async {
    await tester.pumpWidget(
      CupertinoApp(
        home: CupertinoTimerPicker(
          onTimerDurationChanged: (Duration d) { },
128 129
        ),
      ),
130
    );
131
    final dynamic state = tester.state(find.byType(CupertinoTimerPicker));
132
    // Simulates wrong metrics due to font missing.
133
    // ignore: avoid_dynamic_calls
134
    state.numberLabelWidth = 0.0;
135
    // ignore: avoid_dynamic_calls
136
    state.numberLabelHeight = 0.0;
137
    // ignore: avoid_dynamic_calls
138 139 140 141
    state.numberLabelBaseline = 0.0;
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
142
    await ServicesBinding.instance!.defaultBinaryMessenger.handlePlatformMessage(
143 144
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
145
        (ByteData? data) { },
146 147
    );
    // Metrics should be refreshed
148
    // ignore: avoid_dynamic_calls
149
    expect(state.numberLabelWidth - 46.0 < precisionErrorTolerance, isTrue);
150
    // ignore: avoid_dynamic_calls
151
    expect(state.numberLabelHeight - 23.0 < precisionErrorTolerance, isTrue);
152
    // ignore: avoid_dynamic_calls
153 154 155
    expect(state.numberLabelBaseline - 18.400070190429688 < precisionErrorTolerance, isTrue);
    final Element element = tester.element(find.byType(CupertinoTimerPicker));
    expect(element.dirty, isTrue);
156
  }, skip: isBrowser);  // TODO(yjbanov): cupertino does not work on the Web yet: https://github.com/flutter/flutter/issues/41920
157 158 159 160 161 162 163 164 165 166

  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) { },
          ),
        ),
167
      ),
168 169 170 171
    );
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
172
    await ServicesBinding.instance!.defaultBinaryMessenger.handlePlatformMessage(
173 174
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
175
        (ByteData? data) { },
176 177
    );
    final RenderObject renderObject = tester.renderObject(find.byType(RangeSlider));
178

179
    late bool sliderBoxNeedsLayout;
180 181
    renderObject.visitChildren((RenderObject child) {sliderBoxNeedsLayout = child.debugNeedsLayout;});
    expect(sliderBoxNeedsLayout, isTrue);
182 183 184 185 186 187 188 189 190 191 192
  });

  testWidgets('Slider relayout upon system fonts changes', (WidgetTester tester) async {
    await tester.pumpWidget(
      MaterialApp(
        home: Material(
          child: Slider(
            value: 0.0,
            onChanged: (double value) { },
          ),
        ),
193
      ),
194 195 196 197
    );
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
198
    await ServicesBinding.instance!.defaultBinaryMessenger.handlePlatformMessage(
199 200
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
201
        (ByteData? data) { },
202
    );
203 204 205
    // _RenderSlider is the last render object in the tree.
    final RenderObject renderObject = tester.allRenderObjects.last;
    expect(renderObject.debugNeedsLayout, isTrue);
206 207 208 209 210 211 212 213 214
  });

  testWidgets('TimePicker relayout upon system fonts changes', (WidgetTester tester) async {
    await tester.pumpWidget(
      MaterialApp(
        home: Material(
          child: Center(
            child: Builder(
              builder: (BuildContext context) {
215
                return ElevatedButton(
216 217 218 219 220
                  child: const Text('X'),
                  onPressed: () {
                    showTimePicker(
                      context: context,
                      initialTime: const TimeOfDay(hour: 7, minute: 0),
221
                      builder: (BuildContext context, Widget? child) {
222 223 224
                        return Directionality(
                          key: const Key('parent'),
                          textDirection: TextDirection.ltr,
225
                          child: child!,
226 227 228 229 230 231 232 233 234
                        );
                      },
                    );
                  },
                );
              },
            ),
          ),
        ),
235
      ),
236 237 238 239 240 241
    );
    await tester.tap(find.text('X'));
    await tester.pumpAndSettle();
    const Map<String, dynamic> data = <String, dynamic>{
      'type': 'fontsChange',
    };
242
    await ServicesBinding.instance!.defaultBinaryMessenger.handlePlatformMessage(
243 244
      'flutter/system',
      SystemChannels.system.codec.encodeMessage(data),
245
        (ByteData? data) { },
246 247 248 249 250
    );
    final RenderObject renderObject = tester.renderObject(
      find.descendant(
        of: find.byKey(const Key('parent')),
        matching: find.byType(CustomPaint),
251
      ).first,
252 253 254 255
    );
    expect(renderObject.debugNeedsPaint, isTrue);
  });
}