diff --git a/.github/logo-dark.svg b/.github/logo-dark.svg index e0258f10..7319e655 100644 --- a/.github/logo-dark.svg +++ b/.github/logo-dark.svg @@ -1,11 +1 @@ - - - - - - - - - - - + \ No newline at end of file diff --git a/.github/logo-light.svg b/.github/logo-light.svg index dc5a242f..e366b1cd 100644 --- a/.github/logo-light.svg +++ b/.github/logo-light.svg @@ -1,11 +1 @@ - - - - - - - - - - - + \ No newline at end of file diff --git a/package.json b/package.json index becaa2e7..4eb7951a 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "movie-web", - "version": "4.6.1", + "version": "4.6.2", "private": true, "homepage": "https://github.com/movie-web/movie-web", "scripts": { @@ -29,7 +29,7 @@ "@formkit/auto-animate": "^0.8.1", "@headlessui/react": "^1.7.17", "@ladjs/country-language": "^1.0.3", - "@movie-web/providers": "^2.2.2", + "@movie-web/providers": "^2.2.3", "@noble/hashes": "^1.3.3", "@plasmohq/messaging": "^0.6.1", "@react-spring/web": "^9.7.3", @@ -44,7 +44,7 @@ "focus-trap-react": "^10.2.3", "fscreen": "^1.2.0", "fuse.js": "^7.0.0", - "hls.js": "^1.4.14", + "hls.js": "^1.5.7", "i18next": "^23.7.11", "immer": "^10.0.3", "jwt-decode": "^4.0.0", diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index b0cdcb77..313fae85 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -22,8 +22,8 @@ dependencies: specifier: ^1.0.3 version: 1.0.3 '@movie-web/providers': - specifier: ^2.2.2 - version: 2.2.2 + specifier: ^2.2.3 + version: 2.2.3 '@noble/hashes': specifier: ^1.3.3 version: 1.3.3 @@ -67,8 +67,8 @@ dependencies: specifier: ^7.0.0 version: 7.0.0 hls.js: - specifier: ^1.4.14 - version: 1.4.14 + specifier: ^1.5.7 + version: 1.5.7 i18next: specifier: ^23.7.11 version: 23.7.11 @@ -274,7 +274,7 @@ devDependencies: version: 0.5.9(prettier@3.1.1) rollup-plugin-visualizer: specifier: ^5.11.0 - version: 5.11.0(@rollup/wasm-node@4.12.1) + version: 5.11.0(@rollup/wasm-node@4.13.0) tailwind-scrollbar: specifier: ^3.0.5 version: 3.0.5(tailwindcss@3.4.0) @@ -1942,8 +1942,8 @@ packages: engines: {node: '>= 14'} dev: false - /@movie-web/providers@2.2.2: - resolution: {integrity: sha512-pTlErE5bdu+b68mUW2YAKOJKz2hwSx63auGAfTkGQ+0SHDMlCV9QQ8S8O9IoSsvdXps7/YlWJWOMX8pmKuYbPQ==} + /@movie-web/providers@2.2.3: + resolution: {integrity: sha512-0axy02Zzlk7Tvtalc/Ebv9u5vzUPVWmQm0Ts5+6l6KPU41JUdLnFgmOl0yf0lbNeHRNSTx5SDlvWcYNL8rgpyA==} dependencies: cheerio: 1.0.0-rc.12 cookie: 0.6.0 @@ -2068,7 +2068,7 @@ packages: engines: {node: '>=14.0.0'} dev: false - /@rollup/plugin-babel@5.3.1(@babel/core@7.23.6)(@rollup/wasm-node@4.12.1): + /@rollup/plugin-babel@5.3.1(@babel/core@7.23.6)(@rollup/wasm-node@4.13.0): resolution: {integrity: sha512-WFfdLWU/xVWKeRQnKmIAQULUI7Il0gZnBIH/ZFO069wYIfPu+8zrfp/KMW0atmELoRDq8FbiP3VCss9MhCut7Q==} engines: {node: '>= 10.0.0'} peerDependencies: @@ -2081,36 +2081,36 @@ packages: dependencies: '@babel/core': 7.23.6 '@babel/helper-module-imports': 7.22.15 - '@rollup/pluginutils': 3.1.0(@rollup/wasm-node@4.12.1) - rollup: /@rollup/wasm-node@4.12.1 + '@rollup/pluginutils': 3.1.0(@rollup/wasm-node@4.13.0) + rollup: /@rollup/wasm-node@4.13.0 dev: true - /@rollup/plugin-node-resolve@11.2.1(@rollup/wasm-node@4.12.1): + /@rollup/plugin-node-resolve@11.2.1(@rollup/wasm-node@4.13.0): resolution: {integrity: sha512-yc2n43jcqVyGE2sqV5/YCmocy9ArjVAP/BeXyTtADTBBX6V0e5UMqwO8CdQ0kzjb6zu5P1qMzsScCMRvE9OlVg==} engines: {node: '>= 10.0.0'} peerDependencies: rollup: npm:@rollup/wasm-node dependencies: - '@rollup/pluginutils': 3.1.0(@rollup/wasm-node@4.12.1) + '@rollup/pluginutils': 3.1.0(@rollup/wasm-node@4.13.0) '@types/resolve': 1.17.1 builtin-modules: 3.3.0 deepmerge: 4.3.1 is-module: 1.0.0 resolve: 1.22.4 - rollup: /@rollup/wasm-node@4.12.1 + rollup: /@rollup/wasm-node@4.13.0 dev: true - /@rollup/plugin-replace@2.4.2(@rollup/wasm-node@4.12.1): + /@rollup/plugin-replace@2.4.2(@rollup/wasm-node@4.13.0): resolution: {integrity: sha512-IGcu+cydlUMZ5En85jxHH4qj2hta/11BHq95iHEyb2sbgiN0eCdzvUcHw5gt9pBL5lTi4JDYJ1acCoMGpTvEZg==} peerDependencies: rollup: npm:@rollup/wasm-node dependencies: - '@rollup/pluginutils': 3.1.0(@rollup/wasm-node@4.12.1) + '@rollup/pluginutils': 3.1.0(@rollup/wasm-node@4.13.0) magic-string: 0.25.9 - rollup: /@rollup/wasm-node@4.12.1 + rollup: /@rollup/wasm-node@4.13.0 dev: true - /@rollup/pluginutils@3.1.0(@rollup/wasm-node@4.12.1): + /@rollup/pluginutils@3.1.0(@rollup/wasm-node@4.13.0): resolution: {integrity: sha512-GksZ6pr6TpIjHm8h9lSQ8pi8BE9VeubNT0OMJ3B5uZJ8pz73NPiqOtCog/x2/QzM1ENChPKxMDhiQuRHsqc+lg==} engines: {node: '>= 8.0.0'} peerDependencies: @@ -2119,11 +2119,11 @@ packages: '@types/estree': 0.0.39 estree-walker: 1.0.1 picomatch: 2.3.1 - rollup: /@rollup/wasm-node@4.12.1 + rollup: /@rollup/wasm-node@4.13.0 dev: true - /@rollup/wasm-node@4.12.1: - resolution: {integrity: sha512-5j3BVQEccCzCb8fkl++IbDgAsnlsKBPz049C4C//j5s3pFKxKGlybl63QApdJKl1fNLr7HIwQEJcBImQtA3ZHg==} + /@rollup/wasm-node@4.13.0: + resolution: {integrity: sha512-oFX11wzU7RTaiW06WBtRpzIVN/oaG0I3XkevNO0brBklYnY9zpLhTfksN4b+TdBt6CfXV/KdVhdWLbb0fQIR7A==} engines: {node: '>=18.0.0', npm: '>=8.0.0'} hasBin: true dependencies: @@ -4388,8 +4388,8 @@ packages: function-bind: 1.1.2 dev: true - /hls.js@1.4.14: - resolution: {integrity: sha512-UppQjyvPVclg+6t2KY/Rv03h0+bA5u6zwqVoz4LAC/L0fgYmIaCD7ZCrwe8WI1Gv01be1XL0QFsRbSdIHV/Wbw==} + /hls.js@1.5.7: + resolution: {integrity: sha512-Hnyf7ojTBtXHeOW1/t6wCBJSiK1WpoKF9yg7juxldDx8u3iswrkPt2wbOA/1NiwU4j27DSIVoIEJRAhcdMef/A==} dev: false /hoist-non-react-statics@3.3.2: @@ -5112,7 +5112,7 @@ packages: '@babel/plugin-syntax-typescript': 7.23.3(@babel/core@7.23.6) '@babel/types': 7.23.6 kleur: 4.1.5 - rollup: /@rollup/wasm-node@4.12.1 + rollup: /@rollup/wasm-node@4.13.0 unplugin: 1.5.1 transitivePeerDependencies: - supports-color @@ -6040,7 +6040,7 @@ packages: glob: 7.2.3 dev: true - /rollup-plugin-terser@7.0.2(@rollup/wasm-node@4.12.1): + /rollup-plugin-terser@7.0.2(@rollup/wasm-node@4.13.0): resolution: {integrity: sha512-w3iIaU4OxcF52UUXiZNsNeuXIMDvFrr+ZXK6bFZ0Q60qyVfq4uLptoS4bbq3paG3x216eQllFZX7zt6TIImguQ==} deprecated: This package has been deprecated and is no longer maintained. Please use @rollup/plugin-terser peerDependencies: @@ -6048,12 +6048,12 @@ packages: dependencies: '@babel/code-frame': 7.23.5 jest-worker: 26.6.2 - rollup: /@rollup/wasm-node@4.12.1 + rollup: /@rollup/wasm-node@4.13.0 serialize-javascript: 4.0.0 terser: 5.19.3 dev: true - /rollup-plugin-visualizer@5.11.0(@rollup/wasm-node@4.12.1): + /rollup-plugin-visualizer@5.11.0(@rollup/wasm-node@4.13.0): resolution: {integrity: sha512-exM0Ms2SN3AgTzMeW7y46neZQcyLY7eKwWAop1ZoRTCZwyrIRdMMJ6JjToAJbML77X/9N8ZEpmXG4Z/Clb9k8g==} engines: {node: '>=14'} hasBin: true @@ -6065,7 +6065,7 @@ packages: dependencies: open: 8.4.2 picomatch: 2.3.1 - rollup: /@rollup/wasm-node@4.12.1 + rollup: /@rollup/wasm-node@4.13.0 source-map: 0.7.4 yargs: 17.7.2 dev: true @@ -7051,7 +7051,7 @@ packages: '@types/node': 20.10.5 esbuild: 0.19.10 postcss: 8.4.32 - rollup: /@rollup/wasm-node@4.12.1 + rollup: /@rollup/wasm-node@4.13.0 optionalDependencies: fsevents: 2.3.3 dev: true @@ -7313,9 +7313,9 @@ packages: '@babel/core': 7.23.6 '@babel/preset-env': 7.23.6(@babel/core@7.23.6) '@babel/runtime': 7.23.6 - '@rollup/plugin-babel': 5.3.1(@babel/core@7.23.6)(@rollup/wasm-node@4.12.1) - '@rollup/plugin-node-resolve': 11.2.1(@rollup/wasm-node@4.12.1) - '@rollup/plugin-replace': 2.4.2(@rollup/wasm-node@4.12.1) + '@rollup/plugin-babel': 5.3.1(@babel/core@7.23.6)(@rollup/wasm-node@4.13.0) + '@rollup/plugin-node-resolve': 11.2.1(@rollup/wasm-node@4.13.0) + '@rollup/plugin-replace': 2.4.2(@rollup/wasm-node@4.13.0) '@surma/rollup-plugin-off-main-thread': 2.2.3 ajv: 8.12.0 common-tags: 1.8.2 @@ -7324,8 +7324,8 @@ packages: glob: 7.2.3 lodash: 4.17.21 pretty-bytes: 5.6.0 - rollup: /@rollup/wasm-node@4.12.1 - rollup-plugin-terser: 7.0.2(@rollup/wasm-node@4.12.1) + rollup: /@rollup/wasm-node@4.13.0 + rollup-plugin-terser: 7.0.2(@rollup/wasm-node@4.13.0) source-map: 0.8.0-beta.0 stringify-object: 3.3.0 strip-comments: 2.0.1 @@ -7370,6 +7370,7 @@ packages: /workbox-google-analytics@7.0.0: resolution: {integrity: sha512-MEYM1JTn/qiC3DbpvP2BVhyIH+dV/5BjHk756u9VbwuAhu0QHyKscTnisQuz21lfRpOwiS9z4XdqeVAKol0bzg==} + deprecated: It is not compatible with newer versions of GA starting with v4, as long as you are using GAv3 it should be ok, but the package is not longer being maintained dependencies: workbox-background-sync: 7.0.0 workbox-core: 7.0.0 diff --git a/public/android-chrome-192x192.png b/public/android-chrome-192x192.png index f71a0514..fbf54484 100644 Binary files a/public/android-chrome-192x192.png and b/public/android-chrome-192x192.png differ diff --git a/public/android-chrome-512x512.png b/public/android-chrome-512x512.png index b44f4951..c8c1abb0 100644 Binary files a/public/android-chrome-512x512.png and b/public/android-chrome-512x512.png differ diff --git a/public/apple-touch-icon.png b/public/apple-touch-icon.png index f45b8fd4..cb590568 100644 Binary files a/public/apple-touch-icon.png and b/public/apple-touch-icon.png differ diff --git a/public/flags/skull.svg b/public/flags/skull.svg index 02c4741b..b71d596d 100644 --- a/public/flags/skull.svg +++ b/public/flags/skull.svg @@ -1 +1 @@ - + \ No newline at end of file diff --git a/public/flags/tokiPona.svg b/public/flags/tokiPona.svg index d0bea782..89d95846 100644 --- a/public/flags/tokiPona.svg +++ b/public/flags/tokiPona.svg @@ -1,76 +1 @@ - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/lightbar-images/snowflake.svg b/public/lightbar-images/snowflake.svg index 50d9c382..8b928e10 100644 --- a/public/lightbar-images/snowflake.svg +++ b/public/lightbar-images/snowflake.svg @@ -1,45 +1 @@ - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/mstile-150x150.png b/public/mstile-150x150.png index a001008d..e90933d6 100644 Binary files a/public/mstile-150x150.png and b/public/mstile-150x150.png differ diff --git a/public/safari-pinned-tab.svg b/public/safari-pinned-tab.svg index 647fc295..c49ba8e2 100644 --- a/public/safari-pinned-tab.svg +++ b/public/safari-pinned-tab.svg @@ -1,18 +1 @@ - - - - -Created by potrace 1.14, written by Peter Selinger 2001-2017 - - - - - +Created by potrace 1.14, written by Peter Selinger 2001-2017 \ No newline at end of file diff --git a/public/splash_screens/10.2__iPad_landscape.png b/public/splash_screens/10.2__iPad_landscape.png index 62d541fc..e5bd744a 100644 Binary files a/public/splash_screens/10.2__iPad_landscape.png and b/public/splash_screens/10.2__iPad_landscape.png differ diff --git a/public/splash_screens/10.2__iPad_portrait.png b/public/splash_screens/10.2__iPad_portrait.png index 97c5cea9..b2a6e242 100644 Binary files a/public/splash_screens/10.2__iPad_portrait.png and b/public/splash_screens/10.2__iPad_portrait.png differ diff --git a/public/splash_screens/10.5__iPad_Air_landscape.png b/public/splash_screens/10.5__iPad_Air_landscape.png index f8001cf3..929cef7a 100644 Binary files a/public/splash_screens/10.5__iPad_Air_landscape.png and b/public/splash_screens/10.5__iPad_Air_landscape.png differ diff --git a/public/splash_screens/10.5__iPad_Air_portrait.png b/public/splash_screens/10.5__iPad_Air_portrait.png index c6750e7a..5812a334 100644 Binary files a/public/splash_screens/10.5__iPad_Air_portrait.png and b/public/splash_screens/10.5__iPad_Air_portrait.png differ diff --git a/public/splash_screens/10.9__iPad_Air_landscape.png b/public/splash_screens/10.9__iPad_Air_landscape.png index 03a44384..b63e8ef2 100644 Binary files a/public/splash_screens/10.9__iPad_Air_landscape.png and b/public/splash_screens/10.9__iPad_Air_landscape.png differ diff --git a/public/splash_screens/10.9__iPad_Air_portrait.png b/public/splash_screens/10.9__iPad_Air_portrait.png index d7c068b2..2603d66a 100644 Binary files a/public/splash_screens/10.9__iPad_Air_portrait.png and b/public/splash_screens/10.9__iPad_Air_portrait.png differ diff --git a/public/splash_screens/11__iPad_Pro__10.5__iPad_Pro_landscape.png b/public/splash_screens/11__iPad_Pro__10.5__iPad_Pro_landscape.png index c2e02978..a4bee690 100644 Binary files a/public/splash_screens/11__iPad_Pro__10.5__iPad_Pro_landscape.png and b/public/splash_screens/11__iPad_Pro__10.5__iPad_Pro_landscape.png differ diff --git a/public/splash_screens/11__iPad_Pro__10.5__iPad_Pro_portrait.png b/public/splash_screens/11__iPad_Pro__10.5__iPad_Pro_portrait.png index 59178e3f..431c4e55 100644 Binary files a/public/splash_screens/11__iPad_Pro__10.5__iPad_Pro_portrait.png and b/public/splash_screens/11__iPad_Pro__10.5__iPad_Pro_portrait.png differ diff --git a/public/splash_screens/12.9__iPad_Pro_landscape.png b/public/splash_screens/12.9__iPad_Pro_landscape.png index db66590f..5f4e1fcd 100644 Binary files a/public/splash_screens/12.9__iPad_Pro_landscape.png and b/public/splash_screens/12.9__iPad_Pro_landscape.png differ diff --git a/public/splash_screens/12.9__iPad_Pro_portrait.png b/public/splash_screens/12.9__iPad_Pro_portrait.png index 2fc0f2a0..d850c757 100644 Binary files a/public/splash_screens/12.9__iPad_Pro_portrait.png and b/public/splash_screens/12.9__iPad_Pro_portrait.png differ diff --git a/public/splash_screens/4__iPhone_SE__iPod_touch_5th_generation_and_later_landscape.png b/public/splash_screens/4__iPhone_SE__iPod_touch_5th_generation_and_later_landscape.png index c484d5b3..c7f80d17 100644 Binary files a/public/splash_screens/4__iPhone_SE__iPod_touch_5th_generation_and_later_landscape.png and b/public/splash_screens/4__iPhone_SE__iPod_touch_5th_generation_and_later_landscape.png differ diff --git a/public/splash_screens/4__iPhone_SE__iPod_touch_5th_generation_and_later_portrait.png b/public/splash_screens/4__iPhone_SE__iPod_touch_5th_generation_and_later_portrait.png index 2b06bf2e..a462a304 100644 Binary files a/public/splash_screens/4__iPhone_SE__iPod_touch_5th_generation_and_later_portrait.png and b/public/splash_screens/4__iPhone_SE__iPod_touch_5th_generation_and_later_portrait.png differ diff --git a/public/splash_screens/8.3__iPad_Mini_landscape.png b/public/splash_screens/8.3__iPad_Mini_landscape.png index d43bf7c1..5a9a9b56 100644 Binary files a/public/splash_screens/8.3__iPad_Mini_landscape.png and b/public/splash_screens/8.3__iPad_Mini_landscape.png differ diff --git a/public/splash_screens/8.3__iPad_Mini_portrait.png b/public/splash_screens/8.3__iPad_Mini_portrait.png index 83eace82..18e1d650 100644 Binary files a/public/splash_screens/8.3__iPad_Mini_portrait.png and b/public/splash_screens/8.3__iPad_Mini_portrait.png differ diff --git a/public/splash_screens/9.7__iPad_Pro__7.9__iPad_mini__9.7__iPad_Air__9.7__iPad_landscape.png b/public/splash_screens/9.7__iPad_Pro__7.9__iPad_mini__9.7__iPad_Air__9.7__iPad_landscape.png index 64a09053..74a1b83a 100644 Binary files a/public/splash_screens/9.7__iPad_Pro__7.9__iPad_mini__9.7__iPad_Air__9.7__iPad_landscape.png and b/public/splash_screens/9.7__iPad_Pro__7.9__iPad_mini__9.7__iPad_Air__9.7__iPad_landscape.png differ diff --git a/public/splash_screens/9.7__iPad_Pro__7.9__iPad_mini__9.7__iPad_Air__9.7__iPad_portrait.png b/public/splash_screens/9.7__iPad_Pro__7.9__iPad_mini__9.7__iPad_Air__9.7__iPad_portrait.png index 029e83c4..dfaa2111 100644 Binary files a/public/splash_screens/9.7__iPad_Pro__7.9__iPad_mini__9.7__iPad_Air__9.7__iPad_portrait.png and b/public/splash_screens/9.7__iPad_Pro__7.9__iPad_mini__9.7__iPad_Air__9.7__iPad_portrait.png differ diff --git a/public/splash_screens/iPhone_11_Pro_Max__iPhone_XS_Max_landscape.png b/public/splash_screens/iPhone_11_Pro_Max__iPhone_XS_Max_landscape.png index ed0ce416..2306b971 100644 Binary files a/public/splash_screens/iPhone_11_Pro_Max__iPhone_XS_Max_landscape.png and b/public/splash_screens/iPhone_11_Pro_Max__iPhone_XS_Max_landscape.png differ diff --git a/public/splash_screens/iPhone_11_Pro_Max__iPhone_XS_Max_portrait.png b/public/splash_screens/iPhone_11_Pro_Max__iPhone_XS_Max_portrait.png index da53f3e4..c409b3a4 100644 Binary files a/public/splash_screens/iPhone_11_Pro_Max__iPhone_XS_Max_portrait.png and b/public/splash_screens/iPhone_11_Pro_Max__iPhone_XS_Max_portrait.png differ diff --git a/public/splash_screens/iPhone_11__iPhone_XR_landscape.png b/public/splash_screens/iPhone_11__iPhone_XR_landscape.png index 386e0920..11a50873 100644 Binary files a/public/splash_screens/iPhone_11__iPhone_XR_landscape.png and b/public/splash_screens/iPhone_11__iPhone_XR_landscape.png differ diff --git a/public/splash_screens/iPhone_11__iPhone_XR_portrait.png b/public/splash_screens/iPhone_11__iPhone_XR_portrait.png index 590a317e..1922ce36 100644 Binary files a/public/splash_screens/iPhone_11__iPhone_XR_portrait.png and b/public/splash_screens/iPhone_11__iPhone_XR_portrait.png differ diff --git a/public/splash_screens/iPhone_13_mini__iPhone_12_mini__iPhone_11_Pro__iPhone_XS__iPhone_X_landscape.png b/public/splash_screens/iPhone_13_mini__iPhone_12_mini__iPhone_11_Pro__iPhone_XS__iPhone_X_landscape.png index 7bd624b3..c5bcab45 100644 Binary files a/public/splash_screens/iPhone_13_mini__iPhone_12_mini__iPhone_11_Pro__iPhone_XS__iPhone_X_landscape.png and b/public/splash_screens/iPhone_13_mini__iPhone_12_mini__iPhone_11_Pro__iPhone_XS__iPhone_X_landscape.png differ diff --git a/public/splash_screens/iPhone_13_mini__iPhone_12_mini__iPhone_11_Pro__iPhone_XS__iPhone_X_portrait.png b/public/splash_screens/iPhone_13_mini__iPhone_12_mini__iPhone_11_Pro__iPhone_XS__iPhone_X_portrait.png index a70fae80..64cbbeee 100644 Binary files a/public/splash_screens/iPhone_13_mini__iPhone_12_mini__iPhone_11_Pro__iPhone_XS__iPhone_X_portrait.png and b/public/splash_screens/iPhone_13_mini__iPhone_12_mini__iPhone_11_Pro__iPhone_XS__iPhone_X_portrait.png differ diff --git a/public/splash_screens/iPhone_14_Plus__iPhone_13_Pro_Max__iPhone_12_Pro_Max_landscape.png b/public/splash_screens/iPhone_14_Plus__iPhone_13_Pro_Max__iPhone_12_Pro_Max_landscape.png index 1b06793b..fe273cbe 100644 Binary files a/public/splash_screens/iPhone_14_Plus__iPhone_13_Pro_Max__iPhone_12_Pro_Max_landscape.png and b/public/splash_screens/iPhone_14_Plus__iPhone_13_Pro_Max__iPhone_12_Pro_Max_landscape.png differ diff --git a/public/splash_screens/iPhone_14_Plus__iPhone_13_Pro_Max__iPhone_12_Pro_Max_portrait.png b/public/splash_screens/iPhone_14_Plus__iPhone_13_Pro_Max__iPhone_12_Pro_Max_portrait.png index 88a9afdd..d37f3f80 100644 Binary files a/public/splash_screens/iPhone_14_Plus__iPhone_13_Pro_Max__iPhone_12_Pro_Max_portrait.png and b/public/splash_screens/iPhone_14_Plus__iPhone_13_Pro_Max__iPhone_12_Pro_Max_portrait.png differ diff --git a/public/splash_screens/iPhone_14__iPhone_13_Pro__iPhone_13__iPhone_12_Pro__iPhone_12_landscape.png b/public/splash_screens/iPhone_14__iPhone_13_Pro__iPhone_13__iPhone_12_Pro__iPhone_12_landscape.png index d40426eb..f1b91ac0 100644 Binary files a/public/splash_screens/iPhone_14__iPhone_13_Pro__iPhone_13__iPhone_12_Pro__iPhone_12_landscape.png and b/public/splash_screens/iPhone_14__iPhone_13_Pro__iPhone_13__iPhone_12_Pro__iPhone_12_landscape.png differ diff --git a/public/splash_screens/iPhone_14__iPhone_13_Pro__iPhone_13__iPhone_12_Pro__iPhone_12_portrait.png b/public/splash_screens/iPhone_14__iPhone_13_Pro__iPhone_13__iPhone_12_Pro__iPhone_12_portrait.png index 0496f4c7..c05ea1bd 100644 Binary files a/public/splash_screens/iPhone_14__iPhone_13_Pro__iPhone_13__iPhone_12_Pro__iPhone_12_portrait.png and b/public/splash_screens/iPhone_14__iPhone_13_Pro__iPhone_13__iPhone_12_Pro__iPhone_12_portrait.png differ diff --git a/public/splash_screens/iPhone_15_Pro_Max__iPhone_15_Plus__iPhone_14_Pro_Max_landscape.png b/public/splash_screens/iPhone_15_Pro_Max__iPhone_15_Plus__iPhone_14_Pro_Max_landscape.png index 582988cb..f92c7c7f 100644 Binary files a/public/splash_screens/iPhone_15_Pro_Max__iPhone_15_Plus__iPhone_14_Pro_Max_landscape.png and b/public/splash_screens/iPhone_15_Pro_Max__iPhone_15_Plus__iPhone_14_Pro_Max_landscape.png differ diff --git a/public/splash_screens/iPhone_15_Pro_Max__iPhone_15_Plus__iPhone_14_Pro_Max_portrait.png b/public/splash_screens/iPhone_15_Pro_Max__iPhone_15_Plus__iPhone_14_Pro_Max_portrait.png index 4101bc0e..2ef4a328 100644 Binary files a/public/splash_screens/iPhone_15_Pro_Max__iPhone_15_Plus__iPhone_14_Pro_Max_portrait.png and b/public/splash_screens/iPhone_15_Pro_Max__iPhone_15_Plus__iPhone_14_Pro_Max_portrait.png differ diff --git a/public/splash_screens/iPhone_15_Pro__iPhone_15__iPhone_14_Pro_landscape.png b/public/splash_screens/iPhone_15_Pro__iPhone_15__iPhone_14_Pro_landscape.png index a88d6e00..1414d774 100644 Binary files a/public/splash_screens/iPhone_15_Pro__iPhone_15__iPhone_14_Pro_landscape.png and b/public/splash_screens/iPhone_15_Pro__iPhone_15__iPhone_14_Pro_landscape.png differ diff --git a/public/splash_screens/iPhone_15_Pro__iPhone_15__iPhone_14_Pro_portrait.png b/public/splash_screens/iPhone_15_Pro__iPhone_15__iPhone_14_Pro_portrait.png index 2c56a023..b6d5f126 100644 Binary files a/public/splash_screens/iPhone_15_Pro__iPhone_15__iPhone_14_Pro_portrait.png and b/public/splash_screens/iPhone_15_Pro__iPhone_15__iPhone_14_Pro_portrait.png differ diff --git a/public/splash_screens/iPhone_8_Plus__iPhone_7_Plus__iPhone_6s_Plus__iPhone_6_Plus_landscape.png b/public/splash_screens/iPhone_8_Plus__iPhone_7_Plus__iPhone_6s_Plus__iPhone_6_Plus_landscape.png index e1646b9c..a3d35746 100644 Binary files a/public/splash_screens/iPhone_8_Plus__iPhone_7_Plus__iPhone_6s_Plus__iPhone_6_Plus_landscape.png and b/public/splash_screens/iPhone_8_Plus__iPhone_7_Plus__iPhone_6s_Plus__iPhone_6_Plus_landscape.png differ diff --git a/public/splash_screens/iPhone_8_Plus__iPhone_7_Plus__iPhone_6s_Plus__iPhone_6_Plus_portrait.png b/public/splash_screens/iPhone_8_Plus__iPhone_7_Plus__iPhone_6s_Plus__iPhone_6_Plus_portrait.png index 2e1f04b5..44d9d9ac 100644 Binary files a/public/splash_screens/iPhone_8_Plus__iPhone_7_Plus__iPhone_6s_Plus__iPhone_6_Plus_portrait.png and b/public/splash_screens/iPhone_8_Plus__iPhone_7_Plus__iPhone_6s_Plus__iPhone_6_Plus_portrait.png differ diff --git a/public/splash_screens/iPhone_8__iPhone_7__iPhone_6s__iPhone_6__4.7__iPhone_SE_landscape.png b/public/splash_screens/iPhone_8__iPhone_7__iPhone_6s__iPhone_6__4.7__iPhone_SE_landscape.png index c6bd57a4..fba250c6 100644 Binary files a/public/splash_screens/iPhone_8__iPhone_7__iPhone_6s__iPhone_6__4.7__iPhone_SE_landscape.png and b/public/splash_screens/iPhone_8__iPhone_7__iPhone_6s__iPhone_6__4.7__iPhone_SE_landscape.png differ diff --git a/public/splash_screens/iPhone_8__iPhone_7__iPhone_6s__iPhone_6__4.7__iPhone_SE_portrait.png b/public/splash_screens/iPhone_8__iPhone_7__iPhone_6s__iPhone_6__4.7__iPhone_SE_portrait.png index 24788f1a..06b99aca 100644 Binary files a/public/splash_screens/iPhone_8__iPhone_7__iPhone_6s__iPhone_6__4.7__iPhone_SE_portrait.png and b/public/splash_screens/iPhone_8__iPhone_7__iPhone_6s__iPhone_6__4.7__iPhone_SE_portrait.png differ diff --git a/public/splash_screens/icon.png b/public/splash_screens/icon.png index a43a415d..7ff85ce0 100644 Binary files a/public/splash_screens/icon.png and b/public/splash_screens/icon.png differ diff --git a/src/assets/languages.ts b/src/assets/languages.ts index e575d454..8077a5b1 100644 --- a/src/assets/languages.ts +++ b/src/assets/languages.ts @@ -82,7 +82,7 @@ export const locales = { ko, sl, ta, - "zh-HANT": zhhant, + "zh-Hant": zhhant, is, ru, gl, diff --git a/src/assets/locales/pl.json b/src/assets/locales/pl.json index 843e2201..c04669dc 100644 --- a/src/assets/locales/pl.json +++ b/src/assets/locales/pl.json @@ -117,8 +117,7 @@ "loading": "Wczytywanie...", "noResults": "Nie mogliśmy niczego znaleźć!", "placeholder": { - "default": "Co chciałbyś obejrzeć?", - "extra": [] + "default": "Co chciałbyś obejrzeć?" }, "sectionTitle": "Wyniki wyszukiwania" }, @@ -131,11 +130,15 @@ }, "morning": { "default": "Co chciałbyś obejrzeć dziś rano?", - "extra": ["Słyszałem że „Przed wschodem słońca” jest dobre"] + "extra": [ + "Słyszałem że „Przed wschodem słońca” jest dobre" + ] }, "night": { "default": "Co chciałbyś obejrzeć dziś wieczorem?", - "extra": ["Zmęczony? Słyszałem że „Egzorcysta” jest dobry."] + "extra": [ + "Zmęczony? Słyszałem że „Egzorcysta” jest dobry." + ] } } }, @@ -176,7 +179,7 @@ "back": "Wstecz", "explainer": "Korzystając z rozszerzenia przeglądarki, możesz uzyskać najlepsze strumienie. Wystarczy prosta instalacja.", "explainerIos": "Niestety, rozszerzenie przeglądarki nie jest obsługiwane w systemie iOS, naciśnij Wstecz, aby wybrać inną opcję.", - "extensionHelp": "Jeżeli zainstalowałeś rozszerzenie, ale nie zostało ono wykryte. Otwórz rozszerzenie za pomocą menu rozszerzeń przeglądarki i postępuj zgodnie z instrukcjami wyświetlanymi na ekranie.", + "extensionHelp": "Jeżeli zainstalowałeś rozszerzenie, ale nie zostało ono wykryte, otwórz rozszerzenie za pomocą menu rozszerzeń przeglądarki i postępuj zgodnie z instrukcjami wyświetlanymi na ekranie.", "linkChrome": "Zainstaluj rozszerzenie na Chrome", "linkFirefox": "Zainstaluj rozszerzenie na Firefox", "notDetecting": "Zainstalowano na Chrome, ale się nie wyświetla? Spróbuj odświeżyć stronę!", @@ -207,7 +210,7 @@ "title": "Stwórzmy nowe proxy" }, "start": { - "explainer": "Aby uzyskać najlepsze transmisje strumieniowe. Będziesz musiał wybrać metodę strumieniowania, której chcesz użyć.", + "explainer": "Aby uzyskać najlepsze transmisje strumieniowe, będziesz musiał wybrać metodę strumieniowania której chcesz użyć.", "options": { "default": { "text": "Nie chcę dobrej jakości strumieni, <0 /> <1>użyj domyślnej konfiguracji" @@ -524,8 +527,8 @@ } }, "subtitles": { - "backgroundLabel": "Krycie tła", "backgroundBlurLabel": "Rozmycie tła", + "backgroundLabel": "Krycie tła", "colorLabel": "Kolor", "previewQuote": "Nie wolno mi się bać. Strach zabija myślenie.", "textSizeLabel": "Rozmiar czcionki", diff --git a/src/assets/locales/tok.json b/src/assets/locales/tok.json index 7856144e..3e6dfbef 100644 --- a/src/assets/locales/tok.json +++ b/src/assets/locales/tok.json @@ -57,6 +57,8 @@ }, "host": "lawa ilo sina li <0>{{hostname}} - ona li pona tawa sina la sina ken pali e lipu open", "no": "o weka", + "noHost": "lawa ilo ni li open ala li nasin ala la, sina ken ala pali e lipu open", + "noHostTitle": "lawa ilo li open ala a!", "title": "lawa ilo ni li pona tawa sina anu seme?", "yes": "lawa ilo ni li pona" }, @@ -79,7 +81,8 @@ }, "footer": { "legal": { - "disclaimer": "o sona e ni:" + "disclaimer": "o sona e ni:", + "disclaimerText": "ilo Muwi-We li mama ala e ijo sitelen. ona li toki taso tawa ilo ante. utala nasin li lon la o toki tawa ona pi ilo ante. sitelen ale li tan ala ilo Muwi-We" }, "links": { "discord": "kulupu Siko", @@ -117,22 +120,33 @@ "noResults": "ijo li lon ala a!", "placeholder": { "default": "sina wile lukin e seme?", - "extra": [] + "extra": [ + "sina wile alasa e seme?", + "sina wile lukin e seme?", + "sitelen nanpa wan sina li seme?", + "sitelen nanpa wan sina li seme?" + ] }, "sectionTitle": "mi lukin e ni:" }, "titles": { "day": { "default": "tenpo suno ni la sina wile lukin e seme?", - "extra": ["sina pilin alasa la o lukin e sitelen Jurassic Park"] + "extra": [ + "sina pilin alasa la o lukin e sitelen Jurassic Park" + ] }, "morning": { "default": "tenpo sin ni la sina wile lukin e seme?", - "extra": ["ken la sitelen Before Sunrise li pona"] + "extra": [ + "ken la sitelen Before Sunrise li pona" + ] }, "night": { "default": "tenpo pimeja ni la sina wile lukin e seme?", - "extra": ["sina pilin lape anu seme? o alasa lukin e sitelen Exorcist"] + "extra": [ + "sina pilin lape anu seme? o alasa lukin e sitelen Exorcist" + ] } } }, @@ -163,6 +177,9 @@ "title": "mi ken ala lukin e lipu ona" }, "onboarding": { + "defaultConfirm": { + "cancel": "ala" + }, "start": { "title": "o open e ilo Muwi-We" } diff --git a/src/backend/helpers/subs.ts b/src/backend/helpers/subs.ts index 092de6a9..01bf4f7e 100644 --- a/src/backend/helpers/subs.ts +++ b/src/backend/helpers/subs.ts @@ -51,3 +51,15 @@ export async function downloadCaption( downloadCache.set(caption.url, output, expirySeconds); return output; } + +/** + * Downloads the WebVTT content. No different than a simple + * get request with a cache. + */ +export async function downloadWebVTT(url: string): Promise { + const cached = downloadCache.get(url); + if (cached) return cached; + + const data = await fetch(url).then((v) => v.text()); + return data; +} diff --git a/src/components/player/atoms/settings/CaptionsView.tsx b/src/components/player/atoms/settings/CaptionsView.tsx index 92104f1b..8524ecc8 100644 --- a/src/components/player/atoms/settings/CaptionsView.tsx +++ b/src/components/player/atoms/settings/CaptionsView.tsx @@ -122,9 +122,16 @@ export function CaptionsView({ id }: { id: string }) { >(null); const { selectCaptionById, disable } = useCaptions(); const captionList = usePlayerStore((s) => s.captionList); + const getHlsCaptionList = usePlayerStore((s) => s.display?.getCaptionList); + + const captions = useMemo( + () => + captionList.length !== 0 ? captionList : getHlsCaptionList?.() ?? [], + [captionList, getHlsCaptionList], + ); const [searchQuery, setSearchQuery] = useState(""); - const subtitleList = useSubtitleList(captionList, searchQuery); + const subtitleList = useSubtitleList(captions, searchQuery); const [downloadReq, startDownload] = useAsyncFn( async (captionId: string) => { diff --git a/src/components/player/display/base.ts b/src/components/player/display/base.ts index a12661ab..51e6d7bb 100644 --- a/src/components/player/display/base.ts +++ b/src/components/player/display/base.ts @@ -67,6 +67,11 @@ export function makeVideoElementDisplayInterface(): DisplayInterface { let preferenceQuality: SourceQuality | null = null; let lastVolume = 1; + const languagePromises = new Map< + string, + (value: void | PromiseLike) => void + >(); + function reportLevels() { if (!hls) return; const levels = hls.levels; @@ -133,6 +138,7 @@ export function makeVideoElementDisplayInterface(): DisplayInterface { }, }, }, + renderTextTracksNatively: false, }); hls.on(Hls.Events.ERROR, (event, data) => { console.error("HLS error", data); @@ -173,6 +179,16 @@ export function makeVideoElementDisplayInterface(): DisplayInterface { const quality = hlsLevelToQuality(hls.levels[hls.currentLevel]); emit("changedquality", quality); }); + hls.on(Hls.Events.SUBTITLE_TRACK_LOADED, () => { + for (const [lang, resolve] of languagePromises) { + const track = hls?.subtitleTracks.find((t) => t.lang === lang); + if (track) { + resolve(); + languagePromises.delete(lang); + break; + } + } + }); } hls.attachMedia(vid); @@ -413,5 +429,40 @@ export function makeVideoElementDisplayInterface(): DisplayInterface { setPlaybackRate(rate) { if (videoElement) videoElement.playbackRate = rate; }, + getCaptionList() { + return ( + hls?.subtitleTracks.map((track) => { + return { + id: track.id.toString(), + language: track.lang ?? "unknown", + url: track.url, + needsProxy: false, + hls: true, + }; + }) ?? [] + ); + }, + getSubtitleTracks() { + return hls?.subtitleTracks ?? []; + }, + async setSubtitlePreference(lang) { + // default subtitles are already loaded by hls.js + const track = hls?.subtitleTracks.find((t) => t.lang === lang); + if (track?.details !== undefined) return Promise.resolve(); + + // need to wait a moment before hls loads the subtitles + const promise = new Promise((resolve, reject) => { + languagePromises.set(lang, resolve); + + // reject after some time, if hls.js fails to load the subtitles + // for any reason + setTimeout(() => { + reject(); + languagePromises.delete(lang); + }, 5000); + }); + hls?.setSubtitleOption({ lang }); + return promise; + }, }; } diff --git a/src/components/player/display/chromecast.ts b/src/components/player/display/chromecast.ts index dbaab0df..1a318f16 100644 --- a/src/components/player/display/chromecast.ts +++ b/src/components/player/display/chromecast.ts @@ -274,5 +274,14 @@ export function makeChromecastDisplayInterface( playbackRate = rate; setSource(); }, + getCaptionList() { + return []; + }, + getSubtitleTracks() { + return []; + }, + async setSubtitlePreference() { + return Promise.resolve(); + }, }; } diff --git a/src/components/player/display/displayInterface.ts b/src/components/player/display/displayInterface.ts index 8ba8480a..134bef44 100644 --- a/src/components/player/display/displayInterface.ts +++ b/src/components/player/display/displayInterface.ts @@ -1,4 +1,7 @@ +import { MediaPlaylist } from "hls.js"; + import { MWMediaType } from "@/backend/metadata/types/mw"; +import { CaptionListItem } from "@/stores/player/slices/source"; import { LoadableSource, SourceQuality } from "@/stores/player/utils/qualities"; import { Listener } from "@/utils/events"; @@ -70,4 +73,7 @@ export interface DisplayInterface extends Listener { setMeta(meta: DisplayMeta): void; setCaption(caption: DisplayCaption | null): void; getType(): DisplayType; + getCaptionList(): CaptionListItem[]; + getSubtitleTracks(): MediaPlaylist[]; + setSubtitlePreference(lang: string): Promise; } diff --git a/src/components/player/hooks/useCaptions.ts b/src/components/player/hooks/useCaptions.ts index 458c704a..4bbdae4f 100644 --- a/src/components/player/hooks/useCaptions.ts +++ b/src/components/player/hooks/useCaptions.ts @@ -1,9 +1,16 @@ -import { useCallback } from "react"; +import { useCallback, useMemo } from "react"; +import subsrt from "subsrt-ts"; -import { downloadCaption } from "@/backend/helpers/subs"; +import { downloadCaption, downloadWebVTT } from "@/backend/helpers/subs"; +import { Caption } from "@/stores/player/slices/source"; import { usePlayerStore } from "@/stores/player/store"; import { useSubtitleStore } from "@/stores/subtitles"; +import { + filterDuplicateCaptionCues, + parseVttSubtitles, +} from "../utils/captions"; + export function useCaptions() { const setLanguage = useSubtitleStore((s) => s.setLanguage); const enabled = useSubtitleStore((s) => s.enabled); @@ -12,32 +19,85 @@ export function useCaptions() { ); const setCaption = usePlayerStore((s) => s.setCaption); const lastSelectedLanguage = useSubtitleStore((s) => s.lastSelectedLanguage); + const captionList = usePlayerStore((s) => s.captionList); + const getHlsCaptionList = usePlayerStore((s) => s.display?.getCaptionList); + + const getSubtitleTracks = usePlayerStore((s) => s.display?.getSubtitleTracks); + const setSubtitlePreference = usePlayerStore( + (s) => s.display?.setSubtitlePreference, + ); + + const captions = useMemo( + () => + captionList.length !== 0 ? captionList : getHlsCaptionList?.() ?? [], + [captionList, getHlsCaptionList], + ); const selectCaptionById = useCallback( async (captionId: string) => { - const caption = captionList.find((v) => v.id === captionId); + const caption = captions.find((v) => v.id === captionId); if (!caption) return; - const srtData = await downloadCaption(caption); - setCaption({ + + const captionToSet: Caption = { id: caption.id, language: caption.language, - srtData, url: caption.url, - }); + srtData: "", + }; + + if (!caption.hls) { + const srtData = await downloadCaption(caption); + captionToSet.srtData = srtData; + } else { + // request a language change to hls, so it can load the subtitles + await setSubtitlePreference?.(caption.language); + const track = getSubtitleTracks?.().find( + (t) => t.id.toString() === caption.id && t.details !== undefined, + ); + if (!track) return; + + const fragments = + track.details?.fragments?.filter( + (frag) => frag !== null && frag.url !== null, + ) ?? []; + + const vttCaptions = ( + await Promise.all( + fragments.map(async (frag) => { + const vtt = await downloadWebVTT(frag.url); + return parseVttSubtitles(vtt); + }), + ) + ).flat(); + + const filtered = filterDuplicateCaptionCues(vttCaptions); + + const srtData = subsrt.build(filtered, { format: "srt" }); + captionToSet.srtData = srtData; + } + + setCaption(captionToSet); resetSubtitleSpecificSettings(); setLanguage(caption.language); }, - [setLanguage, captionList, setCaption, resetSubtitleSpecificSettings], + [ + setLanguage, + captions, + setCaption, + resetSubtitleSpecificSettings, + getSubtitleTracks, + setSubtitlePreference, + ], ); const selectLanguage = useCallback( async (language: string) => { - const caption = captionList.find((v) => v.language === language); + const caption = captions.find((v) => v.language === language); if (!caption) return; return selectCaptionById(caption.id); }, - [captionList, selectCaptionById], + [captions, selectCaptionById], ); const disable = useCallback(async () => { diff --git a/src/components/player/internals/MediaSession.tsx b/src/components/player/internals/MediaSession.tsx index 1dc7e737..73594793 100644 --- a/src/components/player/internals/MediaSession.tsx +++ b/src/components/player/internals/MediaSession.tsx @@ -32,6 +32,9 @@ export function MediaSession() { const updatePositionState = useCallback( (position: number) => { + // If the browser doesn't support setPositionState, return + if (typeof navigator.mediaSession.setPositionState !== "function") return; + // If the updated position needs to be buffered, queue an update if (position > data.progress.buffered) { shouldUpdatePositionState.current = true; diff --git a/src/components/player/utils/captions.ts b/src/components/player/utils/captions.ts index bc2079db..df64fa5a 100644 --- a/src/components/player/utils/captions.ts +++ b/src/components/player/utils/captions.ts @@ -50,12 +50,30 @@ export function convertSubtitlesToSrt(text: string): string { return srt; } +export function filterDuplicateCaptionCues(cues: ContentCaption[]) { + return cues.reduce((acc: ContentCaption[], cap: ContentCaption) => { + const lastCap = acc[acc.length - 1]; + const isSameAsLast = + lastCap?.start === cap.start && + lastCap?.end === cap.end && + lastCap?.content === cap.content; + if (lastCap === undefined || !isSameAsLast) { + acc.push(cap); + } + return acc; + }, []); +} + +export function parseVttSubtitles(vtt: string) { + return parse(vtt).filter((cue) => cue.type === "caption") as CaptionCueType[]; +} + export function parseSubtitles( text: string, _language?: string, ): CaptionCueType[] { const vtt = convertSubtitlesToVtt(text); - return parse(vtt).filter((cue) => cue.type === "caption") as CaptionCueType[]; + return parseVttSubtitles(vtt); } function stringToBase64(input: string): string { diff --git a/src/stores/player/slices/source.ts b/src/stores/player/slices/source.ts index 0312b8b0..5d04ef49 100644 --- a/src/stores/player/slices/source.ts +++ b/src/stores/player/slices/source.ts @@ -53,6 +53,7 @@ export interface CaptionListItem { language: string; url: string; needsProxy: boolean; + hls?: boolean; } export interface SourceSlice { diff --git a/vercel.json b/vercel.json index 00e7eccd..6ed1f7c3 100644 --- a/vercel.json +++ b/vercel.json @@ -1,3 +1,49 @@ { - "routes": [{ "src": "/[^.]+", "dest": "/", "status": 200 }] + "rewrites": [ + { + "source": "/(.*)", + "destination": "/" + } + ], + "headers": [ + { + "source": "/(.*)", + "headers": [ + { + "key": "X-Content-Type-Options", + "value": "nosniff" + }, + { + "key": "X-Frame-Options", + "value": "DENY" + }, + { + "key": "X-XSS-Protection", + "value": "1; mode=block" + }, + { + "key": "Cache-Control", + "value": "public, max-age=0, s-maxage=0, must-revalidate" + } + ] + }, + { + "source": "/manifest.webmanifest", + "headers": [ + { + "key": "Content-Type", + "value": "application/manifest+json" + } + ] + }, + { + "source": "/assets/(.*)", + "headers": [ + { + "key": "Cache-Control", + "value": "public, max-age=31536000, s-maxage=31536000, immutable" + } + ] + } + ] }