Improve setting rendering and add dynamic section title

This commit is contained in:
Andrew Kingston 2023-12-08 16:10:56 +00:00
parent 1bf556b4f2
commit 173a76dc52
7 changed files with 105 additions and 159 deletions

View File

@ -4,7 +4,6 @@
export let name
export let initiallyShow = false
export let collapsible = true
export let noPadding = false
let show = initiallyShow
@ -29,7 +28,6 @@
class="property-panel"
class:show={show || !collapsible}
class:no-title={!name}
class:no-padding={noPadding}
>
<slot />
</div>
@ -83,10 +81,6 @@
padding: var(--spacing-xl);
}
.property-panel.no-title.no-padding {
padding: 0px;
}
.show {
display: flex;
flex-direction: column;

View File

@ -26,6 +26,7 @@ import FieldConfiguration from "./controls/FieldConfiguration/FieldConfiguration
import ButtonConfiguration from "./controls/ButtonConfiguration/ButtonConfiguration.svelte"
import RelationshipFilterEditor from "./controls/RelationshipFilterEditor.svelte"
import FormStepConfiguration from "./controls/FormStepConfiguration.svelte"
import FormStepControls from "components/design/settings/controls/FormStepControls.svelte"
const componentMap = {
text: DrawerBindableInput,
@ -53,6 +54,7 @@ const componentMap = {
fieldConfiguration: FieldConfiguration,
buttonConfiguration: ButtonConfiguration,
stepConfiguration: FormStepConfiguration,
formStepControls: FormStepControls,
columns: ColumnEditor,
"columns/basic": BasicColumnEditor,
"columns/grid": GridColumnEditor,

View File

@ -4,7 +4,6 @@
import { getDatasourceForProvider } from "builderStore/dataBinding"
import { currentAsset, store } from "builderStore"
import { Helpers } from "@budibase/bbui"
import FormStepControls from "./FormStepControls.svelte"
import { writable } from "svelte/store"
import { buildMultiStepFormBlockButtonConfig } from "@budibase/frontend-core/src/utils/utils"
@ -14,33 +13,32 @@
export let bindings
const dispatch = createEventDispatcher()
let stepState = [...(value || [])]
const stepStore = writable({
stepsCount: stepState?.length || 1,
const multiStepStore = writable({
stepCount: value?.length ?? 0,
currentStep: 0,
})
setContext("step-form-block", stepStore)
setContext("multi-step-form-block", multiStepStore)
$: ({ currentStep } = $stepStore)
$: if (stepState.length) {
stepStore.update(state => ({
...state,
stepsCount: stepState.length || 0,
}))
}
$: currentStep = $multiStepStore.currentStep
$: stepCount = value?.length || 0
$: multiStepStore.update(state => ({ ...state, stepCount }))
$: defaultButtonConfig = buildMultiStepFormBlockButtonConfig({
_id: componentInstance._id,
stepCount: value?.length,
currentStep,
stepCount: stepCount,
currentStep: currentStep,
})
// Step Definition Settings
let compSettings = [
$: dataSource = getDatasourceForProvider($currentAsset, componentInstance)
$: emitCurrentStep(currentStep)
$: sectionName = getSectionName($multiStepStore)
$: stepDef = {
settings: [
{
customType: "formStepControl",
section: true,
name: sectionName,
settings: [
{
type: "formStepControls",
label: "Multi-steps",
key: "steps",
},
@ -68,7 +66,17 @@
wide: true,
nested: true,
},
]
],
},
],
}
const getSectionName = ({ stepCount, currentStep }) => {
if (stepCount <= 1) {
return "Details"
}
return `Details (Step ${currentStep + 1}/${stepCount})`
}
const emitCurrentStep = step => {
store.actions.preview.sendEvent("builder-meta", {
@ -79,70 +87,52 @@
})
}
$: dataSource = getDatasourceForProvider($currentAsset, componentInstance)
$: stepDef = {
component: "@budibase/standard-components/multistepformblock-step",
name: "Formblock step",
settings: compSettings,
}
const addStep = () => {
const newStepIdx = currentStep + 1
stepState = [
...stepState.slice(0, newStepIdx),
const nextStep = currentStep + 1
dispatch("change", [
...value.slice(0, nextStep),
{},
...stepState.slice(newStepIdx),
]
stepStore.update(state => ({
...value.slice(nextStep),
])
multiStepStore.update(state => ({
...state,
currentStep: newStepIdx,
currentStep: nextStep,
}))
dispatch("change", stepState)
emitCurrentStep(newStepIdx)
}
const removeStep = () => {
const clone = stepState.map(x => x)
clone.splice(currentStep, 1)
const targetStepIdx = Math.max(currentStep - 1, 0)
stepState = clone.map(x => x)
stepStore.update(state => ({
dispatch("change", value.toSpliced(currentStep, 1))
const newStep = Math.min(currentStep, stepCount - 2)
multiStepStore.update(state => ({
...state,
currentStep: targetStepIdx,
currentStep: newStep,
}))
dispatch("change", stepState)
emitCurrentStep(targetStepIdx)
}
const previousStep = () => {
const prevStepIdx = Math.max(currentStep - 1, 0)
stepStore.update(state => ({
multiStepStore.update(state => ({
...state,
currentStep: prevStepIdx,
}))
emitCurrentStep(prevStepIdx)
}
const nextStep = () => {
const nextStepIdx = currentStep + 1
stepStore.update(state => ({
multiStepStore.update(state => ({
...state,
currentStep: Math.min(nextStepIdx, stepState.length - 1),
currentStep: Math.min(nextStepIdx, value.length - 1),
}))
emitCurrentStep(nextStepIdx)
}
const updateStep = (field, val) => {
stepState[currentStep] ||= {}
stepState[currentStep][field.key] = val
dispatch("change", stepState)
const newStep = {
...value[currentStep],
[field.key]: val,
}
let newValue = value.slice()
newValue[currentStep] = newStep
dispatch("change", newValue)
}
const handleStepAction = action => {
@ -158,9 +148,6 @@
break
case "previousStep":
previousStep()
break
default:
console.log("Nothing")
}
}
@ -186,32 +173,27 @@
}
}
$: stepConfigInstance = buildPseudoInstance(stepState?.[currentStep] || {})
$: stepConfigInstance = buildPseudoInstance(value[currentStep] || {})
</script>
<span class="settings-wrap">
<div class="nested-section">
<ComponentSettingsSection
includeHidden
componentInstance={stepConfigInstance}
componentDefinition={stepDef}
onUpdateSetting={processUpdate}
getCustomComponent={type => {
const types = { formStepControl: FormStepControls }
return types[type]
}}
getCustomSectionTitle={section => {
console.log(section.name)
if (section.name === "Details" && stepState?.length > 0) {
return `Details (${currentStep}/${stepState?.length})`
}
return section.name
}}
showSectionTitle={false}
showInstanceName={false}
isScreen={false}
noPadding={true}
nested={true}
{bindings}
{componentBindings}
/>
</span>
</div>
<style>
.nested-section {
margin: 0 calc(-1 * var(--spacing-xl)) calc(-1 * var(--spacing-xl))
calc(-1 * var(--spacing-xl));
border-top: var(--border-light);
}
</style>

View File

@ -2,15 +2,10 @@
import { createEventDispatcher, getContext } from "svelte"
import { ActionButton } from "@budibase/bbui"
const stepState = getContext("step-form-block")
const multiStepStore = getContext("multi-step-form-block")
const dispatch = createEventDispatcher()
$: ({ stepsCount, currentStep } = $stepState)
const parseLastIdx = stepsCount => {
return Math.max(stepsCount - 1, 0)
}
$: lastIdx = parseLastIdx(stepsCount)
$: ({ stepCount, currentStep } = $multiStepStore)
const stepAction = action => {
dispatch("change", {
@ -19,7 +14,7 @@
}
</script>
{#if stepsCount === 1}
{#if stepCount === 1}
<ActionButton
icon="MultipleAdd"
secondary
@ -43,7 +38,7 @@
<ActionButton
size="S"
secondary
disabled={currentStep === lastIdx}
disabled={currentStep === stepCount - 1}
icon="ChevronRight"
on:click={() => {
stepAction("nextStep")
@ -53,7 +48,7 @@
size="S"
secondary
icon="Close"
disabled={stepsCount === 1}
disabled={stepCount === 1}
on:click={() => {
stepAction("removeStep")
}}

View File

@ -15,12 +15,9 @@
export let componentBindings
export let isScreen = false
export let onUpdateSetting
export let getCustomComponent
export let getCustomSectionTitle
export let showSectionTitle = true
export let includeHidden = false
export let tag
export let noPadding = false
$: sections = getSections(
componentInstance,
@ -133,30 +130,13 @@
})
}
const resolveComponentByType = setting => {
if (setting.type) {
return getComponentForSetting(setting)
} else if (setting.customType && typeof getCustomComponent === "function") {
return getCustomComponent(setting.customType)
}
}
const resolveSectionName = section => {
console.log(resolveSectionName)
if (typeof getCustomSectionTitle === "function") {
return getCustomSectionTitle(section)
} else {
return section.name
}
}
const canRenderControl = (instance, setting, isScreen, includeHidden) => {
// Prevent rendering on click setting for screens
if (setting?.type === "event" && isScreen) {
return false
}
// Check we have a component to render for this setting
const control = resolveComponentByType(setting)
const control = getComponentForSetting(setting)
if (!control) {
return false
}
@ -172,9 +152,8 @@
{#if section.visible}
<DetailSummary
name={showSectionTitle ? section.name : ""}
show={section.collapsed !== true}
{noPadding}
initiallyShow={section.collapsed !== true}
collapsible={section.name !== "General"}
>
{#if section.info}
<div class="section-info">
@ -191,7 +170,7 @@
{#if setting.visible}
<PropertyControl
type={setting.type}
control={resolveComponentByType(setting)}
control={getComponentForSetting(setting)}
label={setting.label}
labelHidden={setting.labelHidden}
wide={setting.wide}

View File

@ -6056,10 +6056,6 @@
"options": ["Create", "Update", "View"],
"defaultValue": "Create"
},
{
"name": "Details",
"section": true,
"settings": [
{
"type": "stepConfiguration",
"key": "steps",
@ -6068,8 +6064,6 @@
"resetOn": ["dataSource", "actionType"],
"defaultValue": [{}]
}
]
}
],
"actions": [
{

View File

@ -102,7 +102,7 @@
buttons ||
Utils.buildMultiStepFormBlockButtonConfig({
_id: $component.id,
stepCount: steps?.length,
stepCount: steps?.length ?? 0,
currentStep: idx,
}),
}