mirror of
https://github.com/vodemn/m3_lightmeter.git
synced 2024-11-24 00:10:47 +00:00
3bb3f12641
* 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
129 lines
3.8 KiB
Dart
129 lines
3.8 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:lightmeter/utils/selectable_provider.dart';
|
|
import 'package:m3_lightmeter_iap/m3_lightmeter_iap.dart';
|
|
import 'package:m3_lightmeter_resources/m3_lightmeter_resources.dart';
|
|
import 'package:uuid/uuid.dart';
|
|
|
|
class EquipmentProfileProvider extends StatefulWidget {
|
|
final IAPStorageService storageService;
|
|
final Widget child;
|
|
|
|
const EquipmentProfileProvider({
|
|
required this.storageService,
|
|
required this.child,
|
|
super.key,
|
|
});
|
|
|
|
static EquipmentProfileProviderState of(BuildContext context) {
|
|
return context.findAncestorStateOfType<EquipmentProfileProviderState>()!;
|
|
}
|
|
|
|
@override
|
|
State<EquipmentProfileProvider> createState() => EquipmentProfileProviderState();
|
|
}
|
|
|
|
class EquipmentProfileProviderState extends State<EquipmentProfileProvider> {
|
|
static const EquipmentProfile _defaultProfile = EquipmentProfile(
|
|
id: '',
|
|
name: '',
|
|
apertureValues: ApertureValue.values,
|
|
ndValues: NdValue.values,
|
|
shutterSpeedValues: ShutterSpeedValue.values,
|
|
isoValues: IsoValue.values,
|
|
);
|
|
|
|
List<EquipmentProfile> _customProfiles = [];
|
|
String _selectedId = '';
|
|
|
|
EquipmentProfile get _selectedProfile => _customProfiles.firstWhere(
|
|
(e) => e.id == _selectedId,
|
|
orElse: () => _defaultProfile,
|
|
);
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
_selectedId = widget.storageService.selectedEquipmentProfileId;
|
|
_customProfiles = widget.storageService.equipmentProfiles;
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return EquipmentProfiles(
|
|
values: [
|
|
_defaultProfile,
|
|
if (IAPProducts.isPurchased(context, IAPProductType.paidFeatures)) ..._customProfiles,
|
|
],
|
|
selected: IAPProducts.isPurchased(context, IAPProductType.paidFeatures) ? _selectedProfile : _defaultProfile,
|
|
child: widget.child,
|
|
);
|
|
}
|
|
|
|
void setProfile(EquipmentProfile data) {
|
|
if (_selectedId != data.id) {
|
|
setState(() {
|
|
_selectedId = data.id;
|
|
});
|
|
widget.storageService.selectedEquipmentProfileId = _selectedProfile.id;
|
|
}
|
|
}
|
|
|
|
/// Creates a default equipment profile
|
|
void addProfile(String name, [EquipmentProfile? copyFrom]) {
|
|
_customProfiles.add(
|
|
EquipmentProfile(
|
|
id: const Uuid().v1(),
|
|
name: name,
|
|
apertureValues: copyFrom?.apertureValues ?? ApertureValue.values,
|
|
ndValues: copyFrom?.ndValues ?? NdValue.values,
|
|
shutterSpeedValues: copyFrom?.shutterSpeedValues ?? ShutterSpeedValue.values,
|
|
isoValues: copyFrom?.isoValues ?? IsoValue.values,
|
|
),
|
|
);
|
|
_refreshSavedProfiles();
|
|
}
|
|
|
|
void updateProfile(EquipmentProfile data) {
|
|
final indexToUpdate = _customProfiles.indexWhere((element) => element.id == data.id);
|
|
if (indexToUpdate >= 0) {
|
|
_customProfiles[indexToUpdate] = data;
|
|
_refreshSavedProfiles();
|
|
}
|
|
}
|
|
|
|
void deleteProfile(EquipmentProfile data) {
|
|
if (data.id == _selectedId) {
|
|
_selectedId = _defaultProfile.id;
|
|
widget.storageService.selectedEquipmentProfileId = _defaultProfile.id;
|
|
}
|
|
_customProfiles.remove(data);
|
|
_refreshSavedProfiles();
|
|
}
|
|
|
|
void _refreshSavedProfiles() {
|
|
widget.storageService.equipmentProfiles = _customProfiles;
|
|
setState(() {});
|
|
}
|
|
}
|
|
|
|
class EquipmentProfiles extends SelectableInheritedModel<EquipmentProfile> {
|
|
const EquipmentProfiles({
|
|
super.key,
|
|
required super.values,
|
|
required super.selected,
|
|
required super.child,
|
|
});
|
|
|
|
/// [_defaultProfile] + profiles created by the user
|
|
static List<EquipmentProfile> of(BuildContext context) {
|
|
return InheritedModel.inheritFrom<EquipmentProfiles>(context, aspect: SelectableAspect.list)!.values;
|
|
}
|
|
|
|
static EquipmentProfile selectedOf(BuildContext context) {
|
|
return InheritedModel.inheritFrom<EquipmentProfiles>(
|
|
context,
|
|
aspect: SelectableAspect.selected,
|
|
)!
|
|
.selected;
|
|
}
|
|
}
|