Skip to content

Commit

Permalink
Merge pull request #29 from Joery-M/15-sl-ui-library-listgrid-scaling
Browse files Browse the repository at this point in the history
15 sl UI library listgrid scaling
  • Loading branch information
Joery-M authored May 6, 2024
2 parents cea51ee + 7c2d2e7 commit a8be3aa
Show file tree
Hide file tree
Showing 2 changed files with 149 additions and 11 deletions.
135 changes: 127 additions & 8 deletions packages/safelight/src/components/Editor/Library/Library.vue
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
<template>
<DataView
v-if="layout == 'grid'"
:value="sortedAndFiltered"
class="flex h-full flex-col"
data-key="id"
Expand All @@ -12,12 +13,17 @@
},
emptyMessage: {
style: 'height: 100%;'
},
footer: {
style: 'flex-grow: 1; display: flex; align-items: end'
}
}"
layout="grid"
>
<template #header>
<Toolbar class="border-none p-0">
<template #start>
<DataViewLayoutOptions v-model="layout" class="mr-2 min-w-fit" />
<InputGroup class="mr-2">
<InputGroupAddon class="p-0">
<PhMagnifyingGlass />
Expand Down Expand Up @@ -55,18 +61,14 @@
</template>
</Toolbar>
</template>
<template #list="{ items }: { items: Media[] }">
<template #grid="{ items }: { items: Media[] }">
<div
class="grid-nogutter grid h-full select-none overflow-y-auto"
class="flex h-full select-none flex-wrap justify-center overflow-y-auto"
role="grid"
style="
grid-template-columns: repeat(auto-fill, minmax(150px, 1fr));
grid-template-rows: min-content;
"
@dblclick.self="fileDialogOpenDblClick"
>
<template v-for="item in items" :key="item.id">
<LibraryItem :item="item" />
<LibraryGridItem :item="item" :size="gridItemSize" />
</template>
</div>
</template>
Expand All @@ -81,12 +83,118 @@
<label v-else>No media found</label>
</div>
</template>
<template #footer>
<Toolbar class="border-none p-0">
<template #start>
<Slider
v-model="gridItemSize"
orientation="horizontal"
:min="50"
:max="300"
:step="10"
class="w-48 max-w-full"
/>
</template>
</Toolbar>
</template>
</DataView>
<DataTable
v-else-if="layout == 'list'"
class="flex h-full flex-col"
:value="sortedAndFiltered"
:pt="{
header: {
class: 'p-1'
},
wrapper: {
style: 'flex-shrink: 1'
},
table: {
class: 'h-full'
}
}"
sort-field="field"
sort-mode="single"
scrollable
resizable-columns
column-resize-mode="expand"
data-key="id"
>
<Column header="Type" body-class="px-0">
<template #body="{ data }: { data: Media }">
<div class="flex gap-2 pl-2">
<PhVideoCamera
v-if="data.isOfType(MediaType.Video)"
weight="bold"
aria-label="Media has video"
/>
<PhSpeakerHigh
v-if="data.isOfType(MediaType.Audio)"
weight="bold"
aria-label="Media has audio"
/>
<PhSubtitles
v-if="data.isOfType(MediaType.Text)"
weight="bold"
aria-label="Media has subtitles"
/>
<PhImage
v-if="data.isOfType(MediaType.Image)"
weight="bold"
aria-label="Media is an image"
/>
</div>
</template>
</Column>
<Column field="name.value" header="Name" sortable />
<Column field="duration.value" header="Duration" sortable>
<template #body="{ data }: { data: Media }">
<template v-if="data.isOfType(MediaType.Image)">
{{ Timecode.toFormattedTimecode(5000) }}
</template>
<template v-else>
{{ Timecode.toFormattedTimecode(data.duration.value * 1000) }}
</template>
</template>
</Column>
<template #header>
<Toolbar class="border-none p-0">
<template #start>
<DataViewLayoutOptions v-model="layout" class="mr-2 min-w-fit" />
<InputGroup class="mr-2">
<InputGroupAddon class="p-0">
<PhMagnifyingGlass />
</InputGroupAddon>
<InputText v-model="search" placeholder="Search"> </InputText>
</InputGroup>
</template>
<template #end>
<Button title="Load file" rounded @click="fileDialog.open()">
<template #icon>
<PhPlus />
</template>
</Button>
</template>
</Toolbar>
</template>
<template #empty>
<div
class="grid h-full select-none place-items-center opacity-60"
@dblclick="fileDialogOpenDblClick"
>
<label v-if="CurrentProject.project.value?.media.length == 0">
No media imported
</label>
<label v-else>No media found</label>
</div>
</template>
</DataTable>
</template>

