UNPKG

use-fs

Version:

A React hook for integrating with the File System Access API. Enables web applications to seamlessly work with files on a user's local system.

2 lines 49.8 kB
"use strict";var oe=Object.create;var P=Object.defineProperty;var ce=Object.getOwnPropertyDescriptor;var le=Object.getOwnPropertyNames;var ae=Object.getPrototypeOf,ue=Object.prototype.hasOwnProperty;var de=(t,r)=>{for(var n in r)P(t,n,{get:r[n],enumerable:!0})},j=(t,r,n,o)=>{if(r&&typeof r=="object"||typeof r=="function")for(let c of le(r))!ue.call(t,c)&&c!==n&&P(t,c,{get:()=>r[c],enumerable:!(o=ce(r,c))||o.enumerable});return t};var fe=(t,r,n)=>(n=t!=null?oe(ae(t)):{},j(r||!t||!t.__esModule?P(n,"default",{value:t,enumerable:!0}):n,t)),ge=t=>j(P({},"__esModule",{value:!0}),t);var Me={};de(Me,{commonFilters:()=>K,distFilter:()=>q,gitFilter:()=>Z,miscFilter:()=>G,processDirectory:()=>W,useFileSystem:()=>Q,useFs:()=>be});module.exports=ge(Me);var J=fe(require("ignore")),s=require("react"),Z=async()=>{let t=(0,J.default)({allowRelativePaths:!0}),r=!1;return{shouldIncludeFile:async(n,o)=>{if(n.endsWith(".gitignore")&&!r&&o.kind==="file"){let i=await(await o.getFile()).text();return t.add(i),r=!0,!1}let{ignored:c}=t.test(n);return!c},shouldProcessDirectory:async(n,o)=>{let{ignored:c}=t.test(n);return n.endsWith(".git")||n.includes(".git/")?!1:!c}}},q=async()=>{let t=r=>!!(r.includes("/dist/")||r.includes("/out/")||r.includes("/build/")||r.includes("/vendor/")||r.includes("/node_modules/")||r.includes("/.next/"));return{shouldIncludeFile:async(r,n)=>!t(r),shouldProcessDirectory:async(r,n)=>!t(r)}},G=async()=>{let t=r=>!!(r.endsWith(".DS_Store")||r.endsWith(".crswap"));return{shouldIncludeFile:async(r,n)=>!t(r),shouldProcessDirectory:async(r,n)=>!t(r)}},K=[q,G,Z],W=async(t,r,n,o,c)=>{if(c.has(r))return o;for(let i of n)if(await i.shouldProcessDirectory(r,t)===!1)return c.add(r),o;let x=[];for await(let i of t.values())if(i.kind==="file"){let d=`${r}/${i.name}`;c.has(d)||x.push((async()=>{let a=!0;for(let I of n)if(await I.shouldIncludeFile(d,i)===!1){a=!1,c.add(d),o.delete(d);break}a&&o.set(d,i)})())}await Promise.all(x);let b=[];for await(let i of t.values())if(i.kind==="directory"){let d=`${r}/${i.name}`;c.has(d)||b.push(W(i,d,n,o,c))}return await Promise.all(b),o},Fe=(t,r)=>{let[n,o]=(0,s.useState)(t);return(0,s.useEffect)(()=>{let c=setTimeout(()=>{o(t)},r);return()=>{clearTimeout(c)}},[t,r]),n},me=100,we=50,ye=50,pe=5e3,Q=t=>{let{onFilesAdded:r,onFilesChanged:n,onFilesDeleted:o,batchSize:c=we,debounceInterval:x=ye}=t,b=(0,s.useRef)(new Map),i=(0,s.useRef)(new Map),d=(0,s.useRef)(new Set),a=(0,s.useRef)({}),I=(0,s.useRef)({}),H=(0,s.useRef)(null),[X,A]=(0,s.useState)(new Map),[_,$]=(0,s.useState)(!1),[Y,O]=(0,s.useState)(!1),S=(0,s.useRef)(new Map),R=t.fileCacheTtl||pe,U=(0,s.useCallback)(()=>{H.current&&(clearInterval(H.current),H.current=null),b.current.clear(),i.current.clear(),d.current.clear(),a.current={},I.current={},S.current.clear(),A(new Map),O(!1)},[]),N=(0,s.useCallback)(async()=>{let e=new Map,u=new Set,w=t.filters||K,f=[];for(let m of w)f.push(await m());let g=[];b.current.forEach((m,F)=>{g.push(W(m,F,f,e,u))}),await Promise.all(g),e.forEach((m,F)=>{u.has(F)&&e.delete(F)}),i.current=e},[t.filters]),z=(0,s.useCallback)(async()=>{I.current={...a.current||{}};let e=new Set,u=new Map,w=new Set,f=!1,g=Date.now(),m=Array.from(i.current),F={};for(let l=0;l<m.length;l+=c){let L=m.slice(l,l+c);await Promise.all(L.map(async([p,V])=>{if(V.kind==="file"){e.add(p);try{let h=S.current.get(p);if(h&&g-h.timestamp<R){F[p]=h.content,h.content!==I.current[p]&&u.set(p,h.content);return}let T=await(await V.getFile()).text();S.current.set(p,{content:T,timestamp:g}),F[p]=T,T!==I.current[p]&&u.set(p,T)}catch{i.current.delete(p),w.add(p),S.current.delete(p),f=!0}}}))}let M=Array.from(S.current.entries());for(let[l,{timestamp:L}]of M)g-L>R&&S.current.delete(l);let y=Array.from([...Array.from(d.current),...Array.from(w)]).filter(l=>!e.has(l)),v=Array.from(e).filter(l=>!d.current.has(l)),D=new Map;for(let l of v)D.set(l,a.current[l]);let E=new Map;for(let l of y)E.set(l,a.current[l]),i.current.delete(l),delete a.current[l];let B=new Map(Array.from(d.current).map(l=>[l,a.current[l]]));u.size>0&&(f=!0,n==null||n(u,B)),y.length>0&&(f=!0,o==null||o(E,B)),v.length>0&&(f=!0,r==null||r(D,B)),d.current=e,a.current=F,f&&A(new Map(Object.entries(a.current)))},[r,n,o,c,R]),ee=Fe(X,x),k=(0,s.useCallback)(()=>{let e=null;H.current||(H.current=setInterval(async()=>{_||e||($(!0),e=(async()=>{try{await N(),await z()}finally{$(!1),e=null}})())},t.pollInterval||me),O(!0))},[N,z,_,t.pollInterval]),re=(0,s.useCallback)(()=>{H.current&&(clearInterval(H.current),H.current=null,O(!1))},[]),te=(0,s.useCallback)(()=>{b.current.size>0&&k()},[k]),ne=async()=>{try{let e=await window.showDirectoryPicker();if(!e)return;let u=e.name;b.current.set(u,e),k()}catch(e){console.error("Error during directory selection:"),console.error(e)}},C=(0,s.useCallback)(async(e,u,w={})=>{if(!e||typeof e!="string")throw new Error("Invalid file path");let{create:f=!0,truncate:g=!1}=w;if(!i.current.get(e)){if(!f)throw new Error(`File not found: ${e}`);let y=e.substring(0,e.lastIndexOf("/")),v=e.substring(e.lastIndexOf("/")+1);if(!(y&&v))throw new Error("Invalid file path structure");let D=b.current.get(y);if(!D)throw new Error(`Directory not found: ${y}`);let E=await D.getFileHandle(v,{create:!0});i.current.set(e,E)}let F=i.current.get(e);if(!F)throw new Error(`File not found: ${e}`);if(F.name!==e.split("/").pop())throw new Error("File path mismatch");let M=await F.createWritable({keepExistingData:!g});try{await M.write(u),await M.close();let y=typeof u=="string"?u:await new Response(u).text();S.current.set(e,{content:y,timestamp:Date.now()});let v=new Map([[e,y]]),D=new Map(Array.from(d.current).map(E=>[E,a.current[E]]));n==null||n(v,D),a.current[e]=y,A(new Map(Object.entries(a.current)))}catch(y){throw await M.abort(),y}},[n]),se=(0,s.useCallback)(async(e,u)=>{let w=e.substring(0,e.lastIndexOf("/")),f=e.substring(e.lastIndexOf("/")+1),g=b.current.get(w);if(!g)throw new Error(`Directory not found: ${w}`);let m=await g.getFileHandle(f,{create:!0});return i.current.set(e,m),u&&await C(e,u,{truncate:!0}),m},[C]),ie=(0,s.useCallback)(async e=>{if(!i.current.get(e))throw new Error(`File not found: ${e}`);let w=e.substring(0,e.lastIndexOf("/")),f=e.substring(e.lastIndexOf("/")+1),g=b.current.get(w);if(!g)throw new Error(`Directory not found: ${w}`);await g.removeEntry(f),i.current.delete(e),S.current.delete(e);let m=new Map([[e,a.current[e]]]),F=new Map(Array.from(d.current).map(M=>[M,a.current[M]]));delete a.current[e],d.current.delete(e),o==null||o(m,F),A(new Map(Object.entries(a.current)))},[o]);return Se(()=>{U()}),{handles:i.current,onDirectorySelection:ne,onClear:U,files:ee,setFiles:A,isProcessing:_,isBrowserSupported:window!==void 0&&"showDirectoryPicker"in window,writeFile:C,createFile:se,deleteFile:ie,stopPolling:re,startPolling:te,isPolling:Y}},be=Q,He=t=>{(0,s.useEffect)(t,[])},Se=t=>{let r=(0,s.useRef)(t);r.current=t,He(()=>()=>r.current())};0&&(module.exports={commonFilters,distFilter,gitFilter,miscFilter,processDirectory,useFileSystem,useFs}); //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.ts"],
  "sourcesContent": ["import ignore from \"ignore\";\nimport {\n\ttype EffectCallback,\n\tuseCallback,\n\tuseEffect,\n\tuseRef,\n\tuseState,\n} from \"react\";\n\nexport type FileWriteOptions = {\n\t/** Whether to create the file if it doesn't exist */\n\tcreate?: boolean;\n\t/** Whether to truncate the file if it exists */\n\ttruncate?: boolean;\n};\n\nexport type FileOperations = {\n\t/** Write data to a file at the specified path */\n\twriteFile: (\n\t\tpath: string,\n\t\tdata: string | ArrayBuffer | Blob,\n\t\toptions?: FileWriteOptions,\n\t) => Promise<void>;\n\n\t/** Create a new file at the specified path */\n\tcreateFile: (\n\t\tpath: string,\n\t\tinitialData?: string | ArrayBuffer | Blob,\n\t) => Promise<FileSystemFileHandle>;\n\n\t/** Delete a file at the specified path */\n\tdeleteFile: (path: string) => Promise<void>;\n};\n\nexport type Filter = {\n\tshouldIncludeFile: (\n\t\tfilepath: string,\n\t\thandle: FileSystemHandle,\n\t) => Promise<boolean>;\n\tshouldProcessDirectory: (\n\t\tfilepath: string,\n\t\thandle: FileSystemHandle,\n\t) => Promise<boolean>;\n};\n\nexport type FilterFn = () => Promise<Filter>;\n\nexport const gitFilter: FilterFn = async () => {\n\tconst ig = ignore({\n\t\tallowRelativePaths: true,\n\t});\n\n\t// TODO support multiple gitignore files\n\tlet gitIgnoreLoaded = false;\n\n\treturn {\n\t\tshouldIncludeFile: async (filepath: string, handle: FileSystemHandle) => {\n\t\t\tif (\n\t\t\t\tfilepath.endsWith(\".gitignore\") &&\n\t\t\t\t!gitIgnoreLoaded &&\n\t\t\t\thandle.kind === \"file\"\n\t\t\t) {\n\t\t\t\tconst fileHandle = handle as FileSystemFileHandle;\n\t\t\t\tconst file = await fileHandle.getFile();\n\t\t\t\tconst text = await file.text();\n\t\t\t\tig.add(text);\n\t\t\t\tgitIgnoreLoaded = true;\n\t\t\t\treturn false;\n\t\t\t}\n\t\t\tconst { ignored } = ig.test(filepath);\n\n\t\t\treturn !ignored;\n\t\t},\n\t\tshouldProcessDirectory: async (\n\t\t\tfilepath: string,\n\t\t\t_handle: FileSystemHandle,\n\t\t) => {\n\t\t\tconst { ignored } = ig.test(filepath);\n\n\t\t\tif (filepath.endsWith(\".git\")) {\n\t\t\t\treturn false;\n\t\t\t}\n\n\t\t\tif (filepath.includes(\".git/\")) {\n\t\t\t\treturn false;\n\t\t\t}\n\n\t\t\treturn !ignored;\n\t\t},\n\t};\n};\n\nexport const distFilter: FilterFn = async () => {\n\tconst isDist = (filepath: string) => {\n\t\tif (filepath.includes(\"/dist/\")) {\n\t\t\treturn true;\n\t\t}\n\t\tif (filepath.includes(\"/out/\")) {\n\t\t\treturn true;\n\t\t}\n\t\tif (filepath.includes(\"/build/\")) {\n\t\t\treturn true;\n\t\t}\n\t\tif (filepath.includes(\"/vendor/\")) {\n\t\t\treturn true;\n\t\t}\n\t\tif (filepath.includes(\"/node_modules/\")) {\n\t\t\treturn true;\n\t\t}\n\t\tif (filepath.includes(\"/.next/\")) {\n\t\t\treturn true;\n\t\t}\n\t\treturn false;\n\t};\n\treturn {\n\t\tshouldIncludeFile: async (filepath: string, _handle: FileSystemHandle) => {\n\t\t\tif (isDist(filepath)) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t\treturn true;\n\t\t},\n\t\tshouldProcessDirectory: async (\n\t\t\tfilepath: string,\n\t\t\t_handle: FileSystemHandle,\n\t\t) => {\n\t\t\tif (isDist(filepath)) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t\treturn true;\n\t\t},\n\t};\n};\n\nexport const miscFilter: FilterFn = async () => {\n\tconst isMisc = (filepath: string) => {\n\t\tif (filepath.endsWith(\".DS_Store\")) {\n\t\t\treturn true;\n\t\t}\n\t\tif (filepath.endsWith(\".crswap\")) {\n\t\t\treturn true;\n\t\t}\n\t\treturn false;\n\t};\n\treturn {\n\t\tshouldIncludeFile: async (filepath: string, _handle: FileSystemHandle) => {\n\t\t\tif (isMisc(filepath)) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t\treturn true;\n\t\t},\n\t\tshouldProcessDirectory: async (\n\t\t\tfilepath: string,\n\t\t\t_handle: FileSystemHandle,\n\t\t) => {\n\t\t\tif (isMisc(filepath)) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t\treturn true;\n\t\t},\n\t};\n};\n\nexport const commonFilters = [distFilter, miscFilter, gitFilter];\n\n/* eslint-disable no-undef */\nexport const processDirectory = async (\n\tdirectoryHandle: FileSystemDirectoryHandle,\n\tdirectoryPath: string,\n\tfilters: Filter[],\n\tincludeFiles: Map<string, FileSystemFileHandle>,\n\tignoreFilePaths: Set<string>,\n): Promise<Map<string, FileSystemFileHandle>> => {\n\t// Early exit if directory is already ignored\n\tif (ignoreFilePaths.has(directoryPath)) {\n\t\treturn includeFiles;\n\t}\n\n\t// Check directory against filters first\n\tfor (const filter of filters) {\n\t\tif (\n\t\t\t(await filter.shouldProcessDirectory(directoryPath, directoryHandle)) ===\n\t\t\tfalse\n\t\t) {\n\t\t\tignoreFilePaths.add(directoryPath);\n\t\t\treturn includeFiles;\n\t\t}\n\t}\n\n\t// Process files first\n\tconst filePromises: Promise<void>[] = [];\n\t// @ts-ignore\n\tfor await (const entry of directoryHandle.values()) {\n\t\tif (entry.kind === \"file\") {\n\t\t\tconst path = `${directoryPath}/${entry.name}`;\n\t\t\tif (!ignoreFilePaths.has(path)) {\n\t\t\t\tfilePromises.push(\n\t\t\t\t\t(async () => {\n\t\t\t\t\t\tlet shouldInclude = true;\n\t\t\t\t\t\tfor (const filter of filters) {\n\t\t\t\t\t\t\tif (\n\t\t\t\t\t\t\t\t(await filter.shouldIncludeFile(\n\t\t\t\t\t\t\t\t\tpath,\n\t\t\t\t\t\t\t\t\tentry as FileSystemFileHandle,\n\t\t\t\t\t\t\t\t)) === false\n\t\t\t\t\t\t\t) {\n\t\t\t\t\t\t\t\tshouldInclude = false;\n\t\t\t\t\t\t\t\tignoreFilePaths.add(path);\n\t\t\t\t\t\t\t\tincludeFiles.delete(path);\n\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif (shouldInclude) {\n\t\t\t\t\t\t\tincludeFiles.set(path, entry as FileSystemFileHandle);\n\t\t\t\t\t\t}\n\t\t\t\t\t})(),\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\t}\n\tawait Promise.all(filePromises);\n\n\t// Process directories\n\tconst dirPromises: Promise<Map<string, FileSystemFileHandle>>[] = [];\n\t// @ts-ignore\n\tfor await (const entry of directoryHandle.values()) {\n\t\tif (entry.kind === \"directory\") {\n\t\t\tconst path = `${directoryPath}/${entry.name}`;\n\t\t\tif (!ignoreFilePaths.has(path)) {\n\t\t\t\tdirPromises.push(\n\t\t\t\t\tprocessDirectory(\n\t\t\t\t\t\tentry as FileSystemDirectoryHandle,\n\t\t\t\t\t\tpath,\n\t\t\t\t\t\tfilters,\n\t\t\t\t\t\tincludeFiles,\n\t\t\t\t\t\tignoreFilePaths,\n\t\t\t\t\t),\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\t}\n\tawait Promise.all(dirPromises);\n\n\treturn includeFiles;\n};\n\nconst useDebounce = <T>(value: T, delay: number): T => {\n\tconst [debouncedValue, setDebouncedValue] = useState<T>(value);\n\n\tuseEffect(() => {\n\t\tconst handler = setTimeout(() => {\n\t\t\tsetDebouncedValue(value);\n\t\t}, delay);\n\n\t\treturn () => {\n\t\t\tclearTimeout(handler);\n\t\t};\n\t}, [value, delay]);\n\n\treturn debouncedValue;\n};\n\nexport interface UseFileHandlingHookProps {\n\tfilters?: FilterFn[];\n\tonFilesAdded?: (\n\t\tnewFiles: Map<string, string>,\n\t\tpreviousFiles: Map<string, string>,\n\t) => void;\n\tonFilesChanged?: (\n\t\tchangedFiles: Map<string, string>,\n\t\tpreviousFiles: Map<string, string>,\n\t) => void;\n\tonFilesDeleted?: (\n\t\tdeletedFiles: Map<string, string>,\n\t\tpreviousFiles: Map<string, string>,\n\t) => void;\n\tpollInterval?: number;\n\tbatchSize?: number;\n\tdebounceInterval?: number;\n\t/** Time in milliseconds to cache file contents before re-reading from disk. Defaults to 5000ms (5 seconds) */\n\tfileCacheTtl?: number;\n}\n\nconst DEFAULT_POLL_INTERVAL = 100;\nconst DEFAULT_BATCH_SIZE = 50;\nconst DEFAULT_DEBOUNCE_INTERVAL = 50;\nconst DEFAULT_FILE_CACHE_TTL = 5000; // 5 seconds\n\nexport const useFileSystem = (props: UseFileHandlingHookProps) => {\n\tconst {\n\t\tonFilesAdded: onAddFile,\n\t\tonFilesChanged: onChangeFile,\n\t\tonFilesDeleted: onDeleteFile,\n\t\tbatchSize = DEFAULT_BATCH_SIZE,\n\t\tdebounceInterval = DEFAULT_DEBOUNCE_INTERVAL,\n\t} = props;\n\n\tconst watchedDirectoriesRef = useRef<Map<string, FileSystemDirectoryHandle>>(\n\t\tnew Map(),\n\t);\n\tconst handlesRef = useRef<Map<string, FileSystemFileHandle>>(new Map());\n\tconst previousHandlesRef = useRef<Set<string>>(new Set());\n\tconst filesMapRef = useRef<Record<string, string>>({});\n\tconst previousFilesMapRef = useRef<Record<string, string>>({});\n\tconst intervalRef = useRef<NodeJS.Timeout | null>(null);\n\tconst [files, setFiles] = useState<Map<string, string>>(new Map());\n\tconst [isProcessing, setIsProcessing] = useState(false);\n\tconst [isPolling, setIsPolling] = useState(false);\n\n\t// Add a cache for file contents\n\tconst fileContentsCache = useRef<\n\t\tMap<string, { content: string; timestamp: number }>\n\t>(new Map());\n\tconst fileCacheTtl = props.fileCacheTtl || DEFAULT_FILE_CACHE_TTL;\n\n\tconst clearWatchedDirectories = useCallback(() => {\n\t\tif (intervalRef.current) {\n\t\t\tclearInterval(intervalRef.current);\n\t\t\tintervalRef.current = null;\n\t\t}\n\n\t\twatchedDirectoriesRef.current.clear();\n\t\thandlesRef.current.clear();\n\t\tpreviousHandlesRef.current.clear();\n\t\tfilesMapRef.current = {};\n\t\tpreviousFilesMapRef.current = {};\n\t\tfileContentsCache.current.clear();\n\t\tsetFiles(new Map());\n\t\tsetIsPolling(false);\n\t}, []);\n\n\tconst processWatchedDirectories = useCallback(async () => {\n\t\tconst temporaryHandles = new Map<string, FileSystemFileHandle>();\n\t\tconst ignoreFilePaths = new Set<string>();\n\n\t\tconst filterFns: FilterFn[] = props.filters || commonFilters;\n\t\tconst filters: Filter[] = [];\n\n\t\tfor (const filterFn of filterFns) {\n\t\t\tfilters.push(await filterFn());\n\t\t}\n\n\t\tconst dirPromises: Promise<Map<string, FileSystemFileHandle>>[] = [];\n\t\twatchedDirectoriesRef.current.forEach((directoryHandle, directoryPath) => {\n\t\t\tdirPromises.push(\n\t\t\t\tprocessDirectory(\n\t\t\t\t\tdirectoryHandle,\n\t\t\t\t\tdirectoryPath,\n\t\t\t\t\tfilters,\n\t\t\t\t\ttemporaryHandles,\n\t\t\t\t\tignoreFilePaths,\n\t\t\t\t),\n\t\t\t);\n\t\t});\n\n\t\tawait Promise.all(dirPromises);\n\n\t\ttemporaryHandles.forEach((_, filePath) => {\n\t\t\tif (ignoreFilePaths.has(filePath)) {\n\t\t\t\ttemporaryHandles.delete(filePath);\n\t\t\t}\n\t\t});\n\n\t\thandlesRef.current = temporaryHandles;\n\t}, [props.filters]);\n\n\tconst processFiles = useCallback(async () => {\n\t\tpreviousFilesMapRef.current = {\n\t\t\t...(filesMapRef.current || {}),\n\t\t};\n\n\t\tconst seenFiles = new Set<string>();\n\t\tconst changedFiles = new Map<string, string>();\n\t\tconst deletedFilesSet = new Set<string>();\n\t\tlet rerender = false;\n\n\t\tconst now = Date.now();\n\t\tconst handles = Array.from(handlesRef.current);\n\t\tconst newFilesMap: Record<string, string> = {};\n\n\t\tfor (let i = 0; i < handles.length; i += batchSize) {\n\t\t\tconst batch = handles.slice(i, i + batchSize);\n\t\t\tawait Promise.all(\n\t\t\t\tbatch.map(async ([filePath, handle]) => {\n\t\t\t\t\tif (handle.kind === \"file\") {\n\t\t\t\t\t\tseenFiles.add(filePath);\n\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tconst cached = fileContentsCache.current.get(filePath);\n\t\t\t\t\t\t\tif (cached && now - cached.timestamp < fileCacheTtl) {\n\t\t\t\t\t\t\t\tnewFilesMap[filePath] = cached.content;\n\t\t\t\t\t\t\t\tif (cached.content !== previousFilesMapRef.current[filePath]) {\n\t\t\t\t\t\t\t\t\tchangedFiles.set(filePath, cached.content);\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tconst file = await (handle as FileSystemFileHandle).getFile();\n\t\t\t\t\t\t\tconst text = await file.text();\n\t\t\t\t\t\t\tfileContentsCache.current.set(filePath, {\n\t\t\t\t\t\t\t\tcontent: text,\n\t\t\t\t\t\t\t\ttimestamp: now,\n\t\t\t\t\t\t\t});\n\n\t\t\t\t\t\t\tnewFilesMap[filePath] = text;\n\t\t\t\t\t\t\tif (text !== previousFilesMapRef.current[filePath]) {\n\t\t\t\t\t\t\t\tchangedFiles.set(filePath, text);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} catch (error) {\n\t\t\t\t\t\t\thandlesRef.current.delete(filePath);\n\t\t\t\t\t\t\tdeletedFilesSet.add(filePath);\n\t\t\t\t\t\t\tfileContentsCache.current.delete(filePath);\n\t\t\t\t\t\t\trerender = true;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}),\n\t\t\t);\n\t\t}\n\n\t\t// Clean up old cache entries\n\t\tconst cacheEntries = Array.from(fileContentsCache.current.entries());\n\t\tfor (const [path, { timestamp }] of cacheEntries) {\n\t\t\tif (now - timestamp > fileCacheTtl) {\n\t\t\t\tfileContentsCache.current.delete(path);\n\t\t\t}\n\t\t}\n\n\t\tconst deletedFiles = Array.from([\n\t\t\t...Array.from(previousHandlesRef.current),\n\t\t\t...Array.from(deletedFilesSet),\n\t\t]).filter((filePath) => !seenFiles.has(filePath));\n\n\t\tconst addedFiles = Array.from(seenFiles).filter(\n\t\t\t(filePath) => !previousHandlesRef.current.has(filePath),\n\t\t);\n\n\t\tconst addedFilesMap = new Map<string, string>();\n\n\t\tfor (const addedFile of addedFiles) {\n\t\t\taddedFilesMap.set(addedFile, filesMapRef.current[addedFile]);\n\t\t}\n\n\t\tconst deletedFilesMap = new Map<string, string>();\n\n\t\tfor (const deletedFile of deletedFiles) {\n\t\t\tdeletedFilesMap.set(deletedFile, filesMapRef.current[deletedFile]);\n\t\t\thandlesRef.current.delete(deletedFile);\n\t\t\tdelete filesMapRef.current[deletedFile];\n\t\t}\n\n\t\tconst previousFiles = new Map<string, string>(\n\t\t\tArray.from(previousHandlesRef.current).map((filePath) => [\n\t\t\t\tfilePath,\n\t\t\t\tfilesMapRef.current[filePath],\n\t\t\t]),\n\t\t);\n\n\t\tif (changedFiles.size > 0) {\n\t\t\trerender = true;\n\t\t\tonChangeFile?.(changedFiles, previousFiles);\n\t\t}\n\n\t\tif (deletedFiles.length > 0) {\n\t\t\trerender = true;\n\t\t\tonDeleteFile?.(deletedFilesMap, previousFiles);\n\t\t}\n\n\t\tif (addedFiles.length > 0) {\n\t\t\trerender = true;\n\t\t\tonAddFile?.(addedFilesMap, previousFiles);\n\t\t}\n\n\t\tpreviousHandlesRef.current = seenFiles;\n\t\tfilesMapRef.current = newFilesMap;\n\n\t\tif (rerender) {\n\t\t\tsetFiles(new Map(Object.entries(filesMapRef.current)));\n\t\t}\n\t}, [onAddFile, onChangeFile, onDeleteFile, batchSize, fileCacheTtl]);\n\n\t// Debounce the files state update\n\tconst debouncedFiles = useDebounce(files, debounceInterval);\n\n\tconst startPollingWatchedDirectories = useCallback(() => {\n\t\tlet processingPromise: Promise<void> | null = null;\n\n\t\t// Only start a new interval if one isn't already running\n\t\tif (!intervalRef.current) {\n\t\t\tintervalRef.current = setInterval(async () => {\n\t\t\t\tif (isProcessing || processingPromise) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tsetIsProcessing(true);\n\t\t\t\tprocessingPromise = (async () => {\n\t\t\t\t\ttry {\n\t\t\t\t\t\tawait processWatchedDirectories();\n\t\t\t\t\t\tawait processFiles();\n\t\t\t\t\t} finally {\n\t\t\t\t\t\tsetIsProcessing(false);\n\t\t\t\t\t\tprocessingPromise = null;\n\t\t\t\t\t}\n\t\t\t\t})();\n\t\t\t}, props.pollInterval || DEFAULT_POLL_INTERVAL);\n\t\t\tsetIsPolling(true);\n\t\t}\n\t}, [\n\t\tprocessWatchedDirectories,\n\t\tprocessFiles,\n\t\tisProcessing,\n\t\tprops.pollInterval,\n\t]);\n\n\tconst stopPolling = useCallback(() => {\n\t\tif (intervalRef.current) {\n\t\t\tclearInterval(intervalRef.current);\n\t\t\tintervalRef.current = null;\n\t\t\tsetIsPolling(false);\n\t\t}\n\t}, []);\n\n\tconst startPolling = useCallback(() => {\n\t\tif (watchedDirectoriesRef.current.size > 0) {\n\t\t\tstartPollingWatchedDirectories();\n\t\t}\n\t}, [startPollingWatchedDirectories]);\n\n\tconst onDirectorySelection = async () => {\n\t\ttry {\n\t\t\t// @ts-ignore\n\t\t\tconst directoryHandle = await window.showDirectoryPicker();\n\n\t\t\tif (!directoryHandle) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst initialDirectoryPath = directoryHandle.name;\n\t\t\twatchedDirectoriesRef.current.set(initialDirectoryPath, directoryHandle);\n\t\t\tstartPollingWatchedDirectories();\n\t\t} catch (error) {\n\t\t\tconsole.error(\"Error during directory selection:\");\n\t\t\tconsole.error(error);\n\t\t}\n\t};\n\n\tconst writeFile = useCallback(\n\t\tasync (\n\t\t\tpath: string,\n\t\t\tdata: string | ArrayBuffer | Blob,\n\t\t\toptions: FileWriteOptions = {},\n\t\t) => {\n\t\t\t// Validate path\n\t\t\tif (!path || typeof path !== \"string\") {\n\t\t\t\tthrow new Error(\"Invalid file path\");\n\t\t\t}\n\n\t\t\tconst { create = true, truncate = false } = options;\n\t\t\tconst handle = handlesRef.current.get(path);\n\n\t\t\tif (!handle) {\n\t\t\t\tif (!create) {\n\t\t\t\t\tthrow new Error(`File not found: ${path}`);\n\t\t\t\t}\n\t\t\t\t// Create new file\n\t\t\t\tconst directoryPath = path.substring(0, path.lastIndexOf(\"/\"));\n\t\t\t\tconst fileName = path.substring(path.lastIndexOf(\"/\") + 1);\n\n\t\t\t\t// Validate directory path\n\t\t\t\tif (!(directoryPath && fileName)) {\n\t\t\t\t\tthrow new Error(\"Invalid file path structure\");\n\t\t\t\t}\n\n\t\t\t\tconst dirHandle = watchedDirectoriesRef.current.get(directoryPath);\n\n\t\t\t\tif (!dirHandle) {\n\t\t\t\t\tthrow new Error(`Directory not found: ${directoryPath}`);\n\t\t\t\t}\n\n\t\t\t\tconst fileHandle = await dirHandle.getFileHandle(fileName, {\n\t\t\t\t\tcreate: true,\n\t\t\t\t});\n\t\t\t\thandlesRef.current.set(path, fileHandle);\n\t\t\t}\n\n\t\t\tconst fileHandle = handlesRef.current.get(path);\n\t\t\tif (!fileHandle) {\n\t\t\t\tthrow new Error(`File not found: ${path}`);\n\t\t\t}\n\n\t\t\t// Double-check we're writing to the correct file\n\t\t\tif (fileHandle.name !== path.split(\"/\").pop()) {\n\t\t\t\tthrow new Error(\"File path mismatch\");\n\t\t\t}\n\n\t\t\tconst writable = await fileHandle.createWritable({\n\t\t\t\tkeepExistingData: !truncate,\n\t\t\t});\n\n\t\t\ttry {\n\t\t\t\tawait writable.write(data);\n\t\t\t\tawait writable.close();\n\n\t\t\t\t// Update cache and trigger change events\n\t\t\t\tconst content =\n\t\t\t\t\ttypeof data === \"string\" ? data : await new Response(data).text();\n\t\t\t\tfileContentsCache.current.set(path, {\n\t\t\t\t\tcontent,\n\t\t\t\t\ttimestamp: Date.now(),\n\t\t\t\t});\n\n\t\t\t\tconst changedFiles = new Map([[path, content]]);\n\t\t\t\tconst previousFiles = new Map(\n\t\t\t\t\tArray.from(previousHandlesRef.current).map((filePath) => [\n\t\t\t\t\t\tfilePath,\n\t\t\t\t\t\tfilesMapRef.current[filePath],\n\t\t\t\t\t]),\n\t\t\t\t);\n\n\t\t\t\tonChangeFile?.(changedFiles, previousFiles);\n\t\t\t\tfilesMapRef.current[path] = content;\n\t\t\t\tsetFiles(new Map(Object.entries(filesMapRef.current)));\n\t\t\t} catch (error) {\n\t\t\t\tawait writable.abort();\n\t\t\t\tthrow error;\n\t\t\t}\n\t\t},\n\t\t[onChangeFile],\n\t);\n\n\tconst createFile = useCallback(\n\t\tasync (path: string, initialData?: string | ArrayBuffer | Blob) => {\n\t\t\tconst directoryPath = path.substring(0, path.lastIndexOf(\"/\"));\n\t\t\tconst fileName = path.substring(path.lastIndexOf(\"/\") + 1);\n\t\t\tconst dirHandle = watchedDirectoriesRef.current.get(directoryPath);\n\n\t\t\tif (!dirHandle) {\n\t\t\t\tthrow new Error(`Directory not found: ${directoryPath}`);\n\t\t\t}\n\n\t\t\tconst fileHandle = await dirHandle.getFileHandle(fileName, {\n\t\t\t\tcreate: true,\n\t\t\t});\n\t\t\thandlesRef.current.set(path, fileHandle);\n\n\t\t\tif (initialData) {\n\t\t\t\tawait writeFile(path, initialData, { truncate: true });\n\t\t\t}\n\n\t\t\treturn fileHandle;\n\t\t},\n\t\t[writeFile],\n\t);\n\n\tconst deleteFile = useCallback(\n\t\tasync (path: string) => {\n\t\t\tconst handle = handlesRef.current.get(path);\n\t\t\tif (!handle) {\n\t\t\t\tthrow new Error(`File not found: ${path}`);\n\t\t\t}\n\n\t\t\tconst directoryPath = path.substring(0, path.lastIndexOf(\"/\"));\n\t\t\tconst fileName = path.substring(path.lastIndexOf(\"/\") + 1);\n\t\t\tconst dirHandle = watchedDirectoriesRef.current.get(directoryPath);\n\n\t\t\tif (!dirHandle) {\n\t\t\t\tthrow new Error(`Directory not found: ${directoryPath}`);\n\t\t\t}\n\n\t\t\tawait dirHandle.removeEntry(fileName);\n\t\t\thandlesRef.current.delete(path);\n\t\t\tfileContentsCache.current.delete(path);\n\n\t\t\tconst deletedFiles = new Map([[path, filesMapRef.current[path]]]);\n\t\t\tconst previousFiles = new Map(\n\t\t\t\tArray.from(previousHandlesRef.current).map((filePath) => [\n\t\t\t\t\tfilePath,\n\t\t\t\t\tfilesMapRef.current[filePath],\n\t\t\t\t]),\n\t\t\t);\n\n\t\t\tdelete filesMapRef.current[path];\n\t\t\tpreviousHandlesRef.current.delete(path);\n\n\t\t\tonDeleteFile?.(deletedFiles, previousFiles);\n\t\t\tsetFiles(new Map(Object.entries(filesMapRef.current)));\n\t\t},\n\t\t[onDeleteFile],\n\t);\n\n\tuseUnmount(() => {\n\t\tclearWatchedDirectories();\n\t});\n\n\treturn {\n\t\thandles: handlesRef.current,\n\t\tonDirectorySelection,\n\t\tonClear: clearWatchedDirectories,\n\t\tfiles: debouncedFiles,\n\t\tsetFiles,\n\t\tisProcessing,\n\t\tisBrowserSupported: window !== undefined && \"showDirectoryPicker\" in window,\n\t\twriteFile,\n\t\tcreateFile,\n\t\tdeleteFile,\n\t\tstopPolling,\n\t\tstartPolling,\n\t\tisPolling,\n\t};\n};\n\nexport const useFs = useFileSystem;\n\n// below is copied from usehooks-ts\n\nconst useEffectOnce = (effect: EffectCallback) => {\n\tuseEffect(effect, []);\n};\n\nconst useUnmount = (fn: () => void): void => {\n\tconst fnRef = useRef(fn);\n\n\tfnRef.current = fn;\n\n\tuseEffectOnce(() => () => fnRef.current());\n};\n"],
  "mappings": "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",
  "names": ["index_exports", "__export", "commonFilters", "distFilter", "gitFilter", "miscFilter", "processDirectory", "useFileSystem", "useFs", "__toCommonJS", "import_ignore", "import_react", "ig", "ignore", "gitIgnoreLoaded", "filepath", "handle", "text", "ignored", "_handle", "isDist", "isMisc", "directoryHandle", "directoryPath", "filters", "includeFiles", "ignoreFilePaths", "filter", "filePromises", "entry", "path", "shouldInclude", "dirPromises", "useDebounce", "value", "delay", "debouncedValue", "setDebouncedValue", "handler", "DEFAULT_POLL_INTERVAL", "DEFAULT_BATCH_SIZE", "DEFAULT_DEBOUNCE_INTERVAL", "DEFAULT_FILE_CACHE_TTL", "props", "onAddFile", "onChangeFile", "onDeleteFile", "batchSize", "debounceInterval", "watchedDirectoriesRef", "handlesRef", "previousHandlesRef", "filesMapRef", "previousFilesMapRef", "intervalRef", "files", "setFiles", "isProcessing", "setIsProcessing", "isPolling", "setIsPolling", "fileContentsCache", "fileCacheTtl", "clearWatchedDirectories", "processWatchedDirectories", "temporaryHandles", "filterFns", "filterFn", "_", "filePath", "processFiles", "seenFiles", "changedFiles", "deletedFilesSet", "rerender", "now", "handles", "newFilesMap", "i", "batch", "cached", "cacheEntries", "timestamp", "deletedFiles", "addedFiles", "addedFilesMap", "addedFile", "deletedFilesMap", "deletedFile", "previousFiles", "debouncedFiles", "startPollingWatchedDirectories", "processingPromise", "stopPolling", "startPolling", "onDirectorySelection", "initialDirectoryPath", "error", "writeFile", "data", "options", "create", "truncate", "fileName", "dirHandle", "fileHandle", "writable", "content", "createFile", "initialData", "deleteFile", "useUnmount", "useEffectOnce", "effect", "fn", "fnRef"]
}
