1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-19 18:26:46 +01:00

feat(web): show assets without thumbs (#2561)

* feat(web): show assets without thumbnails

* chore: open api
This commit is contained in:
Jason Rasmussen 2023-05-24 22:13:02 -04:00 committed by GitHub
parent d827a6182b
commit 1613ae9185
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
10 changed files with 58 additions and 16 deletions

Binary file not shown.

View file

@ -104,19 +104,23 @@ export class AssetRepository implements IAssetRepository {
return this.getAssetCount(items); return this.getAssetCount(items);
} }
async getAssetByTimeBucket(userId: string, getAssetByTimeBucketDto: GetAssetByTimeBucketDto): Promise<AssetEntity[]> { async getAssetByTimeBucket(userId: string, dto: GetAssetByTimeBucketDto): Promise<AssetEntity[]> {
// Get asset entity from a list of time buckets // Get asset entity from a list of time buckets
return await this.assetRepository let builder = this.assetRepository
.createQueryBuilder('asset') .createQueryBuilder('asset')
.where('asset.ownerId = :userId', { userId: userId }) .where('asset.ownerId = :userId', { userId: userId })
.andWhere(`date_trunc('month', "fileCreatedAt") IN (:...buckets)`, { .andWhere(`date_trunc('month', "fileCreatedAt") IN (:...buckets)`, {
buckets: [...getAssetByTimeBucketDto.timeBucket], buckets: [...dto.timeBucket],
}) })
.andWhere('asset.resizePath is not NULL')
.andWhere('asset.isVisible = true') .andWhere('asset.isVisible = true')
.andWhere('asset.isArchived = false') .andWhere('asset.isArchived = false')
.orderBy('asset.fileCreatedAt', 'DESC') .orderBy('asset.fileCreatedAt', 'DESC');
.getMany();
if (!dto.withoutThumbs) {
builder = builder.andWhere('asset.resizePath is not NULL');
}
return builder.getMany();
} }
async getAssetCountByTimeBucket(userId: string, timeBucket: TimeGroupEnum) { async getAssetCountByTimeBucket(userId: string, timeBucket: TimeGroupEnum) {

View file

@ -1,5 +1,7 @@
import { ApiProperty } from '@nestjs/swagger'; import { ApiProperty } from '@nestjs/swagger';
import { IsNotEmpty, IsOptional, IsUUID } from 'class-validator'; import { Transform } from 'class-transformer';
import { IsBoolean, IsNotEmpty, IsOptional, IsUUID } from 'class-validator';
import { toBoolean } from '../../../utils/transform.util';
export class GetAssetByTimeBucketDto { export class GetAssetByTimeBucketDto {
@IsNotEmpty() @IsNotEmpty()
@ -15,4 +17,12 @@ export class GetAssetByTimeBucketDto {
@IsUUID('4') @IsUUID('4')
@ApiProperty({ format: 'uuid' }) @ApiProperty({ format: 'uuid' })
userId?: string; userId?: string;
/**
* Include assets without thumbnails
*/
@IsOptional()
@IsBoolean()
@Transform(toBoolean)
withoutThumbs?: boolean;
} }

View file

@ -5988,6 +5988,10 @@
"userId": { "userId": {
"type": "string", "type": "string",
"format": "uuid" "format": "uuid"
},
"withoutThumbs": {
"type": "boolean",
"description": "Include assets without thumbnails"
} }
}, },
"required": [ "required": [

View file

@ -1330,6 +1330,12 @@ export interface GetAssetByTimeBucketDto {
* @memberof GetAssetByTimeBucketDto * @memberof GetAssetByTimeBucketDto
*/ */
'userId'?: string; 'userId'?: string;
/**
* Include assets without thumbnails
* @type {boolean}
* @memberof GetAssetByTimeBucketDto
*/
'withoutThumbs'?: boolean;
} }
/** /**
* *

View file

@ -11,6 +11,7 @@
import { createEventDispatcher, onDestroy, onMount } from 'svelte'; import { createEventDispatcher, onDestroy, onMount } from 'svelte';
import ChevronLeft from 'svelte-material-icons/ChevronLeft.svelte'; import ChevronLeft from 'svelte-material-icons/ChevronLeft.svelte';
import ChevronRight from 'svelte-material-icons/ChevronRight.svelte'; import ChevronRight from 'svelte-material-icons/ChevronRight.svelte';
import ImageBrokenVariant from 'svelte-material-icons/ImageBrokenVariant.svelte';
import { fly } from 'svelte/transition'; import { fly } from 'svelte/transition';
import AlbumSelectionModal from '../shared-components/album-selection-modal.svelte'; import AlbumSelectionModal from '../shared-components/album-selection-modal.svelte';
import { import {
@ -350,7 +351,15 @@
<div class="row-start-1 row-span-full col-start-1 col-span-4"> <div class="row-start-1 row-span-full col-start-1 col-span-4">
{#key asset.id} {#key asset.id}
{#if asset.type === AssetTypeEnum.Image} {#if !asset.resizePath}
<div class="h-full w-full flex justify-center">
<div
class="h-full bg-gray-100 dark:bg-immich-dark-gray flex items-center justify-center aspect-square px-auto"
>
<ImageBrokenVariant size="25%" />
</div>
</div>
{:else if asset.type === AssetTypeEnum.Image}
{#if shouldPlayMotionPhoto && asset.livePhotoVideoId} {#if shouldPlayMotionPhoto && asset.livePhotoVideoId}
<VideoViewer <VideoViewer
{publicSharedKey} {publicSharedKey}

View file

@ -10,6 +10,7 @@
import ArchiveArrowDownOutline from 'svelte-material-icons/ArchiveArrowDownOutline.svelte'; import ArchiveArrowDownOutline from 'svelte-material-icons/ArchiveArrowDownOutline.svelte';
import ImageThumbnail from './image-thumbnail.svelte'; import ImageThumbnail from './image-thumbnail.svelte';
import VideoThumbnail from './video-thumbnail.svelte'; import VideoThumbnail from './video-thumbnail.svelte';
import ImageBrokenVariant from 'svelte-material-icons/ImageBrokenVariant.svelte';
const dispatch = createEventDispatcher(); const dispatch = createEventDispatcher();
@ -101,7 +102,7 @@
</div> </div>
<div <div
class="bg-gray-100 dark:bg-immich-dark-gray absolute select-none transition-transform" class="h-full w-full bg-gray-100 dark:bg-immich-dark-gray absolute select-none transition-transform"
class:scale-[0.85]={selected} class:scale-[0.85]={selected}
> >
<!-- Gradient overlay on hover --> <!-- Gradient overlay on hover -->
@ -121,12 +122,19 @@
<ArchiveArrowDownOutline size="24" class="text-white" /> <ArchiveArrowDownOutline size="24" class="text-white" />
</div> </div>
{/if} {/if}
{#if asset.resizePath}
<ImageThumbnail <ImageThumbnail
url={api.getAssetThumbnailUrl(asset.id, format, publicSharedKey)} url={api.getAssetThumbnailUrl(asset.id, format, publicSharedKey)}
altText={asset.originalFileName} altText={asset.originalFileName}
widthStyle="{width}px" widthStyle="{width}px"
heightStyle="{height}px" heightStyle="{height}px"
/> />
{:else}
<div class="w-full h-full p-4 flex items-center justify-center">
<ImageBrokenVariant size="48" />
</div>
{/if}
{#if asset.type === AssetTypeEnum.Video} {#if asset.type === AssetTypeEnum.Video}
<div class="absolute w-full h-full top-0"> <div class="absolute w-full h-full top-0">

View file

@ -67,7 +67,8 @@ function createAssetStore() {
const { data: assets } = await api.assetApi.getAssetByTimeBucket( const { data: assets } = await api.assetApi.getAssetByTimeBucket(
{ {
timeBucket: [bucket], timeBucket: [bucket],
userId: _assetGridState.userId userId: _assetGridState.userId,
withoutThumbs: true
}, },
{ signal: currentBucketData?.cancelToken.signal } { signal: currentBucketData?.cancelToken.signal }
); );