import 'dart:async'; import 'package:flutter/widgets.dart'; import 'package:hydrated_bloc/hydrated_bloc.dart'; import 'package:json_annotation/json_annotation.dart'; import 'package:paperless_api/paperless_api.dart'; import 'package:paperless_mobile/core/notifier/document_changed_notifier.dart'; import 'package:paperless_mobile/core/repository/label_repository.dart'; import 'package:paperless_mobile/core/repository/label_repository_state.dart'; import 'package:paperless_mobile/features/paged_document_view/cubit/paged_documents_state.dart'; import 'package:paperless_mobile/features/paged_document_view/cubit/document_paging_bloc_mixin.dart'; part 'inbox_cubit.g.dart'; part 'inbox_state.dart'; class InboxCubit extends HydratedCubit with DocumentPagingBlocMixin { final LabelRepository _labelRepository; final PaperlessDocumentsApi _documentsApi; @override final DocumentChangedNotifier notifier; final PaperlessServerStatsApi _statsApi; @override PaperlessDocumentsApi get api => _documentsApi; InboxCubit( this._documentsApi, this._statsApi, this._labelRepository, this.notifier, ) : super(InboxState( labels: _labelRepository.state, )) { notifier.addListener( this, onDeleted: remove, onUpdated: (document) { if (document.tags .toSet() .intersection(state.inboxTags.toSet()) .isEmpty) { remove(document); emit(state.copyWith(itemsInInboxCount: state.itemsInInboxCount - 1)); } else { replace(document); } }, ); _labelRepository.addListener( this, onChanged: (labels) { emit(state.copyWith(labels: labels)); }, ); } Future initialize() async { await refreshItemsInInboxCount(false); await loadInbox(); } Future refreshItemsInInboxCount([bool shouldLoadInbox = true]) async { final stats = await _statsApi.getServerStatistics(); if (stats.documentsInInbox != state.itemsInInboxCount && shouldLoadInbox) { await loadInbox(); } emit( state.copyWith( itemsInInboxCount: stats.documentsInInbox, ), ); } /// /// Fetches inbox tag ids and loads the inbox items (documents). /// Future loadInbox() async { if (!isClosed) { debugPrint("Initializing inbox..."); final inboxTags = await _labelRepository.findAllTags().then( (tags) => tags.where((t) => t.isInboxTag).map((t) => t.id!), ); if (inboxTags.isEmpty) { // no inbox tags = no inbox items. return emit( state.copyWith( hasLoaded: true, value: [], inboxTags: [], ), ); } if (!isClosed) { emit(state.copyWith(inboxTags: inboxTags)); updateFilter( filter: DocumentFilter( sortField: SortField.added, tags: TagsQuery.ids(include: inboxTags.toList()), ), ); } } } /// /// Fetches inbox tag ids and loads the inbox items (documents). /// Future reloadInbox() async { emit(state.copyWith(hasLoaded: false, isLoading: true)); final inboxTags = await _labelRepository.findAllTags().then( (tags) => tags.where((t) => t.isInboxTag).map((t) => t.id!), ); if (inboxTags.isEmpty) { // no inbox tags = no inbox items. return emit( state.copyWith( hasLoaded: true, value: [], inboxTags: [], ), ); } emit(state.copyWith(inboxTags: inboxTags)); updateFilter( filter: DocumentFilter( sortField: SortField.added, tags: TagsQuery.ids(include: inboxTags.toList()), ), ); } /// /// Updates the document with all inbox tags removed and removes the document /// from the inbox. /// Future> removeFromInbox(DocumentModel document) async { final tagsToRemove = document.tags.toSet().intersection(state.inboxTags.toSet()); final updatedTags = {...document.tags}..removeAll(tagsToRemove); final updatedDocument = await api.update( document.copyWith(tags: updatedTags), ); // Remove first so document is not replaced first. remove(document); notifier.notifyUpdated(updatedDocument); return tagsToRemove; } /// /// Adds the previously removed tags to the document and performs an update. /// Future undoRemoveFromInbox( DocumentModel document, Iterable removedTags, ) async { final updatedDocument = await _documentsApi.update( document.copyWith( tags: {...document.tags, ...removedTags}, ), ); notifier.notifyUpdated(updatedDocument); emit(state.copyWith(itemsInInboxCount: state.itemsInInboxCount + 1)); return reload(); } /// /// Removes inbox tags from all documents in the inbox. /// Future clearInbox() async { emit(state.copyWith(isLoading: true)); try { await _documentsApi.bulkAction( BulkModifyTagsAction.removeTags( state.documents.map((e) => e.id), state.inboxTags, ), ); emit(state.copyWith( hasLoaded: true, value: [], itemsInInboxCount: 0, )); } finally { emit(state.copyWith(isLoading: false)); } } Future assignAsn(DocumentModel document) async { if (document.archiveSerialNumber == null) { final int asn = await _documentsApi.findNextAsn(); final updatedDocument = await _documentsApi .update(document.copyWith(archiveSerialNumber: () => asn)); replace(updatedDocument); } } void acknowledgeHint() { emit(state.copyWith(isHintAcknowledged: true)); } @override InboxState fromJson(Map json) { return InboxState.fromJson(json); } @override Map toJson(InboxState state) { return state.toJson(); } @override Future close() { _labelRepository.removeListener(this); return super.close(); } @override Future onFilterUpdated(DocumentFilter filter) async {} }