ML-62 Utils tests (#133)

* removed redundant `UserPreferencesService` from `MeteringBloc`

* wip

* post-merge fixes

* `MeasureEvent` tests

* `MeasureEvent` tests revision

* `MeasureEvent` tests added timeout

* added stubs for other `MeteringBloc` events

* rewritten `MeteringBloc` logic

* wip

* `IsoChangedEvent` tests

* refined `IsoChangedEvent` tests

* `NdChangedEvent` tests

* `FilmChangedEvent` tests

* `MeteringCommunicationBloc` tests

* added test run to ci

* overriden `==` for `MeasuredState`

* `LuxMeteringEvent` tests

* refined `LuxMeteringEvent` tests

* rename

* wip

* wip

* `InitializeEvent`/`DeinitializeEvent` tests

* clamp minZoomLevel

* fixed `MeteringCommunicationBloc` tests

* wip

* `ZoomChangedEvent` tests

* `ExposureOffsetChangedEvent`/`ExposureOffsetResetEvent` tests

* renamed test groups

* added test coverage script

* improved `CameraContainerBloc` test coverage

* `EquipmentProfileChangedEvent` tests

* verify response vibration

* fixed running all tests

* `MeteringCommunicationBloc` equality tests

* `CameraContainerBloc` equality tests

* removed generated code from coverage

* `MeteringScreenLayoutFeature` tests

* `SupportedLocale` tests

* `Film` tests

* `CaffeineService` tests

* `UserPreferencesService` tests (wip)

* `LightSensorService` tests (wip)

* `migrateOldKeys()` tests

* ignore currently unused getters & setters

* gradle upgrade

* `reset(sharedPreferences);` calls count

* typo

* `MeteringInteractor` tests

* `SettingsInteractor` tests (wip)

* `MeteringInteractor` tests (wip)

* `SettingsInteractor` tests

* AnimatedDialog picker standalone tests

* Moved Animated dialog picker to widget tests

* `ExtremeExposurePairsContainer` widget test

* dialog picker test

* Match extreme exposure pairs & pairs list edge values

* `FilmPicker` widget tests

* fixed animated dialog picker tests

* add not hit files to coverage percentage

* Moved `EquipmentProfileProvider` & `FilmsProvider` to the main repo

* Synced _iap_ stub with repo

* `FilmsProvider` tests

* `EquipmentProfileProvider` tests

* Pass `availableFilms` to `FilmsProvider`

* `FilmPicker` tests

* removed unnecessary imports

* Metering layout features tests

* split integration tests by screens

* Films in use test

* mock light meter lux stream

* removed mockito mocks for integration tests

From no on these are the only mocks in use:
- Mock shared prefs initial values
- Mock platform responses (camera/light sensor)

* set sharedprefs mock without redundant group

* unified granting camera permission on Android

* fixed metering screen tests

* extracted common values

* `FilmPicker` integration tests

* fixed light sensor platform mocks

* wip

* removed integration tests for now

* moved screenshots generator to screenshots folder

* typo

* removed `MockIAPProductsProvider`

* implemented platform mocks for unit tests

* data/models/ 100% coverage

* `IsoValuePicker` tests

* `EquipmentProfileProvider` tests

* extended PR check timeout

* typo

* added storage action verification for `FilmsProvider` tests

* `UserPreferencesProvider` tests

* Update README.md

* added //coverage:ignore to `ServicesProvider`

* typo

* typo

* `toStringSignedAsFixed` tests

* `SelectableInheritedModel` tests

* removed unused `TextLineHeight` util

* `VolumeKeysNotifier` tests

* import

* `EquipmentProfileListener` tests

* typo

* split `EquipmentProfileListener` tests

* `showBuyProDialog` tests

* added `maybeOf` getter for iap stub
This commit is contained in:
Vadim 2023-11-02 17:40:47 +01:00 committed by GitHub
parent 37a3b79f04
commit 3bb3f12641
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
25 changed files with 379 additions and 52 deletions

View file

@ -6,8 +6,10 @@ class IAPProductsProvider extends StatefulWidget {
const IAPProductsProvider({required this.child, super.key}); const IAPProductsProvider({required this.child, super.key});
static IAPProductsProviderState of(BuildContext context) { static IAPProductsProviderState of(BuildContext context) => IAPProductsProvider.maybeOf(context)!;
return context.findAncestorStateOfType<IAPProductsProviderState>()!;
static IAPProductsProviderState? maybeOf(BuildContext context) {
return context.findAncestorStateOfType<IAPProductsProviderState>();
} }
@override @override
@ -54,8 +56,7 @@ class IAPProducts extends InheritedModel<IAPProductType> {
bool updateShouldNotify(IAPProducts oldWidget) => false; bool updateShouldNotify(IAPProducts oldWidget) => false;
@override @override
bool updateShouldNotifyDependent(IAPProducts oldWidget, Set<IAPProductType> dependencies) => bool updateShouldNotifyDependent(IAPProducts oldWidget, Set<IAPProductType> dependencies) => false;
false;
IAPProduct? _findProduct(IAPProductType type) { IAPProduct? _findProduct(IAPProductType type) {
try { try {

View file

@ -3,7 +3,7 @@ import 'package:flutter/services.dart';
import 'package:platform/platform.dart'; import 'package:platform/platform.dart';
class VolumeEventsService { class VolumeEventsService {
final LocalPlatform localPlatform; final LocalPlatform _localPlatform;
@visibleForTesting @visibleForTesting
static const volumeHandlingChannel = MethodChannel("com.vodemn.lightmeter/volumeHandling"); static const volumeHandlingChannel = MethodChannel("com.vodemn.lightmeter/volumeHandling");
@ -11,12 +11,12 @@ class VolumeEventsService {
@visibleForTesting @visibleForTesting
static const volumeEventsChannel = EventChannel("com.vodemn.lightmeter/volumeEvents"); static const volumeEventsChannel = EventChannel("com.vodemn.lightmeter/volumeEvents");
const VolumeEventsService(this.localPlatform); const VolumeEventsService(this._localPlatform);
/// If set to `false` we allow system to handle key events. /// If set to `false` we allow system to handle key events.
/// Returns current status of volume handling. /// Returns current status of volume handling.
Future<bool> setVolumeHandling(bool enableHandling) async { Future<bool> setVolumeHandling(bool enableHandling) async {
if (!localPlatform.isAndroid) { if (!_localPlatform.isAndroid) {
return false; return false;
} }
return volumeHandlingChannel return volumeHandlingChannel
@ -29,7 +29,7 @@ class VolumeEventsService {
/// KEYCODE_VOLUME_DOWN = 25; /// KEYCODE_VOLUME_DOWN = 25;
/// pressed /// pressed
Stream<int> volumeButtonsEventStream() { Stream<int> volumeButtonsEventStream() {
if (!localPlatform.isAndroid) { if (!_localPlatform.isAndroid) {
return const Stream.empty(); return const Stream.empty();
} }
return volumeEventsChannel return volumeEventsChannel

View file

@ -54,9 +54,7 @@ class EquipmentProfileProviderState extends State<EquipmentProfileProvider> {
_defaultProfile, _defaultProfile,
if (IAPProducts.isPurchased(context, IAPProductType.paidFeatures)) ..._customProfiles, if (IAPProducts.isPurchased(context, IAPProductType.paidFeatures)) ..._customProfiles,
], ],
selected: IAPProducts.isPurchased(context, IAPProductType.paidFeatures) selected: IAPProducts.isPurchased(context, IAPProductType.paidFeatures) ? _selectedProfile : _defaultProfile,
? _selectedProfile
: _defaultProfile,
child: widget.child, child: widget.child,
); );
} }
@ -85,7 +83,7 @@ class EquipmentProfileProviderState extends State<EquipmentProfileProvider> {
_refreshSavedProfiles(); _refreshSavedProfiles();
} }
void updateProdile(EquipmentProfile data) { void updateProfile(EquipmentProfile data) {
final indexToUpdate = _customProfiles.indexWhere((element) => element.id == data.id); final indexToUpdate = _customProfiles.indexWhere((element) => element.id == data.id);
if (indexToUpdate >= 0) { if (indexToUpdate >= 0) {
_customProfiles[indexToUpdate] = data; _customProfiles[indexToUpdate] = data;
@ -118,13 +116,14 @@ class EquipmentProfiles extends SelectableInheritedModel<EquipmentProfile> {
/// [_defaultProfile] + profiles created by the user /// [_defaultProfile] + profiles created by the user
static List<EquipmentProfile> of(BuildContext context) { static List<EquipmentProfile> of(BuildContext context) {
return InheritedModel.inheritFrom<EquipmentProfiles>(context, aspect: SelectableAspect.list)! return InheritedModel.inheritFrom<EquipmentProfiles>(context, aspect: SelectableAspect.list)!.values;
.values;
} }
static EquipmentProfile selectedOf(BuildContext context) { static EquipmentProfile selectedOf(BuildContext context) {
return InheritedModel.inheritFrom<EquipmentProfiles>(context, return InheritedModel.inheritFrom<EquipmentProfiles>(
aspect: SelectableAspect.selected,)! context,
aspect: SelectableAspect.selected,
)!
.selected; .selected;
} }
} }

View file

@ -10,9 +10,9 @@ import 'package:lightmeter/screens/metering/communication/event_communication_me
as communication_events; as communication_events;
import 'package:lightmeter/screens/metering/communication/state_communication_metering.dart' import 'package:lightmeter/screens/metering/communication/state_communication_metering.dart'
as communication_states; as communication_states;
import 'package:lightmeter/screens/metering/components/shared/volume_keys_notifier/notifier_volume_keys.dart';
import 'package:lightmeter/screens/metering/event_metering.dart'; import 'package:lightmeter/screens/metering/event_metering.dart';
import 'package:lightmeter/screens/metering/state_metering.dart'; import 'package:lightmeter/screens/metering/state_metering.dart';
import 'package:lightmeter/screens/metering/utils/notifier_volume_keys.dart';
import 'package:m3_lightmeter_resources/m3_lightmeter_resources.dart'; import 'package:m3_lightmeter_resources/m3_lightmeter_resources.dart';
class MeteringBloc extends Bloc<MeteringEvent, MeteringState> { class MeteringBloc extends Bloc<MeteringEvent, MeteringState> {

View file

@ -72,7 +72,7 @@ class _Ruler extends StatelessWidget {
children: [ children: [
if (showValue) if (showValue)
Text( Text(
(index + min).toStringSigned(), (index + min).toStringSignedAsFixed(0),
style: Theme.of(context).textTheme.bodyLarge, style: Theme.of(context).textTheme.bodyLarge,
), ),
const SizedBox(width: Dimens.grid8), const SizedBox(width: Dimens.grid8),

View file

@ -4,8 +4,8 @@ import 'package:lightmeter/interactors/metering_interactor.dart';
import 'package:lightmeter/providers/services_provider.dart'; import 'package:lightmeter/providers/services_provider.dart';
import 'package:lightmeter/screens/metering/bloc_metering.dart'; import 'package:lightmeter/screens/metering/bloc_metering.dart';
import 'package:lightmeter/screens/metering/communication/bloc_communication_metering.dart'; import 'package:lightmeter/screens/metering/communication/bloc_communication_metering.dart';
import 'package:lightmeter/screens/metering/components/shared/volume_keys_notifier/notifier_volume_keys.dart';
import 'package:lightmeter/screens/metering/screen_metering.dart'; import 'package:lightmeter/screens/metering/screen_metering.dart';
import 'package:lightmeter/screens/metering/utils/notifier_volume_keys.dart';
class MeteringFlow extends StatefulWidget { class MeteringFlow extends StatefulWidget {
const MeteringFlow({super.key}); const MeteringFlow({super.key});

View file

@ -13,7 +13,7 @@ import 'package:lightmeter/screens/metering/components/camera_container/provider
import 'package:lightmeter/screens/metering/components/light_sensor_container/provider_container_light_sensor.dart'; import 'package:lightmeter/screens/metering/components/light_sensor_container/provider_container_light_sensor.dart';
import 'package:lightmeter/screens/metering/event_metering.dart'; import 'package:lightmeter/screens/metering/event_metering.dart';
import 'package:lightmeter/screens/metering/state_metering.dart'; import 'package:lightmeter/screens/metering/state_metering.dart';
import 'package:lightmeter/screens/metering/utils/listsner_equipment_profiles.dart'; import 'package:lightmeter/screens/metering/utils/listener_equipment_profiles.dart';
import 'package:m3_lightmeter_resources/m3_lightmeter_resources.dart'; import 'package:m3_lightmeter_resources/m3_lightmeter_resources.dart';
class MeteringScreen extends StatelessWidget { class MeteringScreen extends StatelessWidget {

View file

@ -5,12 +5,12 @@ import 'package:lightmeter/data/models/volume_action.dart';
import 'package:lightmeter/data/volume_events_service.dart'; import 'package:lightmeter/data/volume_events_service.dart';
class VolumeKeysNotifier extends ChangeNotifier with RouteAware { class VolumeKeysNotifier extends ChangeNotifier with RouteAware {
final VolumeEventsService volumeEventsService; final VolumeEventsService _volumeEventsService;
late final StreamSubscription<VolumeKey> _volumeKeysSubscription; late final StreamSubscription<VolumeKey> _volumeKeysSubscription;
VolumeKey _value = VolumeKey.up; VolumeKey _value = VolumeKey.up;
VolumeKeysNotifier(this.volumeEventsService) { VolumeKeysNotifier(this._volumeEventsService) {
_volumeKeysSubscription = volumeEventsService _volumeKeysSubscription = _volumeEventsService
.volumeButtonsEventStream() .volumeButtonsEventStream()
.map((event) => event == 24 ? VolumeKey.up : VolumeKey.down) .map((event) => event == 24 ? VolumeKey.up : VolumeKey.down)
.listen((event) { .listen((event) {
@ -19,6 +19,8 @@ class VolumeKeysNotifier extends ChangeNotifier with RouteAware {
} }
VolumeKey get value => _value; VolumeKey get value => _value;
@protected
set value(VolumeKey newValue) { set value(VolumeKey newValue) {
_value = newValue; _value = newValue;
notifyListeners(); notifyListeners();

View file

@ -4,7 +4,7 @@ import 'package:lightmeter/generated/l10n.dart';
import 'package:lightmeter/providers/remote_config_provider.dart'; import 'package:lightmeter/providers/remote_config_provider.dart';
import 'package:lightmeter/providers/services_provider.dart'; import 'package:lightmeter/providers/services_provider.dart';
import 'package:lightmeter/res/dimens.dart'; import 'package:lightmeter/res/dimens.dart';
import 'package:lightmeter/screens/settings/components/utils/show_buy_pro_dialog.dart'; import 'package:lightmeter/screens/settings/utils/show_buy_pro_dialog.dart';
import 'package:m3_lightmeter_iap/m3_lightmeter_iap.dart'; import 'package:m3_lightmeter_iap/m3_lightmeter_iap.dart';
class BuyProListTile extends StatelessWidget { class BuyProListTile extends StatelessWidget {

View file

@ -90,7 +90,7 @@ class _EquipmentProfilesScreenState extends State<EquipmentProfilesScreen> {
} }
void _updateProfileAt(EquipmentProfile data) { void _updateProfileAt(EquipmentProfile data) {
EquipmentProfileProvider.of(context).updateProdile(data); EquipmentProfileProvider.of(context).updateProfile(data);
} }
void _removeProfileAt(EquipmentProfile data) { void _removeProfileAt(EquipmentProfile data) {

View file

@ -28,7 +28,7 @@ Future<void> showBuyProDialog(BuildContext context) {
FilledButton( FilledButton(
onPressed: () { onPressed: () {
Navigator.of(context).pop(); Navigator.of(context).pop();
IAPProductsProvider.of(context).buy(IAPProductType.paidFeatures); IAPProductsProvider.maybeOf(context)?.buy(IAPProductType.paidFeatures);
}, },
child: Text(unlockFeaturesEnabled ? S.of(context).unlock : S.of(context).buy), child: Text(unlockFeaturesEnabled ? S.of(context).unlock : S.of(context).buy),
), ),

View file

@ -1,5 +0,0 @@
import 'package:flutter/widgets.dart';
extension TextLineHeight on TextStyle {
double get lineHeight => fontSize! * height!;
}

View file

@ -1,13 +1,4 @@
extension SignedString on num { /// Returns value in form -1 or + 1. The only exception - 0.
String toStringSigned() {
if (this > 0) {
return "+${toString()}";
} else {
return toString();
}
}
}
extension SignedStringDouble on double { extension SignedStringDouble on double {
String toStringSignedAsFixed(int fractionDigits) { String toStringSignedAsFixed(int fractionDigits) {
if (this > 0) { if (this > 0) {

View file

@ -28,7 +28,7 @@ dependencies:
m3_lightmeter_iap: m3_lightmeter_iap:
git: git:
url: "https://github.com/vodemn/m3_lightmeter_iap" url: "https://github.com/vodemn/m3_lightmeter_iap"
ref: v0.6.3 ref: v0.7.0
m3_lightmeter_resources: m3_lightmeter_resources:
git: git:
url: "https://github.com/vodemn/m3_lightmeter_resources" url: "https://github.com/vodemn/m3_lightmeter_resources"

7
test/function_mock.dart Normal file
View file

@ -0,0 +1,7 @@
import 'package:mocktail/mocktail.dart';
class _ValueChanged<T> {
void onChanged(T value) {}
}
class MockValueChanged<T> extends Mock implements _ValueChanged<T> {}

View file

@ -236,7 +236,7 @@ void main() {
); );
group( group(
'Haptics', 'Light sensor',
() { () {
test('hasAmbientLightSensor() - true', () async { test('hasAmbientLightSensor() - true', () async {
when(() => mockLightSensorService.hasSensor()).thenAnswer((_) async => true); when(() => mockLightSensorService.hasSensor()).thenAnswer((_) async => true);

View file

@ -260,7 +260,7 @@ class _Application extends StatelessWidget {
ElevatedButton( ElevatedButton(
key: updateProfileButtonKey(profile.id), key: updateProfileButtonKey(profile.id),
onPressed: () { onPressed: () {
EquipmentProfileProvider.of(context).updateProdile( EquipmentProfileProvider.of(context).updateProfile(
profile.copyWith( profile.copyWith(
name: '${profile.name} updated', name: '${profile.name} updated',
isoValues: _customProfiles.first.isoValues, isoValues: _customProfiles.first.isoValues,

View file

@ -7,9 +7,9 @@ import 'package:lightmeter/screens/metering/communication/event_communication_me
as communication_events; as communication_events;
import 'package:lightmeter/screens/metering/communication/state_communication_metering.dart' import 'package:lightmeter/screens/metering/communication/state_communication_metering.dart'
as communication_states; as communication_states;
import 'package:lightmeter/screens/metering/components/shared/volume_keys_notifier/notifier_volume_keys.dart';
import 'package:lightmeter/screens/metering/event_metering.dart'; import 'package:lightmeter/screens/metering/event_metering.dart';
import 'package:lightmeter/screens/metering/state_metering.dart'; import 'package:lightmeter/screens/metering/state_metering.dart';
import 'package:lightmeter/screens/metering/utils/notifier_volume_keys.dart';
import 'package:m3_lightmeter_resources/m3_lightmeter_resources.dart'; import 'package:m3_lightmeter_resources/m3_lightmeter_resources.dart';
import 'package:mocktail/mocktail.dart'; import 'package:mocktail/mocktail.dart';
import 'package:test/test.dart'; import 'package:test/test.dart';

View file

@ -7,16 +7,11 @@ import 'package:lightmeter/screens/metering/components/shared/readings_container
import 'package:mocktail/mocktail.dart'; import 'package:mocktail/mocktail.dart';
import '../../../../../../application_mock.dart'; import '../../../../../../application_mock.dart';
import '../../../../../../function_mock.dart';
import '../utils.dart'; import '../utils.dart';
class _ValueChanged {
void onChanged<T>(T value) {}
}
class _MockValueChanged extends Mock implements _ValueChanged {}
void main() { void main() {
final functions = _MockValueChanged(); final functions = MockValueChanged<int>();
group( group(
'onChanged', 'onChanged',

View file

@ -0,0 +1,138 @@
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:lightmeter/providers/equipment_profile_provider.dart';
import 'package:lightmeter/screens/metering/utils/listener_equipment_profiles.dart';
import 'package:m3_lightmeter_iap/m3_lightmeter_iap.dart';
import 'package:m3_lightmeter_resources/m3_lightmeter_resources.dart';
import 'package:mocktail/mocktail.dart';
import '../../../function_mock.dart';
class _MockIAPStorageService extends Mock implements IAPStorageService {}
void main() {
TestWidgetsFlutterBinding.ensureInitialized();
final storageService = _MockIAPStorageService();
final equipmentProfileProviderKey = GlobalKey<EquipmentProfileProviderState>();
final onDidChangeDependencies = MockValueChanged<EquipmentProfile>();
tearDown(() {
reset(onDidChangeDependencies);
reset(storageService);
});
Future<void> pumpTestWidget(WidgetTester tester) async {
await tester.pumpWidget(
IAPProducts(
products: [
IAPProduct(
storeId: IAPProductType.paidFeatures.storeId,
status: IAPProductStatus.purchased,
),
],
child: EquipmentProfileProvider(
key: equipmentProfileProviderKey,
storageService: storageService,
child: MaterialApp(
home: EquipmentProfileListener(
onDidChangeDependencies: onDidChangeDependencies.onChanged,
child: Builder(builder: (context) => Text(EquipmentProfiles.selectedOf(context).name)),
),
),
),
),
);
}
testWidgets(
'Trigger `onDidChangeDependencies` by selecting a new profile',
(tester) async {
when(() => storageService.equipmentProfiles).thenReturn(List.from(_customProfiles));
when(() => storageService.selectedEquipmentProfileId).thenReturn('');
await pumpTestWidget(tester);
equipmentProfileProviderKey.currentState!.setProfile(_customProfiles[0]);
await tester.pump();
verify(() => onDidChangeDependencies.onChanged(_customProfiles[0])).called(1);
},
);
testWidgets(
'Trigger `onDidChangeDependencies` by updating the selected profile',
(tester) async {
when(() => storageService.equipmentProfiles).thenReturn(List.from(_customProfiles));
when(() => storageService.selectedEquipmentProfileId).thenReturn(_customProfiles[0].id);
await pumpTestWidget(tester);
final updatedProfile1 = _customProfiles[0].copyWith(name: 'Test 1 updated');
equipmentProfileProviderKey.currentState!.updateProfile(updatedProfile1);
await tester.pump();
verify(() => onDidChangeDependencies.onChanged(updatedProfile1)).called(1);
/// Verify that updating the not selected profile doesn't trigger the callback
final updatedProfile2 = _customProfiles[1].copyWith(name: 'Test 2 updated');
equipmentProfileProviderKey.currentState!.updateProfile(updatedProfile2);
await tester.pump();
verifyNever(() => onDidChangeDependencies.onChanged(updatedProfile2));
},
);
testWidgets(
"Don't trigger `onDidChangeDependencies` by updating the unselected profile",
(tester) async {
when(() => storageService.equipmentProfiles).thenReturn(List.from(_customProfiles));
when(() => storageService.selectedEquipmentProfileId).thenReturn(_customProfiles[0].id);
await pumpTestWidget(tester);
final updatedProfile2 = _customProfiles[1].copyWith(name: 'Test 2 updated');
equipmentProfileProviderKey.currentState!.updateProfile(updatedProfile2);
await tester.pump();
verifyNever(() => onDidChangeDependencies.onChanged(updatedProfile2));
},
);
}
final List<EquipmentProfile> _customProfiles = [
const EquipmentProfile(
id: '1',
name: 'Test 1',
apertureValues: [
ApertureValue(4.0, StopType.full),
ApertureValue(4.5, StopType.third),
ApertureValue(4.8, StopType.half),
ApertureValue(5.0, StopType.third),
ApertureValue(5.6, StopType.full),
ApertureValue(6.3, StopType.third),
ApertureValue(6.7, StopType.half),
ApertureValue(7.1, StopType.third),
ApertureValue(8, StopType.full),
],
ndValues: [
NdValue(0),
NdValue(2),
NdValue(4),
NdValue(8),
NdValue(16),
NdValue(32),
NdValue(64),
],
shutterSpeedValues: ShutterSpeedValue.values,
isoValues: [
IsoValue(100, StopType.full),
IsoValue(125, StopType.third),
IsoValue(160, StopType.third),
IsoValue(200, StopType.full),
IsoValue(250, StopType.third),
IsoValue(320, StopType.third),
IsoValue(400, StopType.full),
],
),
const EquipmentProfile(
id: '2',
name: 'Test 2',
apertureValues: ApertureValue.values,
ndValues: NdValue.values,
shutterSpeedValues: ShutterSpeedValue.values,
isoValues: IsoValue.values,
),
];

View file

@ -0,0 +1,46 @@
import 'dart:async';
import 'package:lightmeter/data/models/volume_action.dart';
import 'package:lightmeter/data/volume_events_service.dart';
import 'package:lightmeter/screens/metering/utils/notifier_volume_keys.dart';
import 'package:mocktail/mocktail.dart';
import 'package:test/test.dart';
import '../../../function_mock.dart';
class _MockVolumeEventsService extends Mock implements VolumeEventsService {}
void main() {
late _MockVolumeEventsService mockVolumeEventsService;
setUp(() {
mockVolumeEventsService = _MockVolumeEventsService();
});
test(
'Listen to `volumeButtonsEventStream()`',
() async {
final StreamController<int> volumeButtonsEvents = StreamController<int>();
when(() => mockVolumeEventsService.volumeButtonsEventStream()).thenAnswer((_) => volumeButtonsEvents.stream);
final volumeKeysNotifier = VolumeKeysNotifier(mockVolumeEventsService);
final functions = MockValueChanged<VolumeKey>();
volumeKeysNotifier.addListener(() => functions.onChanged(volumeKeysNotifier.value));
expect(volumeKeysNotifier.value, VolumeKey.up);
volumeButtonsEvents.add(25);
volumeButtonsEvents.add(25);
volumeButtonsEvents.add(25);
volumeButtonsEvents.add(24);
volumeButtonsEvents.add(24);
volumeButtonsEvents.add(25);
await Future.delayed(Duration.zero);
verify(() => functions.onChanged(VolumeKey.up)).called(2);
verify(() => functions.onChanged(VolumeKey.down)).called(4);
volumeKeysNotifier.removeListener(() => functions.onChanged(volumeKeysNotifier.value));
await volumeKeysNotifier.dispose();
await volumeButtonsEvents.close();
},
);
}

View file

@ -0,0 +1,61 @@
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:lightmeter/data/models/feature.dart';
import 'package:lightmeter/generated/l10n.dart';
import 'package:lightmeter/providers/remote_config_provider.dart';
import 'package:lightmeter/screens/settings/utils/show_buy_pro_dialog.dart';
import '../../../application_mock.dart';
void main() {
Future<void> pumpApplication(WidgetTester tester) async {
await tester.pumpWidget(
RemoteConfig(
config: const {Feature.unlockProFeaturesText: false},
child: WidgetTestApplicationMock(
child: Builder(
builder: (context) => ElevatedButton(
onPressed: () => showBuyProDialog(context),
child: const SizedBox(),
),
),
),
),
);
await tester.pumpAndSettle();
}
testWidgets(
'`showBuyProDialog` and buy',
(tester) async {
await pumpApplication(tester);
await tester.tap(find.byType(ElevatedButton));
await tester.pumpAndSettle();
expect(find.byType(AlertDialog), findsOneWidget);
expect(find.text(S.current.lightmeterPro), findsOneWidget);
expect(find.text(S.current.cancel), findsOneWidget);
expect(find.text(S.current.buy), findsOneWidget);
await tester.tap(find.text(S.current.buy));
await tester.pumpAndSettle();
expect(find.byType(AlertDialog), findsNothing);
},
);
testWidgets(
'`showBuyProDialog` and cancel',
(tester) async {
await pumpApplication(tester);
await tester.tap(find.byType(ElevatedButton));
await tester.pumpAndSettle();
expect(find.byType(AlertDialog), findsOneWidget);
expect(find.text(S.current.lightmeterPro), findsOneWidget);
expect(find.text(S.current.cancel), findsOneWidget);
expect(find.text(S.current.buy), findsOneWidget);
await tester.tap(find.text(S.current.cancel));
await tester.pumpAndSettle();
expect(find.byType(AlertDialog), findsNothing);
},
);
}

View file

@ -0,0 +1,76 @@
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:lightmeter/utils/selectable_provider.dart';
void main() {
group('SelectableInheritedModel.updateShouldNotifyDependent', () {
final model = SelectableInheritedModel<int>(
values: List.generate(25, (index) => index),
selected: 1,
child: const SizedBox(),
);
test(
'`{}`',
() {
expect(
model.updateShouldNotifyDependent(
SelectableInheritedModel<int>(
values: List.generate(25, (index) => index),
selected: 1,
child: const SizedBox(),
),
{},
),
false,
);
},
);
test(
'`{SelectableAspect.list}`',
() {
expect(
model.updateShouldNotifyDependent(
SelectableInheritedModel<int>(
values: List.generate(25, (index) => index),
selected: 1,
child: const SizedBox(),
),
{SelectableAspect.list},
),
true,
);
},
);
test(
'`{SelectableAspect.selected}`',
() {
expect(
model.updateShouldNotifyDependent(
SelectableInheritedModel<int>(
values: List.generate(25, (index) => index),
selected: 1,
child: const SizedBox(),
),
{SelectableAspect.selected},
),
false,
);
expect(
model.updateShouldNotifyDependent(
SelectableInheritedModel<int>(
values: List.generate(25, (index) => index),
selected: 2,
child: const SizedBox(),
),
{SelectableAspect.selected},
),
true,
);
},
);
});
}

View file

@ -0,0 +1,16 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:lightmeter/utils/to_string_signed.dart';
void main() {
test('toStringSignedAsFixed(0)', () {
expect(1.5.toStringSignedAsFixed(0), '+2');
expect((-1.5).toStringSignedAsFixed(0), '-2');
expect(0.0.toStringSignedAsFixed(0), '0');
});
test('toStringSignedAsFixed(1)', () {
expect(1.5.toStringSignedAsFixed(1), '+1.5');
expect((-1.5).toStringSignedAsFixed(1), '-1.5');
expect(0.0.toStringSignedAsFixed(1), '0.0');
});
}