Add proper hot reloading of app preview when styles change

This commit is contained in:
Andrew Kingston 2020-11-24 09:31:54 +00:00
parent aa38f1fe57
commit 853f5d8745
30 changed files with 180 additions and 130 deletions

View File

@ -1,5 +1,6 @@
<script> <script>
import { getContext, setContext } from "svelte" import { getContext, setContext } from "svelte"
import { writable } from "svelte/store"
import * as ComponentLibrary from "@budibase/standard-components" import * as ComponentLibrary from "@budibase/standard-components"
import Router from "./Router.svelte" import Router from "./Router.svelte"
@ -28,21 +29,23 @@
} }
// Extract component definition info // Extract component definition info
const componentName = extractComponentName(definition._component) $: componentName = extractComponentName(definition._component)
const constructor = getComponentConstructor(componentName) $: constructor = getComponentConstructor(componentName)
const componentProps = extractValidProps(definition) $: componentProps = extractValidProps(definition)
const dataContext = getContext("data") $: dataContext = getContext("data")
const enrichedProps = dataContext.actions.enrichDataBindings(componentProps) $: enrichedProps = dataContext.actions.enrichDataBindings(componentProps)
const children = definition._children $: children = definition._children
// Set contexts to be consumed by component // Set observable style context
setContext("style", { ...definition._styles, id: definition._id }) const styleStore = writable({})
setContext("style", styleStore)
$: styleStore.set({ ...definition._styles, id: definition._id })
</script> </script>
{#if constructor} {#if constructor}
<svelte:component this={constructor} {...enrichedProps}> <svelte:component this={constructor} {...enrichedProps}>
{#if children && children.length} {#if children && children.length}
{#each children as child} {#each children as child (child._id)}
<svelte:self definition={child} /> <svelte:self definition={child} />
{/each} {/each}
{/if} {/if}

View File

@ -1,25 +1,19 @@
<script> <script>
import { onMount, getContext, setContext } from "svelte" import { getContext, setContext } from "svelte"
import { createDataContextStore } from "../store" import { createDataContextStore } from "../store"
export let row export let row
// Get current contexts // Get current contexts
const dataContext = getContext("data") const dataContext = getContext("data")
const { id } = getContext("style") const styles = getContext("style")
// Clone current context to this context // Clone current context to this context
const newDataContext = createDataContextStore($dataContext) const newDataContext = createDataContextStore($dataContext)
setContext("data", newDataContext) setContext("data", newDataContext)
// Add additional layer to context // Add additional layer to context
let loaded = false $: newDataContext.actions.addContext(row, $styles.id)
onMount(() => {
newDataContext.actions.addContext(row, id)
loaded = true
})
</script> </script>
{#if loaded} <slot />
<slot />
{/if}

View File

@ -1,7 +1,7 @@
<script> <script>
import { getContext } from "svelte" import { getContext } from "svelte"
import Router from "svelte-spa-router" import Router from "svelte-spa-router"
import { routeStore, screenStore } from "../store" import { routeStore } from "../store"
import Screen from "./Screen.svelte" import Screen from "./Screen.svelte"
const { styleable } = getContext("sdk") const { styleable } = getContext("sdk")
@ -26,7 +26,7 @@
</script> </script>
{#if routerConfig} {#if routerConfig}
<div use:styleable={styles}> <div use:styleable={$styles}>
<Router on:routeLoading={onRouteLoading} routes={routerConfig} /> <Router on:routeLoading={onRouteLoading} routes={routerConfig} />
</div> </div>
{/if} {/if}

View File

@ -11,8 +11,11 @@
// Redirect to home page if no matching route // Redirect to home page if no matching route
$: screenDefinition == null && routeStore.actions.navigate("/") $: screenDefinition == null && routeStore.actions.navigate("/")
// Make a screen array so we can use keying to properly re-render each screen
$: screens = screenDefinition ? [screenDefinition] : []
</script> </script>
{#if screenDefinition} {#each screens as screen (screen._id)}
<Component definition={screenDefinition} /> <Component definition={screen} />
{/if} {/each}

View File

@ -1,16 +1,22 @@
import ClientApp from "./components/ClientApp.svelte" import ClientApp from "./components/ClientApp.svelte"
import { builderStore } from "./store"
let app let app
const loadBudibase = () => { const loadBudibase = () => {
// Destroy old app if one exists // Update builder store with any builder flags
if (app) { builderStore.set({
app.$destroy() inBuilder: !!window["##BUDIBASE_IN_BUILDER##"],
} page: window["##BUDIBASE_PREVIEW_PAGE##"],
// Create new app screen: window["##BUDIBASE_PREVIEW_SCREEN##"],
app = new ClientApp({
target: window.document.body,
}) })
// Create app if one hasn't been created yet
if (!app) {
app = new ClientApp({
target: window.document.body,
})
}
} }
// Attach to window so the HTML template can call this when it loads // Attach to window so the HTML template can call this when it loads

View File

@ -0,0 +1,12 @@
import { writable } from "svelte/store"
const createBuilderStore = () => {
const initialState = {
inBuilder: false,
page: null,
screen: null,
}
return writable(initialState)
}
export const builderStore = createBuilderStore()

View File

@ -2,3 +2,4 @@ export { authStore } from "./auth"
export { routeStore } from "./routes" export { routeStore } from "./routes"
export { screenStore } from "./screens" export { screenStore } from "./screens"
export { createDataContextStore } from "./dataContext" export { createDataContextStore } from "./dataContext"
export { builderStore } from "./builder"

View File

@ -1,5 +1,6 @@
import { writable, derived } from "svelte/store" import { writable, derived } from "svelte/store"
import { routeStore } from "./routes" import { routeStore } from "./routes"
import { builderStore } from "./builder"
import * as API from "../api" import * as API from "../api"
import { getAppId } from "../utils" import { getAppId } from "../utils"
@ -8,36 +9,37 @@ const createScreenStore = () => {
screens: [], screens: [],
page: {}, page: {},
}) })
const store = derived([config, routeStore], ([$config, $routeStore]) => { const store = derived(
const { screens, page } = $config [config, routeStore, builderStore],
const activeScreen = ([$config, $routeStore, $builderStore]) => {
screens.length === 1 let page
? screens[0] let activeScreen
: screens.find( if ($builderStore.inBuilder) {
// Use builder defined definitions if inside the builder preview
page = $builderStore.page
activeScreen = $builderStore.screen
} else {
// Otherwise find the correct screen by matching the current route
page = $config.page
const { screens } = $config
if (screens.length === 1) {
activeScreen = screens[0]
} else {
activeScreen = screens.find(
screen => screen.routing.route === $routeStore.activeRoute screen => screen.routing.route === $routeStore.activeRoute
) )
return { }
screens, }
page, return { page, activeScreen }
activeScreen,
} }
}) )
const fetchScreens = async () => { const fetchScreens = async () => {
let screens const appDefinition = await API.fetchAppDefinition(getAppId())
let page config.set({
const inBuilder = !!window["##BUDIBASE_IN_BUILDER##"] screens: appDefinition.screens,
if (inBuilder) { page: appDefinition.page,
// Load screen and page from the window object if in the builder })
screens = [window["##BUDIBASE_PREVIEW_SCREEN##"]]
page = window["##BUDIBASE_PREVIEW_PAGE##"]
} else {
// Otherwise load from API
const appDefinition = await API.fetchAppDefinition(getAppId())
screens = appDefinition.screens
page = appDefinition.page
}
config.set({ screens, page })
} }
return { return {

View File

@ -1,7 +1,13 @@
import { getContext } from "svelte"
import { get } from "svelte/store"
/**
* Helper to build a CSS string from a style object
*/
const buildStyleString = styles => { const buildStyleString = styles => {
let str = "" let str = ""
Object.entries(styles).forEach(([style, value]) => { Object.entries(styles).forEach(([style, value]) => {
if (style && value) { if (style && value != null) {
str += `${style}: ${value}; ` str += `${style}: ${value}; `
} }
}) })
@ -12,35 +18,52 @@ const buildStyleString = styles => {
* Svelte action to apply correct component styles. * Svelte action to apply correct component styles.
*/ */
export const styleable = (node, styles = {}) => { export const styleable = (node, styles = {}) => {
const normalStyles = styles.normal || {} let applyNormalStyles
const hoverStyles = { let applyHoverStyles
...normalStyles,
...styles.hover, // Creates event listeners and applies initial styles
const setupStyles = newStyles => {
const normalStyles = newStyles.normal || {}
const hoverStyles = {
...normalStyles,
...newStyles.hover,
}
applyNormalStyles = () => {
node.style = buildStyleString(normalStyles)
}
applyHoverStyles = () => {
node.style = buildStyleString(hoverStyles)
}
// Add listeners to toggle hover styles
node.addEventListener("mouseover", applyHoverStyles)
node.addEventListener("mouseout", applyNormalStyles)
node.setAttribute("data-bb-id", newStyles.id)
// Apply initial normal styles
applyNormalStyles()
} }
function applyNormalStyles() { // Removes the current event listeners
node.style = buildStyleString(normalStyles) const removeListeners = () => {
node.removeEventListener("mouseover", applyHoverStyles)
node.removeEventListener("mouseout", applyNormalStyles)
} }
function applyHoverStyles() { // Apply initial styles
node.style = buildStyleString(hoverStyles) setupStyles(styles)
}
// Add listeners to toggle hover styles
node.addEventListener("mouseover", applyHoverStyles)
node.addEventListener("mouseout", applyNormalStyles)
// Apply normal styles initially
applyNormalStyles()
// Also apply data tags so we know how to reference each component
node.setAttribute("data-bb-id", styles.id)
return { return {
// Clean up event listeners when component is destroyed // Clean up old listeners and apply new ones on update
update: newStyles => {
removeListeners()
setupStyles(newStyles)
},
// Clean up listeners when component is destroyed
destroy: () => { destroy: () => {
node.removeEventListener("mouseover", applyHoverStyles) removeListeners()
node.removeEventListener("mouseout", applyNormalStyles)
}, },
} }
} }

View File

@ -9,7 +9,7 @@
export let text export let text
</script> </script>
<button class="default" disabled={disabled || false} use:styleable={styles}> <button class="default" disabled={disabled || false} use:styleable={$styles}>
{text} {text}
</button> </button>

View File

@ -26,7 +26,7 @@
$: showImage = !!imageUrl $: showImage = !!imageUrl
</script> </script>
<div use:cssVars={cssVariables} class="container" use:styleable={styles}> <div use:cssVars={cssVariables} class="container" use:styleable={$styles}>
{#if showImage}<img class="image" src={imageUrl} alt="" />{/if} {#if showImage}<img class="image" src={imageUrl} alt="" />{/if}
<div class="content"> <div class="content">
<h2 class="heading">{heading}</h2> <h2 class="heading">{heading}</h2>

View File

@ -29,7 +29,7 @@
$: showImage = !!imageUrl $: showImage = !!imageUrl
</script> </script>
<div use:cssVars={cssVariables} class="container" use:styleable={styles}> <div use:cssVars={cssVariables} class="container" use:styleable={$styles}>
{#if showImage}<img class="image" src={imageUrl} alt="" />{/if} {#if showImage}<img class="image" src={imageUrl} alt="" />{/if}
<div class="content"> <div class="content">
<main> <main>

View File

@ -9,55 +9,55 @@
</script> </script>
{#if type === 'div'} {#if type === 'div'}
<div use:styleable={styles}> <div use:styleable={$styles}>
<slot /> <slot />
</div> </div>
{:else if type === 'header'} {:else if type === 'header'}
<header use:styleable={styles}> <header use:styleable={$styles}>
<slot /> <slot />
</header> </header>
{:else if type === 'main'} {:else if type === 'main'}
<main use:styleable={styles}> <main use:styleable={$styles}>
<slot /> <slot />
</main> </main>
{:else if type === 'footer'} {:else if type === 'footer'}
<footer use:styleable={styles}> <footer use:styleable={$styles}>
<slot /> <slot />
</footer> </footer>
{:else if type === 'aside'} {:else if type === 'aside'}
<aside use:styleable={styles}> <aside use:styleable={$styles}>
<slot /> <slot />
</aside> </aside>
{:else if type === 'summary'} {:else if type === 'summary'}
<summary use:styleable={styles}> <summary use:styleable={$styles}>
<slot /> <slot />
</summary> </summary>
{:else if type === 'details'} {:else if type === 'details'}
<details use:styleable={styles}> <details use:styleable={$styles}>
<slot /> <slot />
</details> </details>
{:else if type === 'article'} {:else if type === 'article'}
<article use:styleable={styles}> <article use:styleable={$styles}>
<slot /> <slot />
</article> </article>
{:else if type === 'nav'} {:else if type === 'nav'}
<nav use:styleable={styles}> <nav use:styleable={$styles}>
<slot /> <slot />
</nav> </nav>
{:else if type === 'mark'} {:else if type === 'mark'}
<mark use:styleable={styles}> <mark use:styleable={$styles}>
<slot /> <slot />
</mark> </mark>
{:else if type === 'figure'} {:else if type === 'figure'}
<figure use:styleable={styles}> <figure use:styleable={$styles}>
<slot /> <slot />
</figure> </figure>
{:else if type === 'figcaption'} {:else if type === 'figcaption'}
<figcaption use:styleable={styles}> <figcaption use:styleable={$styles}>
<slot /> <slot />
</figcaption> </figcaption>
{:else if type === 'paragraph'} {:else if type === 'paragraph'}
<p use:styleable={styles}> <p use:styleable={$styles}>
<slot /> <slot />
</p> </p>
{/if} {/if}

View File

@ -14,6 +14,6 @@
} }
</script> </script>
<div use:styleable={styles}> <div use:styleable={$styles}>
<DatePicker {placeholder} on:change={handleChange} {value} /> <DatePicker {placeholder} on:change={handleChange} {value} />
</div> </div>

View File

@ -7,7 +7,7 @@
export let embed export let embed
</script> </script>
<div use:styleable={styles}> <div use:styleable={$styles}>
{@html embed} {@html embed}
</div> </div>

View File

@ -27,7 +27,7 @@
} }
</script> </script>
<div class="form-content" use:styleable={styles}> <div class="form-content" use:styleable={$styles}>
<!-- <ErrorsBox errors={$store.saveRowErrors || {}} />--> <!-- <ErrorsBox errors={$store.saveRowErrors || {}} />-->
{#each fields as field} {#each fields as field}
<div class="form-field" class:wide> <div class="form-field" class:wide>

View File

@ -10,15 +10,15 @@
</script> </script>
{#if type === 'h1'} {#if type === 'h1'}
<h1 class={className} use:styleable={styles}>{text}</h1> <h1 class={className} use:styleable={$styles}>{text}</h1>
{:else if type === 'h2'} {:else if type === 'h2'}
<h2 class={className} use:styleable={styles}>{text}</h2> <h2 class={className} use:styleable={$styles}>{text}</h2>
{:else if type === 'h3'} {:else if type === 'h3'}
<h3 class={className} use:styleable={styles}>{text}</h3> <h3 class={className} use:styleable={$styles}>{text}</h3>
{:else if type === 'h4'} {:else if type === 'h4'}
<h4 class={className} use:styleable={styles}>{text}</h4> <h4 class={className} use:styleable={$styles}>{text}</h4>
{:else if type === 'h5'} {:else if type === 'h5'}
<h5 class={className} use:styleable={styles}>{text}</h5> <h5 class={className} use:styleable={$styles}>{text}</h5>
{:else if type === 'h6'} {:else if type === 'h6'}
<h6 class={className} use:styleable={styles}>{text}</h6> <h6 class={className} use:styleable={$styles}>{text}</h6>
{/if} {/if}

View File

@ -13,4 +13,4 @@
<i <i
style={`color: ${color};`} style={`color: ${color};`}
class={`${icon} ${size}`} class={`${icon} ${size}`}
use:styleable={styles} /> use:styleable={$styles} />

View File

@ -17,4 +17,4 @@
class={className} class={className}
src={url} src={url}
alt={description} alt={description}
use:styleable={styles} /> use:styleable={$styles} />

View File

@ -18,4 +18,4 @@
{type} {type}
{value} {value}
on:change={onchange} on:change={onchange}
use:styleable={styles} /> use:styleable={$styles} />

View File

@ -11,7 +11,7 @@
$: target = openInNewTab ? "_blank" : "_self" $: target = openInNewTab ? "_blank" : "_self"
</script> </script>
<a href={url} use:linkable {target} use:styleable={styles}> <a href={url} use:linkable {target} use:styleable={$styles}>
{text} {text}
<slot /> <slot />
</a> </a>

View File

@ -17,7 +17,7 @@
}) })
</script> </script>
<div use:styleable={styles}> <div use:styleable={$styles}>
{#each rows as row} {#each rows as row}
<DataProvider {row}> <DataProvider {row}>
<slot /> <slot />

View File

@ -29,7 +29,7 @@
} }
</script> </script>
<div class="root" use:styleable={styles}> <div class="root" use:styleable={$styles}>
<div class="content"> <div class="content">
{#if logo} {#if logo}
<div class="logo-container"><img src={logo} alt="logo" /></div> <div class="logo-container"><img src={logo} alt="logo" /></div>

View File

@ -12,7 +12,7 @@
} }
</script> </script>
<div class="nav" use:styleable={styles}> <div class="nav" use:styleable={$styles}>
<div class="nav__top"> <div class="nav__top">
<a href="/" use:linkable> <a href="/" use:linkable>
{#if logoUrl} {#if logoUrl}

View File

@ -23,6 +23,6 @@
} }
</script> </script>
<div use:styleable={styles}> <div use:styleable={$styles}>
<RichText bind:content={value} {options} /> <RichText bind:content={value} {options} />
</div> </div>

View File

@ -5,7 +5,7 @@
const { styleable } = getContext("sdk") const { styleable } = getContext("sdk")
</script> </script>
<div use:styleable={styles}> <div use:styleable={$styles}>
<h1>Screen Slot</h1> <h1>Screen Slot</h1>
<span> <span>
The screens that you create will be displayed inside this box. The screens that you create will be displayed inside this box.

View File

@ -14,7 +14,7 @@
$: showImage = !!imageUrl $: showImage = !!imageUrl
</script> </script>
<div class="container" use:styleable={styles}> <div class="container" use:styleable={$styles}>
<a href={destinationUrl}> <a href={destinationUrl}>
<div class="content"> <div class="content">
{#if showImage} {#if showImage}

View File

@ -12,28 +12,28 @@
</script> </script>
{#if isTag('none')} {#if isTag('none')}
<span use:styleable={styles}>{text}</span> <span use:styleable={$styles}>{text}</span>
{:else if isTag('bold')} {:else if isTag('bold')}
<b class={className} use:styleable={styles}>{text}</b> <b class={className} use:styleable={$styles}>{text}</b>
{:else if isTag('strong')} {:else if isTag('strong')}
<strong class={className} use:styleable={styles}>{text}</strong> <strong class={className} use:styleable={$styles}>{text}</strong>
{:else if isTag('italic')} {:else if isTag('italic')}
<i class={className} use:styleable={styles}>{text}</i> <i class={className} use:styleable={$styles}>{text}</i>
{:else if isTag('emphasis')} {:else if isTag('emphasis')}
<em class={className} use:styleable={styles}>{text}</em> <em class={className} use:styleable={$styles}>{text}</em>
{:else if isTag('mark')} {:else if isTag('mark')}
<mark class={className} use:styleable={styles}>{text}</mark> <mark class={className} use:styleable={$styles}>{text}</mark>
{:else if isTag('small')} {:else if isTag('small')}
<small class={className} use:styleable={styles}>{text}</small> <small class={className} use:styleable={$styles}>{text}</small>
{:else if isTag('del')} {:else if isTag('del')}
<del class={className} use:styleable={styles}>{text}</del> <del class={className} use:styleable={$styles}>{text}</del>
{:else if isTag('ins')} {:else if isTag('ins')}
<ins class={className} use:styleable={styles}>{text}</ins> <ins class={className} use:styleable={$styles}>{text}</ins>
{:else if isTag('sub')} {:else if isTag('sub')}
<sub class={className} use:styleable={styles}>{text}</sub> <sub class={className} use:styleable={$styles}>{text}</sub>
{:else if isTag('sup')} {:else if isTag('sup')}
<sup class={className} use:styleable={styles}>{text}</sup> <sup class={className} use:styleable={$styles}>{text}</sup>
{:else}<span use:styleable={styles}>{text}</span>{/if} {:else}<span use:styleable={$styles}>{text}</span>{/if}
<style> <style>
span { span {

View File

@ -9,9 +9,9 @@
</script> </script>
{#if options} {#if options}
<div use:chart={options} use:styleable={styles} /> <div use:chart={options} use:styleable={$styles} />
{:else if options === false} {:else if options === false}
<div use:styleable={styles}>Invalid chart options</div> <div use:styleable={$styles}>Invalid chart options</div>
{/if} {/if}
<style> <style>

View File

@ -27,7 +27,13 @@
export let detailUrl export let detailUrl
// Add setting height as css var to allow grid to use correct height // Add setting height as css var to allow grid to use correct height
styles.normal["--grid-height"] = `${height}px` $: gridStyles = {
...$styles,
normal: {
...$styles.normal,
["--grid-height"]: `${height}px`,
},
}
// These can never change at runtime so don't need to be reactive // These can never change at runtime so don't need to be reactive
let canEdit = editable && datasource && datasource.type !== "view" let canEdit = editable && datasource && datasource.type !== "view"
@ -143,7 +149,7 @@
href="https://cdn.jsdelivr.net/npm/flatpickr/dist/flatpickr.min.css" /> href="https://cdn.jsdelivr.net/npm/flatpickr/dist/flatpickr.min.css" />
</svelte:head> </svelte:head>
<div class="container" use:styleable={styles}> <div class="container" use:styleable={gridStyles}>
{#if dataLoaded} {#if dataLoaded}
{#if canAddDelete} {#if canAddDelete}
<div class="controls"> <div class="controls">