242 lines
5.9 KiB
JavaScript
242 lines
5.9 KiB
JavaScript
import jsdom, { JSDOM } from "jsdom"
|
|
import { loadBudibase } from "../src/index"
|
|
|
|
export const APP_ID = "TEST_APP_ID"
|
|
|
|
export const load = async (page, screens, url, host = "test.com") => {
|
|
screens = screens || []
|
|
url = url || "/"
|
|
|
|
const fullUrl = `http://${host}${url}`
|
|
const cookieJar = new jsdom.CookieJar()
|
|
const cookie = `${btoa("{}")}.${btoa(`{"appId":"${APP_ID}"}`)}.signature`
|
|
cookieJar.setCookie(
|
|
`budibase:${APP_ID}:local=${cookie};domain=${host};path=/`,
|
|
fullUrl,
|
|
{
|
|
looseMode: false,
|
|
},
|
|
() => {}
|
|
)
|
|
|
|
const dom = new JSDOM("<!DOCTYPE html><html><body></body><html>", {
|
|
url: fullUrl,
|
|
cookieJar,
|
|
})
|
|
|
|
autoAssignIds(page.props)
|
|
for (let s of screens) {
|
|
autoAssignIds(s.props)
|
|
}
|
|
setAppDef(dom.window, page, screens)
|
|
addWindowGlobals(dom.window, page, screens, {
|
|
hierarchy: {},
|
|
actions: [],
|
|
triggers: [],
|
|
})
|
|
setComponentCodeMeta(page, screens)
|
|
const app = await loadBudibase({
|
|
componentLibraries: allLibs(dom.window),
|
|
window: dom.window,
|
|
localStorage: createLocalStorage(),
|
|
})
|
|
return { dom, app }
|
|
}
|
|
|
|
const addWindowGlobals = (window, page, screens) => {
|
|
window["##BUDIBASE_FRONTEND_DEFINITION##"] = {
|
|
page,
|
|
screens,
|
|
}
|
|
}
|
|
|
|
export const makePage = props => ({ props })
|
|
export const makeScreen = (route, props) => ({ props, route })
|
|
|
|
export const timeout = ms => new Promise(resolve => setTimeout(resolve, ms))
|
|
|
|
export const walkComponentTree = (node, action) => {
|
|
action(node)
|
|
|
|
// works for nodes or props
|
|
const children = node.children || node._children
|
|
|
|
if (children) {
|
|
for (let child of children) {
|
|
walkComponentTree(child, action)
|
|
}
|
|
}
|
|
}
|
|
|
|
// this happens for real by the builder...
|
|
// ..this only assigns _ids when missing
|
|
const autoAssignIds = (props, count = 0) => {
|
|
if (!props._id) {
|
|
props._id = `auto_id_${count}`
|
|
}
|
|
if (props._children) {
|
|
for (let child of props._children) {
|
|
count += 1
|
|
autoAssignIds(child, count)
|
|
}
|
|
}
|
|
}
|
|
|
|
// any component with an id that include "based_on_store" is
|
|
// assumed to have code that depends on store value
|
|
const setComponentCodeMeta = (page, screens) => {
|
|
const setComponentCodeMeta_single = props => {
|
|
walkComponentTree(props, c => {
|
|
if (c._id.indexOf("based_on_store") >= 0) {
|
|
c._codeMeta = { dependsOnStore: true }
|
|
}
|
|
})
|
|
}
|
|
setComponentCodeMeta_single(page.props)
|
|
for (let s of screens || []) {
|
|
setComponentCodeMeta_single(s.props)
|
|
}
|
|
}
|
|
|
|
const setAppDef = (window, page, screens) => {
|
|
window["##BUDIBASE_FRONTEND_DEFINITION##"] = {
|
|
componentLibraries: [],
|
|
page,
|
|
screens,
|
|
}
|
|
}
|
|
|
|
const allLibs = window => ({
|
|
testlib: maketestlib(window),
|
|
})
|
|
|
|
const createLocalStorage = () => {
|
|
const data = {}
|
|
return {
|
|
getItem: key => data[key],
|
|
setItem: (key, value) => (data[key] = value),
|
|
}
|
|
}
|
|
|
|
const maketestlib = window => ({
|
|
div: function(opts) {
|
|
const node = window.document.createElement("DIV")
|
|
const defaultChild = window.document.createElement("DIV")
|
|
defaultChild.className = "default-child"
|
|
node.appendChild(defaultChild)
|
|
|
|
let currentProps = { ...opts.props }
|
|
let childNodes = []
|
|
|
|
const set = props => {
|
|
currentProps = Object.assign(currentProps, props)
|
|
node.className = currentProps.className || ""
|
|
if (currentProps._children && currentProps._children.length > 0) {
|
|
if (currentProps.append) {
|
|
for (let c of childNodes) {
|
|
node.removeChild(c)
|
|
}
|
|
const components = currentProps._bb.attachChildren(node, {
|
|
hydrate: false,
|
|
})
|
|
childNodes = components.map(c => c.component._element)
|
|
} else {
|
|
currentProps._bb.attachChildren(node)
|
|
}
|
|
}
|
|
}
|
|
|
|
this.$destroy = () => opts.target.removeChild(node)
|
|
|
|
this.$set = set
|
|
this._element = node
|
|
set(opts.props)
|
|
opts.target.appendChild(node)
|
|
},
|
|
|
|
h1: function(opts) {
|
|
const node = window.document.createElement("H1")
|
|
|
|
let currentProps = { ...opts.props }
|
|
|
|
const set = props => {
|
|
currentProps = Object.assign(currentProps, props)
|
|
if (currentProps.text) {
|
|
node.innerText = currentProps.text
|
|
}
|
|
}
|
|
|
|
this.$destroy = () => opts.target.removeChild(node)
|
|
|
|
this.$set = set
|
|
this._element = node
|
|
set(opts.props)
|
|
opts.target.appendChild(node)
|
|
},
|
|
|
|
button: function(opts) {
|
|
const node = window.document.createElement("BUTTON")
|
|
|
|
let currentProps = { ...opts.props }
|
|
|
|
const set = props => {
|
|
currentProps = Object.assign(currentProps, props)
|
|
if (currentProps.onClick) {
|
|
node.addEventListener("click", () => {
|
|
currentProps._bb.call("onClick")
|
|
})
|
|
}
|
|
}
|
|
|
|
this.$destroy = () => opts.target.removeChild(node)
|
|
|
|
this.$set = set
|
|
this._element = node
|
|
set(opts.props)
|
|
opts.target.appendChild(node)
|
|
},
|
|
|
|
list: function(opts) {
|
|
const node = window.document.createElement("DIV")
|
|
|
|
let currentProps = { ...opts.props }
|
|
|
|
const set = props => {
|
|
currentProps = Object.assign(currentProps, props)
|
|
if (currentProps._children && currentProps._children.length > 0) {
|
|
currentProps._bb.attachChildren(node, {
|
|
context: currentProps.data || {},
|
|
})
|
|
}
|
|
}
|
|
|
|
this.$destroy = () => opts.target.removeChild(node)
|
|
|
|
this.$set = set
|
|
this._element = node
|
|
set(opts.props)
|
|
opts.target.appendChild(node)
|
|
},
|
|
|
|
input: function(opts) {
|
|
const node = window.document.createElement("INPUT")
|
|
let currentProps = { ...opts.props }
|
|
|
|
const set = props => {
|
|
currentProps = Object.assign(currentProps, props)
|
|
opts.props._bb.setBinding("value", props.value)
|
|
}
|
|
|
|
node.addEventListener("change", e => {
|
|
opts.props._bb.setBinding("value", e.target.value)
|
|
})
|
|
|
|
this.$destroy = () => opts.target.removeChild(node)
|
|
|
|
this.$set = set
|
|
this._element = node
|
|
set(opts.props)
|
|
opts.target.appendChild(node)
|
|
},
|
|
})
|