2021-04-28 11:49:55 +02:00
|
|
|
import 'dart:async';
|
2021-04-10 06:28:12 +02:00
|
|
|
|
|
|
|
import 'package:bloc/bloc.dart';
|
2021-04-27 11:54:11 +02:00
|
|
|
import 'package:equatable/equatable.dart';
|
2021-04-11 17:38:01 +02:00
|
|
|
import 'package:kiwi/kiwi.dart';
|
2021-04-10 06:28:12 +02:00
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/account.dart';
|
|
|
|
import 'package:nc_photos/entity/file.dart';
|
2021-05-24 09:09:25 +02:00
|
|
|
import 'package:nc_photos/entity/file/data_source.dart';
|
2021-04-10 06:28:12 +02:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2021-04-27 11:54:11 +02:00
|
|
|
class ScanDirBlocQueryBase extends ScanDirBlocEvent {
|
|
|
|
const ScanDirBlocQueryBase(this.account, this.roots);
|
2021-04-10 06:28:12 +02:00
|
|
|
|
|
|
|
@override
|
|
|
|
toString() {
|
|
|
|
return "$runtimeType {"
|
|
|
|
"account: $account, "
|
|
|
|
"roots: ${roots.map((e) => e.path).toReadableString()}, "
|
|
|
|
"}";
|
|
|
|
}
|
|
|
|
|
|
|
|
final Account account;
|
|
|
|
final List<File> roots;
|
|
|
|
}
|
|
|
|
|
2021-04-27 11:54:11 +02:00
|
|
|
class ScanDirBlocQuery extends ScanDirBlocQueryBase with EquatableMixin {
|
|
|
|
const ScanDirBlocQuery(Account account, List<File> roots)
|
|
|
|
: super(account, roots);
|
|
|
|
|
|
|
|
@override
|
|
|
|
get props => [
|
|
|
|
account,
|
|
|
|
roots,
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
class ScanDirBlocRefresh extends ScanDirBlocQueryBase {
|
|
|
|
const ScanDirBlocRefresh(Account account, List<File> roots)
|
|
|
|
: super(account, roots);
|
|
|
|
}
|
|
|
|
|
2021-04-10 06:28:12 +02:00
|
|
|
/// 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<File> get files => _files;
|
|
|
|
|
|
|
|
@override
|
|
|
|
toString() {
|
|
|
|
return "$runtimeType {"
|
|
|
|
"account: $account, "
|
|
|
|
"files: List {length: ${files.length}}, "
|
|
|
|
"}";
|
|
|
|
}
|
|
|
|
|
|
|
|
final Account _account;
|
|
|
|
final List<File> _files;
|
|
|
|
}
|
|
|
|
|
|
|
|
class ScanDirBlocInit extends ScanDirBlocState {
|
|
|
|
const ScanDirBlocInit() : super(null, const []);
|
|
|
|
}
|
|
|
|
|
|
|
|
class ScanDirBlocLoading extends ScanDirBlocState {
|
|
|
|
const ScanDirBlocLoading(Account account, List<File> files)
|
|
|
|
: super(account, files);
|
|
|
|
}
|
|
|
|
|
|
|
|
class ScanDirBlocSuccess extends ScanDirBlocState {
|
|
|
|
const ScanDirBlocSuccess(Account account, List<File> files)
|
|
|
|
: super(account, files);
|
|
|
|
}
|
|
|
|
|
|
|
|
class ScanDirBlocFailure extends ScanDirBlocState {
|
|
|
|
const ScanDirBlocFailure(Account account, List<File> 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<File> files)
|
|
|
|
: super(account, files);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A bloc that return all files under a dir recursively
|
|
|
|
///
|
|
|
|
/// See [ScanDir]
|
|
|
|
class ScanDirBloc extends Bloc<ScanDirBlocEvent, ScanDirBlocState> {
|
|
|
|
ScanDirBloc() : super(ScanDirBlocInit()) {
|
|
|
|
_fileRemovedEventListener =
|
|
|
|
AppEventListener<FileRemovedEvent>(_onFileRemovedEvent);
|
2021-05-28 19:15:09 +02:00
|
|
|
_filePropertyUpdatedEventListener =
|
|
|
|
AppEventListener<FilePropertyUpdatedEvent>(_onFilePropertyUpdatedEvent);
|
2021-04-10 06:28:12 +02:00
|
|
|
_fileRemovedEventListener.begin();
|
2021-05-28 19:15:09 +02:00
|
|
|
_filePropertyUpdatedEventListener.begin();
|
2021-04-10 06:28:12 +02:00
|
|
|
}
|
|
|
|
|
2021-04-11 17:38:01 +02:00
|
|
|
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>("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<ScanDirBloc>(bloc, name: "ScanDirBloc($id)");
|
|
|
|
return bloc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-27 11:54:11 +02:00
|
|
|
@override
|
|
|
|
transformEvents(Stream<ScanDirBlocEvent> 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);
|
|
|
|
}
|
|
|
|
|
2021-04-10 06:28:12 +02:00
|
|
|
@override
|
|
|
|
mapEventToState(ScanDirBlocEvent event) async* {
|
|
|
|
_log.info("[mapEventToState] $event");
|
2021-04-27 11:54:11 +02:00
|
|
|
if (event is ScanDirBlocQueryBase) {
|
2021-04-10 06:28:12 +02:00
|
|
|
yield* _onEventQuery(event);
|
|
|
|
} else if (event is _ScanDirBlocExternalEvent) {
|
|
|
|
yield* _onExternalEvent(event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
close() {
|
|
|
|
_fileRemovedEventListener.end();
|
2021-05-28 19:15:09 +02:00
|
|
|
_filePropertyUpdatedEventListener.end();
|
|
|
|
_propertyUpdatedSubscription?.cancel();
|
2021-04-10 06:28:12 +02:00
|
|
|
return super.close();
|
|
|
|
}
|
|
|
|
|
2021-04-27 11:54:11 +02:00
|
|
|
Stream<ScanDirBlocState> _onEventQuery(ScanDirBlocQueryBase ev) async* {
|
2021-04-10 06:28:12 +02:00
|
|
|
yield ScanDirBlocLoading(ev.account, state.files);
|
2021-04-26 09:39:17 +02:00
|
|
|
bool hasContent = state.files.isNotEmpty;
|
2021-04-10 06:28:12 +02:00
|
|
|
|
2021-04-26 09:39:17 +02:00
|
|
|
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;
|
2021-04-10 06:28:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ScanDirBlocState newState = ScanDirBlocInit();
|
2021-04-26 09:39:17 +02:00
|
|
|
if (!hasContent) {
|
2021-04-10 06:28:12 +02:00
|
|
|
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) {
|
2021-04-26 09:39:17 +02:00
|
|
|
yield ScanDirBlocFailure(ev.account, state.files, newState.exception);
|
2021-04-10 06:28:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Stream<ScanDirBlocState> _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());
|
|
|
|
}
|
|
|
|
|
2021-05-28 19:15:09 +02:00
|
|
|
void _onFilePropertyUpdatedEvent(FilePropertyUpdatedEvent ev) {
|
|
|
|
if (!ev.hasAnyProperties([
|
|
|
|
FilePropertyUpdatedEvent.propMetadata,
|
|
|
|
])) {
|
|
|
|
// not interested
|
|
|
|
return;
|
|
|
|
}
|
2021-04-10 06:28:12 +02:00
|
|
|
if (state is ScanDirBlocInit) {
|
|
|
|
// no data in this bloc, ignore
|
|
|
|
return;
|
|
|
|
}
|
2021-04-28 11:49:55 +02:00
|
|
|
|
2021-05-28 19:15:09 +02:00
|
|
|
_successivePropertyUpdatedCount += 1;
|
|
|
|
_propertyUpdatedSubscription?.cancel();
|
2021-04-28 11:49:55 +02:00
|
|
|
// only trigger the event on the 10th update or 10s after the last update
|
2021-05-28 19:15:09 +02:00
|
|
|
if (_successivePropertyUpdatedCount % 10 == 0) {
|
2021-04-28 11:49:55 +02:00
|
|
|
add(_ScanDirBlocExternalEvent());
|
|
|
|
} else {
|
2021-05-28 19:15:09 +02:00
|
|
|
_propertyUpdatedSubscription =
|
2021-04-28 11:49:55 +02:00
|
|
|
Future.delayed(const Duration(seconds: 10)).asStream().listen((_) {
|
|
|
|
add(_ScanDirBlocExternalEvent());
|
2021-05-28 19:15:09 +02:00
|
|
|
_successivePropertyUpdatedCount = 0;
|
2021-04-28 11:49:55 +02:00
|
|
|
});
|
|
|
|
}
|
2021-04-10 06:28:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Stream<ScanDirBlocState> _queryOffline(
|
2021-04-27 11:54:11 +02:00
|
|
|
ScanDirBlocQueryBase ev, ScanDirBlocState Function() getState) =>
|
2021-04-10 06:28:12 +02:00
|
|
|
_queryWithFileDataSource(ev, getState, FileAppDbDataSource());
|
|
|
|
|
|
|
|
Stream<ScanDirBlocState> _queryOnline(
|
2021-05-24 09:01:42 +02:00
|
|
|
ScanDirBlocQueryBase ev, ScanDirBlocState Function() getState) {
|
|
|
|
final stream = _queryWithFileDataSource(ev, getState,
|
|
|
|
FileCachedDataSource(shouldCheckCache: _shouldCheckCache));
|
|
|
|
_shouldCheckCache = false;
|
|
|
|
return stream;
|
|
|
|
}
|
2021-04-10 06:28:12 +02:00
|
|
|
|
2021-04-27 11:54:11 +02:00
|
|
|
Stream<ScanDirBlocState> _queryWithFileDataSource(ScanDirBlocQueryBase ev,
|
2021-04-10 06:28:12 +02:00
|
|
|
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<FileRemovedEvent> _fileRemovedEventListener;
|
2021-05-28 19:15:09 +02:00
|
|
|
AppEventListener<FilePropertyUpdatedEvent> _filePropertyUpdatedEventListener;
|
2021-04-10 06:28:12 +02:00
|
|
|
|
2021-05-28 19:15:09 +02:00
|
|
|
int _successivePropertyUpdatedCount = 0;
|
|
|
|
StreamSubscription<void> _propertyUpdatedSubscription;
|
2021-04-28 11:49:55 +02:00
|
|
|
|
2021-05-24 09:01:42 +02:00
|
|
|
bool _shouldCheckCache = true;
|
|
|
|
|
2021-04-10 06:28:12 +02:00
|
|
|
static final _log = Logger("bloc.scan_dir.ScanDirBloc");
|
|
|
|
}
|