mirror of
https://github.com/vodemn/m3_lightmeter.git
synced 2024-11-24 00:10:47 +00:00
157 lines
5.3 KiB
Dart
157 lines
5.3 KiB
Dart
import 'dart:async';
|
|
import 'dart:math';
|
|
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
import 'package:lightmeter/data/models/photography_values/aperture_value.dart';
|
|
import 'package:lightmeter/data/models/exposure_pair.dart';
|
|
import 'package:lightmeter/data/models/photography_values/iso_value.dart';
|
|
import 'package:lightmeter/data/models/photography_values/nd_value.dart';
|
|
import 'package:lightmeter/data/models/photography_values/photography_value.dart';
|
|
import 'package:lightmeter/data/models/photography_values/shutter_speed_value.dart';
|
|
import 'package:lightmeter/data/shared_prefs_service.dart';
|
|
import 'package:lightmeter/interactors/metering_interactor.dart';
|
|
import 'package:lightmeter/screens/metering/communication/event_communication_metering.dart'
|
|
as communication_events;
|
|
import 'package:lightmeter/screens/metering/communication/state_communication_metering.dart'
|
|
as communication_states;
|
|
import 'package:lightmeter/utils/log_2.dart';
|
|
|
|
import 'communication/bloc_communication_metering.dart';
|
|
import 'event_metering.dart';
|
|
import 'state_metering.dart';
|
|
|
|
class MeteringBloc extends Bloc<MeteringEvent, MeteringState> {
|
|
final MeteringCommunicationBloc _communicationBloc;
|
|
final UserPreferencesService _userPreferencesService;
|
|
final MeteringInteractor _meteringInteractor;
|
|
late final StreamSubscription<communication_states.ScreenState> _communicationSubscription;
|
|
|
|
List<ApertureValue> get _apertureValues => apertureValues.whereStopType(stopType);
|
|
List<ShutterSpeedValue> get _shutterSpeedValues => shutterSpeedValues.whereStopType(stopType);
|
|
|
|
StopType stopType;
|
|
|
|
late IsoValue _iso = _userPreferencesService.iso;
|
|
late NdValue _nd = _userPreferencesService.ndFilter;
|
|
double _ev = 0.0;
|
|
bool _isMeteringInProgress = false;
|
|
|
|
MeteringBloc(
|
|
this._communicationBloc,
|
|
this._userPreferencesService,
|
|
this._meteringInteractor,
|
|
this.stopType,
|
|
) : super(
|
|
MeteringEndedState(
|
|
iso: _userPreferencesService.iso,
|
|
ev: 0.0,
|
|
nd: _userPreferencesService.ndFilter,
|
|
exposurePairs: [],
|
|
),
|
|
) {
|
|
_communicationSubscription = _communicationBloc.stream
|
|
.where((state) => state is communication_states.ScreenState)
|
|
.map((state) => state as communication_states.ScreenState)
|
|
.listen(_onCommunicationState);
|
|
|
|
on<StopTypeChangedEvent>(_onStopTypeChanged);
|
|
on<IsoChangedEvent>(_onIsoChanged);
|
|
on<NdChangedEvent>(_onNdChanged);
|
|
on<MeasureEvent>(_onMeasure);
|
|
on<MeasuredEvent>(_onMeasured);
|
|
}
|
|
|
|
@override
|
|
Future<void> close() async {
|
|
await _communicationSubscription.cancel();
|
|
return super.close();
|
|
}
|
|
|
|
void _onCommunicationState(communication_states.ScreenState communicationState) {
|
|
if (communicationState is communication_states.MeasuredState) {
|
|
_isMeteringInProgress = communicationState is communication_states.MeteringInProgressState;
|
|
add(MeasuredEvent(communicationState.ev100));
|
|
}
|
|
}
|
|
|
|
void _onStopTypeChanged(StopTypeChangedEvent event, Emitter emit) {
|
|
stopType = event.stopType;
|
|
_emitMeasuredState(emit);
|
|
}
|
|
|
|
void _onIsoChanged(IsoChangedEvent event, Emitter emit) {
|
|
_userPreferencesService.iso = event.isoValue;
|
|
_ev = _ev + log2(event.isoValue.value / _iso.value);
|
|
_iso = event.isoValue;
|
|
_emitMeasuredState(emit);
|
|
}
|
|
|
|
void _onNdChanged(NdChangedEvent event, Emitter emit) {
|
|
_userPreferencesService.ndFilter = event.ndValue;
|
|
_ev = _ev - event.ndValue.stopReduction + _nd.stopReduction;
|
|
_nd = event.ndValue;
|
|
_emitMeasuredState(emit);
|
|
}
|
|
|
|
void _onMeasure(_, Emitter emit) {
|
|
_meteringInteractor.quickVibration();
|
|
_communicationBloc.add(const communication_events.MeasureEvent());
|
|
_isMeteringInProgress = true;
|
|
emit(const LoadingState());
|
|
}
|
|
|
|
void _onMeasured(MeasuredEvent event, Emitter emit) {
|
|
_meteringInteractor.responseVibration();
|
|
_ev = event.ev100 + log2(_iso.value / 100);
|
|
_emitMeasuredState(emit);
|
|
}
|
|
|
|
void _emitMeasuredState(Emitter emit) {
|
|
emit(_isMeteringInProgress
|
|
? MeteringInProgressState(
|
|
iso: _iso,
|
|
ev: _ev,
|
|
nd: _nd,
|
|
exposurePairs: _buildExposureValues(_ev),
|
|
)
|
|
: MeteringEndedState(
|
|
iso: _iso,
|
|
ev: _ev,
|
|
nd: _nd,
|
|
exposurePairs: _buildExposureValues(_ev),
|
|
));
|
|
}
|
|
|
|
List<ExposurePair> _buildExposureValues(double ev) {
|
|
/// Depending on the `stopType` the exposure pairs list length is multiplied by 1,2 or 3
|
|
final int evSteps = (ev * (stopType.index + 1)).round();
|
|
final int evOffset =
|
|
_shutterSpeedValues.indexOf(const ShutterSpeedValue(1, false, StopType.full)) - evSteps;
|
|
|
|
late final int apertureOffset;
|
|
late final int shutterSpeedOffset;
|
|
if (evOffset >= 0) {
|
|
apertureOffset = 0;
|
|
shutterSpeedOffset = evOffset;
|
|
} else {
|
|
apertureOffset = -evOffset;
|
|
shutterSpeedOffset = 0;
|
|
}
|
|
|
|
final int itemsCount = min(_apertureValues.length + shutterSpeedOffset,
|
|
_shutterSpeedValues.length + apertureOffset) -
|
|
max(apertureOffset, shutterSpeedOffset);
|
|
|
|
if (itemsCount < 0) {
|
|
return List.empty(growable: false);
|
|
}
|
|
return List.generate(
|
|
itemsCount,
|
|
(index) => ExposurePair(
|
|
_apertureValues[index + apertureOffset],
|
|
_shutterSpeedValues[index + shutterSpeedOffset],
|
|
),
|
|
growable: false,
|
|
);
|
|
}
|
|
}
|