@fgv/ts-web-extras 5.1.0-3 → 5.1.0-30

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (275) hide show
  1. package/dist/packlets/crypto-utils/browserCryptoProvider.js +330 -18
  2. package/dist/packlets/crypto-utils/browserCryptoProvider.js.map +1 -1
  3. package/dist/packlets/crypto-utils/index.js +12 -0
  4. package/dist/packlets/crypto-utils/index.js.map +1 -1
  5. package/dist/packlets/file-tree/directoryHandleStore.js.map +1 -1
  6. package/dist/packlets/file-tree/fileApiTreeAccessors.js +1 -1
  7. package/dist/packlets/file-tree/fileApiTreeAccessors.js.map +1 -1
  8. package/dist/packlets/file-tree/fileSystemAccessTreeAccessors.js +2 -2
  9. package/dist/packlets/file-tree/fileSystemAccessTreeAccessors.js.map +1 -1
  10. package/dist/packlets/file-tree/httpTreeAccessors.js +74 -44
  11. package/dist/packlets/file-tree/httpTreeAccessors.js.map +1 -1
  12. package/dist/packlets/file-tree/localStorageTreeAccessors.js.map +1 -1
  13. package/dist/packlets/helpers/fileTreeHelpers.js +1 -1
  14. package/dist/packlets/helpers/fileTreeHelpers.js.map +1 -1
  15. package/dist/ts-web-extras.d.ts +141 -10
  16. package/dist/tsdoc-metadata.json +1 -1
  17. package/eslint.config.js +32 -0
  18. package/lib/packlets/crypto-utils/browserCryptoProvider.d.ts +115 -6
  19. package/lib/packlets/crypto-utils/browserCryptoProvider.d.ts.map +1 -1
  20. package/lib/packlets/crypto-utils/browserCryptoProvider.js +329 -17
  21. package/lib/packlets/crypto-utils/browserCryptoProvider.js.map +1 -1
  22. package/lib/packlets/crypto-utils/index.d.ts +13 -0
  23. package/lib/packlets/crypto-utils/index.d.ts.map +1 -1
  24. package/lib/packlets/crypto-utils/index.js +13 -0
  25. package/lib/packlets/crypto-utils/index.js.map +1 -1
  26. package/lib/packlets/file-tree/directoryHandleStore.d.ts +2 -2
  27. package/lib/packlets/file-tree/directoryHandleStore.d.ts.map +1 -1
  28. package/lib/packlets/file-tree/directoryHandleStore.js.map +1 -1
  29. package/lib/packlets/file-tree/fileApiTreeAccessors.d.ts +2 -2
  30. package/lib/packlets/file-tree/fileApiTreeAccessors.d.ts.map +1 -1
  31. package/lib/packlets/file-tree/fileApiTreeAccessors.js +1 -1
  32. package/lib/packlets/file-tree/fileApiTreeAccessors.js.map +1 -1
  33. package/lib/packlets/file-tree/fileSystemAccessTreeAccessors.d.ts.map +1 -1
  34. package/lib/packlets/file-tree/fileSystemAccessTreeAccessors.js +2 -2
  35. package/lib/packlets/file-tree/fileSystemAccessTreeAccessors.js.map +1 -1
  36. package/lib/packlets/file-tree/httpTreeAccessors.d.ts +6 -0
  37. package/lib/packlets/file-tree/httpTreeAccessors.d.ts.map +1 -1
  38. package/lib/packlets/file-tree/httpTreeAccessors.js +74 -44
  39. package/lib/packlets/file-tree/httpTreeAccessors.js.map +1 -1
  40. package/lib/packlets/file-tree/localStorageTreeAccessors.js.map +1 -1
  41. package/lib/packlets/helpers/fileTreeHelpers.d.ts +1 -1
  42. package/lib/packlets/helpers/fileTreeHelpers.d.ts.map +1 -1
  43. package/lib/packlets/helpers/fileTreeHelpers.js +6 -6
  44. package/lib/packlets/helpers/fileTreeHelpers.js.map +1 -1
  45. package/package.json +16 -15
  46. package/.rush/temp/61c1d4a91d98f048b475a5bb3e6541c5d27819de.tar.log +0 -237
  47. package/.rush/temp/chunked-rush-logs/ts-web-extras.build.chunks.jsonl +0 -55
  48. package/.rush/temp/operation/build/all.log +0 -55
  49. package/.rush/temp/operation/build/log-chunks.jsonl +0 -55
  50. package/.rush/temp/operation/build/state.json +0 -3
  51. package/.rush/temp/shrinkwrap-deps.json +0 -635
  52. package/CHANGELOG.md +0 -23
  53. package/config/api-extractor.json +0 -343
  54. package/config/jest.config.json +0 -19
  55. package/config/rig.json +0 -16
  56. package/config/typedoc.json +0 -6
  57. package/dist/test/mocks/idb-keyval.js +0 -6
  58. package/dist/test/mocks/idb-keyval.js.map +0 -1
  59. package/dist/test/setupTests.js +0 -74
  60. package/dist/test/setupTests.js.map +0 -1
  61. package/dist/test/unit/browserHashProvider.test.js +0 -140
  62. package/dist/test/unit/browserHashProvider.test.js.map +0 -1
  63. package/dist/test/unit/directoryHandleStore.test.js +0 -190
  64. package/dist/test/unit/directoryHandleStore.test.js.map +0 -1
  65. package/dist/test/unit/fileApiTreeAccessors.test.js +0 -1188
  66. package/dist/test/unit/fileApiTreeAccessors.test.js.map +0 -1
  67. package/dist/test/unit/fileApiTypes.test.js +0 -472
  68. package/dist/test/unit/fileApiTypes.test.js.map +0 -1
  69. package/dist/test/unit/fileSystemAccessTreeAccessors.test.js +0 -622
  70. package/dist/test/unit/fileSystemAccessTreeAccessors.test.js.map +0 -1
  71. package/dist/test/unit/fileTreeHelpers.test.js +0 -590
  72. package/dist/test/unit/fileTreeHelpers.test.js.map +0 -1
  73. package/dist/test/unit/httpTreeAccessors.test.js +0 -1229
  74. package/dist/test/unit/httpTreeAccessors.test.js.map +0 -1
  75. package/dist/test/unit/localStorageTreeAccessors.test.js +0 -812
  76. package/dist/test/unit/localStorageTreeAccessors.test.js.map +0 -1
  77. package/dist/test/unit/urlParams.test.js +0 -393
  78. package/dist/test/unit/urlParams.test.js.map +0 -1
  79. package/dist/test/utils/fileSystemAccessMocks.js +0 -271
  80. package/dist/test/utils/fileSystemAccessMocks.js.map +0 -1
  81. package/dist/test/utils/testHelpers.js +0 -124
  82. package/dist/test/utils/testHelpers.js.map +0 -1
  83. package/docs/@fgv/namespaces/CryptoUtils/README.md +0 -18
  84. package/docs/@fgv/namespaces/CryptoUtils/classes/BrowserCryptoProvider.md +0 -203
  85. package/docs/@fgv/namespaces/CryptoUtils/classes/BrowserHashProvider.md +0 -63
  86. package/docs/@fgv/namespaces/CryptoUtils/functions/createBrowserCryptoProvider.md +0 -18
  87. package/docs/@fgv/namespaces/FileTreeHelpers/README.md +0 -19
  88. package/docs/@fgv/namespaces/FileTreeHelpers/functions/extractFileListMetadata.md +0 -23
  89. package/docs/@fgv/namespaces/FileTreeHelpers/functions/extractFileMetadata.md +0 -23
  90. package/docs/@fgv/namespaces/FileTreeHelpers/functions/fromDirectoryUpload.md +0 -33
  91. package/docs/@fgv/namespaces/FileTreeHelpers/functions/fromFileList.md +0 -33
  92. package/docs/@fgv/namespaces/FileTreeHelpers/functions/getOriginalFile.md +0 -25
  93. package/docs/@fgv/namespaces/FileTreeHelpers/variables/defaultFileApiTreeInitParams.md +0 -11
  94. package/docs/README.md +0 -78
  95. package/docs/classes/DirectoryHandleStore.md +0 -116
  96. package/docs/classes/FileApiTreeAccessors.md +0 -286
  97. package/docs/classes/FileSystemAccessTreeAccessors.md +0 -557
  98. package/docs/classes/HttpTreeAccessors.md +0 -508
  99. package/docs/classes/LocalStorageTreeAccessors.md +0 -520
  100. package/docs/functions/exportAsJson.md +0 -23
  101. package/docs/functions/exportUsingFileSystemAPI.md +0 -26
  102. package/docs/functions/extractDirectoryPath.md +0 -23
  103. package/docs/functions/isDirectoryHandle.md +0 -23
  104. package/docs/functions/isFileHandle.md +0 -23
  105. package/docs/functions/isFilePath.md +0 -21
  106. package/docs/functions/parseContextFilter.md +0 -22
  107. package/docs/functions/parseQualifierDefaults.md +0 -22
  108. package/docs/functions/parseResourceTypes.md +0 -22
  109. package/docs/functions/parseUrlParameters.md +0 -15
  110. package/docs/functions/safeShowDirectoryPicker.md +0 -24
  111. package/docs/functions/safeShowOpenFilePicker.md +0 -24
  112. package/docs/functions/safeShowSaveFilePicker.md +0 -24
  113. package/docs/functions/supportsFileSystemAccess.md +0 -23
  114. package/docs/interfaces/FilePickerAcceptType.md +0 -16
  115. package/docs/interfaces/FileSystemCreateWritableOptions.md +0 -15
  116. package/docs/interfaces/FileSystemDirectoryHandle.md +0 -187
  117. package/docs/interfaces/FileSystemFileHandle.md +0 -106
  118. package/docs/interfaces/FileSystemGetDirectoryOptions.md +0 -15
  119. package/docs/interfaces/FileSystemGetFileOptions.md +0 -15
  120. package/docs/interfaces/FileSystemHandle.md +0 -69
  121. package/docs/interfaces/FileSystemHandlePermissionDescriptor.md +0 -15
  122. package/docs/interfaces/FileSystemRemoveOptions.md +0 -15
  123. package/docs/interfaces/FileSystemWritableFileStream.md +0 -127
  124. package/docs/interfaces/IDirectoryHandleTreeInitializer.md +0 -17
  125. package/docs/interfaces/IFileHandleTreeInitializer.md +0 -16
  126. package/docs/interfaces/IFileListTreeInitializer.md +0 -15
  127. package/docs/interfaces/IFileMetadata.md +0 -19
  128. package/docs/interfaces/IFileSystemAccessTreeParams.md +0 -30
  129. package/docs/interfaces/IFsAccessApis.md +0 -57
  130. package/docs/interfaces/IHttpTreeParams.md +0 -32
  131. package/docs/interfaces/ILocalStorageTreeParams.md +0 -30
  132. package/docs/interfaces/IUrlConfigOptions.md +0 -27
  133. package/docs/interfaces/ShowDirectoryPickerOptions.md +0 -17
  134. package/docs/interfaces/ShowOpenFilePickerOptions.md +0 -19
  135. package/docs/interfaces/ShowSaveFilePickerOptions.md +0 -19
  136. package/docs/type-aliases/TreeInitializer.md +0 -11
  137. package/docs/type-aliases/WellKnownDirectory.md +0 -11
  138. package/docs/type-aliases/WindowWithFsAccess.md +0 -11
  139. package/docs/variables/DEFAULT_DIRECTORY_HANDLE_DB.md +0 -11
  140. package/docs/variables/DEFAULT_DIRECTORY_HANDLE_STORE.md +0 -11
  141. package/etc/ts-web-extras.api.md +0 -433
  142. package/lib/test/mocks/idb-keyval.d.ts +0 -6
  143. package/lib/test/mocks/idb-keyval.d.ts.map +0 -1
  144. package/lib/test/mocks/idb-keyval.js +0 -9
  145. package/lib/test/mocks/idb-keyval.js.map +0 -1
  146. package/lib/test/setupTests.d.ts +0 -2
  147. package/lib/test/setupTests.d.ts.map +0 -1
  148. package/lib/test/setupTests.js +0 -76
  149. package/lib/test/setupTests.js.map +0 -1
  150. package/lib/test/unit/browserHashProvider.test.d.ts +0 -2
  151. package/lib/test/unit/browserHashProvider.test.d.ts.map +0 -1
  152. package/lib/test/unit/browserHashProvider.test.js +0 -142
  153. package/lib/test/unit/browserHashProvider.test.js.map +0 -1
  154. package/lib/test/unit/directoryHandleStore.test.d.ts +0 -2
  155. package/lib/test/unit/directoryHandleStore.test.d.ts.map +0 -1
  156. package/lib/test/unit/directoryHandleStore.test.js +0 -192
  157. package/lib/test/unit/directoryHandleStore.test.js.map +0 -1
  158. package/lib/test/unit/fileApiTreeAccessors.test.d.ts +0 -2
  159. package/lib/test/unit/fileApiTreeAccessors.test.d.ts.map +0 -1
  160. package/lib/test/unit/fileApiTreeAccessors.test.js +0 -1190
  161. package/lib/test/unit/fileApiTreeAccessors.test.js.map +0 -1
  162. package/lib/test/unit/fileApiTypes.test.d.ts +0 -2
  163. package/lib/test/unit/fileApiTypes.test.d.ts.map +0 -1
  164. package/lib/test/unit/fileApiTypes.test.js +0 -474
  165. package/lib/test/unit/fileApiTypes.test.js.map +0 -1
  166. package/lib/test/unit/fileSystemAccessTreeAccessors.test.d.ts +0 -2
  167. package/lib/test/unit/fileSystemAccessTreeAccessors.test.d.ts.map +0 -1
  168. package/lib/test/unit/fileSystemAccessTreeAccessors.test.js +0 -624
  169. package/lib/test/unit/fileSystemAccessTreeAccessors.test.js.map +0 -1
  170. package/lib/test/unit/fileTreeHelpers.test.d.ts +0 -2
  171. package/lib/test/unit/fileTreeHelpers.test.d.ts.map +0 -1
  172. package/lib/test/unit/fileTreeHelpers.test.js +0 -592
  173. package/lib/test/unit/fileTreeHelpers.test.js.map +0 -1
  174. package/lib/test/unit/httpTreeAccessors.test.d.ts +0 -2
  175. package/lib/test/unit/httpTreeAccessors.test.d.ts.map +0 -1
  176. package/lib/test/unit/httpTreeAccessors.test.js +0 -1231
  177. package/lib/test/unit/httpTreeAccessors.test.js.map +0 -1
  178. package/lib/test/unit/localStorageTreeAccessors.test.d.ts +0 -2
  179. package/lib/test/unit/localStorageTreeAccessors.test.d.ts.map +0 -1
  180. package/lib/test/unit/localStorageTreeAccessors.test.js +0 -814
  181. package/lib/test/unit/localStorageTreeAccessors.test.js.map +0 -1
  182. package/lib/test/unit/urlParams.test.d.ts +0 -2
  183. package/lib/test/unit/urlParams.test.d.ts.map +0 -1
  184. package/lib/test/unit/urlParams.test.js +0 -395
  185. package/lib/test/unit/urlParams.test.js.map +0 -1
  186. package/lib/test/utils/fileSystemAccessMocks.d.ts +0 -53
  187. package/lib/test/utils/fileSystemAccessMocks.d.ts.map +0 -1
  188. package/lib/test/utils/fileSystemAccessMocks.js +0 -277
  189. package/lib/test/utils/fileSystemAccessMocks.js.map +0 -1
  190. package/lib/test/utils/testHelpers.d.ts +0 -51
  191. package/lib/test/utils/testHelpers.d.ts.map +0 -1
  192. package/lib/test/utils/testHelpers.js +0 -133
  193. package/lib/test/utils/testHelpers.js.map +0 -1
  194. package/rush-logs/ts-web-extras.build.cache.log +0 -3
  195. package/rush-logs/ts-web-extras.build.log +0 -55
  196. package/src/index.browser.ts +0 -24
  197. package/src/index.ts +0 -47
  198. package/src/packlets/crypto-utils/browserCryptoProvider.ts +0 -311
  199. package/src/packlets/crypto-utils/browserHashProvider.ts +0 -73
  200. package/src/packlets/crypto-utils/index.ts +0 -29
  201. package/src/packlets/file-api-types/index.ts +0 -366
  202. package/src/packlets/file-tree/directoryHandleStore.ts +0 -136
  203. package/src/packlets/file-tree/fileApiTreeAccessors.ts +0 -528
  204. package/src/packlets/file-tree/fileSystemAccessTreeAccessors.ts +0 -519
  205. package/src/packlets/file-tree/httpTreeAccessors.ts +0 -448
  206. package/src/packlets/file-tree/index.ts +0 -32
  207. package/src/packlets/file-tree/localStorageTreeAccessors.ts +0 -430
  208. package/src/packlets/helpers/fileTreeHelpers.ts +0 -107
  209. package/src/packlets/helpers/index.ts +0 -28
  210. package/src/packlets/url-utils/index.ts +0 -28
  211. package/src/packlets/url-utils/urlParams.ts +0 -245
  212. package/src/test/mocks/idb-keyval.ts +0 -5
  213. package/src/test/setupTests.ts +0 -87
  214. package/src/test/unit/browserHashProvider.test.ts +0 -155
  215. package/src/test/unit/browserHashProvider.test.ts.bak +0 -376
  216. package/src/test/unit/directoryHandleStore.test.ts +0 -251
  217. package/src/test/unit/fileApiTreeAccessors.test.ts +0 -1387
  218. package/src/test/unit/fileApiTypes.test.ts +0 -587
  219. package/src/test/unit/fileSystemAccessTreeAccessors.test.ts +0 -885
  220. package/src/test/unit/fileTreeHelpers.test.ts +0 -694
  221. package/src/test/unit/httpTreeAccessors.test.ts +0 -1571
  222. package/src/test/unit/localStorageTreeAccessors.test.ts +0 -1014
  223. package/src/test/unit/urlParams.test.ts +0 -464
  224. package/src/test/utils/fileSystemAccessMocks.ts +0 -353
  225. package/src/test/utils/testHelpers.ts +0 -155
  226. package/temp/build/typescript/ts_8nwakTlr.json +0 -1
  227. package/temp/coverage/base.css +0 -224
  228. package/temp/coverage/block-navigation.js +0 -87
  229. package/temp/coverage/crypto-utils/browserCryptoProvider.ts.html +0 -1018
  230. package/temp/coverage/crypto-utils/browserHashProvider.ts.html +0 -304
  231. package/temp/coverage/crypto-utils/index.html +0 -131
  232. package/temp/coverage/favicon.png +0 -0
  233. package/temp/coverage/file-tree/directoryHandleStore.ts.html +0 -493
  234. package/temp/coverage/file-tree/fileApiTreeAccessors.ts.html +0 -1669
  235. package/temp/coverage/file-tree/fileSystemAccessTreeAccessors.ts.html +0 -1642
  236. package/temp/coverage/file-tree/httpTreeAccessors.ts.html +0 -1429
  237. package/temp/coverage/file-tree/index.html +0 -176
  238. package/temp/coverage/file-tree/localStorageTreeAccessors.ts.html +0 -1375
  239. package/temp/coverage/helpers/fileTreeHelpers.ts.html +0 -406
  240. package/temp/coverage/helpers/index.html +0 -116
  241. package/temp/coverage/index.html +0 -161
  242. package/temp/coverage/lcov-report/base.css +0 -224
  243. package/temp/coverage/lcov-report/block-navigation.js +0 -87
  244. package/temp/coverage/lcov-report/crypto-utils/browserCryptoProvider.ts.html +0 -1018
  245. package/temp/coverage/lcov-report/crypto-utils/browserHashProvider.ts.html +0 -304
  246. package/temp/coverage/lcov-report/crypto-utils/index.html +0 -131
  247. package/temp/coverage/lcov-report/favicon.png +0 -0
  248. package/temp/coverage/lcov-report/file-tree/directoryHandleStore.ts.html +0 -493
  249. package/temp/coverage/lcov-report/file-tree/fileApiTreeAccessors.ts.html +0 -1669
  250. package/temp/coverage/lcov-report/file-tree/fileSystemAccessTreeAccessors.ts.html +0 -1642
  251. package/temp/coverage/lcov-report/file-tree/httpTreeAccessors.ts.html +0 -1429
  252. package/temp/coverage/lcov-report/file-tree/index.html +0 -176
  253. package/temp/coverage/lcov-report/file-tree/localStorageTreeAccessors.ts.html +0 -1375
  254. package/temp/coverage/lcov-report/helpers/fileTreeHelpers.ts.html +0 -406
  255. package/temp/coverage/lcov-report/helpers/index.html +0 -116
  256. package/temp/coverage/lcov-report/index.html +0 -161
  257. package/temp/coverage/lcov-report/prettify.css +0 -1
  258. package/temp/coverage/lcov-report/prettify.js +0 -2
  259. package/temp/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  260. package/temp/coverage/lcov-report/sorter.js +0 -210
  261. package/temp/coverage/lcov-report/url-utils/index.html +0 -116
  262. package/temp/coverage/lcov-report/url-utils/urlParams.ts.html +0 -820
  263. package/temp/coverage/lcov.info +0 -3597
  264. package/temp/coverage/prettify.css +0 -1
  265. package/temp/coverage/prettify.js +0 -2
  266. package/temp/coverage/sort-arrow-sprite.png +0 -0
  267. package/temp/coverage/sorter.js +0 -210
  268. package/temp/coverage/url-utils/index.html +0 -116
  269. package/temp/coverage/url-utils/urlParams.ts.html +0 -820
  270. package/temp/test/jest/haste-map-7492f1b44480e0cdd1f220078fb3afd8-c8dd6c3430605adeb2f1cadf4f75e791-8c9336785555d572065b28c111982ba4 +0 -0
  271. package/temp/test/jest/jest-transform-cache-7492f1b44480e0cdd1f220078fb3afd8-79ef2876fae7ca75eedb2aa53dc48338/d6/package_d6e3b0fa94752e16b0b2a2777739b973 +0 -53
  272. package/temp/test/jest/perf-cache-7492f1b44480e0cdd1f220078fb3afd8-da39a3ee5e6b4b0d3255bfef95601890 +0 -1
  273. package/temp/ts-web-extras.api.json +0 -8850
  274. package/temp/ts-web-extras.api.md +0 -433
  275. package/tsconfig.json +0 -7
