import 'package:flutter/material.dart'; import 'package:flutter_test/flutter_test.dart'; import 'package:integration_test/integration_test.dart'; import 'package:lightmeter/application.dart'; import 'package:lightmeter/data/caffeine_service.dart'; import 'package:lightmeter/data/haptics_service.dart'; import 'package:lightmeter/data/light_sensor_service.dart'; import 'package:lightmeter/data/models/ev_source_type.dart'; import 'package:lightmeter/data/models/metering_screen_layout_config.dart'; import 'package:lightmeter/data/models/supported_locale.dart'; import 'package:lightmeter/data/models/theme_type.dart'; import 'package:lightmeter/data/models/volume_action.dart'; import 'package:lightmeter/data/permissions_service.dart'; import 'package:lightmeter/data/shared_prefs_service.dart'; import 'package:lightmeter/data/volume_events_service.dart'; import 'package:lightmeter/environment.dart'; import 'package:lightmeter/generated/l10n.dart'; import 'package:lightmeter/providers/services_provider.dart'; import 'package:lightmeter/providers/user_preferences_provider.dart'; import 'package:lightmeter/res/theme.dart'; import 'package:lightmeter/screens/metering/components/bottom_controls/components/measure_button/widget_button_measure.dart'; import 'package:lightmeter/screens/metering/components/shared/exposure_pairs_list/widget_list_exposure_pairs.dart'; import 'package:lightmeter/screens/metering/components/shared/readings_container/components/extreme_exposure_pairs_container/widget_container_extreme_exposure_pairs.dart'; import 'package:lightmeter/screens/metering/components/shared/readings_container/components/film_picker/widget_picker_film.dart'; import 'package:lightmeter/screens/metering/components/shared/readings_container/components/iso_picker/widget_picker_iso.dart'; import 'package:lightmeter/screens/metering/components/shared/readings_container/components/nd_picker/widget_picker_nd.dart'; import 'package:lightmeter/screens/metering/components/shared/readings_container/components/shared/animated_dialog_picker/components/dialog_picker/widget_picker_dialog.dart'; import 'package:lightmeter/screens/metering/screen_metering.dart'; import 'package:lightmeter/screens/shared/icon_placeholder/widget_icon_placeholder.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 'package:permission_handler/permission_handler.dart'; import 'mocks/paid_features_mock.dart'; import 'utils/expectations.dart'; import 'utils/widget_tester_actions.dart'; class _MockUserPreferencesService extends Mock implements UserPreferencesService {} class _MockCaffeineService extends Mock implements CaffeineService {} class _MockHapticsService extends Mock implements HapticsService {} class _MockPermissionsService extends Mock implements PermissionsService {} class _MockLightSensorService extends Mock implements LightSensorService {} class _MockVolumeEventsService extends Mock implements VolumeEventsService {} const _defaultIsoValue = IsoValue(400, StopType.full); //https://stackoverflow.com/a/67186625/13167574 void main() { IntegrationTestWidgetsFlutterBinding.ensureInitialized(); late _MockUserPreferencesService mockUserPreferencesService; late _MockCaffeineService mockCaffeineService; late _MockHapticsService mockHapticsService; late _MockPermissionsService mockPermissionsService; late _MockLightSensorService mockLightSensorService; late _MockVolumeEventsService mockVolumeEventsService; setUpAll(() { mockUserPreferencesService = _MockUserPreferencesService(); when(() => mockUserPreferencesService.evSourceType).thenReturn(EvSourceType.sensor); when(() => mockUserPreferencesService.stopType).thenReturn(StopType.third); when(() => mockUserPreferencesService.locale).thenReturn(SupportedLocale.en); when(() => mockUserPreferencesService.caffeine).thenReturn(true); when(() => mockUserPreferencesService.volumeAction).thenReturn(VolumeAction.shutter); when(() => mockUserPreferencesService.cameraEvCalibration).thenReturn(0.0); when(() => mockUserPreferencesService.lightSensorEvCalibration).thenReturn(0.0); when(() => mockUserPreferencesService.iso).thenReturn(_defaultIsoValue); when(() => mockUserPreferencesService.ndFilter).thenReturn(NdValue.values.first); when(() => mockUserPreferencesService.haptics).thenReturn(true); when(() => mockUserPreferencesService.meteringScreenLayout).thenReturn({ MeteringScreenLayoutFeature.equipmentProfiles: true, MeteringScreenLayoutFeature.extremeExposurePairs: true, MeteringScreenLayoutFeature.filmPicker: true, MeteringScreenLayoutFeature.histogram: false, }); when(() => mockUserPreferencesService.themeType).thenReturn(ThemeType.light); when(() => mockUserPreferencesService.primaryColor).thenReturn(primaryColorsList[5]); when(() => mockUserPreferencesService.dynamicColor).thenReturn(false); mockCaffeineService = _MockCaffeineService(); when(() => mockCaffeineService.isKeepScreenOn()).thenAnswer((_) async => false); when(() => mockCaffeineService.keepScreenOn(true)).thenAnswer((_) async => true); when(() => mockCaffeineService.keepScreenOn(false)).thenAnswer((_) async => false); mockHapticsService = _MockHapticsService(); when(() => mockHapticsService.quickVibration()).thenAnswer((_) async {}); when(() => mockHapticsService.responseVibration()).thenAnswer((_) async {}); when(() => mockHapticsService.errorVibration()).thenAnswer((_) async {}); mockPermissionsService = _MockPermissionsService(); when(() => mockPermissionsService.requestCameraPermission()).thenAnswer((_) async => PermissionStatus.granted); when(() => mockPermissionsService.checkCameraPermission()).thenAnswer((_) async => PermissionStatus.granted); mockLightSensorService = _MockLightSensorService(); when(() => mockLightSensorService.hasSensor()).thenAnswer((_) async => true); when(() => mockLightSensorService.luxStream()).thenAnswer((_) => Stream.fromIterable([100])); mockVolumeEventsService = _MockVolumeEventsService(); when(() => mockVolumeEventsService.setVolumeHandling(true)).thenAnswer((_) async => true); when(() => mockVolumeEventsService.setVolumeHandling(false)).thenAnswer((_) async => false); when(() => mockVolumeEventsService.volumeButtonsEventStream()).thenAnswer((_) => const Stream.empty()); when(() => mockHapticsService.quickVibration()).thenAnswer((_) async {}); when(() => mockHapticsService.responseVibration()).thenAnswer((_) async {}); }); Future pumpApplication( WidgetTester tester, IAPProductStatus purchaseStatus, { String selectedEquipmentProfileId = '', Film selectedFilm = const Film.other(), }) async { await tester.pumpWidget( MockIAPProviders( purchaseStatus: purchaseStatus, selectedEquipmentProfileId: selectedEquipmentProfileId, selectedFilm: selectedFilm, child: ServicesProvider( environment: const Environment.prod().copyWith(hasLightSensor: true), userPreferencesService: mockUserPreferencesService, caffeineService: mockCaffeineService, hapticsService: mockHapticsService, permissionsService: mockPermissionsService, lightSensorService: mockLightSensorService, volumeEventsService: mockVolumeEventsService, child: const UserPreferencesProvider( child: Application(), ), ), ), ); await tester.pumpAndSettle(); } group( 'Match extreme exposure pairs & pairs list edge values', () { void expectExposurePairsListItem(int index, String aperture, String shutterSpeed) { final firstPairRow = find.byWidgetPredicate( (widget) => widget is Row && widget.key == ValueKey(index), ); expect( find.descendant(of: firstPairRow, matching: find.text(aperture)), findsOneWidget, ); expect( find.descendant(of: firstPairRow, matching: find.text(shutterSpeed)), findsOneWidget, ); } setUpAll(() { when(() => mockUserPreferencesService.evSourceType).thenReturn(EvSourceType.sensor); when(() => mockLightSensorService.luxStream()).thenAnswer((_) => Stream.fromIterable([100])); }); testWidgets( 'No exposure pairs', (tester) async { await pumpApplication(tester, IAPProductStatus.purchasable); final pickerFinder = find.byType(ExtremeExposurePairsContainer); expect(pickerFinder, findsOneWidget); expect( find.descendant(of: pickerFinder, matching: find.text(S.current.fastestExposurePair)), findsOneWidget, ); expect( find.descendant(of: pickerFinder, matching: find.text(S.current.slowestExposurePair)), findsOneWidget, ); expect(find.descendant(of: pickerFinder, matching: find.text('-')), findsNWidgets(2)); expect( find.descendant( of: find.byType(ExposurePairsList), matching: find.byWidgetPredicate( (widget) => widget is IconPlaceholder && widget.icon == Icons.not_interested && widget.text == S.current.noExposurePairs, ), ), findsOneWidget, ); }, ); testWidgets( 'Multiple exposure pairs w/o reciprocity', (tester) async { await pumpApplication(tester, IAPProductStatus.purchasable); await tester.toggleIncidentMetering(); expectExposurePairsContainer('f/1.0 - 1/160', 'f/45 - 13"'); expectExposurePairsListItem(0, 'f/1.0', '1/160'); expectMeasureButton(7.3); final exposurePairs = MeteringContainerBuidler.buildExposureValues( 7.3, StopType.third, defaultEquipmentProfile, ); await tester.scrollUntilVisible( find.byWidgetPredicate( (widget) => widget is Row && widget.key == ValueKey(exposurePairs.length - 1), ), 56, scrollable: find.descendant( of: find.byType(ExposurePairsList), matching: find.byType(Scrollable), ), ); expectExposurePairsListItem(exposurePairs.length - 1, 'f/45', '13"'); expectMeasureButton(7.3); }, ); testWidgets( 'Multiple exposure pairs w/ reciprocity', (tester) async { await pumpApplication(tester, IAPProductStatus.purchased, selectedFilm: mockFilms.first); await tester.toggleIncidentMetering(); expectExposurePairsContainer('f/1.0 - 1/160', 'f/45 - 26"'); expectExposurePairsListItem(0, 'f/1.0', '1/160'); expectMeasureButton(7.3); final exposurePairs = MeteringContainerBuidler.buildExposureValues( 7.3, StopType.third, defaultEquipmentProfile, ); await tester.scrollUntilVisible( find.byWidgetPredicate( (widget) => widget is Row && widget.key == ValueKey(exposurePairs.length - 1), ), 56, scrollable: find.descendant( of: find.byType(ExposurePairsList), matching: find.byType(Scrollable), ), ); expectExposurePairsListItem(exposurePairs.length - 1, 'f/45', '26"'); expectMeasureButton(7.3); }, ); }, skip: true, ); group( 'Pickers tests', () { group('Select film', () { testWidgets( 'with the same ISO', (tester) async { await pumpApplication(tester, IAPProductStatus.purchased); expectExposurePairsContainer('f/1.0 - 1/160', 'f/45 - 13"'); expectMeasureButton(7.3); await tester.openAnimatedPicker(); expect(find.byType(DialogPicker), findsOneWidget); await tester.tapRadioListTile('2'); await tester.tapSelectButton(); expectExposurePairsContainer('f/1.0 - 1/160', 'f/45 - 13"'); expectMeasureButton(7.3); /// Make sure, that nothing is changed await tester.tap(find.byType(MeteringMeasureButton)); await tester.tap(find.byType(MeteringMeasureButton)); await tester.pumpAndSettle(); expectExposurePairsContainer('f/1.0 - 1/160', 'f/45 - 13"'); expectMeasureButton(7.3); }, ); }); testWidgets( 'Select ISO +1 EV', (tester) async { await pumpApplication(tester, IAPProductStatus.purchased); expectExposurePairsContainer('f/1.0 - 1/160', 'f/45 - 13"'); expectMeasureButton(7.3); await tester.openAnimatedPicker(); expect(find.byType(DialogPicker), findsOneWidget); await tester.tapRadioListTile('800'); await tester.tapSelectButton(); expectExposurePairsContainer('f/1.0 - 1/320', 'f/45 - 6"'); expectMeasureButton(8.3); /// Make sure, that current ISO is used in metering await tester.tap(find.byType(MeteringMeasureButton)); await tester.tap(find.byType(MeteringMeasureButton)); await tester.pumpAndSettle(); expectExposurePairsContainer('f/1.0 - 1/320', 'f/45 - 6"'); expectMeasureButton(8.3); }, ); testWidgets( 'Select ND -1 EV', (tester) async { await pumpApplication(tester, IAPProductStatus.purchased); expectExposurePairsContainer('f/1.0 - 1/160', 'f/45 - 13"'); expectMeasureButton(7.3); await tester.openAnimatedPicker(); expect(find.byType(DialogPicker), findsOneWidget); await tester.tapRadioListTile('2'); await tester.tapSelectButton(); expectExposurePairsContainer('f/1.0 - 1/80', 'f/36 - 16"'); expectMeasureButton(6.3); /// Make sure, that current ISO is used in metering await tester.tap(find.byType(MeteringMeasureButton)); await tester.tap(find.byType(MeteringMeasureButton)); await tester.pumpAndSettle(); expectExposurePairsContainer('f/1.0 - 1/80', 'f/36 - 16"'); expectMeasureButton(6.3); }, ); }, skip: true, ); } extension _WidgetTesterActions on WidgetTester { Future tapRadioListTile(String value) async { expect(find.descendant(of: find.byType(RadioListTile), matching: find.text(value)), findsOneWidget); await tap(find.descendant(of: find.byType(RadioListTile), matching: find.text(value))); } }