budibase/packages/builder/src/helpers/urlStateSync.js

181 lines
4.7 KiB
JavaScript

import { get } from "svelte/store"
import { isChangingPage } from "@roxi/routify"
export const syncURLToState = options => {
const {
urlParam,
stateKey,
validate,
update,
baseUrl = "..",
fallbackUrl,
store,
routify,
beforeNavigate,
decode,
} = options || {}
if (
!urlParam ||
!stateKey ||
!baseUrl ||
!urlParam ||
!store?.subscribe ||
!routify ||
!routify.params?.subscribe ||
!routify.goto?.subscribe ||
!routify.redirect?.subscribe ||
!routify.page?.subscribe
) {
console.warn("syncURLToState invoked with missing parameters")
return
}
// Decodes encoded URL params if required
const decodeParams = urlParams => {
if (!decode) {
return urlParams
}
let decoded = {}
Object.keys(urlParams || {}).forEach(key => {
decoded[key] = decode(urlParams[key])
})
return decoded
}
// We can't dynamically fetch the value of stateful routify stores so we need
// to just subscribe and cache the latest versions.
// We can grab their initial values as this is during component
// initialisation.
let cachedParams = decodeParams(get(routify.params))
let cachedGoto = get(routify.goto)
let cachedRedirect = get(routify.redirect)
let cachedPage = get(routify.page)
let previousParamsHash = null
let debug = false
const log = (...params) => debug && console.log(`[${urlParam}]`, ...params)
// Navigate to a certain URL
const gotoUrl = (url, params) => {
if (beforeNavigate) {
const res = beforeNavigate(url, params)
if (res?.url) {
url = res.url
}
if (res?.params) {
params = res.params
}
}
log("Navigating to", url, "with params", params)
cachedGoto(url, params)
}
// Redirect to a certain URL
const redirectUrl = url => {
log("Redirecting to", url)
cachedRedirect(url)
}
// Updates state with new URL params
const mapUrlToState = params => {
// Check if we have new URL params
const paramsHash = JSON.stringify(params)
const newParams = paramsHash !== previousParamsHash
previousParamsHash = paramsHash
const urlValue = params?.[urlParam]
const stateValue = get(store)?.[stateKey]
if (!newParams || !urlValue) {
return
}
// Check if new value is valid
if (validate && fallbackUrl) {
if (!validate(urlValue)) {
log("Invalid URL param!", urlValue)
redirectUrl(fallbackUrl)
return
}
}
// Only update state if we have a new value
if (urlValue !== stateValue) {
log(`state.${stateKey} (${stateValue}) <= url.${urlParam} (${urlValue})`)
if (update) {
// Use custom update function if provided
update(urlValue)
} else {
// Otherwise manually update the store
store.update(state => ({
...state,
[stateKey]: urlValue,
}))
}
}
}
// Updates the URL with new state values
const mapStateToUrl = state => {
let needsUpdate = false
const urlValue = cachedParams?.[urlParam]
const stateValue = state?.[stateKey]
if (stateValue !== urlValue) {
needsUpdate = true
log(`url.${urlParam} (${urlValue}) <= state.${stateKey} (${stateValue})`)
if (validate && fallbackUrl) {
if (!validate(stateValue)) {
log("Invalid state param!", stateValue)
redirectUrl(fallbackUrl)
return
}
}
}
// Avoid updating the URL if not necessary to prevent a wasted render
// cycle
if (!needsUpdate) {
return
}
// Navigate to the new URL
if (!get(isChangingPage)) {
const newUrlParams = {
...cachedParams,
[urlParam]: stateValue,
}
gotoUrl(cachedPage.path, newUrlParams)
}
}
// Initially hydrate state from URL
mapUrlToState(cachedParams)
// Subscribe to URL changes and cache them
const unsubscribeParams = routify.params.subscribe($urlParams => {
$urlParams = decodeParams($urlParams)
cachedParams = $urlParams
mapUrlToState($urlParams)
})
// Subscribe to routify store changes and cache them
const unsubscribeGoto = routify.goto.subscribe($goto => {
cachedGoto = $goto
})
const unsubscribeRedirect = routify.redirect.subscribe($redirect => {
cachedRedirect = $redirect
})
const unsubscribePage = routify.page.subscribe($page => {
cachedPage = $page
})
// Subscribe to store changes and keep URL up to date
const unsubscribeStore = store.subscribe(mapStateToUrl)
// Return an unsync function to cancel subscriptions
return () => {
unsubscribeParams()
unsubscribeGoto()
unsubscribeRedirect()
unsubscribePage()
unsubscribeStore()
}
}