-
Marshall Quander authoredMarshall Quander authored
store.js 2.63 KiB
import { Validator } from "jsonschema";
import merge from "deepmerge";
const LOCAL_STORE_KEY = "___hubs_store";
const STORE_STATE_CACHE_KEY = Symbol();
const validator = new Validator();
import { EventTarget } from "event-target-shim";
import { generateDefaultProfile, generateRandomName } from "../utils/identity.js";
// Durable (via local-storage) schema-enforced state that is meant to be consumed via forward data flow.
// (Think flux but with way less incidental complexity, at least for now :))
export const SCHEMA = {
id: "/HubsStore",
definitions: {
profile: {
type: "object",
additionalProperties: false,
properties: {
displayName: { type: "string", pattern: "^[A-Za-z0-9-]{3,32}$" },
avatarId: { type: "string" }
}
},
activity: {
type: "object",
additionalProperties: false,
properties: {
hasFoundFreeze: { type: "boolean" },
hasChangedName: { type: "boolean" },
lastEnteredAt: { type: "string" }
}
},
settings: {
type: "object",
additionalProperties: false,
properties: {
lastUsedMicDeviceId: { type: "string" }
}
}
},
type: "object",
properties: {
profile: { $ref: "#/definitions/profile" },
activity: { $ref: "#/definitions/activity" },
settings: { $ref: "#/definitions/settings" }
},
additionalProperties: false
};
export default class Store extends EventTarget {
constructor() {
super();
if (localStorage.getItem(LOCAL_STORE_KEY) === null) {
localStorage.setItem(LOCAL_STORE_KEY, JSON.stringify({}));
}
}
// Initializes store with any default bits
init = () => {
this.update({
activity: {},
settings: {},
profile: { ...generateDefaultProfile(), ...(this.state.profile || {}) }
});
// Regenerate name to encourage users to change it.
if (!this.state.activity.hasChangedName) {
this.update({ profile: { displayName: generateRandomName() } });
}
};
get state() {
if (!this.hasOwnProperty(STORE_STATE_CACHE_KEY)) {
this[STORE_STATE_CACHE_KEY] = JSON.parse(localStorage.getItem(LOCAL_STORE_KEY));
}
return this[STORE_STATE_CACHE_KEY];
}
update(newState) {
const finalState = merge(this.state, newState);
const isValid = validator.validate(finalState, SCHEMA).valid;
if (!isValid) {
throw new Error(`Write of ${JSON.stringify(finalState)} to store failed schema validation.`);
}
localStorage.setItem(LOCAL_STORE_KEY, JSON.stringify(finalState));
delete this[STORE_STATE_CACHE_KEY];
this.dispatchEvent(new CustomEvent("statechanged"));
return finalState;
}
}