@fgv/ts-web-extras 5.1.0-2 → 5.1.0-21

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 (263) hide show
  1. package/dist/packlets/crypto-utils/browserCryptoProvider.js +208 -18
  2. package/dist/packlets/crypto-utils/browserCryptoProvider.js.map +1 -1
  3. package/dist/packlets/file-tree/fileApiTreeAccessors.js +1 -1
  4. package/dist/packlets/file-tree/fileApiTreeAccessors.js.map +1 -1
  5. package/dist/packlets/file-tree/httpTreeAccessors.js +72 -42
  6. package/dist/packlets/file-tree/httpTreeAccessors.js.map +1 -1
  7. package/dist/ts-web-extras.d.ts +59 -7
  8. package/dist/tsdoc-metadata.json +1 -1
  9. package/lib/packlets/crypto-utils/browserCryptoProvider.d.ts +52 -5
  10. package/lib/packlets/crypto-utils/browserCryptoProvider.d.ts.map +1 -1
  11. package/lib/packlets/crypto-utils/browserCryptoProvider.js +207 -17
  12. package/lib/packlets/crypto-utils/browserCryptoProvider.js.map +1 -1
  13. package/lib/packlets/file-tree/fileApiTreeAccessors.d.ts +1 -1
  14. package/lib/packlets/file-tree/fileApiTreeAccessors.js +1 -1
  15. package/lib/packlets/file-tree/fileApiTreeAccessors.js.map +1 -1
  16. package/lib/packlets/file-tree/httpTreeAccessors.d.ts +6 -0
  17. package/lib/packlets/file-tree/httpTreeAccessors.d.ts.map +1 -1
  18. package/lib/packlets/file-tree/httpTreeAccessors.js +72 -42
  19. package/lib/packlets/file-tree/httpTreeAccessors.js.map +1 -1
  20. package/package.json +27 -26
  21. package/.rush/temp/chunked-rush-logs/ts-web-extras.build.chunks.jsonl +0 -75
  22. package/.rush/temp/chunked-rush-logs/ts-web-extras.test.chunks.jsonl +0 -75
  23. package/.rush/temp/operation/build/all.log +0 -75
  24. package/.rush/temp/operation/build/error.log +0 -18
  25. package/.rush/temp/operation/build/log-chunks.jsonl +0 -75
  26. package/.rush/temp/operation/build/state.json +0 -3
  27. package/.rush/temp/operation/test/all.log +0 -75
  28. package/.rush/temp/operation/test/error.log +0 -18
  29. package/.rush/temp/operation/test/log-chunks.jsonl +0 -75
  30. package/.rush/temp/operation/test/state.json +0 -3
  31. package/.rush/temp/shrinkwrap-deps.json +0 -635
  32. package/CHANGELOG.md +0 -23
  33. package/config/api-extractor.json +0 -343
  34. package/config/jest.config.json +0 -19
  35. package/config/rig.json +0 -16
  36. package/config/typedoc.json +0 -6
  37. package/dist/test/mocks/idb-keyval.js +0 -6
  38. package/dist/test/mocks/idb-keyval.js.map +0 -1
  39. package/dist/test/setupTests.js +0 -74
  40. package/dist/test/setupTests.js.map +0 -1
  41. package/dist/test/unit/browserHashProvider.test.js +0 -140
  42. package/dist/test/unit/browserHashProvider.test.js.map +0 -1
  43. package/dist/test/unit/directoryHandleStore.test.js +0 -190
  44. package/dist/test/unit/directoryHandleStore.test.js.map +0 -1
  45. package/dist/test/unit/fileApiTreeAccessors.test.js +0 -1188
  46. package/dist/test/unit/fileApiTreeAccessors.test.js.map +0 -1
  47. package/dist/test/unit/fileApiTypes.test.js +0 -472
  48. package/dist/test/unit/fileApiTypes.test.js.map +0 -1
  49. package/dist/test/unit/fileSystemAccessTreeAccessors.test.js +0 -622
  50. package/dist/test/unit/fileSystemAccessTreeAccessors.test.js.map +0 -1
  51. package/dist/test/unit/fileTreeHelpers.test.js +0 -590
  52. package/dist/test/unit/fileTreeHelpers.test.js.map +0 -1
  53. package/dist/test/unit/httpTreeAccessors.test.js +0 -1229
  54. package/dist/test/unit/httpTreeAccessors.test.js.map +0 -1
  55. package/dist/test/unit/localStorageTreeAccessors.test.js +0 -812
  56. package/dist/test/unit/localStorageTreeAccessors.test.js.map +0 -1
  57. package/dist/test/unit/urlParams.test.js +0 -393
  58. package/dist/test/unit/urlParams.test.js.map +0 -1
  59. package/dist/test/utils/fileSystemAccessMocks.js +0 -271
  60. package/dist/test/utils/fileSystemAccessMocks.js.map +0 -1
  61. package/dist/test/utils/testHelpers.js +0 -124
  62. package/dist/test/utils/testHelpers.js.map +0 -1
  63. package/docs/@fgv/namespaces/CryptoUtils/README.md +0 -18
  64. package/docs/@fgv/namespaces/CryptoUtils/classes/BrowserCryptoProvider.md +0 -203
  65. package/docs/@fgv/namespaces/CryptoUtils/classes/BrowserHashProvider.md +0 -63
  66. package/docs/@fgv/namespaces/CryptoUtils/functions/createBrowserCryptoProvider.md +0 -18
  67. package/docs/@fgv/namespaces/FileTreeHelpers/README.md +0 -19
  68. package/docs/@fgv/namespaces/FileTreeHelpers/functions/extractFileListMetadata.md +0 -23
  69. package/docs/@fgv/namespaces/FileTreeHelpers/functions/extractFileMetadata.md +0 -23
  70. package/docs/@fgv/namespaces/FileTreeHelpers/functions/fromDirectoryUpload.md +0 -33
  71. package/docs/@fgv/namespaces/FileTreeHelpers/functions/fromFileList.md +0 -33
  72. package/docs/@fgv/namespaces/FileTreeHelpers/functions/getOriginalFile.md +0 -25
  73. package/docs/@fgv/namespaces/FileTreeHelpers/variables/defaultFileApiTreeInitParams.md +0 -11
  74. package/docs/README.md +0 -78
  75. package/docs/classes/DirectoryHandleStore.md +0 -116
  76. package/docs/classes/FileApiTreeAccessors.md +0 -286
  77. package/docs/classes/FileSystemAccessTreeAccessors.md +0 -557
  78. package/docs/classes/HttpTreeAccessors.md +0 -508
  79. package/docs/classes/LocalStorageTreeAccessors.md +0 -520
  80. package/docs/functions/exportAsJson.md +0 -23
  81. package/docs/functions/exportUsingFileSystemAPI.md +0 -26
  82. package/docs/functions/extractDirectoryPath.md +0 -23
  83. package/docs/functions/isDirectoryHandle.md +0 -23
  84. package/docs/functions/isFileHandle.md +0 -23
  85. package/docs/functions/isFilePath.md +0 -21
  86. package/docs/functions/parseContextFilter.md +0 -22
  87. package/docs/functions/parseQualifierDefaults.md +0 -22
  88. package/docs/functions/parseResourceTypes.md +0 -22
  89. package/docs/functions/parseUrlParameters.md +0 -15
  90. package/docs/functions/safeShowDirectoryPicker.md +0 -24
  91. package/docs/functions/safeShowOpenFilePicker.md +0 -24
  92. package/docs/functions/safeShowSaveFilePicker.md +0 -24
  93. package/docs/functions/supportsFileSystemAccess.md +0 -23
  94. package/docs/interfaces/FilePickerAcceptType.md +0 -16
  95. package/docs/interfaces/FileSystemCreateWritableOptions.md +0 -15
  96. package/docs/interfaces/FileSystemDirectoryHandle.md +0 -187
  97. package/docs/interfaces/FileSystemFileHandle.md +0 -106
  98. package/docs/interfaces/FileSystemGetDirectoryOptions.md +0 -15
  99. package/docs/interfaces/FileSystemGetFileOptions.md +0 -15
  100. package/docs/interfaces/FileSystemHandle.md +0 -69
  101. package/docs/interfaces/FileSystemHandlePermissionDescriptor.md +0 -15
  102. package/docs/interfaces/FileSystemRemoveOptions.md +0 -15
  103. package/docs/interfaces/FileSystemWritableFileStream.md +0 -127
  104. package/docs/interfaces/IDirectoryHandleTreeInitializer.md +0 -17
  105. package/docs/interfaces/IFileHandleTreeInitializer.md +0 -16
  106. package/docs/interfaces/IFileListTreeInitializer.md +0 -15
  107. package/docs/interfaces/IFileMetadata.md +0 -19
  108. package/docs/interfaces/IFileSystemAccessTreeParams.md +0 -30
  109. package/docs/interfaces/IFsAccessApis.md +0 -57
  110. package/docs/interfaces/IHttpTreeParams.md +0 -32
  111. package/docs/interfaces/ILocalStorageTreeParams.md +0 -30
  112. package/docs/interfaces/IUrlConfigOptions.md +0 -27
  113. package/docs/interfaces/ShowDirectoryPickerOptions.md +0 -17
  114. package/docs/interfaces/ShowOpenFilePickerOptions.md +0 -19
  115. package/docs/interfaces/ShowSaveFilePickerOptions.md +0 -19
  116. package/docs/type-aliases/TreeInitializer.md +0 -11
  117. package/docs/type-aliases/WellKnownDirectory.md +0 -11
  118. package/docs/type-aliases/WindowWithFsAccess.md +0 -11
  119. package/docs/variables/DEFAULT_DIRECTORY_HANDLE_DB.md +0 -11
  120. package/docs/variables/DEFAULT_DIRECTORY_HANDLE_STORE.md +0 -11
  121. package/etc/ts-web-extras.api.md +0 -433
  122. package/lib/test/mocks/idb-keyval.d.ts +0 -6
  123. package/lib/test/mocks/idb-keyval.d.ts.map +0 -1
  124. package/lib/test/mocks/idb-keyval.js +0 -9
  125. package/lib/test/mocks/idb-keyval.js.map +0 -1
  126. package/lib/test/setupTests.d.ts +0 -2
  127. package/lib/test/setupTests.d.ts.map +0 -1
  128. package/lib/test/setupTests.js +0 -76
  129. package/lib/test/setupTests.js.map +0 -1
  130. package/lib/test/unit/browserHashProvider.test.d.ts +0 -2
  131. package/lib/test/unit/browserHashProvider.test.d.ts.map +0 -1
  132. package/lib/test/unit/browserHashProvider.test.js +0 -142
  133. package/lib/test/unit/browserHashProvider.test.js.map +0 -1
  134. package/lib/test/unit/directoryHandleStore.test.d.ts +0 -2
  135. package/lib/test/unit/directoryHandleStore.test.d.ts.map +0 -1
  136. package/lib/test/unit/directoryHandleStore.test.js +0 -192
  137. package/lib/test/unit/directoryHandleStore.test.js.map +0 -1
  138. package/lib/test/unit/fileApiTreeAccessors.test.d.ts +0 -2
  139. package/lib/test/unit/fileApiTreeAccessors.test.d.ts.map +0 -1
  140. package/lib/test/unit/fileApiTreeAccessors.test.js +0 -1190
  141. package/lib/test/unit/fileApiTreeAccessors.test.js.map +0 -1
  142. package/lib/test/unit/fileApiTypes.test.d.ts +0 -2
  143. package/lib/test/unit/fileApiTypes.test.d.ts.map +0 -1
  144. package/lib/test/unit/fileApiTypes.test.js +0 -474
  145. package/lib/test/unit/fileApiTypes.test.js.map +0 -1
  146. package/lib/test/unit/fileSystemAccessTreeAccessors.test.d.ts +0 -2
  147. package/lib/test/unit/fileSystemAccessTreeAccessors.test.d.ts.map +0 -1
  148. package/lib/test/unit/fileSystemAccessTreeAccessors.test.js +0 -624
  149. package/lib/test/unit/fileSystemAccessTreeAccessors.test.js.map +0 -1
  150. package/lib/test/unit/fileTreeHelpers.test.d.ts +0 -2
  151. package/lib/test/unit/fileTreeHelpers.test.d.ts.map +0 -1
  152. package/lib/test/unit/fileTreeHelpers.test.js +0 -592
  153. package/lib/test/unit/fileTreeHelpers.test.js.map +0 -1
  154. package/lib/test/unit/httpTreeAccessors.test.d.ts +0 -2
  155. package/lib/test/unit/httpTreeAccessors.test.d.ts.map +0 -1
  156. package/lib/test/unit/httpTreeAccessors.test.js +0 -1231
  157. package/lib/test/unit/httpTreeAccessors.test.js.map +0 -1
  158. package/lib/test/unit/localStorageTreeAccessors.test.d.ts +0 -2
  159. package/lib/test/unit/localStorageTreeAccessors.test.d.ts.map +0 -1
  160. package/lib/test/unit/localStorageTreeAccessors.test.js +0 -814
  161. package/lib/test/unit/localStorageTreeAccessors.test.js.map +0 -1
  162. package/lib/test/unit/urlParams.test.d.ts +0 -2
  163. package/lib/test/unit/urlParams.test.d.ts.map +0 -1
  164. package/lib/test/unit/urlParams.test.js +0 -395
  165. package/lib/test/unit/urlParams.test.js.map +0 -1
  166. package/lib/test/utils/fileSystemAccessMocks.d.ts +0 -53
  167. package/lib/test/utils/fileSystemAccessMocks.d.ts.map +0 -1
  168. package/lib/test/utils/fileSystemAccessMocks.js +0 -277
  169. package/lib/test/utils/fileSystemAccessMocks.js.map +0 -1
  170. package/lib/test/utils/testHelpers.d.ts +0 -51
  171. package/lib/test/utils/testHelpers.d.ts.map +0 -1
  172. package/lib/test/utils/testHelpers.js +0 -133
  173. package/lib/test/utils/testHelpers.js.map +0 -1
  174. package/rush-logs/ts-web-extras.build.cache.log +0 -0
  175. package/rush-logs/ts-web-extras.build.error.log +0 -18
  176. package/rush-logs/ts-web-extras.build.log +0 -75
  177. package/rush-logs/ts-web-extras.test.cache.log +0 -1
  178. package/rush-logs/ts-web-extras.test.error.log +0 -18
  179. package/rush-logs/ts-web-extras.test.log +0 -75
  180. package/src/index.browser.ts +0 -24
  181. package/src/index.ts +0 -47
  182. package/src/packlets/crypto-utils/browserCryptoProvider.ts +0 -311
  183. package/src/packlets/crypto-utils/browserHashProvider.ts +0 -73
  184. package/src/packlets/crypto-utils/index.ts +0 -29
  185. package/src/packlets/file-api-types/index.ts +0 -366
  186. package/src/packlets/file-tree/directoryHandleStore.ts +0 -136
  187. package/src/packlets/file-tree/fileApiTreeAccessors.ts +0 -528
  188. package/src/packlets/file-tree/fileSystemAccessTreeAccessors.ts +0 -519
  189. package/src/packlets/file-tree/httpTreeAccessors.ts +0 -448
  190. package/src/packlets/file-tree/index.ts +0 -32
  191. package/src/packlets/file-tree/localStorageTreeAccessors.ts +0 -430
  192. package/src/packlets/helpers/fileTreeHelpers.ts +0 -107
  193. package/src/packlets/helpers/index.ts +0 -28
  194. package/src/packlets/url-utils/index.ts +0 -28
  195. package/src/packlets/url-utils/urlParams.ts +0 -245
  196. package/src/test/mocks/idb-keyval.ts +0 -5
  197. package/src/test/setupTests.ts +0 -87
  198. package/src/test/unit/browserHashProvider.test.ts +0 -155
  199. package/src/test/unit/browserHashProvider.test.ts.bak +0 -376
  200. package/src/test/unit/directoryHandleStore.test.ts +0 -251
  201. package/src/test/unit/fileApiTreeAccessors.test.ts +0 -1387
  202. package/src/test/unit/fileApiTypes.test.ts +0 -587
  203. package/src/test/unit/fileSystemAccessTreeAccessors.test.ts +0 -885
  204. package/src/test/unit/fileTreeHelpers.test.ts +0 -694
  205. package/src/test/unit/httpTreeAccessors.test.ts +0 -1571
  206. package/src/test/unit/localStorageTreeAccessors.test.ts +0 -1014
  207. package/src/test/unit/urlParams.test.ts +0 -464
  208. package/src/test/utils/fileSystemAccessMocks.ts +0 -353
  209. package/src/test/utils/testHelpers.ts +0 -155
  210. package/temp/build/typescript/ts_8nwakTlr.json +0 -1
  211. package/temp/coverage/base.css +0 -224
  212. package/temp/coverage/block-navigation.js +0 -87
  213. package/temp/coverage/crypto/browserHashProvider.ts.html +0 -304
  214. package/temp/coverage/crypto/index.html +0 -116
  215. package/temp/coverage/crypto-utils/browserCryptoProvider.ts.html +0 -1018
  216. package/temp/coverage/crypto-utils/browserHashProvider.ts.html +0 -304
  217. package/temp/coverage/crypto-utils/index.html +0 -131
  218. package/temp/coverage/favicon.png +0 -0
  219. package/temp/coverage/file-tree/directoryHandleStore.ts.html +0 -493
  220. package/temp/coverage/file-tree/fileApiTreeAccessors.ts.html +0 -1669
  221. package/temp/coverage/file-tree/fileSystemAccessTreeAccessors.ts.html +0 -1642
  222. package/temp/coverage/file-tree/httpTreeAccessors.ts.html +0 -1429
  223. package/temp/coverage/file-tree/index.html +0 -176
  224. package/temp/coverage/file-tree/localStorageTreeAccessors.ts.html +0 -1375
  225. package/temp/coverage/helpers/fileTreeHelpers.ts.html +0 -406
  226. package/temp/coverage/helpers/index.html +0 -116
  227. package/temp/coverage/index.html +0 -161
  228. package/temp/coverage/lcov-report/base.css +0 -224
  229. package/temp/coverage/lcov-report/block-navigation.js +0 -87
  230. package/temp/coverage/lcov-report/crypto/browserHashProvider.ts.html +0 -304
  231. package/temp/coverage/lcov-report/crypto/index.html +0 -116
  232. package/temp/coverage/lcov-report/crypto-utils/browserCryptoProvider.ts.html +0 -1018
  233. package/temp/coverage/lcov-report/crypto-utils/browserHashProvider.ts.html +0 -304
  234. package/temp/coverage/lcov-report/crypto-utils/index.html +0 -131
  235. package/temp/coverage/lcov-report/favicon.png +0 -0
  236. package/temp/coverage/lcov-report/file-tree/directoryHandleStore.ts.html +0 -493
  237. package/temp/coverage/lcov-report/file-tree/fileApiTreeAccessors.ts.html +0 -1669
  238. package/temp/coverage/lcov-report/file-tree/fileSystemAccessTreeAccessors.ts.html +0 -1642
  239. package/temp/coverage/lcov-report/file-tree/httpTreeAccessors.ts.html +0 -1429
  240. package/temp/coverage/lcov-report/file-tree/index.html +0 -176
  241. package/temp/coverage/lcov-report/file-tree/localStorageTreeAccessors.ts.html +0 -1375
  242. package/temp/coverage/lcov-report/helpers/fileTreeHelpers.ts.html +0 -406
  243. package/temp/coverage/lcov-report/helpers/index.html +0 -116
  244. package/temp/coverage/lcov-report/index.html +0 -161
  245. package/temp/coverage/lcov-report/prettify.css +0 -1
  246. package/temp/coverage/lcov-report/prettify.js +0 -2
  247. package/temp/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  248. package/temp/coverage/lcov-report/sorter.js +0 -210
  249. package/temp/coverage/lcov-report/url-utils/index.html +0 -116
  250. package/temp/coverage/lcov-report/url-utils/urlParams.ts.html +0 -820
  251. package/temp/coverage/lcov.info +0 -3597
  252. package/temp/coverage/prettify.css +0 -1
  253. package/temp/coverage/prettify.js +0 -2
  254. package/temp/coverage/sort-arrow-sprite.png +0 -0
  255. package/temp/coverage/sorter.js +0 -210
  256. package/temp/coverage/url-utils/index.html +0 -116
  257. package/temp/coverage/url-utils/urlParams.ts.html +0 -820
  258. package/temp/test/jest/haste-map-b931e4e63102f86c5bd4949f7dced44f-9d713eb41149188b4e5c0ae3d86d0a57-2ad8e16b24e391b8cdbe50b55c137169 +0 -0
  259. package/temp/test/jest/jest-transform-cache-b931e4e63102f86c5bd4949f7dced44f-79ef2876fae7ca75eedb2aa53dc48338/b5/package_b5f57afc9ec2c011239b1608ee5bdfa5 +0 -53
  260. package/temp/test/jest/perf-cache-b931e4e63102f86c5bd4949f7dced44f-da39a3ee5e6b4b0d3255bfef95601890 +0 -1
  261. package/temp/ts-web-extras.api.json +0 -8850
  262. package/temp/ts-web-extras.api.md +0 -433
  263. 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
- });