New component panel structure - template update
This commit is contained in:
parent
f58e4c5410
commit
5990d58c82
|
@ -1,11 +1,78 @@
|
||||||
<script>
|
<script>
|
||||||
// This should be fetched from somewhere in the future, rather than be hardcoded.
|
import { splitName } from "./pagesParsing/splitRootComponentName.js"
|
||||||
import components from "./temporaryPanelStructure.js"
|
import components from "./temporaryPanelStructure.js"
|
||||||
|
import ConfirmDialog from "components/common/ConfirmDialog.svelte"
|
||||||
|
import {
|
||||||
|
find,
|
||||||
|
sortBy,
|
||||||
|
groupBy,
|
||||||
|
values,
|
||||||
|
filter,
|
||||||
|
map,
|
||||||
|
uniqBy,
|
||||||
|
flatten,
|
||||||
|
} from "lodash/fp"
|
||||||
|
|
||||||
|
import {
|
||||||
|
getRecordNodes,
|
||||||
|
getIndexNodes,
|
||||||
|
getIndexSchema,
|
||||||
|
pipe,
|
||||||
|
} from "components/common/core"
|
||||||
|
|
||||||
import Tab from "./ComponentTab/Tab.svelte"
|
import Tab from "./ComponentTab/Tab.svelte"
|
||||||
import { store } from "builderStore"
|
import { store } from "builderStore"
|
||||||
|
|
||||||
|
export let toggleTab
|
||||||
|
|
||||||
|
let selectTemplateDialog
|
||||||
|
let selectedTemplateInstance
|
||||||
|
let templateInstances = []
|
||||||
|
|
||||||
|
let selectedComponent = null
|
||||||
|
|
||||||
const categories = components.categories
|
const categories = components.categories
|
||||||
let selectedCategory = categories[0]
|
let selectedCategory = categories[0]
|
||||||
|
|
||||||
|
const onTemplateChosen = template => {
|
||||||
|
selectedComponent = null
|
||||||
|
const { componentName, libName } = splitName(template.name)
|
||||||
|
const templateOptions = {
|
||||||
|
records: getRecordNodes(hierarchy),
|
||||||
|
indexes: getIndexNodes(hierarchy),
|
||||||
|
helpers: {
|
||||||
|
indexSchema: getIndexSchema(hierarchy),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
templateInstances = libraryModules[libName][componentName](templateOptions)
|
||||||
|
if (!templateInstances || templateInstances.length === 0) return
|
||||||
|
selectedTemplateInstance = templateInstances[0].name
|
||||||
|
selectTemplateDialog.show()
|
||||||
|
}
|
||||||
|
|
||||||
|
const onTemplateInstanceChosen = () => {
|
||||||
|
selectedComponent = null
|
||||||
|
const instance = templateInstances.find(
|
||||||
|
i => i.name === selectedTemplateInstance
|
||||||
|
)
|
||||||
|
store.addTemplatedComponent(instance.props)
|
||||||
|
toggleTab()
|
||||||
|
}
|
||||||
|
|
||||||
|
$: templatesByComponent = groupBy(t => t.component)($store.templates)
|
||||||
|
$: hierarchy = $store.hierarchy
|
||||||
|
$: libraryModules = $store.libraries
|
||||||
|
$: standaloneTemplates = pipe(
|
||||||
|
templatesByComponent,
|
||||||
|
[
|
||||||
|
values,
|
||||||
|
flatten,
|
||||||
|
filter(t => !$store.components.some(c => c.name === t.component)),
|
||||||
|
map(t => ({ name: splitName(t.component).componentName, template: t })),
|
||||||
|
uniqBy(t => t.name),
|
||||||
|
]
|
||||||
|
)
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="root">
|
<div class="root">
|
||||||
|
@ -19,10 +86,29 @@
|
||||||
{/each}
|
{/each}
|
||||||
</ul>
|
</ul>
|
||||||
<div class="panel">
|
<div class="panel">
|
||||||
<Tab list={selectedCategory} />
|
<Tab list={selectedCategory} {onTemplateChosen} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<ConfirmDialog
|
||||||
|
bind:this={selectTemplateDialog}
|
||||||
|
title="Choose Template"
|
||||||
|
onCancel={() => (selectedComponent = null)}
|
||||||
|
onOk={onTemplateInstanceChosen}>
|
||||||
|
{#each templateInstances.map(i => i.name) as instance}
|
||||||
|
<div class="uk-margin uk-grid-small uk-child-width-auto uk-grid">
|
||||||
|
<label>
|
||||||
|
<input
|
||||||
|
class="uk-radio"
|
||||||
|
type="radio"
|
||||||
|
bind:group={selectedTemplateInstance}
|
||||||
|
value={instance} />
|
||||||
|
<span class="template-instance-label">{instance}</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
</ConfirmDialog>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.tabs {
|
.tabs {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|
|
@ -2,15 +2,16 @@
|
||||||
import Item from "./Item.svelte"
|
import Item from "./Item.svelte"
|
||||||
import { store } from "builderStore"
|
import { store } from "builderStore"
|
||||||
export let list
|
export let list
|
||||||
|
export let onTemplateChosen
|
||||||
let category = list
|
let category = list
|
||||||
|
|
||||||
const onComponentChosen = component => {
|
const onComponentChosen = component => {
|
||||||
// if (component.template) {
|
if (component.template) {
|
||||||
// onTemplateChosen(component.template)
|
onTemplateChosen(component.template)
|
||||||
// } else {
|
} else {
|
||||||
store.addChildComponent(component._component)
|
store.addChildComponent(component._component)
|
||||||
// toggleTab()
|
toggleTab()
|
||||||
// }
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleClick = component => {
|
const handleClick = component => {
|
||||||
|
|
|
@ -0,0 +1,335 @@
|
||||||
|
<script>
|
||||||
|
import { splitName } from "./pagesParsing/splitRootComponentName.js"
|
||||||
|
import { store } from "builderStore"
|
||||||
|
import {
|
||||||
|
find,
|
||||||
|
sortBy,
|
||||||
|
groupBy,
|
||||||
|
values,
|
||||||
|
filter,
|
||||||
|
map,
|
||||||
|
uniqBy,
|
||||||
|
flatten,
|
||||||
|
} from "lodash/fp"
|
||||||
|
import { ImageIcon, InputIcon, LayoutIcon } from "components/common/Icons/"
|
||||||
|
import Select from "components/common/Select.svelte"
|
||||||
|
import Button from "components/common/PlusButton.svelte"
|
||||||
|
import ConfirmDialog from "components/common/ConfirmDialog.svelte"
|
||||||
|
import {
|
||||||
|
getRecordNodes,
|
||||||
|
getIndexNodes,
|
||||||
|
getIndexSchema,
|
||||||
|
pipe,
|
||||||
|
} from "components/common/core"
|
||||||
|
|
||||||
|
export let toggleTab
|
||||||
|
|
||||||
|
let componentLibraries = []
|
||||||
|
let current_view = "text"
|
||||||
|
let selectedComponent = null
|
||||||
|
let selectedLib
|
||||||
|
let selectTemplateDialog
|
||||||
|
let templateInstances = []
|
||||||
|
let selectedTemplateInstance
|
||||||
|
|
||||||
|
//Info: Components seem to be generated from individual templates. Will this be the same going forward
|
||||||
|
$: templatesByComponent = groupBy(t => t.component)($store.templates)
|
||||||
|
$: hierarchy = $store.hierarchy
|
||||||
|
$: libraryModules = $store.libraries
|
||||||
|
$: standaloneTemplates = pipe(
|
||||||
|
templatesByComponent,
|
||||||
|
[
|
||||||
|
values,
|
||||||
|
flatten,
|
||||||
|
filter(t => !$store.components.some(c => c.name === t.component)),
|
||||||
|
map(t => ({ name: splitName(t.component).componentName, template: t })),
|
||||||
|
uniqBy(t => t.name),
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
const addRootComponent = (component, allComponents) => {
|
||||||
|
const { libName } = splitName(component.name)
|
||||||
|
let group = find(r => r.libName === libName)(allComponents)
|
||||||
|
|
||||||
|
if (!group) {
|
||||||
|
group = {
|
||||||
|
libName,
|
||||||
|
components: [],
|
||||||
|
}
|
||||||
|
|
||||||
|
allComponents.push(group)
|
||||||
|
}
|
||||||
|
|
||||||
|
group.components.push(component)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onComponentChosen = component => {
|
||||||
|
if (component.template) {
|
||||||
|
onTemplateChosen(component.template)
|
||||||
|
} else {
|
||||||
|
store.addChildComponent(component.name)
|
||||||
|
toggleTab()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Info: Called from menu beside components with presets and templates
|
||||||
|
const onTemplateChosen = template => {
|
||||||
|
selectedComponent = null
|
||||||
|
const { componentName, libName } = splitName(template.name)
|
||||||
|
//Info: how will DB changes effect this?
|
||||||
|
const templateOptions = {
|
||||||
|
records: getRecordNodes(hierarchy),
|
||||||
|
indexes: getIndexNodes(hierarchy),
|
||||||
|
helpers: {
|
||||||
|
indexSchema: getIndexSchema(hierarchy),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
//Info: go off and get template instances by library and component name
|
||||||
|
//libraryModules and hierarchies (used above) come from builderStore
|
||||||
|
templateInstances = libraryModules[libName][componentName](templateOptions)
|
||||||
|
if (!templateInstances || templateInstances.length === 0) return
|
||||||
|
selectedTemplateInstance = templateInstances[0].name
|
||||||
|
selectTemplateDialog.show()
|
||||||
|
}
|
||||||
|
|
||||||
|
const onTemplateInstanceChosen = () => {
|
||||||
|
selectedComponent = null
|
||||||
|
const instance = templateInstances.find(
|
||||||
|
i => i.name === selectedTemplateInstance
|
||||||
|
)
|
||||||
|
debugger
|
||||||
|
store.addTemplatedComponent(instance.props)
|
||||||
|
toggleTab()
|
||||||
|
}
|
||||||
|
|
||||||
|
function generate_components_list(components) {
|
||||||
|
debugger
|
||||||
|
return ($store.currentFrontEndType === "page"
|
||||||
|
? $store.builtins.concat(components)
|
||||||
|
: components
|
||||||
|
).concat(standaloneTemplates)
|
||||||
|
}
|
||||||
|
|
||||||
|
$: {
|
||||||
|
const newComponentLibraries = []
|
||||||
|
|
||||||
|
for (let comp of sortBy(["name"])($store.components)) {
|
||||||
|
addRootComponent(comp, newComponentLibraries)
|
||||||
|
}
|
||||||
|
|
||||||
|
componentLibraries = newComponentLibraries
|
||||||
|
if (!selectedLib) selectedLib = newComponentLibraries[0].libName
|
||||||
|
}
|
||||||
|
|
||||||
|
$: componentLibrary = componentLibraries.find(l => l.libName === selectedLib)
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="root">
|
||||||
|
<Select on:change={e => (selectedLib = e.target.value)}>
|
||||||
|
{#each componentLibraries as lib}
|
||||||
|
<option value={lib.libName}>{lib.libName}</option>
|
||||||
|
{/each}
|
||||||
|
</Select>
|
||||||
|
|
||||||
|
<div class="library-container">
|
||||||
|
<!-- <ul>
|
||||||
|
<li>
|
||||||
|
<button
|
||||||
|
class:selected={current_view === 'text'}
|
||||||
|
on:click={() => (current_view = 'text')}>
|
||||||
|
<InputIcon />
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<button
|
||||||
|
class:selected={current_view === 'layout'}
|
||||||
|
on:click={() => (current_view = 'layout')}>
|
||||||
|
<LayoutIcon />
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<button
|
||||||
|
class:selected={current_view === 'media'}
|
||||||
|
on:click={() => (current_view = 'media')}>
|
||||||
|
<ImageIcon />
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
</ul> -->
|
||||||
|
|
||||||
|
{#if componentLibrary}
|
||||||
|
{#each generate_components_list(componentLibrary.components) as component}
|
||||||
|
<div class="component-container">
|
||||||
|
<div class="component" on:click={() => onComponentChosen(component)}>
|
||||||
|
<div class="name">{splitName(component.name).componentName}</div>
|
||||||
|
{#if (component.presets || templatesByComponent[component.name]) && component.name === selectedComponent}
|
||||||
|
<ul class="preset-menu">
|
||||||
|
{#if component.presets}
|
||||||
|
<span>{splitName(component.name).componentName} Presets</span>
|
||||||
|
{#each Object.keys(component.presets) as preset}
|
||||||
|
<li
|
||||||
|
on:click|stopPropagation={() => onComponentChosen(component, preset)}>
|
||||||
|
{preset}
|
||||||
|
</li>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
|
{#if templatesByComponent[component.name]}
|
||||||
|
<span>
|
||||||
|
{splitName(component.name).componentName} Templates
|
||||||
|
</span>
|
||||||
|
{#each templatesByComponent[component.name] as template}
|
||||||
|
<li
|
||||||
|
on:click|stopPropagation={() => onTemplateChosen(template)}>
|
||||||
|
{template.description}
|
||||||
|
</li>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
|
</ul>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
{#if component.presets || templatesByComponent[component.name]}
|
||||||
|
<Button
|
||||||
|
on:click={() => {
|
||||||
|
selectedComponent = selectedComponent ? null : component.name
|
||||||
|
}}>
|
||||||
|
<span
|
||||||
|
class="open-presets"
|
||||||
|
class:open={selectedComponent === component.name}>
|
||||||
|
...
|
||||||
|
</span>
|
||||||
|
</Button>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ConfirmDialog
|
||||||
|
bind:this={selectTemplateDialog}
|
||||||
|
title="Choose Template"
|
||||||
|
onCancel={() => (selectedComponent = null)}
|
||||||
|
onOk={onTemplateInstanceChosen}>
|
||||||
|
{#each templateInstances.map(i => i.name) as instance}
|
||||||
|
<div class="uk-margin uk-grid-small uk-child-width-auto uk-grid">
|
||||||
|
<label>
|
||||||
|
<input
|
||||||
|
class="uk-radio"
|
||||||
|
type="radio"
|
||||||
|
bind:group={selectedTemplateInstance}
|
||||||
|
value={instance} />
|
||||||
|
<span class="template-instance-label">{instance}</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
</ConfirmDialog>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.root {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
.library-container {
|
||||||
|
padding: 0 0 10px 0;
|
||||||
|
flex: 1 1 auto;
|
||||||
|
min-height: 0px;
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.component-container {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.component {
|
||||||
|
position: relative;
|
||||||
|
padding: 0 15px;
|
||||||
|
cursor: pointer;
|
||||||
|
border: 1px solid #d8d8d8;
|
||||||
|
border-radius: 2px;
|
||||||
|
margin: 5px 0;
|
||||||
|
height: 40px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
color: #000333;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
flex: 1;
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.component:hover {
|
||||||
|
background-color: var(--lightslate);
|
||||||
|
}
|
||||||
|
|
||||||
|
.component > .name {
|
||||||
|
color: #000333;
|
||||||
|
display: inline-block;
|
||||||
|
font-size: 13px;
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul {
|
||||||
|
list-style: none;
|
||||||
|
display: flex;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preset-menu {
|
||||||
|
flex-direction: column;
|
||||||
|
position: absolute;
|
||||||
|
top: 25px;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
z-index: 1;
|
||||||
|
background: #fafafa;
|
||||||
|
padding: 10px;
|
||||||
|
border-radius: 2px;
|
||||||
|
color: var(--secondary80);
|
||||||
|
}
|
||||||
|
|
||||||
|
.preset-menu > span {
|
||||||
|
font-size: 13px;
|
||||||
|
text-transform: uppercase;
|
||||||
|
margin-top: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preset-menu li {
|
||||||
|
font-size: 14px;
|
||||||
|
margin-top: 13px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preset-menu li:hover {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
li {
|
||||||
|
margin-right: 20px;
|
||||||
|
background: none;
|
||||||
|
border-radius: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* li button {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
border-radius: 5px;
|
||||||
|
padding: 13px;
|
||||||
|
outline: none;
|
||||||
|
cursor: pointer;
|
||||||
|
} */
|
||||||
|
|
||||||
|
/* .selected {
|
||||||
|
color: var(--button-text);
|
||||||
|
background: var(--background-button) !important;
|
||||||
|
} */
|
||||||
|
|
||||||
|
.open {
|
||||||
|
color: rgba(0, 85, 255, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.template-instance-label {
|
||||||
|
margin-left: 20px;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -140,7 +140,6 @@ export default {
|
||||||
description: 'A component that generates a table from your data.',
|
description: 'A component that generates a table from your data.',
|
||||||
icon: 'ri-archive-drawer-fill',
|
icon: 'ri-archive-drawer-fill',
|
||||||
commonProps: {},
|
commonProps: {},
|
||||||
modelRequired: true,
|
|
||||||
type: []
|
type: []
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -148,7 +147,12 @@ export default {
|
||||||
description: 'A component that generates a form from your data.',
|
description: 'A component that generates a form from your data.',
|
||||||
icon: 'ri-file-edit-fill',
|
icon: 'ri-file-edit-fill',
|
||||||
commonProps: {},
|
commonProps: {},
|
||||||
modelRequired: true,
|
component: "@budibase/materialdesign-components/Form",
|
||||||
|
template: {
|
||||||
|
component: "@budibase/materialdesign-components/Form",
|
||||||
|
description: "Form for saving a record",
|
||||||
|
name: "@budibase/materialdesign-components/recordForm",
|
||||||
|
},
|
||||||
type: []
|
type: []
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
Loading…
Reference in New Issue