mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-26 00:02:34 -04:00 
			
		
		
		
	feat(web): Option to assign people to unassigned faces (#9773)
* added unassigned faces to people edit * svelte fix * fix format * Captialized unassigned person name, removed person id from alttext, fixed problem with multiple faces per person * Added faces to the getAssetInfo API endpoint * Updated openApi clients * Readded the photoeditor dependency * fixed lint/format * fixed photoViewer type * changes getAssetInfo.faces to only include unassigned faces * fix: bad merge * title * logic --------- Co-authored-by: Jan108 <dasJan108@gmail.com> Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
This commit is contained in:
		
							parent
							
								
									588860455f
								
							
						
					
					
						commit
						b2761b12d1
					
				
							
								
								
									
										9
									
								
								mobile/openapi/lib/model/asset_response_dto.dart
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										9
									
								
								mobile/openapi/lib/model/asset_response_dto.dart
									
									
									
										generated
									
									
									
								
							| @ -43,6 +43,7 @@ class AssetResponseDto { | |||||||
|     this.tags = const [], |     this.tags = const [], | ||||||
|     required this.thumbhash, |     required this.thumbhash, | ||||||
|     required this.type, |     required this.type, | ||||||
|  |     this.unassignedFaces = const [], | ||||||
|     required this.updatedAt, |     required this.updatedAt, | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
| @ -126,6 +127,8 @@ class AssetResponseDto { | |||||||
| 
 | 
 | ||||||
|   AssetTypeEnum type; |   AssetTypeEnum type; | ||||||
| 
 | 
 | ||||||
|  |   List<AssetFaceWithoutPersonResponseDto> unassignedFaces; | ||||||
|  | 
 | ||||||
|   DateTime updatedAt; |   DateTime updatedAt; | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
| @ -160,6 +163,7 @@ class AssetResponseDto { | |||||||
|     _deepEquality.equals(other.tags, tags) && |     _deepEquality.equals(other.tags, tags) && | ||||||
|     other.thumbhash == thumbhash && |     other.thumbhash == thumbhash && | ||||||
|     other.type == type && |     other.type == type && | ||||||
|  |     _deepEquality.equals(other.unassignedFaces, unassignedFaces) && | ||||||
|     other.updatedAt == updatedAt; |     other.updatedAt == updatedAt; | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
| @ -195,10 +199,11 @@ class AssetResponseDto { | |||||||
|     (tags.hashCode) + |     (tags.hashCode) + | ||||||
|     (thumbhash == null ? 0 : thumbhash!.hashCode) + |     (thumbhash == null ? 0 : thumbhash!.hashCode) + | ||||||
|     (type.hashCode) + |     (type.hashCode) + | ||||||
|  |     (unassignedFaces.hashCode) + | ||||||
|     (updatedAt.hashCode); |     (updatedAt.hashCode); | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
|   String toString() => 'AssetResponseDto[checksum=$checksum, deviceAssetId=$deviceAssetId, deviceId=$deviceId, duplicateId=$duplicateId, duration=$duration, exifInfo=$exifInfo, fileCreatedAt=$fileCreatedAt, fileModifiedAt=$fileModifiedAt, hasMetadata=$hasMetadata, id=$id, isArchived=$isArchived, isFavorite=$isFavorite, isOffline=$isOffline, isTrashed=$isTrashed, libraryId=$libraryId, livePhotoVideoId=$livePhotoVideoId, localDateTime=$localDateTime, originalFileName=$originalFileName, originalPath=$originalPath, owner=$owner, ownerId=$ownerId, people=$people, resized=$resized, smartInfo=$smartInfo, stack=$stack, stackCount=$stackCount, stackParentId=$stackParentId, tags=$tags, thumbhash=$thumbhash, type=$type, updatedAt=$updatedAt]'; |   String toString() => 'AssetResponseDto[checksum=$checksum, deviceAssetId=$deviceAssetId, deviceId=$deviceId, duplicateId=$duplicateId, duration=$duration, exifInfo=$exifInfo, fileCreatedAt=$fileCreatedAt, fileModifiedAt=$fileModifiedAt, hasMetadata=$hasMetadata, id=$id, isArchived=$isArchived, isFavorite=$isFavorite, isOffline=$isOffline, isTrashed=$isTrashed, libraryId=$libraryId, livePhotoVideoId=$livePhotoVideoId, localDateTime=$localDateTime, originalFileName=$originalFileName, originalPath=$originalPath, owner=$owner, ownerId=$ownerId, people=$people, resized=$resized, smartInfo=$smartInfo, stack=$stack, stackCount=$stackCount, stackParentId=$stackParentId, tags=$tags, thumbhash=$thumbhash, type=$type, unassignedFaces=$unassignedFaces, updatedAt=$updatedAt]'; | ||||||
| 
 | 
 | ||||||
|   Map<String, dynamic> toJson() { |   Map<String, dynamic> toJson() { | ||||||
|     final json = <String, dynamic>{}; |     final json = <String, dynamic>{}; | ||||||
| @ -268,6 +273,7 @@ class AssetResponseDto { | |||||||
|     //  json[r'thumbhash'] = null; |     //  json[r'thumbhash'] = null; | ||||||
|     } |     } | ||||||
|       json[r'type'] = this.type; |       json[r'type'] = this.type; | ||||||
|  |       json[r'unassignedFaces'] = this.unassignedFaces; | ||||||
|       json[r'updatedAt'] = this.updatedAt.toUtc().toIso8601String(); |       json[r'updatedAt'] = this.updatedAt.toUtc().toIso8601String(); | ||||||
|     return json; |     return json; | ||||||
|   } |   } | ||||||
| @ -310,6 +316,7 @@ class AssetResponseDto { | |||||||
|         tags: TagResponseDto.listFromJson(json[r'tags']), |         tags: TagResponseDto.listFromJson(json[r'tags']), | ||||||
|         thumbhash: mapValueOfType<String>(json, r'thumbhash'), |         thumbhash: mapValueOfType<String>(json, r'thumbhash'), | ||||||
|         type: AssetTypeEnum.fromJson(json[r'type'])!, |         type: AssetTypeEnum.fromJson(json[r'type'])!, | ||||||
|  |         unassignedFaces: AssetFaceWithoutPersonResponseDto.listFromJson(json[r'unassignedFaces']), | ||||||
|         updatedAt: mapDateTime(json, r'updatedAt', r'')!, |         updatedAt: mapDateTime(json, r'updatedAt', r'')!, | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -7725,6 +7725,12 @@ | |||||||
|           "type": { |           "type": { | ||||||
|             "$ref": "#/components/schemas/AssetTypeEnum" |             "$ref": "#/components/schemas/AssetTypeEnum" | ||||||
|           }, |           }, | ||||||
|  |           "unassignedFaces": { | ||||||
|  |             "items": { | ||||||
|  |               "$ref": "#/components/schemas/AssetFaceWithoutPersonResponseDto" | ||||||
|  |             }, | ||||||
|  |             "type": "array" | ||||||
|  |           }, | ||||||
|           "updatedAt": { |           "updatedAt": { | ||||||
|             "format": "date-time", |             "format": "date-time", | ||||||
|             "type": "string" |             "type": "string" | ||||||
|  | |||||||
| @ -194,6 +194,7 @@ export type AssetResponseDto = { | |||||||
|     tags?: TagResponseDto[]; |     tags?: TagResponseDto[]; | ||||||
|     thumbhash: string | null; |     thumbhash: string | null; | ||||||
|     "type": AssetTypeEnum; |     "type": AssetTypeEnum; | ||||||
|  |     unassignedFaces?: AssetFaceWithoutPersonResponseDto[]; | ||||||
|     updatedAt: string; |     updatedAt: string; | ||||||
| }; | }; | ||||||
| export type AlbumResponseDto = { | export type AlbumResponseDto = { | ||||||
|  | |||||||
| @ -2,7 +2,12 @@ import { ApiProperty } from '@nestjs/swagger'; | |||||||
| import { PropertyLifecycle } from 'src/decorators'; | import { PropertyLifecycle } from 'src/decorators'; | ||||||
| import { AuthDto } from 'src/dtos/auth.dto'; | import { AuthDto } from 'src/dtos/auth.dto'; | ||||||
| import { ExifResponseDto, mapExif } from 'src/dtos/exif.dto'; | import { ExifResponseDto, mapExif } from 'src/dtos/exif.dto'; | ||||||
| import { PersonWithFacesResponseDto, mapFacesWithoutPerson, mapPerson } from 'src/dtos/person.dto'; | import { | ||||||
|  |   AssetFaceWithoutPersonResponseDto, | ||||||
|  |   PersonWithFacesResponseDto, | ||||||
|  |   mapFacesWithoutPerson, | ||||||
|  |   mapPerson, | ||||||
|  | } from 'src/dtos/person.dto'; | ||||||
| import { TagResponseDto, mapTag } from 'src/dtos/tag.dto'; | import { TagResponseDto, mapTag } from 'src/dtos/tag.dto'; | ||||||
| import { UserResponseDto, mapUser } from 'src/dtos/user.dto'; | import { UserResponseDto, mapUser } from 'src/dtos/user.dto'; | ||||||
| import { AssetFaceEntity } from 'src/entities/asset-face.entity'; | import { AssetFaceEntity } from 'src/entities/asset-face.entity'; | ||||||
| @ -41,6 +46,7 @@ export class AssetResponseDto extends SanitizedAssetResponseDto { | |||||||
|   smartInfo?: SmartInfoResponseDto; |   smartInfo?: SmartInfoResponseDto; | ||||||
|   tags?: TagResponseDto[]; |   tags?: TagResponseDto[]; | ||||||
|   people?: PersonWithFacesResponseDto[]; |   people?: PersonWithFacesResponseDto[]; | ||||||
|  |   unassignedFaces?: AssetFaceWithoutPersonResponseDto[]; | ||||||
|   /**base64 encoded sha1 hash */ |   /**base64 encoded sha1 hash */ | ||||||
|   checksum!: string; |   checksum!: string; | ||||||
|   stackParentId?: string | null; |   stackParentId?: string | null; | ||||||
| @ -116,6 +122,7 @@ export function mapAsset(entity: AssetEntity, options: AssetMapOptions = {}): As | |||||||
|     livePhotoVideoId: entity.livePhotoVideoId, |     livePhotoVideoId: entity.livePhotoVideoId, | ||||||
|     tags: entity.tags?.map(mapTag), |     tags: entity.tags?.map(mapTag), | ||||||
|     people: peopleWithFaces(entity.faces), |     people: peopleWithFaces(entity.faces), | ||||||
|  |     unassignedFaces: entity.faces?.filter((face) => !face.person).map((a) => mapFacesWithoutPerson(a)), | ||||||
|     checksum: entity.checksum.toString('base64'), |     checksum: entity.checksum.toString('base64'), | ||||||
|     stackParentId: withStack ? entity.stack?.primaryAssetId : undefined, |     stackParentId: withStack ? entity.stack?.primaryAssetId : undefined, | ||||||
|     stack: withStack |     stack: withStack | ||||||
|  | |||||||
| @ -27,6 +27,7 @@ | |||||||
|     mdiImageOutline, |     mdiImageOutline, | ||||||
|     mdiInformationOutline, |     mdiInformationOutline, | ||||||
|     mdiPencil, |     mdiPencil, | ||||||
|  |     mdiAccountOff, | ||||||
|   } from '@mdi/js'; |   } from '@mdi/js'; | ||||||
|   import { DateTime } from 'luxon'; |   import { DateTime } from 'luxon'; | ||||||
|   import { createEventDispatcher, onMount } from 'svelte'; |   import { createEventDispatcher, onMount } from 'svelte'; | ||||||
| @ -76,6 +77,7 @@ | |||||||
|     if (newAsset.id && !isSharedLink()) { |     if (newAsset.id && !isSharedLink()) { | ||||||
|       const data = await getAssetInfo({ id: asset.id }); |       const data = await getAssetInfo({ id: asset.id }); | ||||||
|       people = data?.people || []; |       people = data?.people || []; | ||||||
|  |       unassignedFaces = data?.unassignedFaces || []; | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
| @ -93,6 +95,8 @@ | |||||||
|   $: people = asset.people || []; |   $: people = asset.people || []; | ||||||
|   $: showingHiddenPeople = false; |   $: showingHiddenPeople = false; | ||||||
| 
 | 
 | ||||||
|  |   $: unassignedFaces = asset.unassignedFaces || []; | ||||||
|  | 
 | ||||||
|   onMount(() => { |   onMount(() => { | ||||||
|     return websocketEvents.on('on_asset_update', (assetUpdate) => { |     return websocketEvents.on('on_asset_update', (assetUpdate) => { | ||||||
|       if (assetUpdate.id === asset.id) { |       if (assetUpdate.id === asset.id) { | ||||||
| @ -118,6 +122,7 @@ | |||||||
|   const handleRefreshPeople = async () => { |   const handleRefreshPeople = async () => { | ||||||
|     await getAssetInfo({ id: asset.id }).then((data) => { |     await getAssetInfo({ id: asset.id }).then((data) => { | ||||||
|       people = data?.people || []; |       people = data?.people || []; | ||||||
|  |       unassignedFaces = data?.unassignedFaces || []; | ||||||
|     }); |     }); | ||||||
|     showEditFaces = false; |     showEditFaces = false; | ||||||
|   }; |   }; | ||||||
| @ -158,11 +163,20 @@ | |||||||
| 
 | 
 | ||||||
|   <DetailPanelDescription {asset} {isOwner} /> |   <DetailPanelDescription {asset} {isOwner} /> | ||||||
| 
 | 
 | ||||||
|   {#if !isSharedLink() && people.length > 0} |   {#if (!isSharedLink() && unassignedFaces.length > 0) || people.length > 0} | ||||||
|     <section class="px-4 py-4 text-sm"> |     <section class="px-4 py-4 text-sm"> | ||||||
|       <div class="flex h-10 w-full items-center justify-between"> |       <div class="flex h-10 w-full items-center justify-between"> | ||||||
|         <h2>{$t('people').toUpperCase()}</h2> |         <h2>{$t('people').toUpperCase()}</h2> | ||||||
|         <div class="flex gap-2 items-center"> |         <div class="flex gap-2 items-center"> | ||||||
|  |           {#if unassignedFaces.length > 0} | ||||||
|  |             <Icon | ||||||
|  |               ariaLabel="Asset has unassigned faces" | ||||||
|  |               title="Asset has unassigned faces" | ||||||
|  |               color="currentColor" | ||||||
|  |               path={mdiAccountOff} | ||||||
|  |               size="24" | ||||||
|  |             /> | ||||||
|  |           {/if} | ||||||
|           {#if people.some((person) => person.isHidden)} |           {#if people.some((person) => person.isHidden)} | ||||||
|             <CircleIconButton |             <CircleIconButton | ||||||
|               title={$t('show_hidden_people')} |               title={$t('show_hidden_people')} | ||||||
|  | |||||||
| @ -1,24 +1,24 @@ | |||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
|   import { timeBeforeShowLoadingSpinner } from '$lib/constants'; |   import { timeBeforeShowLoadingSpinner } from '$lib/constants'; | ||||||
|   import { photoViewer } from '$lib/stores/assets.store'; |  | ||||||
|   import { getAssetThumbnailUrl, getPeopleThumbnailUrl } from '$lib/utils'; |  | ||||||
|   import { getPersonNameWithHiddenValue } from '$lib/utils/person'; |   import { getPersonNameWithHiddenValue } from '$lib/utils/person'; | ||||||
|  |   import { getPeopleThumbnailUrl } from '$lib/utils'; | ||||||
|   import { AssetTypeEnum, type AssetFaceResponseDto, type PersonResponseDto } from '@immich/sdk'; |   import { AssetTypeEnum, type AssetFaceResponseDto, type PersonResponseDto } from '@immich/sdk'; | ||||||
|   import { mdiArrowLeftThin, mdiClose, mdiMagnify, mdiPlus } from '@mdi/js'; |   import { mdiArrowLeftThin, mdiClose, mdiMagnify, mdiPlus } from '@mdi/js'; | ||||||
|   import { createEventDispatcher } from 'svelte'; |   import { createEventDispatcher } from 'svelte'; | ||||||
|   import { linear } from 'svelte/easing'; |   import { linear } from 'svelte/easing'; | ||||||
|   import { fly } from 'svelte/transition'; |   import { fly } from 'svelte/transition'; | ||||||
|  |   import { photoViewer } from '$lib/stores/assets.store'; | ||||||
|   import ImageThumbnail from '../assets/thumbnail/image-thumbnail.svelte'; |   import ImageThumbnail from '../assets/thumbnail/image-thumbnail.svelte'; | ||||||
|   import LoadingSpinner from '../shared-components/loading-spinner.svelte'; |   import LoadingSpinner from '../shared-components/loading-spinner.svelte'; | ||||||
|   import SearchPeople from '$lib/components/faces-page/people-search.svelte'; |   import SearchPeople from '$lib/components/faces-page/people-search.svelte'; | ||||||
|   import CircleIconButton from '$lib/components/elements/buttons/circle-icon-button.svelte'; |   import CircleIconButton from '$lib/components/elements/buttons/circle-icon-button.svelte'; | ||||||
|  |   import { zoomImageToBase64 } from '$lib/utils/people-utils'; | ||||||
|   import { t } from 'svelte-i18n'; |   import { t } from 'svelte-i18n'; | ||||||
| 
 | 
 | ||||||
|   export let peopleWithFaces: AssetFaceResponseDto[]; |  | ||||||
|   export let allPeople: PersonResponseDto[]; |   export let allPeople: PersonResponseDto[]; | ||||||
|   export let editedPerson: PersonResponseDto; |   export let editedFace: AssetFaceResponseDto; | ||||||
|   export let assetType: AssetTypeEnum; |  | ||||||
|   export let assetId: string; |   export let assetId: string; | ||||||
|  |   export let assetType: AssetTypeEnum; | ||||||
| 
 | 
 | ||||||
|   // loading spinners |   // loading spinners | ||||||
|   let isShowLoadingNewPerson = false; |   let isShowLoadingNewPerson = false; | ||||||
| @ -39,71 +39,11 @@ | |||||||
|   const handleBackButton = () => { |   const handleBackButton = () => { | ||||||
|     dispatch('close'); |     dispatch('close'); | ||||||
|   }; |   }; | ||||||
|   const zoomImageToBase64 = async (face: AssetFaceResponseDto): Promise<string | null> => { |  | ||||||
|     let image: HTMLImageElement | null = null; |  | ||||||
|     if (assetType === AssetTypeEnum.Image) { |  | ||||||
|       image = $photoViewer; |  | ||||||
|     } else if (assetType === AssetTypeEnum.Video) { |  | ||||||
|       const data = getAssetThumbnailUrl(assetId); |  | ||||||
|       const img: HTMLImageElement = new Image(); |  | ||||||
|       img.src = data; |  | ||||||
| 
 |  | ||||||
|       await new Promise<void>((resolve) => { |  | ||||||
|         img.addEventListener('load', () => resolve()); |  | ||||||
|         img.addEventListener('error', () => resolve()); |  | ||||||
|       }); |  | ||||||
| 
 |  | ||||||
|       image = img; |  | ||||||
|     } |  | ||||||
|     if (image === null) { |  | ||||||
|       return null; |  | ||||||
|     } |  | ||||||
|     const { |  | ||||||
|       boundingBoxX1: x1, |  | ||||||
|       boundingBoxX2: x2, |  | ||||||
|       boundingBoxY1: y1, |  | ||||||
|       boundingBoxY2: y2, |  | ||||||
|       imageWidth, |  | ||||||
|       imageHeight, |  | ||||||
|     } = face; |  | ||||||
| 
 |  | ||||||
|     const coordinates = { |  | ||||||
|       x1: (image.naturalWidth / imageWidth) * x1, |  | ||||||
|       x2: (image.naturalWidth / imageWidth) * x2, |  | ||||||
|       y1: (image.naturalHeight / imageHeight) * y1, |  | ||||||
|       y2: (image.naturalHeight / imageHeight) * y2, |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     const faceWidth = coordinates.x2 - coordinates.x1; |  | ||||||
|     const faceHeight = coordinates.y2 - coordinates.y1; |  | ||||||
| 
 |  | ||||||
|     const faceImage = new Image(); |  | ||||||
|     faceImage.src = image.src; |  | ||||||
| 
 |  | ||||||
|     await new Promise((resolve) => { |  | ||||||
|       faceImage.addEventListener('load', resolve); |  | ||||||
|       faceImage.addEventListener('error', () => resolve(null)); |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     const canvas = document.createElement('canvas'); |  | ||||||
|     canvas.width = faceWidth; |  | ||||||
|     canvas.height = faceHeight; |  | ||||||
| 
 |  | ||||||
|     const context = canvas.getContext('2d'); |  | ||||||
|     if (context) { |  | ||||||
|       context.drawImage(faceImage, coordinates.x1, coordinates.y1, faceWidth, faceHeight, 0, 0, faceWidth, faceHeight); |  | ||||||
| 
 |  | ||||||
|       return canvas.toDataURL(); |  | ||||||
|     } else { |  | ||||||
|       return null; |  | ||||||
|     } |  | ||||||
|   }; |  | ||||||
| 
 | 
 | ||||||
|   const handleCreatePerson = async () => { |   const handleCreatePerson = async () => { | ||||||
|     const timeout = setTimeout(() => (isShowLoadingNewPerson = true), timeBeforeShowLoadingSpinner); |     const timeout = setTimeout(() => (isShowLoadingNewPerson = true), timeBeforeShowLoadingSpinner); | ||||||
|     const personToUpdate = peopleWithFaces.find((face) => face.person?.id === editedPerson.id); |  | ||||||
| 
 | 
 | ||||||
|     const newFeaturePhoto = personToUpdate ? await zoomImageToBase64(personToUpdate) : null; |     const newFeaturePhoto = await zoomImageToBase64(editedFace, assetId, assetType, $photoViewer); | ||||||
| 
 | 
 | ||||||
|     dispatch('createPerson', newFeaturePhoto); |     dispatch('createPerson', newFeaturePhoto); | ||||||
| 
 | 
 | ||||||
| @ -161,7 +101,7 @@ | |||||||
|     <h2 class="mb-8 mt-4 uppercase">{$t('all_people')}</h2> |     <h2 class="mb-8 mt-4 uppercase">{$t('all_people')}</h2> | ||||||
|     <div class="immich-scrollbar mt-4 flex flex-wrap gap-2 overflow-y-auto"> |     <div class="immich-scrollbar mt-4 flex flex-wrap gap-2 overflow-y-auto"> | ||||||
|       {#each showPeople as person (person.id)} |       {#each showPeople as person (person.id)} | ||||||
|         {#if person.id !== editedPerson.id} |         {#if !editedFace.person || person.id !== editedFace.person.id} | ||||||
|           <div class="w-fit"> |           <div class="w-fit"> | ||||||
|             <button type="button" class="w-[90px]" on:click={() => dispatch('reassign', person)}> |             <button type="button" class="w-[90px]" on:click={() => dispatch('reassign', person)}> | ||||||
|               <div class="relative"> |               <div class="relative"> | ||||||
|  | |||||||
| @ -7,14 +7,16 @@ | |||||||
|   import { handleError } from '$lib/utils/handle-error'; |   import { handleError } from '$lib/utils/handle-error'; | ||||||
|   import { getPersonNameWithHiddenValue } from '$lib/utils/person'; |   import { getPersonNameWithHiddenValue } from '$lib/utils/person'; | ||||||
|   import { |   import { | ||||||
|     AssetTypeEnum, |  | ||||||
|     createPerson, |     createPerson, | ||||||
|     getAllPeople, |     getAllPeople, | ||||||
|     getFaces, |     getFaces, | ||||||
|     reassignFacesById, |     reassignFacesById, | ||||||
|  |     AssetTypeEnum, | ||||||
|     type AssetFaceResponseDto, |     type AssetFaceResponseDto, | ||||||
|     type PersonResponseDto, |     type PersonResponseDto, | ||||||
|   } from '@immich/sdk'; |   } from '@immich/sdk'; | ||||||
|  |   import { mdiAccountOff } from '@mdi/js'; | ||||||
|  |   import Icon from '$lib/components/elements/icon.svelte'; | ||||||
|   import { mdiArrowLeftThin, mdiMinus, mdiRestart } from '@mdi/js'; |   import { mdiArrowLeftThin, mdiMinus, mdiRestart } from '@mdi/js'; | ||||||
|   import { createEventDispatcher, onMount } from 'svelte'; |   import { createEventDispatcher, onMount } from 'svelte'; | ||||||
|   import { linear } from 'svelte/easing'; |   import { linear } from 'svelte/easing'; | ||||||
| @ -23,6 +25,8 @@ | |||||||
|   import { NotificationType, notificationController } from '../shared-components/notification/notification'; |   import { NotificationType, notificationController } from '../shared-components/notification/notification'; | ||||||
|   import AssignFaceSidePanel from './assign-face-side-panel.svelte'; |   import AssignFaceSidePanel from './assign-face-side-panel.svelte'; | ||||||
|   import CircleIconButton from '$lib/components/elements/buttons/circle-icon-button.svelte'; |   import CircleIconButton from '$lib/components/elements/buttons/circle-icon-button.svelte'; | ||||||
|  |   import { zoomImageToBase64 } from '$lib/utils/people-utils'; | ||||||
|  |   import { photoViewer } from '$lib/stores/assets.store'; | ||||||
|   import { t } from 'svelte-i18n'; |   import { t } from 'svelte-i18n'; | ||||||
| 
 | 
 | ||||||
|   export let assetId: string; |   export let assetId: string; | ||||||
| @ -36,7 +40,6 @@ | |||||||
|   let peopleWithFaces: AssetFaceResponseDto[] = []; |   let peopleWithFaces: AssetFaceResponseDto[] = []; | ||||||
|   let selectedPersonToReassign: Record<string, PersonResponseDto> = {}; |   let selectedPersonToReassign: Record<string, PersonResponseDto> = {}; | ||||||
|   let selectedPersonToCreate: Record<string, string> = {}; |   let selectedPersonToCreate: Record<string, string> = {}; | ||||||
|   let editedPerson: PersonResponseDto; |  | ||||||
|   let editedFace: AssetFaceResponseDto; |   let editedFace: AssetFaceResponseDto; | ||||||
| 
 | 
 | ||||||
|   // loading spinners |   // loading spinners | ||||||
| @ -171,11 +174,8 @@ | |||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   const handleFacePicker = (face: AssetFaceResponseDto) => { |   const handleFacePicker = (face: AssetFaceResponseDto) => { | ||||||
|     if (face.person) { |     editedFace = face; | ||||||
|       editedFace = face; |     showSelectedFaces = true; | ||||||
|       editedPerson = face.person; |  | ||||||
|       showSelectedFaces = true; |  | ||||||
|     } |  | ||||||
|   }; |   }; | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| @ -209,91 +209,125 @@ | |||||||
|         </div> |         </div> | ||||||
|       {:else} |       {:else} | ||||||
|         {#each peopleWithFaces as face, index} |         {#each peopleWithFaces as face, index} | ||||||
|           {#if face.person} |           {@const personName = face.person ? face.person?.name : 'Unassigned'} | ||||||
|             <div class="relative z-[20001] h-[115px] w-[95px]"> |           <div class="relative z-[20001] h-[115px] w-[95px]"> | ||||||
|               <div |             <div | ||||||
|                 role="button" |               role="button" | ||||||
|                 tabindex={index} |               tabindex={index} | ||||||
|                 class="absolute left-0 top-0 h-[90px] w-[90px] cursor-default" |               class="absolute left-0 top-0 h-[90px] w-[90px] cursor-default" | ||||||
|                 on:focus={() => ($boundingBoxesArray = [peopleWithFaces[index]])} |               on:focus={() => ($boundingBoxesArray = [peopleWithFaces[index]])} | ||||||
|                 on:mouseover={() => ($boundingBoxesArray = [peopleWithFaces[index]])} |               on:mouseover={() => ($boundingBoxesArray = [peopleWithFaces[index]])} | ||||||
|                 on:mouseleave={() => ($boundingBoxesArray = [])} |               on:mouseleave={() => ($boundingBoxesArray = [])} | ||||||
|               > |             > | ||||||
|                 <div class="relative"> |               <div class="relative"> | ||||||
|                   {#if selectedPersonToCreate[face.id]} |                 {#if selectedPersonToCreate[face.id]} | ||||||
|  |                   <ImageThumbnail | ||||||
|  |                     curve | ||||||
|  |                     shadow | ||||||
|  |                     url={selectedPersonToCreate[face.id]} | ||||||
|  |                     altText={'New person'} | ||||||
|  |                     title={'New person'} | ||||||
|  |                     widthStyle={thumbnailWidth} | ||||||
|  |                     heightStyle={thumbnailWidth} | ||||||
|  |                   /> | ||||||
|  |                 {:else if selectedPersonToReassign[face.id]} | ||||||
|  |                   <ImageThumbnail | ||||||
|  |                     curve | ||||||
|  |                     shadow | ||||||
|  |                     url={getPeopleThumbnailUrl(selectedPersonToReassign[face.id].id)} | ||||||
|  |                     altText={selectedPersonToReassign[face.id].name} | ||||||
|  |                     title={getPersonNameWithHiddenValue( | ||||||
|  |                       selectedPersonToReassign[face.id].name, | ||||||
|  |                       selectedPersonToReassign[face.id]?.isHidden, | ||||||
|  |                     )} | ||||||
|  |                     widthStyle={thumbnailWidth} | ||||||
|  |                     heightStyle={thumbnailWidth} | ||||||
|  |                     hidden={selectedPersonToReassign[face.id].isHidden} | ||||||
|  |                   /> | ||||||
|  |                 {:else if face.person} | ||||||
|  |                   <ImageThumbnail | ||||||
|  |                     curve | ||||||
|  |                     shadow | ||||||
|  |                     url={getPeopleThumbnailUrl(face.person.id)} | ||||||
|  |                     altText={face.person.name} | ||||||
|  |                     title={getPersonNameWithHiddenValue(face.person.name, face.person.isHidden)} | ||||||
|  |                     widthStyle={thumbnailWidth} | ||||||
|  |                     heightStyle={thumbnailWidth} | ||||||
|  |                     hidden={face.person.isHidden} | ||||||
|  |                   /> | ||||||
|  |                 {:else} | ||||||
|  |                   {#await zoomImageToBase64(face, assetId, assetType, $photoViewer)} | ||||||
|                     <ImageThumbnail |                     <ImageThumbnail | ||||||
|                       curve |                       curve | ||||||
|                       shadow |                       shadow | ||||||
|                       url={selectedPersonToCreate[face.id]} |                       url="/src/lib/assets/no-thumbnail.png" | ||||||
|                       altText={selectedPersonToCreate[face.id]} |                       altText="Unassigned" | ||||||
|                       title={$t('new_person')} |                       title="Unassigned" | ||||||
|                       widthStyle={thumbnailWidth} |                       widthStyle="90px" | ||||||
|                       heightStyle={thumbnailWidth} |                       heightStyle="90px" | ||||||
|  |                       thumbhash={null} | ||||||
|  |                       hidden={false} | ||||||
|                     /> |                     /> | ||||||
|                   {:else if selectedPersonToReassign[face.id]} |                   {:then data} | ||||||
|                     <ImageThumbnail |                     <ImageThumbnail | ||||||
|                       curve |                       curve | ||||||
|                       shadow |                       shadow | ||||||
|                       url={getPeopleThumbnailUrl(selectedPersonToReassign[face.id].id)} |                       url={data === null ? '/src/lib/assets/no-thumbnail.png' : data} | ||||||
|                       altText={selectedPersonToReassign[face.id]?.name || selectedPersonToReassign[face.id].id} |                       altText="Unassigned" | ||||||
|                       title={getPersonNameWithHiddenValue( |                       title="Unassigned" | ||||||
|                         selectedPersonToReassign[face.id].name, |                       widthStyle="90px" | ||||||
|                         face.person?.isHidden, |                       heightStyle="90px" | ||||||
|                       )} |                       thumbhash={null} | ||||||
|                       widthStyle={thumbnailWidth} |                       hidden={false} | ||||||
|                       heightStyle={thumbnailWidth} |  | ||||||
|                       hidden={selectedPersonToReassign[face.id].isHidden} |  | ||||||
|                     /> |                     /> | ||||||
|                   {:else} |                   {/await} | ||||||
|                     <ImageThumbnail |  | ||||||
|                       curve |  | ||||||
|                       shadow |  | ||||||
|                       url={getPeopleThumbnailUrl(face.person.id)} |  | ||||||
|                       altText={face.person.name || face.person.id} |  | ||||||
|                       title={getPersonNameWithHiddenValue(face.person.name, face.person.isHidden)} |  | ||||||
|                       widthStyle={thumbnailWidth} |  | ||||||
|                       heightStyle={thumbnailWidth} |  | ||||||
|                       hidden={face.person.isHidden} |  | ||||||
|                     /> |  | ||||||
|                   {/if} |  | ||||||
|                 </div> |  | ||||||
| 
 |  | ||||||
|                 {#if !selectedPersonToCreate[face.id]} |  | ||||||
|                   <p class="relative mt-1 truncate font-medium" title={face.person?.name}> |  | ||||||
|                     {#if selectedPersonToReassign[face.id]?.id} |  | ||||||
|                       {selectedPersonToReassign[face.id]?.name} |  | ||||||
|                     {:else} |  | ||||||
|                       {face.person?.name} |  | ||||||
|                     {/if} |  | ||||||
|                   </p> |  | ||||||
|                 {/if} |                 {/if} | ||||||
|  |               </div> | ||||||
| 
 | 
 | ||||||
|                 <div class="absolute -right-[5px] -top-[5px] h-[20px] w-[20px] rounded-full"> |               {#if !selectedPersonToCreate[face.id]} | ||||||
|                   {#if selectedPersonToCreate[face.id] || selectedPersonToReassign[face.id]} |                 <p class="relative mt-1 truncate font-medium" title={personName}> | ||||||
|                     <CircleIconButton |                   {#if selectedPersonToReassign[face.id]?.id} | ||||||
|                       color="primary" |                     {selectedPersonToReassign[face.id]?.name} | ||||||
|                       icon={mdiRestart} |  | ||||||
|                       title={$t('reset')} |  | ||||||
|                       size="18" |  | ||||||
|                       padding="1" |  | ||||||
|                       class="absolute left-1/2 top-1/2 translate-x-[-50%] translate-y-[-50%] transform" |  | ||||||
|                       on:click={() => handleReset(face.id)} |  | ||||||
|                     /> |  | ||||||
|                   {:else} |                   {:else} | ||||||
|                     <CircleIconButton |                     <span class={personName == 'Unassigned' ? 'dark:text-gray-500' : ''}>{personName}</span> | ||||||
|                       color="primary" |  | ||||||
|                       icon={mdiMinus} |  | ||||||
|                       title={$t('select_new_face')} |  | ||||||
|                       size="18" |  | ||||||
|                       padding="1" |  | ||||||
|                       class="absolute left-1/2 top-1/2 translate-x-[-50%] translate-y-[-50%] transform" |  | ||||||
|                       on:click={() => handleFacePicker(face)} |  | ||||||
|                     /> |  | ||||||
|                   {/if} |                   {/if} | ||||||
|                 </div> |                 </p> | ||||||
|  |               {/if} | ||||||
|  | 
 | ||||||
|  |               <div class="absolute -right-[5px] -top-[5px] h-[20px] w-[20px] rounded-full"> | ||||||
|  |                 {#if selectedPersonToCreate[face.id] || selectedPersonToReassign[face.id]} | ||||||
|  |                   <CircleIconButton | ||||||
|  |                     color="primary" | ||||||
|  |                     icon={mdiRestart} | ||||||
|  |                     title="Reset" | ||||||
|  |                     size="18" | ||||||
|  |                     padding="1" | ||||||
|  |                     class="absolute left-1/2 top-1/2 translate-x-[-50%] translate-y-[-50%] transform" | ||||||
|  |                     on:click={() => handleReset(face.id)} | ||||||
|  |                   /> | ||||||
|  |                 {:else} | ||||||
|  |                   <CircleIconButton | ||||||
|  |                     color="primary" | ||||||
|  |                     icon={mdiMinus} | ||||||
|  |                     title="Select new face" | ||||||
|  |                     size="18" | ||||||
|  |                     padding="1" | ||||||
|  |                     class="absolute left-1/2 top-1/2 translate-x-[-50%] translate-y-[-50%] transform" | ||||||
|  |                     on:click={() => handleFacePicker(face)} | ||||||
|  |                   /> | ||||||
|  |                 {/if} | ||||||
|  |               </div> | ||||||
|  |               <div class="absolute right-[25px] -top-[5px] h-[20px] w-[20px] rounded-full"> | ||||||
|  |                 {#if !selectedPersonToCreate[face.id] && !selectedPersonToReassign[face.id] && !face.person} | ||||||
|  |                   <div | ||||||
|  |                     class="flex place-content-center place-items-center rounded-full bg-[#d3d3d3] p-1 transition-all absolute left-1/2 top-1/2 translate-x-[-50%] translate-y-[-50%] transform" | ||||||
|  |                   > | ||||||
|  |                     <Icon color="primary" path={mdiAccountOff} ariaLabel="Just a face" size="18" /> | ||||||
|  |                   </div> | ||||||
|  |                 {/if} | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|           {/if} |           </div> | ||||||
|         {/each} |         {/each} | ||||||
|       {/if} |       {/if} | ||||||
|     </div> |     </div> | ||||||
| @ -302,11 +336,10 @@ | |||||||
| 
 | 
 | ||||||
| {#if showSelectedFaces} | {#if showSelectedFaces} | ||||||
|   <AssignFaceSidePanel |   <AssignFaceSidePanel | ||||||
|     {peopleWithFaces} |  | ||||||
|     {allPeople} |     {allPeople} | ||||||
|     {editedPerson} |     {editedFace} | ||||||
|     {assetType} |  | ||||||
|     {assetId} |     {assetId} | ||||||
|  |     {assetType} | ||||||
|     on:close={() => (showSelectedFaces = false)} |     on:close={() => (showSelectedFaces = false)} | ||||||
|     on:createPerson={(event) => handleCreatePerson(event.detail)} |     on:createPerson={(event) => handleCreatePerson(event.detail)} | ||||||
|     on:reassign={(event) => handleReassignFace(event.detail)} |     on:reassign={(event) => handleReassignFace(event.detail)} | ||||||
|  | |||||||
| @ -1,4 +1,6 @@ | |||||||
| import type { Faces } from '$lib/stores/people.store'; | import type { Faces } from '$lib/stores/people.store'; | ||||||
|  | import { getAssetThumbnailUrl } from '$lib/utils'; | ||||||
|  | import { AssetTypeEnum, type AssetFaceResponseDto } from '@immich/sdk'; | ||||||
| import type { ZoomImageWheelState } from '@zoom-image/core'; | import type { ZoomImageWheelState } from '@zoom-image/core'; | ||||||
| 
 | 
 | ||||||
| const getContainedSize = (img: HTMLImageElement): { width: number; height: number } => { | const getContainedSize = (img: HTMLImageElement): { width: number; height: number } => { | ||||||
| @ -69,3 +71,61 @@ export const getBoundingBox = ( | |||||||
|   } |   } | ||||||
|   return boxes; |   return boxes; | ||||||
| }; | }; | ||||||
|  | 
 | ||||||
|  | export const zoomImageToBase64 = async ( | ||||||
|  |   face: AssetFaceResponseDto, | ||||||
|  |   assetId: string, | ||||||
|  |   assetType: AssetTypeEnum, | ||||||
|  |   photoViewer: HTMLImageElement | null, | ||||||
|  | ): Promise<string | null> => { | ||||||
|  |   let image: HTMLImageElement | null = null; | ||||||
|  |   if (assetType === AssetTypeEnum.Image) { | ||||||
|  |     image = photoViewer; | ||||||
|  |   } else if (assetType === AssetTypeEnum.Video) { | ||||||
|  |     const data = getAssetThumbnailUrl(assetId); | ||||||
|  |     const img: HTMLImageElement = new Image(); | ||||||
|  |     img.src = data; | ||||||
|  | 
 | ||||||
|  |     await new Promise<void>((resolve) => { | ||||||
|  |       img.addEventListener('load', () => resolve()); | ||||||
|  |       img.addEventListener('error', () => resolve()); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     image = img; | ||||||
|  |   } | ||||||
|  |   if (image === null) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |   const { boundingBoxX1: x1, boundingBoxX2: x2, boundingBoxY1: y1, boundingBoxY2: y2, imageWidth, imageHeight } = face; | ||||||
|  | 
 | ||||||
|  |   const coordinates = { | ||||||
|  |     x1: (image.naturalWidth / imageWidth) * x1, | ||||||
|  |     x2: (image.naturalWidth / imageWidth) * x2, | ||||||
|  |     y1: (image.naturalHeight / imageHeight) * y1, | ||||||
|  |     y2: (image.naturalHeight / imageHeight) * y2, | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   const faceWidth = coordinates.x2 - coordinates.x1; | ||||||
|  |   const faceHeight = coordinates.y2 - coordinates.y1; | ||||||
|  | 
 | ||||||
|  |   const faceImage = new Image(); | ||||||
|  |   faceImage.src = image.src; | ||||||
|  | 
 | ||||||
|  |   await new Promise((resolve) => { | ||||||
|  |     faceImage.addEventListener('load', resolve); | ||||||
|  |     faceImage.addEventListener('error', () => resolve(null)); | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   const canvas = document.createElement('canvas'); | ||||||
|  |   canvas.width = faceWidth; | ||||||
|  |   canvas.height = faceHeight; | ||||||
|  | 
 | ||||||
|  |   const context = canvas.getContext('2d'); | ||||||
|  |   if (context) { | ||||||
|  |     context.drawImage(faceImage, coordinates.x1, coordinates.y1, faceWidth, faceHeight, 0, 0, faceWidth, faceHeight); | ||||||
|  | 
 | ||||||
|  |     return canvas.toDataURL(); | ||||||
|  |   } else { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user