@@ -1,587 +0,0 @@
1
- /*
2
- * Copyright (c) 2025 Erik Fortune
3
- *
4
- * Permission is hereby granted, free of charge, to any person obtaining a copy
5
- * of this software and associated documentation files (the "Software"), to deal
6
- * in the Software without restriction, including without limitation the rights
7
- * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8
- * copies of the Software, and to permit persons to whom the Software is
9
- * furnished to do so, subject to the following conditions:
10
- *
11
- * The above copyright notice and this permission notice shall be included in all
12
- * copies or substantial portions of the Software.
13
- *
14
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
17
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
19
- * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
20
- * SOFTWARE.
21
- */
22
-
23
- import '@fgv/ts-utils-jest';
24
- import {
25
- supportsFileSystemAccess,
26
- isFileHandle,
27
- isDirectoryHandle,
28
- safeShowOpenFilePicker,
29
- safeShowSaveFilePicker,
30
- safeShowDirectoryPicker,
31
- exportAsJson,
32
- exportUsingFileSystemAPI,
33
- type FileSystemHandle,
34
- type FileSystemFileHandle,
35
- type FileSystemDirectoryHandle,
36
- type WindowWithFsAccess,
37
- type ShowOpenFilePickerOptions,
38
- type ShowSaveFilePickerOptions,
39
- type ShowDirectoryPickerOptions
40
- } from '../../packlets/file-api-types';
41
-
42
- describe('File API Types', () => {
43
- describe('supportsFileSystemAccess', () => {
44
- test('returns true for window with File System Access API support', () => {
45
- const mockWindow = {
46
- showOpenFilePicker: jest.fn(),
47
- showSaveFilePicker: jest.fn(),
48
- showDirectoryPicker: jest.fn()
49
- } as unknown as Window;
50
-
51
- expect(supportsFileSystemAccess(mockWindow)).toBe(true);
52
- });
53
-
54
- test('returns false for window without showOpenFilePicker', () => {
55
- const mockWindow = {
56
- showSaveFilePicker: jest.fn(),
57
- showDirectoryPicker: jest.fn()
58
- } as unknown as Window;
59
-
60
- expect(supportsFileSystemAccess(mockWindow)).toBe(false);
61
- });
62
-
63
- test('returns false for window without showSaveFilePicker', () => {
64
- const mockWindow = {
65
- showOpenFilePicker: jest.fn(),
66
- showDirectoryPicker: jest.fn()
67
- } as unknown as Window;
68
-
69
- expect(supportsFileSystemAccess(mockWindow)).toBe(false);
70
- });
71
-
72
- test('returns false for window without showDirectoryPicker', () => {
73
- const mockWindow = {
74
- showOpenFilePicker: jest.fn(),
75
- showSaveFilePicker: jest.fn()
76
- } as unknown as Window;
77
-
78
- expect(supportsFileSystemAccess(mockWindow)).toBe(false);
79
- });
80
-
81
- test('returns false for window with no File System Access API methods', () => {
82
- const mockWindow = {} as unknown as Window;
83
-
84
- expect(supportsFileSystemAccess(mockWindow)).toBe(false);
85
- });
86
-
87
- test('returns false for standard browser window without File System Access API', () => {
88
- // Simulate older browser window
89
- const mockWindow = {
90
- location: { href: 'https://example.com' },
91
- document: {},
92
- navigator: {}
93
- } as unknown as Window;
94
-
95
- expect(supportsFileSystemAccess(mockWindow)).toBe(false);
96
- });
97
- });
98
-
99
- describe('isFileHandle', () => {
100
- test('returns true for file handle', () => {
101
- const mockFileHandle: FileSystemFileHandle = {
102
- kind: 'file',
103
- name: 'test.txt',
104
- isSameEntry: jest.fn(),
105
- queryPermission: jest.fn(),
106
- requestPermission: jest.fn(),
107
- getFile: jest.fn(),
108
- createWritable: jest.fn()
109
- };
110
-
111
- expect(isFileHandle(mockFileHandle)).toBe(true);
112
- });
113
-
114
- test('returns false for directory handle', () => {
115
- const mockDirectoryHandle: FileSystemDirectoryHandle = {
116
- kind: 'directory',
117
- name: 'test-dir',
118
- isSameEntry: jest.fn(),
119
- queryPermission: jest.fn(),
120
- requestPermission: jest.fn(),
121
- getDirectoryHandle: jest.fn(),
122
- getFileHandle: jest.fn(),
123
- removeEntry: jest.fn(),
124
- resolve: jest.fn(),
125
- keys: jest.fn(),
126
- values: jest.fn(),
127
- entries: jest.fn(),
128
- [Symbol.asyncIterator]: jest.fn()
129
- };
130
-
131
- expect(isFileHandle(mockDirectoryHandle)).toBe(false);
132
- });
133
-
134
- test('handles handle with minimal implementation', () => {
135
- const mockHandle = {
136
- kind: 'file' as const,
137
- name: 'minimal.txt'
138
- } as FileSystemHandle;
139
-
140
- expect(isFileHandle(mockHandle)).toBe(true);
141
- });
142
-
143
- test('handles handle with directory kind', () => {
144
- const mockHandle = {
145
- kind: 'directory' as const,
146
- name: 'minimal-dir'
147
- } as FileSystemHandle;
148
-
149
- expect(isFileHandle(mockHandle)).toBe(false);
150
- });
151
- });
152
-
153
- describe('isDirectoryHandle', () => {
154
- test('returns true for directory handle', () => {
155
- const mockDirectoryHandle: FileSystemDirectoryHandle = {
156
- kind: 'directory',
157
- name: 'test-dir',
158
- isSameEntry: jest.fn(),
159
- queryPermission: jest.fn(),
160
- requestPermission: jest.fn(),
161
- getDirectoryHandle: jest.fn(),
162
- getFileHandle: jest.fn(),
163
- removeEntry: jest.fn(),
164
- resolve: jest.fn(),
165
- keys: jest.fn(),
166
- values: jest.fn(),
167
- entries: jest.fn(),
168
- [Symbol.asyncIterator]: jest.fn()
169
- };
170
-
171
- expect(isDirectoryHandle(mockDirectoryHandle)).toBe(true);
172
- });
173
-
174
- test('returns false for file handle', () => {
175
- const mockFileHandle: FileSystemFileHandle = {
176
- kind: 'file',
177
- name: 'test.txt',
178
- isSameEntry: jest.fn(),
179
- queryPermission: jest.fn(),
180
- requestPermission: jest.fn(),
181
- getFile: jest.fn(),
182
- createWritable: jest.fn()
183
- };
184
-
185
- expect(isDirectoryHandle(mockFileHandle)).toBe(false);
186
- });
187
-
188
- test('handles handle with minimal implementation', () => {
189
- const mockHandle = {
190
- kind: 'directory' as const,
191
- name: 'minimal-dir'
192
- } as FileSystemHandle;
193
-
194
- expect(isDirectoryHandle(mockHandle)).toBe(true);
195
- });
196
-
197
- test('handles handle with file kind', () => {
198
- const mockHandle = {
199
- kind: 'file' as const,
200
- name: 'minimal.txt'
201
- } as FileSystemHandle;
202
-
203
- expect(isDirectoryHandle(mockHandle)).toBe(false);
204
- });
205
- });
206
-
207
- describe('safeShowOpenFilePicker', () => {
208
- test('calls showOpenFilePicker when File System Access API is supported', async () => {
209
- const mockFileHandles = [
210
- {
211
- kind: 'file' as const,
212
- name: 'test.txt',
213
- isSameEntry: jest.fn(),
214
- queryPermission: jest.fn(),
215
- requestPermission: jest.fn(),
216
- getFile: jest.fn(),
217
- createWritable: jest.fn()
218
- }
219
- ] as FileSystemFileHandle[];
220
-
221
- const mockWindow = {
222
- showOpenFilePicker: jest.fn().mockResolvedValue(mockFileHandles),
223
- showSaveFilePicker: jest.fn(),
224
- showDirectoryPicker: jest.fn()
225
- } as unknown as WindowWithFsAccess;
226
-
227
- const options: ShowOpenFilePickerOptions = {
228
- multiple: true,
229
- types: [{ description: 'Text files', accept: { 'text/plain': ['.txt'] } }]
230
- };
231
-
232
- const result = await safeShowOpenFilePicker(mockWindow, options);
233
-
234
- expect(mockWindow.showOpenFilePicker).toHaveBeenCalledWith(options);
235
- expect(result).toBe(mockFileHandles);
236
- });
237
-
238
- test('calls showOpenFilePicker without options when none provided', async () => {
239
- const mockFileHandles = [] as FileSystemFileHandle[];
240
-
241
- const mockWindow = {
242
- showOpenFilePicker: jest.fn().mockResolvedValue(mockFileHandles),
243
- showSaveFilePicker: jest.fn(),
244
- showDirectoryPicker: jest.fn()
245
- } as unknown as WindowWithFsAccess;
246
-
247
- const result = await safeShowOpenFilePicker(mockWindow);
248
-
249
- expect(mockWindow.showOpenFilePicker).toHaveBeenCalledWith(undefined);
250
- expect(result).toBe(mockFileHandles);
251
- });
252
-
253
- test('returns null when File System Access API is not supported', async () => {
254
- const mockWindow = {} as unknown as Window;
255
-
256
- const result = await safeShowOpenFilePicker(mockWindow);
257
-
258
- expect(result).toBeNull();
259
- });
260
-
261
- test('returns null for partially supported window', async () => {
262
- const mockWindow = {
263
- showOpenFilePicker: jest.fn()
264
- // Missing showSaveFilePicker and showDirectoryPicker
265
- } as unknown as Window;
266
-
267
- const result = await safeShowOpenFilePicker(mockWindow);
268
-
269
- expect(result).toBeNull();
270
- });
271
-
272
- test('propagates errors from showOpenFilePicker', async () => {
273
- const mockError = new Error('User cancelled file picker');
274
- const mockWindow = {
275
- showOpenFilePicker: jest.fn().mockRejectedValue(mockError),
276
- showSaveFilePicker: jest.fn(),
277
- showDirectoryPicker: jest.fn()
278
- } as unknown as WindowWithFsAccess;
279
-
280
- await expect(safeShowOpenFilePicker(mockWindow)).rejects.toThrow('User cancelled file picker');
281
- });
282
-
283
- test('returns null when user cancels (AbortError)', async () => {
284
- const abortError = new DOMException('The user aborted a request.', 'AbortError');
285
- const mockWindow = {
286
- showOpenFilePicker: jest.fn().mockRejectedValue(abortError),
287
- showSaveFilePicker: jest.fn(),
288
- showDirectoryPicker: jest.fn()
289
- } as unknown as WindowWithFsAccess;
290
-
291
- const result = await safeShowOpenFilePicker(mockWindow);
292
- expect(result).toBeNull();
293
- });
294
- });
295
-
296
- describe('safeShowSaveFilePicker', () => {
297
- test('calls showSaveFilePicker when File System Access API is supported', async () => {
298
- const mockFileHandle = {
299
- kind: 'file' as const,
300
- name: 'save.txt',
301
- isSameEntry: jest.fn(),
302
- queryPermission: jest.fn(),
303
- requestPermission: jest.fn(),
304
- getFile: jest.fn(),
305
- createWritable: jest.fn()
306
- } as FileSystemFileHandle;
307
-
308
- const mockWindow = {
309
- showOpenFilePicker: jest.fn(),
310
- showSaveFilePicker: jest.fn().mockResolvedValue(mockFileHandle),
311
- showDirectoryPicker: jest.fn()
312
- } as unknown as WindowWithFsAccess;
313
-
314
- const options: ShowSaveFilePickerOptions = {
315
- suggestedName: 'document.txt',
316
- types: [{ description: 'Text files', accept: { 'text/plain': ['.txt'] } }]
317
- };
318
-
319
- const result = await safeShowSaveFilePicker(mockWindow, options);
320
-
321
- expect(mockWindow.showSaveFilePicker).toHaveBeenCalledWith(options);
322
- expect(result).toBe(mockFileHandle);
323
- });
324
-
325
- test('calls showSaveFilePicker without options when none provided', async () => {
326
- const mockFileHandle = {
327
- kind: 'file' as const,
328
- name: 'untitled.txt'
329
- } as FileSystemFileHandle;
330
-
331
- const mockWindow = {
332
- showOpenFilePicker: jest.fn(),
333
- showSaveFilePicker: jest.fn().mockResolvedValue(mockFileHandle),
334
- showDirectoryPicker: jest.fn()
335
- } as unknown as WindowWithFsAccess;
336
-
337
- const result = await safeShowSaveFilePicker(mockWindow);
338
-
339
- expect(mockWindow.showSaveFilePicker).toHaveBeenCalledWith(undefined);
340
- expect(result).toBe(mockFileHandle);
341
- });
342
-
343
- test('returns null when File System Access API is not supported', async () => {
344
- const mockWindow = {} as unknown as Window;
345
-
346
- const result = await safeShowSaveFilePicker(mockWindow);
347
-
348
- expect(result).toBeNull();
349
- });
350
-
351
- test('returns null for partially supported window', async () => {
352
- const mockWindow = {
353
- showSaveFilePicker: jest.fn()
354
- // Missing showOpenFilePicker and showDirectoryPicker
355
- } as unknown as Window;
356
-
357
- const result = await safeShowSaveFilePicker(mockWindow);
358
-
359
- expect(result).toBeNull();
360
- });
361
-
362
- test('propagates errors from showSaveFilePicker', async () => {
363
- const mockError = new Error('Save operation failed');
364
- const mockWindow = {
365
- showOpenFilePicker: jest.fn(),
366
- showSaveFilePicker: jest.fn().mockRejectedValue(mockError),
367
- showDirectoryPicker: jest.fn()
368
- } as unknown as WindowWithFsAccess;
369
-
370
- await expect(safeShowSaveFilePicker(mockWindow)).rejects.toThrow('Save operation failed');
371
- });
372
-
373
- test('returns null when user cancels (AbortError)', async () => {
374
- const abortError = new DOMException('The user aborted a request.', 'AbortError');
375
- const mockWindow = {
376
- showOpenFilePicker: jest.fn(),
377
- showSaveFilePicker: jest.fn().mockRejectedValue(abortError),
378
- showDirectoryPicker: jest.fn()
379
- } as unknown as WindowWithFsAccess;
380
-
381
- const result = await safeShowSaveFilePicker(mockWindow);
382
- expect(result).toBeNull();
383
- });
384
- });
385
-
386
- describe('safeShowDirectoryPicker', () => {
387
- test('calls showDirectoryPicker when File System Access API is supported', async () => {
388
- const mockDirectoryHandle = {
389
- kind: 'directory' as const,
390
- name: 'selected-folder',
391
- isSameEntry: jest.fn(),
392
- queryPermission: jest.fn(),
393
- requestPermission: jest.fn(),
394
- getDirectoryHandle: jest.fn(),
395
- getFileHandle: jest.fn(),
396
- removeEntry: jest.fn(),
397
- resolve: jest.fn(),
398
- keys: jest.fn(),
399
- values: jest.fn(),
400
- entries: jest.fn(),
401
- [Symbol.asyncIterator]: jest.fn()
402
- } as FileSystemDirectoryHandle;
403
-
404
- const mockWindow = {
405
- showOpenFilePicker: jest.fn(),
406
- showSaveFilePicker: jest.fn(),
407
- showDirectoryPicker: jest.fn().mockResolvedValue(mockDirectoryHandle)
408
- } as unknown as WindowWithFsAccess;
409
-
410
- const options: ShowDirectoryPickerOptions = {
411
- mode: 'readwrite',
412
- startIn: 'documents' as const
413
- };
414
-
415
- const result = await safeShowDirectoryPicker(mockWindow, options);
416
-
417
- expect(mockWindow.showDirectoryPicker).toHaveBeenCalledWith(options);
418
- expect(result).toBe(mockDirectoryHandle);
419
- });
420
-
421
- test('calls showDirectoryPicker without options when none provided', async () => {
422
- const mockDirectoryHandle = {
423
- kind: 'directory' as const,
424
- name: 'default-folder'
425
- } as FileSystemDirectoryHandle;
426
-
427
- const mockWindow = {
428
- showOpenFilePicker: jest.fn(),
429
- showSaveFilePicker: jest.fn(),
430
- showDirectoryPicker: jest.fn().mockResolvedValue(mockDirectoryHandle)
431
- } as unknown as WindowWithFsAccess;
432
-
433
- const result = await safeShowDirectoryPicker(mockWindow);
434
-
435
- expect(mockWindow.showDirectoryPicker).toHaveBeenCalledWith(undefined);
436
- expect(result).toBe(mockDirectoryHandle);
437
- });
438
-
439
- test('returns null when File System Access API is not supported', async () => {
440
- const mockWindow = {} as unknown as Window;
441
-
442
- const result = await safeShowDirectoryPicker(mockWindow);
443
-
444
- expect(result).toBeNull();
445
- });
446
-
447
- test('returns null for partially supported window', async () => {
448
- const mockWindow = {
449
- showDirectoryPicker: jest.fn()
450
- // Missing showOpenFilePicker and showSaveFilePicker
451
- } as unknown as Window;
452
-
453
- const result = await safeShowDirectoryPicker(mockWindow);
454
-
455
- expect(result).toBeNull();
456
- });
457
-
458
- test('propagates errors from showDirectoryPicker', async () => {
459
- const mockError = new Error('Directory access denied');
460
- const mockWindow = {
461
- showOpenFilePicker: jest.fn(),
462
- showSaveFilePicker: jest.fn(),
463
- showDirectoryPicker: jest.fn().mockRejectedValue(mockError)
464
- } as unknown as WindowWithFsAccess;
465
-
466
- await expect(safeShowDirectoryPicker(mockWindow)).rejects.toThrow('Directory access denied');
467
- });
468
-
469
- test('returns null when user cancels (AbortError)', async () => {
470
- const abortError = new DOMException('The user aborted a request.', 'AbortError');
471
- const mockWindow = {
472
- showOpenFilePicker: jest.fn(),
473
- showSaveFilePicker: jest.fn(),
474
- showDirectoryPicker: jest.fn().mockRejectedValue(abortError)
475
- } as unknown as WindowWithFsAccess;
476
-
477
- const result = await safeShowDirectoryPicker(mockWindow);
478
- expect(result).toBeNull();
479
- });
480
- });
481
-
482
- describe('integration tests', () => {
483
- test('type guards work with handles returned from safe picker functions', async () => {
484
- const mockFileHandle = {
485
- kind: 'file' as const,
486
- name: 'test.txt',
487
- isSameEntry: jest.fn(),
488
- queryPermission: jest.fn(),
489
- requestPermission: jest.fn(),
490
- getFile: jest.fn(),
491
- createWritable: jest.fn()
492
- } as FileSystemFileHandle;
493
-
494
- const mockDirectoryHandle = {
495
- kind: 'directory' as const,
496
- name: 'test-dir',
497
- isSameEntry: jest.fn(),
498
- queryPermission: jest.fn(),
499
- requestPermission: jest.fn(),
500
- getDirectoryHandle: jest.fn(),
501
- getFileHandle: jest.fn(),
502
- removeEntry: jest.fn(),
503
- resolve: jest.fn(),
504
- keys: jest.fn(),
505
- values: jest.fn(),
506
- entries: jest.fn(),
507
- [Symbol.asyncIterator]: jest.fn()
508
- } as FileSystemDirectoryHandle;
509
-
510
- const mockWindow = {
511
- showOpenFilePicker: jest.fn().mockResolvedValue([mockFileHandle]),
512
- showSaveFilePicker: jest.fn().mockResolvedValue(mockFileHandle),
513
- showDirectoryPicker: jest.fn().mockResolvedValue(mockDirectoryHandle)
514
- } as unknown as WindowWithFsAccess;
515
-
516
- // Test file handle from showOpenFilePicker
517
- const openResult = await safeShowOpenFilePicker(mockWindow);
518
- expect(openResult).not.toBeNull();
519
- if (openResult && openResult.length > 0) {
520
- expect(isFileHandle(openResult[0])).toBe(true);
521
- expect(isDirectoryHandle(openResult[0])).toBe(false);
522
- }
523
-
524
- // Test file handle from showSaveFilePicker
525
- const saveResult = await safeShowSaveFilePicker(mockWindow);
526
- expect(saveResult).not.toBeNull();
527
- if (saveResult) {
528
- expect(isFileHandle(saveResult)).toBe(true);
529
- expect(isDirectoryHandle(saveResult)).toBe(false);
530
- }
531
-
532
- // Test directory handle from showDirectoryPicker
533
- const dirResult = await safeShowDirectoryPicker(mockWindow);
534
- expect(dirResult).not.toBeNull();
535
- if (dirResult) {
536
- expect(isDirectoryHandle(dirResult)).toBe(true);
537
- expect(isFileHandle(dirResult)).toBe(false);
538
- }
539
- });
540
-
541
- test('safe functions respect browser support detection', async () => {
542
- // Mock browser without File System Access API
543
- const unsupportedWindow = {
544
- location: { href: 'https://example.com' }
545
- } as unknown as Window;
546
-
547
- // All safe functions should return null
548
- expect(await safeShowOpenFilePicker(unsupportedWindow)).toBeNull();
549
- expect(await safeShowSaveFilePicker(unsupportedWindow)).toBeNull();
550
- expect(await safeShowDirectoryPicker(unsupportedWindow)).toBeNull();
551
-
552
- // Mock browser with File System Access API
553
- const supportedWindow = {
554
- showOpenFilePicker: jest.fn().mockResolvedValue([]),
555
- showSaveFilePicker: jest.fn().mockResolvedValue(null),
556
- showDirectoryPicker: jest.fn().mockResolvedValue(null)
557
- } as unknown as WindowWithFsAccess;
558
-
559
- // All safe functions should call the underlying API
560
- await safeShowOpenFilePicker(supportedWindow);
561
- await safeShowSaveFilePicker(supportedWindow);
562
- await safeShowDirectoryPicker(supportedWindow);
563
-
564
- expect(supportedWindow.showOpenFilePicker).toHaveBeenCalled();
565
- expect(supportedWindow.showSaveFilePicker).toHaveBeenCalled();
566
- expect(supportedWindow.showDirectoryPicker).toHaveBeenCalled();
567
- });
568
- });
569
-
570
- describe('exportAsJson', () => {
571
- test('is a function', () => {
572
- expect(typeof exportAsJson).toBe('function');
573
- });
574
- });
575
-
576
- describe('exportUsingFileSystemAPI', () => {
577
- test('is a function', () => {
578
- expect(typeof exportUsingFileSystemAPI).toBe('function');
579
- });
580
-
581
- test.skip('API integration tests require browser environment', () => {
582
- // These tests require browser APIs (URL.createObjectURL, document.createElement)
583
- // which are not available in the jest/jsdom test environment.
584
- // The functions work correctly in real browser environments.
585
- });
586
- });
587
- });