From 9dec6bb95332f73e6f7f31b4126fdfc1d10d4f14 Mon Sep 17 00:00:00 2001 From: Justin Walsh Date: Wed, 21 Aug 2024 23:26:26 -0400 Subject: [PATCH 1/6] fix: ensure that useAssets does not infer type from data --- src/hooks/useAsset.ts | 4 ++-- src/hooks/useAssets.ts | 4 ++-- src/typedefs/UseAssetsResult.ts | 4 ++-- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/src/hooks/useAsset.ts b/src/hooks/useAsset.ts index e3a4e5b7..20a0b30d 100644 --- a/src/hooks/useAsset.ts +++ b/src/hooks/useAsset.ts @@ -15,9 +15,9 @@ import type { ErrorCallback } from '../typedefs/ErrorCallback'; const errorCache: Map = new Map(); /** @deprecated Use `useAssets` instead. */ -export function useAsset( +export function useAsset( /** @description Asset options. */ - options: (UnresolvedAsset & AssetRetryOptions) | string, + options: (UnresolvedAsset & AssetRetryOptions) | string, /** @description A function to be called when the asset loader reports loading progress. */ onProgress?: ProgressCallback, /** @description A function to be called when the asset loader reports loading progress. */ diff --git a/src/hooks/useAssets.ts b/src/hooks/useAssets.ts index 8544780d..273336cc 100644 --- a/src/hooks/useAssets.ts +++ b/src/hooks/useAssets.ts @@ -21,14 +21,14 @@ function assetsLoadedTest(asset: UnresolvedAsset) /** Loads assets, returning a hash of assets once they're loaded. */ export function useAssets( /** @description Assets to be loaded. */ - assets: UnresolvedAsset[], + assets: UnresolvedAsset[], /** @description Asset options. */ options: UseAssetsOptions = {}, ): UseAssetsResult { const [state, setState] = useState>({ - assets: Array(assets.length).fill(null), + assets: Array(assets.length).fill(undefined), isError: false, isPending: true, isSuccess: false, diff --git a/src/typedefs/UseAssetsResult.ts b/src/typedefs/UseAssetsResult.ts index e3cfb2b8..b525ed2a 100644 --- a/src/typedefs/UseAssetsResult.ts +++ b/src/typedefs/UseAssetsResult.ts @@ -2,8 +2,8 @@ import type { UseAssetsStatus } from '../constants/UseAssetsStatus'; export interface UseAssetsResult { - /** @description An array of resolved assets, or `null` for assets that are still loading. */ - assets: (T | null)[]; + /** @description An array of resolved assets, or `undefined` for assets that are still loading. */ + assets: (T | undefined)[]; /** @description The error that was encountered. */ error?: Error; From e2f2fb2812644a297e43c2912aa27079ae0eb0f4 Mon Sep 17 00:00:00 2001 From: Justin Walsh Date: Wed, 21 Aug 2024 23:27:13 -0400 Subject: [PATCH 2/6] test: useAssets tests --- package-lock.json | 297 +++++++++++++++++++++-------- package.json | 4 + test/unit/hooks/useAssets.test.tsx | 92 +++++++++ 3 files changed, 310 insertions(+), 83 deletions(-) create mode 100644 test/unit/hooks/useAssets.test.tsx diff --git a/package-lock.json b/package-lock.json index 02efbaf6..8a93640b 100644 --- a/package-lock.json +++ b/package-lock.json @@ -16,6 +16,7 @@ "@rollup/plugin-commonjs": "^25.0.8", "@rollup/plugin-json": "^6.1.0", "@rollup/plugin-node-resolve": "^15.2.3", + "@testing-library/react": "^16.0.0", "@types/eslint": "^8.56.10", "@types/react": "^18.3.2", "@types/react-reconciler": "0.28.8", @@ -2674,22 +2675,6 @@ "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", "dev": true }, - "node_modules/@pixi/extension-scripts/node_modules/rollup": { - "version": "3.29.4", - "resolved": "https://registry.npmjs.org/rollup/-/rollup-3.29.4.tgz", - "integrity": "sha512-oWzmBZwvYrU0iJHtDmhsm662rC15FRXmcjCk1xD771dFDx5jJ02ufAQQTn0etB2emNk4J9EZg/yWKpsn9BWGRw==", - "dev": true, - "bin": { - "rollup": "dist/bin/rollup" - }, - "engines": { - "node": ">=14.18.0", - "npm": ">=8.0.0" - }, - "optionalDependencies": { - "fsevents": "~2.3.2" - } - }, "node_modules/@pixi/extension-scripts/node_modules/rollup-plugin-esbuild": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/rollup-plugin-esbuild/-/rollup-plugin-esbuild-5.0.0.tgz", @@ -2933,9 +2918,9 @@ } }, "node_modules/@rollup/rollup-android-arm-eabi": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.18.0.tgz", - "integrity": "sha512-Tya6xypR10giZV1XzxmH5wr25VcZSncG0pZIjfePT0OVBvqNEurzValetGNarVrGiq66EBVAFn15iYX4w6FKgQ==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.21.0.tgz", + "integrity": "sha512-WTWD8PfoSAJ+qL87lE7votj3syLavxunWhzCnx3XFxFiI/BA/r3X7MUM8dVrH8rb2r4AiO8jJsr3ZjdaftmnfA==", "cpu": [ "arm" ], @@ -2946,9 +2931,9 @@ ] }, "node_modules/@rollup/rollup-android-arm64": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.18.0.tgz", - "integrity": "sha512-avCea0RAP03lTsDhEyfy+hpfr85KfyTctMADqHVhLAF3MlIkq83CP8UfAHUssgXTYd+6er6PaAhx/QGv4L1EiA==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.21.0.tgz", + "integrity": "sha512-a1sR2zSK1B4eYkiZu17ZUZhmUQcKjk2/j9Me2IDjk1GHW7LB5Z35LEzj9iJch6gtUfsnvZs1ZNyDW2oZSThrkA==", "cpu": [ "arm64" ], @@ -2959,9 +2944,9 @@ ] }, "node_modules/@rollup/rollup-darwin-arm64": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.18.0.tgz", - "integrity": "sha512-IWfdwU7KDSm07Ty0PuA/W2JYoZ4iTj3TUQjkVsO/6U+4I1jN5lcR71ZEvRh52sDOERdnNhhHU57UITXz5jC1/w==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.21.0.tgz", + "integrity": "sha512-zOnKWLgDld/svhKO5PD9ozmL6roy5OQ5T4ThvdYZLpiOhEGY+dp2NwUmxK0Ld91LrbjrvtNAE0ERBwjqhZTRAA==", "cpu": [ "arm64" ], @@ -2972,9 +2957,9 @@ ] }, "node_modules/@rollup/rollup-darwin-x64": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.18.0.tgz", - "integrity": "sha512-n2LMsUz7Ynu7DoQrSQkBf8iNrjOGyPLrdSg802vk6XT3FtsgX6JbE8IHRvposskFm9SNxzkLYGSq9QdpLYpRNA==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.21.0.tgz", + "integrity": "sha512-7doS8br0xAkg48SKE2QNtMSFPFUlRdw9+votl27MvT46vo44ATBmdZdGysOevNELmZlfd+NEa0UYOA8f01WSrg==", "cpu": [ "x64" ], @@ -2985,9 +2970,9 @@ ] }, "node_modules/@rollup/rollup-linux-arm-gnueabihf": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.18.0.tgz", - "integrity": "sha512-C/zbRYRXFjWvz9Z4haRxcTdnkPt1BtCkz+7RtBSuNmKzMzp3ZxdM28Mpccn6pt28/UWUCTXa+b0Mx1k3g6NOMA==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.21.0.tgz", + "integrity": "sha512-pWJsfQjNWNGsoCq53KjMtwdJDmh/6NubwQcz52aEwLEuvx08bzcy6tOUuawAOncPnxz/3siRtd8hiQ32G1y8VA==", "cpu": [ "arm" ], @@ -2998,9 +2983,9 @@ ] }, "node_modules/@rollup/rollup-linux-arm-musleabihf": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.18.0.tgz", - "integrity": "sha512-l3m9ewPgjQSXrUMHg93vt0hYCGnrMOcUpTz6FLtbwljo2HluS4zTXFy2571YQbisTnfTKPZ01u/ukJdQTLGh9A==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.21.0.tgz", + "integrity": "sha512-efRIANsz3UHZrnZXuEvxS9LoCOWMGD1rweciD6uJQIx2myN3a8Im1FafZBzh7zk1RJ6oKcR16dU3UPldaKd83w==", "cpu": [ "arm" ], @@ -3011,9 +2996,9 @@ ] }, "node_modules/@rollup/rollup-linux-arm64-gnu": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.18.0.tgz", - "integrity": "sha512-rJ5D47d8WD7J+7STKdCUAgmQk49xuFrRi9pZkWoRD1UeSMakbcepWXPF8ycChBoAqs1pb2wzvbY6Q33WmN2ftw==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.21.0.tgz", + "integrity": "sha512-ZrPhydkTVhyeGTW94WJ8pnl1uroqVHM3j3hjdquwAcWnmivjAwOYjTEAuEDeJvGX7xv3Z9GAvrBkEzCgHq9U1w==", "cpu": [ "arm64" ], @@ -3024,9 +3009,9 @@ ] }, "node_modules/@rollup/rollup-linux-arm64-musl": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.18.0.tgz", - "integrity": "sha512-be6Yx37b24ZwxQ+wOQXXLZqpq4jTckJhtGlWGZs68TgdKXJgw54lUUoFYrg6Zs/kjzAQwEwYbp8JxZVzZLRepQ==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.21.0.tgz", + "integrity": "sha512-cfaupqd+UEFeURmqNP2eEvXqgbSox/LHOyN9/d2pSdV8xTrjdg3NgOFJCtc1vQ/jEke1qD0IejbBfxleBPHnPw==", "cpu": [ "arm64" ], @@ -3037,9 +3022,9 @@ ] }, "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.18.0.tgz", - "integrity": "sha512-hNVMQK+qrA9Todu9+wqrXOHxFiD5YmdEi3paj6vP02Kx1hjd2LLYR2eaN7DsEshg09+9uzWi2W18MJDlG0cxJA==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.21.0.tgz", + "integrity": "sha512-ZKPan1/RvAhrUylwBXC9t7B2hXdpb/ufeu22pG2psV7RN8roOfGurEghw1ySmX/CmDDHNTDDjY3lo9hRlgtaHg==", "cpu": [ "ppc64" ], @@ -3050,9 +3035,9 @@ ] }, "node_modules/@rollup/rollup-linux-riscv64-gnu": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.18.0.tgz", - "integrity": "sha512-ROCM7i+m1NfdrsmvwSzoxp9HFtmKGHEqu5NNDiZWQtXLA8S5HBCkVvKAxJ8U+CVctHwV2Gb5VUaK7UAkzhDjlg==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.21.0.tgz", + "integrity": "sha512-H1eRaCwd5E8eS8leiS+o/NqMdljkcb1d6r2h4fKSsCXQilLKArq6WS7XBLDu80Yz+nMqHVFDquwcVrQmGr28rg==", "cpu": [ "riscv64" ], @@ -3063,9 +3048,9 @@ ] }, "node_modules/@rollup/rollup-linux-s390x-gnu": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.18.0.tgz", - "integrity": "sha512-0UyyRHyDN42QL+NbqevXIIUnKA47A+45WyasO+y2bGJ1mhQrfrtXUpTxCOrfxCR4esV3/RLYyucGVPiUsO8xjg==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.21.0.tgz", + "integrity": "sha512-zJ4hA+3b5tu8u7L58CCSI0A9N1vkfwPhWd/puGXwtZlsB5bTkwDNW/+JCU84+3QYmKpLi+XvHdmrlwUwDA6kqw==", "cpu": [ "s390x" ], @@ -3076,9 +3061,9 @@ ] }, "node_modules/@rollup/rollup-linux-x64-gnu": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.18.0.tgz", - "integrity": "sha512-xuglR2rBVHA5UsI8h8UbX4VJ470PtGCf5Vpswh7p2ukaqBGFTnsfzxUBetoWBWymHMxbIG0Cmx7Y9qDZzr648w==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.21.0.tgz", + "integrity": "sha512-e2hrvElFIh6kW/UNBQK/kzqMNY5mO+67YtEh9OA65RM5IJXYTWiXjX6fjIiPaqOkBthYF1EqgiZ6OXKcQsM0hg==", "cpu": [ "x64" ], @@ -3089,9 +3074,9 @@ ] }, "node_modules/@rollup/rollup-linux-x64-musl": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.18.0.tgz", - "integrity": "sha512-LKaqQL9osY/ir2geuLVvRRs+utWUNilzdE90TpyoX0eNqPzWjRm14oMEE+YLve4k/NAqCdPkGYDaDF5Sw+xBfg==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.21.0.tgz", + "integrity": "sha512-1vvmgDdUSebVGXWX2lIcgRebqfQSff0hMEkLJyakQ9JQUbLDkEaMsPTLOmyccyC6IJ/l3FZuJbmrBw/u0A0uCQ==", "cpu": [ "x64" ], @@ -3102,9 +3087,9 @@ ] }, "node_modules/@rollup/rollup-win32-arm64-msvc": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.18.0.tgz", - "integrity": "sha512-7J6TkZQFGo9qBKH0pk2cEVSRhJbL6MtfWxth7Y5YmZs57Pi+4x6c2dStAUvaQkHQLnEQv1jzBUW43GvZW8OFqA==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.21.0.tgz", + "integrity": "sha512-s5oFkZ/hFcrlAyBTONFY1TWndfyre1wOMwU+6KCpm/iatybvrRgmZVM+vCFwxmC5ZhdlgfE0N4XorsDpi7/4XQ==", "cpu": [ "arm64" ], @@ -3115,9 +3100,9 @@ ] }, "node_modules/@rollup/rollup-win32-ia32-msvc": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.18.0.tgz", - "integrity": "sha512-Txjh+IxBPbkUB9+SXZMpv+b/vnTEtFyfWZgJ6iyCmt2tdx0OF5WhFowLmnh8ENGNpfUlUZkdI//4IEmhwPieNg==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.21.0.tgz", + "integrity": "sha512-G9+TEqRnAA6nbpqyUqgTiopmnfgnMkR3kMukFBDsiyy23LZvUCpiUwjTRx6ezYCjJODXrh52rBR9oXvm+Fp5wg==", "cpu": [ "ia32" ], @@ -3128,9 +3113,9 @@ ] }, "node_modules/@rollup/rollup-win32-x64-msvc": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.18.0.tgz", - "integrity": "sha512-UOo5FdvOL0+eIVTgS4tIdbW+TtnBLWg1YBCcU2KWM7nuNwRz9bksDX1bekJJCpu25N1DVWaCwnT39dVQxzqS8g==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.21.0.tgz", + "integrity": "sha512-2jsCDZwtQvRhejHLfZ1JY6w6kEuEtfF9nzYsZxzSlNVKDX+DpsDJ+Rbjkm74nvg2rdx0gwBS+IMdvwJuq3S9pQ==", "cpu": [ "x64" ], @@ -3303,6 +3288,108 @@ "node": ">=10" } }, + "node_modules/@testing-library/dom": { + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-10.4.0.tgz", + "integrity": "sha512-pemlzrSESWbdAloYml3bAJMEfNh1Z7EduzqPKprCH5S341frlpYnUEW0H72dLxa6IsYr+mPno20GiSm+h9dEdQ==", + "dev": true, + "peer": true, + "dependencies": { + "@babel/code-frame": "^7.10.4", + "@babel/runtime": "^7.12.5", + "@types/aria-query": "^5.0.1", + "aria-query": "5.3.0", + "chalk": "^4.1.0", + "dom-accessibility-api": "^0.5.9", + "lz-string": "^1.5.0", + "pretty-format": "^27.0.2" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/@testing-library/dom/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "peer": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/@testing-library/dom/node_modules/chalk/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "peer": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@testing-library/dom/node_modules/pretty-format": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", + "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", + "dev": true, + "peer": true, + "dependencies": { + "ansi-regex": "^5.0.1", + "ansi-styles": "^5.0.0", + "react-is": "^17.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/@testing-library/dom/node_modules/react-is": { + "version": "17.0.2", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", + "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", + "dev": true, + "peer": true + }, + "node_modules/@testing-library/react": { + "version": "16.0.0", + "resolved": "https://registry.npmjs.org/@testing-library/react/-/react-16.0.0.tgz", + "integrity": "sha512-guuxUKRWQ+FgNX0h0NS0FIq3Q3uLtWVpBzcLOggmfMoUpgBnzBzvLLd4fbm6yS8ydJd94cIfY4yP9qUQjM2KwQ==", + "dev": true, + "dependencies": { + "@babel/runtime": "^7.12.5" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@testing-library/dom": "^10.0.0", + "@types/react": "^18.0.0", + "@types/react-dom": "^18.0.0", + "react": "^18.0.0", + "react-dom": "^18.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, "node_modules/@tootallnate/once": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-2.0.0.tgz", @@ -3346,6 +3433,13 @@ "optional": true, "peer": true }, + "node_modules/@types/aria-query": { + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/@types/aria-query/-/aria-query-5.0.4.tgz", + "integrity": "sha512-rfT93uj5s0PRL7EzccGMs3brplhcrghnDoV26NqKhCAS1hVo+WdNsPvE/yb6ilfr5hi2MEk6d5EWJTKdxg8jVw==", + "dev": true, + "peer": true + }, "node_modules/@types/babel__core": { "version": "7.20.5", "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", @@ -5746,6 +5840,16 @@ "sprintf-js": "~1.0.2" } }, + "node_modules/aria-query": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.0.tgz", + "integrity": "sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==", + "dev": true, + "peer": true, + "dependencies": { + "dequal": "^2.0.3" + } + }, "node_modules/arr-diff": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", @@ -7377,6 +7481,16 @@ "dev": true, "license": "MIT" }, + "node_modules/dequal": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", + "dev": true, + "peer": true, + "engines": { + "node": ">=6" + } + }, "node_modules/detect-file": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/detect-file/-/detect-file-1.0.0.tgz", @@ -7457,6 +7571,13 @@ "node": ">=0.10.0" } }, + "node_modules/dom-accessibility-api": { + "version": "0.5.16", + "resolved": "https://registry.npmjs.org/dom-accessibility-api/-/dom-accessibility-api-0.5.16.tgz", + "integrity": "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg==", + "dev": true, + "peer": true + }, "node_modules/domexception": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/domexception/-/domexception-4.0.0.tgz", @@ -12388,6 +12509,16 @@ "yallist": "^3.0.2" } }, + "node_modules/lz-string": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/lz-string/-/lz-string-1.5.0.tgz", + "integrity": "sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==", + "dev": true, + "peer": true, + "bin": { + "lz-string": "bin/bin.js" + } + }, "node_modules/magic-string": { "version": "0.25.9", "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.9.tgz", @@ -14338,9 +14469,9 @@ "optional": true }, "node_modules/rollup": { - "version": "4.18.0", - "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.18.0.tgz", - "integrity": "sha512-QmJz14PX3rzbJCN1SG4Xe/bAAX2a6NpCP8ab2vfu2GiUr8AQcr2nCV/oEO3yneFarB67zk8ShlIyWb2LGTb3Sg==", + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.21.0.tgz", + "integrity": "sha512-vo+S/lfA2lMS7rZ2Qoubi6I5hwZwzXeUIctILZLbHI+laNtvhhOIon2S1JksA5UEDQ7l3vberd0fxK44lTYjbQ==", "dev": true, "dependencies": { "@types/estree": "1.0.5" @@ -14353,22 +14484,22 @@ "npm": ">=8.0.0" }, "optionalDependencies": { - "@rollup/rollup-android-arm-eabi": "4.18.0", - "@rollup/rollup-android-arm64": "4.18.0", - "@rollup/rollup-darwin-arm64": "4.18.0", - "@rollup/rollup-darwin-x64": "4.18.0", - "@rollup/rollup-linux-arm-gnueabihf": "4.18.0", - "@rollup/rollup-linux-arm-musleabihf": "4.18.0", - "@rollup/rollup-linux-arm64-gnu": "4.18.0", - "@rollup/rollup-linux-arm64-musl": "4.18.0", - "@rollup/rollup-linux-powerpc64le-gnu": "4.18.0", - "@rollup/rollup-linux-riscv64-gnu": "4.18.0", - "@rollup/rollup-linux-s390x-gnu": "4.18.0", - "@rollup/rollup-linux-x64-gnu": "4.18.0", - "@rollup/rollup-linux-x64-musl": "4.18.0", - "@rollup/rollup-win32-arm64-msvc": "4.18.0", - "@rollup/rollup-win32-ia32-msvc": "4.18.0", - "@rollup/rollup-win32-x64-msvc": "4.18.0", + "@rollup/rollup-android-arm-eabi": "4.21.0", + "@rollup/rollup-android-arm64": "4.21.0", + "@rollup/rollup-darwin-arm64": "4.21.0", + "@rollup/rollup-darwin-x64": "4.21.0", + "@rollup/rollup-linux-arm-gnueabihf": "4.21.0", + "@rollup/rollup-linux-arm-musleabihf": "4.21.0", + "@rollup/rollup-linux-arm64-gnu": "4.21.0", + "@rollup/rollup-linux-arm64-musl": "4.21.0", + "@rollup/rollup-linux-powerpc64le-gnu": "4.21.0", + "@rollup/rollup-linux-riscv64-gnu": "4.21.0", + "@rollup/rollup-linux-s390x-gnu": "4.21.0", + "@rollup/rollup-linux-x64-gnu": "4.21.0", + "@rollup/rollup-linux-x64-musl": "4.21.0", + "@rollup/rollup-win32-arm64-msvc": "4.21.0", + "@rollup/rollup-win32-ia32-msvc": "4.21.0", + "@rollup/rollup-win32-x64-msvc": "4.21.0", "fsevents": "~2.3.2" } }, diff --git a/package.json b/package.json index 8f236ba8..1dc2a0b0 100644 --- a/package.json +++ b/package.json @@ -66,6 +66,7 @@ "@rollup/plugin-commonjs": "^25.0.8", "@rollup/plugin-json": "^6.1.0", "@rollup/plugin-node-resolve": "^15.2.3", + "@testing-library/react": "^16.0.0", "@types/eslint": "^8.56.10", "@types/react": "^18.3.2", "@types/react-reconciler": "0.28.8", @@ -90,6 +91,9 @@ "optional": true } }, + "overrides": { + "rollup": "^4.18.0" + }, "publishConfig": { "access": "public" }, diff --git a/test/unit/hooks/useAssets.test.tsx b/test/unit/hooks/useAssets.test.tsx new file mode 100644 index 00000000..f4bec662 --- /dev/null +++ b/test/unit/hooks/useAssets.test.tsx @@ -0,0 +1,92 @@ +import { Assets, Cache, Sprite, Texture, type UnresolvedAsset } from 'pixi.js'; +import { afterAll, afterEach, beforeEach, describe, expect, it, vi } from 'vitest'; +import { extend, useAssets } from '../../../src'; +import { cleanup, renderHook, waitFor } from '@testing-library/react'; + +extend({ Sprite }); + +describe('useAssets', async () => +{ + const assets: UnresolvedAsset[] = [{ src: 'test.png' }, { src: 'test2.png' }]; + + // Mock the Assets.load, Assets.get & Cache.has method + let loaded: Record = {}; + const load = vi.spyOn(Assets, 'load'); + const get = vi.spyOn(Assets, 'get'); + const has = vi.spyOn(Cache, 'has'); + + // Mock the Assets.load to populate the loaded record, and resolve after 100ms + load.mockImplementation((urls) => + { + const assets = urls as UnresolvedAsset[]; + const result = assets.reduce((acc, val) => ({ ...acc, [val.src!.toString()]: Object.assign(Texture.EMPTY, { data: val.data }) }), {}); + + return new Promise((resolve) => + { + setTimeout(() => + { + loaded = { ...loaded, ...result }; + resolve(result); + }, 10); + }); + }); + + // Mock the Assets.get to return the loaded record + get.mockImplementation((keys) => + keys.reduce>((acc, key, idx) => ({ ...acc, [idx]: loaded[key] }), {})); + + // Mock the Cache.has to check if the key is in the loaded record + has.mockImplementation((key) => key in loaded); + + // Load the default results using Assets.load to compare against the results from the useAssets hook + const defaultResults = await Assets.load(assets); + + beforeEach(() => + { + loaded = {}; + }); + + afterEach(() => + { + cleanup(); + }); + + afterAll(() => + { + load.mockRestore(); + get.mockRestore(); + }); + + it('loads assets', async () => + { + const { result } = renderHook(() => useAssets(assets)); + + expect(result.current.isPending).toBe(true); + await waitFor(() => expect(result.current.isSuccess).toBe(true)); + + expect(result.current.assets).toEqual(assets.map(({ src }) => defaultResults[src!.toString()])); + }); + + it('accepts data', async () => + { + type Data = { data: { test: string } }; + const { result } = renderHook(() => useAssets([ + { src: 'test.png', data: { test: 'test' } }, + { src: 'test2.png', data: { test: 'test' } }, + ])); + + expect(result.current.isPending).toBe(true); + await waitFor(() => expect(result.current.isSuccess).toBe(true)); + + const { assets: [texture], isSuccess } = result.current; + + expect(isSuccess).toBe(true); + expect(texture?.data.test).toBe('test'); + + const isTexture = (texture?: Texture) => texture && texture instanceof Texture; + const getData = (texture?: Texture & Data) => texture?.data.test; + + expect(isTexture(texture)).toBe(true); + expect(getData(texture)).toBe('test'); + }); +}); From 800d9e52a39c743e778a126ca785493b423d5b17 Mon Sep 17 00:00:00 2001 From: Justin Walsh Date: Wed, 21 Aug 2024 23:43:09 -0400 Subject: [PATCH 3/6] fix: useSuspenseAssets --- src/hooks/useSuspenseAssets.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/hooks/useSuspenseAssets.ts b/src/hooks/useSuspenseAssets.ts index a19ff2fe..992a8904 100644 --- a/src/hooks/useSuspenseAssets.ts +++ b/src/hooks/useSuspenseAssets.ts @@ -18,7 +18,7 @@ function assetsLoadedTest(asset: UnresolvedAsset) /** Loads assets, returning a hash of assets once they're loaded. Must be inside of a `` component. */ export function useSuspenseAssets( /** @description Assets to be loaded. */ - assets: UnresolvedAsset[], + assets: UnresolvedAsset[], /** @description Asset options. */ options: UseAssetsOptions = {}, ): T[] From 34f844bc9ab6ce6ee80ba4ab7672246fc4681253 Mon Sep 17 00:00:00 2001 From: Justin Walsh Date: Wed, 21 Aug 2024 23:51:50 -0400 Subject: [PATCH 4/6] test: ensure that it works untyped as well --- test/unit/hooks/useAssets.test.tsx | 22 ++++++++++++++++++++++ 1 file changed, 22 insertions(+) diff --git a/test/unit/hooks/useAssets.test.tsx b/test/unit/hooks/useAssets.test.tsx index f4bec662..5b20b361 100644 --- a/test/unit/hooks/useAssets.test.tsx +++ b/test/unit/hooks/useAssets.test.tsx @@ -89,4 +89,26 @@ describe('useAssets', async () => expect(isTexture(texture)).toBe(true); expect(getData(texture)).toBe('test'); }); + + it('is properly typed with data', async () => + { + type Data = { data: { test: string } }; + const { result } = renderHook(() => useAssets([ + { src: 'test.png', data: { test: 'test' } }, + ])); + + expect(result.current.isPending).toBe(true); + await waitFor(() => expect(result.current.isSuccess).toBe(true)); + + const { assets: [texture], isSuccess } = result.current; + + expect(isSuccess).toBe(true); + expect(texture?.data.test).toBe('test'); + + const isTexture = (texture?: Texture) => texture && texture instanceof Texture; + const getData = (texture?: Texture & Data) => texture?.data.test; + + expect(isTexture(texture)).toBe(true); + expect(getData(texture)).toBe('test'); + }); }); From 6a00c77ba849ac1603b13ece88c32cdc1b55cff7 Mon Sep 17 00:00:00 2001 From: Justin Walsh Date: Wed, 21 Aug 2024 23:55:41 -0400 Subject: [PATCH 5/6] test: update timeout in tests --- test/unit/hooks/useAssets.test.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/unit/hooks/useAssets.test.tsx b/test/unit/hooks/useAssets.test.tsx index 5b20b361..4aa24e46 100644 --- a/test/unit/hooks/useAssets.test.tsx +++ b/test/unit/hooks/useAssets.test.tsx @@ -15,7 +15,7 @@ describe('useAssets', async () => const get = vi.spyOn(Assets, 'get'); const has = vi.spyOn(Cache, 'has'); - // Mock the Assets.load to populate the loaded record, and resolve after 100ms + // Mock the Assets.load to populate the loaded record, and resolve after 1ms load.mockImplementation((urls) => { const assets = urls as UnresolvedAsset[]; @@ -27,7 +27,7 @@ describe('useAssets', async () => { loaded = { ...loaded, ...result }; resolve(result); - }, 10); + }, 1); }); }); From 1427db4c32ca16ce167b61d05639bc1de99a5345 Mon Sep 17 00:00:00 2001 From: Justin Walsh Date: Thu, 22 Aug 2024 14:51:10 -0400 Subject: [PATCH 6/6] test: ensure tests do not capture data in the asset mocking actual pixi behavior --- test/unit/hooks/useAssets.test.tsx | 32 +++++++++++++++--------------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/test/unit/hooks/useAssets.test.tsx b/test/unit/hooks/useAssets.test.tsx index 4aa24e46..0f4143be 100644 --- a/test/unit/hooks/useAssets.test.tsx +++ b/test/unit/hooks/useAssets.test.tsx @@ -9,8 +9,11 @@ describe('useAssets', async () => { const assets: UnresolvedAsset[] = [{ src: 'test.png' }, { src: 'test2.png' }]; - // Mock the Assets.load, Assets.get & Cache.has method + // Store the loaded assets and data state to verify the hook results let loaded: Record = {}; + let data: Record = {}; + + // Mock the Assets.load, Assets.get & Cache.has method const load = vi.spyOn(Assets, 'load'); const get = vi.spyOn(Assets, 'get'); const has = vi.spyOn(Cache, 'has'); @@ -19,14 +22,14 @@ describe('useAssets', async () => load.mockImplementation((urls) => { const assets = urls as UnresolvedAsset[]; - const result = assets.reduce((acc, val) => ({ ...acc, [val.src!.toString()]: Object.assign(Texture.EMPTY, { data: val.data }) }), {}); return new Promise((resolve) => { setTimeout(() => { - loaded = { ...loaded, ...result }; - resolve(result); + loaded = { ...loaded, ...assets.reduce((acc, val) => ({ ...acc, [val.src!.toString()]: Texture.EMPTY }), {}) }; + data = { ...data, ...assets.reduce((acc, val) => ({ ...acc, [val.src!.toString()]: val.data }), {}) }; + resolve(loaded); }, 1); }); }); @@ -44,6 +47,7 @@ describe('useAssets', async () => beforeEach(() => { loaded = {}; + data = {}; }); afterEach(() => @@ -69,10 +73,10 @@ describe('useAssets', async () => it('accepts data', async () => { - type Data = { data: { test: string } }; - const { result } = renderHook(() => useAssets([ - { src: 'test.png', data: { test: 'test' } }, - { src: 'test2.png', data: { test: 'test' } }, + // Explicitly type the T in the useAssets hook + const { result } = renderHook(() => useAssets([ + { src: 'test.png', data: { test: '7a1c8bee' } }, + { src: 'test2.png', data: { test: '230a3f41' } }, ])); expect(result.current.isPending).toBe(true); @@ -81,20 +85,19 @@ describe('useAssets', async () => const { assets: [texture], isSuccess } = result.current; expect(isSuccess).toBe(true); - expect(texture?.data.test).toBe('test'); + expect(data['test.png'].test).toBe('7a1c8bee'); + expect(data['test2.png'].test).toBe('230a3f41'); const isTexture = (texture?: Texture) => texture && texture instanceof Texture; - const getData = (texture?: Texture & Data) => texture?.data.test; expect(isTexture(texture)).toBe(true); - expect(getData(texture)).toBe('test'); }); it('is properly typed with data', async () => { - type Data = { data: { test: string } }; + // Do not provide a type for T in the useAssets hook const { result } = renderHook(() => useAssets([ - { src: 'test.png', data: { test: 'test' } }, + { src: 'test.png', data: { test: 'd460dbdd' } }, ])); expect(result.current.isPending).toBe(true); @@ -103,12 +106,9 @@ describe('useAssets', async () => const { assets: [texture], isSuccess } = result.current; expect(isSuccess).toBe(true); - expect(texture?.data.test).toBe('test'); const isTexture = (texture?: Texture) => texture && texture instanceof Texture; - const getData = (texture?: Texture & Data) => texture?.data.test; expect(isTexture(texture)).toBe(true); - expect(getData(texture)).toBe('test'); }); });