1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-04 02:46:47 +01:00

feat(web,server): user avatar color (#4779)

This commit is contained in:
martin 2023-11-14 04:10:35 +01:00 committed by GitHub
parent 14c7187539
commit d25a245049
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
58 changed files with 649 additions and 100 deletions

View file

@ -2355,6 +2355,12 @@ export interface OAuthConfigResponseDto {
* @interface PartnerResponseDto * @interface PartnerResponseDto
*/ */
export interface PartnerResponseDto { export interface PartnerResponseDto {
/**
*
* @type {UserAvatarColor}
* @memberof PartnerResponseDto
*/
'avatarColor': UserAvatarColor;
/** /**
* *
* @type {string} * @type {string}
@ -2440,6 +2446,8 @@ export interface PartnerResponseDto {
*/ */
'updatedAt': string; 'updatedAt': string;
} }
/** /**
* *
* @export * @export
@ -4344,6 +4352,12 @@ export interface UpdateTagDto {
* @interface UpdateUserDto * @interface UpdateUserDto
*/ */
export interface UpdateUserDto { export interface UpdateUserDto {
/**
*
* @type {UserAvatarColor}
* @memberof UpdateUserDto
*/
'avatarColor'?: UserAvatarColor;
/** /**
* *
* @type {string} * @type {string}
@ -4399,6 +4413,8 @@ export interface UpdateUserDto {
*/ */
'storageLabel'?: string; 'storageLabel'?: string;
} }
/** /**
* *
* @export * @export
@ -4436,12 +4452,40 @@ export interface UsageByUserDto {
*/ */
'videos': number; 'videos': number;
} }
/**
*
* @export
* @enum {string}
*/
export const UserAvatarColor = {
Primary: 'primary',
Pink: 'pink',
Red: 'red',
Yellow: 'yellow',
Blue: 'blue',
Green: 'green',
Purple: 'purple',
Orange: 'orange',
Gray: 'gray',
Amber: 'amber'
} as const;
export type UserAvatarColor = typeof UserAvatarColor[keyof typeof UserAvatarColor];
/** /**
* *
* @export * @export
* @interface UserDto * @interface UserDto
*/ */
export interface UserDto { export interface UserDto {
/**
*
* @type {UserAvatarColor}
* @memberof UserDto
*/
'avatarColor': UserAvatarColor;
/** /**
* *
* @type {string} * @type {string}
@ -4467,12 +4511,20 @@ export interface UserDto {
*/ */
'profileImagePath': string; 'profileImagePath': string;
} }
/** /**
* *
* @export * @export
* @interface UserResponseDto * @interface UserResponseDto
*/ */
export interface UserResponseDto { export interface UserResponseDto {
/**
*
* @type {UserAvatarColor}
* @memberof UserResponseDto
*/
'avatarColor': UserAvatarColor;
/** /**
* *
* @type {string} * @type {string}
@ -4552,6 +4604,8 @@ export interface UserResponseDto {
*/ */
'updatedAt': string; 'updatedAt': string;
} }
/** /**
* *
* @export * @export
@ -16477,6 +16531,44 @@ export const UserApiAxiosParamCreator = function (configuration?: Configuration)
options: localVarRequestOptions, options: localVarRequestOptions,
}; };
}, },
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
deleteProfileImage: async (options: AxiosRequestConfig = {}): Promise<RequestArgs> => {
const localVarPath = `/user/profile-image`;
// use dummy base URL string because the URL constructor only accepts absolute URLs.
const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL);
let baseOptions;
if (configuration) {
baseOptions = configuration.baseOptions;
}
const localVarRequestOptions = { method: 'DELETE', ...baseOptions, ...options};
const localVarHeaderParameter = {} as any;
const localVarQueryParameter = {} as any;
// authentication cookie required
// authentication api_key required
await setApiKeyToObject(localVarHeaderParameter, "x-api-key", configuration)
// authentication bearer required
// http bearer authentication required
await setBearerAuthToObject(localVarHeaderParameter, configuration)
setSearchParams(localVarUrlObj, localVarQueryParameter);
let headersFromBaseOptions = baseOptions && baseOptions.headers ? baseOptions.headers : {};
localVarRequestOptions.headers = {...localVarHeaderParameter, ...headersFromBaseOptions, ...options.headers};
return {
url: toPathString(localVarUrlObj),
options: localVarRequestOptions,
};
},
/** /**
* *
* @param {string} id * @param {string} id
@ -16802,6 +16894,15 @@ export const UserApiFp = function(configuration?: Configuration) {
const localVarAxiosArgs = await localVarAxiosParamCreator.createUser(createUserDto, options); const localVarAxiosArgs = await localVarAxiosParamCreator.createUser(createUserDto, options);
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration); return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
}, },
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
async deleteProfileImage(options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<void>> {
const localVarAxiosArgs = await localVarAxiosParamCreator.deleteProfileImage(options);
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
},
/** /**
* *
* @param {string} id * @param {string} id
@ -16899,6 +17000,14 @@ export const UserApiFactory = function (configuration?: Configuration, basePath?
createUser(requestParameters: UserApiCreateUserRequest, options?: AxiosRequestConfig): AxiosPromise<UserResponseDto> { createUser(requestParameters: UserApiCreateUserRequest, options?: AxiosRequestConfig): AxiosPromise<UserResponseDto> {
return localVarFp.createUser(requestParameters.createUserDto, options).then((request) => request(axios, basePath)); return localVarFp.createUser(requestParameters.createUserDto, options).then((request) => request(axios, basePath));
}, },
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
deleteProfileImage(options?: AxiosRequestConfig): AxiosPromise<void> {
return localVarFp.deleteProfileImage(options).then((request) => request(axios, basePath));
},
/** /**
* *
* @param {UserApiDeleteUserRequest} requestParameters Request parameters. * @param {UserApiDeleteUserRequest} requestParameters Request parameters.
@ -17105,6 +17214,16 @@ export class UserApi extends BaseAPI {
return UserApiFp(this.configuration).createUser(requestParameters.createUserDto, options).then((request) => request(this.axios, this.basePath)); return UserApiFp(this.configuration).createUser(requestParameters.createUserDto, options).then((request) => request(this.axios, this.basePath));
} }
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
* @memberof UserApi
*/
public deleteProfileImage(options?: AxiosRequestConfig) {
return UserApiFp(this.configuration).deleteProfileImage(options).then((request) => request(this.axios, this.basePath));
}
/** /**
* *
* @param {UserApiDeleteUserRequest} requestParameters Request parameters. * @param {UserApiDeleteUserRequest} requestParameters Request parameters.

View file

@ -117,12 +117,8 @@ class AlbumOptionsPage extends HookConsumerWidget {
buildOwnerInfo() { buildOwnerInfo() {
return ListTile( return ListTile(
leading: owner != null leading:
? UserCircleAvatar( owner != null ? UserCircleAvatar(user: owner) : const SizedBox(),
user: owner,
useRandomBackgroundColor: true,
)
: const SizedBox(),
title: Text( title: Text(
album.owner.value?.name ?? "", album.owner.value?.name ?? "",
style: const TextStyle( style: const TextStyle(
@ -151,7 +147,6 @@ class AlbumOptionsPage extends HookConsumerWidget {
return ListTile( return ListTile(
leading: UserCircleAvatar( leading: UserCircleAvatar(
user: user, user: user,
useRandomBackgroundColor: true,
radius: 22, radius: 22,
), ),
title: Text( title: Text(

View file

@ -217,7 +217,6 @@ class AlbumViewerPage extends HookConsumerWidget {
user: album.sharedUsers.toList()[index], user: album.sharedUsers.toList()[index],
radius: 18, radius: 18,
size: 36, size: 36,
useRandomBackgroundColor: true,
), ),
); );
}), }),

View file

@ -1,3 +1,5 @@
import 'dart:ui';
import 'package:immich_mobile/shared/models/album.dart'; import 'package:immich_mobile/shared/models/album.dart';
import 'package:immich_mobile/utils/hash.dart'; import 'package:immich_mobile/utils/hash.dart';
import 'package:isar/isar.dart'; import 'package:isar/isar.dart';
@ -16,6 +18,7 @@ class User {
this.isPartnerSharedBy = false, this.isPartnerSharedBy = false,
this.isPartnerSharedWith = false, this.isPartnerSharedWith = false,
this.profileImagePath = '', this.profileImagePath = '',
this.avatarColor = AvatarColorEnum.primary,
this.memoryEnabled = true, this.memoryEnabled = true,
this.inTimeline = false, this.inTimeline = false,
}); });
@ -32,6 +35,7 @@ class User {
profileImagePath = dto.profileImagePath, profileImagePath = dto.profileImagePath,
isAdmin = dto.isAdmin, isAdmin = dto.isAdmin,
memoryEnabled = dto.memoriesEnabled ?? false, memoryEnabled = dto.memoriesEnabled ?? false,
avatarColor = dto.avatarColor.toAvatarColor(),
inTimeline = false; inTimeline = false;
User.fromPartnerDto(PartnerResponseDto dto) User.fromPartnerDto(PartnerResponseDto dto)
@ -44,6 +48,7 @@ class User {
profileImagePath = dto.profileImagePath, profileImagePath = dto.profileImagePath,
isAdmin = dto.isAdmin, isAdmin = dto.isAdmin,
memoryEnabled = dto.memoriesEnabled ?? false, memoryEnabled = dto.memoriesEnabled ?? false,
avatarColor = dto.avatarColor.toAvatarColor(),
inTimeline = dto.inTimeline ?? false; inTimeline = dto.inTimeline ?? false;
@Index(unique: true, replace: false, type: IndexType.hash) @Index(unique: true, replace: false, type: IndexType.hash)
@ -55,6 +60,8 @@ class User {
bool isPartnerSharedWith; bool isPartnerSharedWith;
bool isAdmin; bool isAdmin;
String profileImagePath; String profileImagePath;
@Enumerated(EnumType.ordinal)
AvatarColorEnum avatarColor;
bool memoryEnabled; bool memoryEnabled;
bool inTimeline; bool inTimeline;
@ -68,6 +75,7 @@ class User {
if (other is! User) return false; if (other is! User) return false;
return id == other.id && return id == other.id &&
updatedAt.isAtSameMomentAs(other.updatedAt) && updatedAt.isAtSameMomentAs(other.updatedAt) &&
avatarColor == other.avatarColor &&
email == other.email && email == other.email &&
name == other.name && name == other.name &&
isPartnerSharedBy == other.isPartnerSharedBy && isPartnerSharedBy == other.isPartnerSharedBy &&
@ -88,7 +96,77 @@ class User {
isPartnerSharedBy.hashCode ^ isPartnerSharedBy.hashCode ^
isPartnerSharedWith.hashCode ^ isPartnerSharedWith.hashCode ^
profileImagePath.hashCode ^ profileImagePath.hashCode ^
avatarColor.hashCode ^
isAdmin.hashCode ^ isAdmin.hashCode ^
memoryEnabled.hashCode ^ memoryEnabled.hashCode ^
inTimeline.hashCode; inTimeline.hashCode;
} }
enum AvatarColorEnum {
// do not change this order or reuse indices for other purposes, adding is OK
primary,
pink,
red,
yellow,
blue,
green,
purple,
orange,
gray,
amber,
}
extension AvatarColorEnumHelper on UserAvatarColor {
AvatarColorEnum toAvatarColor() {
switch (this) {
case UserAvatarColor.primary:
return AvatarColorEnum.primary;
case UserAvatarColor.pink:
return AvatarColorEnum.pink;
case UserAvatarColor.red:
return AvatarColorEnum.red;
case UserAvatarColor.yellow:
return AvatarColorEnum.yellow;
case UserAvatarColor.blue:
return AvatarColorEnum.blue;
case UserAvatarColor.green:
return AvatarColorEnum.green;
case UserAvatarColor.purple:
return AvatarColorEnum.purple;
case UserAvatarColor.orange:
return AvatarColorEnum.orange;
case UserAvatarColor.gray:
return AvatarColorEnum.gray;
case UserAvatarColor.amber:
return AvatarColorEnum.amber;
}
return AvatarColorEnum.primary;
}
}
extension AvatarColorToColorHelper on AvatarColorEnum {
Color toColor([bool isDarkTheme = false]) {
switch (this) {
case AvatarColorEnum.primary:
return isDarkTheme ? const Color(0xFFABCBFA) : const Color(0xFF4250AF);
case AvatarColorEnum.pink:
return const Color.fromARGB(255, 244, 114, 182);
case AvatarColorEnum.red:
return const Color.fromARGB(255, 239, 68, 68);
case AvatarColorEnum.yellow:
return const Color.fromARGB(255, 234, 179, 8);
case AvatarColorEnum.blue:
return const Color.fromARGB(255, 59, 130, 246);
case AvatarColorEnum.green:
return const Color.fromARGB(255, 22, 163, 74);
case AvatarColorEnum.purple:
return const Color.fromARGB(255, 147, 51, 234);
case AvatarColorEnum.orange:
return const Color.fromARGB(255, 234, 88, 12);
case AvatarColorEnum.gray:
return const Color.fromARGB(255, 75, 85, 99);
case AvatarColorEnum.amber:
return const Color.fromARGB(255, 217, 119, 6);
}
}
}

Binary file not shown.

View file

@ -22,14 +22,12 @@ class AppBarProfileInfoBox extends HookConsumerWidget {
final user = Store.tryGet(StoreKey.currentUser); final user = Store.tryGet(StoreKey.currentUser);
buildUserProfileImage() { buildUserProfileImage() {
const immichImage = CircleAvatar( if (user == null) {
radius: 20, return const CircleAvatar(
backgroundImage: AssetImage('assets/immich-logo-no-outline.png'), radius: 20,
backgroundColor: Colors.transparent, backgroundImage: AssetImage('assets/immich-logo-no-outline.png'),
); backgroundColor: Colors.transparent,
);
if (authState.profileImagePath.isEmpty || user == null) {
return immichImage;
} }
final userImage = UserCircleAvatar( final userImage = UserCircleAvatar(
@ -38,18 +36,6 @@ class AppBarProfileInfoBox extends HookConsumerWidget {
user: user, user: user,
); );
if (uploadProfileImageStatus == UploadProfileStatus.idle) {
return authState.profileImagePath.isNotEmpty ? userImage : immichImage;
}
if (uploadProfileImageStatus == UploadProfileStatus.success) {
return userImage;
}
if (uploadProfileImageStatus == UploadProfileStatus.failure) {
return immichImage;
}
if (uploadProfileImageStatus == UploadProfileStatus.loading) { if (uploadProfileImageStatus == UploadProfileStatus.loading) {
return const SizedBox( return const SizedBox(
height: 40, height: 40,
@ -58,7 +44,7 @@ class AppBarProfileInfoBox extends HookConsumerWidget {
); );
} }
return immichImage; return userImage;
} }
pickUserProfileImage() async { pickUserProfileImage() async {

View file

@ -4,8 +4,6 @@ import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/shared/models/store.dart'; import 'package:immich_mobile/shared/models/store.dart';
import 'package:immich_mobile/shared/ui/app_bar_dialog/app_bar_dialog.dart'; import 'package:immich_mobile/shared/ui/app_bar_dialog/app_bar_dialog.dart';
import 'package:immich_mobile/shared/ui/user_circle_avatar.dart'; import 'package:immich_mobile/shared/ui/user_circle_avatar.dart';
import 'package:immich_mobile/modules/login/models/authentication_state.model.dart';
import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
import 'package:immich_mobile/routing/router.dart'; import 'package:immich_mobile/routing/router.dart';
import 'package:immich_mobile/modules/backup/models/backup_state.model.dart'; import 'package:immich_mobile/modules/backup/models/backup_state.model.dart';
@ -26,7 +24,6 @@ class ImmichAppBar extends ConsumerWidget implements PreferredSizeWidget {
final bool isEnableAutoBackup = final bool isEnableAutoBackup =
backupState.backgroundBackup || backupState.autoBackup; backupState.backgroundBackup || backupState.autoBackup;
final ServerInfo serverInfoState = ref.watch(serverInfoProvider); final ServerInfo serverInfoState = ref.watch(serverInfoProvider);
AuthenticationState authState = ref.watch(authenticationProvider);
final user = Store.tryGet(StoreKey.currentUser); final user = Store.tryGet(StoreKey.currentUser);
final isDarkTheme = context.isDarkTheme; final isDarkTheme = context.isDarkTheme;
const widgetSize = 30.0; const widgetSize = 30.0;
@ -55,7 +52,7 @@ class ImmichAppBar extends ConsumerWidget implements PreferredSizeWidget {
alignment: Alignment.bottomRight, alignment: Alignment.bottomRight,
isLabelVisible: serverInfoState.isVersionMismatch, isLabelVisible: serverInfoState.isVersionMismatch,
offset: const Offset(2, 2), offset: const Offset(2, 2),
child: authState.profileImagePath.isEmpty || user == null child: user == null
? const Icon( ? const Icon(
Icons.face_outlined, Icons.face_outlined,
size: widgetSize, size: widgetSize,

View file

@ -3,7 +3,6 @@ import 'dart:math';
import 'package:cached_network_image/cached_network_image.dart'; import 'package:cached_network_image/cached_network_image.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/shared/models/store.dart'; import 'package:immich_mobile/shared/models/store.dart';
import 'package:immich_mobile/shared/models/user.dart'; import 'package:immich_mobile/shared/models/user.dart';
import 'package:immich_mobile/shared/ui/transparent_image.dart'; import 'package:immich_mobile/shared/ui/transparent_image.dart';
@ -13,32 +12,17 @@ class UserCircleAvatar extends ConsumerWidget {
final User user; final User user;
double radius; double radius;
double size; double size;
bool useRandomBackgroundColor;
UserCircleAvatar({ UserCircleAvatar({
super.key, super.key,
this.radius = 22, this.radius = 22,
this.size = 44, this.size = 44,
this.useRandomBackgroundColor = false,
required this.user, required this.user,
}); });
@override @override
Widget build(BuildContext context, WidgetRef ref) { Widget build(BuildContext context, WidgetRef ref) {
final randomColors = [ bool isDarkTheme = Theme.of(context).brightness == Brightness.dark;
Colors.red[200],
Colors.blue[200],
Colors.green[200],
Colors.yellow[200],
Colors.purple[200],
Colors.orange[200],
Colors.pink[200],
Colors.teal[200],
Colors.indigo[200],
Colors.cyan[200],
Colors.brown[200],
];
final profileImageUrl = final profileImageUrl =
'${Store.get(StoreKey.serverEndpoint)}/user/profile-image/${user.id}?d=${Random().nextInt(1024)}'; '${Store.get(StoreKey.serverEndpoint)}/user/profile-image/${user.id}?d=${Random().nextInt(1024)}';
@ -46,15 +30,16 @@ class UserCircleAvatar extends ConsumerWidget {
user.name[0].toUpperCase(), user.name[0].toUpperCase(),
style: TextStyle( style: TextStyle(
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
color: context.isDarkTheme ? Colors.black : Colors.white, fontSize: 12,
color: isDarkTheme && user.avatarColor == AvatarColorEnum.primary
? Colors.black
: Colors.white,
), ),
); );
return CircleAvatar( return CircleAvatar(
backgroundColor: useRandomBackgroundColor backgroundColor: user.avatarColor.toColor(),
? randomColors[Random().nextInt(randomColors.length)]
: context.primaryColor,
radius: radius, radius: radius,
child: user.profileImagePath == "" child: user.profileImagePath.isEmpty
? textIcon ? textIcon
: ClipRRect( : ClipRRect(
borderRadius: BorderRadius.circular(50), borderRadius: BorderRadius.circular(50),

View file

@ -166,6 +166,7 @@ doc/UpdateTagDto.md
doc/UpdateUserDto.md doc/UpdateUserDto.md
doc/UsageByUserDto.md doc/UsageByUserDto.md
doc/UserApi.md doc/UserApi.md
doc/UserAvatarColor.md
doc/UserDto.md doc/UserDto.md
doc/UserResponseDto.md doc/UserResponseDto.md
doc/ValidateAccessTokenResponseDto.md doc/ValidateAccessTokenResponseDto.md
@ -343,6 +344,7 @@ lib/model/update_stack_parent_dto.dart
lib/model/update_tag_dto.dart lib/model/update_tag_dto.dart
lib/model/update_user_dto.dart lib/model/update_user_dto.dart
lib/model/usage_by_user_dto.dart lib/model/usage_by_user_dto.dart
lib/model/user_avatar_color.dart
lib/model/user_dto.dart lib/model/user_dto.dart
lib/model/user_response_dto.dart lib/model/user_response_dto.dart
lib/model/validate_access_token_response_dto.dart lib/model/validate_access_token_response_dto.dart
@ -511,6 +513,7 @@ test/update_tag_dto_test.dart
test/update_user_dto_test.dart test/update_user_dto_test.dart
test/usage_by_user_dto_test.dart test/usage_by_user_dto_test.dart
test/user_api_test.dart test/user_api_test.dart
test/user_avatar_color_test.dart
test/user_dto_test.dart test/user_dto_test.dart
test/user_response_dto_test.dart test/user_response_dto_test.dart
test/validate_access_token_response_dto_test.dart test/validate_access_token_response_dto_test.dart

BIN
mobile/openapi/README.md generated

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

BIN
mobile/openapi/doc/UserAvatarColor.md generated Normal file

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

View file

@ -5578,6 +5578,29 @@
} }
}, },
"/user/profile-image": { "/user/profile-image": {
"delete": {
"operationId": "deleteProfileImage",
"parameters": [],
"responses": {
"204": {
"description": ""
}
},
"security": [
{
"bearer": []
},
{
"cookie": []
},
{
"api_key": []
}
],
"tags": [
"User"
]
},
"post": { "post": {
"operationId": "createProfileImage", "operationId": "createProfileImage",
"parameters": [], "parameters": [],
@ -7632,6 +7655,9 @@
}, },
"PartnerResponseDto": { "PartnerResponseDto": {
"properties": { "properties": {
"avatarColor": {
"$ref": "#/components/schemas/UserAvatarColor"
},
"createdAt": { "createdAt": {
"format": "date-time", "format": "date-time",
"type": "string" "type": "string"
@ -7682,6 +7708,7 @@
} }
}, },
"required": [ "required": [
"avatarColor",
"id", "id",
"name", "name",
"email", "email",
@ -9140,6 +9167,9 @@
}, },
"UpdateUserDto": { "UpdateUserDto": {
"properties": { "properties": {
"avatarColor": {
"$ref": "#/components/schemas/UserAvatarColor"
},
"email": { "email": {
"type": "string" "type": "string"
}, },
@ -9202,8 +9232,26 @@
], ],
"type": "object" "type": "object"
}, },
"UserAvatarColor": {
"enum": [
"primary",
"pink",
"red",
"yellow",
"blue",
"green",
"purple",
"orange",
"gray",
"amber"
],
"type": "string"
},
"UserDto": { "UserDto": {
"properties": { "properties": {
"avatarColor": {
"$ref": "#/components/schemas/UserAvatarColor"
},
"email": { "email": {
"type": "string" "type": "string"
}, },
@ -9218,6 +9266,7 @@
} }
}, },
"required": [ "required": [
"avatarColor",
"id", "id",
"name", "name",
"email", "email",
@ -9227,6 +9276,9 @@
}, },
"UserResponseDto": { "UserResponseDto": {
"properties": { "properties": {
"avatarColor": {
"$ref": "#/components/schemas/UserAvatarColor"
},
"createdAt": { "createdAt": {
"format": "date-time", "format": "date-time",
"type": "string" "type": "string"
@ -9274,6 +9326,7 @@
} }
}, },
"required": [ "required": [
"avatarColor",
"id", "id",
"name", "name",
"email", "email",

View file

@ -248,6 +248,7 @@ describe('AuthService', () => {
userMock.getAdmin.mockResolvedValue(null); userMock.getAdmin.mockResolvedValue(null);
userMock.create.mockResolvedValue({ ...dto, id: 'admin', createdAt: new Date('2021-01-01') } as UserEntity); userMock.create.mockResolvedValue({ ...dto, id: 'admin', createdAt: new Date('2021-01-01') } as UserEntity);
await expect(sut.adminSignUp(dto)).resolves.toEqual({ await expect(sut.adminSignUp(dto)).resolves.toEqual({
avatarColor: expect.any(String),
id: 'admin', id: 'admin',
createdAt: new Date('2021-01-01'), createdAt: new Date('2021-01-01'),
email: 'test@immich.com', email: 'test@immich.com',

View file

@ -1,3 +1,4 @@
import { UserAvatarColor } from '@app/infra/entities';
import { BadRequestException } from '@nestjs/common'; import { BadRequestException } from '@nestjs/common';
import { authStub, newPartnerRepositoryMock, partnerStub } from '@test'; import { authStub, newPartnerRepositoryMock, partnerStub } from '@test';
import { IAccessRepository, IPartnerRepository, PartnerDirection } from '../repositories'; import { IAccessRepository, IPartnerRepository, PartnerDirection } from '../repositories';
@ -19,6 +20,7 @@ const responseDto = {
updatedAt: new Date('2021-01-01'), updatedAt: new Date('2021-01-01'),
externalPath: null, externalPath: null,
memoriesEnabled: true, memoriesEnabled: true,
avatarColor: UserAvatarColor.PRIMARY,
inTimeline: true, inTimeline: true,
}, },
user1: <PartnerResponseDto>{ user1: <PartnerResponseDto>{
@ -35,6 +37,7 @@ const responseDto = {
updatedAt: new Date('2021-01-01'), updatedAt: new Date('2021-01-01'),
externalPath: null, externalPath: null,
memoriesEnabled: true, memoriesEnabled: true,
avatarColor: UserAvatarColor.PRIMARY,
inTimeline: true, inTimeline: true,
}, },
}; };

View file

@ -1,6 +1,7 @@
import { UserAvatarColor } from '@app/infra/entities';
import { ApiProperty } from '@nestjs/swagger'; import { ApiProperty } from '@nestjs/swagger';
import { Transform } from 'class-transformer'; import { Transform } from 'class-transformer';
import { IsBoolean, IsEmail, IsNotEmpty, IsString, IsUUID } from 'class-validator'; import { IsBoolean, IsEmail, IsEnum, IsNotEmpty, IsString, IsUUID } from 'class-validator';
import { Optional, toEmail, toSanitized } from '../../domain.util'; import { Optional, toEmail, toSanitized } from '../../domain.util';
export class UpdateUserDto { export class UpdateUserDto {
@ -44,4 +45,9 @@ export class UpdateUserDto {
@Optional() @Optional()
@IsBoolean() @IsBoolean()
memoriesEnabled?: boolean; memoriesEnabled?: boolean;
@Optional()
@IsEnum(UserAvatarColor)
@ApiProperty({ enumName: 'UserAvatarColor', enum: UserAvatarColor })
avatarColor?: UserAvatarColor;
} }

View file

@ -1,10 +1,26 @@
import { UserEntity } from '@app/infra/entities'; import { UserAvatarColor, UserEntity } from '@app/infra/entities';
import { ApiProperty } from '@nestjs/swagger';
import { IsEnum } from 'class-validator';
export const getRandomAvatarColor = (user: UserEntity): UserAvatarColor => {
const values = Object.values(UserAvatarColor);
const randomIndex = Math.floor(
user.email
.split('')
.map((letter) => letter.charCodeAt(0))
.reduce((a, b) => a + b, 0) % values.length,
);
return values[randomIndex] as UserAvatarColor;
};
export class UserDto { export class UserDto {
id!: string; id!: string;
name!: string; name!: string;
email!: string; email!: string;
profileImagePath!: string; profileImagePath!: string;
@IsEnum(UserAvatarColor)
@ApiProperty({ enumName: 'UserAvatarColor', enum: UserAvatarColor })
avatarColor!: UserAvatarColor;
} }
export class UserResponseDto extends UserDto { export class UserResponseDto extends UserDto {
@ -25,6 +41,7 @@ export const mapSimpleUser = (entity: UserEntity): UserDto => {
email: entity.email, email: entity.email,
name: entity.name, name: entity.name,
profileImagePath: entity.profileImagePath, profileImagePath: entity.profileImagePath,
avatarColor: entity.avatarColor ?? getRandomAvatarColor(entity),
}; };
}; };

View file

@ -98,7 +98,6 @@ export class UserCore {
if (payload.storageLabel) { if (payload.storageLabel) {
payload.storageLabel = sanitize(payload.storageLabel); payload.storageLabel = sanitize(payload.storageLabel);
} }
const userEntity = await this.userRepository.create(payload); const userEntity = await this.userRepository.create(payload);
await this.libraryRepository.create({ await this.libraryRepository.create({
owner: { id: userEntity.id } as UserEntity, owner: { id: userEntity.id } as UserEntity,

View file

@ -323,17 +323,52 @@ describe(UserService.name, () => {
const file = { path: '/profile/path' } as Express.Multer.File; const file = { path: '/profile/path' } as Express.Multer.File;
userMock.update.mockResolvedValue({ ...userStub.admin, profileImagePath: file.path }); userMock.update.mockResolvedValue({ ...userStub.admin, profileImagePath: file.path });
await sut.createProfileImage(userStub.admin, file); await expect(sut.createProfileImage(userStub.admin, file)).rejects.toThrowError(BadRequestException);
expect(userMock.update).toHaveBeenCalledWith(userStub.admin.id, { profileImagePath: file.path });
}); });
it('should throw an error if the user profile could not be updated with the new image', async () => { it('should throw an error if the user profile could not be updated with the new image', async () => {
const file = { path: '/profile/path' } as Express.Multer.File; const file = { path: '/profile/path' } as Express.Multer.File;
userMock.get.mockResolvedValue(userStub.profilePath);
userMock.update.mockRejectedValue(new InternalServerErrorException('mocked error')); userMock.update.mockRejectedValue(new InternalServerErrorException('mocked error'));
await expect(sut.createProfileImage(userStub.admin, file)).rejects.toThrowError(InternalServerErrorException); await expect(sut.createProfileImage(userStub.admin, file)).rejects.toThrowError(InternalServerErrorException);
}); });
it('should delete the previous profile image', async () => {
const file = { path: '/profile/path' } as Express.Multer.File;
userMock.get.mockResolvedValue(userStub.profilePath);
const files = [userStub.profilePath.profileImagePath];
userMock.update.mockResolvedValue({ ...userStub.admin, profileImagePath: file.path });
await sut.createProfileImage(userStub.admin, file);
await expect(jobMock.queue.mock.calls).toEqual([[{ name: JobName.DELETE_FILES, data: { files } }]]);
});
it('should not delete the profile image if it has not been set', async () => {
const file = { path: '/profile/path' } as Express.Multer.File;
userMock.get.mockResolvedValue(userStub.admin);
userMock.update.mockResolvedValue({ ...userStub.admin, profileImagePath: file.path });
await sut.createProfileImage(userStub.admin, file);
expect(jobMock.queue).not.toHaveBeenCalled();
});
});
describe('deleteProfileImage', () => {
it('should send an http error has no profile image', async () => {
userMock.get.mockResolvedValue(userStub.admin);
await expect(sut.deleteProfileImage(userStub.admin)).rejects.toBeInstanceOf(BadRequestException);
expect(jobMock.queue).not.toHaveBeenCalled();
});
it('should delete the profile image if user has one', async () => {
userMock.get.mockResolvedValue(userStub.profilePath);
const files = [userStub.profilePath.profileImagePath];
await sut.deleteProfileImage(userStub.admin);
await expect(jobMock.queue.mock.calls).toEqual([[{ name: JobName.DELETE_FILES, data: { files } }]]);
});
}); });
describe('getUserProfileImage', () => { describe('getUserProfileImage', () => {

View file

@ -93,10 +93,23 @@ export class UserService {
authUser: AuthUserDto, authUser: AuthUserDto,
fileInfo: Express.Multer.File, fileInfo: Express.Multer.File,
): Promise<CreateProfileImageResponseDto> { ): Promise<CreateProfileImageResponseDto> {
const { profileImagePath: oldpath } = await this.findOrFail(authUser.id, { withDeleted: false });
const updatedUser = await this.userRepository.update(authUser.id, { profileImagePath: fileInfo.path }); const updatedUser = await this.userRepository.update(authUser.id, { profileImagePath: fileInfo.path });
if (oldpath !== '') {
await this.jobRepository.queue({ name: JobName.DELETE_FILES, data: { files: [oldpath] } });
}
return mapCreateProfileImageResponse(updatedUser.id, updatedUser.profileImagePath); return mapCreateProfileImageResponse(updatedUser.id, updatedUser.profileImagePath);
} }
async deleteProfileImage(authUser: AuthUserDto): Promise<void> {
const user = await this.findOrFail(authUser.id, { withDeleted: false });
if (user.profileImagePath === '') {
throw new BadRequestException("Can't delete a missing profile Image");
}
await this.userRepository.update(authUser.id, { profileImagePath: '' });
await this.jobRepository.queue({ name: JobName.DELETE_FILES, data: { files: [user.profileImagePath] } });
}
async getProfileImage(id: string): Promise<ImmichReadStream> { async getProfileImage(id: string): Promise<ImmichReadStream> {
const user = await this.findOrFail(id, {}); const user = await this.findOrFail(id, {});
if (!user.profileImagePath) { if (!user.profileImagePath) {
@ -111,7 +124,7 @@ export class UserService {
throw new BadRequestException('Admin account does not exist'); throw new BadRequestException('Admin account does not exist');
} }
const providedPassword = await ask(admin); const providedPassword = await ask(mapUser(admin));
const password = providedPassword || randomBytes(24).toString('base64').replace(/\W/g, ''); const password = providedPassword || randomBytes(24).toString('base64').replace(/\W/g, '');
await this.userCore.updateUser(admin, admin.id, { password }); await this.userCore.updateUser(admin, admin.id, { password });

View file

@ -12,6 +12,7 @@ import {
SignUpDto, SignUpDto,
UserResponseDto, UserResponseDto,
ValidateAccessTokenResponseDto, ValidateAccessTokenResponseDto,
mapUser,
} from '@app/domain'; } from '@app/domain';
import { Body, Controller, Delete, Get, HttpCode, HttpStatus, Param, Post, Req, Res } from '@nestjs/common'; import { Body, Controller, Delete, Get, HttpCode, HttpStatus, Param, Post, Req, Res } from '@nestjs/common';
import { ApiTags } from '@nestjs/swagger'; import { ApiTags } from '@nestjs/swagger';
@ -71,7 +72,7 @@ export class AuthController {
@Post('change-password') @Post('change-password')
@HttpCode(HttpStatus.OK) @HttpCode(HttpStatus.OK)
changePassword(@AuthUser() authUser: AuthUserDto, @Body() dto: ChangePasswordDto): Promise<UserResponseDto> { changePassword(@AuthUser() authUser: AuthUserDto, @Body() dto: ChangePasswordDto): Promise<UserResponseDto> {
return this.service.changePassword(authUser, dto); return this.service.changePassword(authUser, dto).then(mapUser);
} }
@Post('logout') @Post('logout')

View file

@ -13,6 +13,8 @@ import {
Delete, Delete,
Get, Get,
Header, Header,
HttpCode,
HttpStatus,
Param, Param,
Post, Post,
Put, Put,
@ -54,6 +56,12 @@ export class UserController {
return this.service.create(createUserDto); return this.service.create(createUserDto);
} }
@Delete('profile-image')
@HttpCode(HttpStatus.NO_CONTENT)
deleteProfileImage(@AuthUser() authUser: AuthUserDto): Promise<void> {
return this.service.deleteProfileImage(authUser);
}
@AdminRoute() @AdminRoute()
@Delete(':id') @Delete(':id')
deleteUser(@AuthUser() authUser: AuthUserDto, @Param() { id }: UUIDParamDto): Promise<UserResponseDto> { deleteUser(@AuthUser() authUser: AuthUserDto, @Param() { id }: UUIDParamDto): Promise<UserResponseDto> {

View file

@ -10,6 +10,19 @@ import {
import { AssetEntity } from './asset.entity'; import { AssetEntity } from './asset.entity';
import { TagEntity } from './tag.entity'; import { TagEntity } from './tag.entity';
export enum UserAvatarColor {
PRIMARY = 'primary',
PINK = 'pink',
RED = 'red',
YELLOW = 'yellow',
BLUE = 'blue',
GREEN = 'green',
PURPLE = 'purple',
ORANGE = 'orange',
GRAY = 'gray',
AMBER = 'amber',
}
@Entity('users') @Entity('users')
export class UserEntity { export class UserEntity {
@PrimaryGeneratedColumn('uuid') @PrimaryGeneratedColumn('uuid')
@ -18,6 +31,9 @@ export class UserEntity {
@Column({ default: '' }) @Column({ default: '' })
name!: string; name!: string;
@Column({ type: 'varchar', nullable: true })
avatarColor!: UserAvatarColor | null;
@Column({ default: false }) @Column({ default: false })
isAdmin!: boolean; isAdmin!: boolean;

View file

@ -0,0 +1,14 @@
import { MigrationInterface, QueryRunner } from "typeorm";
export class AddAvatarColor1699889987493 implements MigrationInterface {
name = 'AddAvatarColor1699889987493'
public async up(queryRunner: QueryRunner): Promise<void> {
await queryRunner.query(`ALTER TABLE "users" ADD "avatarColor" character varying`);
}
public async down(queryRunner: QueryRunner): Promise<void> {
await queryRunner.query(`ALTER TABLE "users" DROP COLUMN "avatarColor"`);
}
}

View file

@ -18,6 +18,7 @@ const password = 'Password123';
const email = 'admin@immich.app'; const email = 'admin@immich.app';
const adminSignupResponse = { const adminSignupResponse = {
avatarColor: expect.any(String),
id: expect.any(String), id: expect.any(String),
name: 'Immich Admin', name: 'Immich Admin',
email: 'admin@immich.app', email: 'admin@immich.app',

View file

@ -1,4 +1,4 @@
import { UserEntity } from '@app/infra/entities'; import { UserAvatarColor, UserEntity } from '@app/infra/entities';
import { authStub } from './auth.stub'; import { authStub } from './auth.stub';
export const userStub = { export const userStub = {
@ -17,6 +17,7 @@ export const userStub = {
tags: [], tags: [],
assets: [], assets: [],
memoriesEnabled: true, memoriesEnabled: true,
avatarColor: UserAvatarColor.PRIMARY,
}), }),
user1: Object.freeze<UserEntity>({ user1: Object.freeze<UserEntity>({
...authStub.user1, ...authStub.user1,
@ -33,6 +34,7 @@ export const userStub = {
tags: [], tags: [],
assets: [], assets: [],
memoriesEnabled: true, memoriesEnabled: true,
avatarColor: UserAvatarColor.PRIMARY,
}), }),
user2: Object.freeze<UserEntity>({ user2: Object.freeze<UserEntity>({
...authStub.user2, ...authStub.user2,
@ -49,6 +51,7 @@ export const userStub = {
tags: [], tags: [],
assets: [], assets: [],
memoriesEnabled: true, memoriesEnabled: true,
avatarColor: UserAvatarColor.PRIMARY,
}), }),
storageLabel: Object.freeze<UserEntity>({ storageLabel: Object.freeze<UserEntity>({
...authStub.user1, ...authStub.user1,
@ -65,6 +68,7 @@ export const userStub = {
tags: [], tags: [],
assets: [], assets: [],
memoriesEnabled: true, memoriesEnabled: true,
avatarColor: UserAvatarColor.PRIMARY,
}), }),
externalPath1: Object.freeze<UserEntity>({ externalPath1: Object.freeze<UserEntity>({
...authStub.user1, ...authStub.user1,
@ -81,6 +85,7 @@ export const userStub = {
tags: [], tags: [],
assets: [], assets: [],
memoriesEnabled: true, memoriesEnabled: true,
avatarColor: UserAvatarColor.PRIMARY,
}), }),
externalPath2: Object.freeze<UserEntity>({ externalPath2: Object.freeze<UserEntity>({
...authStub.user1, ...authStub.user1,
@ -97,6 +102,7 @@ export const userStub = {
tags: [], tags: [],
assets: [], assets: [],
memoriesEnabled: true, memoriesEnabled: true,
avatarColor: UserAvatarColor.PRIMARY,
}), }),
profilePath: Object.freeze<UserEntity>({ profilePath: Object.freeze<UserEntity>({
...authStub.user1, ...authStub.user1,
@ -113,5 +119,6 @@ export const userStub = {
tags: [], tags: [],
assets: [], assets: [],
memoriesEnabled: true, memoriesEnabled: true,
avatarColor: UserAvatarColor.PRIMARY,
}), }),
}; };

View file

@ -2355,6 +2355,12 @@ export interface OAuthConfigResponseDto {
* @interface PartnerResponseDto * @interface PartnerResponseDto
*/ */
export interface PartnerResponseDto { export interface PartnerResponseDto {
/**
*
* @type {UserAvatarColor}
* @memberof PartnerResponseDto
*/
'avatarColor': UserAvatarColor;
/** /**
* *
* @type {string} * @type {string}
@ -2440,6 +2446,8 @@ export interface PartnerResponseDto {
*/ */
'updatedAt': string; 'updatedAt': string;
} }
/** /**
* *
* @export * @export
@ -4344,6 +4352,12 @@ export interface UpdateTagDto {
* @interface UpdateUserDto * @interface UpdateUserDto
*/ */
export interface UpdateUserDto { export interface UpdateUserDto {
/**
*
* @type {UserAvatarColor}
* @memberof UpdateUserDto
*/
'avatarColor'?: UserAvatarColor;
/** /**
* *
* @type {string} * @type {string}
@ -4399,6 +4413,8 @@ export interface UpdateUserDto {
*/ */
'storageLabel'?: string; 'storageLabel'?: string;
} }
/** /**
* *
* @export * @export
@ -4436,12 +4452,40 @@ export interface UsageByUserDto {
*/ */
'videos': number; 'videos': number;
} }
/**
*
* @export
* @enum {string}
*/
export const UserAvatarColor = {
Primary: 'primary',
Pink: 'pink',
Red: 'red',
Yellow: 'yellow',
Blue: 'blue',
Green: 'green',
Purple: 'purple',
Orange: 'orange',
Gray: 'gray',
Amber: 'amber'
} as const;
export type UserAvatarColor = typeof UserAvatarColor[keyof typeof UserAvatarColor];
/** /**
* *
* @export * @export
* @interface UserDto * @interface UserDto
*/ */
export interface UserDto { export interface UserDto {
/**
*
* @type {UserAvatarColor}
* @memberof UserDto
*/
'avatarColor': UserAvatarColor;
/** /**
* *
* @type {string} * @type {string}
@ -4467,12 +4511,20 @@ export interface UserDto {
*/ */
'profileImagePath': string; 'profileImagePath': string;
} }
/** /**
* *
* @export * @export
* @interface UserResponseDto * @interface UserResponseDto
*/ */
export interface UserResponseDto { export interface UserResponseDto {
/**
*
* @type {UserAvatarColor}
* @memberof UserResponseDto
*/
'avatarColor': UserAvatarColor;
/** /**
* *
* @type {string} * @type {string}
@ -4552,6 +4604,8 @@ export interface UserResponseDto {
*/ */
'updatedAt': string; 'updatedAt': string;
} }
/** /**
* *
* @export * @export
@ -16477,6 +16531,44 @@ export const UserApiAxiosParamCreator = function (configuration?: Configuration)
options: localVarRequestOptions, options: localVarRequestOptions,
}; };
}, },
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
deleteProfileImage: async (options: AxiosRequestConfig = {}): Promise<RequestArgs> => {
const localVarPath = `/user/profile-image`;
// use dummy base URL string because the URL constructor only accepts absolute URLs.
const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL);
let baseOptions;
if (configuration) {
baseOptions = configuration.baseOptions;
}
const localVarRequestOptions = { method: 'DELETE', ...baseOptions, ...options};
const localVarHeaderParameter = {} as any;
const localVarQueryParameter = {} as any;
// authentication cookie required
// authentication api_key required
await setApiKeyToObject(localVarHeaderParameter, "x-api-key", configuration)
// authentication bearer required
// http bearer authentication required
await setBearerAuthToObject(localVarHeaderParameter, configuration)
setSearchParams(localVarUrlObj, localVarQueryParameter);
let headersFromBaseOptions = baseOptions && baseOptions.headers ? baseOptions.headers : {};
localVarRequestOptions.headers = {...localVarHeaderParameter, ...headersFromBaseOptions, ...options.headers};
return {
url: toPathString(localVarUrlObj),
options: localVarRequestOptions,
};
},
/** /**
* *
* @param {string} id * @param {string} id
@ -16802,6 +16894,15 @@ export const UserApiFp = function(configuration?: Configuration) {
const localVarAxiosArgs = await localVarAxiosParamCreator.createUser(createUserDto, options); const localVarAxiosArgs = await localVarAxiosParamCreator.createUser(createUserDto, options);
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration); return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
}, },
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
async deleteProfileImage(options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<void>> {
const localVarAxiosArgs = await localVarAxiosParamCreator.deleteProfileImage(options);
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
},
/** /**
* *
* @param {string} id * @param {string} id
@ -16899,6 +17000,14 @@ export const UserApiFactory = function (configuration?: Configuration, basePath?
createUser(requestParameters: UserApiCreateUserRequest, options?: AxiosRequestConfig): AxiosPromise<UserResponseDto> { createUser(requestParameters: UserApiCreateUserRequest, options?: AxiosRequestConfig): AxiosPromise<UserResponseDto> {
return localVarFp.createUser(requestParameters.createUserDto, options).then((request) => request(axios, basePath)); return localVarFp.createUser(requestParameters.createUserDto, options).then((request) => request(axios, basePath));
}, },
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
deleteProfileImage(options?: AxiosRequestConfig): AxiosPromise<void> {
return localVarFp.deleteProfileImage(options).then((request) => request(axios, basePath));
},
/** /**
* *
* @param {UserApiDeleteUserRequest} requestParameters Request parameters. * @param {UserApiDeleteUserRequest} requestParameters Request parameters.
@ -17105,6 +17214,16 @@ export class UserApi extends BaseAPI {
return UserApiFp(this.configuration).createUser(requestParameters.createUserDto, options).then((request) => request(this.axios, this.basePath)); return UserApiFp(this.configuration).createUser(requestParameters.createUserDto, options).then((request) => request(this.axios, this.basePath));
} }
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
* @memberof UserApi
*/
public deleteProfileImage(options?: AxiosRequestConfig) {
return UserApiFp(this.configuration).deleteProfileImage(options).then((request) => request(this.axios, this.basePath));
}
/** /**
* *
* @param {UserApiDeleteUserRequest} requestParameters Request parameters. * @param {UserApiDeleteUserRequest} requestParameters Request parameters.

View file

@ -77,7 +77,7 @@
<section class="immich-scrollbar max-h-[400px] overflow-y-auto pb-4"> <section class="immich-scrollbar max-h-[400px] overflow-y-auto pb-4">
<div class="flex w-full place-items-center justify-between gap-4 p-5"> <div class="flex w-full place-items-center justify-between gap-4 p-5">
<div class="flex place-items-center gap-4"> <div class="flex place-items-center gap-4">
<UserAvatar user={album.owner} size="md" autoColor /> <UserAvatar user={album.owner} size="md" />
<p class="text-sm font-medium">{album.owner.name}</p> <p class="text-sm font-medium">{album.owner.name}</p>
</div> </div>
@ -90,7 +90,7 @@
class="flex w-full place-items-center justify-between gap-4 p-5 transition-colors hover:bg-gray-50 dark:hover:bg-gray-700" class="flex w-full place-items-center justify-between gap-4 p-5 transition-colors hover:bg-gray-50 dark:hover:bg-gray-700"
> >
<div class="flex place-items-center gap-4"> <div class="flex place-items-center gap-4">
<UserAvatar {user} size="md" autoColor /> <UserAvatar {user} size="md" />
<p class="text-sm font-medium">{user.name}</p> <p class="text-sm font-medium">{user.name}</p>
</div> </div>

View file

@ -71,7 +71,7 @@
on:click={() => handleUnselect(user)} on:click={() => handleUnselect(user)}
class="flex place-items-center gap-1 rounded-full border border-gray-400 p-1 transition-colors hover:bg-gray-200 dark:hover:bg-gray-700" class="flex place-items-center gap-1 rounded-full border border-gray-400 p-1 transition-colors hover:bg-gray-200 dark:hover:bg-gray-700"
> >
<UserAvatar {user} size="sm" autoColor /> <UserAvatar {user} size="sm" />
<p class="text-xs font-medium">{user.name}</p> <p class="text-xs font-medium">{user.name}</p>
</button> </button>
{/key} {/key}
@ -94,7 +94,7 @@
>✓</span >✓</span
> >
{:else} {:else}
<UserAvatar {user} size="md" autoColor /> <UserAvatar {user} size="md" />
{/if} {/if}
<div class="text-left"> <div class="text-left">

View file

@ -333,7 +333,7 @@
<p class="text-sm">SHARED BY</p> <p class="text-sm">SHARED BY</p>
<div class="flex gap-4 pt-4"> <div class="flex gap-4 pt-4">
<div> <div>
<UserAvatar user={asset.owner} size="md" autoColor /> <UserAvatar user={asset.owner} size="md" />
</div> </div>
<div class="mb-auto mt-auto"> <div class="mb-auto mt-auto">

View file

@ -1,16 +1,48 @@
<script lang="ts"> <script lang="ts">
import Button from '$lib/components/elements/buttons/button.svelte'; import Button from '$lib/components/elements/buttons/button.svelte';
import { AppRoute } from '$lib/constants'; import { AppRoute } from '$lib/constants';
import type { UserResponseDto } from '@api'; import { api, UserAvatarColor, type UserResponseDto } from '@api';
import { createEventDispatcher } from 'svelte'; import { createEventDispatcher } from 'svelte';
import Icon from '$lib/components/elements/icon.svelte'; import Icon from '$lib/components/elements/icon.svelte';
import { fade } from 'svelte/transition'; import { fade } from 'svelte/transition';
import UserAvatar from '../user-avatar.svelte'; import UserAvatar from '../user-avatar.svelte';
import { mdiCog, mdiLogout } from '@mdi/js'; import { mdiCog, mdiLogout, mdiPencil } from '@mdi/js';
import { notificationController, NotificationType } from '../notification/notification';
import { handleError } from '$lib/utils/handle-error';
import AvatarSelector from './avatar-selector.svelte';
export let user: UserResponseDto; export let user: UserResponseDto;
let isShowSelectAvatar = false;
const dispatch = createEventDispatcher(); const dispatch = createEventDispatcher();
const handleSaveProfile = async (color: UserAvatarColor) => {
try {
if (user.profileImagePath !== '') {
await api.userApi.deleteProfileImage();
}
const { data } = await api.userApi.updateUser({
updateUserDto: {
id: user.id,
email: user.email,
name: user.name,
avatarColor: color,
},
});
user = data;
isShowSelectAvatar = false;
notificationController.show({
message: 'Saved profile',
type: NotificationType.Info,
});
} catch (error) {
handleError(error, 'Unable to save profile');
}
};
</script> </script>
<div <div
@ -22,8 +54,22 @@
<div <div
class="mx-4 mt-4 flex flex-col items-center justify-center gap-4 rounded-3xl bg-white p-4 dark:bg-immich-dark-primary/10" class="mx-4 mt-4 flex flex-col items-center justify-center gap-4 rounded-3xl bg-white p-4 dark:bg-immich-dark-primary/10"
> >
<UserAvatar size="xl" {user} /> <div class="relative">
{#key user}
<UserAvatar {user} size="xl" />
<div
class="absolute z-10 bottom-0 right-0 rounded-full w-6 h-6 border dark:border-immich-dark-primary bg-immich-primary"
>
<button
class="flex items-center justify-center w-full h-full text-white"
on:click={() => (isShowSelectAvatar = true)}
>
<Icon path={mdiPencil} />
</button>
</div>
{/key}
</div>
<div> <div>
<p class="text-center text-lg font-medium text-immich-primary dark:text-immich-dark-primary"> <p class="text-center text-lg font-medium text-immich-primary dark:text-immich-dark-primary">
{user.name} {user.name}
@ -51,3 +97,10 @@
> >
</div> </div>
</div> </div>
{#if isShowSelectAvatar}
<AvatarSelector
{user}
on:close={() => (isShowSelectAvatar = false)}
on:choose={({ detail: color }) => handleSaveProfile(color)}
/>
{/if}

View file

@ -0,0 +1,39 @@
<script lang="ts">
import { mdiClose } from '@mdi/js';
import { createEventDispatcher } from 'svelte';
import { UserAvatarColor, UserResponseDto } from '@api';
import CircleIconButton from '$lib/components/elements/buttons/circle-icon-button.svelte';
import FullScreenModal from '../full-screen-modal.svelte';
import UserAvatar from '../user-avatar.svelte';
export let user: UserResponseDto;
const dispatch = createEventDispatcher();
const colors: UserAvatarColor[] = Object.values(UserAvatarColor);
</script>
<FullScreenModal on:clickOutside={() => dispatch('close')} on:escape={() => dispatch('close')}>
<div class="flex h-full w-full place-content-center place-items-center overflow-hidden">
<div
class=" rounded-3xl border bg-immich-bg shadow-sm dark:border-immich-dark-gray dark:bg-immich-dark-gray dark:text-immich-dark-fg p-4"
>
<div class="flex items-center">
<h1 class="px-4 w-full self-center font-medium text-immich-primary dark:text-immich-dark-primary text-sm">
SELECT AVATAR COLOR
</h1>
<div>
<CircleIconButton icon={mdiClose} on:click={() => dispatch('close')} />
</div>
</div>
<div class="flex items-center justify-center p-4 mt-4">
<div class="grid grid-cols-2 md:grid-cols-5 gap-4">
{#each colors as color}
<button on:click={() => dispatch('choose', color)}>
<UserAvatar {user} {color} size="xl" showProfileImage={false} />
</button>
{/each}
</div>
</div>
</div>
</div>
</FullScreenModal>

View file

@ -124,7 +124,9 @@
on:mouseleave={() => (shouldShowAccountInfo = false)} on:mouseleave={() => (shouldShowAccountInfo = false)}
on:click={() => (shouldShowAccountInfoPanel = !shouldShowAccountInfoPanel)} on:click={() => (shouldShowAccountInfoPanel = !shouldShowAccountInfoPanel)}
> >
<UserAvatar {user} size="lg" showTitle={false} interactive /> {#key user}
<UserAvatar {user} size="lg" showTitle={false} interactive />
{/key}
</button> </button>
{#if shouldShowAccountInfo && !shouldShowAccountInfoPanel} {#if shouldShowAccountInfo && !shouldShowAccountInfoPanel}
@ -139,7 +141,7 @@
{/if} {/if}
{#if shouldShowAccountInfoPanel} {#if shouldShowAccountInfoPanel}
<AccountInfoPanel {user} on:logout={logOut} /> <AccountInfoPanel bind:user on:logout={logOut} />
{/if} {/if}
</div> </div>
</section> </section>

View file

@ -1,35 +1,40 @@
<script lang="ts" context="module"> <script lang="ts" context="module">
export type Color = 'primary' | 'pink' | 'red' | 'yellow' | 'blue' | 'green'; export type Size = 'full' | 'sm' | 'md' | 'lg' | 'xl' | 'xxl' | 'xxxl';
export type Size = 'full' | 'sm' | 'md' | 'lg' | 'xl';
</script> </script>
<script lang="ts"> <script lang="ts">
import { imageLoad } from '$lib/utils/image-load'; import { imageLoad } from '$lib/utils/image-load';
import { api } from '@api'; import { UserAvatarColor, api } from '@api';
interface User { interface User {
id: string; id: string;
name: string; name: string;
email: string; email: string;
profileImagePath: string; profileImagePath: string;
avatarColor: UserAvatarColor;
} }
export let user: User; export let user: User;
export let color: Color = 'primary'; export let color: UserAvatarColor = user.avatarColor;
export let size: Size = 'full'; export let size: Size = 'full';
export let rounded = true; export let rounded = true;
export let interactive = false; export let interactive = false;
export let showTitle = true; export let showTitle = true;
export let autoColor = false; export let showProfileImage = true;
let showFallback = true; let showFallback = true;
const colorClasses: Record<Color, string> = { const colorClasses: Record<UserAvatarColor, string> = {
primary: 'bg-immich-primary dark:bg-immich-dark-primary text-immich-dark-fg dark:text-immich-fg', primary: 'bg-immich-primary dark:bg-immich-dark-primary text-immich-dark-fg dark:text-immich-fg',
pink: 'bg-pink-400 text-immich-bg', pink: 'bg-pink-400 text-immich-bg',
red: 'bg-red-500 text-immich-bg', red: 'bg-red-500 text-immich-bg',
yellow: 'bg-yellow-500 text-immich-bg', yellow: 'bg-yellow-500 text-immich-bg',
blue: 'bg-blue-500 text-immich-bg', blue: 'bg-blue-500 text-immich-bg',
green: 'bg-green-600 text-immich-bg', green: 'bg-green-600 text-immich-bg',
purple: 'bg-purple-600 text-immich-bg',
orange: 'bg-orange-600 text-immich-bg',
gray: 'bg-gray-600 text-immich-bg',
amber: 'bg-amber-600 text-immich-bg',
}; };
const sizeClasses: Record<Size, string> = { const sizeClasses: Record<Size, string> = {
@ -37,18 +42,12 @@
sm: 'w-7 h-7', sm: 'w-7 h-7',
md: 'w-10 h-10', md: 'w-10 h-10',
lg: 'w-12 h-12', lg: 'w-12 h-12',
xl: 'w-20 h-20', xl: 'w-16 h-16',
xxl: 'w-24 h-24',
xxxl: 'w-28 h-28',
}; };
// Get color based on the user UUID. $: colorClass = colorClasses[color];
function getUserColor() {
const seed = parseInt(user.id.split('-')[0], 16);
const colors = Object.keys(colorClasses).filter((color) => color !== 'primary') as Color[];
const randomIndex = seed % colors.length;
return colors[randomIndex];
}
$: colorClass = colorClasses[autoColor ? getUserColor() : color];
$: sizeClass = sizeClasses[size]; $: sizeClass = sizeClasses[size];
$: title = `${user.name} (${user.email})`; $: title = `${user.name} (${user.email})`;
$: interactiveClass = interactive $: interactiveClass = interactive
@ -61,7 +60,7 @@
class:rounded-full={rounded} class:rounded-full={rounded}
title={showTitle ? title : undefined} title={showTitle ? title : undefined}
> >
{#if user.profileImagePath} {#if showProfileImage && user.profileImagePath}
<img <img
src={api.getProfileImageUrl(user.id)} src={api.getProfileImageUrl(user.id)}
alt="Profile image of {title}" alt="Profile image of {title}"
@ -74,12 +73,12 @@
{/if} {/if}
{#if showFallback} {#if showFallback}
<span <span
class="flex h-full w-full select-none items-center justify-center" class="flex h-full w-full select-none items-center justify-center font-medium"
class:text-xs={size === 'sm'} class:text-xs={size === 'sm'}
class:text-lg={size === 'lg'} class:text-lg={size === 'lg'}
class:text-xl={size === 'xl'} class:text-xl={size === 'xl'}
class:font-medium={!autoColor} class:text-2xl={size === 'xxl'}
class:font-semibold={autoColor} class:text-3xl={size === 'xxxl'}
> >
{(user.name[0] || '').toUpperCase()} {(user.name[0] || '').toUpperCase()}
</span> </span>

View file

@ -56,7 +56,7 @@
>✓</span >✓</span
> >
{:else} {:else}
<UserAvatar {user} size="lg" autoColor /> <UserAvatar {user} size="lg" />
{/if} {/if}
<div class="text-left"> <div class="text-left">

View file

@ -113,7 +113,7 @@
<div class="rounded-2xl border border-gray-200 dark:border-gray-800 mt-6 bg-slate-50 dark:bg-gray-900 p-5"> <div class="rounded-2xl border border-gray-200 dark:border-gray-800 mt-6 bg-slate-50 dark:bg-gray-900 p-5">
<div class="flex gap-4 rounded-lg pb-4 transition-all justify-between"> <div class="flex gap-4 rounded-lg pb-4 transition-all justify-between">
<div class="flex gap-4"> <div class="flex gap-4">
<UserAvatar user={partner.user} size="md" autoColor /> <UserAvatar user={partner.user} size="md" />
<div class="text-left"> <div class="text-left">
<p class="text-immich-fg dark:text-immich-dark-fg"> <p class="text-immich-fg dark:text-immich-dark-fg">
{partner.user.name} {partner.user.name}

View file

@ -603,13 +603,13 @@
<!-- owner --> <!-- owner -->
<button on:click={() => (viewMode = ViewMode.VIEW_USERS)}> <button on:click={() => (viewMode = ViewMode.VIEW_USERS)}>
<UserAvatar user={album.owner} size="md" autoColor /> <UserAvatar user={album.owner} size="md" />
</button> </button>
<!-- users --> <!-- users -->
{#each album.sharedUsers as user (user.id)} {#each album.sharedUsers as user (user.id)}
<button on:click={() => (viewMode = ViewMode.VIEW_USERS)}> <button on:click={() => (viewMode = ViewMode.VIEW_USERS)}>
<UserAvatar {user} size="md" autoColor /> <UserAvatar {user} size="md" />
</button> </button>
{/each} {/each}

View file

@ -69,7 +69,7 @@
href="/partners/{partner.id}" href="/partners/{partner.id}"
class="flex gap-4 rounded-lg px-5 py-4 transition-all hover:bg-gray-200 dark:hover:bg-gray-700" class="flex gap-4 rounded-lg px-5 py-4 transition-all hover:bg-gray-200 dark:hover:bg-gray-700"
> >
<UserAvatar user={partner} size="lg" autoColor /> <UserAvatar user={partner} size="lg" />
<div class="text-left"> <div class="text-left">
<p class="text-immich-fg dark:text-immich-dark-fg"> <p class="text-immich-fg dark:text-immich-dark-fg">
{partner.name} {partner.name}

View file

@ -1,4 +1,4 @@
import type { UserResponseDto } from '@api'; import { UserAvatarColor, type UserResponseDto } from '@api';
import { faker } from '@faker-js/faker'; import { faker } from '@faker-js/faker';
import { Sync } from 'factory.ts'; import { Sync } from 'factory.ts';
@ -16,4 +16,5 @@ export const userFactory = Sync.makeFactory<UserResponseDto>({
updatedAt: Sync.each(() => faker.date.past().toISOString()), updatedAt: Sync.each(() => faker.date.past().toISOString()),
memoriesEnabled: true, memoriesEnabled: true,
oauthId: '', oauthId: '',
avatarColor: UserAvatarColor.Primary,
}); });