2023-07-03 19:23:42 +02:00
|
|
|
import 'dart:async';
|
|
|
|
|
|
|
|
import 'package:copy_with/copy_with.dart';
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/account.dart';
|
|
|
|
import 'package:nc_photos/controller/account_pref_controller.dart';
|
|
|
|
import 'package:nc_photos/di_container.dart';
|
|
|
|
import 'package:nc_photos/entity/person.dart';
|
2024-10-31 16:50:54 +01:00
|
|
|
import 'package:nc_photos/exception_event.dart';
|
2023-07-03 19:23:42 +02:00
|
|
|
import 'package:nc_photos/use_case/person/list_person.dart';
|
|
|
|
import 'package:np_codegen/np_codegen.dart';
|
|
|
|
import 'package:rxdart/rxdart.dart';
|
|
|
|
|
|
|
|
part 'persons_controller.g.dart';
|
|
|
|
|
|
|
|
@genCopyWith
|
|
|
|
class PersonStreamEvent {
|
|
|
|
const PersonStreamEvent({
|
|
|
|
required this.data,
|
|
|
|
required this.hasNext,
|
|
|
|
});
|
|
|
|
|
|
|
|
final List<Person> data;
|
|
|
|
|
|
|
|
/// If true, the results are intermediate values and may not represent the
|
|
|
|
/// latest state
|
|
|
|
final bool hasNext;
|
|
|
|
}
|
|
|
|
|
|
|
|
@npLog
|
|
|
|
class PersonsController {
|
|
|
|
PersonsController(
|
|
|
|
this._c, {
|
|
|
|
required this.account,
|
|
|
|
required this.accountPrefController,
|
2023-12-13 17:50:00 +01:00
|
|
|
}) {
|
|
|
|
_subscriptions
|
|
|
|
.add(accountPrefController.personProvider.distinct().listen((event) {
|
|
|
|
reload();
|
|
|
|
}));
|
|
|
|
}
|
2023-07-03 19:23:42 +02:00
|
|
|
|
|
|
|
void dispose() {
|
2023-12-13 17:50:00 +01:00
|
|
|
for (final s in _subscriptions) {
|
|
|
|
s.cancel();
|
|
|
|
}
|
2023-07-03 19:23:42 +02:00
|
|
|
_personStreamContorller.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return a stream of [Person]s associated with [account]
|
|
|
|
///
|
|
|
|
/// There's no guarantee that the returned list is always sorted in some ways,
|
|
|
|
/// callers must sort it by themselves if the ordering is important
|
|
|
|
ValueStream<PersonStreamEvent> get stream {
|
|
|
|
if (!_isPersonStreamInited) {
|
|
|
|
_isPersonStreamInited = true;
|
|
|
|
unawaited(_load());
|
|
|
|
}
|
|
|
|
return _personStreamContorller.stream;
|
|
|
|
}
|
|
|
|
|
2024-10-31 16:50:54 +01:00
|
|
|
Stream<ExceptionEvent> get errorStream => _personErrorStreamController.stream;
|
|
|
|
|
2023-07-03 19:23:42 +02:00
|
|
|
Future<void> reload() async {
|
|
|
|
if (_isPersonStreamInited) {
|
|
|
|
return _load();
|
|
|
|
} else {
|
|
|
|
_log.warning("[reload] Not inited, ignore");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _load() async {
|
|
|
|
var lastData = _personStreamContorller.value.copyWith(hasNext: true);
|
|
|
|
_personStreamContorller.add(lastData);
|
|
|
|
final completer = Completer();
|
2023-12-13 17:49:12 +01:00
|
|
|
ListPerson(_c.withLocalRepo())(
|
2024-02-25 05:09:57 +01:00
|
|
|
account, accountPrefController.personProviderValue)
|
2023-12-13 17:49:12 +01:00
|
|
|
.listen(
|
2023-07-03 19:23:42 +02:00
|
|
|
(results) {
|
|
|
|
lastData = PersonStreamEvent(
|
|
|
|
data: results,
|
|
|
|
hasNext: true,
|
|
|
|
);
|
|
|
|
_personStreamContorller.add(lastData);
|
|
|
|
},
|
2024-10-31 16:50:54 +01:00
|
|
|
onError: _personErrorStreamController.add,
|
2023-07-03 19:23:42 +02:00
|
|
|
onDone: () => completer.complete(),
|
|
|
|
);
|
|
|
|
await completer.future;
|
|
|
|
_personStreamContorller.add(lastData.copyWith(hasNext: false));
|
|
|
|
}
|
|
|
|
|
|
|
|
final DiContainer _c;
|
|
|
|
final Account account;
|
|
|
|
final AccountPrefController accountPrefController;
|
|
|
|
|
2023-12-13 17:50:00 +01:00
|
|
|
final _subscriptions = <StreamSubscription>[];
|
|
|
|
|
2023-07-03 19:23:42 +02:00
|
|
|
var _isPersonStreamInited = false;
|
|
|
|
final _personStreamContorller = BehaviorSubject.seeded(
|
|
|
|
const PersonStreamEvent(data: [], hasNext: true),
|
|
|
|
);
|
2024-10-31 16:50:54 +01:00
|
|
|
final _personErrorStreamController =
|
|
|
|
StreamController<ExceptionEvent>.broadcast();
|
2023-07-03 19:23:42 +02:00
|
|
|
}
|