<script setup lang="ts">
import { ProjectFeatures } from '@safelight/shared/base/Project';
import Media from '@safelight/shared/Media/Media';
import Media, { MediaType } from '@safelight/shared/Media/Media';
import Timecode from '@safelight/shared/Timecode';
import fuzzysearch from 'fuzzysearch';
import MimeMatcher from 'mime-matcher';
import InputGroup from 'primevue/inputgroup';
Expand Down Expand Up @@ -125,9 +233,20 @@ fileDialog.onChange((fileList) => {
const search = ref('');
const sortBy = ref<sortOptions>('Name');
const sortDescending = ref(false);
const layout = ref<string>('grid');
const gridItemSize = ref(176); // 11rem
const sortedAndFiltered = shallowRef<Media[]>([]);
// Reset sorting when changing to list
// List has its own sorting
watchEffect(() => {
if (layout.value == 'list') {
sortBy.value = 'Name';
sortDescending.value = false;
}
});
watchDebounced(
[CurrentProject.project.value?.media, search, sortBy, sortDescending],
sortAndFilter,
Expand Down
Original file line number Diff line number Diff line change
@@ -1,9 +1,18 @@
<template>
<div
role="gridcell"
class="border-round m-1 flex min-h-44 select-text flex-col rounded-md border-solid border-white/10"
class="border-round m-1 flex select-text flex-col rounded-md border-solid border-white/10"
style="border-width: 1px"
:style="{
width: size + 'px'
}"
:aria-label="item.name.value"
@contextmenu.prevent="
(ev) => {
closeOtherOverlays();
overlay?.toggle(ev);
}
"
>
<div
class="bg-checkerboard relative flex aspect-video w-full items-center justify-center overflow-clip rounded-t-md"
Expand All @@ -20,7 +29,7 @@
height="100%"
width="100%"
/>
<div class="mediaType">
<div v-if="size >= 96" class="mediaType">
<PhVideoCamera
v-if="item.isOfType(MediaType.Video)"
weight="bold"
Expand Down Expand Up @@ -51,6 +60,7 @@
{{ item.name.value }}
</p>
<Button
v-if="size >= 96"
title="Options"
text
rounded
Expand Down Expand Up @@ -112,10 +122,11 @@ import type Media from '@safelight/shared/Media/Media';
import { MediaType } from '@safelight/shared/Media/Media';
import type Menu from 'primevue/menu';
import type { MenuItem } from 'primevue/menuitem';
import type OverlayPanel from 'primevue/overlaypanel';
import OverlayPanel from 'primevue/overlaypanel';
const props = defineProps<{
item: Media;
size: number;
}>();
const menuItems = ref<MenuItem[]>([
Expand All @@ -134,6 +145,14 @@ const hasItemInTimeline = computed(
const alertt = (text: string) => window.alert(text);
const overlay = ref<OverlayPanel>();
function closeOtherOverlays() {
if (document.activeElement && 'blur' in document.activeElement) {
(document.activeElement as HTMLElement).blur();
}
// Weird, but fine
document.body.click();
}
</script>

<style lang="scss" scoped>
Expand Down

0 comments on commit a8be3aa

Please sign in to comment.