blob: 1ec3c64e35f0b0fd25457799564396a6ad31f729 [file] [log] [blame]
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import 'package:flutter/foundation.dart';
import 'package:flutter_test/flutter_test.dart';
class TestNotifier extends ChangeNotifier {
void notify() {
notifyListeners();
}
bool get isListenedTo => hasListeners;
}
class HasListenersTester<T> extends ValueNotifier<T> {
HasListenersTester(T value) : super(value);
bool get testHasListeners => hasListeners;
}
class A {
bool result = false;
void test() { result = true; }
}
class B extends A with ChangeNotifier {
@override
void test() {
notifyListeners();
super.test();
}
}
void main() {
testWidgets('ChangeNotifier', (WidgetTester tester) async {
final List<String> log = <String>[];
final VoidCallback listener = () { log.add('listener'); };
final VoidCallback listener1 = () { log.add('listener1'); };
final VoidCallback listener2 = () { log.add('listener2'); };
final VoidCallback badListener = () {
log.add('badListener');
throw ArgumentError();
};
final TestNotifier test = TestNotifier();
test.addListener(listener);
test.addListener(listener);
test.notify();
expect(log, <String>['listener', 'listener']);
log.clear();
test.removeListener(listener);
test.notify();
expect(log, <String>['listener']);
log.clear();
test.removeListener(listener);
test.notify();
expect(log, <String>[]);
log.clear();
test.removeListener(listener);
test.notify();
expect(log, <String>[]);
log.clear();
test.addListener(listener);
test.notify();
expect(log, <String>['listener']);
log.clear();
test.addListener(listener1);
test.notify();
expect(log, <String>['listener', 'listener1']);
log.clear();
test.addListener(listener2);
test.notify();
expect(log, <String>['listener', 'listener1', 'listener2']);
log.clear();
test.removeListener(listener1);
test.notify();
expect(log, <String>['listener', 'listener2']);
log.clear();
test.addListener(listener1);
test.notify();
expect(log, <String>['listener', 'listener2', 'listener1']);
log.clear();
test.addListener(badListener);
test.notify();
expect(log, <String>['listener', 'listener2', 'listener1', 'badListener']);
expect(tester.takeException(), isArgumentError);
log.clear();
test.addListener(listener1);
test.removeListener(listener);
test.removeListener(listener1);
test.removeListener(listener2);
test.addListener(listener2);
test.notify();
expect(log, <String>['badListener', 'listener1', 'listener2']);
expect(tester.takeException(), isArgumentError);
log.clear();
});
test('ChangeNotifier with mutating listener', () {
final TestNotifier test = TestNotifier();
final List<String> log = <String>[];
final VoidCallback listener1 = () { log.add('listener1'); };
final VoidCallback listener3 = () { log.add('listener3'); };
final VoidCallback listener4 = () { log.add('listener4'); };
final VoidCallback listener2 = () {
log.add('listener2');
test.removeListener(listener1);
test.removeListener(listener3);
test.addListener(listener4);
};
test.addListener(listener1);
test.addListener(listener2);
test.addListener(listener3);
test.notify();
expect(log, <String>['listener1', 'listener2']);
log.clear();
test.notify();
expect(log, <String>['listener2', 'listener4']);
log.clear();
test.notify();
expect(log, <String>['listener2', 'listener4', 'listener4']);
log.clear();
});
test('During notifyListeners, a listener was added and removed immediately', () {
final TestNotifier source = TestNotifier();
final List<String> log = <String>[];
final VoidCallback listener3 = () { log.add('listener3'); };
final VoidCallback listener2 = () { log.add('listener2'); };
void listener1() {
log.add('listener1');
source.addListener(listener2);
source.removeListener(listener2);
source.addListener(listener3);
}
source.addListener(listener1);
source.notify();
expect(log, <String>['listener1']);
});
test(
'If a listener in the middle of the list of listeners removes itself, '
'notifyListeners still notifies all listeners', () {
final TestNotifier source = TestNotifier();
final List<String> log = <String>[];
void selfRemovingListener() {
log.add('selfRemovingListener');
source.removeListener(selfRemovingListener);
}
final VoidCallback listener1 = () { log.add('listener1'); };
source.addListener(listener1);
source.addListener(selfRemovingListener);
source.addListener(listener1);
source.notify();
expect(log, <String>['listener1', 'selfRemovingListener', 'listener1']);
});
test('If the first listener removes itself, notifyListeners still notify all listeners', () {
final TestNotifier source = TestNotifier();
final List<String> log = <String>[];
void selfRemovingListener() {
log.add('selfRemovingListener');
source.removeListener(selfRemovingListener);
}
void listener1() {
log.add('listener1');
}
source.addListener(selfRemovingListener);
source.addListener(listener1);
source.notifyListeners();
expect(log, <String>['selfRemovingListener', 'listener1']);
});
test('Merging change notifiers', () {
final TestNotifier source1 = TestNotifier();
final TestNotifier source2 = TestNotifier();
final TestNotifier source3 = TestNotifier();
final List<String> log = <String>[];
final Listenable merged = Listenable.merge(<Listenable>[source1, source2]);
final VoidCallback listener1 = () { log.add('listener1'); };
final VoidCallback listener2 = () { log.add('listener2'); };
merged.addListener(listener1);
source1.notify();
source2.notify();
source3.notify();
expect(log, <String>['listener1', 'listener1']);
log.clear();
merged.removeListener(listener1);
source1.notify();
source2.notify();
source3.notify();
expect(log, isEmpty);
log.clear();
merged.addListener(listener1);
merged.addListener(listener2);
source1.notify();
source2.notify();
source3.notify();
expect(log, <String>['listener1', 'listener2', 'listener1', 'listener2']);
log.clear();
});
test('Merging change notifiers ignores null', () {
final TestNotifier source1 = TestNotifier();
final TestNotifier source2 = TestNotifier();
final List<String> log = <String>[];
final Listenable merged = Listenable.merge(<Listenable?>[null, source1, null, source2, null]);
final VoidCallback listener = () { log.add('listener'); };
merged.addListener(listener);
source1.notify();
source2.notify();
expect(log, <String>['listener', 'listener']);
log.clear();
});
test('Can remove from merged notifier', () {
final TestNotifier source1 = TestNotifier();
final TestNotifier source2 = TestNotifier();
final List<String> log = <String>[];
final Listenable merged = Listenable.merge(<Listenable>[source1, source2]);
final VoidCallback listener = () { log.add('listener'); };
merged.addListener(listener);
source1.notify();
source2.notify();
expect(log, <String>['listener', 'listener']);
log.clear();
merged.removeListener(listener);
source1.notify();
source2.notify();
expect(log, isEmpty);
});
test('Cannot use a disposed ChangeNotifier', () {
final TestNotifier source = TestNotifier();
source.dispose();
expect(() { source.addListener(() { }); }, throwsFlutterError);
expect(() { source.removeListener(() { }); }, throwsFlutterError);
expect(() { source.dispose(); }, throwsFlutterError);
expect(() { source.notify(); }, throwsFlutterError);
});
test('Value notifier', () {
final ValueNotifier<double> notifier = ValueNotifier<double>(2.0);
final List<double> log = <double>[];
final VoidCallback listener = () { log.add(notifier.value); };
notifier.addListener(listener);
notifier.value = 3.0;
expect(log, equals(<double>[ 3.0 ]));
log.clear();
notifier.value = 3.0;
expect(log, isEmpty);
});
test('Listenable.merge toString', () {
final TestNotifier source1 = TestNotifier();
final TestNotifier source2 = TestNotifier();
Listenable listenableUnderTest = Listenable.merge(<Listenable>[]);
expect(listenableUnderTest.toString(), 'Listenable.merge([])');
listenableUnderTest = Listenable.merge(<Listenable?>[null]);
expect(listenableUnderTest.toString(), 'Listenable.merge([null])');
listenableUnderTest = Listenable.merge(<Listenable>[source1]);
expect(
listenableUnderTest.toString(),
"Listenable.merge([Instance of 'TestNotifier'])",
);
listenableUnderTest = Listenable.merge(<Listenable>[source1, source2]);
expect(
listenableUnderTest.toString(),
"Listenable.merge([Instance of 'TestNotifier', Instance of 'TestNotifier'])",
);
listenableUnderTest = Listenable.merge(<Listenable?>[null, source2]);
expect(
listenableUnderTest.toString(),
"Listenable.merge([null, Instance of 'TestNotifier'])",
);
});
test('Listenable.merge does not leak', () {
// Regression test for https://github.com/flutter/flutter/issues/25163.
final TestNotifier source1 = TestNotifier();
final TestNotifier source2 = TestNotifier();
final VoidCallback fakeListener = () { };
final Listenable listenableUnderTest = Listenable.merge(<Listenable>[source1, source2]);
expect(source1.isListenedTo, isFalse);
expect(source2.isListenedTo, isFalse);
listenableUnderTest.addListener(fakeListener);
expect(source1.isListenedTo, isTrue);
expect(source2.isListenedTo, isTrue);
listenableUnderTest.removeListener(fakeListener);
expect(source1.isListenedTo, isFalse);
expect(source2.isListenedTo, isFalse);
});
test('hasListeners', () {
final HasListenersTester<bool> notifier = HasListenersTester<bool>(true);
expect(notifier.testHasListeners, isFalse);
void test1() { }
void test2() { }
notifier.addListener(test1);
expect(notifier.testHasListeners, isTrue);
notifier.addListener(test1);
expect(notifier.testHasListeners, isTrue);
notifier.removeListener(test1);
expect(notifier.testHasListeners, isTrue);
notifier.removeListener(test1);
expect(notifier.testHasListeners, isFalse);
notifier.addListener(test1);
expect(notifier.testHasListeners, isTrue);
notifier.addListener(test2);
expect(notifier.testHasListeners, isTrue);
notifier.removeListener(test1);
expect(notifier.testHasListeners, isTrue);
notifier.removeListener(test2);
expect(notifier.testHasListeners, isFalse);
});
test('ChangeNotifier as a mixin', () {
// We document that this is a valid way to use this class.
final B b = B();
int notifications = 0;
b.addListener(() {
notifications += 1;
});
expect(b.result, isFalse);
expect(notifications, 0);
b.test();
expect(b.result, isTrue);
expect(notifications, 1);
});
test('Throws FlutterError when disposed and called', () {
final TestNotifier testNotifier = TestNotifier();
testNotifier.dispose();
FlutterError? error;
try {
testNotifier.dispose();
} on FlutterError catch (e) {
error = e;
}
expect(error, isNotNull);
expect(error!, isFlutterError);
expect(error.toStringDeep(), equalsIgnoringHashCodes(
'FlutterError\n'
' A TestNotifier was used after being disposed.\n'
' Once you have called dispose() on a TestNotifier, it can no\n'
' longer be used.\n'
));
});
}