2021-02-10 20:42:56 +01:00
|
|
|
import { writable, derived } from "svelte/store"
|
2021-11-08 15:35:58 +01:00
|
|
|
import { hashString } from "../utils/helpers"
|
2021-02-01 19:51:22 +01:00
|
|
|
|
2021-05-04 12:32:22 +02:00
|
|
|
export const createContextStore = oldContext => {
|
2021-02-10 20:42:56 +01:00
|
|
|
const newContext = writable({})
|
|
|
|
const contexts = oldContext ? [oldContext, newContext] : [newContext]
|
2021-05-04 12:32:22 +02:00
|
|
|
const totalContext = derived(contexts, $contexts => {
|
2021-06-25 16:04:27 +02:00
|
|
|
// The key is the serialized representation of context
|
|
|
|
let key = ""
|
|
|
|
for (let i = 0; i < $contexts.length - 1; i++) {
|
|
|
|
key += $contexts[i].key
|
|
|
|
}
|
2021-11-08 15:35:58 +01:00
|
|
|
key = hashString(key + JSON.stringify($contexts[$contexts.length - 1]))
|
2021-06-25 16:04:27 +02:00
|
|
|
|
|
|
|
// Reduce global state
|
|
|
|
const reducer = (total, context) => ({ ...total, ...context })
|
|
|
|
const context = $contexts.reduce(reducer, {})
|
|
|
|
|
|
|
|
return {
|
|
|
|
...context,
|
|
|
|
key,
|
|
|
|
}
|
2021-02-10 20:42:56 +01:00
|
|
|
})
|
2021-02-01 19:51:22 +01:00
|
|
|
|
|
|
|
// Adds a data context layer to the tree
|
2021-02-10 20:42:56 +01:00
|
|
|
const provideData = (providerId, data) => {
|
|
|
|
if (!providerId || data === undefined) {
|
|
|
|
return
|
|
|
|
}
|
2021-05-04 12:32:22 +02:00
|
|
|
newContext.update(state => {
|
2021-02-10 20:42:56 +01:00
|
|
|
state[providerId] = data
|
2021-02-02 15:32:58 +01:00
|
|
|
|
2021-02-05 12:44:33 +01:00
|
|
|
// Keep track of the closest component ID so we can later hydrate a "data" prop.
|
|
|
|
// This is only required for legacy bindings that used "data" rather than a
|
|
|
|
// component ID.
|
2021-02-10 20:42:56 +01:00
|
|
|
state.closestComponentId = providerId
|
|
|
|
|
|
|
|
return state
|
|
|
|
})
|
2021-02-01 19:51:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Adds an action context layer to the tree
|
2021-02-05 17:16:41 +01:00
|
|
|
const provideAction = (providerId, actionType, callback) => {
|
|
|
|
if (!providerId || !actionType) {
|
2021-02-05 12:44:33 +01:00
|
|
|
return
|
|
|
|
}
|
2021-05-04 12:32:22 +02:00
|
|
|
newContext.update(state => {
|
2021-02-05 17:16:41 +01:00
|
|
|
state[`${providerId}_${actionType}`] = callback
|
2021-02-01 19:51:22 +01:00
|
|
|
return state
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
2021-02-10 20:42:56 +01:00
|
|
|
subscribe: totalContext.subscribe,
|
2021-02-01 19:51:22 +01:00
|
|
|
actions: { provideData, provideAction },
|
|
|
|
}
|
|
|
|
}
|