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/dimens.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/equipment_profile_picker/widget_picker_equipment_profiles.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/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'; 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.camera); 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) async { await tester.pumpWidget( MockIAPProviders( purchaseStatus: purchaseStatus, 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', () { 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', (tester) async { await pumpApplication(tester, IAPProductStatus.purchasable); await tester.tap(find.byType(MeteringMeasureButton)); await tester.tap(find.byType(MeteringMeasureButton)); await tester.pumpAndSettle(); 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('f/1.0 - 1/160')), findsOneWidget); expect(find.descendant(of: pickerFinder, matching: find.text('f/45 - 13"')), findsOneWidget); final firstPairRow = find.byWidgetPredicate((widget) => widget is Row && widget.key == const ValueKey(0)); expect(find.descendant(of: firstPairRow, matching: find.text('f/1.0')), findsOneWidget); expect(find.descendant(of: firstPairRow, matching: find.text('1/160')), findsOneWidget); final lastPairRow = find.byWidgetPredicate( (widget) => widget is Row && widget.key == ValueKey(ApertureValue.values.whereStopType(StopType.third).length - 1), ); final listFinder = find.descendant(of: find.byType(ExposurePairsList), matching: find.byType(Scrollable)); await tester.scrollUntilVisible(lastPairRow, 56, scrollable: listFinder); expect(find.descendant(of: lastPairRow, matching: find.text('f/45')), findsOneWidget); expect(find.descendant(of: lastPairRow, matching: find.text('13"')), findsOneWidget); }, ); }); group( 'Free version', () { testWidgets( 'Initial state', (tester) async { await pumpApplication(tester, IAPProductStatus.purchasable); expectAnimatedPicker(S.current.equipmentProfile, S.current.none); expectAnimatedPicker(S.current.film, S.current.none); expectAnimatedPicker(S.current.iso, '400'); expectAnimatedPicker(S.current.nd, S.current.none); await tester.openAnimatedPicker(); expect(find.byType(DialogPicker), findsOneWidget); // expect None selected and no other profiles present await tester.tapCancelButton(); expect(find.byType(DialogPicker), findsNothing); await tester.openAnimatedPicker(); await tester.openAnimatedPicker(); await tester.openAnimatedPicker(); }, ); }, skip: true, ); group( 'Pro version', () { testWidgets( 'Initial state', (tester) async { await pumpApplication(tester, IAPProductStatus.purchased); }, ); testWidgets( 'Film (push/pull)', (tester) async { await pumpApplication(tester, IAPProductStatus.purchased); }, ); }, skip: true, ); } extension WidgetTesterActions on WidgetTester { Future openAnimatedPicker() async { await tap(find.byType(T)); await pumpAndSettle(Dimens.durationL); } Future tapSelectButton() async { final cancelButton = find.byWidgetPredicate( (widget) => widget is TextButton && widget.child is Text && (widget.child as Text?)?.data == S.current.select, ); expect(cancelButton, findsOneWidget); await tap(cancelButton); await pumpAndSettle(); } Future tapCancelButton() async { final cancelButton = find.byWidgetPredicate( (widget) => widget is TextButton && widget.child is Text && (widget.child as Text?)?.data == S.current.cancel, ); expect(cancelButton, findsOneWidget); await tap(cancelButton); await pumpAndSettle(); } }