platform/packages/ui/src/components/ListView.svelte
Alexander Platov 97abd2fc12
UBER-675: updated pop-ups and components layout (#3631)
Signed-off-by: Alexander Platov <sas_lord@mail.ru>
2023-08-25 16:49:14 +07:00

113 lines
2.8 KiB
Svelte

<!--
// Copyright © 2022 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
-->
<script lang="ts">
import { createEventDispatcher } from 'svelte'
import { mouseAttractor, resizeObserver } from '..'
export let selection: number = 0
export let count: number
export let addClass: string | undefined = undefined
export let noScroll: boolean = false
export let kind: 'default' | 'thin' = 'default'
const refs: HTMLElement[] = []
const dispatch = createEventDispatcher()
let oldSelection = Date.now()
function onRow (item: number): void {
if (updateSelection(item)) {
dispatch('on-select', item)
}
}
function updateSelection (item: number): boolean {
const now = Date.now()
if (now - oldSelection >= 25) {
selection = item
oldSelection = now
return true
}
return false
}
export function select (pos: number): void {
if (pos < 0) {
pos = 0
}
if (pos >= count) {
pos = count - 1
}
const r = refs[pos]
onRow(pos)
if (r !== undefined) {
r?.scrollIntoView({ behavior: 'auto', block: 'nearest' })
}
}
</script>
{#if count}
<div
class="list-container {kind} flex-col flex-grow"
style:overflow={noScroll ? 'visible' : 'auto'}
use:resizeObserver={() => {
dispatch('changeContent')
}}
>
{#each Array(count) as _, row}
<slot name="category" item={row} />
<!-- svelte-ignore a11y-click-events-have-key-events -->
<div
class="list-item{addClass ? ` ${addClass}` : ''}"
class:selection={row === selection}
on:mouseover={mouseAttractor(() => onRow(row))}
on:mouseenter={mouseAttractor(() => onRow(row))}
on:focus={() => {}}
bind:this={refs[row]}
on:click={() => dispatch('click', row)}
>
<slot name="item" item={row} />
</div>
{/each}
</div>
{/if}
<style lang="scss">
.list-container {
min-width: 0;
// border-radius: 0.25rem;
user-select: none;
.list-item {
margin: 0 0.5rem;
min-width: 0;
border-radius: 0.25rem;
}
&.thin {
.list-item {
margin: 0 0.375rem;
border-radius: 0.375rem;
}
.list-item + .list-item {
margin-top: 0.375rem;
}
}
.selection {
background-color: var(--theme-popup-hover);
}
}
</style>