budibase/packages/builder/src/builderStore/fetchBindableProperties.js

159 lines
5.1 KiB
JavaScript
Raw Normal View History

2020-08-04 17:11:46 +02:00
import { cloneDeep, difference, fill } from "lodash"
2020-08-04 12:10:02 +02:00
const stubBindings = [
{
// type: instance represents a bindable property of a component
type: "instance",
instance: {} /** a component instance **/,
// how the binding expression persists, and is used in the app at runtime
runtimeBinding: "state.<component instance Id>.<component property name>",
// how the binding exressions looks to the user of the builder
readableBinding: "<component instance name>",
},
{
type: "context",
2020-08-04 17:11:46 +02:00
instance: {
/** a component instance **/
},
2020-08-04 12:10:02 +02:00
// how the binding expression persists, and is used in the app at runtime
runtimeBinding: "context._parent.<key of model/record>",
// how the binding exressions looks to the user of the builder
readableBinding: "<component instance name>.<model/view name>.<key>",
},
]
2020-08-03 16:06:51 +02:00
export default function({ componentInstanceId, screen, components, models }) {
2020-08-04 17:11:46 +02:00
const walkResult = walk({
// cloning so we are free to mutate props (e.g. by adding _contexts)
instance: cloneDeep(screen.props),
targetId: componentInstanceId,
components,
models,
})
2020-08-03 16:06:51 +02:00
return [
2020-08-04 17:11:46 +02:00
...walkResult.bindableInstances
.filter(isInstanceInSharedContext(walkResult))
.map(componentInstanceToBindable(walkResult)),
...walkResult.target._contexts.map(contextToBindables(walkResult)).flat(),
2020-08-03 16:06:51 +02:00
]
}
2020-08-04 17:11:46 +02:00
const isInstanceInSharedContext = walkResult => i =>
// should cover
// - neither are in any context
// - both in same context
// - instance is in ancestor context of target
i.instance._contexts.length <= walkResult.target._contexts.length &&
difference(i.instance._contexts, walkResult.target._contexts).length === 0
2020-08-03 16:06:51 +02:00
// turns a component instance prop into binding expressions
// used by the UI
2020-08-04 17:11:46 +02:00
const componentInstanceToBindable = walkResult => i => {
const lastContext =
i.instance._contexts.length &&
i.instance._contexts[i.instance._contexts.length - 1]
const contextParentPath = lastContext
? getParentPath(walkResult, lastContext)
: ""
// if component is inside context, then the component lives
// in context at runtime (otherwise, in state)
const stateOrContext = lastContext ? "context" : "state"
return {
type: "instance",
instance: i.instance,
// how the binding expression persists, and is used in the app at runtime
runtimeBinding: `${stateOrContext}.${contextParentPath}${i.instance._id}.${i.prop}`,
// how the binding exressions looks to the user of the builder
readableBinding: `${i.instance._instanceName}`,
}
}
const contextToBindables = walkResult => c => {
const contextParentPath = getParentPath(walkResult, c)
return Object.keys(c.model.schema).map(k => ({
2020-08-03 16:06:51 +02:00
type: "context",
instance: c.instance,
// how the binding expression persists, and is used in the app at runtime
2020-08-04 17:11:46 +02:00
runtimeBinding: `context.${contextParentPath}data.${k}`,
2020-08-03 16:06:51 +02:00
// how the binding exressions looks to the user of the builder
2020-08-04 17:11:46 +02:00
readableBinding: `${c.instance._instanceName}.${c.model.name}.${k}`,
2020-08-03 16:06:51 +02:00
}))
}
2020-08-04 17:11:46 +02:00
const getParentPath = (walkResult, context) => {
// describes the number of "_parent" in the path
// clone array first so original array is not mtated
const contextParentNumber = [...walkResult.target._contexts]
.reverse()
.indexOf(context)
return (
new Array(contextParentNumber).fill("_parent").join(".") +
// trailing . if has parents
(contextParentNumber ? "." : "")
)
}
2020-08-03 16:06:51 +02:00
const walk = ({ instance, targetId, components, models, result }) => {
if (!result) {
result = {
target: null,
bindableInstances: [],
2020-08-04 17:11:46 +02:00
allContexts: [],
currentContexts: [],
2020-08-03 16:06:51 +02:00
}
}
2020-08-04 17:11:46 +02:00
if (!instance._contexts) instance._contexts = []
2020-08-03 16:06:51 +02:00
// "component" is the component definition (object in component.json)
const component = components[instance._component]
if (instance._id === targetId) {
// found it
result.target = instance
} else {
2020-08-04 12:10:02 +02:00
if (component.bindable) {
2020-08-03 16:06:51 +02:00
// pushing all components in here initially
// but this will not be correct, as some of
// these components will be in another context
// but we dont know this until the end of the walk
// so we will filter in another metod
result.bindableInstances.push({
instance,
2020-08-04 12:10:02 +02:00
prop: component.bindable,
2020-08-03 16:06:51 +02:00
})
}
}
2020-08-04 17:11:46 +02:00
2020-08-03 16:06:51 +02:00
// a component that provides context to it's children
const contextualInstance = component.context && instance[component.context]
if (contextualInstance) {
// add to currentContexts (ancestory of context)
// before walking children
2020-08-04 17:11:46 +02:00
const model = models.find(m => m._id === instance[component.context])
result.currentContexts.push({ instance, model })
2020-08-03 16:06:51 +02:00
}
2020-08-04 17:11:46 +02:00
const currentContexts = [...result.currentContexts]
2020-08-03 16:06:51 +02:00
for (let child of instance._children || []) {
2020-08-04 17:11:46 +02:00
// attaching _contexts of components, for eas comparison later
// these have been deep cloned above, so shouln't modify the
// original component instances
child._contexts = currentContexts
2020-08-04 12:10:02 +02:00
walk({ instance: child, targetId, components, models, result })
2020-08-03 16:06:51 +02:00
}
if (contextualInstance) {
// child walk done, remove from currentContexts
result.currentContexts.pop()
}
return result
}