UNPKG

jotai

Version:

👻 Next gen state management that will spook you

127 lines (121 loc) • 4.2 kB
import path from 'path'; import templateBuilder from '@babel/template'; function isAtom(t, callee) { if (t.isIdentifier(callee) && atomFunctionNames.includes(callee.name)) { return true; } if (t.isMemberExpression(callee)) { const { property } = callee; if (t.isIdentifier(property) && atomFunctionNames.includes(property.name)) { return true; } } return false; } const atomFunctionNames = [ "atom", "atomFamily", "atomWithDefault", "atomWithObservable", "atomWithReducer", "atomWithReset", "atomWithStorage", "freezeAtom", "loadable", "selectAtom", "splitAtom" ]; function debugLabelPlugin({ types: t }) { return { visitor: { ExportDefaultDeclaration(nodePath, state) { const { node } = nodePath; if (t.isCallExpression(node.declaration) && isAtom(t, node.declaration.callee)) { const filename = state.filename || "unknown"; let displayName = path.basename(filename, path.extname(filename)); if (displayName === "index") { displayName = path.basename(path.dirname(filename)); } const buildExport = templateBuilder(` const %%atomIdentifier%% = %%atom%%; export default %%atomIdentifier%% `); const ast = buildExport({ atomIdentifier: t.identifier(displayName), atom: node.declaration }); nodePath.replaceWithMultiple(ast); } }, VariableDeclarator(path2) { if (t.isIdentifier(path2.node.id) && t.isCallExpression(path2.node.init) && isAtom(t, path2.node.init.callee)) { path2.parentPath.insertAfter(t.expressionStatement(t.assignmentExpression("=", t.memberExpression(t.identifier(path2.node.id.name), t.identifier("debugLabel")), t.stringLiteral(path2.node.id.name)))); } } } }; } function reactRefreshPlugin({ types: t }) { return { pre({ opts }) { if (!opts.filename) { throw new Error("Filename must be available"); } }, visitor: { Program: { exit(path) { const jotaiAtomCache = templateBuilder(` globalThis.jotaiAtomCache = globalThis.jotaiAtomCache || { cache: new Map(), get(name, inst) { if (this.cache.has(name)) { return this.cache.get(name) } this.cache.set(name, inst) return inst }, }`)(); path.unshiftContainer("body", jotaiAtomCache); } }, ExportDefaultDeclaration(nodePath, state) { const { node } = nodePath; if (t.isCallExpression(node.declaration) && isAtom(t, node.declaration.callee)) { const filename = state.filename || "unknown"; const atomKey = `${filename}/defaultExport`; const buildExport = templateBuilder(`export default globalThis.jotaiAtomCache.get(%%atomKey%%, %%atom%%)`); const ast = buildExport({ atomKey: t.stringLiteral(atomKey), atom: node.declaration }); nodePath.replaceWith(ast); } }, VariableDeclarator(nodePath, state) { var _a, _b; if (t.isIdentifier(nodePath.node.id) && t.isCallExpression(nodePath.node.init) && isAtom(t, nodePath.node.init.callee) && (((_a = nodePath.parentPath.parentPath) == null ? void 0 : _a.isProgram()) || ((_b = nodePath.parentPath.parentPath) == null ? void 0 : _b.isExportNamedDeclaration()))) { const filename = state.filename || "unknown"; const atomKey = `${filename}/${nodePath.node.id.name}`; const buildAtomDeclaration = templateBuilder(`const %%atomIdentifier%% = globalThis.jotaiAtomCache.get(%%atomKey%%, %%atom%%)`); const ast = buildAtomDeclaration({ atomIdentifier: t.identifier(nodePath.node.id.name), atomKey: t.stringLiteral(atomKey), atom: nodePath.node.init }); nodePath.parentPath.replaceWith(ast); } } } }; } function jotaiPreset() { return { plugins: [debugLabelPlugin, reactRefreshPlugin] }; } export { jotaiPreset as default };