feat: browse images in fullscreen (#352)

This commit is contained in:
Vojtech Simetka 2023-04-02 23:57:37 +02:00 committed by GitHub
parent 01de584171
commit f3abf95f86
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 106 additions and 10 deletions

View File

@ -0,0 +1,13 @@
<script lang="ts">
import type { IconProps } from '$lib/types'
type $$Props = IconProps
let cls: string | undefined = undefined
export { cls as class }
export let size = 20
</script>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 32 32" width={size} height={size} class={cls}>
<polygon points="14 26 15.41 24.59 7.83 17 28 17 28 15 7.83 15 15.41 7.41 14 6 4 16 14 26" />
</svg>

View File

@ -3,23 +3,80 @@
import { formatDateAndTime } from '$lib/utils/format' import { formatDateAndTime } from '$lib/utils/format'
import type { DraftPost, Post } from '$lib/stores/post' import type { DraftPost, Post } from '$lib/stores/post'
import adapter from '$lib/adapters' import adapter from '$lib/adapters'
import Button from './button.svelte'
import Close from './icons/close.svelte'
import ArrowRight from './icons/arrow-right.svelte'
import ArrowLeft from './icons/arrow-left.svelte'
let cls: string | undefined = undefined let cls: string | undefined = undefined
export let noHover: boolean | undefined = undefined export let noHover: boolean | undefined = undefined
export { cls as class } export { cls as class }
export let post: Post | DraftPost export let post: Post | DraftPost
let openedImageIndex: number | undefined = undefined
const MAX_IMAGES = 3
function nextImage() {
if (openedImageIndex === undefined) return
openedImageIndex += openedImageIndex === post.images.length - 1 ? 0 : 1
}
function previousImage() {
if (openedImageIndex === undefined) return
openedImageIndex -= openedImageIndex === 0 ? 0 : 1
}
function closeImage() {
openedImageIndex = undefined
}
</script> </script>
<svelte:window
on:keydown={({ key }) => {
if (!openedImageIndex) return
switch (key) {
case 'ArrowLeft':
previousImage()
break
case 'ArrowRight':
nextImage()
break
case 'Escape':
closeImage()
break
}
}}
/>
{#if openedImageIndex !== undefined}
<!-- Show image in full screen modal -->
<div class="fullscreen-image">
<img src={adapter.getPicture(post.images[openedImageIndex])} alt="post" />
</div>
<div class="button-close">
<Button icon={Close} variant="overlay" on:click={closeImage} />
</div>
<div class="button-next">
<Button icon={ArrowRight} variant="overlay" on:click={nextImage} />
</div>
<div class="button-previous">
<Button icon={ArrowLeft} variant="overlay" on:click={previousImage} />
</div>
{/if}
<Card on:click class={` ${cls}`} {noHover}> <Card on:click class={` ${cls}`} {noHover}>
<div class={`content-wrapper`}> <div class={`content-wrapper`}>
<div class="parent parent--adjusted-width"> <div class="parent parent--adjusted-width">
{#each post.images as image, index} {#each post.images as image, index}
{#if index <= 2} {#if index < MAX_IMAGES}
<div class="child"> <!-- svelte-ignore a11y-click-events-have-key-events -->
<div class="child" on:click={() => (openedImageIndex = index)}>
<img src={adapter.getPicture(image)} alt="post" /> <img src={adapter.getPicture(image)} alt="post" />
{#if index === 2 && post.images.length > 3} {#if index === MAX_IMAGES - 1 && post.images.length > MAX_IMAGES}
<div class="more">+{post.images.length - 3}</div> <div class="more">+{post.images.length - MAX_IMAGES}</div>
{/if} {/if}
</div> </div>
{/if} {/if}
@ -36,6 +93,38 @@
</Card> </Card>
<style lang="scss"> <style lang="scss">
.fullscreen-image {
position: fixed;
top: 0;
left: 0;
bottom: 0;
right: 0;
z-index: 1000;
background-color: black;
display: flex;
justify-content: center;
align-items: center;
}
.button-close {
position: fixed;
top: 24px;
right: 24px;
z-index: 1005;
}
.button-next {
position: fixed;
top: 50%;
transform: translateY(-50%);
right: 24px;
z-index: 1005;
}
.button-previous {
position: fixed;
top: 50%;
transform: translateY(-50%);
left: 24px;
z-index: 1005;
}
.btns { .btns {
display: flex; display: flex;
flex-direction: row; flex-direction: row;
@ -143,12 +232,6 @@
width: 100%; width: 100%;
height: 100%; height: 100%;
} }
.icon {
position: absolute;
right: var(--spacing-12);
top: var(--spacing-12);
}
} }
.user-info { .user-info {