mirror of
				https://github.com/Mastermindzh/tidal-hifi.git
				synced 2025-10-26 06:18:28 +01:00 
			
		
		
		
	Compare commits
	
		
			253 Commits
		
	
	
		
			5.15.0
			...
			renovate/e
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 39f113ec95 | ||
| 3c782c460c | |||
|  | feb9e2cb53 | ||
| be2c00f3b4 | |||
|  | f992ca3ed0 | ||
| c89fe5e573 | |||
|  | 255b59c87c | ||
| 2bbca7df24 | |||
| eecae61405 | |||
|  | 15dbadcc73 | ||
|  | 14b92cc5de | ||
| 959fc6631f | |||
| 91f70752c8 | |||
|  | 4f0dcee359 | ||
|  | a4a963c17b | ||
| 2ecacff0e1 | |||
|  | b426f0857e | ||
| 36c9887552 | |||
|  | 9efdbb8234 | ||
| 4d5b6a35e2 | |||
|  | 7923303e18 | ||
| cc95089201 | |||
| 9b010ea2bf | |||
| 3828591177 | |||
|  | 1e471df549 | ||
|  | 96acbd3559 | ||
|  | 224689b15b | ||
| 5f1fc8731a | |||
| 02ad1289c5 | |||
| b684d290bd | |||
|  | f35fd542b2 | ||
|  | 6831e55681 | ||
| 69ad060485 | |||
| 985831e961 | |||
|  | 218f7652cd | ||
|  | e46b34e076 | ||
|  | 5f49d0d3a9 | ||
| c1850436e1 | |||
|  | 7efbd52553 | ||
| 5ea91bbfde | |||
| 76b8adb038 | |||
| a746e53f7a | |||
|  | 149c0c975f | ||
|  | 98339444f7 | ||
|  | fd5bc505da | ||
| 5f94cfbf63 | |||
| 466c39134f | |||
| da8baebf29 | |||
|  | aa6a8b3417 | ||
| 4a1152175c | |||
|  | 721c3f5047 | ||
|  | d4b0552c14 | ||
|  | 44c8c01b8b | ||
| 8d4dd1ec9d | |||
| d012ecb04e | |||
|  | fa4b4da9a5 | ||
|  | 8f1a5dc344 | ||
| 604f34b31f | |||
| 3ff37f78e8 | |||
| 8eaca2221f | |||
| 8e9e9fed4f | |||
|  | d839dba58d | ||
| bbbd919655 | |||
| d4104b21e4 | |||
| 6a23873d3d | |||
| 8d6f08a7bb | |||
| 521b8b2170 | |||
|  | 99e4cb7f48 | ||
|  | 4cd130ee0d | ||
|  | a2a2c800b6 | ||
|  | 83210f9e1d | ||
| 5d40d2d736 | |||
| 405c2d7c9f | |||
|  | 9f860d2eec | ||
|  | 86487863f8 | ||
| 3312211df2 | |||
|  | a84bb9efde | ||
| 31c37b6c7f | |||
| be9724f0cb | |||
|  | 7277482c06 | ||
|  | 323b591f4f | ||
| c63b46ee06 | |||
|  | b4a8e155af | ||
| 2702d99aca | |||
|  | 7cf9ae5c36 | ||
| 3a1402b47a | |||
|  | 4366caa16a | ||
|  | 9389b4195e | ||
| 87d8bc07ad | |||
|  | 316bf054f5 | ||
| 0d4975ce62 | |||
|  | 912f064fed | ||
| bff95bcf10 | |||
| b59dce9c9b | |||
|  | f71fb60d38 | ||
|  | fab566e9da | ||
| d2e93e7062 | |||
|  | 52a78a6a01 | ||
| f878371fab | |||
| e81aaf1384 | |||
|  | 98b4068b62 | ||
|  | b7c1cee64d | ||
| 350b4434be | |||
| 702a16357f | |||
| c6619be068 | |||
| bf7cd70f20 | |||
| 03cb14e31e | |||
| 697298da38 | |||
| 5f5b3b11a8 | |||
| d474b7b78f | |||
| ef374ea283 | |||
|  | 96ff46737b | ||
|  | dee0044876 | ||
| 3b1f456d4c | |||
|  | f380839631 | ||
| fca5194873 | |||
|  | 92d1fc159d | ||
| b632c287b0 | |||
| 5a98b13e70 | |||
| 3543253f3a | |||
|  | fff8399c9b | ||
|  | 78b9b32dbd | ||
|  | c773e59432 | ||
| 535de65e17 | |||
| b80599eab2 | |||
| c161ef8f60 | |||
| 7ed160f266 | |||
| b23975227f | |||
| 314e69dd4b | |||
|  | 10251e2f30 | ||
|  | cf06655008 | ||
|  | d5804ed4da | ||
|  | 76500ca6b8 | ||
|  | f5c80e97c5 | ||
|  | 154698c084 | ||
| 9837e77768 | |||
| 340bd82fe6 | |||
| b765ab9f48 | |||
| 6562549897 | |||
| 9089bc1d4e | |||
| 5a3609676e | |||
| 76c3082d5c | |||
| 7f7a8b7f5b | |||
|  | 2d49504818 | ||
|  | 5f4f0c67d5 | ||
|  | 3a5f1f155a | ||
|  | 3c9328fb92 | ||
|  | f262c54be2 | ||
|  | cb6e3e8b18 | ||
|  | 7f9b8dcee4 | ||
|  | 99740f4335 | ||
| 985bc0f1d5 | |||
| 1f9b3a297a | |||
| 98ca73d17d | |||
| 396ed223d3 | |||
|  | b0458f4de5 | ||
|  | b875d091ec | ||
|  | 1a0b69d042 | ||
|  | e23a71a1a6 | ||
| 510ffa1509 | |||
|  | e17ef32cde | ||
| d4c3999d9c | |||
| b0695600b1 | |||
|  | 22ecc7ade7 | ||
| a5102ebd03 | |||
|  | d01f08508f | ||
| e87d8f6922 | |||
|  | c044476014 | ||
| 75df43864b | |||
|  | 174d0a38a1 | ||
|  | cf1aeefe67 | ||
| 280d927a03 | |||
|  | f62294ef20 | ||
| 237e354f80 | |||
|  | 3a5172cb23 | ||
|  | f8f12cca62 | ||
| 79bbab6a82 | |||
| 7376df7f30 | |||
| 7f8760c4e9 | |||
|  | 1cf5a01ad7 | ||
| 8fc49f500e | |||
|  | 49f5a01b58 | ||
| 48f4fe47ef | |||
| a819e1eb45 | |||
| 57e6342b5f | |||
| 58a543a3c8 | |||
| 1c915d99fe | |||
|  | 8994360415 | ||
| a5c0d9e6e8 | |||
| f8eb36f4c7 | |||
| 5ca90c25d3 | |||
| cf86969cfc | |||
| afd05ae88b | |||
| 72c25dfdc1 | |||
|  | 25b30edf31 | ||
| 35ed4807e3 | |||
| deff9524a8 | |||
| 34c7777eeb | |||
| 048e949f30 | |||
|  | 4a976bc58c | ||
| ff4c51234b | |||
|  | c4ee6b51b9 | ||
| 45fe336598 | |||
|  | fe9f50aaf5 | ||
| aaf7a1b662 | |||
| 755be0ee30 | |||
|  | 9d736b2bd9 | ||
| f4d4b1a1df | |||
|  | 0c27c815f5 | ||
|  | a3aa45a96b | ||
| ae699887b2 | |||
| a8c635932f | |||
| 66d0d004bf | |||
| d1e0321058 | |||
| 2ab5a556ab | |||
| baf719fc60 | |||
| 15c8f6a418 | |||
| f73521e2e5 | |||
|  | 0f5e00c4df | ||
|  | 974877ea4f | ||
|  | 6be5774001 | ||
|  | f96cf2e8da | ||
|  | f832bd2712 | ||
|  | cb8667bd41 | ||
| 91156e5936 | |||
|  | 57e9a2dcb8 | ||
|  | 21edcd6ad5 | ||
|  | 3dc42eceb0 | ||
|  | b1830f5684 | ||
|  | 1a0c15e17f | ||
| 6e59d59a1d | |||
|  | fab7497311 | ||
| fccbcc77ea | |||
|  | 041c19fb52 | ||
| e36c562afa | |||
| 1589aa5251 | |||
| 8c672dd1eb | |||
|  | 3769550f24 | ||
|  | 5a06b6c53f | ||
|  | 1d4ef66d27 | ||
| e02f07401b | |||
| 118f92e75a | |||
|  | 788d302ce8 | ||
|  | cc09f35b49 | ||
| ef13933c66 | |||
| 4f72e1b35d | |||
| 2c1c76d2d0 | |||
| e2a84e119a | |||
| 5b85e59fc3 | |||
| c3b772919a | |||
|  | 70f2f5c248 | ||
|  | ffcb563b35 | ||
|  | 2dd96dd48e | 
| @@ -4,12 +4,12 @@ name: default | |||||||
|  |  | ||||||
| steps: | steps: | ||||||
|   - name: install |   - name: install | ||||||
|     image: node:19.4.0 |     image: node:22.17.0 | ||||||
|     commands: |     commands: | ||||||
|       - npm install |       - npm install | ||||||
|  |  | ||||||
|   - name: build_with_linux |   - name: build_with_linux | ||||||
|     image: node:19.4.0 |     image: node:22.17.0 | ||||||
|     commands: |     commands: | ||||||
|       - apt-get update && apt-get upgrade -y |       - apt-get update && apt-get upgrade -y | ||||||
|       - apt-get install -y libarchive-tools rpm |       - apt-get install -y libarchive-tools rpm | ||||||
|   | |||||||
							
								
								
									
										42
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										42
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -21,26 +21,30 @@ jobs: | |||||||
|       - uses: actions/checkout@master |       - uses: actions/checkout@master | ||||||
|       - uses: actions/setup-node@master |       - uses: actions/setup-node@master | ||||||
|         with: |         with: | ||||||
|           node-version: 22.4 |           node-version: 22.12.0 | ||||||
|       - run: npm install |       - run: npm install | ||||||
|       - run: npm run build |       - run: npm run build | ||||||
|  |       # - uses: actions/upload-artifact@master | ||||||
|  |       #   with: | ||||||
|  |       #     name: linux-builds | ||||||
|  |       #     path: dist/ | ||||||
|  |  | ||||||
|   build_on_mac: |   # build_on_mac: | ||||||
|     runs-on: macos-latest |   #   runs-on: macos-latest | ||||||
|     steps: |   #   steps: | ||||||
|       - uses: actions/checkout@master |   #     - uses: actions/checkout@master | ||||||
|       - uses: actions/setup-node@master |   #     - uses: actions/setup-node@master | ||||||
|         with: |   #       with: | ||||||
|           node-version: 22.4 |   #         node-version: 22.4 | ||||||
|       - run: npm install |   #     - run: npm install | ||||||
|       - run: npm run build |   #     - run: npm run build | ||||||
|  |  | ||||||
|   build_on_win: |   # build_on_win: | ||||||
|     runs-on: windows-latest |   #   runs-on: windows-latest | ||||||
|     steps: |   #   steps: | ||||||
|       - uses: actions/checkout@master |   #     - uses: actions/checkout@master | ||||||
|       - uses: actions/setup-node@master |   #     - uses: actions/setup-node@master | ||||||
|         with: |   #       with: | ||||||
|           node-version: 22.4 |   #         node-version: 22.4 | ||||||
|       - run: npm install |   #     - run: npm install | ||||||
|       - run: npm run build |   #     - run: npm run build | ||||||
|   | |||||||
							
								
								
									
										54
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										54
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -21,7 +21,7 @@ jobs: | |||||||
|       - uses: actions/checkout@master |       - uses: actions/checkout@master | ||||||
|       - uses: actions/setup-node@master |       - uses: actions/setup-node@master | ||||||
|         with: |         with: | ||||||
|           node-version: 22.4 |           node-version: 22.12.0 | ||||||
|       - run: npm install |       - run: npm install | ||||||
|       - run: npm run build |       - run: npm run build | ||||||
|       - uses: actions/upload-artifact@master |       - uses: actions/upload-artifact@master | ||||||
| @@ -29,30 +29,30 @@ jobs: | |||||||
|           name: linux-builds |           name: linux-builds | ||||||
|           path: dist/ |           path: dist/ | ||||||
|  |  | ||||||
|   build_on_mac: |   # build_on_mac: | ||||||
|     runs-on: macos-latest |   #   runs-on: macos-latest | ||||||
|     steps: |   #   steps: | ||||||
|       - uses: actions/checkout@master |   #     - uses: actions/checkout@master | ||||||
|       - uses: actions/setup-node@master |   #     - uses: actions/setup-node@master | ||||||
|         with: |   #       with: | ||||||
|           node-version: 22.4 |   #         node-version: 22.4 | ||||||
|       - run: npm install |   #     - run: npm install | ||||||
|       - run: npm run build |   #     - run: npm run build | ||||||
|       - uses: actions/upload-artifact@master |   #     - uses: actions/upload-artifact@master | ||||||
|         with: |   #       with: | ||||||
|           name: mac-builds |   #         name: mac-builds | ||||||
|           path: ./dist/ |   #         path: ./dist/ | ||||||
|  |  | ||||||
|   build_on_win: |   # build_on_win: | ||||||
|     runs-on: windows-latest |   #   runs-on: windows-latest | ||||||
|     steps: |   #   steps: | ||||||
|       - uses: actions/checkout@master |   #     - uses: actions/checkout@master | ||||||
|       - uses: actions/setup-node@master |   #     - uses: actions/setup-node@master | ||||||
|         with: |   #       with: | ||||||
|           node-version: 22.4 |   #         node-version: 22.4 | ||||||
|       - run: npm install |   #     - run: npm install | ||||||
|       - run: npm run build |   #     - run: npm run build | ||||||
|       - uses: actions/upload-artifact@master |   #     - uses: actions/upload-artifact@master | ||||||
|         with: |   #       with: | ||||||
|           name: windows-builds |   #         name: windows-builds | ||||||
|           path: dist/ |   #         path: dist/ | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -14,11 +14,12 @@ | |||||||
|     "rescrobbler", |     "rescrobbler", | ||||||
|     "scrobble", |     "scrobble", | ||||||
|     "scrobbling", |     "scrobbling", | ||||||
|     "Songwhip", |  | ||||||
|     "trackid", |     "trackid", | ||||||
|     "tracklist", |     "tracklist", | ||||||
|     "widevine", |     "widevine", | ||||||
|     "xesam" |     "wvcus", | ||||||
|  |     "xesam", | ||||||
|  |     "xhayper" | ||||||
|   ], |   ], | ||||||
|   "sonarlint.connectedMode.project": { |   "sonarlint.connectedMode.project": { | ||||||
|     "connectionId": "public-sonarcloud", |     "connectionId": "public-sonarcloud", | ||||||
|   | |||||||
							
								
								
									
										49
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										49
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -4,6 +4,55 @@ All notable changes to this project will be documented in this file. | |||||||
| The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | ||||||
| and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | ||||||
|  |  | ||||||
|  | ## [5.20.1] | ||||||
|  |  | ||||||
|  | - Updated electron to 37.2.5 | ||||||
|  |  | ||||||
|  | ## [5.20.0] | ||||||
|  |  | ||||||
|  | - Removes the `--enable-features=UseOzonePlatform` flag, as the Ozone platform has been the default on Linux since Electron 28 and this flag is no longer necessary. | ||||||
|  | - Adds the `--enable-wayland-ime` flag to enable Input Method Editor (IME) support in Wayland environments, improving the input experience for CJK and other users. | ||||||
|  | - Updated various dependencies | ||||||
|  | - Updated Electron to 37, potentially fixing [#580](https://github.com/Mastermindzh/tidal-hifi/issues/580) | ||||||
|  |  | ||||||
|  | ## [5.19.0] | ||||||
|  |  | ||||||
|  | - Fixed the issue where media updates would cease to work after album names can't be found | ||||||
|  |   - Will simply report an empty string when it can't find the album | ||||||
|  | - Updated various dependencies | ||||||
|  |  | ||||||
|  | ## [5.18.2] | ||||||
|  |  | ||||||
|  | - Reverted to sass 1.79.4 to fix `Nix` builds | ||||||
|  | - Changed electron-builder.base.yml to now generate the correct .desktop entries again | ||||||
|  |   - Should fix flatpak build | ||||||
|  |  | ||||||
|  | ## [5.18.1] | ||||||
|  |  | ||||||
|  | - Fixed the login bug | ||||||
|  | - Upgraded electron to 35.1.1 | ||||||
|  | - Added Widevine/CDM info to startup | ||||||
|  | - delayed remote electron initializer | ||||||
|  |  | ||||||
|  | ## [5.18.0] | ||||||
|  |  | ||||||
|  | - [Dianoga](https://github.com/Dianoga) fixed the duration selector, restoring mpris & partial API data. | ||||||
|  |   - PR: #554 | ||||||
|  | - Added `xesam:url` property to mpris metadata fixes [#506](https://github.com/Mastermindzh/tidal-hifi/issues/506) | ||||||
|  |  | ||||||
|  | ## [5.17.0] | ||||||
|  |  | ||||||
|  | - Added an option to disable the dynamic title and set it to a static one, [#491](https://github.com/Mastermindzh/tidal-hifi/pull/491) | ||||||
|  | - Discord integration now says "Listening to" instead of "playing" [#488](https://github.com/Mastermindzh/tidal-hifi/pull/488) && [#454](https://github.com/Mastermindzh/tidal-hifi/pull/454) | ||||||
|  | - Fixed several element names in the dom scraper | ||||||
|  | - Removed the Songwhip (they shut down) integration and replaced it with TIDAL's universal link system | ||||||
|  |  | ||||||
|  | ## [5.16.0] | ||||||
|  |  | ||||||
|  | - Fix issue #449 Discord RPC stuck on "Browsing Tidal". | ||||||
|  | - Fix issue #448 Add option to disable the discord rpc idle text | ||||||
|  | - Notifications are now send at the end of the update process, allowing other events to happen sooner. | ||||||
|  |  | ||||||
| ## [5.15.0] | ## [5.15.0] | ||||||
|  |  | ||||||
| - Added all missing swagger/openApi info with the help of [Times-Z](https://github.com/Times-Z) | - Added all missing swagger/openApi info with the help of [Times-Z](https://github.com/Times-Z) | ||||||
|   | |||||||
| @@ -48,7 +48,6 @@ The web version of [listen.tidal.com](https://listen.tidal.com) running in elect | |||||||
| - AlbumArt in integrations ([best-effort](https://github.com/Mastermindzh/tidal-hifi/pull/88#pullrequestreview-840814847)) | - AlbumArt in integrations ([best-effort](https://github.com/Mastermindzh/tidal-hifi/pull/88#pullrequestreview-840814847)) | ||||||
| - Custom [integrations](#integrations) | - Custom [integrations](#integrations) | ||||||
|   - [ListenBrainz](https://listenbrainz.org/?redirect=false) integration |   - [ListenBrainz](https://listenbrainz.org/?redirect=false) integration | ||||||
|   - Songwhip.com integration (hotkey `ctrl + w`) |  | ||||||
|   - Discord RPC integration (showing "now listening", "Browsing", etc) |   - Discord RPC integration (showing "now listening", "Browsing", etc) | ||||||
|     - Flatpak version only works if both Discord and Tidal-HiFi are flatpaks |     - Flatpak version only works if both Discord and Tidal-HiFi are flatpaks | ||||||
|   - MPRIS integration |   - MPRIS integration | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| appId: com.rickvanlieshout.tidal-hifi | appId: com.rickvanlieshout.tidal-hifi | ||||||
| electronVersion: 28.1.1 | electronVersion: 37.2.5 | ||||||
| electronDownload: | electronDownload: | ||||||
|   version: 28.1.1+wvcus |   version: 37.2.5+wvcus | ||||||
|   mirror: https://github.com/castlabs/electron-releases/releases/download/v |   mirror: https://github.com/castlabs/electron-releases/releases/download/v | ||||||
| snap: | snap: | ||||||
|   plugs: |   plugs: | ||||||
| @@ -17,24 +17,25 @@ linux: | |||||||
|   executableName: tidal-hifi |   executableName: tidal-hifi | ||||||
|   executableArgs: |   executableArgs: | ||||||
|     [ |     [ | ||||||
|       "--enable-features=UseOzonePlatform", |  | ||||||
|       "--ozone-platform-hint=auto", |       "--ozone-platform-hint=auto", | ||||||
|       "--enable-features=WaylandWindowDecorations", |       "--enable-features=WaylandWindowDecorations", | ||||||
|  |       "--enable-wayland-ime", | ||||||
|  |       "--use-angle", | ||||||
|     ] |     ] | ||||||
|   desktop: |   desktop: | ||||||
|     Encoding: UTF-8 |     entry: | ||||||
|     Name: TIDAL Hi-Fi |       Encoding: "UTF-8" | ||||||
|     GenericName: TIDAL Hi-Fi |       Name: "TIDAL Hi-Fi" | ||||||
|     Comment: The web version of listen.tidal.com running in electron with hifi support thanks to widevine. |       GenericName: "TIDAL Hi-Fi" | ||||||
|     Icon: tidal-hifi |       Comment: "The web version of listen.tidal.com running in electron with hifi support thanks to widevine." | ||||||
|     StartupNotify: true |       Icon: "tidal-hifi" | ||||||
|     Terminal: false |       StartupNotify: "true" | ||||||
|     Type: Application |       Terminal: "false" | ||||||
|     Categories: Network;Application;AudioVideo;Audio;Video |       Type: "Application" | ||||||
|     StartupWMClass: tidal-hifi |       Categories: "Network;Application;AudioVideo;Audio;Video" | ||||||
|     X-PulseAudio-Properties: media.role=music |       StartupWMClass: "tidal-hifi" | ||||||
|     MimeType: x-scheme-handler/tidal; |       X-PulseAudio-Properties: "media.role=music" | ||||||
|  |       MimeType: "x-scheme-handler/tidal;" | ||||||
| mac: | mac: | ||||||
|   category: public.app-category.entertainment |   category: public.app-category.entertainment | ||||||
| win: | win: | ||||||
|   | |||||||
							
								
								
									
										5537
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										5537
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										52
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										52
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,10 +1,10 @@ | |||||||
| { | { | ||||||
|   "name": "tidal-hifi", |   "name": "tidal-hifi", | ||||||
|   "version": "5.15.0", |   "version": "5.20.1", | ||||||
|   "description": "Tidal on Electron with widevine(hifi) support", |   "description": "Tidal on Electron with widevine(hifi) support", | ||||||
|   "main": "ts-dist/main.js", |   "main": "ts-dist/main.js", | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "start": "electron --inspect=0.0.0.0:5858 .", |     "start": "electron --inspect=0.0.0.0:5858 --remote-debugging-port=8315 --remote-allow-origins=* .", | ||||||
|     "watchStart": "nodemon dist -x \"npm run start\"", |     "watchStart": "nodemon dist -x \"npm run start\"", | ||||||
|     "compile": "tsc && npm run sass-and-copy", |     "compile": "tsc && npm run sass-and-copy", | ||||||
|     "deps": "npm run watch", |     "deps": "npm run watch", | ||||||
| @@ -40,45 +40,45 @@ | |||||||
|   "homepage": "https://github.com/Mastermindzh/tidal-hifi", |   "homepage": "https://github.com/Mastermindzh/tidal-hifi", | ||||||
|   "license": "MIT", |   "license": "MIT", | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@electron/remote": "^2.1.2", |     "@electron/remote": "^2.1.3", | ||||||
|     "@types/swagger-jsdoc": "^6.0.4", |     "@types/swagger-jsdoc": "^6.0.4", | ||||||
|     "axios": "^1.7.2", |     "@xhayper/discord-rpc": "1.3.0", | ||||||
|  |     "axios": "^1.12.0", | ||||||
|     "cors": "^2.8.5", |     "cors": "^2.8.5", | ||||||
|     "discord-rpc": "^4.0.1", |  | ||||||
|     "electron-store": "^8.2.0", |     "electron-store": "^8.2.0", | ||||||
|     "express": "^4.19.2", |     "express": "^5.1.0", | ||||||
|     "hotkeys-js": "^3.13.7", |     "hotkeys-js": "^3.13.15", | ||||||
|     "mpris-service": "^2.1.2", |     "mpris-service": "^2.1.2", | ||||||
|     "request": "^2.88.2", |     "request": "^2.88.2", | ||||||
|     "sass": "^1.77.7", |     "sass": "1.93.2", | ||||||
|     "swagger-ui-express": "^5.0.1" |     "swagger-ui-express": "^5.0.1" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@mastermindzh/prettier-config": "^1.0.0", |     "@mastermindzh/prettier-config": "^1.0.0", | ||||||
|     "@types/cors": "^2.8.17", |     "@types/cors": "^2.8.19", | ||||||
|     "@types/discord-rpc": "^4.0.8", |     "@types/express": "^5.0.3", | ||||||
|     "@types/express": "^4.17.21", |     "@types/node": "^22.16.2", | ||||||
|     "@types/node": "^20.14.10", |  | ||||||
|     "@types/request": "^2.48.12", |     "@types/request": "^2.48.12", | ||||||
|     "@types/swagger-ui-express": "^4.1.6", |     "@types/swagger-ui-express": "^4.1.8", | ||||||
|     "@typescript-eslint/eslint-plugin": "^7.16.0", |     "@typescript-eslint/eslint-plugin": "^8.36.0", | ||||||
|     "@typescript-eslint/parser": "^7.15.0", |     "@typescript-eslint/parser": "^8.36.0", | ||||||
|     "copyfiles": "^2.4.1", |     "copyfiles": "^2.4.1", | ||||||
|     "electron": "git+https://github.com/castlabs/electron-releases#v31.1.0+wvcus", |     "electron": "github:castlabs/electron-releases#v37.2.5+wvcus", | ||||||
|     "electron-builder": "~24.9.4", |     "electron-builder": "~26.0.12", | ||||||
|     "eslint": "^8.57.0", |     "eslint": "^9.30.1", | ||||||
|     "js-yaml": "^4.1.0", |     "js-yaml": "^4.1.0", | ||||||
|     "markdown-toc": "^1.2.0", |     "markdown-toc": "^1.2.0", | ||||||
|     "nodemon": "^3.1.4", |     "node-abi": "^4.12.0", | ||||||
|     "prettier": "^3.3.2", |     "nodemon": "^3.1.10", | ||||||
|     "stylelint": "^16.6.1", |     "prettier": "^3.6.2", | ||||||
|     "stylelint-config-standard": "^36.0.1", |     "stylelint": "^16.21.1", | ||||||
|     "stylelint-config-standard-scss": "^13.1.0", |     "stylelint-config-standard": "^39.0.0", | ||||||
|     "stylelint-prettier": "^5.0.0", |     "stylelint-config-standard-scss": "^16.0.0", | ||||||
|  |     "stylelint-prettier": "^5.0.3", | ||||||
|     "swagger-jsdoc": "^6.2.8", |     "swagger-jsdoc": "^6.2.8", | ||||||
|     "ts-node": "^10.9.2", |     "ts-node": "^10.9.2", | ||||||
|     "tsc-watch": "^6.2.0", |     "tsc-watch": "^7.1.1", | ||||||
|     "typescript": "^5.5.3" |     "typescript": "^5.8.3" | ||||||
|   }, |   }, | ||||||
|   "prettier": "@mastermindzh/prettier-config" |   "prettier": "@mastermindzh/prettier-config" | ||||||
| } | } | ||||||
							
								
								
									
										4
									
								
								renovate.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								renovate.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | { | ||||||
|  |   "$schema": "https://docs.renovatebot.com/renovate-schema.json", | ||||||
|  |   "ignoreDeps": ["@types/node", "electron-store", "@xhayper/discord-rpc"] | ||||||
|  | } | ||||||
| @@ -16,17 +16,19 @@ | |||||||
|     search: '[class^="searchField"]', |     search: '[class^="searchField"]', | ||||||
|     shuffle: '*[data-test="shuffle"]', |     shuffle: '*[data-test="shuffle"]', | ||||||
|     repeat: '*[data-test="repeat"]', |     repeat: '*[data-test="repeat"]', | ||||||
|     account: '*[class^="profileOptions"]', |     account: '*[data-test^="profile-image-button"]', | ||||||
|     settings: '*[data-test^="open-settings"]', |     settings: '*[data-test^="sidebar-menu-button"]', | ||||||
|  |     openSettings: '*[data-test^="open-settings"]', | ||||||
|     media: '*[data-test="current-media-imagery"]', |     media: '*[data-test="current-media-imagery"]', | ||||||
|     image: "img", |     image: "img", | ||||||
|     current: '*[data-test="current-time"]', |     current: '*[data-test="current-time"]', | ||||||
|     duration: '*[class^=playbackControlsContainer] *[data-test="duration"]', |     duration: '*[class^=_playbackControlsContainer] *[data-test="duration"]', | ||||||
|     bar: '*[data-test="progress-bar"]', |     bar: '*[data-test="progress-bar"]', | ||||||
|     footer: "#footerPlayer", |     footer: "#footerPlayer", | ||||||
|     mediaItem: "[data-type='mediaItem']", |     mediaItem: "[data-type='mediaItem']", | ||||||
|     album_header_title: '*[class^="playingFrom"] span:nth-child(2)', |     album_header_title: '*[class^="_playingFrom"] span:nth-child(2)', | ||||||
|     playingFrom: '*[class^="playingFrom"] span:nth-child(2)', |     playing_from: '*[class^="_playingFrom"] span:nth-child(2)', | ||||||
|  |     queue_album: "*[class^=playQueueItemsContainer] *[class^=groupTitle] span:nth-child(2)", | ||||||
|     currentlyPlaying: "[class^='isPlayingIcon'], [data-test-is-playing='true']", |     currentlyPlaying: "[class^='isPlayingIcon'], [data-test-is-playing='true']", | ||||||
|     album_name_cell: '[class^="album"]', |     album_name_cell: '[class^="album"]', | ||||||
|     tracklist_row: '[data-test="tracklist-row"]', |     tracklist_row: '[data-test="tracklist-row"]', | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ export const globalEvents = { | |||||||
|   showSettings: "showSettings", |   showSettings: "showSettings", | ||||||
|   storeChanged: "storeChanged", |   storeChanged: "storeChanged", | ||||||
|   error: "error", |   error: "error", | ||||||
|   whip: "whip", |   getUniversalLink: "getUniversalLink", | ||||||
|   log: "log", |   log: "log", | ||||||
|   toggleFavorite: "toggleFavorite", |   toggleFavorite: "toggleFavorite", | ||||||
|   toggleShuffle: "toggleShuffle", |   toggleShuffle: "toggleShuffle", | ||||||
|   | |||||||
| @@ -30,6 +30,7 @@ export const settings = { | |||||||
|     buttonText: "discord.buttonText", |     buttonText: "discord.buttonText", | ||||||
|     includeTimestamps: "discord.includeTimestamps", |     includeTimestamps: "discord.includeTimestamps", | ||||||
|     showSong: "discord.showSong", |     showSong: "discord.showSong", | ||||||
|  |     showIdle: "discord.showIdle", | ||||||
|     idleText: "discord.idleText", |     idleText: "discord.idleText", | ||||||
|     usingText: "discord.usingText", |     usingText: "discord.usingText", | ||||||
|   }, |   }, | ||||||
| @@ -54,6 +55,7 @@ export const settings = { | |||||||
|   singleInstance: "singleInstance", |   singleInstance: "singleInstance", | ||||||
|   skipArtists: "skipArtists", |   skipArtists: "skipArtists", | ||||||
|   skippedArtists: "skippedArtists", |   skippedArtists: "skippedArtists", | ||||||
|  |   staticWindowTitle: "staticWindowTitle", | ||||||
|   theme: "theme", |   theme: "theme", | ||||||
|   trayIcon: "trayIcon", |   trayIcon: "trayIcon", | ||||||
|   updateFrequency: "updateFrequency", |   updateFrequency: "updateFrequency", | ||||||
|   | |||||||
| @@ -88,8 +88,9 @@ export const addCurrentInfo = (expressApp: Router) => { | |||||||
|    *             schema: |    *             schema: | ||||||
|    *               $ref: '#/components/schemas/MediaInfo' |    *               $ref: '#/components/schemas/MediaInfo' | ||||||
|    */ |    */ | ||||||
|   expressApp.get("/current", (req, res) => res.json({ ...mediaInfo, artist: mediaInfo.artists })); |   expressApp.get("/current", (_req, res) => { | ||||||
|  |     res.json({ ...mediaInfo, artist: mediaInfo.artists }); | ||||||
|  |   }); | ||||||
|   /** |   /** | ||||||
|    * @swagger |    * @swagger | ||||||
|    * /current/image: |    * /current/image: | ||||||
|   | |||||||
| @@ -21,8 +21,12 @@ export const startApi = (mainWindow: BrowserWindow) => { | |||||||
|   expressApp.use(cors()); |   expressApp.use(cors()); | ||||||
|   expressApp.use(express.json()); |   expressApp.use(express.json()); | ||||||
|   expressApp.use("/docs", swaggerUi.serve, swaggerUi.setup(swaggerSpec)); |   expressApp.use("/docs", swaggerUi.serve, swaggerUi.setup(swaggerSpec)); | ||||||
|   expressApp.get("/", (req, res) => res.send("Hello World!")); |   expressApp.get("/", (req, res) => { | ||||||
|   expressApp.get("/swagger.json", (req, res) => res.json(swaggerSpec)); |     res.send("Hello World!"); | ||||||
|  |   }); | ||||||
|  |   expressApp.get("/swagger.json", (req, res) => { | ||||||
|  |     res.json(swaggerSpec); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|   // add features |   // add features | ||||||
|   addLegacyApi(expressApp, mainWindow); |   addLegacyApi(expressApp, mainWindow); | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|   "openapi": "3.1.0", |   "openapi": "3.1.0", | ||||||
|   "info": { |   "info": { | ||||||
|     "title": "TIDAL Hi-Fi API", |     "title": "TIDAL Hi-Fi API", | ||||||
|     "version": "5.15.0", |     "version": "5.20.1", | ||||||
|     "description": "", |     "description": "", | ||||||
|     "license": { |     "license": { | ||||||
|       "name": "MIT", |       "name": "MIT", | ||||||
| @@ -21,9 +21,7 @@ | |||||||
|     "/current": { |     "/current": { | ||||||
|       "get": { |       "get": { | ||||||
|         "summary": "Get current media info", |         "summary": "Get current media info", | ||||||
|         "tags": [ |         "tags": ["current"], | ||||||
|           "current" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Current media info", |             "description": "Current media info", | ||||||
| @@ -41,9 +39,7 @@ | |||||||
|     "/current/image": { |     "/current/image": { | ||||||
|       "get": { |       "get": { | ||||||
|         "summary": "Get current media image", |         "summary": "Get current media image", | ||||||
|         "tags": [ |         "tags": ["current"], | ||||||
|           "current" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Current media image", |             "description": "Current media image", | ||||||
| @@ -65,9 +61,7 @@ | |||||||
|     "/player/play": { |     "/player/play": { | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Play the current media", |         "summary": "Play the current media", | ||||||
|         "tags": [ |         "tags": ["player"], | ||||||
|           "player" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Ok", |             "description": "Ok", | ||||||
| @@ -85,9 +79,7 @@ | |||||||
|     "/player/favorite/toggle": { |     "/player/favorite/toggle": { | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Add the current media to your favorites, or remove it if its already added to your favorites", |         "summary": "Add the current media to your favorites, or remove it if its already added to your favorites", | ||||||
|         "tags": [ |         "tags": ["player"], | ||||||
|           "player" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Ok", |             "description": "Ok", | ||||||
| @@ -105,9 +97,7 @@ | |||||||
|     "/player/pause": { |     "/player/pause": { | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Pause the current media", |         "summary": "Pause the current media", | ||||||
|         "tags": [ |         "tags": ["player"], | ||||||
|           "player" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Ok", |             "description": "Ok", | ||||||
| @@ -125,9 +115,7 @@ | |||||||
|     "/player/next": { |     "/player/next": { | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Play the next song", |         "summary": "Play the next song", | ||||||
|         "tags": [ |         "tags": ["player"], | ||||||
|           "player" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Ok", |             "description": "Ok", | ||||||
| @@ -145,9 +133,7 @@ | |||||||
|     "/player/previous": { |     "/player/previous": { | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Play the previous song", |         "summary": "Play the previous song", | ||||||
|         "tags": [ |         "tags": ["player"], | ||||||
|           "player" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Ok", |             "description": "Ok", | ||||||
| @@ -165,9 +151,7 @@ | |||||||
|     "/player/shuffle/toggle": { |     "/player/shuffle/toggle": { | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Play the previous song", |         "summary": "Play the previous song", | ||||||
|         "tags": [ |         "tags": ["player"], | ||||||
|           "player" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Ok", |             "description": "Ok", | ||||||
| @@ -185,9 +169,7 @@ | |||||||
|     "/player/repeat/toggle": { |     "/player/repeat/toggle": { | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Toggle the repeat status, toggles between \"off\" , \"single\" and \"all\"", |         "summary": "Toggle the repeat status, toggles between \"off\" , \"single\" and \"all\"", | ||||||
|         "tags": [ |         "tags": ["player"], | ||||||
|           "player" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Ok", |             "description": "Ok", | ||||||
| @@ -205,9 +187,7 @@ | |||||||
|     "/player/playpause": { |     "/player/playpause": { | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Start playing the media if paused, or pause the media if playing", |         "summary": "Start playing the media if paused, or pause the media if playing", | ||||||
|         "tags": [ |         "tags": ["player"], | ||||||
|           "player" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Ok", |             "description": "Ok", | ||||||
| @@ -225,9 +205,7 @@ | |||||||
|     "/settings/skipped-artists": { |     "/settings/skipped-artists": { | ||||||
|       "get": { |       "get": { | ||||||
|         "summary": "get a list of artists that TIDAL Hi-Fi will skip if skipping is enabled", |         "summary": "get a list of artists that TIDAL Hi-Fi will skip if skipping is enabled", | ||||||
|         "tags": [ |         "tags": ["settings"], | ||||||
|           "settings" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "The list book.", |             "description": "The list book.", | ||||||
| @@ -243,9 +221,7 @@ | |||||||
|       }, |       }, | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Add new artists to the list of skipped artists", |         "summary": "Add new artists to the list of skipped artists", | ||||||
|         "tags": [ |         "tags": ["settings"], | ||||||
|           "settings" |  | ||||||
|         ], |  | ||||||
|         "requestBody": { |         "requestBody": { | ||||||
|           "required": true, |           "required": true, | ||||||
|           "content": { |           "content": { | ||||||
| @@ -266,9 +242,7 @@ | |||||||
|     "/settings/skipped-artists/delete": { |     "/settings/skipped-artists/delete": { | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Remove artists from the list of skipped artists", |         "summary": "Remove artists from the list of skipped artists", | ||||||
|         "tags": [ |         "tags": ["settings"], | ||||||
|           "settings" |  | ||||||
|         ], |  | ||||||
|         "requestBody": { |         "requestBody": { | ||||||
|           "required": true, |           "required": true, | ||||||
|           "content": { |           "content": { | ||||||
| @@ -289,9 +263,7 @@ | |||||||
|     "/settings/skipped-artists/current": { |     "/settings/skipped-artists/current": { | ||||||
|       "post": { |       "post": { | ||||||
|         "summary": "Add the current artist to the list of skipped artists", |         "summary": "Add the current artist to the list of skipped artists", | ||||||
|         "tags": [ |         "tags": ["settings"], | ||||||
|           "settings" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Ok" |             "description": "Ok" | ||||||
| @@ -300,9 +272,7 @@ | |||||||
|       }, |       }, | ||||||
|       "delete": { |       "delete": { | ||||||
|         "summary": "Remove the current artist from the list of skipped artists", |         "summary": "Remove the current artist from the list of skipped artists", | ||||||
|         "tags": [ |         "tags": ["settings"], | ||||||
|           "settings" |  | ||||||
|         ], |  | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
|             "description": "Ok" |             "description": "Ok" | ||||||
| @@ -313,9 +283,7 @@ | |||||||
|     "/image": { |     "/image": { | ||||||
|       "get": { |       "get": { | ||||||
|         "summary": "Get current image", |         "summary": "Get current image", | ||||||
|         "tags": [ |         "tags": ["legacy"], | ||||||
|           "legacy" |  | ||||||
|         ], |  | ||||||
|         "deprecated": true, |         "deprecated": true, | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
| @@ -338,9 +306,7 @@ | |||||||
|     "/play": { |     "/play": { | ||||||
|       "get": { |       "get": { | ||||||
|         "summary": "Play the current media", |         "summary": "Play the current media", | ||||||
|         "tags": [ |         "tags": ["legacy"], | ||||||
|           "legacy" |  | ||||||
|         ], |  | ||||||
|         "deprecated": true, |         "deprecated": true, | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
| @@ -359,9 +325,7 @@ | |||||||
|     "/favorite/toggle": { |     "/favorite/toggle": { | ||||||
|       "get": { |       "get": { | ||||||
|         "summary": "Add the current media to your favorites, or remove it if its already added to your favorites", |         "summary": "Add the current media to your favorites, or remove it if its already added to your favorites", | ||||||
|         "tags": [ |         "tags": ["legacy"], | ||||||
|           "legacy" |  | ||||||
|         ], |  | ||||||
|         "deprecated": true, |         "deprecated": true, | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
| @@ -380,9 +344,7 @@ | |||||||
|     "/pause": { |     "/pause": { | ||||||
|       "get": { |       "get": { | ||||||
|         "summary": "Pause the current media", |         "summary": "Pause the current media", | ||||||
|         "tags": [ |         "tags": ["legacy"], | ||||||
|           "legacy" |  | ||||||
|         ], |  | ||||||
|         "deprecated": true, |         "deprecated": true, | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
| @@ -401,9 +363,7 @@ | |||||||
|     "/next": { |     "/next": { | ||||||
|       "get": { |       "get": { | ||||||
|         "summary": "Play the next song", |         "summary": "Play the next song", | ||||||
|         "tags": [ |         "tags": ["legacy"], | ||||||
|           "legacy" |  | ||||||
|         ], |  | ||||||
|         "deprecated": true, |         "deprecated": true, | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
| @@ -422,9 +382,7 @@ | |||||||
|     "/previous": { |     "/previous": { | ||||||
|       "get": { |       "get": { | ||||||
|         "summary": "Play the previous song", |         "summary": "Play the previous song", | ||||||
|         "tags": [ |         "tags": ["legacy"], | ||||||
|           "legacy" |  | ||||||
|         ], |  | ||||||
|         "deprecated": true, |         "deprecated": true, | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
| @@ -443,9 +401,7 @@ | |||||||
|     "/playpause": { |     "/playpause": { | ||||||
|       "get": { |       "get": { | ||||||
|         "summary": "Toggle play/pause", |         "summary": "Toggle play/pause", | ||||||
|         "tags": [ |         "tags": ["legacy"], | ||||||
|           "legacy" |  | ||||||
|         ], |  | ||||||
|         "deprecated": true, |         "deprecated": true, | ||||||
|         "responses": { |         "responses": { | ||||||
|           "200": { |           "200": { | ||||||
| @@ -558,10 +514,7 @@ | |||||||
|         "items": { |         "items": { | ||||||
|           "type": "string" |           "type": "string" | ||||||
|         }, |         }, | ||||||
|         "example": [ |         "example": ["Artist1", "Artist2"] | ||||||
|           "Artist1", |  | ||||||
|           "Artist2" |  | ||||||
|         ] |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								src/features/sharingService/sharingService.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/features/sharingService/sharingService.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | export class SharingService { | ||||||
|  |   /** | ||||||
|  |    * Retrieve the universal link given a regular track link | ||||||
|  |    * @param currentUrl | ||||||
|  |    * @returns | ||||||
|  |    */ | ||||||
|  |   public static getUniversalLink(currentUrl: string): string { | ||||||
|  |     return `${currentUrl}?u`; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,21 +0,0 @@ | |||||||
| import { ServiceLinks } from "./ServiceLinks"; |  | ||||||
|  |  | ||||||
| export interface Artist { |  | ||||||
|   type: string; |  | ||||||
|   id: number; |  | ||||||
|   path: string; |  | ||||||
|   name: string; |  | ||||||
|   sourceUrl: string; |  | ||||||
|   sourceCountry: string; |  | ||||||
|   url: string; |  | ||||||
|   image: string; |  | ||||||
|   createdAt: string; |  | ||||||
|   updatedAt: string; |  | ||||||
|   refreshedAt: string; |  | ||||||
|   serviceIds: { [key: string]: string }; |  | ||||||
|   orchardId: string; |  | ||||||
|   spotifyId: string; |  | ||||||
|   links: { [key: string]: ServiceLinks[] }; |  | ||||||
|   linksCountries: string[]; |  | ||||||
|   description: string; |  | ||||||
| } |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| export interface ServiceLinks { |  | ||||||
|   link: string; |  | ||||||
|   countries: string[]; |  | ||||||
| } |  | ||||||
| @@ -1,27 +0,0 @@ | |||||||
| import { Artist } from "./Artist"; |  | ||||||
| import { ServiceLinks } from "./ServiceLinks"; |  | ||||||
|  |  | ||||||
| export interface WhippedResult { |  | ||||||
|   status: string; |  | ||||||
|   data: { |  | ||||||
|     item: { |  | ||||||
|       type: string; |  | ||||||
|       id: number; |  | ||||||
|       path: string; |  | ||||||
|       name: string; |  | ||||||
|       url: string; |  | ||||||
|       sourceUrl: string; |  | ||||||
|       sourceCountry: string; |  | ||||||
|       releaseDate: string; |  | ||||||
|       createdAt: string; |  | ||||||
|       updatedAt: string; |  | ||||||
|       refreshedAt: string; |  | ||||||
|       image: string; |  | ||||||
|       isrc: string; |  | ||||||
|       isExplicit: boolean; |  | ||||||
|       links: { [key: string]: ServiceLinks[] }; |  | ||||||
|       linksCountries: string[]; |  | ||||||
|       artists: Artist[]; |  | ||||||
|     }; |  | ||||||
|   }; |  | ||||||
| } |  | ||||||
| @@ -1,32 +0,0 @@ | |||||||
| import { WhippedResult } from "./models/whip"; |  | ||||||
| import axios from "axios"; |  | ||||||
|  |  | ||||||
| export class Songwhip { |  | ||||||
|   /** |  | ||||||
|    * Call the songwhip API and create a shareable songwhip page |  | ||||||
|    * @param currentUrl |  | ||||||
|    * @returns |  | ||||||
|    */ |  | ||||||
|   public static async whip(currentUrl: string): Promise<WhippedResult> { |  | ||||||
|     try { |  | ||||||
|       const response = await axios.post("https://songwhip.com/api/songwhip/create", { |  | ||||||
|         url: currentUrl, |  | ||||||
|         // doesn't actually matter.. returns everything the same way anyway |  | ||||||
|         country: "NL", |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       return response.data; |  | ||||||
|     } catch (error) { |  | ||||||
|       console.log(JSON.stringify(error)); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Transform a songwhip response into a shareable url |  | ||||||
|    * @param response |  | ||||||
|    * @returns |  | ||||||
|    */ |  | ||||||
|   public static getWhipUrl(response: WhippedResult) { |  | ||||||
|     return `https://songwhip.com${response.data.item.url}`; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
							
								
								
									
										12
									
								
								src/main.ts
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								src/main.ts
									
									
									
									
									
								
							| @@ -10,7 +10,7 @@ import { | |||||||
|   releaseInhibitorIfActive, |   releaseInhibitorIfActive, | ||||||
| } from "./features/idleInhibitor/idleInhibitor"; | } from "./features/idleInhibitor/idleInhibitor"; | ||||||
| import { Logger } from "./features/logger"; | import { Logger } from "./features/logger"; | ||||||
| import { Songwhip } from "./features/songwhip/songwhip"; | import { SharingService } from "./features/sharingService/sharingService"; | ||||||
| import { MediaInfo } from "./models/mediaInfo"; | import { MediaInfo } from "./models/mediaInfo"; | ||||||
| import { MediaStatus } from "./models/mediaStatus"; | import { MediaStatus } from "./models/mediaStatus"; | ||||||
| import { initRPC, rpc, unRPC } from "./scripts/discord"; | import { initRPC, rpc, unRPC } from "./scripts/discord"; | ||||||
| @@ -26,7 +26,6 @@ import { | |||||||
| import { addTray, refreshTray } from "./scripts/tray"; | import { addTray, refreshTray } from "./scripts/tray"; | ||||||
| let mainInhibitorId = -1; | let mainInhibitorId = -1; | ||||||
|  |  | ||||||
| initialize(); |  | ||||||
| let mainWindow: BrowserWindow; | let mainWindow: BrowserWindow; | ||||||
| const icon = path.join(__dirname, "../assets/icon.png"); | const icon = path.join(__dirname, "../assets/icon.png"); | ||||||
| const PROTOCOL_PREFIX = "tidal"; | const PROTOCOL_PREFIX = "tidal"; | ||||||
| @@ -98,6 +97,7 @@ function createWindow(options = { x: 0, y: 0, backgroundColor: "white" }) { | |||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   enable(mainWindow.webContents); |   enable(mainWindow.webContents); | ||||||
|   registerHttpProtocols(); |   registerHttpProtocols(); | ||||||
|   syncMenuBarWithStore(); |   syncMenuBarWithStore(); | ||||||
| @@ -126,6 +126,7 @@ function createWindow(options = { x: 0, y: 0, backgroundColor: "white" }) { | |||||||
|     } |     } | ||||||
|     return false; |     return false; | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   // Emitted when the window is closed. |   // Emitted when the window is closed. | ||||||
|   mainWindow.on("closed", function () { |   mainWindow.on("closed", function () { | ||||||
|     releaseInhibitorIfActive(mainInhibitorId); |     releaseInhibitorIfActive(mainInhibitorId); | ||||||
| @@ -178,6 +179,7 @@ app.on("ready", async () => { | |||||||
|  |  | ||||||
|   if (isMainInstance() || isMultipleInstancesAllowed()) { |   if (isMainInstance() || isMultipleInstancesAllowed()) { | ||||||
|     await components.whenReady(); |     await components.whenReady(); | ||||||
|  |     initialize(); | ||||||
|  |  | ||||||
|     // Adblock |     // Adblock | ||||||
|     if (settingsStore.get(settings.adBlock)) { |     if (settingsStore.get(settings.adBlock)) { | ||||||
| @@ -188,6 +190,8 @@ app.on("ready", async () => { | |||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     Logger.log("components ready:", components.status()); | ||||||
|  |  | ||||||
|     createWindow(); |     createWindow(); | ||||||
|     addMenu(mainWindow); |     addMenu(mainWindow); | ||||||
|     createSettingsWindow(); |     createSettingsWindow(); | ||||||
| @@ -250,8 +254,8 @@ ipcMain.on(globalEvents.error, (event) => { | |||||||
|   console.log(event); |   console.log(event); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| ipcMain.handle(globalEvents.whip, async (event, url) => { | ipcMain.handle(globalEvents.getUniversalLink, async (event, url) => { | ||||||
|   return Songwhip.whip(url); |   return SharingService.getUniversalLink(url); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| Logger.watch(ipcMain); | Logger.watch(ipcMain); | ||||||
|   | |||||||
| @@ -46,6 +46,7 @@ let adBlock: HTMLInputElement, | |||||||
|   singleInstance: HTMLInputElement, |   singleInstance: HTMLInputElement, | ||||||
|   skipArtists: HTMLInputElement, |   skipArtists: HTMLInputElement, | ||||||
|   skippedArtists: HTMLInputElement, |   skippedArtists: HTMLInputElement, | ||||||
|  |   staticWindowTitle: HTMLInputElement, | ||||||
|   theme: HTMLSelectElement, |   theme: HTMLSelectElement, | ||||||
|   trayIcon: HTMLInputElement, |   trayIcon: HTMLInputElement, | ||||||
|   updateFrequency: HTMLInputElement, |   updateFrequency: HTMLInputElement, | ||||||
| @@ -58,6 +59,7 @@ let adBlock: HTMLInputElement, | |||||||
|   discord_include_timestamps: HTMLInputElement, |   discord_include_timestamps: HTMLInputElement, | ||||||
|   discord_button_text: HTMLInputElement, |   discord_button_text: HTMLInputElement, | ||||||
|   discord_show_song: HTMLInputElement, |   discord_show_song: HTMLInputElement, | ||||||
|  |   discord_show_idle: HTMLInputElement, | ||||||
|   discord_idle_text: HTMLInputElement, |   discord_idle_text: HTMLInputElement, | ||||||
|   discord_using_text: HTMLInputElement; |   discord_using_text: HTMLInputElement; | ||||||
|  |  | ||||||
| @@ -139,8 +141,9 @@ function refreshSettings() { | |||||||
|     port.value = settingsStore.get(settings.apiSettings.port); |     port.value = settingsStore.get(settings.apiSettings.port); | ||||||
|     singleInstance.checked = settingsStore.get(settings.singleInstance); |     singleInstance.checked = settingsStore.get(settings.singleInstance); | ||||||
|     skipArtists.checked = settingsStore.get(settings.skipArtists); |     skipArtists.checked = settingsStore.get(settings.skipArtists); | ||||||
|     theme.value = settingsStore.get(settings.theme); |  | ||||||
|     skippedArtists.value = settingsStore.get<string, string[]>(settings.skippedArtists).join("\n"); |     skippedArtists.value = settingsStore.get<string, string[]>(settings.skippedArtists).join("\n"); | ||||||
|  |     staticWindowTitle.checked = settingsStore.get(settings.staticWindowTitle); | ||||||
|  |     theme.value = settingsStore.get(settings.theme); | ||||||
|     trayIcon.checked = settingsStore.get(settings.trayIcon); |     trayIcon.checked = settingsStore.get(settings.trayIcon); | ||||||
|     updateFrequency.value = settingsStore.get(settings.updateFrequency); |     updateFrequency.value = settingsStore.get(settings.updateFrequency); | ||||||
|     enableListenBrainz.checked = settingsStore.get(settings.ListenBrainz.enabled); |     enableListenBrainz.checked = settingsStore.get(settings.ListenBrainz.enabled); | ||||||
| @@ -151,6 +154,7 @@ function refreshSettings() { | |||||||
|     discord_include_timestamps.checked = settingsStore.get(settings.discord.includeTimestamps); |     discord_include_timestamps.checked = settingsStore.get(settings.discord.includeTimestamps); | ||||||
|     discord_button_text.value = settingsStore.get(settings.discord.buttonText); |     discord_button_text.value = settingsStore.get(settings.discord.buttonText); | ||||||
|     discord_show_song.checked = settingsStore.get(settings.discord.showSong); |     discord_show_song.checked = settingsStore.get(settings.discord.showSong); | ||||||
|  |     discord_show_idle.checked = settingsStore.get(settings.discord.showIdle); | ||||||
|     discord_idle_text.value = settingsStore.get(settings.discord.idleText); |     discord_idle_text.value = settingsStore.get(settings.discord.idleText); | ||||||
|     discord_using_text.value = settingsStore.get(settings.discord.usingText); |     discord_using_text.value = settingsStore.get(settings.discord.usingText); | ||||||
|  |  | ||||||
| @@ -259,6 +263,7 @@ window.addEventListener("DOMContentLoaded", () => { | |||||||
|   trayIcon = get("trayIcon"); |   trayIcon = get("trayIcon"); | ||||||
|   skipArtists = get("skipArtists"); |   skipArtists = get("skipArtists"); | ||||||
|   skippedArtists = get("skippedArtists"); |   skippedArtists = get("skippedArtists"); | ||||||
|  |   staticWindowTitle = get("staticWindowTitle"); | ||||||
|   singleInstance = get("singleInstance"); |   singleInstance = get("singleInstance"); | ||||||
|   updateFrequency = get("updateFrequency"); |   updateFrequency = get("updateFrequency"); | ||||||
|   enableListenBrainz = get("enableListenBrainz"); |   enableListenBrainz = get("enableListenBrainz"); | ||||||
| @@ -269,6 +274,7 @@ window.addEventListener("DOMContentLoaded", () => { | |||||||
|   listenbrainz_delay = get("listenbrainz_delay"); |   listenbrainz_delay = get("listenbrainz_delay"); | ||||||
|   discord_button_text = get("discord_button_text"); |   discord_button_text = get("discord_button_text"); | ||||||
|   discord_show_song = get("discord_show_song"); |   discord_show_song = get("discord_show_song"); | ||||||
|  |   discord_show_idle = get("discord_show_idle"); | ||||||
|   discord_using_text = get("discord_using_text"); |   discord_using_text = get("discord_using_text"); | ||||||
|   discord_idle_text = get("discord_idle_text"); |   discord_idle_text = get("discord_idle_text"); | ||||||
|  |  | ||||||
| @@ -292,6 +298,7 @@ window.addEventListener("DOMContentLoaded", () => { | |||||||
|   addInputListener(port, settings.apiSettings.port); |   addInputListener(port, settings.apiSettings.port); | ||||||
|   addInputListener(skipArtists, settings.skipArtists); |   addInputListener(skipArtists, settings.skipArtists); | ||||||
|   addTextAreaListener(skippedArtists, settings.skippedArtists); |   addTextAreaListener(skippedArtists, settings.skippedArtists); | ||||||
|  |   addInputListener(staticWindowTitle, settings.staticWindowTitle); | ||||||
|   addInputListener(singleInstance, settings.singleInstance); |   addInputListener(singleInstance, settings.singleInstance); | ||||||
|   addSelectListener(theme, settings.theme); |   addSelectListener(theme, settings.theme); | ||||||
|   addInputListener(trayIcon, settings.trayIcon); |   addInputListener(trayIcon, settings.trayIcon); | ||||||
| @@ -312,6 +319,7 @@ window.addEventListener("DOMContentLoaded", () => { | |||||||
|     settings.discord.showSong, |     settings.discord.showSong, | ||||||
|     switchesWithSettings.discord_show_song |     switchesWithSettings.discord_show_song | ||||||
|   ); |   ); | ||||||
|  |   addInputListener(discord_show_idle, settings.discord.showIdle); | ||||||
|   addInputListener(discord_idle_text, settings.discord.idleText); |   addInputListener(discord_idle_text, settings.discord.idleText); | ||||||
|   addInputListener(discord_using_text, settings.discord.usingText); |   addInputListener(discord_using_text, settings.discord.usingText); | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ | |||||||
|   <meta name="viewport" content="width=device-width, initial-scale=1.0" /> |   <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||||
|   <meta http-equiv="X-UA-Compatible" content="ie=edge" /> |   <meta http-equiv="X-UA-Compatible" content="ie=edge" /> | ||||||
|   <link rel="stylesheet" href="./settings.css" /> |   <link rel="stylesheet" href="./settings.css" /> | ||||||
|   <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css"> |   <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/7.0.1/css/font-awesome.min.css"> | ||||||
| </head> | </head> | ||||||
|  |  | ||||||
| <body class="settings-window"> | <body class="settings-window"> | ||||||
| @@ -106,6 +106,16 @@ | |||||||
|                 <span class="switch__slider"></span> |                 <span class="switch__slider"></span> | ||||||
|               </label> |               </label> | ||||||
|             </div> |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Static Window Title</h4> | ||||||
|  |                 <p>Makes the window title "TIDAL Hi-Fi" instead of changing to the currently playing song.</p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="staticWindowTitle" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|             <div class="group__option"> |             <div class="group__option"> | ||||||
|               <div class="group__description"> |               <div class="group__description"> | ||||||
|                 <h4>Minimize on Close</h4> |                 <h4>Minimize on Close</h4> | ||||||
| @@ -227,6 +237,17 @@ | |||||||
|             </div> |             </div> | ||||||
|             <div id="discord_options"> |             <div id="discord_options"> | ||||||
|  |  | ||||||
|  |               <div class="group__option" class="hidden"> | ||||||
|  |                 <div class="group__description"> | ||||||
|  |                   <h4>Show Idle Text</h4> | ||||||
|  |                   <p>Should the idle text be shown when idle?</p> | ||||||
|  |                 </div> | ||||||
|  |                 <label class="switch"> | ||||||
|  |                   <input id="discord_show_idle" type="checkbox" /> | ||||||
|  |                   <span class="switch__slider"></span> | ||||||
|  |                 </label> | ||||||
|  |               </div> | ||||||
|  |  | ||||||
|               <div class="group__option" class="hidden"> |               <div class="group__option" class="hidden"> | ||||||
|                 <div class="group__description"> |                 <div class="group__description"> | ||||||
|                   <h4>Idle Text</h4> |                   <h4>Idle Text</h4> | ||||||
| @@ -457,7 +478,7 @@ | |||||||
|           <h4>TIDAL Hi-Fi</h4> |           <h4>TIDAL Hi-Fi</h4> | ||||||
|           <div class="about-section__version"> |           <div class="about-section__version"> | ||||||
|             <a target="_blank" rel="noopener" |             <a target="_blank" rel="noopener" | ||||||
|               href="https://github.com/Mastermindzh/tidal-hifi/releases/tag/5.15.0">5.15.0</a> |               href="https://github.com/Mastermindzh/tidal-hifi/releases/tag/5.20.1">5.20.1</a> | ||||||
|           </div> |           </div> | ||||||
|           <div class="about-section__links"> |           <div class="about-section__links"> | ||||||
|             <a target="_blank" rel="noopener" href="https://github.com/mastermindzh/tidal-hifi/" |             <a target="_blank" rel="noopener" href="https://github.com/mastermindzh/tidal-hifi/" | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ import { | |||||||
| } from "./features/listenbrainz/listenbrainz"; | } from "./features/listenbrainz/listenbrainz"; | ||||||
| import { StoreData } from "./features/listenbrainz/models/storeData"; | import { StoreData } from "./features/listenbrainz/models/storeData"; | ||||||
| import { Logger } from "./features/logger"; | import { Logger } from "./features/logger"; | ||||||
| import { Songwhip } from "./features/songwhip/songwhip"; | import { SharingService } from "./features/sharingService/sharingService"; | ||||||
| import { addCustomCss } from "./features/theming/theming"; | import { addCustomCss } from "./features/theming/theming"; | ||||||
| import { convertDurationToSeconds } from "./features/time/parse"; | import { convertDurationToSeconds } from "./features/time/parse"; | ||||||
| import { MediaInfo } from "./models/mediaInfo"; | import { MediaInfo } from "./models/mediaInfo"; | ||||||
| @@ -48,17 +48,18 @@ const elements = { | |||||||
|   search: '[class^="searchField"]', |   search: '[class^="searchField"]', | ||||||
|   shuffle: '*[data-test="shuffle"]', |   shuffle: '*[data-test="shuffle"]', | ||||||
|   repeat: '*[data-test="repeat"]', |   repeat: '*[data-test="repeat"]', | ||||||
|   account: '*[class^="profileOptions"]', |   account: '*[data-test^="profile-image-button"]', | ||||||
|   settings: '*[data-test^="open-settings"]', |   settings: '*[data-test^="sidebar-menu-button"]', | ||||||
|  |   openSettings: '*[data-test^="open-settings"]', | ||||||
|   media: '*[data-test="current-media-imagery"]', |   media: '*[data-test="current-media-imagery"]', | ||||||
|   image: "img", |   image: "img", | ||||||
|   current: '*[data-test="current-time"]', |   current: '*[data-test="current-time"]', | ||||||
|   duration: '*[class^=playbackControlsContainer] *[data-test="duration"]', |   duration: '*[class^=_playbackControlsContainer] *[data-test="duration"]', | ||||||
|   bar: '*[data-test="progress-bar"]', |   bar: '*[data-test="progress-bar"]', | ||||||
|   footer: "#footerPlayer", |   footer: "#footerPlayer", | ||||||
|   mediaItem: "[data-type='mediaItem']", |   mediaItem: "[data-type='mediaItem']", | ||||||
|   album_header_title: '*[class^="playingFrom"] span:nth-child(2)', |   album_header_title: '*[class^="_playingFrom"] span:nth-child(2)', | ||||||
|   playing_from: '*[class^="playingFrom"] span:nth-child(2)', |   playing_from: '*[class^="_playingFrom"] span:nth-child(2)', | ||||||
|   queue_album: "*[class^=playQueueItemsContainer] *[class^=groupTitle] span:nth-child(2)", |   queue_album: "*[class^=playQueueItemsContainer] *[class^=groupTitle] span:nth-child(2)", | ||||||
|   currentlyPlaying: "[class^='isPlayingIcon'], [data-test-is-playing='true']", |   currentlyPlaying: "[class^='isPlayingIcon'], [data-test-is-playing='true']", | ||||||
|   album_name_cell: '[class^="album"]', |   album_name_cell: '[class^="album"]', | ||||||
| @@ -114,18 +115,19 @@ const elements = { | |||||||
|   }, |   }, | ||||||
|  |  | ||||||
|   getAlbumName: function () { |   getAlbumName: function () { | ||||||
|  |     try { | ||||||
|       //If listening to an album, get its name from the header title |       //If listening to an album, get its name from the header title | ||||||
|     if (window.location.href.includes("/album/")) { |       if (globalThis.location.href.includes("/album/")) { | ||||||
|       const albumName = window.document.querySelector(this.album_header_title); |         const albumName = globalThis.document.querySelector(this.album_header_title); | ||||||
|         if (albumName) { |         if (albumName) { | ||||||
|           return albumName.textContent; |           return albumName.textContent; | ||||||
|         } |         } | ||||||
|         //If listening to a playlist or a mix, get album name from the list |         //If listening to a playlist or a mix, get album name from the list | ||||||
|       } else if ( |       } else if ( | ||||||
|       window.location.href.includes("/playlist/") || |         globalThis.location.href.includes("/playlist/") || | ||||||
|       window.location.href.includes("/mix/") |         globalThis.location.href.includes("/mix/") | ||||||
|       ) { |       ) { | ||||||
|       if (currentPlayStatus === MediaStatus.playing) { |         if (this.currentlyPlaying === MediaStatus.playing) { | ||||||
|           // find the currently playing element from the list (which might be in an album icon), traverse back up to the mediaItem (row) and select the album cell. |           // find the currently playing element from the list (which might be in an album icon), traverse back up to the mediaItem (row) and select the album cell. | ||||||
|           // document.querySelector("[class^='isPlayingIcon'], [data-test-is-playing='true']").closest('[data-type="mediaItem"]').querySelector('[class^="album"]').textContent |           // document.querySelector("[class^='isPlayingIcon'], [data-test-is-playing='true']").closest('[data-type="mediaItem"]').querySelector('[class^="album"]').textContent | ||||||
|           const row = window.document.querySelector(this.currentlyPlaying).closest(this.mediaItem); |           const row = window.document.querySelector(this.currentlyPlaying).closest(this.mediaItem); | ||||||
| @@ -136,12 +138,15 @@ const elements = { | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       // see whether we're on the queue page and get it from there |       // see whether we're on the queue page and get it from there | ||||||
|     const queueAlbumName = elements.getText("queue_album"); |       const queueAlbumName = this.getText("queue_album"); | ||||||
|       if (queueAlbumName) { |       if (queueAlbumName) { | ||||||
|         return queueAlbumName; |         return queueAlbumName; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       return ""; |       return ""; | ||||||
|  |     } catch { | ||||||
|  |       return ""; | ||||||
|  |     } | ||||||
|   }, |   }, | ||||||
|  |  | ||||||
|   isMuted: function () { |   isMuted: function () { | ||||||
| @@ -220,9 +225,9 @@ ListenBrainzStore.clear(); | |||||||
| function addHotKeys() { | function addHotKeys() { | ||||||
|   if (settingsStore.get(settings.enableCustomHotkeys)) { |   if (settingsStore.get(settings.enableCustomHotkeys)) { | ||||||
|     addHotkey("Control+p", function () { |     addHotkey("Control+p", function () { | ||||||
|       elements.click("account"); |  | ||||||
|       setTimeout(() => { |  | ||||||
|       elements.click("settings"); |       elements.click("settings"); | ||||||
|  |       setTimeout(() => { | ||||||
|  |         elements.click("openSettings"); | ||||||
|       }, 100); |       }, 100); | ||||||
|     }); |     }); | ||||||
|     addHotkey("Control+l", function () { |     addHotkey("Control+l", function () { | ||||||
| @@ -254,11 +259,10 @@ function addHotKeys() { | |||||||
|       elements.click("repeat"); |       elements.click("repeat"); | ||||||
|     }); |     }); | ||||||
|     addHotkey("control+w", async function () { |     addHotkey("control+w", async function () { | ||||||
|       const result = await ipcRenderer.invoke(globalEvents.whip, getTrackURL()); |       const url = SharingService.getUniversalLink(getTrackURL()); | ||||||
|       const url = Songwhip.getWhipUrl(result); |  | ||||||
|       clipboard.writeText(url); |       clipboard.writeText(url); | ||||||
|       new Notification({ |       new Notification({ | ||||||
|         title: `Successfully whipped: `, |         title: `Universal link generated: `, | ||||||
|         body: `URL copied to clipboard: ${url}`, |         body: `URL copied to clipboard: ${url}`, | ||||||
|       }).show(); |       }).show(); | ||||||
|     }); |     }); | ||||||
| @@ -394,8 +398,24 @@ function updateMediaInfo(mediaInfo: MediaInfo, notify: boolean) { | |||||||
|   if (mediaInfo) { |   if (mediaInfo) { | ||||||
|     currentMediaInfo = mediaInfo; |     currentMediaInfo = mediaInfo; | ||||||
|     ipcRenderer.send(globalEvents.updateInfo, mediaInfo); |     ipcRenderer.send(globalEvents.updateInfo, mediaInfo); | ||||||
|     if (settingsStore.get(settings.notifications) && notify) { |     updateMpris(mediaInfo); | ||||||
|       if (currentNotification) currentNotification.close(); |     updateListenBrainz(mediaInfo); | ||||||
|  |     if (notify) { | ||||||
|  |       sendNotification(mediaInfo); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * send a desktop notification if enabled in settings | ||||||
|  |  * @param mediaInfo | ||||||
|  |  * @param notify Whether to notify | ||||||
|  |  */ | ||||||
|  | async function sendNotification(mediaInfo: MediaInfo) { | ||||||
|  |   if (settingsStore.get(settings.notifications)) { | ||||||
|  |     if (currentNotification) { | ||||||
|  |       currentNotification.close(); | ||||||
|  |     } | ||||||
|     currentNotification = new Notification({ |     currentNotification = new Notification({ | ||||||
|       title: mediaInfo.title, |       title: mediaInfo.title, | ||||||
|       body: mediaInfo.artists, |       body: mediaInfo.artists, | ||||||
| @@ -403,10 +423,6 @@ function updateMediaInfo(mediaInfo: MediaInfo, notify: boolean) { | |||||||
|     }); |     }); | ||||||
|     currentNotification.show(); |     currentNotification.show(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|     updateMpris(mediaInfo); |  | ||||||
|     updateListenBrainz(mediaInfo); |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  |  | ||||||
| function addMPRIS() { | function addMPRIS() { | ||||||
| @@ -471,6 +487,7 @@ function updateMpris(mediaInfo: MediaInfo) { | |||||||
|         "xesam:title": mediaInfo.title, |         "xesam:title": mediaInfo.title, | ||||||
|         "xesam:artist": [mediaInfo.artists], |         "xesam:artist": [mediaInfo.artists], | ||||||
|         "xesam:album": mediaInfo.album, |         "xesam:album": mediaInfo.album, | ||||||
|  |         "xesam:url": mediaInfo.url, | ||||||
|         "mpris:artUrl": mediaInfo.image, |         "mpris:artUrl": mediaInfo.image, | ||||||
|         "mpris:length": convertDuration(mediaInfo.duration) * 1000 * 1000, |         "mpris:length": convertDuration(mediaInfo.duration) * 1000 * 1000, | ||||||
|         "mpris:trackid": "/org/mpris/MediaPlayer2/track/" + getTrackID(), |         "mpris:trackid": "/org/mpris/MediaPlayer2/track/" + getTrackID(), | ||||||
| @@ -538,6 +555,7 @@ setInterval(function () { | |||||||
|   const artistsArray = elements.getArtistsArray(); |   const artistsArray = elements.getArtistsArray(); | ||||||
|   const artistsString = elements.getArtistsString(artistsArray); |   const artistsString = elements.getArtistsString(artistsArray); | ||||||
|   const songDashArtistTitle = `${title} - ${artistsString}`; |   const songDashArtistTitle = `${title} - ${artistsString}`; | ||||||
|  |   const staticTitle = "TIDAL Hi-Fi"; | ||||||
|   const titleOrArtistsChanged = currentSong !== songDashArtistTitle; |   const titleOrArtistsChanged = currentSong !== songDashArtistTitle; | ||||||
|   const current = elements.getText("current"); |   const current = elements.getText("current"); | ||||||
|   const currentStatus = getCurrentlyPlayingStatus(); |   const currentStatus = getCurrentlyPlayingStatus(); | ||||||
| @@ -582,7 +600,9 @@ setInterval(function () { | |||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     // update title, url and play info with new info |     // update title, url and play info with new info | ||||||
|     setTitle(songDashArtistTitle); |     settingsStore.get(settings.staticWindowTitle) | ||||||
|  |       ? setTitle(staticTitle) | ||||||
|  |       : setTitle(songDashArtistTitle); | ||||||
|     getTrackURL(); |     getTrackURL(); | ||||||
|     currentSong = songDashArtistTitle; |     currentSong = songDashArtistTitle; | ||||||
|     currentPlayStatus = currentStatus; |     currentPlayStatus = currentStatus; | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { Client, Presence } from "discord-rpc"; | import { Client, SetActivity } from "@xhayper/discord-rpc"; | ||||||
| import { app, ipcMain } from "electron"; | import { app, ipcMain } from "electron"; | ||||||
| import { globalEvents } from "../constants/globalEvents"; | import { globalEvents } from "../constants/globalEvents"; | ||||||
| import { settings } from "../constants/settings"; | import { settings } from "../constants/settings"; | ||||||
| @@ -12,9 +12,13 @@ const clientId = "833617820704440341"; | |||||||
|  |  | ||||||
| export let rpc: Client; | export let rpc: Client; | ||||||
|  |  | ||||||
|  | const ACTIVITY_LISTENING = 2; | ||||||
|  | const MAX_RETRIES = 5; | ||||||
|  | const RETRY_DELAY = 10000; | ||||||
|  |  | ||||||
| const observer = () => { | const observer = () => { | ||||||
|   if (rpc) { |   if (rpc) { | ||||||
|     rpc.setActivity(getActivity()); |     updateActivity(); | ||||||
|   } |   } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -22,10 +26,20 @@ const defaultPresence = { | |||||||
|   largeImageKey: "tidal-hifi-icon", |   largeImageKey: "tidal-hifi-icon", | ||||||
|   largeImageText: `TIDAL Hi-Fi ${app.getVersion()}`, |   largeImageText: `TIDAL Hi-Fi ${app.getVersion()}`, | ||||||
|   instance: false, |   instance: false, | ||||||
|  |   type: ACTIVITY_LISTENING | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const getActivity = (): Presence => { | const updateActivity = () => { | ||||||
|   const presence: Presence = { ...defaultPresence }; |   const showIdle = settingsStore.get<string, boolean>(settings.discord.showIdle) ?? true; | ||||||
|  |   if (mediaInfo.status === MediaStatus.paused && !showIdle) { | ||||||
|  |     rpc.user?.clearActivity(); | ||||||
|  |   } else { | ||||||
|  |     rpc.user?.setActivity(getActivity()); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const getActivity = (): SetActivity => { | ||||||
|  |   const presence: SetActivity = { ...defaultPresence }; | ||||||
|  |  | ||||||
|   if (mediaInfo.status === MediaStatus.paused) { |   if (mediaInfo.status === MediaStatus.paused) { | ||||||
|     presence.details = |     presence.details = | ||||||
| @@ -41,6 +55,7 @@ const getActivity = (): Presence => { | |||||||
|         settingsStore.get<string, string>(settings.discord.usingText) ?? "Playing media on TIDAL"; |         settingsStore.get<string, string>(settings.discord.usingText) ?? "Playing media on TIDAL"; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return presence; |   return presence; | ||||||
|  |  | ||||||
|   function getFromStore() { |   function getFromStore() { | ||||||
| @@ -54,18 +69,33 @@ const getActivity = (): Presence => { | |||||||
|     return { includeTimestamps, detailsPrefix, buttonText }; |     return { includeTimestamps, detailsPrefix, buttonText }; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   function setPresenceFromMediaInfo(detailsPrefix: string, buttonText: string) { |   /** | ||||||
|     if (mediaInfo.url) { |    * Pad a string using spaces to at least 2 characters | ||||||
|       presence.details = `${detailsPrefix}${mediaInfo.title}`; |    * @param input string to pad with 2 characters | ||||||
|       presence.state = mediaInfo.artists ? mediaInfo.artists : "unknown artist(s)"; |    * @returns | ||||||
|       presence.largeImageKey = mediaInfo.image; |    */ | ||||||
|       if (mediaInfo.album) { |   function pad(input: string): string { | ||||||
|         presence.largeImageText = mediaInfo.album; |     return input.padEnd(2, " "); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   function setPresenceFromMediaInfo(detailsPrefix: string, buttonText: string) { | ||||||
|  |     // discord requires a minimum of 2 characters | ||||||
|  |     const title = pad(mediaInfo.title); | ||||||
|  |     const album = pad(mediaInfo.album); | ||||||
|  |     const artists = pad(mediaInfo.artists); | ||||||
|  |  | ||||||
|  |     if (mediaInfo.url) { | ||||||
|  |       presence.details = `${detailsPrefix}${title}`; | ||||||
|  |       presence.state = artists ? artists : "unknown artist(s)"; | ||||||
|  |       presence.largeImageKey = mediaInfo.image; | ||||||
|  |       if (album) { | ||||||
|  |         presence.largeImageText = album; | ||||||
|  |       } | ||||||
|  |  | ||||||
|       presence.buttons = [{ label: buttonText, url: mediaInfo.url }]; |       presence.buttons = [{ label: buttonText, url: mediaInfo.url }]; | ||||||
|     } else { |     } else { | ||||||
|       presence.details = `Watching ${mediaInfo.title}`; |       presence.details = `Watching ${title}`; | ||||||
|       presence.state = mediaInfo.artists; |       presence.state = artists; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -73,11 +103,29 @@ const getActivity = (): Presence => { | |||||||
|     if (includeTimestamps) { |     if (includeTimestamps) { | ||||||
|       const currentSeconds = convertDurationToSeconds(mediaInfo.current); |       const currentSeconds = convertDurationToSeconds(mediaInfo.current); | ||||||
|       const durationSeconds = convertDurationToSeconds(mediaInfo.duration); |       const durationSeconds = convertDurationToSeconds(mediaInfo.duration); | ||||||
|       const date = new Date(); |       const now = Math.trunc((Date.now() + 500) / 1000); | ||||||
|       const now = (date.getTime() / 1000) | 0; |       presence.startTimestamp = now - currentSeconds; | ||||||
|       const remaining = date.setSeconds(date.getSeconds() + (durationSeconds - currentSeconds)); |       presence.endTimestamp = presence.startTimestamp + durationSeconds; | ||||||
|       presence.startTimestamp = now; |     } | ||||||
|       presence.endTimestamp = remaining; |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Try to login to RPC and retry if it errors | ||||||
|  |  * @param retryCount Max retry count | ||||||
|  |  */ | ||||||
|  | const connectWithRetry = async (retryCount = 0) => { | ||||||
|  |   try { | ||||||
|  |     await rpc.login(); | ||||||
|  |     Logger.log('Connected to Discord'); | ||||||
|  |     rpc.on("ready", updateActivity); | ||||||
|  |     Object.values(globalEvents).forEach(event => ipcMain.on(event, observer)); | ||||||
|  |   } catch (error) { | ||||||
|  |     if (retryCount < MAX_RETRIES) { | ||||||
|  |       Logger.log(`Failed to connect to Discord, retrying in ${RETRY_DELAY/1000} seconds... (Attempt ${retryCount + 1}/${MAX_RETRIES})`); | ||||||
|  |       setTimeout(() => connectWithRetry(retryCount + 1), RETRY_DELAY); | ||||||
|  |     } else { | ||||||
|  |       Logger.log('Failed to connect to Discord after maximum retry attempts'); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| }; | }; | ||||||
| @@ -86,18 +134,8 @@ const getActivity = (): Presence => { | |||||||
|  * Set up the discord rpc and listen on globalEvents.updateInfo |  * Set up the discord rpc and listen on globalEvents.updateInfo | ||||||
|  */ |  */ | ||||||
| export const initRPC = () => { | export const initRPC = () => { | ||||||
|   rpc = new Client({ transport: "ipc" }); |   rpc = new Client({ transport: {type: "ipc"}, clientId }); | ||||||
|   rpc.login({ clientId }).then( |   connectWithRetry(); | ||||||
|     () => { |  | ||||||
|       rpc.on("ready", () => { |  | ||||||
|         rpc.setActivity(getActivity()); |  | ||||||
|       }); |  | ||||||
|       ipcMain.on(globalEvents.updateInfo, observer); |  | ||||||
|     }, |  | ||||||
|     () => { |  | ||||||
|       Logger.log("Can't connect to Discord, is it running?"); |  | ||||||
|     } |  | ||||||
|   ); |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -105,7 +143,7 @@ export const initRPC = () => { | |||||||
|  */ |  */ | ||||||
| export const unRPC = () => { | export const unRPC = () => { | ||||||
|   if (rpc) { |   if (rpc) { | ||||||
|     rpc.clearActivity(); |     rpc.user?.clearActivity(); | ||||||
|     rpc.destroy(); |     rpc.destroy(); | ||||||
|     rpc = null; |     rpc = null; | ||||||
|     ipcMain.removeListener(globalEvents.updateInfo, observer); |     ipcMain.removeListener(globalEvents.updateInfo, observer); | ||||||
|   | |||||||
| @@ -43,6 +43,7 @@ export const settingsStore = new Store({ | |||||||
|     enableDiscord: false, |     enableDiscord: false, | ||||||
|     discord: { |     discord: { | ||||||
|       showSong: true, |       showSong: true, | ||||||
|  |       showIdle: true, | ||||||
|       idleText: "Browsing Tidal", |       idleText: "Browsing Tidal", | ||||||
|       usingText: "Playing media on TIDAL", |       usingText: "Playing media on TIDAL", | ||||||
|       includeTimestamps: true, |       includeTimestamps: true, | ||||||
| @@ -68,6 +69,7 @@ export const settingsStore = new Store({ | |||||||
|     singleInstance: true, |     singleInstance: true, | ||||||
|     skipArtists: false, |     skipArtists: false, | ||||||
|     skippedArtists: [""], |     skippedArtists: [""], | ||||||
|  |     staticWindowTitle: false, | ||||||
|     theme: "none", |     theme: "none", | ||||||
|     trayIcon: true, |     trayIcon: true, | ||||||
|     updateFrequency: 500, |     updateFrequency: 500, | ||||||
| @@ -115,6 +117,9 @@ export const settingsStore = new Store({ | |||||||
|         { key: settings.advanced.tidalUrl, value: "https://listen.tidal.com" }, |         { key: settings.advanced.tidalUrl, value: "https://listen.tidal.com" }, | ||||||
|       ]); |       ]); | ||||||
|     }, |     }, | ||||||
|  |     "5.16.0": (migrationStore) => { | ||||||
|  |       buildMigration("5.16.0", migrationStore, [{ key: settings.discord.showIdle, value: "true" }]); | ||||||
|  |     }, | ||||||
|   }, |   }, | ||||||
| }); | }); | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user