diff --git a/iap/lib/src/data/models/iap_product.dart b/iap/lib/src/data/models/iap_product.dart index b075ff6..4ee2b24 100644 --- a/iap/lib/src/data/models/iap_product.dart +++ b/iap/lib/src/data/models/iap_product.dart @@ -9,15 +9,18 @@ enum IAPProductType { paidFeatures } class IAPProduct { final String storeId; final IAPProductStatus status; + final String price; const IAPProduct({ required this.storeId, this.status = IAPProductStatus.purchasable, + required this.price, }); IAPProduct copyWith({IAPProductStatus? status}) => IAPProduct( storeId: storeId, status: status ?? this.status, + price: price, ); } diff --git a/iap/lib/src/providers/iap_products_provider.dart b/iap/lib/src/providers/iap_products_provider.dart index 2c4ca5c..936f296 100644 --- a/iap/lib/src/providers/iap_products_provider.dart +++ b/iap/lib/src/providers/iap_products_provider.dart @@ -25,6 +25,7 @@ class IAPProductsProviderState extends State { IAPProduct( storeId: IAPProductType.paidFeatures.storeId, status: IAPProductStatus.purchased, + price: '0.0\$', ) ], child: widget.child, diff --git a/integration_test/mocks/iap_products_mock.dart b/integration_test/mocks/iap_products_mock.dart index 1a79bc2..9742c8a 100644 --- a/integration_test/mocks/iap_products_mock.dart +++ b/integration_test/mocks/iap_products_mock.dart @@ -27,6 +27,7 @@ class MockIAPProductsProviderState extends State { IAPProduct( storeId: IAPProductType.paidFeatures.storeId, status: _purchased ? IAPProductStatus.purchased : IAPProductStatus.purchasable, + price: '0.0\$', ), ]), child: widget.child, diff --git a/lib/application.dart b/lib/application.dart index 7512644..b8939e4 100644 --- a/lib/application.dart +++ b/lib/application.dart @@ -5,6 +5,7 @@ import 'package:lightmeter/data/models/supported_locale.dart'; import 'package:lightmeter/generated/l10n.dart'; import 'package:lightmeter/platform_config.dart'; import 'package:lightmeter/providers/user_preferences_provider.dart'; +import 'package:lightmeter/screens/lightmeter_pro/screen_lightmeter_pro.dart'; import 'package:lightmeter/screens/metering/flow_metering.dart'; import 'package:lightmeter/screens/settings/flow_settings.dart'; import 'package:lightmeter/screens/shared/release_notes_dialog/flow_dialog_release_notes.dart'; @@ -44,6 +45,7 @@ class Application extends StatelessWidget { routes: { "metering": (_) => const ReleaseNotesFlow(child: MeteringFlow()), "settings": (_) => const SettingsFlow(), + "lightmeterPro": (_) => LightmeterProScreen(), "timer": (context) => TimerFlow(args: ModalRoute.of(context)!.settings.arguments! as TimerFlowArgs), }, ), diff --git a/lib/data/models/app_feature.dart b/lib/data/models/app_feature.dart new file mode 100644 index 0000000..f9be63a --- /dev/null +++ b/lib/data/models/app_feature.dart @@ -0,0 +1,56 @@ +import 'package:flutter/material.dart'; +import 'package:lightmeter/generated/l10n.dart'; + +enum AppFeature { + reflectedLightMetering, + incidedntLightMetering, + isoAndNdValues, + themeEngine, + spotMetering, + histogram, + listOfFilms, + equipmentProfiles, + timer, + mainScreenCustomization; + + static List get androidFeatures => values; + + static List get iosFeatures => values.where((f) => f != AppFeature.incidedntLightMetering).toList(); + + String name(BuildContext context) { + switch (this) { + case AppFeature.reflectedLightMetering: + return S.of(context).featureReflectedLightMetering; + case AppFeature.incidedntLightMetering: + return S.of(context).featureIncidentLightMetering; + case AppFeature.isoAndNdValues: + return S.of(context).featureIsoAndNdValues; + case AppFeature.themeEngine: + return S.of(context).featureTheme; + case AppFeature.spotMetering: + return S.of(context).featureSpotMetering; + case AppFeature.histogram: + return S.of(context).featureHistogram; + case AppFeature.listOfFilms: + return S.of(context).featureListOfFilms; + case AppFeature.equipmentProfiles: + return S.of(context).featureEquipmentProfiles; + case AppFeature.timer: + return S.of(context).featureTimer; + case AppFeature.mainScreenCustomization: + return S.of(context).featureMeteringScreenLayout; + } + } + + bool get isFree { + switch (this) { + case AppFeature.reflectedLightMetering: + case AppFeature.incidedntLightMetering: + case AppFeature.isoAndNdValues: + case AppFeature.themeEngine: + return true; + default: + return false; + } + } +} diff --git a/lib/l10n/intl_en.arb b/lib/l10n/intl_en.arb index 0b973c9..14f3950 100644 --- a/lib/l10n/intl_en.arb +++ b/lib/l10n/intl_en.arb @@ -103,10 +103,31 @@ } } }, - "proFeatures": "Pro features", - "unlockProFeatures": "Unlock Pro features", - "unlockProFeaturesDescription": "Unlock professional features:\n \u2022 Equipment profiles containing filters for aperture, shutter speed, and more\n \u2022 List of films with compensation for what's known as reciprocity failure\n \u2022 Spot metering & Histogram\n \u2022 And more!\n\nBy unlocking Pro features you support the development and make it possible to add new features to the app.", - "unlock": "Unlock", + "proFeaturesTitle": "Lightmeter Pro", + "getPro": "Get Pro", + "featuresFree": "Free", + "featuresPro": "Pro", + "proFeaturesPromoText": "Lightmeter Pro delivers everything you need to get the best shots!", + "proFeaturesWhatsIncluded": "What's included?", + "featureReflectedLightMetering": "Reflected light metering", + "featureIncidentLightMetering": "Incident light metering", + "featureIsoAndNdValues": "Wide range of ISO and ND filters values", + "featureTheme": "Theme customization", + "featureSpotMetering": "Spot metering", + "featureHistogram": "Histogram", + "featureListOfFilms": "List of 20+ films with reciprocity formulas", + "featureEquipmentProfiles": "Equipment profiles", + "featureTimer": "Built-in timer for long exposure", + "featureMeteringScreenLayout": "Customizable main screen", + "proFeaturesSupportText": "By purchasing Lightmeter Pro you support the development and make it possible to add new features to the app.", + "getNowFor": "Get now for {price}", + "@getNowFor": { + "price": { + "version": { + "type": "String" + } + } + }, "tooltipAdd": "Add", "tooltipClose": "Close", "tooltipExpand": "Expand", diff --git a/lib/l10n/intl_fr.arb b/lib/l10n/intl_fr.arb index 3d0a684..e002a0c 100644 --- a/lib/l10n/intl_fr.arb +++ b/lib/l10n/intl_fr.arb @@ -103,10 +103,32 @@ } } }, - "proFeatures": "Fonctionnalités professionnelles", - "unlockProFeatures": "Déverrouiller les fonctionnalités professionnelles", - "unlockProFeaturesDescription": "Déverrouillez des fonctions professionnelles:\n \u2022 Profils d'équipement contenant des filtres pour l'ouverture, la vitesse d'obturation et plus encore, ainsi qu'une liste de films avec compensation pour ce que l'on appelle l'échec de réciprocité\n \u2022 Mesure spot & Histogramme\n \u2022 Et plus encore!\n\nEn débloquant les fonctionnalités Pro, vous soutenez le développement et permettez d'ajouter de nouvelles fonctionnalités à l'application.", + "proFeaturesTitle": "Lightmeter Pro", + "getPro": "Acheter Pro", "unlock": "Déverrouiller", + "featuresFree": "Gratuit", + "featuresPro": "Pro", + "proFeaturesPromoText": "Lightmeter Pro offre tout ce dont vous avez besoin pour obtenir les meilleurs clichés!", + "proFeaturesWhatsIncluded": "Qu'est-ce qui est inclus?", + "featureReflectedLightMetering": "Mesure de la lumière réfléchie", + "featureIncidentLightMetering": "Mesure de la lumière incidente", + "featureIsoAndNdValues": "Large gamme de valeurs ISO et de filtres ND", + "featureTheme": "Personnalisation du thème", + "featureSpotMetering": "Mesure spot", + "featureHistogram": "Histogramme", + "featureListOfFilms": "Liste de plus de 20 films avec des formules de correction", + "featureEquipmentProfiles": "Profils de l'équipement", + "featureTimer": "Minuteur intégré pour longues expositions", + "featureMeteringScreenLayout": "Écran principal personnalisable", + "proFeaturesSupportText": "En achetant Lightmeter Pro, vous soutenez le développement et permettez l'ajout de nouvelles fonctionnalités à l'application.", + "getNowFor": "Acheter maintenant {price}", + "@getNowFor": { + "price": { + "version": { + "type": "String" + } + } + }, "tooltipAdd": "Ajouter", "tooltipClose": "Fermer", "tooltipExpand": "Élargir", diff --git a/lib/l10n/intl_ru.arb b/lib/l10n/intl_ru.arb index d005f77..2e60b8d 100644 --- a/lib/l10n/intl_ru.arb +++ b/lib/l10n/intl_ru.arb @@ -67,8 +67,8 @@ "equipmentProfile": "Оборудование", "equipmentProfiles": "Профили оборудования", "tapToAdd": "Нажмите, чтобы добавить", - "filmsInUse": "Используемые пленки", - "filmsInUseDescription": "Выберите пленки, которыми вы пользуетесь.", + "filmsInUse": "Используемые плёнки", + "filmsInUseDescription": "Выберите плёнки, которыми вы пользуетесь.", "general": "Общие", "keepScreenOn": "Запрет блокировки", "haptics": "Вибрация", @@ -103,10 +103,31 @@ } } }, - "proFeatures": "Профессиональные настройки", - "unlockProFeatures": "Разблокировать профессиональные настройки", - "unlockProFeaturesDescription": "Вы можете разблокировать профессиональные настройки:\n \u2022 Профили оборудования, содержащие фильтры для диафрагмы, выдержки и других значений\n \u2022 Список пленок с компенсацией эффекта Шварцшильда\n \u2022 Точечный замер и гистограмма\n \u2022 И другие возможности!\n\nПолучая доступ к профессиональным настройкам, вы поддерживаете разработку и делаете возможным появление новых функций в приложении.", - "unlock": "Разблокировать", + "proFeaturesTitle": "Lightmeter Pro", + "getPro": "Купить Pro", + "featuresFree": "Бесплатно", + "featuresPro": "Pro", + "proFeaturesPromoText": "Lightmeter Pro предоставляет все необходимое для получения лучших снимков!", + "proFeaturesWhatsIncluded": "Что включено?", + "featureReflectedLightMetering": "Замер отраженного света", + "featureIncidentLightMetering": "Замер падающего света", + "featureIsoAndNdValues": "Широкий диапазон значений ISO и фильтров ND", + "featureTheme": "Настройка темы", + "featureSpotMetering": "Точечный замер", + "featureHistogram": "Гистограмма", + "featureListOfFilms": "Список из 20+ плёнок с формулами коррекции", + "featureEquipmentProfiles": "Профили оборудования", + "featureTimer": "Встроенный таймер для длинных выдержек", + "featureMeteringScreenLayout": "Настраиваемый главный экран", + "proFeaturesSupportText": "Покупая Lightmeter Pro, вы поддерживаете разработку и делаете возможным добавление новых функций в приложение.", + "getNowFor": "Купить за {price}", + "@getNowFor": { + "price": { + "version": { + "type": "String" + } + } + }, "tooltipAdd": "Добавить", "tooltipClose": "Закрыть", "tooltipExpand": "Развернуть", diff --git a/lib/l10n/intl_zh.arb b/lib/l10n/intl_zh.arb index aa17748..76df1fb 100644 --- a/lib/l10n/intl_zh.arb +++ b/lib/l10n/intl_zh.arb @@ -103,10 +103,30 @@ } } }, - "proFeatures": "专业功能", - "unlockProFeatures": "解锁专业功能", - "unlockProFeaturesDescription": "\n \u2022 配置文件,其中包含光圈、快门速度等参数\n \u2022 胶卷列表,对胶片倒易率失效进行曝光补偿\n \u2022 点测光和直方图\n \u2022 和更多\n\n通过解锁专业版功能,您可以支持开发工作,帮助为应用程序添加新功能。", - "unlock": "解锁", + "getPro": "购买专业版", + "featuresFree": "免费", + "featuresPro": "专业版", + "proFeaturesPromoText": "Lightmeter Pro 提供您需要的一切,助您拍出最佳照片!", + "proFeaturesWhatsIncluded": "包括哪些内容?", + "featureReflectedLightMetering": "反射光测光", + "featureIncidentLightMetering": "入射光测光", + "featureIsoAndNdValues": "广泛的ISO和ND滤镜值范围", + "featureTheme": "主题自定义", + "featureSpotMetering": "点测光", + "featureHistogram": "直方图", + "featureListOfFilms": "20多部电影的修正公式列表", + "featureEquipmentProfiles": "设备配置文件", + "featureTimer": "内置长曝光计时器", + "featureMeteringScreenLayout": "可自定义的主屏幕", + "proFeaturesSupportText": "通过购买Lightmeter Pro,您支持开发工作,并使添加新功能成为可能。", + "getNowFor": "立即获取 {price}", + "@getNowFor": { + "price": { + "version": { + "type": "String" + } + } + }, "tooltipAdd": "添加", "tooltipClose": "关闭", "tooltipExpand": "展开", diff --git a/lib/runner.dart b/lib/runner.dart index 8eb31b2..b43d7bf 100644 --- a/lib/runner.dart +++ b/lib/runner.dart @@ -25,7 +25,12 @@ Future runLightmeterApp(Environment env) async { runApp( env.buildType == BuildType.dev ? IAPProducts( - products: [IAPProduct(storeId: IAPProductType.paidFeatures.storeId)], + products: [ + IAPProduct( + storeId: IAPProductType.paidFeatures.storeId, + price: '0.0\$', + ), + ], child: application, ) : IAPProductsProvider( diff --git a/lib/screens/lightmeter_pro/screen_lightmeter_pro.dart b/lib/screens/lightmeter_pro/screen_lightmeter_pro.dart new file mode 100644 index 0000000..35bdf83 --- /dev/null +++ b/lib/screens/lightmeter_pro/screen_lightmeter_pro.dart @@ -0,0 +1,229 @@ +import 'package:flutter/foundation.dart'; +import 'package:flutter/material.dart'; +import 'package:lightmeter/data/models/app_feature.dart'; +import 'package:lightmeter/generated/l10n.dart'; +import 'package:lightmeter/providers/services_provider.dart'; +import 'package:lightmeter/res/dimens.dart'; +import 'package:lightmeter/res/theme.dart'; +import 'package:lightmeter/screens/shared/sliver_screen/screen_sliver.dart'; +import 'package:lightmeter/utils/text_height.dart'; +import 'package:m3_lightmeter_iap/m3_lightmeter_iap.dart'; + +class LightmeterProScreen extends StatelessWidget { + final features = + defaultTargetPlatform == TargetPlatform.android ? AppFeature.androidFeatures : AppFeature.iosFeatures; + + LightmeterProScreen({super.key}); + + @override + Widget build(BuildContext context) { + return Column( + children: [ + Expanded( + child: SliverScreen( + title: S.of(context).proFeaturesTitle, + slivers: [ + SliverToBoxAdapter( + child: Padding( + padding: const EdgeInsets.all(Dimens.paddingM), + child: Text( + S.of(context).proFeaturesPromoText, + style: Theme.of(context).textTheme.bodyLarge, + ), + ), + ), + SliverToBoxAdapter( + child: Padding( + padding: const EdgeInsets.fromLTRB( + Dimens.paddingM, + 0, + Dimens.paddingM, + Dimens.paddingS, + ), + child: Text( + S.of(context).proFeaturesWhatsIncluded, + style: Theme.of(context).textTheme.headlineSmall, + ), + ), + ), + const SliverToBoxAdapter(child: _FeaturesHeader()), + SliverList.separated( + itemCount: features.length, + itemBuilder: (_, index) => _FeatureItem(feature: features[index]), + separatorBuilder: (_, __) => const Padding( + padding: EdgeInsets.symmetric(horizontal: Dimens.paddingM), + child: Divider(), + ), + ), + SliverToBoxAdapter( + child: Padding( + padding: const EdgeInsets.all(Dimens.paddingM), + child: Text(S.of(context).proFeaturesSupportText), + ), + ), + ], + ), + ), + Container( + color: Theme.of(context).colorScheme.surfaceElevated1, + width: MediaQuery.sizeOf(context).width, + padding: EdgeInsets.fromLTRB( + Dimens.paddingM, + Dimens.paddingM, + Dimens.paddingM, + Dimens.paddingM + MediaQuery.paddingOf(context).bottom, + ), + child: FilledButton( + onPressed: () { + ServicesProvider.maybeOf(context) + ?.analytics + .setCustomKey('iap_product_type', IAPProductType.paidFeatures.storeId); + IAPProductsProvider.maybeOf(context)?.buy(IAPProductType.paidFeatures); + Navigator.of(context).pop(); + }, + child: Text(S.of(context).getNowFor(IAPProducts.productOf(context, IAPProductType.paidFeatures)!.price)), + ), + ), + ], + ); + } +} + +class _FeaturesHeader extends StatelessWidget { + const _FeaturesHeader(); + + @override + Widget build(BuildContext context) { + return Padding( + padding: const EdgeInsets.symmetric(horizontal: Dimens.paddingM), + child: Row( + children: [ + const Spacer(), + _FeatureHighlight(child: Text(S.of(context).featuresFree)), + _FeatureHighlight( + roundedTop: true, + highlight: true, + child: Text( + S.of(context).featuresPro, + style: Theme.of(context) + .textTheme + .bodyMedium! + .copyWith(color: Theme.of(context).colorScheme.onSecondaryContainer), + ), + ), + ], + ), + ); + } +} + +class _FeatureItem extends StatelessWidget { + final AppFeature feature; + + const _FeatureItem({ + required this.feature, + }); + + @override + Widget build(BuildContext context) { + return ConstrainedBox( + constraints: const BoxConstraints(minHeight: Dimens.grid48), + child: IntrinsicHeight( + child: Row( + crossAxisAlignment: CrossAxisAlignment.stretch, + mainAxisSize: MainAxisSize.min, + children: [ + Expanded( + child: Align( + alignment: Alignment.centerLeft, + child: Padding( + padding: const EdgeInsets.symmetric( + horizontal: Dimens.paddingM, + vertical: Dimens.paddingS, + ), + child: Text( + feature.name(context), + style: Theme.of(context).textTheme.bodyLarge, + ), + ), + ), + ), + Opacity( + opacity: feature.isFree ? 1 : 0, + child: const _FeatureHighlight( + child: _CheckBox(highlight: false), + ), + ), + _FeatureHighlight( + highlight: true, + roundedBottom: feature == AppFeature.values.last, + child: const _CheckBox(highlight: true), + ), + const SizedBox(width: Dimens.grid16), + ], + ), + ), + ); + } +} + +class _FeatureHighlight extends StatelessWidget { + final bool highlight; + final bool roundedTop; + final bool roundedBottom; + final Widget child; + + const _FeatureHighlight({ + this.highlight = false, + this.roundedTop = false, + this.roundedBottom = false, + required this.child, + }); + + @override + Widget build(BuildContext context) { + return Container( + constraints: BoxConstraints( + minWidth: textSize( + highlight ? S.of(context).featuresPro : S.of(context).featuresFree, + Theme.of(context).textTheme.bodyMedium, + MediaQuery.sizeOf(context).width, + ).width + + Dimens.paddingM * 2, + ), + padding: const EdgeInsets.symmetric( + horizontal: Dimens.paddingM, + vertical: Dimens.paddingS, + ), + decoration: BoxDecoration( + color: highlight ? Theme.of(context).colorScheme.secondaryContainer : null, + borderRadius: roundedTop + ? const BorderRadius.only( + topLeft: Radius.circular(Dimens.borderRadiusM), + topRight: Radius.circular(Dimens.borderRadiusM), + ) + : roundedBottom + ? const BorderRadius.only( + bottomLeft: Radius.circular(Dimens.borderRadiusM), + bottomRight: Radius.circular(Dimens.borderRadiusM), + ) + : null, + ), + child: child, + ); + } +} + +class _CheckBox extends StatelessWidget { + final bool highlight; + + const _CheckBox({required this.highlight}); + + @override + Widget build(BuildContext context) { + return Icon( + Icons.check_outlined, + color: highlight ? Theme.of(context).colorScheme.onSecondaryContainer : null, + ); + } +} diff --git a/lib/screens/metering/components/shared/readings_container/components/lightmeter_pro/widget_lightmeter_pro.dart b/lib/screens/metering/components/shared/readings_container/components/lightmeter_pro/widget_lightmeter_pro.dart index 9d5218a..245bab3 100644 --- a/lib/screens/metering/components/shared/readings_container/components/lightmeter_pro/widget_lightmeter_pro.dart +++ b/lib/screens/metering/components/shared/readings_container/components/lightmeter_pro/widget_lightmeter_pro.dart @@ -1,27 +1,26 @@ import 'package:flutter/material.dart'; import 'package:lightmeter/generated/l10n.dart'; -import 'package:lightmeter/screens/metering/components/shared/readings_container/components/shared/animated_dialog_picker/components/animated_dialog/widget_dialog_animated.dart'; import 'package:lightmeter/screens/metering/components/shared/readings_container/components/shared/reading_value_container/widget_container_reading_value.dart'; -import 'package:lightmeter/screens/shared/pro_features_dialog/widget_dialog_pro_features.dart'; class LightmeterProAnimatedDialog extends StatelessWidget { const LightmeterProAnimatedDialog({super.key}); @override Widget build(BuildContext context) { - return AnimatedDialog( - closedChild: ReadingValueContainer( - color: Theme.of(context).colorScheme.errorContainer, - textColor: Theme.of(context).colorScheme.onErrorContainer, + return GestureDetector( + onTap: () { + Navigator.of(context).pushNamed("lightmeterPro"); + }, + child: ReadingValueContainer( + color: Theme.of(context).colorScheme.secondary, + textColor: Theme.of(context).colorScheme.onSecondary, values: [ ReadingValue( - label: S.of(context).proFeatures, - value: S.of(context).unlock, + label: S.of(context).proFeaturesTitle, + value: S.of(context).getPro, ), ], ), - openedChild: const ProFeaturesDialog(), - openedSize: Size.fromHeight(const ProFeaturesDialog().height(context)), ); } } diff --git a/lib/screens/settings/components/about/components/restore_purchases/widget_list_tile_restore_purchases.dart b/lib/screens/settings/components/about/components/restore_purchases/widget_list_tile_restore_purchases.dart index b07257e..23f778d 100644 --- a/lib/screens/settings/components/about/components/restore_purchases/widget_list_tile_restore_purchases.dart +++ b/lib/screens/settings/components/about/components/restore_purchases/widget_list_tile_restore_purchases.dart @@ -10,7 +10,7 @@ class RestorePurchasesListTile extends StatelessWidget { return ListTile( leading: const Icon(Icons.restore_outlined), title: Text(S.of(context).restorePurchases), - onTap: IAPProductsProvider.of(context).restorePurchases, + onTap: IAPProductsProvider.maybeOf(context)?.restorePurchases, ); } } diff --git a/lib/screens/settings/components/lightmeter_pro/components/buy_pro/widget_list_tile_buy_pro.dart b/lib/screens/settings/components/lightmeter_pro/components/buy_pro/widget_list_tile_buy_pro.dart index cc026a9..3ab7ae5 100644 --- a/lib/screens/settings/components/lightmeter_pro/components/buy_pro/widget_list_tile_buy_pro.dart +++ b/lib/screens/settings/components/lightmeter_pro/components/buy_pro/widget_list_tile_buy_pro.dart @@ -1,7 +1,6 @@ import 'package:flutter/material.dart'; import 'package:lightmeter/generated/l10n.dart'; import 'package:lightmeter/res/dimens.dart'; -import 'package:lightmeter/screens/shared/pro_features_dialog/widget_dialog_pro_features.dart'; import 'package:m3_lightmeter_iap/m3_lightmeter_iap.dart'; class BuyProListTile extends StatelessWidget { @@ -12,14 +11,11 @@ class BuyProListTile extends StatelessWidget { final status = IAPProducts.productOf(context, IAPProductType.paidFeatures)?.status; final isPending = status == IAPProductStatus.purchased || status == null; return ListTile( - leading: const Icon(Icons.star_outlined), - title: Text(S.of(context).unlockProFeatures), + leading: const Icon(Icons.bolt), + title: Text(S.of(context).getPro), onTap: !isPending ? () { - showDialog( - context: context, - builder: (_) => const Dialog(child: ProFeaturesDialog()), - ); + Navigator.of(context).pushNamed("lightmeterPro"); } : null, trailing: isPending diff --git a/lib/screens/settings/components/lightmeter_pro/widget_settings_section_lightmeter_pro.dart b/lib/screens/settings/components/lightmeter_pro/widget_settings_section_lightmeter_pro.dart index 57d5d14..501398c 100644 --- a/lib/screens/settings/components/lightmeter_pro/widget_settings_section_lightmeter_pro.dart +++ b/lib/screens/settings/components/lightmeter_pro/widget_settings_section_lightmeter_pro.dart @@ -9,7 +9,9 @@ class LightmeterProSettingsSection extends StatelessWidget { @override Widget build(BuildContext context) { return SettingsSection( - title: S.of(context).proFeatures, + backgroundColor: Theme.of(context).colorScheme.secondary, + foregroundColor: Theme.of(context).colorScheme.onSecondary, + title: S.of(context).proFeaturesTitle, children: const [BuyProListTile()], ); } diff --git a/lib/screens/settings/components/shared/settings_section/widget_settings_section.dart b/lib/screens/settings/components/shared/settings_section/widget_settings_section.dart index be62847..718c67c 100644 --- a/lib/screens/settings/components/shared/settings_section/widget_settings_section.dart +++ b/lib/screens/settings/components/shared/settings_section/widget_settings_section.dart @@ -4,10 +4,14 @@ import 'package:lightmeter/res/dimens.dart'; class SettingsSection extends StatelessWidget { final String title; final List children; + final Color? backgroundColor; + final Color? foregroundColor; const SettingsSection({ required this.title, required this.children, + this.backgroundColor, + this.foregroundColor, super.key, }); @@ -21,24 +25,33 @@ class SettingsSection extends StatelessWidget { Dimens.paddingM, ), child: Card( + color: backgroundColor, child: Padding( padding: const EdgeInsets.symmetric(vertical: Dimens.paddingM), - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - mainAxisSize: MainAxisSize.min, - children: [ - Padding( - padding: const EdgeInsets.symmetric(horizontal: Dimens.paddingM), - child: Text( - title, - style: Theme.of(context) - .textTheme - .labelLarge - ?.copyWith(color: Theme.of(context).colorScheme.onSurface), + child: Theme( + data: Theme.of(context).copyWith( + listTileTheme: Theme.of(context).listTileTheme.copyWith( + iconColor: foregroundColor, + textColor: foregroundColor, + ), + ), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisSize: MainAxisSize.min, + children: [ + Padding( + padding: const EdgeInsets.symmetric(horizontal: Dimens.paddingM), + child: Text( + title, + style: Theme.of(context) + .textTheme + .labelLarge + ?.copyWith(color: foregroundColor ?? Theme.of(context).colorScheme.onSurface), + ), ), - ), - ...children, - ], + ...children, + ], + ), ), ), ), diff --git a/lib/screens/shared/pro_features_dialog/widget_dialog_pro_features.dart b/lib/screens/shared/pro_features_dialog/widget_dialog_pro_features.dart deleted file mode 100644 index cdf76b0..0000000 --- a/lib/screens/shared/pro_features_dialog/widget_dialog_pro_features.dart +++ /dev/null @@ -1,62 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:lightmeter/generated/l10n.dart'; -import 'package:lightmeter/providers/services_provider.dart'; -import 'package:lightmeter/res/dimens.dart'; -import 'package:lightmeter/screens/metering/components/shared/readings_container/components/shared/animated_dialog_picker/components/animated_dialog/widget_dialog_animated.dart'; -import 'package:lightmeter/screens/shared/transparent_dialog/widget_dialog_transparent.dart'; -import 'package:lightmeter/utils/text_height.dart'; -import 'package:m3_lightmeter_iap/m3_lightmeter_iap.dart'; - -class ProFeaturesDialog extends StatelessWidget { - const ProFeaturesDialog({super.key}); - - double height(BuildContext context) => TransparentDialog.height( - context, - title: S.of(context).proFeatures, - contextHeight: dialogTextHeight( - context, - S.of(context).unlockProFeaturesDescription, - Theme.of(context).textTheme.bodyMedium, - Dimens.paddingL * 2, - ), - ); - - @override - Widget build(BuildContext context) { - return TransparentDialog( - icon: Icons.star_outlined, - title: S.of(context).proFeatures, - scrollableContent: false, - content: Flexible( - child: SingleChildScrollView( - child: Padding( - padding: const EdgeInsets.symmetric(horizontal: Dimens.paddingL), - child: Text( - S.of(context).unlockProFeaturesDescription, - style: Theme.of(context).textTheme.bodyMedium, - ), - ), - ), - ), - actions: [ - TextButton( - onPressed: () => _close(context), - child: Text(S.of(context).cancel), - ), - FilledButton( - onPressed: () { - _close(context).then((_) { - ServicesProvider.maybeOf(context) - ?.analytics - .setCustomKey('iap_product_type', IAPProductType.paidFeatures.storeId); - IAPProductsProvider.maybeOf(context)?.buy(IAPProductType.paidFeatures); - }); - }, - child: Text(S.of(context).unlock), - ), - ], - ); - } - - Future _close(BuildContext context) async => AnimatedDialog.maybeClose(context) ?? Navigator.of(context).pop(); -} diff --git a/lib/utils/text_height.dart b/lib/utils/text_height.dart index f5a0fe1..6c3afa4 100644 --- a/lib/utils/text_height.dart +++ b/lib/utils/text_height.dart @@ -17,6 +17,13 @@ double textHeight( String text, TextStyle? style, double maxWidth, +) => + textSize(text, style, maxWidth).height; + +Size textSize( + String text, + TextStyle? style, + double maxWidth, ) { final TextPainter titlePainter = TextPainter( text: TextSpan( @@ -25,5 +32,5 @@ double textHeight( ), textDirection: TextDirection.ltr, )..layout(maxWidth: maxWidth); - return titlePainter.height; + return titlePainter.size; } diff --git a/pubspec.yaml b/pubspec.yaml index bd98a04..cbc36ec 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -29,7 +29,7 @@ dependencies: m3_lightmeter_iap: git: url: "https://github.com/vodemn/m3_lightmeter_iap" - ref: v0.10.0 + ref: v0.11.0 m3_lightmeter_resources: git: url: "https://github.com/vodemn/m3_lightmeter_resources" diff --git a/test/application_mock.dart b/test/application_mock.dart index a1bffef..a7882cc 100644 --- a/test/application_mock.dart +++ b/test/application_mock.dart @@ -91,6 +91,7 @@ class _GoldenTestApplicationMockState extends State { IAPProduct( storeId: IAPProductType.paidFeatures.storeId, status: widget.productStatus, + price: '0.0\$', ), ], child: ApplicationWrapper( diff --git a/test/providers/equipment_profile_provider_test.dart b/test/providers/equipment_profile_provider_test.dart index fb329cd..15472dc 100644 --- a/test/providers/equipment_profile_provider_test.dart +++ b/test/providers/equipment_profile_provider_test.dart @@ -26,6 +26,7 @@ void main() { IAPProduct( storeId: IAPProductType.paidFeatures.storeId, status: productStatus, + price: '0.0\$', ), ], child: EquipmentProfileProvider( diff --git a/test/providers/films_provider_test.dart b/test/providers/films_provider_test.dart index 47c35c1..f898579 100644 --- a/test/providers/films_provider_test.dart +++ b/test/providers/films_provider_test.dart @@ -26,6 +26,7 @@ void main() { IAPProduct( storeId: IAPProductType.paidFeatures.storeId, status: productStatus, + price: '0.0\$', ), ], child: FilmsProvider( diff --git a/test/screens/lightmeter_pro/goldens/lightmeter_pro_screen.png b/test/screens/lightmeter_pro/goldens/lightmeter_pro_screen.png new file mode 100644 index 0000000..7db028a Binary files /dev/null and b/test/screens/lightmeter_pro/goldens/lightmeter_pro_screen.png differ diff --git a/test/screens/lightmeter_pro/lightmeter_pro_screen_golden_test.dart b/test/screens/lightmeter_pro/lightmeter_pro_screen_golden_test.dart new file mode 100644 index 0000000..a70327f --- /dev/null +++ b/test/screens/lightmeter_pro/lightmeter_pro_screen_golden_test.dart @@ -0,0 +1,49 @@ +import 'package:flutter/widgets.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:golden_toolkit/golden_toolkit.dart'; +import 'package:lightmeter/data/models/theme_type.dart'; +import 'package:lightmeter/screens/lightmeter_pro/screen_lightmeter_pro.dart'; +import 'package:m3_lightmeter_iap/m3_lightmeter_iap.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import '../../application_mock.dart'; +import '../../utils/golden_test_set_theme.dart'; + +void main() { + setUpAll(() { + SharedPreferences.setMockInitialValues({}); + }); + + testGoldens( + 'LightmeterProScreen golden test', + (tester) async { + final builder = DeviceBuilder(); + builder.addScenario( + name: 'Get Pro', + widget: const _MockLightmeterProFlow(), + onCreate: (scenarioWidgetKey) async { + if (scenarioWidgetKey.toString().contains('Dark')) { + await setTheme(tester, scenarioWidgetKey, ThemeType.dark); + } + }, + ); + await tester.pumpDeviceBuilder(builder); + await screenMatchesGolden( + tester, + 'lightmeter_pro_screen', + ); + }, + ); +} + +class _MockLightmeterProFlow extends StatelessWidget { + const _MockLightmeterProFlow(); + + @override + Widget build(BuildContext context) { + return GoldenTestApplicationMock( + productStatus: IAPProductStatus.purchasable, + child: LightmeterProScreen(), + ); + } +} diff --git a/test/screens/metering/components/shared/readings_container/equipment_profile_picker_test.dart b/test/screens/metering/components/shared/readings_container/equipment_profile_picker_test.dart index 5eec978..8c7d96b 100644 --- a/test/screens/metering/components/shared/readings_container/equipment_profile_picker_test.dart +++ b/test/screens/metering/components/shared/readings_container/equipment_profile_picker_test.dart @@ -28,6 +28,7 @@ void main() { IAPProduct( storeId: IAPProductType.paidFeatures.storeId, status: IAPProductStatus.purchased, + price: '0.0\$', ), ], child: EquipmentProfileProvider( diff --git a/test/screens/metering/components/shared/readings_container/film_picker_test.dart b/test/screens/metering/components/shared/readings_container/film_picker_test.dart index 03f50ed..765ca51 100644 --- a/test/screens/metering/components/shared/readings_container/film_picker_test.dart +++ b/test/screens/metering/components/shared/readings_container/film_picker_test.dart @@ -27,6 +27,7 @@ void main() { IAPProduct( storeId: IAPProductType.paidFeatures.storeId, status: IAPProductStatus.purchased, + price: '0.0\$', ), ], child: FilmsProvider( diff --git a/test/screens/metering/goldens/metering_screen.png b/test/screens/metering/goldens/metering_screen.png index a88dc2c..4810bc9 100644 Binary files a/test/screens/metering/goldens/metering_screen.png and b/test/screens/metering/goldens/metering_screen.png differ diff --git a/test/screens/metering/screen_metering_golden_test.dart b/test/screens/metering/screen_metering_golden_test.dart index 8c7e15f..9ca908f 100644 --- a/test/screens/metering/screen_metering_golden_test.dart +++ b/test/screens/metering/screen_metering_golden_test.dart @@ -15,6 +15,7 @@ import 'package:shared_preferences/shared_preferences.dart'; import '../../../integration_test/utils/finder_actions.dart'; import '../../../integration_test/utils/platform_channel_mock.dart'; import '../../application_mock.dart'; +import '../../utils/golden_test_set_theme.dart'; class _MeteringScreenConfig { final IAPProductStatus iapProductStatus; @@ -54,16 +55,6 @@ void main() { await tester.pumpAndSettle(); } - Future setTheme(WidgetTester tester, Key scenarioWidgetKey, ThemeType themeType) async { - final flow = find.descendant( - of: find.byKey(scenarioWidgetKey), - matching: find.byType(MeteringFlow), - ); - final BuildContext context = tester.element(flow); - UserPreferencesProvider.of(context).setThemeType(themeType); - await tester.pumpAndSettle(); - } - Future takePhoto(WidgetTester tester, Key scenarioWidgetKey) async { final button = find.descendant( of: find.byKey(scenarioWidgetKey), @@ -110,7 +101,7 @@ void main() { onCreate: (scenarioWidgetKey) async { await setEvSource(tester, scenarioWidgetKey, scenario.evSourceType); if (scenarioWidgetKey.toString().contains('Dark')) { - await setTheme(tester, scenarioWidgetKey, ThemeType.dark); + await setTheme(tester, scenarioWidgetKey, ThemeType.dark); } if (scenario.evSourceType == EvSourceType.camera) { await takePhoto(tester, scenarioWidgetKey); diff --git a/test/screens/metering/utils/listener_equipment_profiles_test.dart b/test/screens/metering/utils/listener_equipment_profiles_test.dart index ae6cb87..38e2a14 100644 --- a/test/screens/metering/utils/listener_equipment_profiles_test.dart +++ b/test/screens/metering/utils/listener_equipment_profiles_test.dart @@ -28,6 +28,7 @@ void main() { IAPProduct( storeId: IAPProductType.paidFeatures.storeId, status: IAPProductStatus.purchased, + price: '0.0\$', ), ], child: EquipmentProfileProvider( diff --git a/test/screens/settings/components/widget_settings_section_lightmeter_pro_test.dart b/test/screens/settings/components/widget_settings_section_lightmeter_pro_test.dart deleted file mode 100644 index 1ae99f4..0000000 --- a/test/screens/settings/components/widget_settings_section_lightmeter_pro_test.dart +++ /dev/null @@ -1,60 +0,0 @@ -import 'package:flutter_test/flutter_test.dart'; -import 'package:lightmeter/generated/l10n.dart'; -import 'package:lightmeter/screens/settings/components/lightmeter_pro/components/buy_pro/widget_list_tile_buy_pro.dart'; -import 'package:lightmeter/screens/settings/components/lightmeter_pro/widget_settings_section_lightmeter_pro.dart'; -import 'package:lightmeter/screens/shared/transparent_dialog/widget_dialog_transparent.dart'; -import 'package:m3_lightmeter_iap/m3_lightmeter_iap.dart'; - -import '../../../application_mock.dart'; - -void main() { - Future pumpApplication(WidgetTester tester) async { - await tester.pumpWidget( - IAPProducts( - products: [ - IAPProduct( - storeId: IAPProductType.paidFeatures.storeId, - ), - ], - child: const WidgetTestApplicationMock( - child: LightmeterProSettingsSection(), - ), - ), - ); - await tester.pumpAndSettle(); - } - - testWidgets( - '`showBuyProDialog` and buy', - (tester) async { - await pumpApplication(tester); - await tester.tap(find.byType(BuyProListTile)); - await tester.pumpAndSettle(); - expect(find.byType(TransparentDialog), findsOneWidget); - expect(find.text(S.current.proFeatures), findsNWidgets(2)); - expect(find.text(S.current.cancel), findsOneWidget); - expect(find.text(S.current.unlock), findsOneWidget); - - await tester.tap(find.text(S.current.unlock)); - await tester.pumpAndSettle(); - expect(find.byType(TransparentDialog), findsNothing); - }, - ); - - testWidgets( - '`showBuyProDialog` and cancel', - (tester) async { - await pumpApplication(tester); - await tester.tap(find.byType(BuyProListTile)); - await tester.pumpAndSettle(); - expect(find.byType(TransparentDialog), findsOneWidget); - expect(find.text(S.current.proFeatures), findsNWidgets(2)); - expect(find.text(S.current.cancel), findsOneWidget); - expect(find.text(S.current.unlock), findsOneWidget); - - await tester.tap(find.text(S.current.cancel)); - await tester.pumpAndSettle(); - expect(find.byType(TransparentDialog), findsNothing); - }, - ); -} diff --git a/test/screens/settings/goldens/settings_screen.png b/test/screens/settings/goldens/settings_screen.png index 1c65c19..025453b 100644 Binary files a/test/screens/settings/goldens/settings_screen.png and b/test/screens/settings/goldens/settings_screen.png differ diff --git a/test/screens/settings/settings_screen_golden_test.dart b/test/screens/settings/settings_screen_golden_test.dart index c1deae5..6c34522 100644 --- a/test/screens/settings/settings_screen_golden_test.dart +++ b/test/screens/settings/settings_screen_golden_test.dart @@ -7,13 +7,13 @@ import 'package:lightmeter/data/models/ev_source_type.dart'; import 'package:lightmeter/data/models/metering_screen_layout_config.dart'; import 'package:lightmeter/data/models/theme_type.dart'; import 'package:lightmeter/data/shared_prefs_service.dart'; -import 'package:lightmeter/providers/user_preferences_provider.dart'; import 'package:lightmeter/screens/settings/flow_settings.dart'; import 'package:m3_lightmeter_iap/m3_lightmeter_iap.dart'; import 'package:package_info_plus/package_info_plus.dart'; import 'package:shared_preferences/shared_preferences.dart'; import '../../application_mock.dart'; +import '../../utils/golden_test_set_theme.dart'; class _SettingsScreenConfig { final IAPProductStatus iapProductStatus; @@ -33,16 +33,6 @@ final _testScenarios = [IAPProductStatus.purchased, IAPProductStatus.purchasable ); void main() { - Future setTheme(WidgetTester tester, Key scenarioWidgetKey, ThemeType themeType) async { - final flow = find.descendant( - of: find.byKey(scenarioWidgetKey), - matching: find.byType(SettingsFlow), - ); - final BuildContext context = tester.element(flow); - UserPreferencesProvider.of(context).setThemeType(themeType); - await tester.pumpAndSettle(); - } - setUpAll(() { SharedPreferences.setMockInitialValues({ UserPreferencesService.evSourceTypeKey: EvSourceType.camera.index, @@ -73,7 +63,7 @@ void main() { widget: _MockSettingsFlow(productStatus: scenario.iapProductStatus), onCreate: (scenarioWidgetKey) async { if (scenarioWidgetKey.toString().contains('Dark')) { - await setTheme(tester, scenarioWidgetKey, ThemeType.dark); + await setTheme(tester, scenarioWidgetKey, ThemeType.dark); } }, ); diff --git a/test/screens/timer/screen_timer_golden_test.dart b/test/screens/timer/screen_timer_golden_test.dart index 41b682a..f0b601b 100644 --- a/test/screens/timer/screen_timer_golden_test.dart +++ b/test/screens/timer/screen_timer_golden_test.dart @@ -4,7 +4,6 @@ import 'package:golden_toolkit/golden_toolkit.dart'; import 'package:lightmeter/data/models/exposure_pair.dart'; import 'package:lightmeter/data/models/theme_type.dart'; import 'package:lightmeter/data/shared_prefs_service.dart'; -import 'package:lightmeter/providers/user_preferences_provider.dart'; import 'package:lightmeter/res/dimens.dart'; import 'package:lightmeter/screens/shared/animated_circular_button/widget_button_circular_animated.dart'; import 'package:lightmeter/screens/timer/flow_timer.dart'; @@ -13,6 +12,7 @@ import 'package:m3_lightmeter_resources/m3_lightmeter_resources.dart'; import 'package:shared_preferences/shared_preferences.dart'; import '../../application_mock.dart'; +import '../../utils/golden_test_set_theme.dart'; class _TimerScreenConfig { final ShutterSpeedValue shutterSpeedValue; @@ -52,16 +52,6 @@ final _testScenarios = [ ); void main() { - Future setTheme(WidgetTester tester, Key scenarioWidgetKey, ThemeType themeType) async { - final flow = find.descendant( - of: find.byKey(scenarioWidgetKey), - matching: find.byType(TimerFlow), - ); - final BuildContext context = tester.element(flow); - UserPreferencesProvider.of(context).setThemeType(themeType); - await tester.pumpAndSettle(); - } - Future toggleTimer(WidgetTester tester, Key scenarioWidgetKey) async { final button = find.descendant( of: find.byKey(scenarioWidgetKey), @@ -98,7 +88,7 @@ void main() { widget: _MockTimerFlow(scenario.shutterSpeedValue), onCreate: (scenarioWidgetKey) async { if (scenarioWidgetKey.toString().contains('Dark')) { - await setTheme(tester, scenarioWidgetKey, ThemeType.dark); + await setTheme(tester, scenarioWidgetKey, ThemeType.dark); } if (!scenario.isStopped) { await toggleTimer(tester, scenarioWidgetKey); diff --git a/test/utils/golden_test_set_theme.dart b/test/utils/golden_test_set_theme.dart new file mode 100644 index 0000000..dbeef13 --- /dev/null +++ b/test/utils/golden_test_set_theme.dart @@ -0,0 +1,14 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:lightmeter/data/models/theme_type.dart'; +import 'package:lightmeter/providers/user_preferences_provider.dart'; + +Future setTheme(WidgetTester tester, Key scenarioWidgetKey, ThemeType themeType) async { + final flow = find.descendant( + of: find.byKey(scenarioWidgetKey), + matching: find.byType(T), + ); + final BuildContext context = tester.element(flow); + UserPreferencesProvider.of(context).setThemeType(themeType); + await tester.pumpAndSettle(); +}