From 7ae29a603072a9d00981a0ad569614829ea77cd4 Mon Sep 17 00:00:00 2001 From: Michael Shanks Date: Fri, 31 Jan 2020 23:11:50 +0000 Subject: [PATCH] #24 - Control Flow (#79) * removed binding references to array type * refactored initialiseChildren into seperate file * render function, with code blocks - tested simple cases * few mores tests for control flow * md components - getting TestApp to work * new render wrapper - bug fix * client: providing access to component root elements * code editor working * code editor improvements --- .../bootstrap-components/dist/generators.js | 2 +- packages/builder/package.json | 1 + .../src/builderStore/buildCodeForScreens.js | 35 + packages/builder/src/builderStore/store.js | 28 +- .../builder/src/builderStore/store.js.orig | 799 ++++++++++++++++++ .../src/common/Icons/CircleIndicator.svelte | 3 + packages/builder/src/common/Icons/index.js | 3 +- .../builder/src/common/Icons/index.js.orig | 12 + packages/builder/src/index.html | 2 + packages/builder/src/main.js | 2 + .../src/userInterface/CodeEditor.svelte | 158 ++-- .../src/userInterface/ComponentPanel.svelte | 25 +- .../userInterface/ComponentPanel.svelte.orig | 149 ++++ .../userInterface/CurrentItemPreview.svelte | 3 + .../src/userInterface/PropsView.svelte | 2 +- .../src/userInterface/PropsView.svelte.orig | 74 ++ .../builder/tests/buildCodeForScreen.spec.js | 64 ++ packages/client/src/index.js | 2 + .../client/src/render/initialiseChildren.js | 2 +- packages/client/src/render/renderComponent.js | 2 +- packages/server/utilities/builder/buildApp.js | 5 +- .../standard-components/dist/generators.js | 2 +- 22 files changed, 1276 insertions(+), 99 deletions(-) create mode 100644 packages/builder/src/builderStore/buildCodeForScreens.js create mode 100644 packages/builder/src/builderStore/store.js.orig create mode 100644 packages/builder/src/common/Icons/CircleIndicator.svelte create mode 100644 packages/builder/src/common/Icons/index.js.orig create mode 100644 packages/builder/src/userInterface/ComponentPanel.svelte.orig create mode 100644 packages/builder/src/userInterface/PropsView.svelte.orig create mode 100644 packages/builder/tests/buildCodeForScreen.spec.js diff --git a/packages/bootstrap-components/dist/generators.js b/packages/bootstrap-components/dist/generators.js index 6f18e32cae..1021b09753 100644 --- a/packages/bootstrap-components/dist/generators.js +++ b/packages/bootstrap-components/dist/generators.js @@ -447,4 +447,4 @@ const navItem = ({record}) => ({ }); export { app, forms, indexTables, recordHomePageComponents as recordHomepages }; -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"generators.js","sources":["../src/generators/buttonGenerators.js","../src/generators/formsGenerator.js","../src/generators/getRecordPath.js","../src/generators/indexTablesGenerator.js","../src/generators/recordHomePageGenerator.js","../src/generators/selectedNavContentGenerator.js","../src/generators/appGenerator.js"],"sourcesContent":["export const buttons = () => [\n    {\n        name: \"common/Primary Button\",\n        description: \"Bootstrap primary button \",\n        inherits: \"@budibase/standard-components/button\",\n        props: {\n            className: \"btn btn-primary\"\n        }\n    },\n    {\n        name: \"common/Default Button\",\n        description: \"Bootstrap default button\",\n        inherits: \"@budibase/standard-components/button\",\n        props: {\n            className: \"btn btn-secondary\"\n        }\n    }\n]","import {buttons} from \"./buttonGenerators\";\n\nexport const forms = ({records, indexes, helpers}) => \n    [\n        ...records.map(root),\n        ...buttons({records, indexes, helpers})\n    ];\n\nexport const formName = record =>  `${record.name}/${record.name} Form`;\n\nconst root = record => ({\n    name: formName(record),\n    description: `Control for creating/updating '${record.nodeKey()}' `,\n    inherits: \"@budibase/standard-components/div\",\n    props: {\n        className:\"p-1\",\n        children: [\n            {\n                component: {\n                    _component: \"@budibase/standard-components/h3\",\n                    text: `Edit ${record.name}`,\n                }\n            },\n            form(record),\n            saveCancelButtons(record)\n        ]\n    }\n}) \n\nconst form = record => ({\n    component: {\n        _component: \"@budibase/standard-components/form\",\n        formControls: \n            record.fields.map(f => formControl(record, f))\n    }\n})\n\nconst formControl = (record, field) => {\n    if(field.type === \"string\" && field.typeOptions.values && field.typeOptions.values.length > 0) {\n        return ({\n            control: {\n                _component: \"@budibase/standard-components/select\",\n                options: field.typeOptions.values.map(v => ({id:v, value:v})),\n                value: {\n                    \"##bbstate\":`${record.name}.${field.name}`,\n                    \"##bbsource\":\"store\"\n                },\n                className: \"form-control\"\n            },\n            label: field.label\n        });\n    } else {\n        return ({\n            control: {\n                _component: \"@budibase/standard-components/input\",\n                value: {\n                    \"##bbstate\":`${record.name}.${field.name}`,\n                    \"##bbsource\":\"store\"\n                },\n                className: \"form-control\",\n                type: field.type === \"string\" ? \"text\"\n                    : field.type === \"datetime\" ? \"date\"\n                    : field.type === \"number\" ? \"number\"\n                    : \"text\"\n            },\n            label: field.label\n        });\n    }\n}\n\nconst saveCancelButtons = (record) => ({\n    component: {\n        _component: \"@budibase/standard-components/stackpanel\",\n        direction: \"horizontal\",\n        children: [\n            paddedPanelForButton({\n                _component: \"common/Primary Button\",\n                contentText: `Save ${record.name}`,\n                onClick: [                \n                    {\n                        \"##eventHandlerType\": \"Save Record\",\n                        parameters: {\n                            statePath: `${record.name}`,\n                        }\n                    },\n                    {\n                        \"##eventHandlerType\": \"Set State\",\n                        parameters: {\n                            path: `isEditing${record.name}`,\n                            value: \"\"\n                        }\n                    }\n                ]\n            }),\n            paddedPanelForButton({\n                _component: \"common/Default Button\",\n                contentText: `Cancel`,\n                onClick: [\n                    {\n                        \"##eventHandlerType\": \"Set State\",\n                        parameters: {\n                            path: `isEditing${record.name}`,\n                            value: \"\"\n                        }\n                    }\n                ]\n            })\n        ]\n    }\n})\n\nconst paddedPanelForButton = (button) => ({\n    control: {\n        _component: \"@budibase/standard-components/div\",\n        className: \"btn-group\",\n        children: [\n            {\n                component: button\n            }\n        ]\n    }\n});\n\n","export const getRecordPath = (record) => {\n\n    const parts = [];\n\n    const add = (current) => {\n        parts.push(current.name);\n        if(current.parent().type === \"root\") {\n            return;\n        }\n\n        add(current.parent());\n    }\n\n    return parts.reverse().join(\"/\");\n}","import { getRecordPath } from \"./getRecordPath\";\n\nexport const indexTables = ({indexes, helpers}) => \n    indexes.map(i => indexTable(i, helpers));\n\nconst excludedColumns = [\"id\", \"isNew\", \"key\", \"type\", \"sortKey\"];\n\nexport const indexTableProps = (index, helpers) => ({\n    data: {\n        \"##bbstate\":index.nodeKey(),\n        \"##bbsource\":\"store\"\n    },\n    tableClass: \"table table-hover\",\n    theadClass: \"thead-dark\",\n    columns: helpers\n                .indexSchema(index)\n                .filter(c => !excludedColumns.includes(c.name))\n                .map(column),\n    onRowClick: [\n        {\n            \"##eventHandlerType\": \"Set State\",\n            parameters: {\n                path: `selectedrow_${index.name}`,\n                value: {\n                    \"##bbstate\": \"key\",\n                    \"##bbsource\": \"event\"\n                }\n            },\n        }\n    ]\n});\n\nexport const getIndexTableName = (index, record) => {\n    record = record \n             || index.parent().type === \"record\" ? index.parent() : null;\n    \n    return (record\n            ? `${getRecordPath(record)}/${index.name} Table`\n            : `${index.name} Table`);\n}\n\nconst indexTable = (index, helpers) => ({\n    name: getIndexTableName(index),\n    inherits: \"@budibase/standard-components/table\",\n    props: indexTableProps(index, helpers)\n});\n\nconst column = (col) => ({\n    title: col.name,\n    value: {\n        \"##bbstate\": col.name,\n        \"##bbsource\":\"context\"\n    }\n})","import {\n    getIndexTableName, indexTables\n} from \"./indexTablesGenerator\";\n\nimport {\n    buttons\n} from \"./buttonGenerators\";\n\nexport const recordHomePageComponents = ({indexes, records, helpers}) => \n    [   \n        ...recordHomepages({indexes, records})\n          .map(component),\n\n        ...recordHomepages({indexes, records})\n            .map(homePageButtons),\n        \n        ...indexTables({indexes, records, helpers}),\n\n        ...buttons({indexes, buttons, helpers})\n    ]\n\n\nconst findIndexForRecord = (indexes, record) => {\n    const forRecord = indexes.filter(i => i.allowedRecordNodeIds.includes(record.nodeId));\n    if(forRecord.length === 0) return;\n    if(forRecord.length === 1) return forRecord[0];\n    const noMap = forRecord.filter(i => !i.filter || !i.filter.trim());\n    if(noMap.length === 0) forRecord[0];\n    return noMap[0];\n}\n\nexport const recordHomepages = ({indexes, records}) => \n    records.filter(r => r.parent().type === \"root\")\n        .map(r =>({\n            record:r, \n            index:findIndexForRecord(indexes, r)\n        }))\n        .filter(r => r.index);\n\n\nexport const  homepageComponentName = (record) => \n    `${record.name}/${record.name} homepage`;\n\nconst component = ({record, index}) => ({\n    inherits: \"@budibase/standard-components/div\",\n    name: homepageComponentName(record),\n    props: {\n        className: \"d-flex flex-column h-100\",\n        children: [\n            {\n                component: {\n                    _component: `${record.name}/homepage buttons`,\n                }\n            },\n            {\n                component: {\n                    _component: getIndexTableName(index)\n                },\n                className: \"flex-gow-1 overflow-auto\"\n            }\n        ],\n        onLoad: [\n            {\n                \"##eventHandlerType\": \"Set State\",\n                parameters: {\n                    path: `isEditing${record.name}`,\n                    value: \"\"\n                }\n            },\n            {\n                \"##eventHandlerType\": \"List Records\",\n                parameters: {\n                    statePath: index.nodeKey(),\n                    indexKey: index.nodeKey()\n                }\n            }\n        ]\n    }\n\n});\n\nconst homePageButtons = ({index, record}) => ({\n    inherits: \"@budibase/standard-components/div\",\n    name: `${record.name}/homepage buttons`,\n    props: {\n        className: \"btn-toolbar mt-4 mb-2\",\n        children: [\n            {\n                component: {\n                    _component: \"@budibase/standard-components/div\",\n                    className: \"btn-group mr-3\",\n                    children: [\n                        {\n                            component: {\n                                _component: \"common/Default Button\",\n                                contentText: `Create ${record.name}`,\n                                onClick: [\n                                    {\n                                        \"##eventHandlerType\": \"Get New Record\",\n                                        parameters: {\n                                            statePath: record.name,\n                                            collectionKey: `/${record.collectionName}`,\n                                            childRecordType: record.name\n                                        }\n                                    }, \n                                    {\n                                        \"##eventHandlerType\": \"Set State\",\n                                        parameters: {\n                                            path: `isEditing${record.name}`,\n                                            value: \"true\"\n                                        }\n                                    }\n                                ]\n                            }\n                        },\n                        {\n                            component: {\n                                _component: \"common/Default Button\",\n                                contentText: `Refresh`,\n                                onClick: [\n                                    {\n                                        \"##eventHandlerType\": \"List Records\",\n                                        parameters: {\n                                            statePath: index.nodeKey(),\n                                            indexKey: index.nodeKey()\n                                        }\n                                    }\n                                ]\n                            }\n                        }\n                    ]\n                }\n            },\n            {\n                component: {\n                    _component: \"@budibase/standard-components/if\",\n                    condition: `$store.selectedrow_${index.name} && $store.selectedrow_${index.name}.length > 0`,\n                    thenComponent: {\n                        _component: \"@budibase/standard-components/div\",\n                        className: \"btn-group\",\n                        children: [\n                            {\n                                component: {\n                                    _component: \"common/Default Button\",\n                                    contentText: `Edit ${record.name}`,\n                                    onClick: [\n                                        {\n                                            \"##eventHandlerType\": \"Load Record\",\n                                            parameters: {\n                                                statePath: record.name,\n                                                recordKey: {\n                                                    \"##bbstate\" : `selectedrow_${index.name}`,\n                                                    \"##source\": \"store\"\n                                                }\n                                            }\n                                        }, \n                                        {\n                                            \"##eventHandlerType\": \"Set State\",\n                                            parameters: {\n                                                path: `isEditing${record.name}`,\n                                                value: \"true\"\n                                            }\n                                        }\n                                    ]\n                                }\n                            },\n                            {\n                                component: {\n                                    _component: \"common/Default Button\",\n                                    contentText: `Delete ${record.name}`,\n                                    onClick: [\n                                        {\n                                            \"##eventHandlerType\": \"Delete Record\",\n                                            parameters: {\n                                                recordKey: {\n                                                    \"##bbstate\" : `selectedrow_${index.name}`,\n                                                    \"##source\": \"store\"\n                                                }\n                                            }\n                                        }\n                                    ]\n                                }\n                            }\n                        ]\n                    }\n                }\n            }\n        ]\n    }\n})","import { \n    recordHomepages, \n    homepageComponentName,\n    recordHomePageComponents\n} from \"./recordHomePageGenerator\";\nimport { formName, forms } from \"./formsGenerator\";\n\nexport const selectNavContent = ({indexes, records, helpers}) => \n    [\n        ...recordHomepages({indexes, records})\n            .map(component),\n\n        ...recordHomePageComponents({indexes, records, helpers}),\n\n        ...forms({indexes, records, helpers})\n\n    ]\n\n\nexport const navContentComponentName = record =>\n    `${record.name}/${record.name} Nav Content`;\n\nconst component = ({record, index}) => ({\n    inherits: \"@budibase/standard-components/if\",\n    description: `the component that gets displayed when the ${record.collectionName} nav is selected`,\n    name: navContentComponentName(record),\n    props: {\n        condition: `$store.isEditing${record.name}`,\n        thenComponent: {\n            _component: formName(record)\n        },\n        elseComponent: {\n            _component: homepageComponentName(record)\n        }\n    }\n});","import { navContentComponentName, selectNavContent } from \"./selectedNavContentGenerator\";\nimport { recordHomepages } from \"./recordHomePageGenerator\";\nexport const app = ({records, indexes, helpers}) => [\n    {\n        name: \"Application Root\",\n        inherits: \"@budibase/bootstrap-components/nav\",\n        props: {\n            items: recordHomepages({indexes, records})\n                    .map(navItem),\n            orientation: \"horizontal\",\n            alignment: \"start\",\n            fill: false,\n            pills: true,\n            selectedItem: {\n                \"##bbstate\":\"selectedNav\",\n                \"##bbstatefallback\":`${records[0].name}`,\n                \"##bbsource\": \"store\"\n            },\n            className: \"p-3\"\n        }\n    },\n    {\n        name: \"Login\",\n        inherits: \"@budibase/standard-components/login\",\n        props: {}\n    },\n    ...selectNavContent({records, indexes, helpers})\n]\n\n\nexport const navItem = ({record}) => ({\n    title: record.collectionName,\n    component : {\n        _component: navContentComponentName(record)\n    }\n})\n\n"],"names":["component"],"mappings":"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"} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"generators.js","sources":["../src/generators/buttonGenerators.js","../src/generators/formsGenerator.js","../src/generators/getRecordPath.js","../src/generators/indexTablesGenerator.js","../src/generators/recordHomePageGenerator.js","../src/generators/selectedNavContentGenerator.js","../src/generators/appGenerator.js"],"sourcesContent":["export const buttons = () => [\r\n    {\r\n        name: \"common/Primary Button\",\r\n        description: \"Bootstrap primary button \",\r\n        inherits: \"@budibase/standard-components/button\",\r\n        props: {\r\n            className: \"btn btn-primary\"\r\n        }\r\n    },\r\n    {\r\n        name: \"common/Default Button\",\r\n        description: \"Bootstrap default button\",\r\n        inherits: \"@budibase/standard-components/button\",\r\n        props: {\r\n            className: \"btn btn-secondary\"\r\n        }\r\n    }\r\n]","import {buttons} from \"./buttonGenerators\";\r\n\r\nexport const forms = ({records, indexes, helpers}) => \r\n    [\r\n        ...records.map(root),\r\n        ...buttons({records, indexes, helpers})\r\n    ];\r\n\r\nexport const formName = record =>  `${record.name}/${record.name} Form`;\r\n\r\nconst root = record => ({\r\n    name: formName(record),\r\n    description: `Control for creating/updating '${record.nodeKey()}' `,\r\n    inherits: \"@budibase/standard-components/div\",\r\n    props: {\r\n        className:\"p-1\",\r\n        children: [\r\n            {\r\n                component: {\r\n                    _component: \"@budibase/standard-components/h3\",\r\n                    text: `Edit ${record.name}`,\r\n                }\r\n            },\r\n            form(record),\r\n            saveCancelButtons(record)\r\n        ]\r\n    }\r\n}) \r\n\r\nconst form = record => ({\r\n    component: {\r\n        _component: \"@budibase/standard-components/form\",\r\n        formControls: \r\n            record.fields.map(f => formControl(record, f))\r\n    }\r\n})\r\n\r\nconst formControl = (record, field) => {\r\n    if(field.type === \"string\" && field.typeOptions.values && field.typeOptions.values.length > 0) {\r\n        return ({\r\n            control: {\r\n                _component: \"@budibase/standard-components/select\",\r\n                options: field.typeOptions.values.map(v => ({id:v, value:v})),\r\n                value: {\r\n                    \"##bbstate\":`${record.name}.${field.name}`,\r\n                    \"##bbsource\":\"store\"\r\n                },\r\n                className: \"form-control\"\r\n            },\r\n            label: field.label\r\n        });\r\n    } else {\r\n        return ({\r\n            control: {\r\n                _component: \"@budibase/standard-components/input\",\r\n                value: {\r\n                    \"##bbstate\":`${record.name}.${field.name}`,\r\n                    \"##bbsource\":\"store\"\r\n                },\r\n                className: \"form-control\",\r\n                type: field.type === \"string\" ? \"text\"\r\n                    : field.type === \"datetime\" ? \"date\"\r\n                    : field.type === \"number\" ? \"number\"\r\n                    : \"text\"\r\n            },\r\n            label: field.label\r\n        });\r\n    }\r\n}\r\n\r\nconst saveCancelButtons = (record) => ({\r\n    component: {\r\n        _component: \"@budibase/standard-components/stackpanel\",\r\n        direction: \"horizontal\",\r\n        children: [\r\n            paddedPanelForButton({\r\n                _component: \"common/Primary Button\",\r\n                contentText: `Save ${record.name}`,\r\n                onClick: [                \r\n                    {\r\n                        \"##eventHandlerType\": \"Save Record\",\r\n                        parameters: {\r\n                            statePath: `${record.name}`,\r\n                        }\r\n                    },\r\n                    {\r\n                        \"##eventHandlerType\": \"Set State\",\r\n                        parameters: {\r\n                            path: `isEditing${record.name}`,\r\n                            value: \"\"\r\n                        }\r\n                    }\r\n                ]\r\n            }),\r\n            paddedPanelForButton({\r\n                _component: \"common/Default Button\",\r\n                contentText: `Cancel`,\r\n                onClick: [\r\n                    {\r\n                        \"##eventHandlerType\": \"Set State\",\r\n                        parameters: {\r\n                            path: `isEditing${record.name}`,\r\n                            value: \"\"\r\n                        }\r\n                    }\r\n                ]\r\n            })\r\n        ]\r\n    }\r\n})\r\n\r\nconst paddedPanelForButton = (button) => ({\r\n    control: {\r\n        _component: \"@budibase/standard-components/div\",\r\n        className: \"btn-group\",\r\n        children: [\r\n            {\r\n                component: button\r\n            }\r\n        ]\r\n    }\r\n});\r\n\r\n","export const getRecordPath = (record) => {\r\n\r\n    const parts = [];\r\n\r\n    const add = (current) => {\r\n        parts.push(current.name);\r\n        if(current.parent().type === \"root\") {\r\n            return;\r\n        }\r\n\r\n        add(current.parent());\r\n    }\r\n\r\n    return parts.reverse().join(\"/\");\r\n}","import { getRecordPath } from \"./getRecordPath\";\r\n\r\nexport const indexTables = ({indexes, helpers}) => \r\n    indexes.map(i => indexTable(i, helpers));\r\n\r\nconst excludedColumns = [\"id\", \"isNew\", \"key\", \"type\", \"sortKey\"];\r\n\r\nexport const indexTableProps = (index, helpers) => ({\r\n    data: {\r\n        \"##bbstate\":index.nodeKey(),\r\n        \"##bbsource\":\"store\"\r\n    },\r\n    tableClass: \"table table-hover\",\r\n    theadClass: \"thead-dark\",\r\n    columns: helpers\r\n                .indexSchema(index)\r\n                .filter(c => !excludedColumns.includes(c.name))\r\n                .map(column),\r\n    onRowClick: [\r\n        {\r\n            \"##eventHandlerType\": \"Set State\",\r\n            parameters: {\r\n                path: `selectedrow_${index.name}`,\r\n                value: {\r\n                    \"##bbstate\": \"key\",\r\n                    \"##bbsource\": \"event\"\r\n                }\r\n            },\r\n        }\r\n    ]\r\n});\r\n\r\nexport const getIndexTableName = (index, record) => {\r\n    record = record \r\n             || index.parent().type === \"record\" ? index.parent() : null;\r\n    \r\n    return (record\r\n            ? `${getRecordPath(record)}/${index.name} Table`\r\n            : `${index.name} Table`);\r\n}\r\n\r\nconst indexTable = (index, helpers) => ({\r\n    name: getIndexTableName(index),\r\n    inherits: \"@budibase/standard-components/table\",\r\n    props: indexTableProps(index, helpers)\r\n});\r\n\r\nconst column = (col) => ({\r\n    title: col.name,\r\n    value: {\r\n        \"##bbstate\": col.name,\r\n        \"##bbsource\":\"context\"\r\n    }\r\n})","import {\r\n    getIndexTableName, indexTables\r\n} from \"./indexTablesGenerator\";\r\n\r\nimport {\r\n    buttons\r\n} from \"./buttonGenerators\";\r\n\r\nexport const recordHomePageComponents = ({indexes, records, helpers}) => \r\n    [   \r\n        ...recordHomepages({indexes, records})\r\n          .map(component),\r\n\r\n        ...recordHomepages({indexes, records})\r\n            .map(homePageButtons),\r\n        \r\n        ...indexTables({indexes, records, helpers}),\r\n\r\n        ...buttons({indexes, buttons, helpers})\r\n    ]\r\n\r\n\r\nconst findIndexForRecord = (indexes, record) => {\r\n    const forRecord = indexes.filter(i => i.allowedRecordNodeIds.includes(record.nodeId));\r\n    if(forRecord.length === 0) return;\r\n    if(forRecord.length === 1) return forRecord[0];\r\n    const noMap = forRecord.filter(i => !i.filter || !i.filter.trim());\r\n    if(noMap.length === 0) forRecord[0];\r\n    return noMap[0];\r\n}\r\n\r\nexport const recordHomepages = ({indexes, records}) => \r\n    records.filter(r => r.parent().type === \"root\")\r\n        .map(r =>({\r\n            record:r, \r\n            index:findIndexForRecord(indexes, r)\r\n        }))\r\n        .filter(r => r.index);\r\n\r\n\r\nexport const  homepageComponentName = (record) => \r\n    `${record.name}/${record.name} homepage`;\r\n\r\nconst component = ({record, index}) => ({\r\n    inherits: \"@budibase/standard-components/div\",\r\n    name: homepageComponentName(record),\r\n    props: {\r\n        className: \"d-flex flex-column h-100\",\r\n        children: [\r\n            {\r\n                component: {\r\n                    _component: `${record.name}/homepage buttons`,\r\n                }\r\n            },\r\n            {\r\n                component: {\r\n                    _component: getIndexTableName(index)\r\n                },\r\n                className: \"flex-gow-1 overflow-auto\"\r\n            }\r\n        ],\r\n        onLoad: [\r\n            {\r\n                \"##eventHandlerType\": \"Set State\",\r\n                parameters: {\r\n                    path: `isEditing${record.name}`,\r\n                    value: \"\"\r\n                }\r\n            },\r\n            {\r\n                \"##eventHandlerType\": \"List Records\",\r\n                parameters: {\r\n                    statePath: index.nodeKey(),\r\n                    indexKey: index.nodeKey()\r\n                }\r\n            }\r\n        ]\r\n    }\r\n\r\n});\r\n\r\nconst homePageButtons = ({index, record}) => ({\r\n    inherits: \"@budibase/standard-components/div\",\r\n    name: `${record.name}/homepage buttons`,\r\n    props: {\r\n        className: \"btn-toolbar mt-4 mb-2\",\r\n        children: [\r\n            {\r\n                component: {\r\n                    _component: \"@budibase/standard-components/div\",\r\n                    className: \"btn-group mr-3\",\r\n                    children: [\r\n                        {\r\n                            component: {\r\n                                _component: \"common/Default Button\",\r\n                                contentText: `Create ${record.name}`,\r\n                                onClick: [\r\n                                    {\r\n                                        \"##eventHandlerType\": \"Get New Record\",\r\n                                        parameters: {\r\n                                            statePath: record.name,\r\n                                            collectionKey: `/${record.collectionName}`,\r\n                                            childRecordType: record.name\r\n                                        }\r\n                                    }, \r\n                                    {\r\n                                        \"##eventHandlerType\": \"Set State\",\r\n                                        parameters: {\r\n                                            path: `isEditing${record.name}`,\r\n                                            value: \"true\"\r\n                                        }\r\n                                    }\r\n                                ]\r\n                            }\r\n                        },\r\n                        {\r\n                            component: {\r\n                                _component: \"common/Default Button\",\r\n                                contentText: `Refresh`,\r\n                                onClick: [\r\n                                    {\r\n                                        \"##eventHandlerType\": \"List Records\",\r\n                                        parameters: {\r\n                                            statePath: index.nodeKey(),\r\n                                            indexKey: index.nodeKey()\r\n                                        }\r\n                                    }\r\n                                ]\r\n                            }\r\n                        }\r\n                    ]\r\n                }\r\n            },\r\n            {\r\n                component: {\r\n                    _component: \"@budibase/standard-components/if\",\r\n                    condition: `$store.selectedrow_${index.name} && $store.selectedrow_${index.name}.length > 0`,\r\n                    thenComponent: {\r\n                        _component: \"@budibase/standard-components/div\",\r\n                        className: \"btn-group\",\r\n                        children: [\r\n                            {\r\n                                component: {\r\n                                    _component: \"common/Default Button\",\r\n                                    contentText: `Edit ${record.name}`,\r\n                                    onClick: [\r\n                                        {\r\n                                            \"##eventHandlerType\": \"Load Record\",\r\n                                            parameters: {\r\n                                                statePath: record.name,\r\n                                                recordKey: {\r\n                                                    \"##bbstate\" : `selectedrow_${index.name}`,\r\n                                                    \"##source\": \"store\"\r\n                                                }\r\n                                            }\r\n                                        }, \r\n                                        {\r\n                                            \"##eventHandlerType\": \"Set State\",\r\n                                            parameters: {\r\n                                                path: `isEditing${record.name}`,\r\n                                                value: \"true\"\r\n                                            }\r\n                                        }\r\n                                    ]\r\n                                }\r\n                            },\r\n                            {\r\n                                component: {\r\n                                    _component: \"common/Default Button\",\r\n                                    contentText: `Delete ${record.name}`,\r\n                                    onClick: [\r\n                                        {\r\n                                            \"##eventHandlerType\": \"Delete Record\",\r\n                                            parameters: {\r\n                                                recordKey: {\r\n                                                    \"##bbstate\" : `selectedrow_${index.name}`,\r\n                                                    \"##source\": \"store\"\r\n                                                }\r\n                                            }\r\n                                        }\r\n                                    ]\r\n                                }\r\n                            }\r\n                        ]\r\n                    }\r\n                }\r\n            }\r\n        ]\r\n    }\r\n})","import { \r\n    recordHomepages, \r\n    homepageComponentName,\r\n    recordHomePageComponents\r\n} from \"./recordHomePageGenerator\";\r\nimport { formName, forms } from \"./formsGenerator\";\r\n\r\nexport const selectNavContent = ({indexes, records, helpers}) => \r\n    [\r\n        ...recordHomepages({indexes, records})\r\n            .map(component),\r\n\r\n        ...recordHomePageComponents({indexes, records, helpers}),\r\n\r\n        ...forms({indexes, records, helpers})\r\n\r\n    ]\r\n\r\n\r\nexport const navContentComponentName = record =>\r\n    `${record.name}/${record.name} Nav Content`;\r\n\r\nconst component = ({record, index}) => ({\r\n    inherits: \"@budibase/standard-components/if\",\r\n    description: `the component that gets displayed when the ${record.collectionName} nav is selected`,\r\n    name: navContentComponentName(record),\r\n    props: {\r\n        condition: `$store.isEditing${record.name}`,\r\n        thenComponent: {\r\n            _component: formName(record)\r\n        },\r\n        elseComponent: {\r\n            _component: homepageComponentName(record)\r\n        }\r\n    }\r\n});","import { navContentComponentName, selectNavContent } from \"./selectedNavContentGenerator\";\r\nimport { recordHomepages } from \"./recordHomePageGenerator\";\r\nexport const app = ({records, indexes, helpers}) => [\r\n    {\r\n        name: \"Application Root\",\r\n        inherits: \"@budibase/bootstrap-components/nav\",\r\n        props: {\r\n            items: recordHomepages({indexes, records})\r\n                    .map(navItem),\r\n            orientation: \"horizontal\",\r\n            alignment: \"start\",\r\n            fill: false,\r\n            pills: true,\r\n            selectedItem: {\r\n                \"##bbstate\":\"selectedNav\",\r\n                \"##bbstatefallback\":`${records[0].name}`,\r\n                \"##bbsource\": \"store\"\r\n            },\r\n            className: \"p-3\"\r\n        }\r\n    },\r\n    {\r\n        name: \"Login\",\r\n        inherits: \"@budibase/standard-components/login\",\r\n        props: {}\r\n    },\r\n    ...selectNavContent({records, indexes, helpers})\r\n]\r\n\r\n\r\nexport const navItem = ({record}) => ({\r\n    title: record.collectionName,\r\n    component : {\r\n        _component: navContentComponentName(record)\r\n    }\r\n})\r\n\r\n"],"names":["component"],"mappings":"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"} diff --git a/packages/builder/package.json b/packages/builder/package.json index 9595da46f6..904a609ca6 100644 --- a/packages/builder/package.json +++ b/packages/builder/package.json @@ -36,6 +36,7 @@ "dependencies": { "@budibase/client": "^0.0.16", "@nx-js/compiler-util": "^2.0.0", + "codemirror": "^5.51.0", "date-fns": "^1.29.0", "feather-icons": "^4.21.0", "flatpickr": "^4.5.7", diff --git a/packages/builder/src/builderStore/buildCodeForScreens.js b/packages/builder/src/builderStore/buildCodeForScreens.js new file mode 100644 index 0000000000..fdd2c8d797 --- /dev/null +++ b/packages/builder/src/builderStore/buildCodeForScreens.js @@ -0,0 +1,35 @@ + + +const buildCodeForSingleScreen = (screen) => { + let code = ""; + const walkProps = (props) => { + if(props._code && props._code.trim().length > 0) { + code += buildComponentCode(props) + } + + if(!props._children) return; + + for(let child of props._children) { + walkProps(child); + } + } + + walkProps(screen.props); + + return code; +} + +export const buildCodeForScreens = screens => { + let allfunctions = ""; + for(let screen of screens) { + allfunctions += buildCodeForSingleScreen(screen); + } + + return (`return ({ ${allfunctions} });`); +} + +const buildComponentCode = (componentProps) => +`"${componentProps._id}" : (render, context) => { +${componentProps._code} +}, +`; \ No newline at end of file diff --git a/packages/builder/src/builderStore/store.js b/packages/builder/src/builderStore/store.js index c9dc3da87b..7e52a0e881 100644 --- a/packages/builder/src/builderStore/store.js +++ b/packages/builder/src/builderStore/store.js @@ -22,6 +22,7 @@ import { import { loadLibs, loadLibUrls, loadGeneratorLibs } from "./loadComponentLibraries"; +import { buildCodeForScreens } from "./buildCodeForScreens"; import { uuid } from './uuid'; import { generate_screen_css } from './generate_css'; @@ -97,6 +98,7 @@ export const getStore = () => { store.selectComponent = selectComponent(store); store.setComponentProp = setComponentProp(store); store.setComponentStyle = setComponentStyle(store); + store.setComponentCode = setComponentCode(store); return store; } @@ -669,7 +671,8 @@ const savePackage = (store, s) => { s.components, s.screens, s.pages.unauthenticated.appBody) - } + }, + uiFunctions: buildCodeForScreens(s.screens) }; const data = { @@ -687,6 +690,7 @@ const setCurrentScreen = store => screenName => { s.currentFrontEndItem = screen; s.currentFrontEndType = "screen"; s.currentComponentInfo = getScreenInfo(s.components, screen); + setCurrentScreenFunctions(s); return s; }) } @@ -695,8 +699,9 @@ const setCurrentPage = store => pageName => { store.update(s => { s.currentFrontEndType = "page"; s.currentPageName = pageName; + setCurrentScreenFunctions(s); return s; - }) + }); } const addChildComponent = store => component => { @@ -770,3 +775,22 @@ const setComponentStyle = store => (type, name, value) => { return s; }) } + +const setComponentCode = store => (code) => { + store.update(s => { + s.currentComponentInfo._code = code; + + setCurrentScreenFunctions(s); + // save without messing with the store + _save(s.appname, s.currentFrontEndItem, store, s) + + return s; + }) +} + +const setCurrentScreenFunctions = (s) => { + s.currentScreenFunctions = + s.currentFrontEndItem === "screen" + ? buildCodeForScreens([s.currentFrontEndItem]) + : "({});"; +} diff --git a/packages/builder/src/builderStore/store.js.orig b/packages/builder/src/builderStore/store.js.orig new file mode 100644 index 0000000000..9c753a180e --- /dev/null +++ b/packages/builder/src/builderStore/store.js.orig @@ -0,0 +1,799 @@ +import { + hierarchy as hierarchyFunctions, +} from "../../../core/src"; +import { + filter, cloneDeep, sortBy, + map, last, keys, concat, keyBy, + find, isEmpty, values, +} from "lodash/fp"; +import { + pipe, getNode, validate, + constructHierarchy, templateApi +} from "../common/core"; +import { writable } from "svelte/store"; +import { defaultPagesObject } from "../userInterface/pagesParsing/defaultPagesObject" +import { buildPropsHierarchy } from "../userInterface/pagesParsing/buildPropsHierarchy" +import api from "./api"; +import { isRootComponent, getExactComponent } from "../userInterface/pagesParsing/searchComponents"; +import { rename } from "../userInterface/pagesParsing/renameScreen"; +import { + getNewComponentInfo, getScreenInfo, +} from "../userInterface/pagesParsing/createProps"; +import { + loadLibs, loadLibUrls, loadGeneratorLibs +} from "./loadComponentLibraries"; +<<<<<<< HEAD +import { buildCodeForScreens } from "./buildCodeForScreens"; +======= +import { uuid } from './uuid'; +import { generate_screen_css } from './generate_css'; +>>>>>>> master + +let appname = ""; + +export const getStore = () => { + + const initial = { + apps: [], + appname: "", + hierarchy: {}, + actions: [], + triggers: [], + pages: defaultPagesObject(), + mainUi: {}, + unauthenticatedUi: {}, + components: [], + currentFrontEndItem: null, + currentComponentInfo: null, + currentFrontEndType: "none", + currentPageName: "", + currentComponentProps: null, + currentNodeIsNew: false, + errors: [], + activeNav: "database", + isBackend: true, + hasAppPackage: false, + accessLevels: { version: 0, levels: [] }, + currentNode: null, + libraries: null, + showSettings: false, + useAnalytics: true, + }; + + const store = writable(initial); + + store.initialise = initialise(store, initial); + store.newChildRecord = newRecord(store, false); + store.newRootRecord = newRecord(store, true); + store.selectExistingNode = selectExistingNode(store); + store.newChildIndex = newIndex(store, false); + store.newRootIndex = newIndex(store, true); + store.saveCurrentNode = saveCurrentNode(store); + store.importAppDefinition = importAppDefinition(store); + store.deleteCurrentNode = deleteCurrentNode(store); + store.saveField = saveField(store); + store.deleteField = deleteField(store); + store.saveAction = saveAction(store); + store.deleteAction = deleteAction(store); + store.saveTrigger = saveTrigger(store); + store.deleteTrigger = deleteTrigger(store); + store.saveLevel = saveLevel(store); + store.deleteLevel = deleteLevel(store); + store.setActiveNav = setActiveNav(store); + store.saveScreen = saveScreen(store); + store.refreshComponents = refreshComponents(store); + store.addComponentLibrary = addComponentLibrary(store); + store.renameScreen = renameScreen(store); + store.deleteScreen = deleteScreen(store); + store.setCurrentScreen = setCurrentScreen(store); + store.setCurrentPage = setCurrentPage(store); + store.createScreen = createScreen(store); + store.removeComponentLibrary = removeComponentLibrary(store); + store.addStylesheet = addStylesheet(store); + store.removeStylesheet = removeStylesheet(store); + store.savePage = savePage(store); + store.showFrontend = showFrontend(store); + store.showBackend = showBackend(store); + store.showSettings = showSettings(store); + store.useAnalytics = useAnalytics(store); + store.createGeneratedComponents = createGeneratedComponents(store); + store.addChildComponent = addChildComponent(store); + store.selectComponent = selectComponent(store); + store.setComponentProp = setComponentProp(store); + store.setComponentStyle = setComponentStyle(store); + store.setComponentCode = setComponentCode(store); + return store; +} + +export default getStore; + +const initialise = (store, initial) => async () => { + + appname = window.location.hash + ? last(window.location.hash.substr(1).split("/")) + : ""; + + if (!appname) { + initial.apps = await api.get(`/_builder/api/apps`).then(r => r.json()); + initial.hasAppPackage = false; + store.set(initial); + return initial; + } + + const pkg = await api.get(`/_builder/api/${appname}/appPackage`) + .then(r => r.json()); + + initial.libraries = await loadLibs(appname, pkg); + initial.generatorLibraries = await loadGeneratorLibs(appname, pkg); + initial.loadLibraryUrls = () => loadLibUrls(appname, pkg); + initial.appname = appname; + initial.pages = pkg.pages; + initial.hasAppPackage = true; + initial.hierarchy = pkg.appDefinition.hierarchy; + initial.accessLevels = pkg.accessLevels; + initial.screens = values(pkg.screens); + initial.generators = generatorsArray(pkg.components.generators); + initial.components = values(pkg.components.components); + initial.actions = values(pkg.appDefinition.actions); + initial.triggers = pkg.appDefinition.triggers; + + if (!!initial.hierarchy && !isEmpty(initial.hierarchy)) { + initial.hierarchy = constructHierarchy(initial.hierarchy); + const shadowHierarchy = createShadowHierarchy(initial.hierarchy); + if (initial.currentNode !== null) + initial.currentNode = getNode( + shadowHierarchy, initial.currentNode.nodeId + ); + } + + store.set(initial); + return initial; +} + +const generatorsArray = generators => + pipe(generators, [ + keys, + filter(k => k !== "_lib"), + map(k => generators[k]) + ]); + + +const showSettings = store => show => { + store.update(s => { + s.showSettings = !s.showSettings; + return s; + }); +} + +const useAnalytics = store => useAnalytics => { + store.update(s => { + s.useAnalytics = !s.useAnalytics; + return s; + }); +} + +const showBackend = store => () => { + store.update(s => { + s.isBackend = true; + return s; + }) +} + +const showFrontend = store => () => { + store.update(s => { + s.isBackend = false; + return s; + }) +} + +const newRecord = (store, useRoot) => () => { + store.update(s => { + s.currentNodeIsNew = true; + const shadowHierarchy = createShadowHierarchy(s.hierarchy); + parent = useRoot ? shadowHierarchy + : getNode( + shadowHierarchy, + s.currentNode.nodeId); + s.errors = []; + s.currentNode = templateApi(shadowHierarchy) + .getNewRecordTemplate(parent, "", true); + return s; + }); +} + + +const selectExistingNode = (store) => (nodeId) => { + store.update(s => { + const shadowHierarchy = createShadowHierarchy(s.hierarchy); + s.currentNode = getNode( + shadowHierarchy, nodeId + ); + s.currentNodeIsNew = false; + s.errors = []; + s.activeNav = "database"; + return s; + }) +} + +const newIndex = (store, useRoot) => () => { + store.update(s => { + s.currentNodeIsNew = true; + s.errors = []; + const shadowHierarchy = createShadowHierarchy(s.hierarchy); + parent = useRoot ? shadowHierarchy + : getNode( + shadowHierarchy, + s.currentNode.nodeId); + + s.currentNode = templateApi(shadowHierarchy) + .getNewIndexTemplate(parent); + return s; + }); +} + +const saveCurrentNode = (store) => () => { + store.update(s => { + + const errors = validate.node(s.currentNode); + s.errors = errors; + if (errors.length > 0) { + return s; + } + + const parentNode = getNode( + s.hierarchy, s.currentNode.parent().nodeId); + + const existingNode = getNode( + s.hierarchy, s.currentNode.nodeId); + + let index = parentNode.children.length; + if (!!existingNode) { + // remove existing + index = existingNode.parent().children.indexOf(existingNode); + existingNode.parent().children = pipe(existingNode.parent().children, [ + filter(c => c.nodeId !== existingNode.nodeId) + ]); + } + + // should add node into existing hierarchy + const cloned = cloneDeep(s.currentNode); + templateApi(s.hierarchy).constructNode( + parentNode, + cloned + ); + + const newIndexOfchild = child => { + if (child === cloned) return index; + const currentIndex = parentNode.children.indexOf(child); + return currentIndex >= index ? currentIndex + 1 : currentIndex; + } + + parentNode.children = pipe(parentNode.children, [ + sortBy(newIndexOfchild) + ]); + + if (!existingNode && s.currentNode.type === "record") { + const defaultIndex = templateApi(s.hierarchy) + .getNewIndexTemplate(cloned.parent()); + defaultIndex.name = `all_${cloned.collectionName}`; + defaultIndex.allowedRecordNodeIds = [cloned.nodeId]; + } + + s.currentNodeIsNew = false; + + savePackage(store, s); + + return s; + }); +} + +const importAppDefinition = store => appDefinition => { + store.update(s => { + s.hierarchy = appDefinition.hierarchy; + s.currentNode = appDefinition.hierarchy.children.length > 0 + ? appDefinition.hierarchy.children[0] + : null; + s.actions = appDefinition.actions; + s.triggers = appDefinition.triggers; + s.currentNodeIsNew = false; + return s; + }); +} + +const deleteCurrentNode = store => () => { + store.update(s => { + const nodeToDelete = getNode(s.hierarchy, s.currentNode.nodeId); + s.currentNode = hierarchyFunctions.isRoot(nodeToDelete.parent()) + ? find(n => n != s.currentNode) + (s.hierarchy.children) + : nodeToDelete.parent(); + if (hierarchyFunctions.isRecord(nodeToDelete)) { + nodeToDelete.parent().children = filter(c => c.nodeId !== nodeToDelete.nodeId) + (nodeToDelete.parent().children); + } else { + nodeToDelete.parent().indexes = filter(c => c.nodeId !== nodeToDelete.nodeId) + (nodeToDelete.parent().indexes); + } + s.errors = []; + savePackage(store, s); + return s; + }); +} + +const saveField = databaseStore => (field) => { + databaseStore.update(db => { + db.currentNode.fields = filter(f => f.name !== field.name) + (db.currentNode.fields); + + templateApi(db.hierarchy).addField(db.currentNode, field); + return db; + }); +} + + +const deleteField = databaseStore => field => { + databaseStore.update(db => { + db.currentNode.fields = filter(f => f.name !== field.name) + (db.currentNode.fields); + + return db; + }); +} + + +const saveAction = store => (newAction, isNew, oldAction = null) => { + store.update(s => { + + const existingAction = isNew + ? null + : find(a => a.name === oldAction.name)(s.actions); + + if (existingAction) { + s.actions = pipe(s.actions, [ + map(a => a === existingAction ? newAction : a) + ]); + } else { + s.actions.push(newAction); + } + savePackage(store, s); + return s; + }); +} + +const deleteAction = store => action => { + store.update(s => { + s.actions = filter(a => a.name !== action.name)(s.actions); + savePackage(store, s); + return s; + }); +} + +const saveTrigger = store => (newTrigger, isNew, oldTrigger = null) => { + store.update(s => { + + const existingTrigger = isNew + ? null + : find(a => a.name === oldTrigger.name)(s.triggers); + + if (existingTrigger) { + s.triggers = pipe(s.triggers, [ + map(a => a === existingTrigger ? newTrigger : a) + ]); + } else { + s.triggers.push(newTrigger); + } + savePackage(store, s); + return s; + }); +} + +const deleteTrigger = store => trigger => { + store.update(s => { + s.triggers = filter(t => t.name !== trigger.name)(s.triggers); + return s; + }); +} + +const incrementAccessLevelsVersion = (s) => + s.accessLevels.version = (s.accessLevels.version || 0) + 1; + +const saveLevel = store => (newLevel, isNew, oldLevel = null) => { + store.update(s => { + + const levels = s.accessLevels.levels; + + const existingLevel = isNew + ? null + : find(a => a.name === oldLevel.name)(levels); + + if (existingLevel) { + s.accessLevels.levels = pipe(levels, [ + map(a => a === existingLevel ? newLevel : a) + ]); + } else { + s.accessLevels.levels.push(newLevel); + } + + incrementAccessLevelsVersion(s); + + savePackage(store, s); + return s; + }); +} + +const deleteLevel = store => level => { + store.update(s => { + s.accessLevels.levels = filter(t => t.name !== level.name)(s.accessLevels.levels); + incrementAccessLevelsVersion(s); + savePackage(store, s); + return s; + }); +} + +const setActiveNav = store => navName => { + store.update(s => { + s.activeNav = navName; + return s; + }); +} + +const createShadowHierarchy = hierarchy => + constructHierarchy(JSON.parse(JSON.stringify(hierarchy))); + +const saveScreen = store => (screen) => { + store.update(s => { + return _saveScreen(store, s, screen); + }) +}; + +const _saveScreen = (store, s, screen) => { + const screens = pipe(s.screens, [ + filter(c => c.name !== screen.name), + concat([screen]) + ]); + + s.screens = screens; + s.currentFrontEndItem = screen; + s.currentComponentInfo = getScreenInfo( + s.components, screen); + + api.post(`/_builder/api/${s.appname}/screen`, screen) + .then(() => savePackage(store, s)); + + return s; +} + +const _save = (appname, screen, store, s) => + api.post(`/_builder/api/${appname}/screen`, screen) + .then(() => savePackage(store, s)); + +const createScreen = store => (screenName, layoutComponentName) => { + store.update(s => { + const newComponentInfo = getNewComponentInfo( + s.components, layoutComponentName, screenName); + + s.currentFrontEndItem = newComponentInfo.component; + s.currentComponentInfo = newComponentInfo; + s.currentFrontEndType = "screen"; + + return _saveScreen(store, s, newComponentInfo.component); + }); +}; + +const createGeneratedComponents = store => components => { + store.update(s => { + s.components = [...s.components, ...components]; + s.screens = [...s.screens, ...components]; + + const doCreate = async () => { + for (let c of components) { + await api.post(`/_builder/api/${s.appname}/screen`, c); + } + + await savePackage(store, s); + } + + doCreate(); + + return s; + }); +}; + +const deleteScreen = store => name => { + store.update(s => { + + const components = pipe(s.components, [ + filter(c => c.name !== name) + ]); + + const screens = pipe(s.screens, [ + filter(c => c.name !== name) + ]); + + s.components = components; + s.screens = screens; + if (s.currentFrontEndItem.name === name) { + s.currentFrontEndItem = null; + s.currentFrontEndType = ""; + } + + api.delete(`/_builder/api/${s.appname}/screen/${name}`); + + return s; + }) +} + +const renameScreen = store => (oldname, newname) => { + store.update(s => { + + const { + screens, pages, error, changedScreens + } = rename(s.pages, s.screens, oldname, newname); + + if (error) { + // should really do something with this + return s; + } + + s.screens = screens; + s.pages = pages; + if (s.currentFrontEndItem.name === oldname) + s.currentFrontEndItem.name = newname; + + const saveAllChanged = async () => { + for (let screenName of changedScreens) { + const changedScreen + = getExactComponent(screens, screenName); + await api.post(`/_builder/api/${s.appname}/screen`, changedScreen); + } + } + + api.patch(`/_builder/api/${s.appname}/screen`, { + oldname, newname + }) + .then(() => saveAllChanged()) + .then(() => { + savePackage(store, s); + }); + + return s; + }) +} + +const savePage = store => async page => { + store.update(s => { + if (s.currentFrontEndType !== "page" || !s.currentPageName) { + return s; + } + + s.pages[s.currentPageName] = page; + savePackage(store, s); + return s; + }); +} + +const addComponentLibrary = store => async lib => { + + const response = + await api.get(`/_builder/api/${appname}/componentlibrary?lib=${encodeURI(lib)}`, undefined, false); + + const success = response.status === 200; + + const error = response.status === 404 + ? `Could not find library ${lib}` + : success + ? "" + : response.statusText; + + const components = success + ? await response.json() + : []; + + store.update(s => { + if (success) { + + const componentsArray = []; + for (let c in components) { + componentsArray.push(components[c]); + } + + s.components = pipe(s.components, [ + filter(c => !c.name.startsWith(`${lib}/`)), + concat(componentsArray) + ]); + + s.pages.componentLibraries.push(lib); + savePackage(store, s); + } + + return s; + }) +} + +const removeComponentLibrary = store => lib => { + store.update(s => { + + + s.pages.componentLibraries = filter(l => l !== lib)( + s.pages.componentLibraries); + savePackage(store, s); + + + return s; + }) +} + +const addStylesheet = store => stylesheet => { + store.update(s => { + s.pages.stylesheets.push(stylesheet); + savePackage(store, s); + return s; + }) +} + +const removeStylesheet = store => stylesheet => { + store.update(s => { + s.pages.stylesheets = filter(s => s !== stylesheet)(s.pages.stylesheets); + savePackage(store, s); + return s; + }); +} + +const refreshComponents = store => async () => { + + const componentsAndGenerators = + await api.get(`/_builder/api/${db.appname}/components`).then(r => r.json()); + + const components = pipe(componentsAndGenerators.components, [ + keys, + map(k => ({ ...componentsAndGenerators[k], name: k })) + ]); + + store.update(s => { + s.components = pipe(s.components, [ + filter(c => !isRootComponent(c)), + concat(components) + ]); + s.generators = componentsAndGenerators.generators; + return s; + }); +}; + +const savePackage = (store, s) => { + + const appDefinition = { + hierarchy: s.hierarchy, + triggers: s.triggers, + actions: keyBy("name")(s.actions), + props: { + main: buildPropsHierarchy( + s.components, + s.screens, + s.pages.main.appBody), + unauthenticated: buildPropsHierarchy( + s.components, + s.screens, + s.pages.unauthenticated.appBody) + }, + uiFunctions: buildCodeForScreens(s.screens) + }; + + const data = { + appDefinition, + accessLevels: s.accessLevels, + pages: s.pages, + } + + return api.post(`/_builder/api/${s.appname}/appPackage`, data); +} + +const setCurrentScreen = store => screenName => { + store.update(s => { + const screen = getExactComponent(s.screens, screenName); + s.currentFrontEndItem = screen; + s.currentFrontEndType = "screen"; + s.currentComponentInfo = getScreenInfo(s.components, screen); + setCurrentScreenFunctions(s); + return s; + }) +} + +const setCurrentPage = store => pageName => { + store.update(s => { + s.currentFrontEndType = "page"; + s.currentPageName = pageName; + setCurrentScreenFunctions(s); + return s; + }); +} + +const addChildComponent = store => component => { + + store.update(s => { + const newComponent = getNewComponentInfo( + s.components, component); + + let children = s.currentComponentInfo.component ? + s.currentComponentInfo.component.props._children : + s.currentComponentInfo._children; + + const component_definition = Object.assign( + cloneDeep(newComponent.fullProps), { + _component: component, + _styles: { position: {}, layout: {} }, + _id: uuid() + }) + + if (children) { + if (s.currentComponentInfo.component) { + s.currentComponentInfo.component.props._children = children.concat(component_definition); + } else { + s.currentComponentInfo._children = children.concat(component_definition) + } + } else { + if (s.currentComponentInfo.component) { + s.currentComponentInfo.component.props._children = [component_definition]; + } else { + s.currentComponentInfo._children = [component_definition] + } + } + + _saveScreen(store, s, s.currentFrontEndItem); + + _saveScreen(store, s, s.currentFrontEndItem); + + return s; + }) +} + +const selectComponent = store => component => { + store.update(s => { + s.currentComponentInfo = component; + return s; + }) + +} + +const setComponentProp = store => (name, value) => { + store.update(s => { + const current_component = s.currentComponentInfo; + s.currentComponentInfo[name] = value; + _saveScreen(store, s, s.currentFrontEndItem); + s.currentComponentInfo = current_component; + return s; + }) +} + +const setComponentStyle = store => (type, name, value) => { + store.update(s => { + if (!s.currentComponentInfo._styles) { + s.currentComponentInfo._styles = {}; + } + s.currentComponentInfo._styles[type][name] = value; + s.currentFrontEndItem._css = generate_screen_css(s.currentFrontEndItem.props._children) + + // save without messing with the store + _save(s.appname, s.currentFrontEndItem, store, s) + + return s; + }) +} + +const setComponentCode = store => (code) => { + store.update(s => { + s.currentComponentInfo._code = code; + + setCurrentScreenFunctions(s); + // save without messing with the store + _save(s.appname, s.currentFrontEndItem, store, s) + + return s; + }) +} + +const setCurrentScreenFunctions = (s) => { + s.currentScreenFunctions = + s.currentFrontEndItem === "screen" + ? buildCodeForScreens([s.currentFrontEndItem]) + : "({});"; +} diff --git a/packages/builder/src/common/Icons/CircleIndicator.svelte b/packages/builder/src/common/Icons/CircleIndicator.svelte new file mode 100644 index 0000000000..9c0fe034f4 --- /dev/null +++ b/packages/builder/src/common/Icons/CircleIndicator.svelte @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/packages/builder/src/common/Icons/index.js b/packages/builder/src/common/Icons/index.js index c00ebb3ce4..2aeb386bd7 100644 --- a/packages/builder/src/common/Icons/index.js +++ b/packages/builder/src/common/Icons/index.js @@ -4,5 +4,6 @@ export { default as TerminalIcon } from './Terminal.svelte'; export { default as InputIcon } from './Input.svelte'; export { default as ImageIcon } from './Image.svelte'; export { default as ArrowDownIcon } from './ArrowDown.svelte'; -export { default as EventsIcon } from './Events.svelte'; +export { default as CircleIndicator } from './CircleIndicator.svelte'; export { default as PencilIcon } from './Pencil.svelte'; +export { default as EventsIcon } from './Events.svelte'; \ No newline at end of file diff --git a/packages/builder/src/common/Icons/index.js.orig b/packages/builder/src/common/Icons/index.js.orig new file mode 100644 index 0000000000..69f195a161 --- /dev/null +++ b/packages/builder/src/common/Icons/index.js.orig @@ -0,0 +1,12 @@ +export { default as LayoutIcon } from './Layout.svelte'; +export { default as PaintIcon } from './Paint.svelte'; +export { default as TerminalIcon } from './Terminal.svelte'; +export { default as InputIcon } from './Input.svelte'; +export { default as ImageIcon } from './Image.svelte'; +export { default as ArrowDownIcon } from './ArrowDown.svelte'; +<<<<<<< HEAD +export { default as CircleIndicator } from './CircleIndicator.svelte'; +======= +export { default as EventsIcon } from './Events.svelte'; +export { default as PencilIcon } from './Pencil.svelte'; +>>>>>>> master diff --git a/packages/builder/src/index.html b/packages/builder/src/index.html index 184103a067..8ce642e160 100644 --- a/packages/builder/src/index.html +++ b/packages/builder/src/index.html @@ -8,6 +8,8 @@ + + diff --git a/packages/builder/src/main.js b/packages/builder/src/main.js index 723e771208..9c2a6b5066 100644 --- a/packages/builder/src/main.js +++ b/packages/builder/src/main.js @@ -10,6 +10,8 @@ import "/assets/budibase-logo.png"; import "/assets/budibase-logo-only.png"; import "uikit/dist/css/uikit.min.css"; import "uikit/dist/js/uikit.min.js"; +import "codemirror/lib/codemirror.css"; +import 'codemirror/theme/monokai.css'; const app = new App({ target: document.getElementById("app") diff --git a/packages/builder/src/userInterface/CodeEditor.svelte b/packages/builder/src/userInterface/CodeEditor.svelte index f4f2d7aa93..dbe87d8eff 100644 --- a/packages/builder/src/userInterface/CodeEditor.svelte +++ b/packages/builder/src/userInterface/CodeEditor.svelte @@ -1,47 +1,83 @@ -

