mirror of
https://github.com/immich-app/immich.git
synced 2025-01-21 00:52:43 -05:00
chore(mobile): remove obsolete files (#2482)
This commit is contained in:
parent
c8e649f190
commit
89edbcacfa
9 changed files with 0 additions and 808 deletions
|
@ -1,77 +0,0 @@
|
||||||
import 'package:collection/collection.dart';
|
|
||||||
import 'package:immich_mobile/shared/models/asset.dart';
|
|
||||||
|
|
||||||
class AssetSelectionState {
|
|
||||||
final Set<String> selectedMonths;
|
|
||||||
final Set<Asset> selectedNewAssetsForAlbum;
|
|
||||||
final Set<Asset> selectedAdditionalAssetsForAlbum;
|
|
||||||
final Set<Asset> selectedAssetsInAlbumViewer;
|
|
||||||
final bool isMultiselectEnable;
|
|
||||||
|
|
||||||
/// Indicate the asset selection page is navigated from existing album
|
|
||||||
final bool isAlbumExist;
|
|
||||||
AssetSelectionState({
|
|
||||||
required this.selectedMonths,
|
|
||||||
required this.selectedNewAssetsForAlbum,
|
|
||||||
required this.selectedAdditionalAssetsForAlbum,
|
|
||||||
required this.selectedAssetsInAlbumViewer,
|
|
||||||
required this.isMultiselectEnable,
|
|
||||||
required this.isAlbumExist,
|
|
||||||
});
|
|
||||||
|
|
||||||
AssetSelectionState copyWith({
|
|
||||||
Set<String>? selectedMonths,
|
|
||||||
Set<Asset>? selectedNewAssetsForAlbum,
|
|
||||||
Set<Asset>? selectedAdditionalAssetsForAlbum,
|
|
||||||
Set<Asset>? selectedAssetsInAlbumViewer,
|
|
||||||
bool? isMultiselectEnable,
|
|
||||||
bool? isAlbumExist,
|
|
||||||
}) {
|
|
||||||
return AssetSelectionState(
|
|
||||||
selectedMonths: selectedMonths ?? this.selectedMonths,
|
|
||||||
selectedNewAssetsForAlbum:
|
|
||||||
selectedNewAssetsForAlbum ?? this.selectedNewAssetsForAlbum,
|
|
||||||
selectedAdditionalAssetsForAlbum: selectedAdditionalAssetsForAlbum ??
|
|
||||||
this.selectedAdditionalAssetsForAlbum,
|
|
||||||
selectedAssetsInAlbumViewer:
|
|
||||||
selectedAssetsInAlbumViewer ?? this.selectedAssetsInAlbumViewer,
|
|
||||||
isMultiselectEnable: isMultiselectEnable ?? this.isMultiselectEnable,
|
|
||||||
isAlbumExist: isAlbumExist ?? this.isAlbumExist,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
String toString() {
|
|
||||||
return 'AssetSelectionState(selectedMonths: $selectedMonths, selectedNewAssetsForAlbum: $selectedNewAssetsForAlbum, selectedAdditionalAssetsForAlbum: $selectedAdditionalAssetsForAlbum, selectedAssetsInAlbumViewer: $selectedAssetsInAlbumViewer, isMultiselectEnable: $isMultiselectEnable, isAlbumExist: $isAlbumExist)';
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
bool operator ==(Object other) {
|
|
||||||
if (identical(this, other)) return true;
|
|
||||||
final setEquals = const DeepCollectionEquality().equals;
|
|
||||||
|
|
||||||
return other is AssetSelectionState &&
|
|
||||||
setEquals(other.selectedMonths, selectedMonths) &&
|
|
||||||
setEquals(other.selectedNewAssetsForAlbum, selectedNewAssetsForAlbum) &&
|
|
||||||
setEquals(
|
|
||||||
other.selectedAdditionalAssetsForAlbum,
|
|
||||||
selectedAdditionalAssetsForAlbum,
|
|
||||||
) &&
|
|
||||||
setEquals(
|
|
||||||
other.selectedAssetsInAlbumViewer,
|
|
||||||
selectedAssetsInAlbumViewer,
|
|
||||||
) &&
|
|
||||||
other.isMultiselectEnable == isMultiselectEnable &&
|
|
||||||
other.isAlbumExist == isAlbumExist;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
int get hashCode {
|
|
||||||
return selectedMonths.hashCode ^
|
|
||||||
selectedNewAssetsForAlbum.hashCode ^
|
|
||||||
selectedAdditionalAssetsForAlbum.hashCode ^
|
|
||||||
selectedAssetsInAlbumViewer.hashCode ^
|
|
||||||
isMultiselectEnable.hashCode ^
|
|
||||||
isAlbumExist.hashCode;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,110 +0,0 @@
|
||||||
import 'dart:convert';
|
|
||||||
|
|
||||||
class LogInReponse {
|
|
||||||
final String accessToken;
|
|
||||||
final String userId;
|
|
||||||
final String userEmail;
|
|
||||||
final String firstName;
|
|
||||||
final String lastName;
|
|
||||||
final String profileImagePath;
|
|
||||||
final bool isAdmin;
|
|
||||||
final bool shouldChangePassword;
|
|
||||||
|
|
||||||
LogInReponse({
|
|
||||||
required this.accessToken,
|
|
||||||
required this.userId,
|
|
||||||
required this.userEmail,
|
|
||||||
required this.firstName,
|
|
||||||
required this.lastName,
|
|
||||||
required this.profileImagePath,
|
|
||||||
required this.isAdmin,
|
|
||||||
required this.shouldChangePassword,
|
|
||||||
});
|
|
||||||
|
|
||||||
LogInReponse copyWith({
|
|
||||||
String? accessToken,
|
|
||||||
String? userId,
|
|
||||||
String? userEmail,
|
|
||||||
String? firstName,
|
|
||||||
String? lastName,
|
|
||||||
String? profileImagePath,
|
|
||||||
bool? isAdmin,
|
|
||||||
bool? shouldChangePassword,
|
|
||||||
}) {
|
|
||||||
return LogInReponse(
|
|
||||||
accessToken: accessToken ?? this.accessToken,
|
|
||||||
userId: userId ?? this.userId,
|
|
||||||
userEmail: userEmail ?? this.userEmail,
|
|
||||||
firstName: firstName ?? this.firstName,
|
|
||||||
lastName: lastName ?? this.lastName,
|
|
||||||
profileImagePath: profileImagePath ?? this.profileImagePath,
|
|
||||||
isAdmin: isAdmin ?? this.isAdmin,
|
|
||||||
shouldChangePassword: shouldChangePassword ?? this.shouldChangePassword,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Map<String, dynamic> toMap() {
|
|
||||||
final result = <String, dynamic>{};
|
|
||||||
|
|
||||||
result.addAll({'accessToken': accessToken});
|
|
||||||
result.addAll({'userId': userId});
|
|
||||||
result.addAll({'userEmail': userEmail});
|
|
||||||
result.addAll({'firstName': firstName});
|
|
||||||
result.addAll({'lastName': lastName});
|
|
||||||
result.addAll({'profileImagePath': profileImagePath});
|
|
||||||
result.addAll({'isAdmin': isAdmin});
|
|
||||||
result.addAll({'shouldChangePassword': shouldChangePassword});
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
factory LogInReponse.fromMap(Map<String, dynamic> map) {
|
|
||||||
return LogInReponse(
|
|
||||||
accessToken: map['accessToken'] ?? '',
|
|
||||||
userId: map['userId'] ?? '',
|
|
||||||
userEmail: map['userEmail'] ?? '',
|
|
||||||
firstName: map['firstName'] ?? '',
|
|
||||||
lastName: map['lastName'] ?? '',
|
|
||||||
profileImagePath: map['profileImagePath'] ?? '',
|
|
||||||
isAdmin: map['isAdmin'] ?? false,
|
|
||||||
shouldChangePassword: map['shouldChangePassword'] ?? false,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
String toJson() => json.encode(toMap());
|
|
||||||
|
|
||||||
factory LogInReponse.fromJson(String source) =>
|
|
||||||
LogInReponse.fromMap(json.decode(source));
|
|
||||||
|
|
||||||
@override
|
|
||||||
String toString() {
|
|
||||||
return 'LogInReponse(accessToken: $accessToken, userId: $userId, userEmail: $userEmail, firstName: $firstName, lastName: $lastName, profileImagePath: $profileImagePath, isAdmin: $isAdmin, shouldChangePassword: $shouldChangePassword)';
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
bool operator ==(Object other) {
|
|
||||||
if (identical(this, other)) return true;
|
|
||||||
|
|
||||||
return other is LogInReponse &&
|
|
||||||
other.accessToken == accessToken &&
|
|
||||||
other.userId == userId &&
|
|
||||||
other.userEmail == userEmail &&
|
|
||||||
other.firstName == firstName &&
|
|
||||||
other.lastName == lastName &&
|
|
||||||
other.profileImagePath == profileImagePath &&
|
|
||||||
other.isAdmin == isAdmin &&
|
|
||||||
other.shouldChangePassword == shouldChangePassword;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
int get hashCode {
|
|
||||||
return accessToken.hashCode ^
|
|
||||||
userId.hashCode ^
|
|
||||||
userEmail.hashCode ^
|
|
||||||
firstName.hashCode ^
|
|
||||||
lastName.hashCode ^
|
|
||||||
profileImagePath.hashCode ^
|
|
||||||
isAdmin.hashCode ^
|
|
||||||
shouldChangePassword.hashCode;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,150 +0,0 @@
|
||||||
import 'package:easy_localization/easy_localization.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_hooks/flutter_hooks.dart';
|
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
||||||
import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
|
|
||||||
import 'package:immich_mobile/modules/settings/ui/cache_settings/cache_settings_slider_pref.dart';
|
|
||||||
import 'package:immich_mobile/shared/services/cache.service.dart';
|
|
||||||
import 'package:immich_mobile/utils/bytes_units.dart';
|
|
||||||
|
|
||||||
class CacheSettings extends HookConsumerWidget {
|
|
||||||
const CacheSettings({
|
|
||||||
Key? key,
|
|
||||||
}) : super(key: key);
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context, WidgetRef ref) {
|
|
||||||
final CacheService cacheService = ref.watch(cacheServiceProvider);
|
|
||||||
final clearCacheState = useState(false);
|
|
||||||
|
|
||||||
Future<void> clearCache() async {
|
|
||||||
await cacheService.emptyAllCaches();
|
|
||||||
clearCacheState.value = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget cacheStatisticsRow(String name, CacheType type) {
|
|
||||||
final cacheSize = useState(0);
|
|
||||||
final cacheAssets = useState(0);
|
|
||||||
|
|
||||||
if (!clearCacheState.value) {
|
|
||||||
final repo = cacheService.getCacheRepo(type);
|
|
||||||
|
|
||||||
repo.open().then((_) {
|
|
||||||
cacheSize.value = repo.getCacheSize();
|
|
||||||
cacheAssets.value = repo.getNumberOfCachedObjects();
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
cacheSize.value = 0;
|
|
||||||
cacheAssets.value = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Container(
|
|
||||||
margin: const EdgeInsets.only(left: 20, bottom: 10),
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
name,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const Text(
|
|
||||||
"cache_settings_statistics_assets",
|
|
||||||
style: TextStyle(color: Colors.grey),
|
|
||||||
).tr(
|
|
||||||
args: ["${cacheAssets.value}", formatBytes(cacheSize.value)],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return ExpansionTile(
|
|
||||||
expandedCrossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
textColor: Theme.of(context).primaryColor,
|
|
||||||
title: const Text(
|
|
||||||
'cache_settings_title',
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
).tr(),
|
|
||||||
subtitle: const Text(
|
|
||||||
'cache_settings_subtitle',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 13,
|
|
||||||
),
|
|
||||||
).tr(),
|
|
||||||
children: [
|
|
||||||
const CacheSettingsSliderPref(
|
|
||||||
setting: AppSettingsEnum.thumbnailCacheSize,
|
|
||||||
translationKey: "cache_settings_thumbnail_size",
|
|
||||||
min: 1000,
|
|
||||||
max: 20000,
|
|
||||||
divisions: 19,
|
|
||||||
),
|
|
||||||
const CacheSettingsSliderPref(
|
|
||||||
setting: AppSettingsEnum.imageCacheSize,
|
|
||||||
translationKey: "cache_settings_image_cache_size",
|
|
||||||
min: 0,
|
|
||||||
max: 1000,
|
|
||||||
divisions: 20,
|
|
||||||
),
|
|
||||||
const CacheSettingsSliderPref(
|
|
||||||
setting: AppSettingsEnum.albumThumbnailCacheSize,
|
|
||||||
translationKey: "cache_settings_album_thumbnails",
|
|
||||||
min: 0,
|
|
||||||
max: 1000,
|
|
||||||
divisions: 20,
|
|
||||||
),
|
|
||||||
ListTile(
|
|
||||||
title: const Text(
|
|
||||||
"cache_settings_statistics_title",
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 12,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
).tr(),
|
|
||||||
),
|
|
||||||
cacheStatisticsRow(
|
|
||||||
"cache_settings_statistics_thumbnail".tr(),
|
|
||||||
CacheType.thumbnail,
|
|
||||||
),
|
|
||||||
cacheStatisticsRow(
|
|
||||||
"cache_settings_statistics_album".tr(),
|
|
||||||
CacheType.albumThumbnail,
|
|
||||||
),
|
|
||||||
cacheStatisticsRow(
|
|
||||||
"cache_settings_statistics_shared".tr(),
|
|
||||||
CacheType.sharedAlbumThumbnail,
|
|
||||||
),
|
|
||||||
cacheStatisticsRow(
|
|
||||||
"cache_settings_statistics_full".tr(),
|
|
||||||
CacheType.imageViewerFull,
|
|
||||||
),
|
|
||||||
ListTile(
|
|
||||||
title: const Text(
|
|
||||||
"cache_settings_clear_cache_button_title",
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 12,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
).tr(),
|
|
||||||
),
|
|
||||||
Container(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
child: ElevatedButton(
|
|
||||||
onPressed: clearCache,
|
|
||||||
child: const Text(
|
|
||||||
"cache_settings_clear_cache_button",
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 12,
|
|
||||||
),
|
|
||||||
).tr(),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,63 +0,0 @@
|
||||||
import 'package:easy_localization/easy_localization.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_hooks/flutter_hooks.dart';
|
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
||||||
import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart';
|
|
||||||
import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
|
|
||||||
|
|
||||||
class CacheSettingsSliderPref extends HookConsumerWidget {
|
|
||||||
final AppSettingsEnum<int> setting;
|
|
||||||
final String translationKey;
|
|
||||||
final int min;
|
|
||||||
final int max;
|
|
||||||
final int divisions;
|
|
||||||
|
|
||||||
const CacheSettingsSliderPref({
|
|
||||||
Key? key,
|
|
||||||
required this.setting,
|
|
||||||
required this.translationKey,
|
|
||||||
required this.min,
|
|
||||||
required this.max,
|
|
||||||
required this.divisions,
|
|
||||||
}) : super(key: key);
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context, WidgetRef ref) {
|
|
||||||
final appSettingService = ref.watch(appSettingsServiceProvider);
|
|
||||||
|
|
||||||
final itemsValue = useState(appSettingService.getSetting<int>(setting));
|
|
||||||
|
|
||||||
void sliderChanged(double value) {
|
|
||||||
itemsValue.value = value.toInt();
|
|
||||||
}
|
|
||||||
|
|
||||||
void sliderChangedEnd(double value) {
|
|
||||||
appSettingService.setSetting(setting, value.toInt());
|
|
||||||
}
|
|
||||||
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
ListTile(
|
|
||||||
title: Text(
|
|
||||||
translationKey,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontSize: 12,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
).tr(args: ["${itemsValue.value.toInt()}"]),
|
|
||||||
),
|
|
||||||
Slider(
|
|
||||||
onChangeEnd: sliderChangedEnd,
|
|
||||||
onChanged: sliderChanged,
|
|
||||||
value: itemsValue.value.toDouble(),
|
|
||||||
min: min.toDouble(),
|
|
||||||
max: max.toDouble(),
|
|
||||||
divisions: divisions,
|
|
||||||
label: "${itemsValue.value.toInt()}",
|
|
||||||
activeColor: Theme.of(context).primaryColor,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,48 +0,0 @@
|
||||||
import 'package:easy_localization/easy_localization.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
||||||
|
|
||||||
class ExperimentalSettings extends HookConsumerWidget {
|
|
||||||
const ExperimentalSettings({
|
|
||||||
Key? key,
|
|
||||||
}) : super(key: key);
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context, WidgetRef ref) {
|
|
||||||
return ExpansionTile(
|
|
||||||
textColor: Theme.of(context).primaryColor,
|
|
||||||
title: const Text(
|
|
||||||
'experimental_settings_title',
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
).tr(),
|
|
||||||
subtitle: const Text(
|
|
||||||
'experimental_settings_subtitle',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 13,
|
|
||||||
),
|
|
||||||
).tr(),
|
|
||||||
children: const [
|
|
||||||
// SwitchListTile.adaptive(
|
|
||||||
// activeColor: Theme.of(context).primaryColor,
|
|
||||||
// title: const Text(
|
|
||||||
// "experimental_settings_new_asset_list_title",
|
|
||||||
// style: TextStyle(
|
|
||||||
// fontSize: 12,
|
|
||||||
// fontWeight: FontWeight.bold,
|
|
||||||
// ),
|
|
||||||
// ).tr(),
|
|
||||||
// subtitle: const Text(
|
|
||||||
// "experimental_settings_new_asset_list_subtitle",
|
|
||||||
// style: TextStyle(
|
|
||||||
// fontSize: 12,
|
|
||||||
// ),
|
|
||||||
// ).tr(),
|
|
||||||
// value: useExperimentalAssetGrid.value,
|
|
||||||
// onChanged: changeUseExperimentalAssetGrid,
|
|
||||||
// ),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,11 +0,0 @@
|
||||||
class ImageViewerPageData {
|
|
||||||
final String heroTag;
|
|
||||||
final String imageUrl;
|
|
||||||
final String thumbnailUrl;
|
|
||||||
|
|
||||||
ImageViewerPageData({
|
|
||||||
required this.heroTag,
|
|
||||||
required this.imageUrl,
|
|
||||||
required this.thumbnailUrl,
|
|
||||||
});
|
|
||||||
}
|
|
|
@ -1,57 +0,0 @@
|
||||||
import 'dart:convert';
|
|
||||||
|
|
||||||
class UploadProfileImageResponse {
|
|
||||||
final String userId;
|
|
||||||
final String profileImagePath;
|
|
||||||
UploadProfileImageResponse({
|
|
||||||
required this.userId,
|
|
||||||
required this.profileImagePath,
|
|
||||||
});
|
|
||||||
|
|
||||||
UploadProfileImageResponse copyWith({
|
|
||||||
String? userId,
|
|
||||||
String? profileImagePath,
|
|
||||||
}) {
|
|
||||||
return UploadProfileImageResponse(
|
|
||||||
userId: userId ?? this.userId,
|
|
||||||
profileImagePath: profileImagePath ?? this.profileImagePath,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Map<String, dynamic> toMap() {
|
|
||||||
final result = <String, dynamic>{};
|
|
||||||
|
|
||||||
result.addAll({'userId': userId});
|
|
||||||
result.addAll({'profileImagePath': profileImagePath});
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
factory UploadProfileImageResponse.fromMap(Map<String, dynamic> map) {
|
|
||||||
return UploadProfileImageResponse(
|
|
||||||
userId: map['userId'] ?? '',
|
|
||||||
profileImagePath: map['profileImagePath'] ?? '',
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
String toJson() => json.encode(toMap());
|
|
||||||
|
|
||||||
factory UploadProfileImageResponse.fromJson(String source) =>
|
|
||||||
UploadProfileImageResponse.fromMap(json.decode(source));
|
|
||||||
|
|
||||||
@override
|
|
||||||
String toString() =>
|
|
||||||
'UploadProfileImageReponse(userId: $userId, profileImagePath: $profileImagePath)';
|
|
||||||
|
|
||||||
@override
|
|
||||||
bool operator ==(Object other) {
|
|
||||||
if (identical(this, other)) return true;
|
|
||||||
|
|
||||||
return other is UploadProfileImageResponse &&
|
|
||||||
other.userId == userId &&
|
|
||||||
other.profileImagePath == profileImagePath;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
int get hashCode => userId.hashCode ^ profileImagePath.hashCode;
|
|
||||||
}
|
|
|
@ -1,82 +0,0 @@
|
||||||
// ignore: depend_on_referenced_packages
|
|
||||||
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
|
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
||||||
import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart';
|
|
||||||
import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
|
|
||||||
import 'package:immich_mobile/utils/immich_cache_info_repository.dart';
|
|
||||||
|
|
||||||
enum CacheType {
|
|
||||||
// Shared cache for asset thumbnails in various modules
|
|
||||||
thumbnail,
|
|
||||||
imageViewerPreview,
|
|
||||||
imageViewerFull,
|
|
||||||
albumThumbnail,
|
|
||||||
sharedAlbumThumbnail;
|
|
||||||
}
|
|
||||||
|
|
||||||
final cacheServiceProvider = Provider(
|
|
||||||
(ref) => CacheService(ref.watch(appSettingsServiceProvider)),
|
|
||||||
);
|
|
||||||
|
|
||||||
class CacheService {
|
|
||||||
final AppSettingsService _settingsService;
|
|
||||||
final _cacheRepositoryInstances = <CacheType, ImmichCacheRepository>{};
|
|
||||||
|
|
||||||
CacheService(this._settingsService);
|
|
||||||
|
|
||||||
BaseCacheManager getCache(CacheType type) {
|
|
||||||
return _getDefaultCache(
|
|
||||||
type.name,
|
|
||||||
_getCacheSize(type) + 1,
|
|
||||||
getCacheRepo(type),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
ImmichCacheRepository getCacheRepo(CacheType type) {
|
|
||||||
if (!_cacheRepositoryInstances.containsKey(type)) {
|
|
||||||
final repo = ImmichCacheInfoRepository(
|
|
||||||
"cache_${type.name}",
|
|
||||||
"cacheKeys_${type.name}",
|
|
||||||
);
|
|
||||||
_cacheRepositoryInstances[type] = repo;
|
|
||||||
}
|
|
||||||
|
|
||||||
return _cacheRepositoryInstances[type]!;
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> emptyAllCaches() async {
|
|
||||||
for (var type in CacheType.values) {
|
|
||||||
await getCache(type).emptyCache();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
int _getCacheSize(CacheType type) {
|
|
||||||
switch (type) {
|
|
||||||
case CacheType.thumbnail:
|
|
||||||
return _settingsService.getSetting(AppSettingsEnum.thumbnailCacheSize);
|
|
||||||
case CacheType.imageViewerPreview:
|
|
||||||
case CacheType.imageViewerFull:
|
|
||||||
return _settingsService.getSetting(AppSettingsEnum.imageCacheSize);
|
|
||||||
case CacheType.sharedAlbumThumbnail:
|
|
||||||
case CacheType.albumThumbnail:
|
|
||||||
return _settingsService
|
|
||||||
.getSetting(AppSettingsEnum.albumThumbnailCacheSize);
|
|
||||||
default:
|
|
||||||
return 200;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
BaseCacheManager _getDefaultCache(
|
|
||||||
String cacheName,
|
|
||||||
int size,
|
|
||||||
CacheInfoRepository repo,
|
|
||||||
) {
|
|
||||||
return CacheManager(
|
|
||||||
Config(
|
|
||||||
cacheName,
|
|
||||||
maxNrOfCacheObjects: size,
|
|
||||||
repo: repo,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,210 +0,0 @@
|
||||||
// ignore_for_file: depend_on_referenced_packages, implementation_imports
|
|
||||||
|
|
||||||
import 'dart:io';
|
|
||||||
import 'dart:math';
|
|
||||||
|
|
||||||
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
|
|
||||||
import 'package:flutter_cache_manager/src/storage/cache_object.dart';
|
|
||||||
import 'package:hive/hive.dart';
|
|
||||||
import 'package:path/path.dart' as p;
|
|
||||||
import 'package:path_provider/path_provider.dart';
|
|
||||||
|
|
||||||
// Implementation of a CacheInfoRepository based on Hive
|
|
||||||
abstract class ImmichCacheRepository extends CacheInfoRepository {
|
|
||||||
int getNumberOfCachedObjects();
|
|
||||||
int getCacheSize();
|
|
||||||
}
|
|
||||||
|
|
||||||
class ImmichCacheInfoRepository extends ImmichCacheRepository {
|
|
||||||
final String hiveBoxName;
|
|
||||||
final String keyLookupHiveBoxName;
|
|
||||||
|
|
||||||
// To circumvent some of the limitations of a non-relational key-value database,
|
|
||||||
// we use two hive boxes per cache.
|
|
||||||
// [cacheObjectLookupBox] maps ids to cache objects.
|
|
||||||
// [keyLookupHiveBox] maps keys to ids.
|
|
||||||
// The lookup of a cache object by key therefore involves two steps:
|
|
||||||
// id = keyLookupHiveBox[key]
|
|
||||||
// object = cacheObjectLookupBox[id]
|
|
||||||
late Box<Map<dynamic, dynamic>> cacheObjectLookupBox;
|
|
||||||
late Box<int> keyLookupHiveBox;
|
|
||||||
|
|
||||||
ImmichCacheInfoRepository(this.hiveBoxName, this.keyLookupHiveBoxName);
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<bool> close() async {
|
|
||||||
await cacheObjectLookupBox.close();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<int> delete(int id) async {
|
|
||||||
if (cacheObjectLookupBox.containsKey(id)) {
|
|
||||||
await cacheObjectLookupBox.delete(id);
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<int> deleteAll(Iterable<int> ids) async {
|
|
||||||
int deleted = 0;
|
|
||||||
for (var id in ids) {
|
|
||||||
if (cacheObjectLookupBox.containsKey(id)) {
|
|
||||||
deleted++;
|
|
||||||
await cacheObjectLookupBox.delete(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return deleted;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<void> deleteDataFile() async {
|
|
||||||
await cacheObjectLookupBox.clear();
|
|
||||||
await keyLookupHiveBox.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<bool> exists() async {
|
|
||||||
return cacheObjectLookupBox.isNotEmpty && keyLookupHiveBox.isNotEmpty;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<CacheObject?> get(String key) async {
|
|
||||||
if (!keyLookupHiveBox.containsKey(key)) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
int id = keyLookupHiveBox.get(key)!;
|
|
||||||
if (!cacheObjectLookupBox.containsKey(id)) {
|
|
||||||
keyLookupHiveBox.delete(key);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return _deserialize(cacheObjectLookupBox.get(id)!);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<List<CacheObject>> getAllObjects() async {
|
|
||||||
return cacheObjectLookupBox.values.map(_deserialize).toList();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<List<CacheObject>> getObjectsOverCapacity(int capacity) async {
|
|
||||||
if (cacheObjectLookupBox.length <= capacity) {
|
|
||||||
return List.empty();
|
|
||||||
}
|
|
||||||
var values = cacheObjectLookupBox.values.map(_deserialize).toList();
|
|
||||||
values.sort((CacheObject a, CacheObject b) {
|
|
||||||
final aTouched = a.touched ?? DateTime.fromMicrosecondsSinceEpoch(0);
|
|
||||||
final bTouched = b.touched ?? DateTime.fromMicrosecondsSinceEpoch(0);
|
|
||||||
|
|
||||||
return aTouched.compareTo(bTouched);
|
|
||||||
});
|
|
||||||
return values.skip(capacity).take(10).toList();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<List<CacheObject>> getOldObjects(Duration maxAge) async {
|
|
||||||
return cacheObjectLookupBox.values
|
|
||||||
.map(_deserialize)
|
|
||||||
.where((CacheObject element) {
|
|
||||||
DateTime touched =
|
|
||||||
element.touched ?? DateTime.fromMicrosecondsSinceEpoch(0);
|
|
||||||
return touched.isBefore(DateTime.now().subtract(maxAge));
|
|
||||||
}).toList();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<CacheObject> insert(
|
|
||||||
CacheObject cacheObject, {
|
|
||||||
bool setTouchedToNow = true,
|
|
||||||
}) async {
|
|
||||||
int newId = keyLookupHiveBox.length == 0
|
|
||||||
? 0
|
|
||||||
: keyLookupHiveBox.values.reduce(max) + 1;
|
|
||||||
cacheObject = cacheObject.copyWith(id: newId);
|
|
||||||
|
|
||||||
keyLookupHiveBox.put(cacheObject.key, newId);
|
|
||||||
cacheObjectLookupBox.put(newId, cacheObject.toMap());
|
|
||||||
|
|
||||||
return cacheObject;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<bool> open() async {
|
|
||||||
cacheObjectLookupBox = await Hive.openBox(hiveBoxName);
|
|
||||||
keyLookupHiveBox = await Hive.openBox(keyLookupHiveBoxName);
|
|
||||||
|
|
||||||
// The cache might have cleared by the operating system.
|
|
||||||
// This could create inconsistencies between the file system cache and database.
|
|
||||||
// To check whether the cache was cleared, a file within the cache directory
|
|
||||||
// is created for each database. If the file is absent, the cache was cleared and therefore
|
|
||||||
// the database has to be cleared as well.
|
|
||||||
if (!await _checkAndCreateAnchorFile()) {
|
|
||||||
await cacheObjectLookupBox.clear();
|
|
||||||
await keyLookupHiveBox.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
return cacheObjectLookupBox.isOpen;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<int> update(
|
|
||||||
CacheObject cacheObject, {
|
|
||||||
bool setTouchedToNow = true,
|
|
||||||
}) async {
|
|
||||||
if (cacheObject.id != null) {
|
|
||||||
cacheObjectLookupBox.put(cacheObject.id, cacheObject.toMap());
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future updateOrInsert(CacheObject cacheObject) {
|
|
||||||
if (cacheObject.id == null) {
|
|
||||||
return insert(cacheObject);
|
|
||||||
} else {
|
|
||||||
return update(cacheObject);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
int getNumberOfCachedObjects() {
|
|
||||||
return cacheObjectLookupBox.length;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
int getCacheSize() {
|
|
||||||
final cacheElementsWithSize =
|
|
||||||
cacheObjectLookupBox.values.map(_deserialize).map((e) => e.length ?? 0);
|
|
||||||
|
|
||||||
if (cacheElementsWithSize.isEmpty) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return cacheElementsWithSize.reduce((value, element) => value + element);
|
|
||||||
}
|
|
||||||
|
|
||||||
CacheObject _deserialize(Map serData) {
|
|
||||||
Map<String, dynamic> converted = {};
|
|
||||||
|
|
||||||
serData.forEach((key, value) {
|
|
||||||
converted[key.toString()] = value;
|
|
||||||
});
|
|
||||||
|
|
||||||
return CacheObject.fromMap(converted);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<bool> _checkAndCreateAnchorFile() async {
|
|
||||||
final tmpDir = await getTemporaryDirectory();
|
|
||||||
final cacheFile = File(p.join(tmpDir.path, "$hiveBoxName.tmp"));
|
|
||||||
|
|
||||||
if (await cacheFile.exists()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
await cacheFile.create();
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Add table
Reference in a new issue