mirror of
https://github.com/immich-app/immich.git
synced 2025-01-19 18:26:46 +01:00
feat(web, mobile): Options to show archived assets in map (#4293)
* Add include archive setting to map on web * open api * better naming for web isArchived variable * add withArchived setting to mobile * (e2e): tests for mapMarker endpoint and isArchived * isArchived to mobile * chore: cleanup test * chore: optimize e2e --------- Co-authored-by: shalong-tanwen <139912620+shalong-tanwen@users.noreply.github.com> Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
This commit is contained in:
parent
e5b4d09827
commit
e571880c16
22 changed files with 169 additions and 28 deletions
23
cli/src/api/open-api/api.ts
generated
23
cli/src/api/open-api/api.ts
generated
|
@ -6278,13 +6278,14 @@ export const AssetApiAxiosParamCreator = function (configuration?: Configuration
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
* @param {boolean} [isArchived]
|
||||||
* @param {boolean} [isFavorite]
|
* @param {boolean} [isFavorite]
|
||||||
* @param {string} [fileCreatedAfter]
|
* @param {string} [fileCreatedAfter]
|
||||||
* @param {string} [fileCreatedBefore]
|
* @param {string} [fileCreatedBefore]
|
||||||
* @param {*} [options] Override http request option.
|
* @param {*} [options] Override http request option.
|
||||||
* @throws {RequiredError}
|
* @throws {RequiredError}
|
||||||
*/
|
*/
|
||||||
getMapMarkers: async (isFavorite?: boolean, fileCreatedAfter?: string, fileCreatedBefore?: string, options: AxiosRequestConfig = {}): Promise<RequestArgs> => {
|
getMapMarkers: async (isArchived?: boolean, isFavorite?: boolean, fileCreatedAfter?: string, fileCreatedBefore?: string, options: AxiosRequestConfig = {}): Promise<RequestArgs> => {
|
||||||
const localVarPath = `/asset/map-marker`;
|
const localVarPath = `/asset/map-marker`;
|
||||||
// use dummy base URL string because the URL constructor only accepts absolute URLs.
|
// use dummy base URL string because the URL constructor only accepts absolute URLs.
|
||||||
const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL);
|
const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL);
|
||||||
|
@ -6306,6 +6307,10 @@ export const AssetApiAxiosParamCreator = function (configuration?: Configuration
|
||||||
// http bearer authentication required
|
// http bearer authentication required
|
||||||
await setBearerAuthToObject(localVarHeaderParameter, configuration)
|
await setBearerAuthToObject(localVarHeaderParameter, configuration)
|
||||||
|
|
||||||
|
if (isArchived !== undefined) {
|
||||||
|
localVarQueryParameter['isArchived'] = isArchived;
|
||||||
|
}
|
||||||
|
|
||||||
if (isFavorite !== undefined) {
|
if (isFavorite !== undefined) {
|
||||||
localVarQueryParameter['isFavorite'] = isFavorite;
|
localVarQueryParameter['isFavorite'] = isFavorite;
|
||||||
}
|
}
|
||||||
|
@ -7134,14 +7139,15 @@ export const AssetApiFp = function(configuration?: Configuration) {
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
* @param {boolean} [isArchived]
|
||||||
* @param {boolean} [isFavorite]
|
* @param {boolean} [isFavorite]
|
||||||
* @param {string} [fileCreatedAfter]
|
* @param {string} [fileCreatedAfter]
|
||||||
* @param {string} [fileCreatedBefore]
|
* @param {string} [fileCreatedBefore]
|
||||||
* @param {*} [options] Override http request option.
|
* @param {*} [options] Override http request option.
|
||||||
* @throws {RequiredError}
|
* @throws {RequiredError}
|
||||||
*/
|
*/
|
||||||
async getMapMarkers(isFavorite?: boolean, fileCreatedAfter?: string, fileCreatedBefore?: string, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<Array<MapMarkerResponseDto>>> {
|
async getMapMarkers(isArchived?: boolean, isFavorite?: boolean, fileCreatedAfter?: string, fileCreatedBefore?: string, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<Array<MapMarkerResponseDto>>> {
|
||||||
const localVarAxiosArgs = await localVarAxiosParamCreator.getMapMarkers(isFavorite, fileCreatedAfter, fileCreatedBefore, options);
|
const localVarAxiosArgs = await localVarAxiosParamCreator.getMapMarkers(isArchived, isFavorite, fileCreatedAfter, fileCreatedBefore, options);
|
||||||
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
|
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
|
@ -7428,7 +7434,7 @@ export const AssetApiFactory = function (configuration?: Configuration, basePath
|
||||||
* @throws {RequiredError}
|
* @throws {RequiredError}
|
||||||
*/
|
*/
|
||||||
getMapMarkers(requestParameters: AssetApiGetMapMarkersRequest = {}, options?: AxiosRequestConfig): AxiosPromise<Array<MapMarkerResponseDto>> {
|
getMapMarkers(requestParameters: AssetApiGetMapMarkersRequest = {}, options?: AxiosRequestConfig): AxiosPromise<Array<MapMarkerResponseDto>> {
|
||||||
return localVarFp.getMapMarkers(requestParameters.isFavorite, requestParameters.fileCreatedAfter, requestParameters.fileCreatedBefore, options).then((request) => request(axios, basePath));
|
return localVarFp.getMapMarkers(requestParameters.isArchived, requestParameters.isFavorite, requestParameters.fileCreatedAfter, requestParameters.fileCreatedBefore, options).then((request) => request(axios, basePath));
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
@ -7846,6 +7852,13 @@ export interface AssetApiGetDownloadInfoRequest {
|
||||||
* @interface AssetApiGetMapMarkersRequest
|
* @interface AssetApiGetMapMarkersRequest
|
||||||
*/
|
*/
|
||||||
export interface AssetApiGetMapMarkersRequest {
|
export interface AssetApiGetMapMarkersRequest {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @type {boolean}
|
||||||
|
* @memberof AssetApiGetMapMarkers
|
||||||
|
*/
|
||||||
|
readonly isArchived?: boolean
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @type {boolean}
|
* @type {boolean}
|
||||||
|
@ -8374,7 +8387,7 @@ export class AssetApi extends BaseAPI {
|
||||||
* @memberof AssetApi
|
* @memberof AssetApi
|
||||||
*/
|
*/
|
||||||
public getMapMarkers(requestParameters: AssetApiGetMapMarkersRequest = {}, options?: AxiosRequestConfig) {
|
public getMapMarkers(requestParameters: AssetApiGetMapMarkersRequest = {}, options?: AxiosRequestConfig) {
|
||||||
return AssetApiFp(this.configuration).getMapMarkers(requestParameters.isFavorite, requestParameters.fileCreatedAfter, requestParameters.fileCreatedBefore, options).then((request) => request(this.axios, this.basePath));
|
return AssetApiFp(this.configuration).getMapMarkers(requestParameters.isArchived, requestParameters.isFavorite, requestParameters.fileCreatedAfter, requestParameters.fileCreatedBefore, options).then((request) => request(this.axios, this.basePath));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -312,6 +312,7 @@
|
||||||
"map_settings_dialog_title": "Map Settings",
|
"map_settings_dialog_title": "Map Settings",
|
||||||
"map_settings_dark_mode": "Dark mode",
|
"map_settings_dark_mode": "Dark mode",
|
||||||
"map_settings_only_show_favorites": "Show Favorite Only",
|
"map_settings_only_show_favorites": "Show Favorite Only",
|
||||||
|
"map_settings_include_show_archived": "Include Archived",
|
||||||
"map_settings_only_relative_range": "Date range",
|
"map_settings_only_relative_range": "Date range",
|
||||||
"map_settings_dialog_cancel": "Cancel",
|
"map_settings_dialog_cancel": "Cancel",
|
||||||
"map_settings_dialog_save": "Save",
|
"map_settings_dialog_save": "Save",
|
||||||
|
|
|
@ -1,29 +1,33 @@
|
||||||
class MapState {
|
class MapState {
|
||||||
final bool isDarkTheme;
|
final bool isDarkTheme;
|
||||||
final bool showFavoriteOnly;
|
final bool showFavoriteOnly;
|
||||||
|
final bool includeArchived;
|
||||||
final int relativeTime;
|
final int relativeTime;
|
||||||
|
|
||||||
MapState({
|
MapState({
|
||||||
this.isDarkTheme = false,
|
this.isDarkTheme = false,
|
||||||
this.showFavoriteOnly = false,
|
this.showFavoriteOnly = false,
|
||||||
|
this.includeArchived = false,
|
||||||
this.relativeTime = 0,
|
this.relativeTime = 0,
|
||||||
});
|
});
|
||||||
|
|
||||||
MapState copyWith({
|
MapState copyWith({
|
||||||
bool? isDarkTheme,
|
bool? isDarkTheme,
|
||||||
bool? showFavoriteOnly,
|
bool? showFavoriteOnly,
|
||||||
|
bool? includeArchived,
|
||||||
int? relativeTime,
|
int? relativeTime,
|
||||||
}) {
|
}) {
|
||||||
return MapState(
|
return MapState(
|
||||||
isDarkTheme: isDarkTheme ?? this.isDarkTheme,
|
isDarkTheme: isDarkTheme ?? this.isDarkTheme,
|
||||||
showFavoriteOnly: showFavoriteOnly ?? this.showFavoriteOnly,
|
showFavoriteOnly: showFavoriteOnly ?? this.showFavoriteOnly,
|
||||||
|
includeArchived: includeArchived ?? this.includeArchived,
|
||||||
relativeTime: relativeTime ?? this.relativeTime,
|
relativeTime: relativeTime ?? this.relativeTime,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String toString() {
|
String toString() {
|
||||||
return 'MapSettingsState(isDarkTheme: $isDarkTheme, showFavoriteOnly: $showFavoriteOnly, relativeTime: $relativeTime)';
|
return 'MapSettingsState(isDarkTheme: $isDarkTheme, showFavoriteOnly: $showFavoriteOnly, relativeTime: $relativeTime, includeArchived: $includeArchived)';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
|
@ -33,13 +37,15 @@ class MapState {
|
||||||
return other is MapState &&
|
return other is MapState &&
|
||||||
other.isDarkTheme == isDarkTheme &&
|
other.isDarkTheme == isDarkTheme &&
|
||||||
other.showFavoriteOnly == showFavoriteOnly &&
|
other.showFavoriteOnly == showFavoriteOnly &&
|
||||||
other.relativeTime == relativeTime;
|
other.relativeTime == relativeTime &&
|
||||||
|
other.includeArchived == includeArchived;
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
int get hashCode {
|
int get hashCode {
|
||||||
return isDarkTheme.hashCode ^
|
return isDarkTheme.hashCode ^
|
||||||
showFavoriteOnly.hashCode ^
|
showFavoriteOnly.hashCode ^
|
||||||
relativeTime.hashCode;
|
relativeTime.hashCode ^
|
||||||
|
includeArchived.hashCode;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@ final mapMarkersProvider =
|
||||||
final mapState = ref.read(mapStateNotifier);
|
final mapState = ref.read(mapStateNotifier);
|
||||||
DateTime? fileCreatedAfter;
|
DateTime? fileCreatedAfter;
|
||||||
bool? isFavorite;
|
bool? isFavorite;
|
||||||
|
bool? isIncludeArchived;
|
||||||
|
|
||||||
if (mapState.relativeTime != 0) {
|
if (mapState.relativeTime != 0) {
|
||||||
fileCreatedAfter =
|
fileCreatedAfter =
|
||||||
|
@ -20,8 +21,13 @@ final mapMarkersProvider =
|
||||||
isFavorite = true;
|
isFavorite = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!mapState.includeArchived) {
|
||||||
|
isIncludeArchived = false;
|
||||||
|
}
|
||||||
|
|
||||||
final markers = await service.getMapMarkers(
|
final markers = await service.getMapMarkers(
|
||||||
isFavorite: isFavorite,
|
isFavorite: isFavorite,
|
||||||
|
withArchived: isIncludeArchived,
|
||||||
fileCreatedAfter: fileCreatedAfter,
|
fileCreatedAfter: fileCreatedAfter,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -11,6 +11,8 @@ class MapStateNotifier extends StateNotifier<MapState> {
|
||||||
.getSetting<bool>(AppSettingsEnum.mapThemeMode),
|
.getSetting<bool>(AppSettingsEnum.mapThemeMode),
|
||||||
showFavoriteOnly: appSettingsProvider
|
showFavoriteOnly: appSettingsProvider
|
||||||
.getSetting<bool>(AppSettingsEnum.mapShowFavoriteOnly),
|
.getSetting<bool>(AppSettingsEnum.mapShowFavoriteOnly),
|
||||||
|
includeArchived: appSettingsProvider
|
||||||
|
.getSetting<bool>(AppSettingsEnum.mapIncludeArchived),
|
||||||
relativeTime: appSettingsProvider
|
relativeTime: appSettingsProvider
|
||||||
.getSetting<int>(AppSettingsEnum.mapRelativeDate),
|
.getSetting<int>(AppSettingsEnum.mapRelativeDate),
|
||||||
),
|
),
|
||||||
|
@ -31,11 +33,19 @@ class MapStateNotifier extends StateNotifier<MapState> {
|
||||||
void switchFavoriteOnly(bool isFavoriteOnly) {
|
void switchFavoriteOnly(bool isFavoriteOnly) {
|
||||||
appSettingsProvider.setSetting(
|
appSettingsProvider.setSetting(
|
||||||
AppSettingsEnum.mapShowFavoriteOnly,
|
AppSettingsEnum.mapShowFavoriteOnly,
|
||||||
appSettingsProvider,
|
isFavoriteOnly,
|
||||||
);
|
);
|
||||||
state = state.copyWith(showFavoriteOnly: isFavoriteOnly);
|
state = state.copyWith(showFavoriteOnly: isFavoriteOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void switchIncludeArchived(bool isIncludeArchived) {
|
||||||
|
appSettingsProvider.setSetting(
|
||||||
|
AppSettingsEnum.mapIncludeArchived,
|
||||||
|
isIncludeArchived,
|
||||||
|
);
|
||||||
|
state = state.copyWith(includeArchived: isIncludeArchived);
|
||||||
|
}
|
||||||
|
|
||||||
void setRelativeTime(int relativeTime) {
|
void setRelativeTime(int relativeTime) {
|
||||||
appSettingsProvider.setSetting(
|
appSettingsProvider.setSetting(
|
||||||
AppSettingsEnum.mapRelativeDate,
|
AppSettingsEnum.mapRelativeDate,
|
||||||
|
|
|
@ -23,12 +23,14 @@ class MapSerivce {
|
||||||
|
|
||||||
Future<List<MapMarkerResponseDto>> getMapMarkers({
|
Future<List<MapMarkerResponseDto>> getMapMarkers({
|
||||||
bool? isFavorite,
|
bool? isFavorite,
|
||||||
|
bool? withArchived,
|
||||||
DateTime? fileCreatedAfter,
|
DateTime? fileCreatedAfter,
|
||||||
DateTime? fileCreatedBefore,
|
DateTime? fileCreatedBefore,
|
||||||
}) async {
|
}) async {
|
||||||
try {
|
try {
|
||||||
final markers = await _apiService.assetApi.getMapMarkers(
|
final markers = await _apiService.assetApi.getMapMarkers(
|
||||||
isFavorite: isFavorite,
|
isFavorite: isFavorite,
|
||||||
|
isArchived: withArchived,
|
||||||
fileCreatedAfter: fileCreatedAfter,
|
fileCreatedAfter: fileCreatedAfter,
|
||||||
fileCreatedBefore: fileCreatedBefore,
|
fileCreatedBefore: fileCreatedBefore,
|
||||||
);
|
);
|
||||||
|
|
|
@ -13,6 +13,7 @@ class MapSettingsDialog extends HookConsumerWidget {
|
||||||
final mapSettings = ref.read(mapStateNotifier);
|
final mapSettings = ref.read(mapStateNotifier);
|
||||||
final isDarkMode = useState(mapSettings.isDarkTheme);
|
final isDarkMode = useState(mapSettings.isDarkTheme);
|
||||||
final showFavoriteOnly = useState(mapSettings.showFavoriteOnly);
|
final showFavoriteOnly = useState(mapSettings.showFavoriteOnly);
|
||||||
|
final showIncludeArchived = useState(mapSettings.includeArchived);
|
||||||
final showRelativeDate = useState(mapSettings.relativeTime);
|
final showRelativeDate = useState(mapSettings.relativeTime);
|
||||||
final ThemeData theme = Theme.of(context);
|
final ThemeData theme = Theme.of(context);
|
||||||
|
|
||||||
|
@ -48,6 +49,22 @@ class MapSettingsDialog extends HookConsumerWidget {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Widget buildIncludeArchivedSetting() {
|
||||||
|
return SwitchListTile.adaptive(
|
||||||
|
value: showIncludeArchived.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
showIncludeArchived.value = value;
|
||||||
|
},
|
||||||
|
activeColor: theme.primaryColor,
|
||||||
|
dense: true,
|
||||||
|
title: Text(
|
||||||
|
"map_settings_include_show_archived".tr(),
|
||||||
|
style:
|
||||||
|
theme.textTheme.labelLarge?.copyWith(fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
Widget buildDateRangeSetting() {
|
Widget buildDateRangeSetting() {
|
||||||
final now = DateTime.now();
|
final now = DateTime.now();
|
||||||
return DropdownMenu(
|
return DropdownMenu(
|
||||||
|
@ -127,6 +144,8 @@ class MapSettingsDialog extends HookConsumerWidget {
|
||||||
mapSettingsNotifier.switchTheme(isDarkMode.value);
|
mapSettingsNotifier.switchTheme(isDarkMode.value);
|
||||||
mapSettingsNotifier.switchFavoriteOnly(showFavoriteOnly.value);
|
mapSettingsNotifier.switchFavoriteOnly(showFavoriteOnly.value);
|
||||||
mapSettingsNotifier.setRelativeTime(showRelativeDate.value);
|
mapSettingsNotifier.setRelativeTime(showRelativeDate.value);
|
||||||
|
mapSettingsNotifier
|
||||||
|
.switchIncludeArchived(showIncludeArchived.value);
|
||||||
Navigator.of(context).pop();
|
Navigator.of(context).pop();
|
||||||
},
|
},
|
||||||
style: TextButton.styleFrom(
|
style: TextButton.styleFrom(
|
||||||
|
@ -166,6 +185,7 @@ class MapSettingsDialog extends HookConsumerWidget {
|
||||||
children: [
|
children: [
|
||||||
buildMapThemeSetting(),
|
buildMapThemeSetting(),
|
||||||
buildFavoriteOnlySetting(),
|
buildFavoriteOnlySetting(),
|
||||||
|
buildIncludeArchivedSetting(),
|
||||||
const SizedBox(
|
const SizedBox(
|
||||||
height: 10,
|
height: 10,
|
||||||
),
|
),
|
||||||
|
|
|
@ -155,7 +155,8 @@ class MapPageState extends ConsumerState<MapPage> {
|
||||||
ref.listen(mapStateNotifier, (previous, next) {
|
ref.listen(mapStateNotifier, (previous, next) {
|
||||||
bool shouldRefetch =
|
bool shouldRefetch =
|
||||||
previous?.showFavoriteOnly != next.showFavoriteOnly ||
|
previous?.showFavoriteOnly != next.showFavoriteOnly ||
|
||||||
previous?.relativeTime != next.relativeTime;
|
previous?.relativeTime != next.relativeTime ||
|
||||||
|
previous?.includeArchived != next.includeArchived;
|
||||||
if (shouldRefetch) {
|
if (shouldRefetch) {
|
||||||
refetchMarkers.value = shouldRefetch;
|
refetchMarkers.value = shouldRefetch;
|
||||||
ref.invalidate(mapMarkersProvider);
|
ref.invalidate(mapMarkersProvider);
|
||||||
|
|
|
@ -48,6 +48,7 @@ enum AppSettingsEnum<T> {
|
||||||
preferRemoteImage<bool>(StoreKey.preferRemoteImage, null, false),
|
preferRemoteImage<bool>(StoreKey.preferRemoteImage, null, false),
|
||||||
mapThemeMode<bool>(StoreKey.mapThemeMode, null, false),
|
mapThemeMode<bool>(StoreKey.mapThemeMode, null, false),
|
||||||
mapShowFavoriteOnly<bool>(StoreKey.mapShowFavoriteOnly, null, false),
|
mapShowFavoriteOnly<bool>(StoreKey.mapShowFavoriteOnly, null, false),
|
||||||
|
mapIncludeArchived<bool>(StoreKey.mapIncludeArchived, null, false),
|
||||||
mapRelativeDate<int>(StoreKey.mapRelativeDate, null, 0),
|
mapRelativeDate<int>(StoreKey.mapRelativeDate, null, 0),
|
||||||
allowSelfSignedSSLCert<bool>(StoreKey.selfSignedCert, null, false),
|
allowSelfSignedSSLCert<bool>(StoreKey.selfSignedCert, null, false),
|
||||||
;
|
;
|
||||||
|
|
|
@ -179,6 +179,7 @@ enum StoreKey<T> {
|
||||||
mapShowFavoriteOnly<bool>(118, type: bool),
|
mapShowFavoriteOnly<bool>(118, type: bool),
|
||||||
mapRelativeDate<int>(119, type: int),
|
mapRelativeDate<int>(119, type: int),
|
||||||
selfSignedCert<bool>(120, type: bool),
|
selfSignedCert<bool>(120, type: bool),
|
||||||
|
mapIncludeArchived<bool>(121, type: bool),
|
||||||
;
|
;
|
||||||
|
|
||||||
const StoreKey(
|
const StoreKey(
|
||||||
|
|
BIN
mobile/openapi/doc/AssetApi.md
generated
BIN
mobile/openapi/doc/AssetApi.md
generated
Binary file not shown.
BIN
mobile/openapi/lib/api/asset_api.dart
generated
BIN
mobile/openapi/lib/api/asset_api.dart
generated
Binary file not shown.
BIN
mobile/openapi/test/asset_api_test.dart
generated
BIN
mobile/openapi/test/asset_api_test.dart
generated
Binary file not shown.
|
@ -1406,6 +1406,14 @@
|
||||||
"get": {
|
"get": {
|
||||||
"operationId": "getMapMarkers",
|
"operationId": "getMapMarkers",
|
||||||
"parameters": [
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "isArchived",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "boolean"
|
||||||
|
}
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "isFavorite",
|
"name": "isFavorite",
|
||||||
"required": false,
|
"required": false,
|
||||||
|
|
|
@ -22,6 +22,7 @@ export interface LivePhotoSearchOptions {
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface MapMarkerSearchOptions {
|
export interface MapMarkerSearchOptions {
|
||||||
|
isArchived?: boolean;
|
||||||
isFavorite?: boolean;
|
isFavorite?: boolean;
|
||||||
fileCreatedBefore?: Date;
|
fileCreatedBefore?: Date;
|
||||||
fileCreatedAfter?: Date;
|
fileCreatedAfter?: Date;
|
||||||
|
|
|
@ -4,6 +4,12 @@ import { IsBoolean, IsDate } from 'class-validator';
|
||||||
import { Optional, toBoolean } from '../../domain.util';
|
import { Optional, toBoolean } from '../../domain.util';
|
||||||
|
|
||||||
export class MapMarkerDto {
|
export class MapMarkerDto {
|
||||||
|
@ApiProperty()
|
||||||
|
@Optional()
|
||||||
|
@IsBoolean()
|
||||||
|
@Transform(toBoolean)
|
||||||
|
isArchived?: boolean;
|
||||||
|
|
||||||
@ApiProperty()
|
@ApiProperty()
|
||||||
@Optional()
|
@Optional()
|
||||||
@IsBoolean()
|
@IsBoolean()
|
||||||
|
|
|
@ -355,7 +355,7 @@ export class AssetRepository implements IAssetRepository {
|
||||||
}
|
}
|
||||||
|
|
||||||
async getMapMarkers(ownerId: string, options: MapMarkerSearchOptions = {}): Promise<MapMarker[]> {
|
async getMapMarkers(ownerId: string, options: MapMarkerSearchOptions = {}): Promise<MapMarker[]> {
|
||||||
const { isFavorite, fileCreatedAfter, fileCreatedBefore } = options;
|
const { isArchived, isFavorite, fileCreatedAfter, fileCreatedBefore } = options;
|
||||||
|
|
||||||
const assets = await this.repository.find({
|
const assets = await this.repository.find({
|
||||||
select: {
|
select: {
|
||||||
|
@ -368,7 +368,7 @@ export class AssetRepository implements IAssetRepository {
|
||||||
where: {
|
where: {
|
||||||
ownerId,
|
ownerId,
|
||||||
isVisible: true,
|
isVisible: true,
|
||||||
isArchived: false,
|
isArchived,
|
||||||
exifInfo: {
|
exifInfo: {
|
||||||
latitude: Not(IsNull()),
|
latitude: Not(IsNull()),
|
||||||
longitude: Not(IsNull()),
|
longitude: Not(IsNull()),
|
||||||
|
|
|
@ -56,6 +56,7 @@ const createAsset = (
|
||||||
deviceAssetId: `test_${id}`,
|
deviceAssetId: `test_${id}`,
|
||||||
deviceId: 'e2e-test',
|
deviceId: 'e2e-test',
|
||||||
libraryId,
|
libraryId,
|
||||||
|
isVisible: true,
|
||||||
fileCreatedAt: createdAt,
|
fileCreatedAt: createdAt,
|
||||||
fileModifiedAt: new Date(),
|
fileModifiedAt: new Date(),
|
||||||
type: AssetType.IMAGE,
|
type: AssetType.IMAGE,
|
||||||
|
@ -89,19 +90,25 @@ describe(`${AssetController.name} (e2e)`, () => {
|
||||||
await api.authApi.adminSignUp(server);
|
await api.authApi.adminSignUp(server);
|
||||||
const admin = await api.authApi.adminLogin(server);
|
const admin = await api.authApi.adminLogin(server);
|
||||||
|
|
||||||
const libraries = await api.libraryApi.getAll(server, admin.accessToken);
|
const [libraries] = await Promise.all([
|
||||||
|
api.libraryApi.getAll(server, admin.accessToken),
|
||||||
|
api.userApi.create(server, admin.accessToken, user1Dto),
|
||||||
|
api.userApi.create(server, admin.accessToken, user2Dto),
|
||||||
|
]);
|
||||||
|
|
||||||
const defaultLibrary = libraries[0];
|
const defaultLibrary = libraries[0];
|
||||||
|
|
||||||
await api.userApi.create(server, admin.accessToken, user1Dto);
|
[user1, user2] = await Promise.all([
|
||||||
user1 = await api.authApi.login(server, { email: user1Dto.email, password: user1Dto.password });
|
api.authApi.login(server, { email: user1Dto.email, password: user1Dto.password }),
|
||||||
|
api.authApi.login(server, { email: user2Dto.email, password: user2Dto.password }),
|
||||||
|
]);
|
||||||
|
|
||||||
asset1 = await createAsset(assetRepository, user1, defaultLibrary.id, new Date('1970-01-01'));
|
[asset1, asset2, asset3, asset4] = await Promise.all([
|
||||||
asset2 = await createAsset(assetRepository, user1, defaultLibrary.id, new Date('1970-01-02'));
|
createAsset(assetRepository, user1, defaultLibrary.id, new Date('1970-01-01')),
|
||||||
asset3 = await createAsset(assetRepository, user1, defaultLibrary.id, new Date('1970-02-01'));
|
createAsset(assetRepository, user1, defaultLibrary.id, new Date('1970-01-02')),
|
||||||
|
createAsset(assetRepository, user1, defaultLibrary.id, new Date('1970-02-01')),
|
||||||
await api.userApi.create(server, admin.accessToken, user2Dto);
|
createAsset(assetRepository, user2, defaultLibrary.id, new Date('1970-01-01')),
|
||||||
user2 = await api.authApi.login(server, { email: user2Dto.email, password: user2Dto.password });
|
]);
|
||||||
asset4 = await createAsset(assetRepository, user2, defaultLibrary.id, new Date('1970-01-01'));
|
|
||||||
});
|
});
|
||||||
|
|
||||||
afterAll(async () => {
|
afterAll(async () => {
|
||||||
|
@ -515,4 +522,45 @@ describe(`${AssetController.name} (e2e)`, () => {
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('GET /asset/map-marker', () => {
|
||||||
|
beforeEach(async () => {
|
||||||
|
await assetRepository.save({ id: asset1.id, isArchived: true });
|
||||||
|
await assetRepository.upsertExif({ assetId: asset1.id, latitude: 0, longitude: 0 });
|
||||||
|
await assetRepository.upsertExif({ assetId: asset2.id, latitude: 0, longitude: 0 });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should require authentication', async () => {
|
||||||
|
const { status, body } = await request(server).get('/asset/map-marker');
|
||||||
|
|
||||||
|
expect(status).toBe(401);
|
||||||
|
expect(body).toEqual(errorStub.unauthorized);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should get map markers for all non-archived assets', async () => {
|
||||||
|
const { status, body } = await request(server)
|
||||||
|
.get('/asset/map-marker')
|
||||||
|
.set('Authorization', `Bearer ${user1.accessToken}`);
|
||||||
|
|
||||||
|
expect(status).toBe(200);
|
||||||
|
expect(body).toHaveLength(2);
|
||||||
|
expect(body).toEqual(
|
||||||
|
expect.arrayContaining([
|
||||||
|
expect.objectContaining({ id: asset1.id }),
|
||||||
|
expect.objectContaining({ id: asset2.id }),
|
||||||
|
]),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should get all map markers', async () => {
|
||||||
|
const { status, body } = await request(server)
|
||||||
|
.get('/asset/map-marker')
|
||||||
|
.set('Authorization', `Bearer ${user1.accessToken}`)
|
||||||
|
.query({ isArchived: false });
|
||||||
|
|
||||||
|
expect(status).toBe(200);
|
||||||
|
expect(body).toHaveLength(1);
|
||||||
|
expect(body).toEqual([expect.objectContaining({ id: asset2.id })]);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
23
web/src/api/open-api/api.ts
generated
23
web/src/api/open-api/api.ts
generated
|
@ -6278,13 +6278,14 @@ export const AssetApiAxiosParamCreator = function (configuration?: Configuration
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
* @param {boolean} [isArchived]
|
||||||
* @param {boolean} [isFavorite]
|
* @param {boolean} [isFavorite]
|
||||||
* @param {string} [fileCreatedAfter]
|
* @param {string} [fileCreatedAfter]
|
||||||
* @param {string} [fileCreatedBefore]
|
* @param {string} [fileCreatedBefore]
|
||||||
* @param {*} [options] Override http request option.
|
* @param {*} [options] Override http request option.
|
||||||
* @throws {RequiredError}
|
* @throws {RequiredError}
|
||||||
*/
|
*/
|
||||||
getMapMarkers: async (isFavorite?: boolean, fileCreatedAfter?: string, fileCreatedBefore?: string, options: AxiosRequestConfig = {}): Promise<RequestArgs> => {
|
getMapMarkers: async (isArchived?: boolean, isFavorite?: boolean, fileCreatedAfter?: string, fileCreatedBefore?: string, options: AxiosRequestConfig = {}): Promise<RequestArgs> => {
|
||||||
const localVarPath = `/asset/map-marker`;
|
const localVarPath = `/asset/map-marker`;
|
||||||
// use dummy base URL string because the URL constructor only accepts absolute URLs.
|
// use dummy base URL string because the URL constructor only accepts absolute URLs.
|
||||||
const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL);
|
const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL);
|
||||||
|
@ -6306,6 +6307,10 @@ export const AssetApiAxiosParamCreator = function (configuration?: Configuration
|
||||||
// http bearer authentication required
|
// http bearer authentication required
|
||||||
await setBearerAuthToObject(localVarHeaderParameter, configuration)
|
await setBearerAuthToObject(localVarHeaderParameter, configuration)
|
||||||
|
|
||||||
|
if (isArchived !== undefined) {
|
||||||
|
localVarQueryParameter['isArchived'] = isArchived;
|
||||||
|
}
|
||||||
|
|
||||||
if (isFavorite !== undefined) {
|
if (isFavorite !== undefined) {
|
||||||
localVarQueryParameter['isFavorite'] = isFavorite;
|
localVarQueryParameter['isFavorite'] = isFavorite;
|
||||||
}
|
}
|
||||||
|
@ -7134,14 +7139,15 @@ export const AssetApiFp = function(configuration?: Configuration) {
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
* @param {boolean} [isArchived]
|
||||||
* @param {boolean} [isFavorite]
|
* @param {boolean} [isFavorite]
|
||||||
* @param {string} [fileCreatedAfter]
|
* @param {string} [fileCreatedAfter]
|
||||||
* @param {string} [fileCreatedBefore]
|
* @param {string} [fileCreatedBefore]
|
||||||
* @param {*} [options] Override http request option.
|
* @param {*} [options] Override http request option.
|
||||||
* @throws {RequiredError}
|
* @throws {RequiredError}
|
||||||
*/
|
*/
|
||||||
async getMapMarkers(isFavorite?: boolean, fileCreatedAfter?: string, fileCreatedBefore?: string, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<Array<MapMarkerResponseDto>>> {
|
async getMapMarkers(isArchived?: boolean, isFavorite?: boolean, fileCreatedAfter?: string, fileCreatedBefore?: string, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<Array<MapMarkerResponseDto>>> {
|
||||||
const localVarAxiosArgs = await localVarAxiosParamCreator.getMapMarkers(isFavorite, fileCreatedAfter, fileCreatedBefore, options);
|
const localVarAxiosArgs = await localVarAxiosParamCreator.getMapMarkers(isArchived, isFavorite, fileCreatedAfter, fileCreatedBefore, options);
|
||||||
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
|
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
|
@ -7428,7 +7434,7 @@ export const AssetApiFactory = function (configuration?: Configuration, basePath
|
||||||
* @throws {RequiredError}
|
* @throws {RequiredError}
|
||||||
*/
|
*/
|
||||||
getMapMarkers(requestParameters: AssetApiGetMapMarkersRequest = {}, options?: AxiosRequestConfig): AxiosPromise<Array<MapMarkerResponseDto>> {
|
getMapMarkers(requestParameters: AssetApiGetMapMarkersRequest = {}, options?: AxiosRequestConfig): AxiosPromise<Array<MapMarkerResponseDto>> {
|
||||||
return localVarFp.getMapMarkers(requestParameters.isFavorite, requestParameters.fileCreatedAfter, requestParameters.fileCreatedBefore, options).then((request) => request(axios, basePath));
|
return localVarFp.getMapMarkers(requestParameters.isArchived, requestParameters.isFavorite, requestParameters.fileCreatedAfter, requestParameters.fileCreatedBefore, options).then((request) => request(axios, basePath));
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
@ -7846,6 +7852,13 @@ export interface AssetApiGetDownloadInfoRequest {
|
||||||
* @interface AssetApiGetMapMarkersRequest
|
* @interface AssetApiGetMapMarkersRequest
|
||||||
*/
|
*/
|
||||||
export interface AssetApiGetMapMarkersRequest {
|
export interface AssetApiGetMapMarkersRequest {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @type {boolean}
|
||||||
|
* @memberof AssetApiGetMapMarkers
|
||||||
|
*/
|
||||||
|
readonly isArchived?: boolean
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @type {boolean}
|
* @type {boolean}
|
||||||
|
@ -8374,7 +8387,7 @@ export class AssetApi extends BaseAPI {
|
||||||
* @memberof AssetApi
|
* @memberof AssetApi
|
||||||
*/
|
*/
|
||||||
public getMapMarkers(requestParameters: AssetApiGetMapMarkersRequest = {}, options?: AxiosRequestConfig) {
|
public getMapMarkers(requestParameters: AssetApiGetMapMarkersRequest = {}, options?: AxiosRequestConfig) {
|
||||||
return AssetApiFp(this.configuration).getMapMarkers(requestParameters.isFavorite, requestParameters.fileCreatedAfter, requestParameters.fileCreatedBefore, options).then((request) => request(this.axios, this.basePath));
|
return AssetApiFp(this.configuration).getMapMarkers(requestParameters.isArchived, requestParameters.isFavorite, requestParameters.fileCreatedAfter, requestParameters.fileCreatedBefore, options).then((request) => request(this.axios, this.basePath));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -32,6 +32,7 @@
|
||||||
>
|
>
|
||||||
<SettingSwitch title="Allow dark mode" bind:checked={settings.allowDarkMode} />
|
<SettingSwitch title="Allow dark mode" bind:checked={settings.allowDarkMode} />
|
||||||
<SettingSwitch title="Only favorites" bind:checked={settings.onlyFavorites} />
|
<SettingSwitch title="Only favorites" bind:checked={settings.onlyFavorites} />
|
||||||
|
<SettingSwitch title="Include archived" bind:checked={settings.includeArchived} />
|
||||||
{#if customDateRange}
|
{#if customDateRange}
|
||||||
<div in:fly={{ y: 10, duration: 200 }} class="flex flex-col gap-4">
|
<div in:fly={{ y: 10, duration: 200 }} class="flex flex-col gap-4">
|
||||||
<div class="flex items-center justify-between gap-8">
|
<div class="flex items-center justify-between gap-8">
|
||||||
|
|
|
@ -21,6 +21,7 @@ export const locale = persisted<string | undefined>('locale', undefined, {
|
||||||
|
|
||||||
export interface MapSettings {
|
export interface MapSettings {
|
||||||
allowDarkMode: boolean;
|
allowDarkMode: boolean;
|
||||||
|
includeArchived: boolean;
|
||||||
onlyFavorites: boolean;
|
onlyFavorites: boolean;
|
||||||
relativeDate: string;
|
relativeDate: string;
|
||||||
dateAfter: string;
|
dateAfter: string;
|
||||||
|
@ -29,6 +30,7 @@ export interface MapSettings {
|
||||||
|
|
||||||
export const mapSettings = persisted<MapSettings>('map-settings', {
|
export const mapSettings = persisted<MapSettings>('map-settings', {
|
||||||
allowDarkMode: true,
|
allowDarkMode: true,
|
||||||
|
includeArchived: false,
|
||||||
onlyFavorites: false,
|
onlyFavorites: false,
|
||||||
relativeDate: '',
|
relativeDate: '',
|
||||||
dateAfter: '',
|
dateAfter: '',
|
||||||
|
|
|
@ -44,11 +44,12 @@
|
||||||
}
|
}
|
||||||
abortController = new AbortController();
|
abortController = new AbortController();
|
||||||
|
|
||||||
const { onlyFavorites } = $mapSettings;
|
const { includeArchived, onlyFavorites } = $mapSettings;
|
||||||
const { fileCreatedAfter, fileCreatedBefore } = getFileCreatedDates();
|
const { fileCreatedAfter, fileCreatedBefore } = getFileCreatedDates();
|
||||||
|
|
||||||
const { data } = await api.assetApi.getMapMarkers(
|
const { data } = await api.assetApi.getMapMarkers(
|
||||||
{
|
{
|
||||||
|
isArchived: includeArchived && undefined,
|
||||||
isFavorite: onlyFavorites || undefined,
|
isFavorite: onlyFavorites || undefined,
|
||||||
fileCreatedAfter: fileCreatedAfter || undefined,
|
fileCreatedAfter: fileCreatedAfter || undefined,
|
||||||
fileCreatedBefore,
|
fileCreatedBefore,
|
||||||
|
|
Loading…
Reference in a new issue