Merge branch 'couchdb-api' of github.com:Budibase/budibase into couchdb-api
This commit is contained in:
commit
e2b393a077
File diff suppressed because it is too large
Load Diff
|
@ -2,6 +2,7 @@
|
||||||
"name": "root",
|
"name": "root",
|
||||||
"private": true,
|
"private": true,
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@rollup/plugin-json": "^4.0.2",
|
||||||
"babel-eslint": "^10.0.3",
|
"babel-eslint": "^10.0.3",
|
||||||
"eslint": "^6.8.0",
|
"eslint": "^6.8.0",
|
||||||
"eslint-plugin-prettier": "^3.1.2",
|
"eslint-plugin-prettier": "^3.1.2",
|
||||||
|
@ -17,7 +18,7 @@
|
||||||
"build": "lerna run build",
|
"build": "lerna run build",
|
||||||
"initialise": "lerna run initialise",
|
"initialise": "lerna run initialise",
|
||||||
"publishdev": "lerna run publishdev",
|
"publishdev": "lerna run publishdev",
|
||||||
"publishnpm":"yarn build && lerna publish",
|
"publishnpm": "yarn build && lerna publish",
|
||||||
"clean": "lerna clean",
|
"clean": "lerna clean",
|
||||||
"dev": "lerna run --parallel --stream dev:builder",
|
"dev": "lerna run --parallel --stream dev:builder",
|
||||||
"test": "lerna run test",
|
"test": "lerna run test",
|
||||||
|
|
|
@ -9,6 +9,8 @@ import builtins from "rollup-plugin-node-builtins"
|
||||||
import nodeglobals from "rollup-plugin-node-globals"
|
import nodeglobals from "rollup-plugin-node-globals"
|
||||||
import copy from "rollup-plugin-copy"
|
import copy from "rollup-plugin-copy"
|
||||||
import replace from "rollup-plugin-replace"
|
import replace from "rollup-plugin-replace"
|
||||||
|
import json from '@rollup/plugin-json';
|
||||||
|
|
||||||
|
|
||||||
import path from "path"
|
import path from "path"
|
||||||
|
|
||||||
|
@ -133,10 +135,16 @@ export default {
|
||||||
plugins: [
|
plugins: [
|
||||||
alias({
|
alias({
|
||||||
entries: [
|
entries: [
|
||||||
{ find: "components", replacement: path.resolve(projectRootDir, 'src/components') },
|
{
|
||||||
{ find: "builderStore", replacement: path.resolve(projectRootDir, 'src/builderStore') }
|
find: "components",
|
||||||
|
replacement: path.resolve(projectRootDir, "src/components"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
find: "builderStore",
|
||||||
|
replacement: path.resolve(projectRootDir, "src/builderStore"),
|
||||||
|
},
|
||||||
],
|
],
|
||||||
customResolver
|
customResolver,
|
||||||
}),
|
}),
|
||||||
copy({
|
copy({
|
||||||
targets: [
|
targets: [
|
||||||
|
@ -206,5 +214,6 @@ export default {
|
||||||
// If we're building for production (npm run build
|
// If we're building for production (npm run build
|
||||||
// instead of npm run dev), minify
|
// instead of npm run dev), minify
|
||||||
production && terser(),
|
production && terser(),
|
||||||
|
json(),
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,8 +27,8 @@ export async function loadRecord(key, { appname, instanceId }) {
|
||||||
return await response.json()
|
return await response.json()
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function saveRecord({ record, instanceId, modelId }) {
|
export async function saveRecord(record, instanceId) {
|
||||||
const SAVE_RECORDS_URL = `/api/${instanceId}/${modelId}/records`
|
const SAVE_RECORDS_URL = `/api/${instanceId}/records`
|
||||||
const response = await api.post(SAVE_RECORDS_URL, record)
|
const response = await api.post(SAVE_RECORDS_URL, record)
|
||||||
return await response.json()
|
return await response.json()
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,19 +43,22 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
async function saveRecord() {
|
async function saveRecord() {
|
||||||
const recordResponse = await api.saveRecord({
|
const recordResponse = await api.saveRecord(
|
||||||
record,
|
{
|
||||||
instanceId,
|
...record,
|
||||||
modelId: $backendUiStore.selectedModel._id
|
modelId: $backendUiStore.selectedModel._id,
|
||||||
})
|
},
|
||||||
|
instanceId
|
||||||
|
)
|
||||||
if (recordResponse.errors) {
|
if (recordResponse.errors) {
|
||||||
errors = recordResponse.errors;
|
errors = recordResponse.errors
|
||||||
return;
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
backendUiStore.update(state => {
|
backendUiStore.update(state => {
|
||||||
state.selectedView = state.selectedView
|
state.selectedView = state.selectedView
|
||||||
return state
|
return state
|
||||||
|
onClosed();
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -14,7 +14,6 @@
|
||||||
async function createUser() {
|
async function createUser() {
|
||||||
const user = { name: username, username, password }
|
const user = { name: username, username, password }
|
||||||
const response = await api.createUser(user, instanceId);
|
const response = await api.createUser(user, instanceId);
|
||||||
console.log(response);
|
|
||||||
backendUiStore.actions.users.create(response)
|
backendUiStore.actions.users.create(response)
|
||||||
onClosed()
|
onClosed()
|
||||||
}
|
}
|
||||||
|
@ -22,11 +21,14 @@
|
||||||
|
|
||||||
<form on:submit|preventDefault class="uk-form-stacked">
|
<form on:submit|preventDefault class="uk-form-stacked">
|
||||||
<div>
|
<div>
|
||||||
<label class="uk-form-label" for="form-stacked-text">Username</label>
|
<div class="uk-margin">
|
||||||
<input class="uk-input" type="text" bind:value={username} />
|
<label class="uk-form-label" for="form-stacked-text">Username</label>
|
||||||
<label class="uk-form-label" for="form-stacked-text">Password</label>
|
<input class="uk-input" type="text" bind:value={username} />
|
||||||
<input class="uk-input" type="password" bind:value={password} />
|
</div>
|
||||||
<label class="uk-form-label" for="form-stacked-text">Access Levels</label>
|
<div class="uk-margin">
|
||||||
|
<label class="uk-form-label" for="form-stacked-text">Password</label>
|
||||||
|
<input class="uk-input" type="password" bind:value={password} />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<footer>
|
<footer>
|
||||||
<ActionButton alert on:click={onClosed}>Cancel</ActionButton>
|
<ActionButton alert on:click={onClosed}>Cancel</ActionButton>
|
||||||
|
|
|
@ -146,6 +146,7 @@
|
||||||
|
|
||||||
.root {
|
.root {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
|
padding: 20px;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<script>
|
<script>
|
||||||
import { splitName } from "./pagesParsing/splitRootComponentName.js"
|
import { splitName } from "./pagesParsing/splitRootComponentName.js"
|
||||||
import { store } from "builderStore"
|
import components from "./temporaryPanelStructure.js"
|
||||||
|
import ConfirmDialog from "components/common/ConfirmDialog.svelte"
|
||||||
import {
|
import {
|
||||||
find,
|
find,
|
||||||
sortBy,
|
sortBy,
|
||||||
|
@ -11,10 +12,7 @@
|
||||||
uniqBy,
|
uniqBy,
|
||||||
flatten,
|
flatten,
|
||||||
} from "lodash/fp"
|
} from "lodash/fp"
|
||||||
import { ImageIcon, InputIcon, LayoutIcon } from "components/common/Icons/"
|
|
||||||
import Select from "components/common/Select.svelte"
|
|
||||||
import Button from "components/common/PlusButton.svelte"
|
|
||||||
import ConfirmDialog from "components/common/ConfirmDialog.svelte"
|
|
||||||
import {
|
import {
|
||||||
getRecordNodes,
|
getRecordNodes,
|
||||||
getIndexNodes,
|
getIndexNodes,
|
||||||
|
@ -22,15 +20,54 @@
|
||||||
pipe,
|
pipe,
|
||||||
} from "components/common/core"
|
} from "components/common/core"
|
||||||
|
|
||||||
|
import Tab from "./ItemTab/Tab.svelte"
|
||||||
|
import { store } from "builderStore"
|
||||||
|
|
||||||
export let toggleTab
|
export let toggleTab
|
||||||
|
|
||||||
let componentLibraries = []
|
|
||||||
let current_view = "text"
|
|
||||||
let selectedComponent = null
|
|
||||||
let selectedLib
|
|
||||||
let selectTemplateDialog
|
let selectTemplateDialog
|
||||||
let templateInstances = []
|
|
||||||
let selectedTemplateInstance
|
let selectedTemplateInstance
|
||||||
|
let templateInstances = []
|
||||||
|
|
||||||
|
let selectedComponent = null
|
||||||
|
|
||||||
|
const categories = components.categories
|
||||||
|
let selectedCategory = categories[0]
|
||||||
|
|
||||||
|
const onTemplateChosen = template => {
|
||||||
|
selectedComponent = null
|
||||||
|
const { componentName, libName } = splitName(template.name)
|
||||||
|
const templateOptions = {
|
||||||
|
records: getRecordNodes(hierarchy),
|
||||||
|
indexes: getIndexNodes(hierarchy),
|
||||||
|
helpers: {
|
||||||
|
indexSchema: getIndexSchema(hierarchy),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
templateInstances = libraryModules[libName][componentName](templateOptions)
|
||||||
|
if (!templateInstances || templateInstances.length === 0) return
|
||||||
|
selectedTemplateInstance = templateInstances[0].name
|
||||||
|
selectTemplateDialog.show()
|
||||||
|
}
|
||||||
|
|
||||||
|
const onComponentChosen = component => {
|
||||||
|
if (component.template) {
|
||||||
|
onTemplateChosen(component.template)
|
||||||
|
} else {
|
||||||
|
store.addChildComponent(component._component)
|
||||||
|
toggleTab()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const onTemplateInstanceChosen = () => {
|
||||||
|
selectedComponent = null
|
||||||
|
const instance = templateInstances.find(
|
||||||
|
i => i.name === selectedTemplateInstance
|
||||||
|
)
|
||||||
|
store.addTemplatedComponent(instance.props)
|
||||||
|
toggleTab()
|
||||||
|
}
|
||||||
|
|
||||||
$: templatesByComponent = groupBy(t => t.component)($store.templates)
|
$: templatesByComponent = groupBy(t => t.component)($store.templates)
|
||||||
$: hierarchy = $store.hierarchy
|
$: hierarchy = $store.hierarchy
|
||||||
|
@ -45,157 +82,25 @@
|
||||||
uniqBy(t => t.name),
|
uniqBy(t => t.name),
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
||||||
const addRootComponent = (component, allComponents) => {
|
|
||||||
const { libName } = splitName(component.name)
|
|
||||||
let group = find(r => r.libName === libName)(allComponents)
|
|
||||||
|
|
||||||
if (!group) {
|
|
||||||
group = {
|
|
||||||
libName,
|
|
||||||
components: [],
|
|
||||||
}
|
|
||||||
|
|
||||||
allComponents.push(group)
|
|
||||||
}
|
|
||||||
|
|
||||||
group.components.push(component)
|
|
||||||
}
|
|
||||||
|
|
||||||
const onComponentChosen = component => {
|
|
||||||
if (component.template) {
|
|
||||||
onTemplateChosen(component.template)
|
|
||||||
} else {
|
|
||||||
store.addChildComponent(component.name)
|
|
||||||
toggleTab()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const onTemplateChosen = template => {
|
|
||||||
selectedComponent = null
|
|
||||||
const { componentName, libName } = splitName(template.name)
|
|
||||||
const templateOptions = {
|
|
||||||
records: getRecordNodes(hierarchy),
|
|
||||||
indexes: getIndexNodes(hierarchy),
|
|
||||||
helpers: {
|
|
||||||
indexSchema: getIndexSchema(hierarchy),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
templateInstances = libraryModules[libName][componentName](templateOptions)
|
|
||||||
if (!templateInstances || templateInstances.length === 0) return
|
|
||||||
selectedTemplateInstance = templateInstances[0].name
|
|
||||||
selectTemplateDialog.show()
|
|
||||||
}
|
|
||||||
|
|
||||||
const onTemplateInstanceChosen = () => {
|
|
||||||
selectedComponent = null
|
|
||||||
const instance = templateInstances.find(
|
|
||||||
i => i.name === selectedTemplateInstance
|
|
||||||
)
|
|
||||||
store.addTemplatedComponent(instance.props)
|
|
||||||
toggleTab()
|
|
||||||
}
|
|
||||||
|
|
||||||
function generate_components_list(components) {
|
|
||||||
return ($store.currentFrontEndType === "page"
|
|
||||||
? $store.builtins.concat(components)
|
|
||||||
: components
|
|
||||||
).concat(standaloneTemplates)
|
|
||||||
}
|
|
||||||
|
|
||||||
$: {
|
|
||||||
const newComponentLibraries = []
|
|
||||||
|
|
||||||
for (let comp of sortBy(["name"])($store.components)) {
|
|
||||||
addRootComponent(comp, newComponentLibraries)
|
|
||||||
}
|
|
||||||
|
|
||||||
componentLibraries = newComponentLibraries
|
|
||||||
if (!selectedLib) selectedLib = newComponentLibraries[0].libName
|
|
||||||
}
|
|
||||||
|
|
||||||
$: componentLibrary = componentLibraries.find(l => l.libName === selectedLib)
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="root">
|
<div class="root">
|
||||||
<Select on:change={e => (selectedLib = e.target.value)}>
|
<ul class="tabs">
|
||||||
{#each componentLibraries as lib}
|
{#each categories as category}
|
||||||
<option value={lib.libName}>{lib.libName}</option>
|
<li
|
||||||
|
on:click={() => (selectedCategory = category)}
|
||||||
|
class:active={selectedCategory === category}>
|
||||||
|
{category.name}
|
||||||
|
</li>
|
||||||
{/each}
|
{/each}
|
||||||
</Select>
|
</ul>
|
||||||
|
<div class="panel">
|
||||||
<div class="library-container">
|
<Tab
|
||||||
<!-- <ul>
|
list={selectedCategory}
|
||||||
<li>
|
on:selectItem={e => onComponentChosen(e.detail)}
|
||||||
<button
|
{onTemplateChosen}
|
||||||
class:selected={current_view === 'text'}
|
{toggleTab} />
|
||||||
on:click={() => (current_view = 'text')}>
|
|
||||||
<InputIcon />
|
|
||||||
</button>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<button
|
|
||||||
class:selected={current_view === 'layout'}
|
|
||||||
on:click={() => (current_view = 'layout')}>
|
|
||||||
<LayoutIcon />
|
|
||||||
</button>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<button
|
|
||||||
class:selected={current_view === 'media'}
|
|
||||||
on:click={() => (current_view = 'media')}>
|
|
||||||
<ImageIcon />
|
|
||||||
</button>
|
|
||||||
</li>
|
|
||||||
</ul> -->
|
|
||||||
|
|
||||||
{#if componentLibrary}
|
|
||||||
{#each generate_components_list(componentLibrary.components) as component}
|
|
||||||
<div class="component-container">
|
|
||||||
<div class="component" on:click={() => onComponentChosen(component)}>
|
|
||||||
<div class="name">{splitName(component.name).componentName}</div>
|
|
||||||
{#if (component.presets || templatesByComponent[component.name]) && component.name === selectedComponent}
|
|
||||||
<ul class="preset-menu">
|
|
||||||
{#if component.presets}
|
|
||||||
<span>{splitName(component.name).componentName} Presets</span>
|
|
||||||
{#each Object.keys(component.presets) as preset}
|
|
||||||
<li
|
|
||||||
on:click|stopPropagation={() => onComponentChosen(component, preset)}>
|
|
||||||
{preset}
|
|
||||||
</li>
|
|
||||||
{/each}
|
|
||||||
{/if}
|
|
||||||
{#if templatesByComponent[component.name]}
|
|
||||||
<span>
|
|
||||||
{splitName(component.name).componentName} Templates
|
|
||||||
</span>
|
|
||||||
{#each templatesByComponent[component.name] as template}
|
|
||||||
<li
|
|
||||||
on:click|stopPropagation={() => onTemplateChosen(template)}>
|
|
||||||
{template.description}
|
|
||||||
</li>
|
|
||||||
{/each}
|
|
||||||
{/if}
|
|
||||||
</ul>
|
|
||||||
{/if}
|
|
||||||
</div>
|
|
||||||
{#if component.presets || templatesByComponent[component.name]}
|
|
||||||
<Button
|
|
||||||
on:click={() => {
|
|
||||||
selectedComponent = selectedComponent ? null : component.name
|
|
||||||
}}>
|
|
||||||
<span
|
|
||||||
class="open-presets"
|
|
||||||
class:open={selectedComponent === component.name}>
|
|
||||||
...
|
|
||||||
</span>
|
|
||||||
</Button>
|
|
||||||
{/if}
|
|
||||||
</div>
|
|
||||||
{/each}
|
|
||||||
{/if}
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<ConfirmDialog
|
<ConfirmDialog
|
||||||
|
@ -218,111 +123,32 @@
|
||||||
</ConfirmDialog>
|
</ConfirmDialog>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.root {
|
.tabs {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
justify-content: center;
|
||||||
}
|
|
||||||
|
|
||||||
.library-container {
|
|
||||||
padding: 0 0 10px 0;
|
|
||||||
flex: 1 1 auto;
|
|
||||||
min-height: 0px;
|
|
||||||
margin-top: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.component-container {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.component {
|
|
||||||
position: relative;
|
|
||||||
padding: 0 15px;
|
|
||||||
cursor: pointer;
|
|
||||||
border: 1px solid #d8d8d8;
|
|
||||||
border-radius: 2px;
|
|
||||||
margin: 5px 0;
|
|
||||||
height: 40px;
|
|
||||||
box-sizing: border-box;
|
|
||||||
color: #000333;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
flex: 1;
|
|
||||||
margin-right: 5px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.component:hover {
|
|
||||||
background-color: var(--lightslate);
|
|
||||||
}
|
|
||||||
|
|
||||||
.component > .name {
|
|
||||||
color: #000333;
|
|
||||||
display: inline-block;
|
|
||||||
font-size: 13px;
|
|
||||||
opacity: 0.8;
|
|
||||||
}
|
|
||||||
|
|
||||||
ul {
|
|
||||||
list-style: none;
|
list-style: none;
|
||||||
display: flex;
|
margin: 0 auto;
|
||||||
padding: 0;
|
padding: 0 30px;
|
||||||
}
|
border-bottom: 1px solid #d8d8d8;
|
||||||
|
|
||||||
.preset-menu {
|
|
||||||
flex-direction: column;
|
|
||||||
position: absolute;
|
|
||||||
top: 25px;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
z-index: 1;
|
|
||||||
background: #fafafa;
|
|
||||||
padding: 10px;
|
|
||||||
border-radius: 2px;
|
|
||||||
color: var(--secondary80);
|
|
||||||
}
|
|
||||||
|
|
||||||
.preset-menu > span {
|
|
||||||
font-size: 13px;
|
|
||||||
text-transform: uppercase;
|
|
||||||
margin-top: 5px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.preset-menu li {
|
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
margin-top: 13px;
|
font-weight: 500;
|
||||||
}
|
letter-spacing: 0.14px;
|
||||||
|
|
||||||
.preset-menu li:hover {
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
li {
|
li {
|
||||||
margin-right: 20px;
|
color: #808192;
|
||||||
background: none;
|
margin: 0 5px;
|
||||||
border-radius: 5px;
|
padding: 0 8px;
|
||||||
}
|
|
||||||
|
|
||||||
/* li button {
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
border-radius: 5px;
|
|
||||||
padding: 13px;
|
|
||||||
outline: none;
|
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
} */
|
|
||||||
|
|
||||||
/* .selected {
|
|
||||||
color: var(--button-text);
|
|
||||||
background: var(--background-button) !important;
|
|
||||||
} */
|
|
||||||
|
|
||||||
.open {
|
|
||||||
color: rgba(0, 85, 255, 1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.template-instance-label {
|
.panel {
|
||||||
margin-left: 20px;
|
padding: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.active {
|
||||||
|
border-bottom: solid 3px #0055ff;
|
||||||
|
color: #393c44;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -60,9 +60,7 @@
|
||||||
.switcher {
|
.switcher {
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
margin-bottom: 20px;
|
margin: 20px;
|
||||||
padding: 0 20px 20px;
|
|
||||||
border-bottom: 1px solid #d8d8d8;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.switcher > button {
|
.switcher > button {
|
||||||
|
@ -85,11 +83,4 @@
|
||||||
color: var(--secondary100);
|
color: var(--secondary100);
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
|
|
||||||
.panel {
|
|
||||||
flex: 1 1 auto;
|
|
||||||
height: 0px;
|
|
||||||
overflow-y: auto;
|
|
||||||
padding: 0 20px 40px 20px;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -133,15 +133,13 @@
|
||||||
grid-column-start: middle;
|
grid-column-start: middle;
|
||||||
position: relative;
|
position: relative;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
padding-top: 56.25%;
|
|
||||||
margin: auto;
|
margin: auto;
|
||||||
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.component-container iframe {
|
.component-container iframe {
|
||||||
border: 0;
|
border: 0;
|
||||||
height: 100%;
|
|
||||||
left: 0;
|
left: 0;
|
||||||
position: absolute;
|
|
||||||
top: 0;
|
top: 0;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,66 @@
|
||||||
|
<script>
|
||||||
|
export let item
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="item-item" on:click>
|
||||||
|
<div class="item-icon">
|
||||||
|
<i class={item.icon} />
|
||||||
|
</div>
|
||||||
|
<div class="item-text">
|
||||||
|
<div class="item-name">{item.name}</div>
|
||||||
|
<div class="item-description">
|
||||||
|
<p>{item.description}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.item-item {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
padding: 10px 0px 8px 10px;
|
||||||
|
align-items: center;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-item:hover {
|
||||||
|
background: #fbfbfb;
|
||||||
|
border-radius: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-icon {
|
||||||
|
flex: 0 0 40px;
|
||||||
|
background: #f1f4fc;
|
||||||
|
height: 40px;
|
||||||
|
border-radius: 5px;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-text {
|
||||||
|
display: flex;
|
||||||
|
padding-left: 16px;
|
||||||
|
padding-top: 8px;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-name {
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-description {
|
||||||
|
font-size: 12px;
|
||||||
|
color: #808192;
|
||||||
|
}
|
||||||
|
|
||||||
|
p {
|
||||||
|
line-height: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
i {
|
||||||
|
font-size: 24px;
|
||||||
|
color: #808192;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,44 @@
|
||||||
|
<script>
|
||||||
|
import { createEventDispatcher } from "svelte"
|
||||||
|
const dispatch = createEventDispatcher()
|
||||||
|
|
||||||
|
import Item from "./Item.svelte"
|
||||||
|
import { store } from "builderStore"
|
||||||
|
export let list
|
||||||
|
|
||||||
|
let category = list
|
||||||
|
|
||||||
|
const handleClick = item => {
|
||||||
|
if (item.children && item.children.length > 0) {
|
||||||
|
list = item
|
||||||
|
} else {
|
||||||
|
dispatch("selectItem", item)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const goBack = () => {
|
||||||
|
list = category
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{#if !list.isCategory}
|
||||||
|
<button class="back-button" on:click={() => (list = category)}>Back</button>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#each list.children as item}
|
||||||
|
<Item {item} on:click={() => handleClick(item)} />
|
||||||
|
{/each}
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.back-button {
|
||||||
|
font-size: 16px;
|
||||||
|
width: 100%;
|
||||||
|
text-align: center;
|
||||||
|
height: 40px;
|
||||||
|
border-radius: 3px;
|
||||||
|
border: solid 1px #e8e8ef;
|
||||||
|
background: white;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,335 @@
|
||||||
|
<script>
|
||||||
|
import { splitName } from "./pagesParsing/splitRootComponentName.js"
|
||||||
|
import { store } from "builderStore"
|
||||||
|
import {
|
||||||
|
find,
|
||||||
|
sortBy,
|
||||||
|
groupBy,
|
||||||
|
values,
|
||||||
|
filter,
|
||||||
|
map,
|
||||||
|
uniqBy,
|
||||||
|
flatten,
|
||||||
|
} from "lodash/fp"
|
||||||
|
import { ImageIcon, InputIcon, LayoutIcon } from "components/common/Icons/"
|
||||||
|
import Select from "components/common/Select.svelte"
|
||||||
|
import Button from "components/common/PlusButton.svelte"
|
||||||
|
import ConfirmDialog from "components/common/ConfirmDialog.svelte"
|
||||||
|
import {
|
||||||
|
getRecordNodes,
|
||||||
|
getIndexNodes,
|
||||||
|
getIndexSchema,
|
||||||
|
pipe,
|
||||||
|
} from "components/common/core"
|
||||||
|
|
||||||
|
export let toggleTab
|
||||||
|
|
||||||
|
let componentLibraries = []
|
||||||
|
let current_view = "text"
|
||||||
|
let selectedComponent = null
|
||||||
|
let selectedLib
|
||||||
|
let selectTemplateDialog
|
||||||
|
let templateInstances = []
|
||||||
|
let selectedTemplateInstance
|
||||||
|
|
||||||
|
//Info: Components seem to be generated from individual templates. Will this be the same going forward
|
||||||
|
$: templatesByComponent = groupBy(t => t.component)($store.templates)
|
||||||
|
$: hierarchy = $store.hierarchy
|
||||||
|
$: libraryModules = $store.libraries
|
||||||
|
$: standaloneTemplates = pipe(
|
||||||
|
templatesByComponent,
|
||||||
|
[
|
||||||
|
values,
|
||||||
|
flatten,
|
||||||
|
filter(t => !$store.components.some(c => c.name === t.component)),
|
||||||
|
map(t => ({ name: splitName(t.component).componentName, template: t })),
|
||||||
|
uniqBy(t => t.name),
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
const addRootComponent = (component, allComponents) => {
|
||||||
|
const { libName } = splitName(component.name)
|
||||||
|
let group = find(r => r.libName === libName)(allComponents)
|
||||||
|
|
||||||
|
if (!group) {
|
||||||
|
group = {
|
||||||
|
libName,
|
||||||
|
components: [],
|
||||||
|
}
|
||||||
|
|
||||||
|
allComponents.push(group)
|
||||||
|
}
|
||||||
|
|
||||||
|
group.components.push(component)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onComponentChosen = component => {
|
||||||
|
if (component.template) {
|
||||||
|
onTemplateChosen(component.template)
|
||||||
|
} else {
|
||||||
|
store.addChildComponent(component.name)
|
||||||
|
toggleTab()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Info: Called from menu beside components with presets and templates
|
||||||
|
const onTemplateChosen = template => {
|
||||||
|
selectedComponent = null
|
||||||
|
const { componentName, libName } = splitName(template.name)
|
||||||
|
//Info: how will DB changes effect this?
|
||||||
|
const templateOptions = {
|
||||||
|
records: getRecordNodes(hierarchy),
|
||||||
|
indexes: getIndexNodes(hierarchy),
|
||||||
|
helpers: {
|
||||||
|
indexSchema: getIndexSchema(hierarchy),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
//Info: go off and get template instances by library and component name
|
||||||
|
//libraryModules and hierarchies (used above) come from builderStore
|
||||||
|
templateInstances = libraryModules[libName][componentName](templateOptions)
|
||||||
|
if (!templateInstances || templateInstances.length === 0) return
|
||||||
|
selectedTemplateInstance = templateInstances[0].name
|
||||||
|
selectTemplateDialog.show()
|
||||||
|
}
|
||||||
|
|
||||||
|
const onTemplateInstanceChosen = () => {
|
||||||
|
selectedComponent = null
|
||||||
|
const instance = templateInstances.find(
|
||||||
|
i => i.name === selectedTemplateInstance
|
||||||
|
)
|
||||||
|
debugger
|
||||||
|
store.addTemplatedComponent(instance.props)
|
||||||
|
toggleTab()
|
||||||
|
}
|
||||||
|
|
||||||
|
function generate_components_list(components) {
|
||||||
|
debugger
|
||||||
|
return ($store.currentFrontEndType === "page"
|
||||||
|
? $store.builtins.concat(components)
|
||||||
|
: components
|
||||||
|
).concat(standaloneTemplates)
|
||||||
|
}
|
||||||
|
|
||||||
|
$: {
|
||||||
|
const newComponentLibraries = []
|
||||||
|
|
||||||
|
for (let comp of sortBy(["name"])($store.components)) {
|
||||||
|
addRootComponent(comp, newComponentLibraries)
|
||||||
|
}
|
||||||
|
|
||||||
|
componentLibraries = newComponentLibraries
|
||||||
|
if (!selectedLib) selectedLib = newComponentLibraries[0].libName
|
||||||
|
}
|
||||||
|
|
||||||
|
$: componentLibrary = componentLibraries.find(l => l.libName === selectedLib)
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="root">
|
||||||
|
<Select on:change={e => (selectedLib = e.target.value)}>
|
||||||
|
{#each componentLibraries as lib}
|
||||||
|
<option value={lib.libName}>{lib.libName}</option>
|
||||||
|
{/each}
|
||||||
|
</Select>
|
||||||
|
|
||||||
|
<div class="library-container">
|
||||||
|
<!-- <ul>
|
||||||
|
<li>
|
||||||
|
<button
|
||||||
|
class:selected={current_view === 'text'}
|
||||||
|
on:click={() => (current_view = 'text')}>
|
||||||
|
<InputIcon />
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<button
|
||||||
|
class:selected={current_view === 'layout'}
|
||||||
|
on:click={() => (current_view = 'layout')}>
|
||||||
|
<LayoutIcon />
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<button
|
||||||
|
class:selected={current_view === 'media'}
|
||||||
|
on:click={() => (current_view = 'media')}>
|
||||||
|
<ImageIcon />
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
</ul> -->
|
||||||
|
|
||||||
|
{#if componentLibrary}
|
||||||
|
{#each generate_components_list(componentLibrary.components) as component}
|
||||||
|
<div class="component-container">
|
||||||
|
<div class="component" on:click={() => onComponentChosen(component)}>
|
||||||
|
<div class="name">{splitName(component.name).componentName}</div>
|
||||||
|
{#if (component.presets || templatesByComponent[component.name]) && component.name === selectedComponent}
|
||||||
|
<ul class="preset-menu">
|
||||||
|
{#if component.presets}
|
||||||
|
<span>{splitName(component.name).componentName} Presets</span>
|
||||||
|
{#each Object.keys(component.presets) as preset}
|
||||||
|
<li
|
||||||
|
on:click|stopPropagation={() => onComponentChosen(component, preset)}>
|
||||||
|
{preset}
|
||||||
|
</li>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
|
{#if templatesByComponent[component.name]}
|
||||||
|
<span>
|
||||||
|
{splitName(component.name).componentName} Templates
|
||||||
|
</span>
|
||||||
|
{#each templatesByComponent[component.name] as template}
|
||||||
|
<li
|
||||||
|
on:click|stopPropagation={() => onTemplateChosen(template)}>
|
||||||
|
{template.description}
|
||||||
|
</li>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
|
</ul>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
{#if component.presets || templatesByComponent[component.name]}
|
||||||
|
<Button
|
||||||
|
on:click={() => {
|
||||||
|
selectedComponent = selectedComponent ? null : component.name
|
||||||
|
}}>
|
||||||
|
<span
|
||||||
|
class="open-presets"
|
||||||
|
class:open={selectedComponent === component.name}>
|
||||||
|
...
|
||||||
|
</span>
|
||||||
|
</Button>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ConfirmDialog
|
||||||
|
bind:this={selectTemplateDialog}
|
||||||
|
title="Choose Template"
|
||||||
|
onCancel={() => (selectedComponent = null)}
|
||||||
|
onOk={onTemplateInstanceChosen}>
|
||||||
|
{#each templateInstances.map(i => i.name) as instance}
|
||||||
|
<div class="uk-margin uk-grid-small uk-child-width-auto uk-grid">
|
||||||
|
<label>
|
||||||
|
<input
|
||||||
|
class="uk-radio"
|
||||||
|
type="radio"
|
||||||
|
bind:group={selectedTemplateInstance}
|
||||||
|
value={instance} />
|
||||||
|
<span class="template-instance-label">{instance}</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
</ConfirmDialog>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.root {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
.library-container {
|
||||||
|
padding: 0 0 10px 0;
|
||||||
|
flex: 1 1 auto;
|
||||||
|
min-height: 0px;
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.component-container {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.component {
|
||||||
|
position: relative;
|
||||||
|
padding: 0 15px;
|
||||||
|
cursor: pointer;
|
||||||
|
border: 1px solid #d8d8d8;
|
||||||
|
border-radius: 2px;
|
||||||
|
margin: 5px 0;
|
||||||
|
height: 40px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
color: #000333;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
flex: 1;
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.component:hover {
|
||||||
|
background-color: var(--lightslate);
|
||||||
|
}
|
||||||
|
|
||||||
|
.component > .name {
|
||||||
|
color: #000333;
|
||||||
|
display: inline-block;
|
||||||
|
font-size: 13px;
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul {
|
||||||
|
list-style: none;
|
||||||
|
display: flex;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preset-menu {
|
||||||
|
flex-direction: column;
|
||||||
|
position: absolute;
|
||||||
|
top: 25px;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
z-index: 1;
|
||||||
|
background: #fafafa;
|
||||||
|
padding: 10px;
|
||||||
|
border-radius: 2px;
|
||||||
|
color: var(--secondary80);
|
||||||
|
}
|
||||||
|
|
||||||
|
.preset-menu > span {
|
||||||
|
font-size: 13px;
|
||||||
|
text-transform: uppercase;
|
||||||
|
margin-top: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preset-menu li {
|
||||||
|
font-size: 14px;
|
||||||
|
margin-top: 13px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.preset-menu li:hover {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
li {
|
||||||
|
margin-right: 20px;
|
||||||
|
background: none;
|
||||||
|
border-radius: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* li button {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
border-radius: 5px;
|
||||||
|
padding: 13px;
|
||||||
|
outline: none;
|
||||||
|
cursor: pointer;
|
||||||
|
} */
|
||||||
|
|
||||||
|
/* .selected {
|
||||||
|
color: var(--button-text);
|
||||||
|
background: var(--background-button) !important;
|
||||||
|
} */
|
||||||
|
|
||||||
|
.open {
|
||||||
|
color: rgba(0, 85, 255, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.template-instance-label {
|
||||||
|
margin-left: 20px;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -135,7 +135,7 @@
|
||||||
background-color: var(--white);
|
background-color: var(--white);
|
||||||
height: calc(100vh - 49px);
|
height: calc(100vh - 49px);
|
||||||
padding: 0;
|
padding: 0;
|
||||||
overflow: hidden;
|
overflow: scroll;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
|
@ -152,7 +152,7 @@
|
||||||
grid-column: 3;
|
grid-column: 3;
|
||||||
background-color: var(--white);
|
background-color: var(--white);
|
||||||
min-height: 0px;
|
min-height: 0px;
|
||||||
overflow-y: hidden;
|
overflow-y: scroll;
|
||||||
}
|
}
|
||||||
|
|
||||||
.components-nav-page {
|
.components-nav-page {
|
||||||
|
@ -235,7 +235,6 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.components-list-container {
|
.components-list-container {
|
||||||
overflow: auto;
|
|
||||||
padding: 20px 0px 0 0;
|
padding: 20px 0px 0 0;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -0,0 +1,162 @@
|
||||||
|
export default {
|
||||||
|
categories: [
|
||||||
|
{
|
||||||
|
name: 'Basic',
|
||||||
|
isCategory: true,
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
_component: "@budibase/standard-components/container",
|
||||||
|
name: 'Container',
|
||||||
|
description: 'This component contains things within itself',
|
||||||
|
icon: 'ri-layout-row-fill',
|
||||||
|
commonProps: {},
|
||||||
|
children: []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Text',
|
||||||
|
description: 'This is a simple text component',
|
||||||
|
icon: 'ri-t-box-fill',
|
||||||
|
commonProps: {},
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
_component: '@budibase/standard-components/heading',
|
||||||
|
name: 'Headline',
|
||||||
|
description: "A component for displaying heading text",
|
||||||
|
icon: "ri-heading",
|
||||||
|
props: {
|
||||||
|
type: {
|
||||||
|
type: "options",
|
||||||
|
options: ["h1", "h2", "h3", "h4", "h5", "h6"],
|
||||||
|
default: "h1",
|
||||||
|
},
|
||||||
|
text: "string",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
_component: '@budibase/standard-components/text',
|
||||||
|
name: 'Paragraph',
|
||||||
|
description: "A component for displaying paragraph text.",
|
||||||
|
icon: 'ri-paragraph',
|
||||||
|
props: {}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Input',
|
||||||
|
description: "These components handle user input.",
|
||||||
|
icon: 'ri-edit-box-line',
|
||||||
|
commonProps: {},
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
_component: "@budibase/standard-components/textfield",
|
||||||
|
name: "Textfield",
|
||||||
|
description: "A textfield component that allows the user to input text.",
|
||||||
|
icon: 'ri-edit-box-line',
|
||||||
|
props: {}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
_component: "@budibase/standard-components/checkbox",
|
||||||
|
name: "Checkbox",
|
||||||
|
description: "A selectable checkbox component",
|
||||||
|
icon: 'ri-checkbox-line',
|
||||||
|
props: {}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
_component: "@budibase/standard-components/radiobutton",
|
||||||
|
name: "Radiobutton",
|
||||||
|
description: "A selectable radiobutton component",
|
||||||
|
icon: 'ri-radio-button-line',
|
||||||
|
props: {}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
_component: "@budibase/standard-components/select",
|
||||||
|
name: "Select",
|
||||||
|
description: "A select component for choosing from different options",
|
||||||
|
icon: 'ri-file-list-line',
|
||||||
|
props: {}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
_component: "@budibase/standard-components/button",
|
||||||
|
name: 'Button',
|
||||||
|
description: 'A basic html button that is ready for styling',
|
||||||
|
icon: 'ri-radio-button-fill',
|
||||||
|
commonProps: {},
|
||||||
|
children: []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
_component: "@budibase/standard-components/icon",
|
||||||
|
name: 'Icon',
|
||||||
|
description: 'A basic component for displaying icons',
|
||||||
|
icon: 'ri-sun-fill',
|
||||||
|
commonProps: {},
|
||||||
|
children: []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
_component: "@budibase/standard-components/link",
|
||||||
|
name: 'Link',
|
||||||
|
description: 'A basic link component for internal and external links',
|
||||||
|
icon: 'ri-link',
|
||||||
|
commonProps: {},
|
||||||
|
children: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Blocks',
|
||||||
|
isCategory: true,
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
_component: "@budibase/materialdesign-components/BasicCard",
|
||||||
|
name: 'Card',
|
||||||
|
description: 'A basic card component that can contain content and actions.',
|
||||||
|
icon: 'ri-layout-bottom-line',
|
||||||
|
commonProps: {},
|
||||||
|
children: []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Login',
|
||||||
|
description: 'A component that automatically generates a login screen for your app.',
|
||||||
|
icon: 'ri-login-box-fill',
|
||||||
|
commonProps: {},
|
||||||
|
children: []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Navigation Bar",
|
||||||
|
_component: "@budibase/standard-components/Navigation",
|
||||||
|
description: "A component for handling the navigation within your app.",
|
||||||
|
icon: "ri-navigation-fill",
|
||||||
|
commonProps: {},
|
||||||
|
children: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Data',
|
||||||
|
isCategory: true,
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
name: 'Table',
|
||||||
|
description: 'A component that generates a table from your data.',
|
||||||
|
icon: 'ri-archive-drawer-fill',
|
||||||
|
commonProps: {},
|
||||||
|
children: []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Form',
|
||||||
|
description: 'A component that generates a form from your data.',
|
||||||
|
icon: 'ri-file-edit-fill',
|
||||||
|
commonProps: {},
|
||||||
|
component: "@budibase/materialdesign-components/Form",
|
||||||
|
template: {
|
||||||
|
component: "@budibase/materialdesign-components/Form",
|
||||||
|
description: "Form for saving a record",
|
||||||
|
name: "@budibase/materialdesign-components/recordForm",
|
||||||
|
},
|
||||||
|
children: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
]
|
||||||
|
}
|
|
@ -0,0 +1,75 @@
|
||||||
|
{
|
||||||
|
"categories": [
|
||||||
|
{
|
||||||
|
"name": "Basic",
|
||||||
|
"components": [
|
||||||
|
{
|
||||||
|
"component": "Container",
|
||||||
|
"description": "This component contains things within itself",
|
||||||
|
"icon": "ri-layout-row-fill",
|
||||||
|
"commonProps": {},
|
||||||
|
"type": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"component": "Text",
|
||||||
|
"description": "This is a simple text component",
|
||||||
|
"icon": "ri-t-box-fill",
|
||||||
|
"commonProps": {
|
||||||
|
},
|
||||||
|
"type": [
|
||||||
|
{
|
||||||
|
"_component": "@budibase/standard-components/header",
|
||||||
|
"name": "Headline",
|
||||||
|
"icon": "headline",
|
||||||
|
"props": {
|
||||||
|
"type": {
|
||||||
|
"type": "options",
|
||||||
|
"options": [
|
||||||
|
"h1",
|
||||||
|
"h2"
|
||||||
|
],
|
||||||
|
"default": "h1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"_component": "@budibase/standard-components/text",
|
||||||
|
"name": "Paragraph",
|
||||||
|
"icon": "paragraph",
|
||||||
|
"props": {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"component": "Button",
|
||||||
|
"description": "A basic html button that is ready for styling",
|
||||||
|
"icon": "ri-radio-button-fill",
|
||||||
|
"commonProps": {},
|
||||||
|
"type": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"component": "Icon",
|
||||||
|
"description": "A basic component for displaying icons",
|
||||||
|
"icon": "ri-sun-fill",
|
||||||
|
"commonProps": {},
|
||||||
|
"type": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"component": "Avatar",
|
||||||
|
"description": "A basic component for rendering an avatar",
|
||||||
|
"icon": "ri-user-smile-fill",
|
||||||
|
"commonProps": {},
|
||||||
|
"type": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"component": "Link",
|
||||||
|
"description": "A basic link component for internal and external links",
|
||||||
|
"icon": "ri-link",
|
||||||
|
"commonProps": {},
|
||||||
|
"type": []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
|
@ -5,7 +5,8 @@ var getNewRecord = function getNewRecord(schema, modelName) {
|
||||||
|
|
||||||
var record = {
|
var record = {
|
||||||
_id: (0, _shortid.generate)(),
|
_id: (0, _shortid.generate)(),
|
||||||
modelId: model._id };
|
modelId: model._id
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
for (var field in model.schema.properties) {
|
for (var field in model.schema.properties) {
|
||||||
|
|
|
@ -115,6 +115,21 @@
|
||||||
},
|
},
|
||||||
"tags": []
|
"tags": []
|
||||||
},
|
},
|
||||||
|
"BasicCard": {
|
||||||
|
"name": "BasicCard",
|
||||||
|
"description": "This is a basic card",
|
||||||
|
"props": {
|
||||||
|
"heading": "string",
|
||||||
|
"subheading": "string",
|
||||||
|
"content": "string",
|
||||||
|
"imageUrl": "string",
|
||||||
|
"button1Text": "string",
|
||||||
|
"button2Text": "string",
|
||||||
|
"cardClick": "event",
|
||||||
|
"button1Click": "event",
|
||||||
|
"button2Click": "event"
|
||||||
|
}
|
||||||
|
},
|
||||||
"Card": {
|
"Card": {
|
||||||
"name": "Card",
|
"name": "Card",
|
||||||
"description": "A Material Card container. Accepts CardHeader, CardBody and CardFooter as possible children",
|
"description": "A Material Card container. Accepts CardHeader, CardBody and CardFooter as possible children",
|
||||||
|
|
|
@ -0,0 +1,67 @@
|
||||||
|
<script>
|
||||||
|
export let heading = ""
|
||||||
|
export let subheading = ""
|
||||||
|
export let content = ""
|
||||||
|
export let imageUrl = ""
|
||||||
|
export let button1Text = ""
|
||||||
|
export let button2Text = ""
|
||||||
|
export let cardClick = () => {}
|
||||||
|
export let button1Click = () => {}
|
||||||
|
export let button2Click = () => {}
|
||||||
|
|
||||||
|
$: showImage = !!imageUrl
|
||||||
|
$: showButton1 = !!button1Text
|
||||||
|
$: showButton2 = !!button2Text
|
||||||
|
$: showButtons = !!showButton1 && !!showButton2
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="mdc-card" on:click={cardClick}>
|
||||||
|
<div class="mdc-card__primary-action demo-card__primary-action" tabindex="0">
|
||||||
|
{#if showImage}
|
||||||
|
<div
|
||||||
|
class="mdc-card__media mdc-card__media--16-9 demo-card__media"
|
||||||
|
style="background-image: url("{imageUrl}");" />
|
||||||
|
{/if}
|
||||||
|
<div class="pad">
|
||||||
|
<div class="demo-card__primary">
|
||||||
|
<h2 class="demo-card__title mdc-typography mdc-typography--headline6">
|
||||||
|
{heading}
|
||||||
|
</h2>
|
||||||
|
<h3
|
||||||
|
class="demo-card__subtitle mdc-typography mdc-typography--subtitle2">
|
||||||
|
{subheading}
|
||||||
|
</h3>
|
||||||
|
</div>
|
||||||
|
<div class="demo-card__secondary mdc-typography mdc-typography--body2">
|
||||||
|
{content}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{#if showButtons}
|
||||||
|
<div class="mdc-card__actions">
|
||||||
|
<div class="mdc-card__action-buttons">
|
||||||
|
{#if showButton1}
|
||||||
|
<button class="mdc-button mdc-card__action mdc-card__action--button">
|
||||||
|
<span class="mdc-button__ripple" on:click={button1Click} />
|
||||||
|
{button1Text}
|
||||||
|
</button>
|
||||||
|
{/if}
|
||||||
|
{#if showButton2}
|
||||||
|
<button
|
||||||
|
class="mdc-button mdc-card__action mdc-card__action--button"
|
||||||
|
on:click={button2Click}>
|
||||||
|
<span class="mdc-button__ripple" />
|
||||||
|
{button2Text}
|
||||||
|
</button>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.pad {
|
||||||
|
padding: 10px;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -1,5 +1,6 @@
|
||||||
import "./_styles.scss"
|
import "./_styles.scss"
|
||||||
export { default as Card } from "./Card.svelte"
|
export { default as Card } from "./Card.svelte"
|
||||||
|
export { default as BasicCard } from "./BasicCard.svelte"
|
||||||
export { default as CardBody } from "./CardBody.svelte"
|
export { default as CardBody } from "./CardBody.svelte"
|
||||||
export { default as CardFooter } from "./CardFooter.svelte"
|
export { default as CardFooter } from "./CardFooter.svelte"
|
||||||
export { default as CardHeader } from "./CardHeader.svelte"
|
export { default as CardHeader } from "./CardHeader.svelte"
|
||||||
|
|
|
@ -22,7 +22,7 @@ export { Menu } from "./Menu"
|
||||||
export { Select } from "./Select"
|
export { Select } from "./Select"
|
||||||
export { DatePicker } from "./DatePicker"
|
export { DatePicker } from "./DatePicker"
|
||||||
export { IconButton } from "./IconButton"
|
export { IconButton } from "./IconButton"
|
||||||
export { Card, CardHeader, CardImage, CardBody, CardFooter } from "./Card"
|
export { Card, CardHeader, CardImage, CardBody, CardFooter, BasicCard } from "./Card"
|
||||||
export { Dialog, DialogHeader, DialogContent, DialogActions } from "./Dialog"
|
export { Dialog, DialogHeader, DialogContent, DialogActions } from "./Dialog"
|
||||||
export { Switch } from "./Switch"
|
export { Switch } from "./Switch"
|
||||||
export { Slider } from "./Slider"
|
export { Slider } from "./Slider"
|
||||||
|
|
|
@ -10,7 +10,7 @@ exports.create = async function(ctx) {
|
||||||
by_type: {
|
by_type: {
|
||||||
map: function(doc) {
|
map: function(doc) {
|
||||||
emit([doc.type], doc._id);
|
emit([doc.type], doc._id);
|
||||||
}
|
}.toString()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -5,7 +5,7 @@ exports.create = async function(ctx) {
|
||||||
// await couchdb.db.create(instanceName);
|
// await couchdb.db.create(instanceName);
|
||||||
|
|
||||||
const { clientId, applicationId } = ctx.params;
|
const { clientId, applicationId } = ctx.params;
|
||||||
const db = new CouchDB(ctx.config)(instanceName);
|
const db = new CouchDB(instanceName);
|
||||||
await db.put({
|
await db.put({
|
||||||
_id: "_design/database",
|
_id: "_design/database",
|
||||||
metadata: {
|
metadata: {
|
||||||
|
@ -17,7 +17,7 @@ exports.create = async function(ctx) {
|
||||||
map: function(doc) {
|
map: function(doc) {
|
||||||
emit([doc.type], doc._id);
|
emit([doc.type], doc._id);
|
||||||
}.toString()
|
}.toString()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -46,7 +46,7 @@ exports.destroy = async function(ctx) {
|
||||||
const clientDb = new CouchDB(metadata.clientId);
|
const clientDb = new CouchDB(metadata.clientId);
|
||||||
const budibaseApp = await clientDb.get(metadata.applicationId);
|
const budibaseApp = await clientDb.get(metadata.applicationId);
|
||||||
budibaseApp.instances = budibaseApp.instances.filter(instance => instance !== ctx.params.instanceId);
|
budibaseApp.instances = budibaseApp.instances.filter(instance => instance !== ctx.params.instanceId);
|
||||||
await clientDb.put(budibaseApp);
|
const updatedApp = await clientDb.put(budibaseApp);
|
||||||
|
|
||||||
ctx.body = {
|
ctx.body = {
|
||||||
message: `Instance Database ${ctx.params.instanceId} successfully destroyed.`,
|
message: `Instance Database ${ctx.params.instanceId} successfully destroyed.`,
|
||||||
|
|
|
@ -24,9 +24,12 @@ exports.save = async function(ctx) {
|
||||||
if (existingRecord) {
|
if (existingRecord) {
|
||||||
const response = await db.put({ ...record, _id: existingRecord._id });
|
const response = await db.put({ ...record, _id: existingRecord._id });
|
||||||
ctx.body = {
|
ctx.body = {
|
||||||
message: "Record updated successfully.",
|
message: `${model.name} updated successfully.`,
|
||||||
status: 200,
|
status: 200,
|
||||||
record: response
|
record: {
|
||||||
|
...record,
|
||||||
|
...response
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -40,7 +43,7 @@ exports.save = async function(ctx) {
|
||||||
|
|
||||||
ctx.body = record
|
ctx.body = record
|
||||||
ctx.status = 200
|
ctx.status = 200
|
||||||
ctx.message = `${model.name} ${record._rev ? "updated" : "created"} successfully`
|
ctx.message = `${model.name} created successfully`
|
||||||
}
|
}
|
||||||
|
|
||||||
exports.fetch = async function(ctx) {
|
exports.fetch = async function(ctx) {
|
||||||
|
@ -62,5 +65,5 @@ exports.find = async function(ctx) {
|
||||||
exports.destroy = async function(ctx) {
|
exports.destroy = async function(ctx) {
|
||||||
const databaseId = ctx.params.instanceId;
|
const databaseId = ctx.params.instanceId;
|
||||||
const db = new CouchDB(databaseId)
|
const db = new CouchDB(databaseId)
|
||||||
ctx.body = await db.destroy(ctx.params.recordId, ctx.params.revId);
|
ctx.body = await db.remove(ctx.params.recordId, ctx.params.revId);
|
||||||
};
|
};
|
||||||
|
|
|
@ -20,8 +20,8 @@ describe("/applications", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
afterAll(async () => {
|
afterAll(async () => {
|
||||||
await destroyDatabase(CLIENT_DB_ID)
|
|
||||||
server.close();
|
server.close();
|
||||||
|
await destroyDatabase(CLIENT_DB_ID)
|
||||||
})
|
})
|
||||||
|
|
||||||
describe("create", () => {
|
describe("create", () => {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
const supertest = require("supertest");
|
const supertest = require("supertest");
|
||||||
const app = require("../../../../app");
|
const app = require("../../../../app");
|
||||||
const { createClientDatabase, destroyDatabase } = require("./couchTestUtils")
|
const { createClientDatabase, destroyClientDatabase } = require("./couchTestUtils")
|
||||||
|
|
||||||
|
|
||||||
const CLIENT_DB_ID = "client-testing";
|
const CLIENT_DB_ID = "client-testing";
|
||||||
|
@ -8,6 +8,7 @@ const CLIENT_DB_ID = "client-testing";
|
||||||
describe("/clients", () => {
|
describe("/clients", () => {
|
||||||
let request;
|
let request;
|
||||||
let server;
|
let server;
|
||||||
|
let db;
|
||||||
|
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
server = await app({
|
server = await app({
|
||||||
|
@ -24,7 +25,7 @@ describe("/clients", () => {
|
||||||
|
|
||||||
describe("create", () => {
|
describe("create", () => {
|
||||||
afterEach(async () => {
|
afterEach(async () => {
|
||||||
await destroyDatabase(CLIENT_DB_ID);
|
await destroyClientDatabase();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns a success message when the client database is successfully created", done => {
|
it("returns a success message when the client database is successfully created", done => {
|
||||||
|
@ -43,7 +44,7 @@ describe("/clients", () => {
|
||||||
|
|
||||||
describe("destroy", () => {
|
describe("destroy", () => {
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await createClientDatabase();
|
db = await createClientDatabase();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns a success message when the client database is successfully destroyed", async done => {
|
it("returns a success message when the client database is successfully destroyed", async done => {
|
||||||
|
|
|
@ -1,11 +1,7 @@
|
||||||
const couchdb = require("../../../../db")
|
const CouchDB = require("../../../../db");
|
||||||
const createClientDb = require("../../../../db/initialiseClientDb")
|
const CLIENT_DB_ID = "client-testing";
|
||||||
|
|
||||||
const CLIENT_DB_ID = "client-testing"
|
|
||||||
const TEST_APP_ID = "test-app";
|
const TEST_APP_ID = "test-app";
|
||||||
|
|
||||||
exports.destroyDatabase = couchdb.db.destroy;
|
|
||||||
|
|
||||||
exports.createModel = async (instanceId, model) => {
|
exports.createModel = async (instanceId, model) => {
|
||||||
model = model || {
|
model = model || {
|
||||||
"name": "TestModel",
|
"name": "TestModel",
|
||||||
|
@ -15,8 +11,8 @@ exports.createModel = async (instanceId, model) => {
|
||||||
"name": { "type": "string" }
|
"name": { "type": "string" }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const db = couchdb.db.use(instanceId);
|
const db = new CouchDB(instanceId);
|
||||||
const response = await db.insert(model);
|
const response = await db.post(model);
|
||||||
|
|
||||||
const designDoc = await db.get("_design/database");
|
const designDoc = await db.get("_design/database");
|
||||||
designDoc.views = {
|
designDoc.views = {
|
||||||
|
@ -29,7 +25,7 @@ exports.createModel = async (instanceId, model) => {
|
||||||
}`
|
}`
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
await db.insert(designDoc, designDoc._id);
|
await db.put(designDoc);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...response,
|
...response,
|
||||||
|
@ -37,20 +33,36 @@ exports.createModel = async (instanceId, model) => {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
exports.createClientDatabase = async () =>
|
exports.createClientDatabase = async () => {
|
||||||
await createClientDb({
|
const db = new CouchDB(CLIENT_DB_ID);
|
||||||
database: "couch",
|
|
||||||
clientId: CLIENT_DB_ID,
|
|
||||||
})
|
|
||||||
|
|
||||||
exports.destroyClientDatabase = async () => await couchdb.db.destroy(CLIENT_DB_ID);
|
await db.put({
|
||||||
|
_id: "_design/client",
|
||||||
|
views: {
|
||||||
|
by_type: {
|
||||||
|
map: function(doc) {
|
||||||
|
emit([doc.type], doc._id);
|
||||||
|
}
|
||||||
|
}.toString()
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
await db.put({
|
||||||
|
_id: TEST_APP_ID,
|
||||||
|
type: "app",
|
||||||
|
instances: []
|
||||||
|
});
|
||||||
|
|
||||||
|
return db;
|
||||||
|
}
|
||||||
|
|
||||||
|
exports.destroyClientDatabase = async () => new CouchDB(CLIENT_DB_ID).destroy();
|
||||||
|
|
||||||
exports.createInstanceDatabase = async instanceId => {
|
exports.createInstanceDatabase = async instanceId => {
|
||||||
await couchdb.db.create(instanceId);
|
const db = new CouchDB(instanceId);
|
||||||
|
|
||||||
const db = couchdb.db.use(instanceId);
|
await db.put({
|
||||||
|
_id: "_design/database",
|
||||||
await db.insert({
|
|
||||||
metadata: {
|
metadata: {
|
||||||
clientId: CLIENT_DB_ID,
|
clientId: CLIENT_DB_ID,
|
||||||
applicationId: TEST_APP_ID
|
applicationId: TEST_APP_ID
|
||||||
|
@ -59,17 +71,17 @@ exports.createInstanceDatabase = async instanceId => {
|
||||||
by_type: {
|
by_type: {
|
||||||
map: function(doc) {
|
map: function(doc) {
|
||||||
emit([doc.type], doc._id);
|
emit([doc.type], doc._id);
|
||||||
}
|
}.toString()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, '_design/database');
|
});
|
||||||
|
|
||||||
return instanceId;
|
return db;
|
||||||
}
|
}
|
||||||
|
|
||||||
exports.insertDocument = async (databaseId, document) => {
|
exports.insertDocument = async (databaseId, document) => {
|
||||||
const { id, ...documentFields } = document;
|
const { id, ...documentFields } = document;
|
||||||
await couchdb.db.use(databaseId).insert(documentFields, id);
|
await new CouchDB(databaseId).put({ _id: id, ...documentFields });
|
||||||
}
|
}
|
||||||
|
|
||||||
exports.createSchema = async (request, instanceId, schema) => {
|
exports.createSchema = async (request, instanceId, schema) => {
|
||||||
|
|
|
@ -2,7 +2,6 @@ const supertest = require("supertest");
|
||||||
const app = require("../../../../app");
|
const app = require("../../../../app");
|
||||||
const {
|
const {
|
||||||
createInstanceDatabase,
|
createInstanceDatabase,
|
||||||
destroyDatabase,
|
|
||||||
createClientDatabase,
|
createClientDatabase,
|
||||||
destroyClientDatabase
|
destroyClientDatabase
|
||||||
} = require("./couchTestUtils");
|
} = require("./couchTestUtils");
|
||||||
|
@ -29,13 +28,13 @@ describe("/instances", () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
describe("create", () => {
|
describe("create", () => {
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await createClientDatabase();
|
clientDb = await createClientDatabase();
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(async () => {
|
afterEach(async () => {
|
||||||
await destroyClientDatabase();
|
await destroyClientDatabase();
|
||||||
await destroyDatabase(TEST_INSTANCE_ID);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns a success message when the instance database is successfully created", done => {
|
it("returns a success message when the instance database is successfully created", done => {
|
||||||
|
@ -55,7 +54,7 @@ describe("/instances", () => {
|
||||||
describe("destroy", () => {
|
describe("destroy", () => {
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await createClientDatabase();
|
await createClientDatabase();
|
||||||
await createInstanceDatabase(TEST_INSTANCE_ID);
|
instanceDb = await createInstanceDatabase(TEST_INSTANCE_ID);
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(async () => {
|
afterEach(async () => {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
const supertest = require("supertest");
|
const supertest = require("supertest");
|
||||||
const app = require("../../../../app");
|
const app = require("../../../../app");
|
||||||
const { createInstanceDatabase, createModel, destroyDatabase } = require("./couchTestUtils");
|
const { createInstanceDatabase, createModel } = require("./couchTestUtils");
|
||||||
|
|
||||||
|
|
||||||
const TEST_INSTANCE_ID = "testing-123";
|
const TEST_INSTANCE_ID = "testing-123";
|
||||||
|
@ -23,12 +23,14 @@ describe("/models", () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
describe("create", () => {
|
describe("create", () => {
|
||||||
|
let db;
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await createInstanceDatabase(TEST_INSTANCE_ID);
|
db = await createInstanceDatabase(TEST_INSTANCE_ID);
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(async () => {
|
afterEach(async () => {
|
||||||
await destroyDatabase(TEST_INSTANCE_ID);
|
await db.destroy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns a success message when the model is successfully created", done => {
|
it("returns a success message when the model is successfully created", done => {
|
||||||
|
@ -54,14 +56,15 @@ describe("/models", () => {
|
||||||
|
|
||||||
describe("fetch", () => {
|
describe("fetch", () => {
|
||||||
let testModel;
|
let testModel;
|
||||||
|
let db;
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await createInstanceDatabase(TEST_INSTANCE_ID);
|
db = await createInstanceDatabase(TEST_INSTANCE_ID);
|
||||||
testModel = await createModel(TEST_INSTANCE_ID);
|
testModel = await createModel(TEST_INSTANCE_ID);
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(async () => {
|
afterEach(async () => {
|
||||||
await destroyDatabase(TEST_INSTANCE_ID);
|
await db.destroy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns all the models for that instance in the response body", done => {
|
it("returns all the models for that instance in the response body", done => {
|
||||||
|
@ -81,14 +84,15 @@ describe("/models", () => {
|
||||||
|
|
||||||
describe("destroy", () => {
|
describe("destroy", () => {
|
||||||
let testModel;
|
let testModel;
|
||||||
|
let db;
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await createInstanceDatabase(TEST_INSTANCE_ID);
|
db = await createInstanceDatabase(TEST_INSTANCE_ID);
|
||||||
testModel = await createModel(TEST_INSTANCE_ID);
|
testModel = await createModel(TEST_INSTANCE_ID);
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(async () => {
|
afterEach(async () => {
|
||||||
await destroyDatabase(TEST_INSTANCE_ID);
|
await db.destroy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns a success response when a model is deleted.", done => {
|
it("returns a success response when a model is deleted.", done => {
|
||||||
|
|
|
@ -1,14 +1,23 @@
|
||||||
const supertest = require("supertest");
|
const supertest = require("supertest");
|
||||||
const app = require("../../../../app");
|
const app = require("../../../../app");
|
||||||
const { createInstanceDatabase, createModel, destroyDatabase } = require("./couchTestUtils");
|
const { createInstanceDatabase, createModel } = require("./couchTestUtils");
|
||||||
const { getNewRecord } = require("../../../../../common/lib/records/getNewRecord")
|
|
||||||
const { testSchema } = require("../../../../../common/lib/testUtils/testSchema")
|
|
||||||
|
|
||||||
const TEST_INSTANCE_ID = "testing-123";
|
const TEST_INSTANCE_ID = "testing-123";
|
||||||
|
|
||||||
|
const CONTACT_MODEL = {
|
||||||
|
"name": "Contact",
|
||||||
|
"type": "model",
|
||||||
|
"key": "name",
|
||||||
|
"schema": {
|
||||||
|
"name": { "type": "string" },
|
||||||
|
"age": { "type": "number" }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
describe("/records", () => {
|
describe("/records", () => {
|
||||||
let request;
|
let request;
|
||||||
let server;
|
let server;
|
||||||
|
let db;
|
||||||
|
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
server = await app({
|
server = await app({
|
||||||
|
@ -24,21 +33,23 @@ describe("/records", () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
describe("save, load, update, delete", () => {
|
describe("save, load, update, delete", () => {
|
||||||
const schema = testSchema()
|
let record;
|
||||||
|
let model;
|
||||||
|
|
||||||
beforeAll(async () => {
|
beforeEach(async () => {
|
||||||
await createInstanceDatabase(TEST_INSTANCE_ID);
|
db = await createInstanceDatabase(TEST_INSTANCE_ID);
|
||||||
await createModel(TEST_INSTANCE_ID, schema.findModel("Contact"))
|
model = await createModel(TEST_INSTANCE_ID, CONTACT_MODEL)
|
||||||
|
record = {
|
||||||
|
name: "Test Contact",
|
||||||
|
status: "new",
|
||||||
|
modelId: model.id
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
afterAll(async () => {
|
afterEach(async () => {
|
||||||
await destroyDatabase(TEST_INSTANCE_ID);
|
await db.destroy();
|
||||||
});
|
});
|
||||||
|
|
||||||
let record = getNewRecord(schema, "Contact")
|
|
||||||
record.name = "Test Contact"
|
|
||||||
record.Status = "new"
|
|
||||||
|
|
||||||
it("returns a success message when the record is created", done => {
|
it("returns a success message when the record is created", done => {
|
||||||
request
|
request
|
||||||
.post(`/api/${TEST_INSTANCE_ID}/records`)
|
.post(`/api/${TEST_INSTANCE_ID}/records`)
|
||||||
|
@ -47,69 +58,65 @@ describe("/records", () => {
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
.expect(200)
|
.expect(200)
|
||||||
.end(async (err, res) => {
|
.end(async (err, res) => {
|
||||||
expect(res.res.statusMessage.toLowerCase()).toEqual("contact created successfully")
|
expect(res.res.statusMessage).toEqual("Contact created successfully")
|
||||||
expect(res.body.name).toEqual("Test Contact")
|
expect(res.body.name).toEqual("Test Contact")
|
||||||
expect(res.body._rev).toBeDefined()
|
expect(res.body._rev).toBeDefined()
|
||||||
record = res.body
|
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
|
|
||||||
it("updates a record successfully", async () => {
|
it("updates a record successfully", async () => {
|
||||||
record.name = "Updated Name"
|
const existing = await db.post(record);
|
||||||
|
|
||||||
const res = await request
|
const res = await request
|
||||||
.post(`/api/${TEST_INSTANCE_ID}/records`)
|
.post(`/api/${TEST_INSTANCE_ID}/records`)
|
||||||
.send(record)
|
.send({
|
||||||
|
_id: existing.id,
|
||||||
|
_rev: existing.rev,
|
||||||
|
modelId: model.id,
|
||||||
|
name: "Updated Name",
|
||||||
|
})
|
||||||
.set("Accept", "application/json")
|
.set("Accept", "application/json")
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
.expect(200)
|
.expect(200)
|
||||||
|
|
||||||
expect(res.res.statusMessage.toLowerCase()).toEqual("contact updated successfully")
|
expect(res.body.message).toEqual("Contact updated successfully.")
|
||||||
expect(res.body.name).toEqual("Updated Name")
|
expect(res.body.record.name).toEqual("Updated Name")
|
||||||
record = res.body
|
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should load a record", async () => {
|
it("should load a record", async () => {
|
||||||
|
const existing = await db.post(record);
|
||||||
|
|
||||||
const res = await request
|
const res = await request
|
||||||
.get(`/api/${TEST_INSTANCE_ID}/records/${record._id}`)
|
.get(`/api/${TEST_INSTANCE_ID}/records/${existing.id}`)
|
||||||
.set("Accept", "application/json")
|
.set("Accept", "application/json")
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
.expect(200)
|
.expect(200)
|
||||||
expect(res.body).toEqual(record)
|
|
||||||
|
expect(res.body).toEqual({
|
||||||
|
...record,
|
||||||
|
_id: existing.id,
|
||||||
|
_rev: existing.rev
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should list all records for given modelId", async () => {
|
it("should list all records for given modelId", async () => {
|
||||||
const secondRecord = getNewRecord(schema, "Contact")
|
const newRecord = {
|
||||||
secondRecord.name = "Second Contact"
|
modelId: model.id,
|
||||||
secondRecord.Status = "new"
|
name: "Second Contact",
|
||||||
|
status: "new"
|
||||||
|
}
|
||||||
|
|
||||||
await request
|
await db.post(newRecord);
|
||||||
.post(`/api/${TEST_INSTANCE_ID}/records`)
|
|
||||||
.send(secondRecord)
|
|
||||||
.set("Accept", "application/json")
|
|
||||||
|
|
||||||
const res = await request
|
const res = await request
|
||||||
.get(`/api/${TEST_INSTANCE_ID}/${record.modelId}/records`)
|
.get(`/api/${TEST_INSTANCE_ID}/all_${newRecord.modelId}/records`)
|
||||||
.set("Accept", "application/json")
|
.set("Accept", "application/json")
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
.expect(200)
|
.expect(200)
|
||||||
|
|
||||||
expect(res.body.length).toBe(2)
|
expect(res.body.length).toBe(1)
|
||||||
console.log(JSON.stringify(res.body, undefined, 2))
|
expect(res.body[0].name).toEqual(newRecord.name);
|
||||||
expect(res.body.find(c => c.name === record.name)).toBeDefined()
|
|
||||||
expect(res.body.find(c => c.name === secondRecord.name)).toBeDefined()
|
|
||||||
})
|
|
||||||
|
|
||||||
it("should return 404 when load, after a delete", async () => {
|
|
||||||
await request
|
|
||||||
.delete(`/api/${TEST_INSTANCE_ID}/records/${record._id}/${record._rev}`)
|
|
||||||
.expect(200)
|
|
||||||
|
|
||||||
await request
|
|
||||||
.get(`/api/${TEST_INSTANCE_ID}/records/${record._id}`)
|
|
||||||
.set("Accept", "application/json")
|
|
||||||
.expect('Content-Type', /json/)
|
|
||||||
.expect(404)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
it("load should return 404 when record does not exist", async () => {
|
it("load should return 404 when record does not exist", async () => {
|
||||||
|
@ -119,6 +126,5 @@ describe("/records", () => {
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
.expect(404)
|
.expect(404)
|
||||||
})
|
})
|
||||||
|
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
const supertest = require("supertest");
|
const supertest = require("supertest");
|
||||||
const app = require("../../../../app");
|
const app = require("../../../../app");
|
||||||
const {
|
const {
|
||||||
createInstanceDatabase,
|
createInstanceDatabase
|
||||||
destroyDatabase
|
|
||||||
} = require("./couchTestUtils");
|
} = require("./couchTestUtils");
|
||||||
|
|
||||||
|
|
||||||
|
@ -29,12 +28,14 @@ describe("/users", () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
describe("fetch", () => {
|
describe("fetch", () => {
|
||||||
|
let db;
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await createInstanceDatabase(TEST_INSTANCE_ID);
|
db = await createInstanceDatabase(TEST_INSTANCE_ID);
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(async () => {
|
afterEach(async () => {
|
||||||
await destroyDatabase(TEST_INSTANCE_ID);
|
await db.destroy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns a list of users from an instance db", done => {
|
it("returns a list of users from an instance db", done => {
|
||||||
|
@ -52,18 +53,20 @@ describe("/users", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("create", () => {
|
describe("create", () => {
|
||||||
|
let db;
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await createInstanceDatabase(TEST_INSTANCE_ID);
|
db = await createInstanceDatabase(TEST_INSTANCE_ID);
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(async () => {
|
afterEach(async () => {
|
||||||
await destroyDatabase(TEST_INSTANCE_ID);
|
await db.destroy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns a success message when a user is successfully created", done => {
|
it("returns a success message when a user is successfully created", done => {
|
||||||
request
|
request
|
||||||
.post(`/api/${TEST_INSTANCE_ID}/users`)
|
.post(`/api/${TEST_INSTANCE_ID}/users`)
|
||||||
.send({ name: "John" })
|
.send({ name: "Bill", username: "bill1", password: "password" })
|
||||||
.set("Accept", "application/json")
|
.set("Accept", "application/json")
|
||||||
.expect('Content-Type', /json/)
|
.expect('Content-Type', /json/)
|
||||||
.expect(200)
|
.expect(200)
|
||||||
|
|
|
@ -8,6 +8,7 @@ const TEST_INSTANCE_ID = "testing-123";
|
||||||
describe("/views", () => {
|
describe("/views", () => {
|
||||||
let request;
|
let request;
|
||||||
let server;
|
let server;
|
||||||
|
let db;
|
||||||
|
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
server = await app({
|
server = await app({
|
||||||
|
@ -24,11 +25,11 @@ describe("/views", () => {
|
||||||
|
|
||||||
describe("create", () => {
|
describe("create", () => {
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await createInstanceDatabase(TEST_INSTANCE_ID);
|
db = await createInstanceDatabase(TEST_INSTANCE_ID);
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(async () => {
|
afterEach(async () => {
|
||||||
await destroyDatabase(TEST_INSTANCE_ID);
|
await db.destroy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns a success message when the view is successfully created", done => {
|
it("returns a success message when the view is successfully created", done => {
|
||||||
|
@ -55,13 +56,15 @@ describe("/views", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("fetch", () => {
|
describe("fetch", () => {
|
||||||
|
let db;
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await createInstanceDatabase(TEST_INSTANCE_ID);
|
db = await createInstanceDatabase(TEST_INSTANCE_ID);
|
||||||
await createModel(TEST_INSTANCE_ID);
|
await createModel(TEST_INSTANCE_ID);
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(async () => {
|
afterEach(async () => {
|
||||||
await destroyDatabase(TEST_INSTANCE_ID);
|
await db.destroy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns a list of all the views that exist in the instance database", done => {
|
it("returns a list of all the views that exist in the instance database", done => {
|
||||||
|
|
|
@ -1,16 +1,32 @@
|
||||||
{
|
{
|
||||||
"_lib": "./dist/index.js",
|
"_lib": "./dist/index.js",
|
||||||
"_templates" : {
|
"_templates": {
|
||||||
"saveRecordButton" : {
|
"saveRecordButton": {
|
||||||
"description": "Save record button",
|
"description": "Save record button",
|
||||||
"component": "button"
|
"component": "button"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"button" : {
|
"Navigation": {
|
||||||
|
"name": "Navigation",
|
||||||
|
"description": "A basic header navigation component",
|
||||||
|
"props": {
|
||||||
|
"logoUrl": "string",
|
||||||
|
"title": "string",
|
||||||
|
"backgroundColor": "string",
|
||||||
|
"color": "string",
|
||||||
|
"borderWidth": "string",
|
||||||
|
"borderColor": "string",
|
||||||
|
"borderStyle": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"button": {
|
||||||
"name": "Button",
|
"name": "Button",
|
||||||
"description": "an html <button />",
|
"description": "an html <button />",
|
||||||
"props": {
|
"props": {
|
||||||
"contentText": { "type": "string", "default": "Button" },
|
"contentText": {
|
||||||
|
"type": "string",
|
||||||
|
"default": "Button"
|
||||||
|
},
|
||||||
"className": "string",
|
"className": "string",
|
||||||
"disabled": "bool",
|
"disabled": "bool",
|
||||||
"onClick": "event",
|
"onClick": "event",
|
||||||
|
@ -22,76 +38,110 @@
|
||||||
"hoverBackground": "string",
|
"hoverBackground": "string",
|
||||||
"hoverBorder": "string"
|
"hoverBorder": "string"
|
||||||
},
|
},
|
||||||
"tags": ["layout"],
|
"tags": [
|
||||||
|
"layout"
|
||||||
|
],
|
||||||
"presets": {
|
"presets": {
|
||||||
"primary": {
|
"primary": {
|
||||||
"contentText": "Primary Button Preset",
|
"contentText": "Primary Button Preset",
|
||||||
"color": "papayawhip",
|
"color": "papayawhip",
|
||||||
"padding": "20px",
|
"padding": "20px",
|
||||||
"background": "blue"
|
"background": "blue"
|
||||||
},
|
},
|
||||||
"secondary": {
|
"secondary": {
|
||||||
"contentText": "Secondary Button Preset",
|
"contentText": "Secondary Button Preset",
|
||||||
"color": "rebeccapurple",
|
"color": "rebeccapurple",
|
||||||
"padding": "10px",
|
"padding": "10px",
|
||||||
"background": "#fff",
|
"background": "#fff",
|
||||||
"border": "1px solid red"
|
"border": "1px solid red"
|
||||||
},
|
},
|
||||||
"error": {
|
"error": {
|
||||||
"contentText": "ERROR",
|
"contentText": "ERROR",
|
||||||
"color": "red",
|
"color": "red",
|
||||||
"padding": "10px",
|
"padding": "10px",
|
||||||
"border": "1px solid red"
|
"border": "1px solid red"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"login" : {
|
"login": {
|
||||||
"name": "Login Control",
|
"name": "Login Control",
|
||||||
"description": "A control that accepts username, password an also handles password resets",
|
"description": "A control that accepts username, password an also handles password resets",
|
||||||
"props" : {
|
"props": {
|
||||||
"logo": "asset",
|
"logo": "asset",
|
||||||
"loginRedirect": "string",
|
"loginRedirect": "string",
|
||||||
"usernameLabel": {"type":"string", "default": "Username"},
|
"usernameLabel": {
|
||||||
"passwordLabel": {"type":"string", "default": "Password"},
|
"type": "string",
|
||||||
"loginButtonLabel": {"type":"string", "default": "Login"},
|
"default": "Username"
|
||||||
|
},
|
||||||
|
"passwordLabel": {
|
||||||
|
"type": "string",
|
||||||
|
"default": "Password"
|
||||||
|
},
|
||||||
|
"loginButtonLabel": {
|
||||||
|
"type": "string",
|
||||||
|
"default": "Login"
|
||||||
|
},
|
||||||
"buttonClass": "string",
|
"buttonClass": "string",
|
||||||
"inputClass": "string"
|
"inputClass": "string"
|
||||||
},
|
},
|
||||||
"tags": ["login", "credentials", "password", "logon"]
|
"tags": [
|
||||||
|
"login",
|
||||||
|
"credentials",
|
||||||
|
"password",
|
||||||
|
"logon"
|
||||||
|
]
|
||||||
},
|
},
|
||||||
"input" : {
|
"input": {
|
||||||
"name": "Input",
|
"name": "Input",
|
||||||
"description": "An HTML input",
|
"description": "An HTML input",
|
||||||
"props" : {
|
"props": {
|
||||||
"value": "string",
|
"value": "string",
|
||||||
"type": {
|
"type": {
|
||||||
"type":"options",
|
"type": "options",
|
||||||
"options":[
|
"options": [
|
||||||
"text", "password", "checkbox", "color",
|
"text",
|
||||||
"date", "datetime-local", "email",
|
"password",
|
||||||
"file", "hidden", "image", "month", "number",
|
"checkbox",
|
||||||
"radio", "range", "reset", "search", "submit",
|
"color",
|
||||||
"tel", "time", "week"],
|
"date",
|
||||||
"default":"text"
|
"datetime-local",
|
||||||
|
"email",
|
||||||
|
"file",
|
||||||
|
"hidden",
|
||||||
|
"image",
|
||||||
|
"month",
|
||||||
|
"number",
|
||||||
|
"radio",
|
||||||
|
"range",
|
||||||
|
"reset",
|
||||||
|
"search",
|
||||||
|
"submit",
|
||||||
|
"tel",
|
||||||
|
"time",
|
||||||
|
"week"
|
||||||
|
],
|
||||||
|
"default": "text"
|
||||||
},
|
},
|
||||||
"onChange": "event",
|
"onChange": "event",
|
||||||
"className": "string"
|
"className": "string"
|
||||||
},
|
},
|
||||||
"tags": ["form"]
|
"tags": [
|
||||||
|
"form"
|
||||||
|
]
|
||||||
},
|
},
|
||||||
"select" : {
|
"select": {
|
||||||
"name": "Select",
|
"name": "Select",
|
||||||
"description": "An HTML <select> (dropdown)",
|
"description": "An HTML <select> (dropdown)",
|
||||||
"props" : {
|
"props": {
|
||||||
"value": "string",
|
"value": "string",
|
||||||
"className": "string"
|
"className": "string"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"option" : {
|
"option": {
|
||||||
"name": "Option",
|
"name": "Option",
|
||||||
"description": "An HTML <option>, to be used with <select>",
|
"description": "An HTML <option>, to be used with <select>",
|
||||||
"children": false,
|
"children": false,
|
||||||
"props" : {
|
"props": {
|
||||||
"value": "string",
|
"value": "string",
|
||||||
"text": "string"
|
"text": "string"
|
||||||
}
|
}
|
||||||
|
@ -100,27 +150,32 @@
|
||||||
"name": "Text",
|
"name": "Text",
|
||||||
"description": "stylable block of text",
|
"description": "stylable block of text",
|
||||||
"children": false,
|
"children": false,
|
||||||
"props" : {
|
"props": {
|
||||||
"text": "string",
|
"text": "string",
|
||||||
"font": "string",
|
"font": "string",
|
||||||
|
"fontSize": "string",
|
||||||
"color": "string",
|
"color": "string",
|
||||||
"textAlign": {
|
"textAlign": {
|
||||||
"type": "options",
|
"type": "options",
|
||||||
"default":"inline",
|
"default": "inline",
|
||||||
"options": [
|
"options": [
|
||||||
"left", "center", "right"
|
"left",
|
||||||
|
"center",
|
||||||
|
"right"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"verticalAlign": {
|
"verticalAlign": {
|
||||||
"type": "options",
|
"type": "options",
|
||||||
"default":"inline",
|
"default": "inline",
|
||||||
"options": [
|
"options": [
|
||||||
"top", "middle", "bottom"
|
"top",
|
||||||
|
"middle",
|
||||||
|
"bottom"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"formattingTag": {
|
"formattingTag": {
|
||||||
"type": "options",
|
"type": "options",
|
||||||
"default":"none",
|
"default": "none",
|
||||||
"options": [
|
"options": [
|
||||||
"none",
|
"none",
|
||||||
"<b> - bold",
|
"<b> - bold",
|
||||||
|
@ -136,7 +191,46 @@
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"tags": ["div", "container"]
|
"tags": [
|
||||||
|
"div",
|
||||||
|
"container"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"textfield": {
|
||||||
|
"name": "Textfield",
|
||||||
|
"description": "A component that allows the user to input text.",
|
||||||
|
"props": {
|
||||||
|
"label": "string",
|
||||||
|
"value": "string",
|
||||||
|
"onchange": "event"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"checkbox": {
|
||||||
|
"name": "Checkbox",
|
||||||
|
"description": "A selectable checkbox component",
|
||||||
|
"props": {
|
||||||
|
"label": "string",
|
||||||
|
"checked": "bool",
|
||||||
|
"value": "string",
|
||||||
|
"onchange": "event"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"radiobutton": {
|
||||||
|
"name": "Radiobutton",
|
||||||
|
"description": "A selectable radiobutton component",
|
||||||
|
"props": {
|
||||||
|
"label": "string",
|
||||||
|
"checked": "bool",
|
||||||
|
"value": "string",
|
||||||
|
"onchange": "event"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"icon": {
|
||||||
|
"description": "A HTML icon tag",
|
||||||
|
"props": {
|
||||||
|
"icon": "string",
|
||||||
|
"fontSize": "string"
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"link": {
|
"link": {
|
||||||
"description": "an HTML anchor <a> tag",
|
"description": "an HTML anchor <a> tag",
|
||||||
|
@ -163,11 +257,11 @@
|
||||||
"container": {
|
"container": {
|
||||||
"name": "Container",
|
"name": "Container",
|
||||||
"description": "An element that contains and lays out other elements. e.g. <div>, <header> etc",
|
"description": "An element that contains and lays out other elements. e.g. <div>, <header> etc",
|
||||||
"props" : {
|
"props": {
|
||||||
"className":"string",
|
"className": "string",
|
||||||
"onLoad": "event",
|
"onLoad": "event",
|
||||||
"type": {
|
"type": {
|
||||||
"type": "options",
|
"type": "options",
|
||||||
"options": [
|
"options": [
|
||||||
"article",
|
"article",
|
||||||
"aside",
|
"aside",
|
||||||
|
@ -190,7 +284,7 @@
|
||||||
"borderWidth": "string",
|
"borderWidth": "string",
|
||||||
"borderColor": "string",
|
"borderColor": "string",
|
||||||
"borderStyle": {
|
"borderStyle": {
|
||||||
"type":"options",
|
"type": "options",
|
||||||
"options": [
|
"options": [
|
||||||
"none",
|
"none",
|
||||||
"solid",
|
"solid",
|
||||||
|
@ -204,20 +298,31 @@
|
||||||
],
|
],
|
||||||
"default": "none"
|
"default": "none"
|
||||||
}
|
}
|
||||||
|
|
||||||
},
|
},
|
||||||
"container": true,
|
"container": true,
|
||||||
"tags": ["div", "container", "layout"]
|
"tags": [
|
||||||
|
"div",
|
||||||
|
"container",
|
||||||
|
"layout"
|
||||||
|
]
|
||||||
},
|
},
|
||||||
"heading": {
|
"heading": {
|
||||||
"name": "Heading",
|
"name": "Heading",
|
||||||
"description": "An HTML H1 - H6 tag",
|
"description": "An HTML H1 - H6 tag",
|
||||||
"props" : {
|
"props": {
|
||||||
"className":"string",
|
"className": "string",
|
||||||
|
"text": "string",
|
||||||
"type": {
|
"type": {
|
||||||
"type": "options",
|
"type": "options",
|
||||||
"default": "h1",
|
"default": "h1",
|
||||||
"options": ["h1","h2","h3","h4","h5","h6"]
|
"options": [
|
||||||
|
"h1",
|
||||||
|
"h2",
|
||||||
|
"h3",
|
||||||
|
"h4",
|
||||||
|
"h5",
|
||||||
|
"h6"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"tags": []
|
"tags": []
|
||||||
|
@ -225,15 +330,15 @@
|
||||||
"thead": {
|
"thead": {
|
||||||
"name": "TableHead",
|
"name": "TableHead",
|
||||||
"description": "an HTML <thead> tab",
|
"description": "an HTML <thead> tab",
|
||||||
"props" : {
|
"props": {
|
||||||
"className":"string"
|
"className": "string"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"tbody": {
|
"tbody": {
|
||||||
"name": "TableBody",
|
"name": "TableBody",
|
||||||
"description": "an HTML <tbody> tab",
|
"description": "an HTML <tbody> tab",
|
||||||
"props" : {
|
"props": {
|
||||||
"className":"string"
|
"className": "string"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -0,0 +1,11 @@
|
||||||
|
<script>
|
||||||
|
import Input from "./Input.svelte"
|
||||||
|
export let _bb
|
||||||
|
|
||||||
|
export let label = ""
|
||||||
|
export let checked = false
|
||||||
|
export let value = ""
|
||||||
|
export let onchange = () => {}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<Input type="checkbox" {_bb} {checked} {label} {value} {onchange} />
|
|
@ -2,22 +2,22 @@
|
||||||
export let className = ""
|
export let className = ""
|
||||||
export let type
|
export let type
|
||||||
export let _bb
|
export let _bb
|
||||||
|
export let text = ""
|
||||||
let containerElement
|
let containerElement
|
||||||
|
|
||||||
$: containerElement && _bb.attachChildren(containerElement)
|
$: containerElement && !text && _bb.attachChildren(containerElement)
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if type === 'h1'}
|
{#if type === 'h1'}
|
||||||
<h1 class={className} bind:this={containerElement} />
|
<h1 class={className} bind:this={containerElement}>{text}</h1>
|
||||||
{:else if type === 'h2'}
|
{:else if type === 'h2'}
|
||||||
<h2 class={className} bind:this={containerElement} />
|
<h2 class={className} bind:this={containerElement}>{text}</h2>
|
||||||
{:else if type === 'h3'}
|
{:else if type === 'h3'}
|
||||||
<h3 class={className} bind:this={containerElement} />
|
<h3 class={className} bind:this={containerElement}>{text}</h3>
|
||||||
{:else if type === 'h4'}
|
{:else if type === 'h4'}
|
||||||
<h4 class={className} bind:this={containerElement} />
|
<h4 class={className} bind:this={containerElement}>{text}</h4>
|
||||||
{:else if type === 'h5'}
|
{:else if type === 'h5'}
|
||||||
<h5 class={className} bind:this={containerElement} />
|
<h5 class={className} bind:this={containerElement}>{text}</h5>
|
||||||
{:else if type === 'h6'}
|
{:else if type === 'h6'}
|
||||||
<h6 class={className} bind:this={containerElement} />
|
<h6 class={className} bind:this={containerElement}>{text}</h6>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
<script>
|
||||||
|
export let icon = ""
|
||||||
|
export let fontSize = "1em"
|
||||||
|
export let _bb
|
||||||
|
|
||||||
|
$: style = { fontSize }
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<i class={icon} {style} />
|
|
@ -1,7 +1,10 @@
|
||||||
<script>
|
<script>
|
||||||
|
export let id = ""
|
||||||
export let value = ""
|
export let value = ""
|
||||||
export let className = ""
|
export let className = ""
|
||||||
export let type = "text"
|
export let type = "text"
|
||||||
|
export let label = ""
|
||||||
|
export let checked = false
|
||||||
|
|
||||||
export let _bb
|
export let _bb
|
||||||
|
|
||||||
|
@ -14,4 +17,5 @@
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<input class={className} {type} {value} on:change={onchange} />
|
<label for={id}>{label}</label>
|
||||||
|
<input {id} class={className} {type} {value} {checked} on:change={onchange} />
|
||||||
|
|
|
@ -0,0 +1,75 @@
|
||||||
|
<script>
|
||||||
|
import { cssVars, createClasses } from "./cssVars"
|
||||||
|
|
||||||
|
export let className = ""
|
||||||
|
export let onLoad
|
||||||
|
export let backgroundColor
|
||||||
|
export let color
|
||||||
|
export let borderWidth
|
||||||
|
export let borderColor
|
||||||
|
export let borderStyle
|
||||||
|
export let logoUrl
|
||||||
|
export let title
|
||||||
|
export let _bb
|
||||||
|
|
||||||
|
let itemContainer
|
||||||
|
let hasLoaded
|
||||||
|
let currentChildren
|
||||||
|
|
||||||
|
$: cssVariables = {
|
||||||
|
backgroundColor,
|
||||||
|
color,
|
||||||
|
borderWidth,
|
||||||
|
borderColor,
|
||||||
|
borderStyle,
|
||||||
|
}
|
||||||
|
|
||||||
|
$: {
|
||||||
|
if (itemContainer) {
|
||||||
|
_bb.attachChildren(itemContainer)
|
||||||
|
if (!hasLoaded) {
|
||||||
|
_bb.call(onLoad)
|
||||||
|
hasLoaded = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<nav use:cssVars={cssVariables}>
|
||||||
|
<a href="/">
|
||||||
|
<img class="logo" alt="logo" src={logoUrl} height="30" />
|
||||||
|
<span>{title}</span>
|
||||||
|
</a>
|
||||||
|
<div class="menu-items" bind:this={itemContainer} />
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
nav {
|
||||||
|
color: var(--color);
|
||||||
|
background-color: var(--backgroundColor);
|
||||||
|
align-items: center;
|
||||||
|
display: flex;
|
||||||
|
font-weight: bold;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 20px 0 20px 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav > a {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
font-size: 1.5em;
|
||||||
|
color: var(--color);
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav a img {
|
||||||
|
border-radius: 15px;
|
||||||
|
margin-right: 15px;
|
||||||
|
}
|
||||||
|
.menu-items {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
.menu-items > :global(*) {
|
||||||
|
margin: 0 10px;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,11 @@
|
||||||
|
<script>
|
||||||
|
import Input from "./Input.svelte"
|
||||||
|
export let _bb
|
||||||
|
|
||||||
|
export let label = ""
|
||||||
|
export let checked = false
|
||||||
|
export let value = ""
|
||||||
|
export let onchange = () => {}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<Input type="radio" {_bb} {checked} {label} {value} {onchange} />
|
|
@ -7,29 +7,26 @@
|
||||||
export let formattingTag = ""
|
export let formattingTag = ""
|
||||||
|
|
||||||
export let font = ""
|
export let font = ""
|
||||||
|
export let fontSize = "1em"
|
||||||
export let textAlign = ""
|
export let textAlign = ""
|
||||||
export let verticalAlign = ""
|
export let verticalAlign = ""
|
||||||
export let color = ""
|
export let color = ""
|
||||||
|
|
||||||
export let _bb
|
export let _bb
|
||||||
|
|
||||||
let style = ""
|
|
||||||
|
|
||||||
const isTag = tag => (formattingTag || "").indexOf(tag) > -1
|
const isTag = tag => (formattingTag || "").indexOf(tag) > -1
|
||||||
|
|
||||||
$: {
|
$: style = buildStyle({
|
||||||
style = buildStyle({
|
font: `${fontSize} ${font}`,
|
||||||
font,
|
verticalAlign,
|
||||||
verticalAlign,
|
color,
|
||||||
color,
|
"text-align": textAlign,
|
||||||
"text-align": textAlign,
|
"vertical-align": verticalAlign,
|
||||||
"vertical-align": verticalAlign,
|
})
|
||||||
})
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if isTag('none')}
|
{#if isTag('none')}
|
||||||
<span>{text}</span>
|
<span {style}>{text}</span>
|
||||||
{:else if isTag('<b>')}
|
{:else if isTag('<b>')}
|
||||||
<b class={className} {style}>{text}</b>
|
<b class={className} {style}>{text}</b>
|
||||||
{:else if isTag('<strong>')}
|
{:else if isTag('<strong>')}
|
||||||
|
|
|
@ -0,0 +1,10 @@
|
||||||
|
<script>
|
||||||
|
import Input from "./Input.svelte"
|
||||||
|
export let _bb
|
||||||
|
|
||||||
|
export let label = ""
|
||||||
|
export let value = ""
|
||||||
|
export let onchange = () => {}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<Input type="text" {_bb} {label} {value} {onchange} />
|
|
@ -3,9 +3,14 @@ export { default as text } from "./Text.svelte"
|
||||||
export { default as heading } from "./Heading.svelte"
|
export { default as heading } from "./Heading.svelte"
|
||||||
export { default as input } from "./Input.svelte"
|
export { default as input } from "./Input.svelte"
|
||||||
export { default as select } from "./Select.svelte"
|
export { default as select } from "./Select.svelte"
|
||||||
|
export { default as textfield } from "./Textfield.svelte"
|
||||||
|
export { default as checkbox } from "./Checkbox.svelte"
|
||||||
|
export { default as radiobutton } from "./Radiobutton.svelte"
|
||||||
export { default as option } from "./Option.svelte"
|
export { default as option } from "./Option.svelte"
|
||||||
export { default as button } from "./Button.svelte"
|
export { default as button } from "./Button.svelte"
|
||||||
export { default as login } from "./Login.svelte"
|
export { default as login } from "./Login.svelte"
|
||||||
export { default as saveRecordButton } from "./Templates/saveRecordButton"
|
export { default as saveRecordButton } from "./Templates/saveRecordButton"
|
||||||
export { default as link } from "./Link.svelte"
|
export { default as link } from "./Link.svelte"
|
||||||
export { default as image } from "./Image.svelte"
|
export { default as image } from "./Image.svelte"
|
||||||
|
export { default as icon } from "./Icon.svelte"
|
||||||
|
export { default as Navigation } from "./Navigation.svelte"
|
||||||
|
|
65
yarn.lock
65
yarn.lock
|
@ -918,6 +918,27 @@
|
||||||
dependencies:
|
dependencies:
|
||||||
"@types/node" ">= 8"
|
"@types/node" ">= 8"
|
||||||
|
|
||||||
|
"@rollup/plugin-json@^4.0.2":
|
||||||
|
version "4.0.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/@rollup/plugin-json/-/plugin-json-4.0.2.tgz#482185ee36ac7dd21c346e2dbcc22ffed0c6f2d6"
|
||||||
|
integrity sha512-t4zJMc98BdH42mBuzjhQA7dKh0t4vMJlUka6Fz0c+iO5IVnWaEMiYBy1uBj9ruHZzXBW23IPDGL9oCzBkQ9Udg==
|
||||||
|
dependencies:
|
||||||
|
"@rollup/pluginutils" "^3.0.4"
|
||||||
|
|
||||||
|
"@rollup/pluginutils@^3.0.4":
|
||||||
|
version "3.0.9"
|
||||||
|
resolved "https://registry.yarnpkg.com/@rollup/pluginutils/-/pluginutils-3.0.9.tgz#aa6adca2c45e5a1b950103a999e3cddfe49fd775"
|
||||||
|
integrity sha512-TLZavlfPAZYI7v33wQh4mTP6zojne14yok3DNSLcjoG/Hirxfkonn6icP5rrNWRn8nZsirJBFFpijVOJzkUHDg==
|
||||||
|
dependencies:
|
||||||
|
"@types/estree" "0.0.39"
|
||||||
|
estree-walker "^1.0.1"
|
||||||
|
micromatch "^4.0.2"
|
||||||
|
|
||||||
|
"@types/estree@0.0.39":
|
||||||
|
version "0.0.39"
|
||||||
|
resolved "https://registry.yarnpkg.com/@types/estree/-/estree-0.0.39.tgz#e177e699ee1b8c22d23174caaa7422644389509f"
|
||||||
|
integrity sha512-EYNwp3bU+98cpU4lAWYYL7Zz+2gryWH1qbdDTidVd6hkiR6weksdbMadyXKXNPEkQFhXM+hVO9ZygomHXp+AIw==
|
||||||
|
|
||||||
"@types/node@>= 8":
|
"@types/node@>= 8":
|
||||||
version "13.5.3"
|
version "13.5.3"
|
||||||
resolved "https://registry.yarnpkg.com/@types/node/-/node-13.5.3.tgz#37f1f539b7535b9fb4ef77d59db1847a837b7f17"
|
resolved "https://registry.yarnpkg.com/@types/node/-/node-13.5.3.tgz#37f1f539b7535b9fb4ef77d59db1847a837b7f17"
|
||||||
|
@ -1216,6 +1237,13 @@ braces@^2.3.1:
|
||||||
split-string "^3.0.2"
|
split-string "^3.0.2"
|
||||||
to-regex "^3.0.1"
|
to-regex "^3.0.1"
|
||||||
|
|
||||||
|
braces@^3.0.1:
|
||||||
|
version "3.0.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107"
|
||||||
|
integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==
|
||||||
|
dependencies:
|
||||||
|
fill-range "^7.0.1"
|
||||||
|
|
||||||
btoa-lite@^1.0.0:
|
btoa-lite@^1.0.0:
|
||||||
version "1.0.0"
|
version "1.0.0"
|
||||||
resolved "https://registry.yarnpkg.com/btoa-lite/-/btoa-lite-1.0.0.tgz#337766da15801210fdd956c22e9c6891ab9d0337"
|
resolved "https://registry.yarnpkg.com/btoa-lite/-/btoa-lite-1.0.0.tgz#337766da15801210fdd956c22e9c6891ab9d0337"
|
||||||
|
@ -2065,6 +2093,11 @@ estree-walker@^0.6.1:
|
||||||
resolved "https://registry.yarnpkg.com/estree-walker/-/estree-walker-0.6.1.tgz#53049143f40c6eb918b23671d1fe3219f3a1b362"
|
resolved "https://registry.yarnpkg.com/estree-walker/-/estree-walker-0.6.1.tgz#53049143f40c6eb918b23671d1fe3219f3a1b362"
|
||||||
integrity sha512-SqmZANLWS0mnatqbSfRP5g8OXZC12Fgg1IwNtLsyHDzJizORW4khDfjPqJZsemPWBB2uqykUah5YpQ6epsqC/w==
|
integrity sha512-SqmZANLWS0mnatqbSfRP5g8OXZC12Fgg1IwNtLsyHDzJizORW4khDfjPqJZsemPWBB2uqykUah5YpQ6epsqC/w==
|
||||||
|
|
||||||
|
estree-walker@^1.0.1:
|
||||||
|
version "1.0.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/estree-walker/-/estree-walker-1.0.1.tgz#31bc5d612c96b704106b477e6dd5d8aa138cb700"
|
||||||
|
integrity sha512-1fMXF3YP4pZZVozF8j/ZLfvnR8NSIljt56UhbZ5PeeDmmGHpgpdwQt7ITlGvYaQukCvuBRMLEiKiYC+oeIg4cg==
|
||||||
|
|
||||||
esutils@^2.0.2:
|
esutils@^2.0.2:
|
||||||
version "2.0.3"
|
version "2.0.3"
|
||||||
resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64"
|
resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64"
|
||||||
|
@ -2222,6 +2255,13 @@ fill-range@^4.0.0:
|
||||||
repeat-string "^1.6.1"
|
repeat-string "^1.6.1"
|
||||||
to-regex-range "^2.1.0"
|
to-regex-range "^2.1.0"
|
||||||
|
|
||||||
|
fill-range@^7.0.1:
|
||||||
|
version "7.0.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40"
|
||||||
|
integrity sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==
|
||||||
|
dependencies:
|
||||||
|
to-regex-range "^5.0.1"
|
||||||
|
|
||||||
find-up@^1.0.0:
|
find-up@^1.0.0:
|
||||||
version "1.1.2"
|
version "1.1.2"
|
||||||
resolved "https://registry.yarnpkg.com/find-up/-/find-up-1.1.2.tgz#6b2e9822b1a2ce0a60ab64d610eccad53cb24d0f"
|
resolved "https://registry.yarnpkg.com/find-up/-/find-up-1.1.2.tgz#6b2e9822b1a2ce0a60ab64d610eccad53cb24d0f"
|
||||||
|
@ -2930,6 +2970,11 @@ is-number@^3.0.0:
|
||||||
dependencies:
|
dependencies:
|
||||||
kind-of "^3.0.2"
|
kind-of "^3.0.2"
|
||||||
|
|
||||||
|
is-number@^7.0.0:
|
||||||
|
version "7.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b"
|
||||||
|
integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==
|
||||||
|
|
||||||
is-obj@^1.0.0:
|
is-obj@^1.0.0:
|
||||||
version "1.0.1"
|
version "1.0.1"
|
||||||
resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-1.0.1.tgz#3e4729ac1f5fde025cd7d83a896dab9f4f67db0f"
|
resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-1.0.1.tgz#3e4729ac1f5fde025cd7d83a896dab9f4f67db0f"
|
||||||
|
@ -3464,6 +3509,14 @@ micromatch@^3.1.10:
|
||||||
snapdragon "^0.8.1"
|
snapdragon "^0.8.1"
|
||||||
to-regex "^3.0.2"
|
to-regex "^3.0.2"
|
||||||
|
|
||||||
|
micromatch@^4.0.2:
|
||||||
|
version "4.0.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.2.tgz#4fcb0999bf9fbc2fcbdd212f6d629b9a56c39259"
|
||||||
|
integrity sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==
|
||||||
|
dependencies:
|
||||||
|
braces "^3.0.1"
|
||||||
|
picomatch "^2.0.5"
|
||||||
|
|
||||||
mime-db@1.43.0:
|
mime-db@1.43.0:
|
||||||
version "1.43.0"
|
version "1.43.0"
|
||||||
resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.43.0.tgz#0a12e0502650e473d735535050e7c8f4eb4fae58"
|
resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.43.0.tgz#0a12e0502650e473d735535050e7c8f4eb4fae58"
|
||||||
|
@ -4174,6 +4227,11 @@ performance-now@^2.1.0:
|
||||||
resolved "https://registry.yarnpkg.com/performance-now/-/performance-now-2.1.0.tgz#6309f4e0e5fa913ec1c69307ae364b4b377c9e7b"
|
resolved "https://registry.yarnpkg.com/performance-now/-/performance-now-2.1.0.tgz#6309f4e0e5fa913ec1c69307ae364b4b377c9e7b"
|
||||||
integrity sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=
|
integrity sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=
|
||||||
|
|
||||||
|
picomatch@^2.0.5:
|
||||||
|
version "2.2.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.2.2.tgz#21f333e9b6b8eaff02468f5146ea406d345f4dad"
|
||||||
|
integrity sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==
|
||||||
|
|
||||||
pify@^2.0.0, pify@^2.3.0:
|
pify@^2.0.0, pify@^2.3.0:
|
||||||
version "2.3.0"
|
version "2.3.0"
|
||||||
resolved "https://registry.yarnpkg.com/pify/-/pify-2.3.0.tgz#ed141a6ac043a849ea588498e7dca8b15330e90c"
|
resolved "https://registry.yarnpkg.com/pify/-/pify-2.3.0.tgz#ed141a6ac043a849ea588498e7dca8b15330e90c"
|
||||||
|
@ -5141,6 +5199,13 @@ to-regex-range@^2.1.0:
|
||||||
is-number "^3.0.0"
|
is-number "^3.0.0"
|
||||||
repeat-string "^1.6.1"
|
repeat-string "^1.6.1"
|
||||||
|
|
||||||
|
to-regex-range@^5.0.1:
|
||||||
|
version "5.0.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4"
|
||||||
|
integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==
|
||||||
|
dependencies:
|
||||||
|
is-number "^7.0.0"
|
||||||
|
|
||||||
to-regex@^3.0.1, to-regex@^3.0.2:
|
to-regex@^3.0.1, to-regex@^3.0.2:
|
||||||
version "3.0.2"
|
version "3.0.2"
|
||||||
resolved "https://registry.yarnpkg.com/to-regex/-/to-regex-3.0.2.tgz#13cfdd9b336552f30b51f33a8ae1b42a7a7599ce"
|
resolved "https://registry.yarnpkg.com/to-regex/-/to-regex-3.0.2.tgz#13cfdd9b336552f30b51f33a8ae1b42a7a7599ce"
|
||||||
|
|
Loading…
Reference in New Issue