import 'dart:async'; import 'package:bloc/bloc.dart'; import 'package:equatable/equatable.dart'; import 'package:kiwi/kiwi.dart'; import 'package:logging/logging.dart'; import 'package:nc_photos/account.dart'; import 'package:nc_photos/entity/file.dart'; import 'package:nc_photos/entity/file/data_source.dart'; import 'package:nc_photos/event/event.dart'; import 'package:nc_photos/iterable_extension.dart'; import 'package:nc_photos/use_case/scan_dir.dart'; abstract class ScanDirBlocEvent { const ScanDirBlocEvent(); } class ScanDirBlocQueryBase extends ScanDirBlocEvent with EquatableMixin { const ScanDirBlocQueryBase(this.account, this.roots); @override toString() { return "$runtimeType {" "account: $account, " "roots: ${roots.map((e) => e.path).toReadableString()}, " "}"; } @override get props => [ account, roots, ]; final Account account; final List roots; } class ScanDirBlocQuery extends ScanDirBlocQueryBase { const ScanDirBlocQuery(Account account, List roots) : super(account, roots); } class ScanDirBlocRefresh extends ScanDirBlocQueryBase { const ScanDirBlocRefresh(Account account, List roots) : super(account, roots); } /// An external event has happened and may affect the state of this bloc class _ScanDirBlocExternalEvent extends ScanDirBlocEvent { const _ScanDirBlocExternalEvent(); @override toString() { return "$runtimeType {" "}"; } } abstract class ScanDirBlocState { const ScanDirBlocState(this._account, this._files); Account get account => _account; List get files => _files; @override toString() { return "$runtimeType {" "account: $account, " "files: List {length: ${files.length}}, " "}"; } final Account _account; final List _files; } class ScanDirBlocInit extends ScanDirBlocState { const ScanDirBlocInit() : super(null, const []); } class ScanDirBlocLoading extends ScanDirBlocState { const ScanDirBlocLoading(Account account, List files) : super(account, files); } class ScanDirBlocSuccess extends ScanDirBlocState { const ScanDirBlocSuccess(Account account, List files) : super(account, files); } class ScanDirBlocFailure extends ScanDirBlocState { const ScanDirBlocFailure(Account account, List files, this.exception) : super(account, files); @override toString() { return "$runtimeType {" "super: ${super.toString()}, " "exception: $exception, " "}"; } final dynamic exception; } /// The state of this bloc is inconsistent. This typically means that the data /// may have been changed externally class ScanDirBlocInconsistent extends ScanDirBlocState { const ScanDirBlocInconsistent(Account account, List files) : super(account, files); } /// A bloc that return all files under a dir recursively /// /// See [ScanDir] class ScanDirBloc extends Bloc { ScanDirBloc() : super(ScanDirBlocInit()) { _fileRemovedEventListener = AppEventListener(_onFileRemovedEvent); _filePropertyUpdatedEventListener = AppEventListener(_onFilePropertyUpdatedEvent); _fileRemovedEventListener.begin(); _filePropertyUpdatedEventListener.begin(); } static ScanDirBloc of(Account account) { final id = "${account.scheme}://${account.username}@${account.address}?${account.roots.join('&')}"; try { _log.fine("[of] Resolving bloc for '$id'"); return KiwiContainer().resolve("ScanDirBloc($id)"); } catch (_) { // no created instance for this account, make a new one _log.info("[of] New bloc instance for account: $account"); final bloc = ScanDirBloc(); KiwiContainer() .registerInstance(bloc, name: "ScanDirBloc($id)"); return bloc; } } @override transformEvents(Stream events, transitionFn) { return super.transformEvents(events.distinct((a, b) { // only handle ScanDirBlocQuery final r = a is ScanDirBlocQuery && b is ScanDirBlocQuery && a == b; if (r) { _log.fine("[transformEvents] Skip identical ScanDirBlocQuery event"); } return r; }), transitionFn); } @override mapEventToState(ScanDirBlocEvent event) async* { _log.info("[mapEventToState] $event"); if (event is ScanDirBlocQueryBase) { yield* _onEventQuery(event); } else if (event is _ScanDirBlocExternalEvent) { yield* _onExternalEvent(event); } } @override close() { _fileRemovedEventListener.end(); _filePropertyUpdatedEventListener.end(); _propertyUpdatedSubscription?.cancel(); return super.close(); } Stream _onEventQuery(ScanDirBlocQueryBase ev) async* { yield ScanDirBlocLoading(ev.account, state.files); bool hasContent = state.files.isNotEmpty; if (!hasContent) { // show something instantly on first load ScanDirBlocState cacheState = ScanDirBlocInit(); await for (final s in _queryOffline(ev, () => cacheState)) { cacheState = s; } yield ScanDirBlocLoading(ev.account, cacheState.files); hasContent = cacheState.files.isNotEmpty; } ScanDirBlocState newState = ScanDirBlocInit(); if (!hasContent) { await for (final s in _queryOnline(ev, () => newState)) { newState = s; yield s; } } else { await for (final s in _queryOnline(ev, () => newState)) { newState = s; } if (newState is ScanDirBlocSuccess) { yield newState; } else if (newState is ScanDirBlocFailure) { yield ScanDirBlocFailure(ev.account, state.files, newState.exception); } } } Stream _onExternalEvent( _ScanDirBlocExternalEvent ev) async* { yield ScanDirBlocInconsistent(state.account, state.files); } void _onFileRemovedEvent(FileRemovedEvent ev) { if (state is ScanDirBlocInit) { // no data in this bloc, ignore return; } add(_ScanDirBlocExternalEvent()); } void _onFilePropertyUpdatedEvent(FilePropertyUpdatedEvent ev) { if (!ev.hasAnyProperties([ FilePropertyUpdatedEvent.propMetadata, FilePropertyUpdatedEvent.propIsArchived, FilePropertyUpdatedEvent.propOverrideDateTime, ])) { // not interested return; } if (state is ScanDirBlocInit) { // no data in this bloc, ignore return; } _successivePropertyUpdatedCount += 1; _propertyUpdatedSubscription?.cancel(); // only trigger the event on the 10th update or 10s after the last update if (_successivePropertyUpdatedCount % 10 == 0) { add(_ScanDirBlocExternalEvent()); } else { if (ev.hasAnyProperties([ FilePropertyUpdatedEvent.propIsArchived, FilePropertyUpdatedEvent.propOverrideDateTime, ])) { _propertyUpdatedSubscription = Future.delayed(const Duration(seconds: 2)).asStream().listen((_) { add(_ScanDirBlocExternalEvent()); _successivePropertyUpdatedCount = 0; }); } else { _propertyUpdatedSubscription = Future.delayed(const Duration(seconds: 10)).asStream().listen((_) { add(_ScanDirBlocExternalEvent()); _successivePropertyUpdatedCount = 0; }); } } } Stream _queryOffline( ScanDirBlocQueryBase ev, ScanDirBlocState Function() getState) => _queryWithFileDataSource(ev, getState, FileAppDbDataSource()); Stream _queryOnline( ScanDirBlocQueryBase ev, ScanDirBlocState Function() getState) { final stream = _queryWithFileDataSource(ev, getState, FileCachedDataSource(shouldCheckCache: _shouldCheckCache)); _shouldCheckCache = false; return stream; } Stream _queryWithFileDataSource(ScanDirBlocQueryBase ev, ScanDirBlocState Function() getState, FileDataSource dataSrc) async* { try { for (final r in ev.roots) { final dataStream = ScanDir(FileRepo(dataSrc))(ev.account, r); await for (final d in dataStream) { if (d is Exception || d is Error) { throw d; } yield ScanDirBlocLoading(ev.account, getState().files + d); } } yield ScanDirBlocSuccess(ev.account, getState().files); } catch (e) { _log.severe("[_queryWithFileDataSource] Exception while request", e); yield ScanDirBlocFailure(ev.account, getState().files, e); } } AppEventListener _fileRemovedEventListener; AppEventListener _filePropertyUpdatedEventListener; int _successivePropertyUpdatedCount = 0; StreamSubscription _propertyUpdatedSubscription; bool _shouldCheckCache = true; static final _log = Logger("bloc.scan_dir.ScanDirBloc"); }