1
0
mirror of https://github.com/pocketbase/pocketbase.git synced 2025-04-11 19:58:04 +02:00
pocketbase/ui/src/components/records/RecordsPicker.svelte

349 lines
10 KiB
Svelte
Raw Normal View History

<script>
import { createEventDispatcher } from "svelte";
import CommonHelper from "@/utils/CommonHelper";
import ApiClient from "@/utils/ApiClient";
import scrollend from "@/actions/scrollend";
import tooltip from "@/actions/tooltip";
import OverlayPanel from "@/components/base/OverlayPanel.svelte";
import Searchbar from "@/components/base/Searchbar.svelte";
import Draggable from "@/components/base/Draggable.svelte";
import RecordInfo from "@/components/records/RecordInfo.svelte";
import RecordUpsertPanel from "@/components/records/RecordUpsertPanel.svelte";
import { collections } from "@/stores/collections";
const dispatch = createEventDispatcher();
const uniqueId = "picker_" + CommonHelper.randomString(5);
const batchSize = 100;
export let value;
export let field;
let pickerPanel;
let upsertPanel;
let filter = "";
let list = [];
let selected = [];
let currentPage = 1;
let totalItems = 0;
let isLoadingList = false;
let isLoadingSelected = false;
$: maxSelect = field?.options?.maxSelect || null;
$: collectionId = field?.options?.collectionId;
$: displayFields = field?.options?.displayFields;
$: collection = $collections.find((c) => c.id == collectionId) || null;
$: if (typeof filter !== "undefined" && !isLoadingSelected && pickerPanel?.isActive()) {
loadList(true); // reset list on filter or list change
}
$: isLoading = isLoadingList || isLoadingSelected;
$: canLoadMore = totalItems > list.length;
$: canSelectMore = maxSelect === null || maxSelect > selected.length;
export function show() {
filter = "";
list = [];
selected = [];
loadSelected();
loadList(true);
return pickerPanel?.show();
}
export function hide() {
return pickerPanel?.hide();
}
async function loadSelected() {
const selectedIds = CommonHelper.toArray(value);
if (!collectionId || !selectedIds.length) {
return;
}
isLoadingSelected = true;
let loadedItems = [];
// batch load all selected records to avoid parser stack overflow errors
const filterIds = selectedIds.slice();
const loadPromises = [];
while (filterIds.length > 0) {
const filters = [];
for (const id of filterIds.splice(0, batchSize)) {
filters.push(`id="${id}"`);
}
loadPromises.push(
ApiClient.collection(collectionId).getFullList(batchSize, {
filter: filters.join("||"),
$autoCancel: false,
})
);
}
try {
await Promise.all(loadPromises).then((values) => {
loadedItems = loadedItems.concat(...values);
});
// preserve selected order
selected = [];
for (const id of selectedIds) {
const item = CommonHelper.findByKey(loadedItems, "id", id);
if (item) {
selected.push(item);
}
}
if (!filter.trim()) {
// add the selected models to the list (if not already)
list = CommonHelper.filterDuplicatesByKey(selected.concat(list));
}
} catch (err) {
2023-05-13 22:10:14 +03:00
ApiClient.error(err);
}
isLoadingSelected = false;
}
async function loadList(reset = false) {
if (!collectionId) {
return;
}
isLoadingList = true;
if (reset) {
if (!filter.trim()) {
// prepend the loaded selected items
list = CommonHelper.toArray(selected).slice();
} else {
list = [];
}
}
try {
const page = reset ? 1 : currentPage + 1;
const fallbackSearchFields = CommonHelper.getAllCollectionIdentifiers(collection);
const result = await ApiClient.collection(collectionId).getList(page, batchSize, {
filter: CommonHelper.normalizeSearchFilter(filter, fallbackSearchFields),
2023-03-27 19:43:46 +03:00
sort: !collection?.$isView ? "-created" : "",
$cancelKey: uniqueId + "loadList",
});
list = CommonHelper.filterDuplicatesByKey(list.concat(result.items));
currentPage = result.page;
totalItems = result.totalItems;
} catch (err) {
2023-05-13 22:10:14 +03:00
ApiClient.error(err);
}
isLoadingList = false;
}
$: isSelected = function (record) {
return CommonHelper.findByKey(selected, "id", record.id);
};
function select(record) {
if (maxSelect == 1) {
selected = [record];
} else if (canSelectMore) {
CommonHelper.pushOrReplaceByKey(selected, record);
selected = selected;
}
}
function deselect(record) {
CommonHelper.removeByKey(selected, "id", record.id);
selected = selected;
}
function toggle(record) {
if (isSelected(record)) {
deselect(record);
} else {
select(record);
}
}
function save() {
if (maxSelect != 1) {
value = selected.map((r) => r.id);
} else {
value = selected?.[0]?.id || "";
}
dispatch("save", selected);
hide();
}
</script>
<OverlayPanel bind:this={pickerPanel} popup class="overlay-panel-xl" on:hide on:show {...$$restProps}>
<svelte:fragment slot="header">
<h4>
Select <strong>{collection?.name || ""}</strong> records
</h4>
</svelte:fragment>
<div class="flex m-b-base flex-gap-10">
<Searchbar
value={filter}
autocompleteCollection={collection}
on:submit={(e) => (filter = e.detail)}
/>
2023-03-27 19:43:46 +03:00
{#if !collection?.$isView}
<button
type="button"
class="btn btn-transparent btn-hint p-l-sm p-r-sm"
on:click={() => upsertPanel?.show()}
>
<div class="txt">New record</div>
</button>
{/if}
</div>
<div
class="list picker-list m-b-base"
use:scrollend={() => {
if (canLoadMore && !isLoadingList) {
loadList();
}
}}
>
{#each list as record (record.id)}
{@const selected = isSelected(record)}
<div
tabindex="0"
class="list-item handle"
class:selected
class:disabled={!selected && maxSelect > 1 && !canSelectMore}
on:click={() => toggle(record)}
on:keydown={(e) => {
if (e.code === "Enter" || e.code === "Space") {
e.preventDefault();
e.stopPropagation();
toggle(record);
}
}}
>
{#if selected}
<i class="ri-checkbox-circle-fill txt-success" />
{:else}
<i class="ri-checkbox-blank-circle-line txt-disabled" />
{/if}
<div class="content">
<RecordInfo {record} {displayFields} />
</div>
2023-03-27 19:43:46 +03:00
{#if !collection?.$isView}
<div class="actions nonintrusive">
<button
type="button"
class="btn btn-sm btn-circle btn-transparent btn-hint m-l-auto"
use:tooltip={"Edit"}
on:keydown|stopPropagation
on:click|stopPropagation={() => upsertPanel?.show(record)}
>
<i class="ri-pencil-line" />
</button>
</div>
{/if}
</div>
{:else}
<div class="list-item">
{#if isLoading}
<div class="block txt-center">
<span class="loader loader-sm active" />
</div>
{:else}
<span class="txt txt-hint">No records found.</span>
{#if filter?.length}
<button type="button" class="btn btn-hint btn-sm" on:click={() => (filter = "")}>
<span class="txt">Clear filters</span>
</button>
{/if}
{/if}
</div>
{/each}
</div>
<h5 class="section-title">
Selected
{#if maxSelect > 1}
({selected.length} of MAX {maxSelect})
{/if}
</h5>
{#if selected.length}
<div class="selected-list">
{#each selected as record, i}
<Draggable bind:list={selected} index={i} let:dragging let:dragover>
<span class="label" class:label-danger={dragging} class:label-warning={dragover}>
<RecordInfo {record} {displayFields} />
<button
type="button"
title="Remove"
class="btn btn-circle btn-transparent btn-hint btn-xs"
on:click={() => deselect(record)}
>
<i class="ri-close-line" />
</button>
</span>
</Draggable>
{/each}
</div>
{:else}
<p class="txt-hint">No selected records.</p>
{/if}
<svelte:fragment slot="footer">
<button type="button" class="btn btn-transparent" on:click={() => hide()}>
<span class="txt">Cancel</span>
</button>
<button type="button" class="btn" on:click={() => save()}>
<span class="txt">Set selection</span>
</button>
</svelte:fragment>
</OverlayPanel>
<RecordUpsertPanel
bind:this={upsertPanel}
{collection}
on:save={(e) => {
CommonHelper.removeByKey(list, "id", e.detail.id);
list.unshift(e.detail);
list = list;
select(e.detail);
}}
on:delete={(e) => {
CommonHelper.removeByKey(list, "id", e.detail.id);
list = list;
deselect(e.detail);
}}
/>
<style lang="scss">
.picker-list {
max-height: 380px;
}
.selected-list {
display: flex;
flex-wrap: wrap;
align-items: center;
gap: 10px;
max-height: 220px;
overflow: auto;
}
</style>