Tidy up import data modal and change to be a detail popover

This commit is contained in:
Andrew Kingston 2024-10-25 16:24:40 +01:00
parent c110cb628d
commit dae550c21e
No known key found for this signature in database
3 changed files with 145 additions and 142 deletions

View File

@ -1,17 +1,91 @@
<script> <script>
import { ActionButton, Modal } from "@budibase/bbui" import {
import ImportModal from "../modals/ImportModal.svelte" ActionButton,
Label,
Button,
Body,
Layout,
notifications,
} from "@budibase/bbui"
import DetailPopover from "components/common/DetailPopover.svelte"
import TableDataImport from "components/backend/TableNavigator/TableDataImport.svelte"
import { createEventDispatcher } from "svelte"
import { API } from "api"
export let tableId export let tableId
export let tableType export let tableType
export let disabled export let disabled
let modal const dispatch = createEventDispatcher()
let popover
let rows = []
let allValid = false
let displayColumn = null
let identifierFields = []
let loading = false
const openPopover = () => {
rows = []
allValid = false
displayColumn = null
identifierFields = []
loading = false
popover.show()
}
const importData = async () => {
try {
loading = true
await API.importTableData({
tableId,
rows,
identifierFields,
})
notifications.success("Rows successfully imported")
popover.hide()
} catch (error) {
console.error(error)
notifications.error("Unable to import data")
} finally {
loading = false
}
// Always refresh rows just to be sure
dispatch("importrows")
}
</script> </script>
<ActionButton icon="DataUpload" quiet on:click={modal.show} {disabled}> <DetailPopover title="Import data" bind:this={popover}>
<svelte:fragment slot="anchor" let:open>
<ActionButton
icon="DataUpload"
quiet
on:click={openPopover}
{disabled}
selected={open}
>
Import Import
</ActionButton> </ActionButton>
<Modal bind:this={modal}> </svelte:fragment>
<ImportModal {tableId} {tableType} on:importrows /> <Body size="S">
</Modal> Import rows to an existing table from a CSV or JSON file. Only columns from
the file which exist in the table will be imported.
</Body>
<Layout gap="XS" noPadding>
<Label grey extraSmall>CSV or JSON file to import</Label>
<TableDataImport
{tableId}
{tableType}
bind:rows
bind:allValid
bind:displayColumn
bind:identifierFields
/>
</Layout>
<div>
<Button cta disabled={loading || !allValid} on:click={importData}>
Import
</Button>
</div>
</DetailPopover>

View File

@ -1,61 +0,0 @@
<script>
import {
ModalContent,
Label,
notifications,
Body,
Layout,
} from "@budibase/bbui"
import TableDataImport from "../../TableNavigator/ExistingTableDataImport.svelte"
import { API } from "api"
import { createEventDispatcher } from "svelte"
const dispatch = createEventDispatcher()
export let tableId
export let tableType
let rows = []
let allValid = false
let displayColumn = null
let identifierFields = []
async function importData() {
try {
await API.importTableData({
tableId,
rows,
identifierFields,
})
notifications.success("Rows successfully imported")
} catch (error) {
notifications.error("Unable to import data")
}
// Always refresh rows just to be sure
dispatch("importrows")
}
</script>
<ModalContent
title="Import Data"
confirmText="Import"
onConfirm={importData}
disabled={!allValid}
>
<Body size="S">
Import rows to an existing table from a CSV or JSON file. Only columns from
the file which exist in the table will be imported.
</Body>
<Layout gap="XS" noPadding>
<Label grey extraSmall>CSV or JSON file to import</Label>
<TableDataImport
{tableId}
{tableType}
bind:rows
bind:allValid
bind:displayColumn
bind:identifierFields
/>
</Layout>
</ModalContent>

View File

@ -1,5 +1,5 @@
<script> <script>
import { Select, Icon } from "@budibase/bbui" import { Select, Icon, Layout } from "@budibase/bbui"
import { FIELDS } from "constants/backend" import { FIELDS } from "constants/backend"
import { utils } from "@budibase/shared-core" import { utils } from "@budibase/shared-core"
import { canBeDisplayColumn } from "@budibase/frontend-core" import { canBeDisplayColumn } from "@budibase/frontend-core"
@ -184,7 +184,8 @@
} }
</script> </script>
<div class="dropzone"> <Layout noPadding gap="S">
<div class="dropzone">
<input <input
bind:this={fileInput} bind:this={fileInput}
disabled={loading} disabled={loading}
@ -202,9 +203,10 @@
Upload Upload
{/if} {/if}
</label> </label>
</div> </div>
{#if rawRows.length > 0 && !error}
<div class="schema-fields"> {#if rawRows.length > 0 && !error}
<div>
{#each Object.entries(schema) as [name, column]} {#each Object.entries(schema) as [name, column]}
<div class="field"> <div class="field">
<span>{column.name}</span> <span>{column.name}</span>
@ -239,15 +241,14 @@
</div> </div>
{/each} {/each}
</div> </div>
<div class="display-column">
<Select <Select
label="Display Column" label="Display Column"
bind:value={displayColumn} bind:value={displayColumn}
options={displayColumnOptions} options={displayColumnOptions}
sort sort
/> />
</div> {/if}
{/if} </Layout>
<style> <style>
.dropzone { .dropzone {
@ -269,7 +270,6 @@
box-sizing: border-box; box-sizing: border-box;
overflow: hidden; overflow: hidden;
border-radius: var(--border-radius-s); border-radius: var(--border-radius-s);
color: var(--ink);
padding: var(--spacing-m) var(--spacing-l); padding: var(--spacing-m) var(--spacing-l);
transition: all 0.2s ease 0s; transition: all 0.2s ease 0s;
display: inline-flex; display: inline-flex;
@ -283,20 +283,14 @@
align-items: center; align-items: center;
justify-content: center; justify-content: center;
width: 100%; width: 100%;
background-color: var(--grey-2); background-color: var(--spectrum-global-color-gray-300);
font-size: var(--font-size-xs); font-size: var(--font-size-s);
line-height: normal; line-height: normal;
border: var(--border-transparent); border: var(--border-transparent);
} }
.uploaded { .uploaded {
color: var(--blue); color: var(--spectrum-global-color-blue-600);
} }
.schema-fields {
margin-top: var(--spacing-xl);
}
.field { .field {
display: grid; display: grid;
grid-template-columns: 2fr 2fr 1fr auto; grid-template-columns: 2fr 2fr 1fr auto;
@ -322,8 +316,4 @@
.fieldStatusFailure :global(.spectrum-Icon) { .fieldStatusFailure :global(.spectrum-Icon) {
width: 12px; width: 12px;
} }
.display-column {
margin-top: var(--spacing-xl);
}
</style> </style>