grungnet/src/contexts/TokenDataContext.js

193 lines
5.3 KiB
JavaScript
Raw Normal View History

import React, { useEffect, useState, useContext, useCallback } from "react";
import { decode } from "@msgpack/msgpack";
import { useAuth } from "./AuthContext";
import { useDatabase } from "./DatabaseContext";
2021-05-25 01:47:52 -04:00
import { applyObservableChange } from "../helpers/dexie";
const TokenDataContext = React.createContext();
export function TokenDataProvider({ children }) {
const { database, databaseStatus, worker } = useDatabase();
const { userId } = useAuth();
const [tokens, setTokens] = useState([]);
const [tokensLoading, setTokensLoading] = useState(true);
const [tokenGroups, setTokenGroups] = useState([]);
useEffect(() => {
2020-10-23 07:16:18 -04:00
if (!userId || !database || databaseStatus === "loading") {
return;
}
async function loadTokens() {
let storedTokens = [];
// Try to load tokens with worker, fallback to database if failed
const packedTokens = await worker.loadData("tokens");
if (packedTokens) {
storedTokens = decode(packedTokens);
} else {
console.warn("Unable to load tokens with worker, loading may be slow");
await database.table("tokens").each((token) => {
2021-04-29 19:54:35 -04:00
storedTokens.push(token);
});
}
setTokens(storedTokens);
const group = await database.table("groups").get("tokens");
2021-05-14 04:02:50 -04:00
const storedGroups = group.items;
setTokenGroups(storedGroups);
setTokensLoading(false);
}
loadTokens();
}, [userId, database, databaseStatus, worker]);
const getToken = useCallback(
async (tokenId) => {
let token = await database.table("tokens").get(tokenId);
return token;
},
[database]
);
// Add token and add it to the token group
const addToken = useCallback(
async (token) => {
await database.table("tokens").add(token);
const group = await database.table("groups").get("tokens");
2021-05-14 04:02:50 -04:00
await database.table("groups").update("tokens", {
items: [{ id: token.id, type: "item" }, ...group.items],
});
},
[database]
);
const removeTokens = useCallback(
async (ids) => {
2021-04-29 19:54:35 -04:00
const tokens = await database.table("tokens").bulkGet(ids);
let assetIds = [];
for (let token of tokens) {
if (token.type === "file") {
assetIds.push(token.file);
assetIds.push(token.thumbnail);
}
}
await database.table("tokens").bulkDelete(ids);
2021-04-29 19:54:35 -04:00
await database.table("assets").bulkDelete(assetIds);
},
[database]
);
const updateToken = useCallback(
async (id, update) => {
const change = { lastModified: Date.now(), ...update };
await database.table("tokens").update(id, change);
},
[database]
);
const updateTokens = useCallback(
async (ids, update) => {
const change = { lastModified: Date.now(), ...update };
await Promise.all(
ids.map((id) => database.table("tokens").update(id, change))
);
},
[database]
);
const updateTokenGroups = useCallback(
async (groups) => {
// Update group state immediately to avoid animation delay
setTokenGroups(groups);
2021-05-14 04:02:50 -04:00
await database.table("groups").update("tokens", { items: groups });
},
[database]
);
// Create DB observable to sync creating and deleting
useEffect(() => {
if (!database || databaseStatus === "loading") {
return;
}
function handleTokenChanges(changes) {
for (let change of changes) {
if (change.table === "tokens") {
if (change.type === 1) {
// Created
const token = change.obj;
setTokens((prevTokens) => [token, ...prevTokens]);
} else if (change.type === 2) {
// Updated
const token = change.obj;
setTokens((prevTokens) => {
const newTokens = [...prevTokens];
const i = newTokens.findIndex((t) => t.id === token.id);
if (i > -1) {
newTokens[i] = token;
}
return newTokens;
});
} else if (change.type === 3) {
// Deleted
const id = change.key;
setTokens((prevTokens) => {
const filtered = prevTokens.filter((token) => token.id !== id);
return filtered;
});
}
}
if (change.table === "groups") {
if (change.type === 2 && change.key === "tokens") {
2021-05-25 01:47:52 -04:00
const group = applyObservableChange(change);
const groups = group.items.filter((item) => item !== null);
setTokenGroups(groups);
}
}
}
}
database.on("changes", handleTokenChanges);
return () => {
database.on("changes").unsubscribe(handleTokenChanges);
};
}, [database, databaseStatus]);
const tokensById = tokens.reduce((obj, token) => {
obj[token.id] = token;
return obj;
}, {});
const value = {
tokens,
addToken,
tokenGroups,
removeTokens,
updateToken,
updateTokens,
tokensById,
tokensLoading,
getToken,
updateTokenGroups,
};
return (
<TokenDataContext.Provider value={value}>
{children}
</TokenDataContext.Provider>
);
}
export function useTokenData() {
const context = useContext(TokenDataContext);
if (context === undefined) {
throw new Error("useTokenData must be used within a TokenDataProvider");
}
return context;
}
export default TokenDataContext;