Code

-

Use the code box below to add snippets of javascript to enhance your webapp

+
+
-
- +
+
+ {"}"} +
+
+
+ + + + + + diff --git a/packages/builder/src/userInterface/ComponentPanel.svelte b/packages/builder/src/userInterface/ComponentPanel.svelte index d4d982f2e2..1de496bb1a 100644 --- a/packages/builder/src/userInterface/ComponentPanel.svelte +++ b/packages/builder/src/userInterface/ComponentPanel.svelte @@ -2,12 +2,13 @@ import PropsView from "./PropsView.svelte"; import { store } from "../builderStore"; import IconButton from "../common/IconButton.svelte"; - import { LayoutIcon, PaintIcon, TerminalIcon, EventsIcon } from '../common/Icons/'; + import { LayoutIcon, PaintIcon, TerminalIcon, CircleIndicator, EventsIcon } from '../common/Icons/'; import CodeEditor from './CodeEditor.svelte'; import LayoutEditor from './LayoutEditor.svelte'; import EventsEditor from "./EventsEditor"; let current_view = 'props'; + let codeEditor; $: component = $store.currentComponentInfo; $: originalName = component.name; @@ -34,7 +35,12 @@
  • -
  • @@ -54,10 +60,13 @@ {:else if current_view === 'events'} - {:else} - {/if} + + {:else}

    This is a screen, this will be dealt with later

    @@ -113,6 +122,7 @@ li button { padding: 12px; outline: none; cursor: pointer; + position: relative; } .selected { @@ -120,4 +130,11 @@ li button { background: var(--background-button)!important; } +.button-indicator { + position: absolute; + top: 8px; + right: 10px; + color: var(--button-text); +} + diff --git a/packages/builder/src/userInterface/ComponentPanel.svelte.orig b/packages/builder/src/userInterface/ComponentPanel.svelte.orig new file mode 100644 index 0000000000..88464032da --- /dev/null +++ b/packages/builder/src/userInterface/ComponentPanel.svelte.orig @@ -0,0 +1,149 @@ + + +
    +
      +
    • + +
    • +
    • + +
    • +
    • + +
    • +
    • + +
    • +
    + + {#if !componentInfo.component} +
    + + {#if current_view === 'props'} + + {:else if current_view === 'layout'} + +<<<<<<< HEAD +======= + {:else if current_view === 'events'} + + {:else} + +>>>>>>> master + {/if} + + + +
    + {:else} +

    This is a screen, this will be dealt with later

    + {/if} + +
    + + + diff --git a/packages/builder/src/userInterface/CurrentItemPreview.svelte b/packages/builder/src/userInterface/CurrentItemPreview.svelte index bb586aa474..84420c7bdd 100644 --- a/packages/builder/src/userInterface/CurrentItemPreview.svelte +++ b/packages/builder/src/userInterface/CurrentItemPreview.svelte @@ -24,6 +24,7 @@ hierarchy: $store.hierarchy, appRootPath: "" }; + @@ -39,6 +40,8 @@ ${stylesheetLinks} + +
    + +
    + {#each propDefs as [prop_name, prop_value], index} + +
    + + + +
    + + {/each} + +
    + +
    + + + diff --git a/packages/builder/tests/buildCodeForScreen.spec.js b/packages/builder/tests/buildCodeForScreen.spec.js new file mode 100644 index 0000000000..4d6282ee81 --- /dev/null +++ b/packages/builder/tests/buildCodeForScreen.spec.js @@ -0,0 +1,64 @@ +import { buildCodeForScreens } from "../src/builderStore/buildCodeForScreens"; + +describe("buildCodeForScreen",() => { + + it("should package _code into runnable function, for simple screen props", () => { + const screen = { + props: { + _id: "1234", + _code: "render('render argument');" + } + } + + let renderArg; + const render = (arg) => { + renderArg = arg; + } + const uiFunctions = getFunctions(screen); + + const targetfunction = uiFunctions[screen.props._id]; + expect(targetfunction).toBeDefined(); + + targetfunction(render); + + expect(renderArg).toBe("render argument"); + + }); + + it("should package _code into runnable function, for _children ", () => { + const screen = { + props: { + _id: "parent", + _code: "render('parent argument');", + _children: [ + { + _id: "child1", + _code: "render('child 1 argument');" + }, + { + _id: "child2", + _code: "render('child 2 argument');" + } + ] + } + } + + let renderArg; + const render = (arg) => { + renderArg = arg; + } + const uiFunctions = getFunctions(screen); + + const targetfunction = uiFunctions["child2"]; + expect(targetfunction).toBeDefined(); + + targetfunction(render); + + expect(renderArg).toBe("child 2 argument"); + + }) + +}); + +const getFunctions = (screen) => + new Function(buildCodeForScreens([screen]))(); diff --git a/packages/client/src/index.js b/packages/client/src/index.js index 7114cde443..8da1657cf6 100644 --- a/packages/client/src/index.js +++ b/packages/client/src/index.js @@ -6,6 +6,8 @@ export const loadBudibase = async ({ window, localStorage, uiFunctions }) => { const appDefinition = window["##BUDIBASE_APPDEFINITION##"]; + const uiFunctionsFromWindow = window["##BUDIBASE_APPDEFINITION##"]; + uiFunctions = uiFunctionsFromWindow || uiFunctions; const userFromStorage = localStorage.getItem("budibase:user") diff --git a/packages/client/src/render/initialiseChildren.js b/packages/client/src/render/initialiseChildren.js index a466b9dcaf..3c6cec74ae 100644 --- a/packages/client/src/render/initialiseChildren.js +++ b/packages/client/src/render/initialiseChildren.js @@ -48,7 +48,7 @@ export const _initialiseChildren = (initialiseOpts) => parentNode: treeNode, componentConstructor,uiFunctions, htmlElement, anchor, initialProps, - bb, document}); + bb}); for(let comp of renderedComponentsThisIteration) { comp.unsubscribe = bind(comp.component); diff --git a/packages/client/src/render/renderComponent.js b/packages/client/src/render/renderComponent.js index ea8e2e281d..ea91045546 100644 --- a/packages/client/src/render/renderComponent.js +++ b/packages/client/src/render/renderComponent.js @@ -2,7 +2,7 @@ export const renderComponent = ({ componentConstructor, uiFunctions, htmlElement, anchor, props, - initialProps, bb, document, + initialProps, bb, parentNode}) => { const func = initialProps._id diff --git a/packages/server/utilities/builder/buildApp.js b/packages/server/utilities/builder/buildApp.js index f2791c29f8..76a0ef9342 100644 --- a/packages/server/utilities/builder/buildApp.js +++ b/packages/server/utilities/builder/buildApp.js @@ -83,11 +83,9 @@ const buildIndexHtml = async (config, appname, appPath, pages, pageName) => { const buildClientAppDefinition = async (config, appname, appdefinition, appPath, pages, pageName) => { - const appPublicPath = publicPath(appPath, pageName); const appRootPath = rootPath(config, appname); - const componentLibraries = []; @@ -129,6 +127,7 @@ const buildClientAppDefinition = async (config, appname, appdefinition, appPath, } await writeFile(filename, - `window['##BUDIBASE_APPDEFINITION##'] = ${JSON.stringify(clientAppDefObj)}`); +`window['##BUDIBASE_APPDEFINITION##'] = ${JSON.stringify(clientAppDefObj)}; +window['##BUDIBASE_UIFUNCTIONS##'] = ${appdefinition.uiFunctions}`); } \ No newline at end of file diff --git a/packages/standard-components/dist/generators.js b/packages/standard-components/dist/generators.js index 921321c87a..7fd547ea88 100644 --- a/packages/standard-components/dist/generators.js +++ b/packages/standard-components/dist/generators.js @@ -204,4 +204,4 @@ const buttons = () => [ ]; export { app, buttons, forms, headers, indexTables, nav }; -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"generators.js","sources":["../src/generators/headersGenerator.js","../src/generators/formsGenerator.js","../src/generators/indexTablesGenerator.js","../src/generators/navGenerator.js","../src/generators/appGenerator.js","../src/generators/buttonsGenerator.js"],"sourcesContent":["export const headers = () => [\n    {\n        name: \"common/H1\",\n        description: \"Header 1\",\n        inherits: \"@budibase/standard-components/text\",\n        props: {\n            font: \"20pt\",\n        }\n    },\n    {\n        name: \"common/H2\",\n        description: \"Header 2\",\n        inherits: \"@budibase/standard-components/text\",\n        props: {\n            font: \"15pt\",\n        }\n    },\n    {\n        name: \"common/H3\",\n        description: \"Header 3\",\n        inherits: \"@budibase/standard-components/text\",\n        props: {\n            font: \"12pt bold\",\n        }\n    },\n    {\n        name: \"common/H4\",\n        description: \"Header 4\",\n        inherits: \"@budibase/standard-components/text\",\n        props: {\n            font: \"10pt bold\",\n        }\n    }\n]","import {headers} from \"./headersGenerator\";\n\nexport const forms = ({records, indexes}) => \n    [...headers({records, indexes}),\n    ...records.map(root)];\n\nconst root = record => ({\n    name: `${record.name} Form`,\n    description: `All fields on record '${record.nodeKey()}' `,\n    inherits: \"@budibase/standard-components/stackpanel\",\n    props: {\n        direction: \"vertical\",\n        children: [\n            {\n                control: {\n                    _component: \"common/H1\",\n                    value: `Edit ${record.name}`,\n                }\n            },\n            form(record),\n            saveCancelButtons(record)\n        ]\n    }\n}) \n\nconst form = record => ({\n    control: {\n        _component: \"@budibase/standard-components/form\",\n        formControls: \n            record.fields.map(f => ({\n                label: f.label,\n                control: {\n                    _component: \"@budibase/standard-components/textbox\",\n                    value: {\n                        \"##bbstate\":`current${record.name}.${f.name}`,\n                        \"##bbsource\":\"store\"\n                    }\n                }\n            }))\n    }\n})\n\nconst saveCancelButtons = (record) => ({\n    control: {\n        _component: \"@budibase/standard-components/stackpanel\",\n        direction: \"horizontal\",\n        children: [\n            paddedPanelForButton({\n                _component: \"common/Primary Button\",\n                contentText: `Save ${record.name}`,\n                onClick: [                \n                    {\n                        \"##eventHandlerType\": \"Save Record\",\n                        parameters: {\n                            statePath: `current${record.name}`,\n                        }\n                    }\n                ]\n            }),\n            paddedPanelForButton({\n                _component: \"common/Secondary Button\",\n                contentText: `Cancel`,\n                onClick: [\n                    {\n                        \"##eventHandlerType\": \"Save Record\",\n                        parameters: {\n                            statePath: `current${record.name}`,\n                        }\n                    }\n                ]\n            })\n        ]\n    }\n})\n\nconst paddedPanelForButton = (button) => ({\n    control: {\n        _component: \"@budibase/standard-components/panel\",\n        padding: \"20px\",\n        component: button\n    }\n});\n\n","export const indexTables = ({indexes, helpers}) => \n    indexes.filter(i => i.parent().type === \"root\")\n           .map(i => indexTable(i, helpers));\n\nexport const indexTableProps = (index, helpers) => ({\n    data: {\n        \"##bbstate\":index.nodeKey(),\n        \"##bbsource\":\"store\"\n    },\n    columns: helpers.indexSchema(index).map(column)\n});\n\nconst indexTable = (index, helpers) => ({\n    name: `tables/${index.name} Table`,\n    inherits: \"@budibase/standard-components/table\",\n    props: indexTableProps(index, helpers)\n});\n\nconst column = (col) => ({\n    title: col.name,\n    value: {\n        \"##bbstate\": col.name,\n        \"##bbsource\":\"context\"\n    }\n})","import {indexTables} from \"./indexTablesGenerator\";\n\nexport const nav = ({records, indexes, helpers}) => [\n    {\n        name: \"Application Root\",\n        inherits: \"@budibase/standard-components/nav\",\n        props: {\n            items: indexes\n                    .filter(i => i.parent().type === \"root\")\n                    .map(navItem),\n            selectedItem: {\n                \"##bbstate\": \"selectedNav\",\n                \"##bbstatefallback\": records[0].collectionName,\n                \"##bbsource\":\"store\"\n            }\n        },\n    },\n    ...indexTables({records, indexes, helpers})\n]\n\n\nexport const navItem = (index) => ({\n    title: index.name,\n    component : {\n        _component: `tables/${index.name} Table`\n    }\n})\n\n","import { forms } from \"./formsGenerator\";\nimport { nav } from \"./navGenerator\";\n\nexport const app = (params) => {\n\n    return [\n        ...nav(params),\n        ...forms(params)\n    ];\n}\n\n","export const buttons = () => [\n    {\n        name: \"common/Primary Button\",\n        description: \"a styled button\",\n        inherits: \"@budibase/standard-components/button\",\n        props: {\n            padding: \"5px 7px\",\n            border: \"1px solid #EEE\",\n            color: \"#5F6368\",\n            background: \"##f2f2f2\",\n            hoverColor: \"black\",\n            hoverBackground: \"#cccccc\"\n        }\n    },\n    {\n        name: \"common/Secondary Button\",\n        description: \"a styled button\",\n        inherits: \"@budibase/standard-components/button\",\n        props: {\n            padding: \"5px 7px\",\n            border: \"1px solid #EEE\",\n            color: \"#5F6368\",\n            background: \"##f2f2f2\",\n            hoverColor: \"black\",\n            hoverBackground: \"#cccccc\"\n        }\n    }\n]"],"names":[],"mappings":"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"} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"generators.js","sources":["../src/generators/headersGenerator.js","../src/generators/formsGenerator.js","../src/generators/indexTablesGenerator.js","../src/generators/navGenerator.js","../src/generators/appGenerator.js","../src/generators/buttonsGenerator.js"],"sourcesContent":["export const headers = () => [\r\n    {\r\n        name: \"common/H1\",\r\n        description: \"Header 1\",\r\n        inherits: \"@budibase/standard-components/text\",\r\n        props: {\r\n            font: \"20pt\",\r\n        }\r\n    },\r\n    {\r\n        name: \"common/H2\",\r\n        description: \"Header 2\",\r\n        inherits: \"@budibase/standard-components/text\",\r\n        props: {\r\n            font: \"15pt\",\r\n        }\r\n    },\r\n    {\r\n        name: \"common/H3\",\r\n        description: \"Header 3\",\r\n        inherits: \"@budibase/standard-components/text\",\r\n        props: {\r\n            font: \"12pt bold\",\r\n        }\r\n    },\r\n    {\r\n        name: \"common/H4\",\r\n        description: \"Header 4\",\r\n        inherits: \"@budibase/standard-components/text\",\r\n        props: {\r\n            font: \"10pt bold\",\r\n        }\r\n    }\r\n]","import {headers} from \"./headersGenerator\";\r\n\r\nexport const forms = ({records, indexes}) => \r\n    [...headers({records, indexes}),\r\n    ...records.map(root)];\r\n\r\nconst root = record => ({\r\n    name: `${record.name} Form`,\r\n    description: `All fields on record '${record.nodeKey()}' `,\r\n    inherits: \"@budibase/standard-components/stackpanel\",\r\n    props: {\r\n        direction: \"vertical\",\r\n        children: [\r\n            {\r\n                control: {\r\n                    _component: \"common/H1\",\r\n                    value: `Edit ${record.name}`,\r\n                }\r\n            },\r\n            form(record),\r\n            saveCancelButtons(record)\r\n        ]\r\n    }\r\n}) \r\n\r\nconst form = record => ({\r\n    control: {\r\n        _component: \"@budibase/standard-components/form\",\r\n        formControls: \r\n            record.fields.map(f => ({\r\n                label: f.label,\r\n                control: {\r\n                    _component: \"@budibase/standard-components/textbox\",\r\n                    value: {\r\n                        \"##bbstate\":`current${record.name}.${f.name}`,\r\n                        \"##bbsource\":\"store\"\r\n                    }\r\n                }\r\n            }))\r\n    }\r\n})\r\n\r\nconst saveCancelButtons = (record) => ({\r\n    control: {\r\n        _component: \"@budibase/standard-components/stackpanel\",\r\n        direction: \"horizontal\",\r\n        children: [\r\n            paddedPanelForButton({\r\n                _component: \"common/Primary Button\",\r\n                contentText: `Save ${record.name}`,\r\n                onClick: [                \r\n                    {\r\n                        \"##eventHandlerType\": \"Save Record\",\r\n                        parameters: {\r\n                            statePath: `current${record.name}`,\r\n                        }\r\n                    }\r\n                ]\r\n            }),\r\n            paddedPanelForButton({\r\n                _component: \"common/Secondary Button\",\r\n                contentText: `Cancel`,\r\n                onClick: [\r\n                    {\r\n                        \"##eventHandlerType\": \"Save Record\",\r\n                        parameters: {\r\n                            statePath: `current${record.name}`,\r\n                        }\r\n                    }\r\n                ]\r\n            })\r\n        ]\r\n    }\r\n})\r\n\r\nconst paddedPanelForButton = (button) => ({\r\n    control: {\r\n        _component: \"@budibase/standard-components/panel\",\r\n        padding: \"20px\",\r\n        component: button\r\n    }\r\n});\r\n\r\n","export const indexTables = ({indexes, helpers}) => \r\n    indexes.filter(i => i.parent().type === \"root\")\r\n           .map(i => indexTable(i, helpers));\r\n\r\nexport const indexTableProps = (index, helpers) => ({\r\n    data: {\r\n        \"##bbstate\":index.nodeKey(),\r\n        \"##bbsource\":\"store\"\r\n    },\r\n    columns: helpers.indexSchema(index).map(column)\r\n});\r\n\r\nconst indexTable = (index, helpers) => ({\r\n    name: `tables/${index.name} Table`,\r\n    inherits: \"@budibase/standard-components/table\",\r\n    props: indexTableProps(index, helpers)\r\n});\r\n\r\nconst column = (col) => ({\r\n    title: col.name,\r\n    value: {\r\n        \"##bbstate\": col.name,\r\n        \"##bbsource\":\"context\"\r\n    }\r\n})","import {indexTables} from \"./indexTablesGenerator\";\r\n\r\nexport const nav = ({records, indexes, helpers}) => [\r\n    {\r\n        name: \"Application Root\",\r\n        inherits: \"@budibase/standard-components/nav\",\r\n        props: {\r\n            items: indexes\r\n                    .filter(i => i.parent().type === \"root\")\r\n                    .map(navItem),\r\n            selectedItem: {\r\n                \"##bbstate\": \"selectedNav\",\r\n                \"##bbstatefallback\": records[0].collectionName,\r\n                \"##bbsource\":\"store\"\r\n            }\r\n        },\r\n    },\r\n    ...indexTables({records, indexes, helpers})\r\n]\r\n\r\n\r\nexport const navItem = (index) => ({\r\n    title: index.name,\r\n    component : {\r\n        _component: `tables/${index.name} Table`\r\n    }\r\n})\r\n\r\n","import { forms } from \"./formsGenerator\";\r\nimport { nav } from \"./navGenerator\";\r\n\r\nexport const app = (params) => {\r\n\r\n    return [\r\n        ...nav(params),\r\n        ...forms(params)\r\n    ];\r\n}\r\n\r\n","export const buttons = () => [\r\n    {\r\n        name: \"common/Primary Button\",\r\n        description: \"a styled button\",\r\n        inherits: \"@budibase/standard-components/button\",\r\n        props: {\r\n            padding: \"5px 7px\",\r\n            border: \"1px solid #EEE\",\r\n            color: \"#5F6368\",\r\n            background: \"##f2f2f2\",\r\n            hoverColor: \"black\",\r\n            hoverBackground: \"#cccccc\"\r\n        }\r\n    },\r\n    {\r\n        name: \"common/Secondary Button\",\r\n        description: \"a styled button\",\r\n        inherits: \"@budibase/standard-components/button\",\r\n        props: {\r\n            padding: \"5px 7px\",\r\n            border: \"1px solid #EEE\",\r\n            color: \"#5F6368\",\r\n            background: \"##f2f2f2\",\r\n            hoverColor: \"black\",\r\n            hoverBackground: \"#cccccc\"\r\n        }\r\n    }\r\n]"],"names":[],"mappings":"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"}