Compare commits
	
		
			127 Commits
		
	
	
		
			2.1.0
			...
			62244f432a
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 62244f432a | |||
| 2c5d2b9530 | |||
| d823f07ed8 | |||
| 53e4711c39 | |||
| e8509d42e7 | |||
| 46d030cf8e | |||
| 412f1ae3e3 | |||
| 68f0c89ec2 | |||
| 8d44ff8afb | |||
| bccc979f43 | |||
| 6849952c41 | |||
| 07be74af9f | |||
| fc6adc25ca | |||
| 4498e8a73e | |||
| 3d2a9c3992 | |||
| af6bfaf55e | |||
|  | 8bac90e0f1 | ||
|  | 887c75f61a | ||
|  | cde7408cc4 | ||
| 05b422e045 | |||
| 35289d8216 | |||
| ea42b79cd8 | |||
| 6d859cf780 | |||
| af20092053 | |||
| 166ca353cf | |||
| b807aa2f76 | |||
|  | ef8ffe47f5 | ||
| ba7b2a5717 | |||
| 0d93bedb4d | |||
|  | 1de71aa82b | ||
|  | b2e68f5a8f | ||
|  | a2a2023853 | ||
|  | 26c8a38350 | ||
| eb93fbc35d | |||
|  | d3c56fa445 | ||
| 6998992011 | |||
| 108e1d65d4 | |||
| 1097f83911 | |||
| 8c734777cc | |||
|  | de17ac6113 | ||
|  | ced41c00d7 | ||
|  | 744016f307 | ||
|  | ad8ef71c6b | ||
| d0f9a34f9c | |||
| 3b316f2301 | |||
| c0d9cd2834 | |||
| 0620d87d8b | |||
| 57b7f9148f | |||
| 63ccff97ea | |||
| 3a4d23738f | |||
| c96bdb0d28 | |||
|  | 115d8c6c5c | ||
|  | cd2a068470 | ||
| bf260b14e0 | |||
| d161a68c95 | |||
|  | 9de8cea50e | ||
| 5f330a7c48 | |||
| 732710c3ef | |||
| 4941aae950 | |||
| 1439a11969 | |||
|  | 3a3e0e1a2d | ||
| fa9ab22867 | |||
| 207a61d199 | |||
|  | 7b18322e17 | ||
| 8f47756244 | |||
|  | cdcf9431bf | ||
| 374f3da740 | |||
| 3965ada0a2 | |||
| 79ff02d06c | |||
| 7b2afd2290 | |||
| 5fde20ace1 | |||
| 7f5f5e7f62 | |||
| 6a1a1efe74 | |||
| 94e1bb1780 | |||
| d66dd8cc9e | |||
|  | de97ac8a00 | ||
|  | 82ac5edf22 | ||
|  | 909c8ee8ba | ||
|  | 15b6b13e14 | ||
| 89589b75e1 | |||
| 6a7b3eefd4 | |||
|  | 0583c4a188 | ||
|  | 8855e7f89f | ||
|  | 4fbb598c50 | ||
| 101fe967d3 | |||
|  | 53468e0dc3 | ||
| 53cecbcd18 | |||
| 5a65f60cc5 | |||
| d51d5cdc24 | |||
| 0dec967e71 | |||
|  | c940d0991d | ||
| 662ef6ad7b | |||
| 5313ab13d3 | |||
| f43f227191 | |||
|  | ae25d88e94 | ||
| 5ef6074015 | |||
| 8fea5265e7 | |||
| 8d2e03ca6b | |||
| 1074de228b | |||
|  | 64d1aa4041 | ||
|  | 6608330ed3 | ||
| aa562c4a30 | |||
| d6f63ac560 | |||
| cdc0f49789 | |||
| 2f290f83fd | |||
| d34ddfeb75 | |||
| b7f163c1a1 | |||
|  | 791a92a446 | ||
|  | 10c1e57680 | ||
|  | c65d1a56c8 | ||
|  | d8f2dbd0c2 | ||
|  | 43ce85bb28 | ||
|  | 8201e23e4b | ||
|  | 3cc288e014 | ||
|  | 08ec7fadac | ||
|  | 7a30b125ec | ||
|  | cac5db123f | ||
|  | 59f8b2d0b5 | ||
|  | 81a536bbdb | ||
|  | 5e952e3899 | ||
|  | df887b8628 | ||
|  | 31d90a342c | ||
|  | 8607337580 | ||
| 4fe42a3671 | |||
| 148d1746ad | |||
| ae51f9610c | |||
| 5eb3b8d95f | 
							
								
								
									
										12
									
								
								.eslintrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | |||||||
|  | { | ||||||
|  |   "root": true, | ||||||
|  |   "parser": "@typescript-eslint/parser", | ||||||
|  |   "plugins": [ | ||||||
|  |     "@typescript-eslint" | ||||||
|  |   ], | ||||||
|  |   "extends": [ | ||||||
|  |     "eslint:recommended", | ||||||
|  |     "plugin:@typescript-eslint/eslint-recommended", | ||||||
|  |     "plugin:@typescript-eslint/recommended" | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										12
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -9,20 +9,24 @@ jobs: | |||||||
|   build_on_linux: |   build_on_linux: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|  |       - name: update apt | ||||||
|  |         run: sudo apt-get update | ||||||
|  |       - name: Install libarchive-tools | ||||||
|  |         run: sudo apt-get install -y libarchive-tools | ||||||
|       - uses: actions/checkout@master |       - uses: actions/checkout@master | ||||||
|       - uses: actions/setup-node@master |       - uses: actions/setup-node@master | ||||||
|         with: |         with: | ||||||
|           node-version: 12 |           node-version: 19 | ||||||
|       - run: npm install |       - run: npm install | ||||||
|       - run: npm run build |       - run: npm run build | ||||||
|  |  | ||||||
|   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: 12 |           node-version: 19 | ||||||
|       - run: npm install |       - run: npm install | ||||||
|       - run: npm run build |       - run: npm run build | ||||||
|  |  | ||||||
| @@ -32,6 +36,6 @@ jobs: | |||||||
|       - uses: actions/checkout@master |       - uses: actions/checkout@master | ||||||
|       - uses: actions/setup-node@master |       - uses: actions/setup-node@master | ||||||
|         with: |         with: | ||||||
|           node-version: 12 |           node-version: 19 | ||||||
|       - run: npm install |       - run: npm install | ||||||
|       - run: npm run build |       - run: npm run build | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -5,14 +5,22 @@ on: | |||||||
|     branches: |     branches: | ||||||
|       - master |       - master | ||||||
|       - develop |       - develop | ||||||
|  |   pull_request: | ||||||
|  |     branches: | ||||||
|  |       - master | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build_on_linux: |   build_on_linux: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|  |       - name: update apt | ||||||
|  |         run: sudo apt-get update | ||||||
|  |       - name: Install libarchive-tools | ||||||
|  |         run: sudo apt-get install -y libarchive-tools | ||||||
|       - uses: actions/checkout@master |       - uses: actions/checkout@master | ||||||
|       - uses: actions/setup-node@master |       - uses: actions/setup-node@master | ||||||
|         with: |         with: | ||||||
|           node-version: 12 |           node-version: 19 | ||||||
|       - run: npm install |       - run: npm install | ||||||
|       - run: npm run build |       - run: npm run build | ||||||
|       - uses: actions/upload-artifact@master |       - uses: actions/upload-artifact@master | ||||||
| @@ -21,12 +29,12 @@ jobs: | |||||||
|           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: 12 |           node-version: 19 | ||||||
|       - run: npm install |       - run: npm install | ||||||
|       - run: npm run build |       - run: npm run build | ||||||
|       - uses: actions/upload-artifact@master |       - uses: actions/upload-artifact@master | ||||||
| @@ -40,7 +48,7 @@ jobs: | |||||||
|       - uses: actions/checkout@master |       - uses: actions/checkout@master | ||||||
|       - uses: actions/setup-node@master |       - uses: actions/setup-node@master | ||||||
|         with: |         with: | ||||||
|           node-version: 12 |           node-version: 19 | ||||||
|       - run: npm install |       - run: npm install | ||||||
|       - run: npm run build |       - run: npm run build | ||||||
|       - uses: actions/upload-artifact@master |       - uses: actions/upload-artifact@master | ||||||
|   | |||||||
							
								
								
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -7,3 +7,10 @@ build/linux/arch/* | |||||||
| !build/linux/arch/.SRCINFO | !build/linux/arch/.SRCINFO | ||||||
| !build/linux/arch/tidal-hifi.desktop | !build/linux/arch/tidal-hifi.desktop | ||||||
| !build/linux/arch/install.sh | !build/linux/arch/install.sh | ||||||
|  | *.css | ||||||
|  | *.css.map | ||||||
|  |  | ||||||
|  | # JetBrains IDE configuration | ||||||
|  | .idea | ||||||
|  | ts-dist/** | ||||||
|  | ts-dist | ||||||
|   | |||||||
							
								
								
									
										13
									
								
								.stylelintrc.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,13 @@ | |||||||
|  | { | ||||||
|  |   "plugins": [ | ||||||
|  |     "stylelint-prettier" | ||||||
|  |   ], | ||||||
|  |   "extends": [ | ||||||
|  |     "stylelint-config-standard-scss" | ||||||
|  |   ], | ||||||
|  |   "rules": { | ||||||
|  |     "prettier/prettier": true, | ||||||
|  |     "scss/at-extend-no-missing-placeholder": null, | ||||||
|  |     "no-descending-specificity": null | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										13
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,13 @@ | |||||||
|  | { | ||||||
|  |   "cSpell.words": [ | ||||||
|  |     "flac", | ||||||
|  |     "geqnfr", | ||||||
|  |     "hifi", | ||||||
|  |     "playpause", | ||||||
|  |     "rescrobbler", | ||||||
|  |     "trackid", | ||||||
|  |     "tracklist", | ||||||
|  |     "widevine", | ||||||
|  |     "xesam" | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										184
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						| @@ -4,6 +4,186 @@ 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.2.0 | ||||||
|  |  | ||||||
|  | - moved from Javascript to Typescript for all files | ||||||
|  |   - use `npm run watch` to watch for changes & recompile typescript and sass files | ||||||
|  |  | ||||||
|  | ## 5.1.0 | ||||||
|  |  | ||||||
|  | ### New features | ||||||
|  |  | ||||||
|  | - Added proper updates through the MediaSession API | ||||||
|  | - You can now add custom CSS in the "advanced" settings tab | ||||||
|  | - You can now configure the updateFrequency in the settings window | ||||||
|  |   - Default value is set to 500 and will overwrite the hardcoded value of 100 | ||||||
|  |  | ||||||
|  | ### Fixes | ||||||
|  |  | ||||||
|  | - Any songs **including** an artist listed in the `skipped artists` setting will now be skipped even if the song is a collaboration. | ||||||
|  | - Linux desktop icons have been fixed. See [#222](https://github.com/Mastermindzh/tidal-hifi/pull/222) for details. | ||||||
|  |  | ||||||
|  | ## 5.0.0 | ||||||
|  |  | ||||||
|  | - Replaced "muting artists" with a full implementation of an Adblock mechanism | ||||||
|  |  | ||||||
|  |   > Disabled audio & visual ads, unlocked lyrics, suggested track, track info, unlimited skips thanks to uBlockOrigin custom filters ([source](https://github.com/uBlockOrigin/uAssets/issues/17495)) | ||||||
|  |  | ||||||
|  | - @thanasistrisp updated Electron to 24.1.2 and fixed the tray bug :) | ||||||
|  |  | ||||||
|  | ## 4.4.0 | ||||||
|  |  | ||||||
|  | - Updated shortcut hint on the menubar to reflect the new `ctrl+=` shortcut. | ||||||
|  | - Reverted icon path to `icon.png` instead of the hardcoded linux path. | ||||||
|  | - Add support to autoHide the menubar and showing it with the `alt` key. | ||||||
|  | - Move the quit command from the system sub-menu to the main menu | ||||||
|  | - Added single click focus/show on the tray icon | ||||||
|  |   - Doesn't work on all platforms. Nothing I can do about that unfortunately! | ||||||
|  | - Added a list of artists to automatically skip. | ||||||
|  |   - I don't like the vast majority of dutch music so I added one of them to my list to test: [./docs/no-dutch-music.mp4](./docs/no-dutch-music.mp4) | ||||||
|  |  | ||||||
|  | ## 4.3.1 | ||||||
|  |  | ||||||
|  | - fix: App always requests a default-url-handler-scheme change on start | ||||||
|  |  | ||||||
|  | ## 4.3.0 | ||||||
|  |  | ||||||
|  | - Added a setting to disable background throttling ([docs](https://www.electronjs.org/docs/latest/api/browser-window)) | ||||||
|  |  | ||||||
|  | ## 4.2.0 | ||||||
|  |  | ||||||
|  | - New settings window by BlueManCZ | ||||||
|  | - Fixed the desktop files in electron-builder | ||||||
|  |   - icon is set to new static path based on Arch/Debian | ||||||
|  |   - Name has changed to Tidal-Hifi | ||||||
|  |  | ||||||
|  | ## 4.1.2 | ||||||
|  |  | ||||||
|  | - Changed the category of the desktop file to AudioVideo | ||||||
|  | - Changed desktop file name to "TIDAL Hi-Fi" | ||||||
|  |  | ||||||
|  | ## 4.1.1 | ||||||
|  |  | ||||||
|  | - Fixed `cannot read property of undefined` error because of not passing mainWindow around. | ||||||
|  | - vincens2005, fixed inconsistent auto muting | ||||||
|  |  | ||||||
|  | ## 4.1.0 | ||||||
|  |  | ||||||
|  | - Added `tidal://` protocol support | ||||||
|  | - Switched icon strategies to fix bugs with icons | ||||||
|  | - Fixed tray icon bugs | ||||||
|  |   - Menu now shows in KDE as well | ||||||
|  |   - Toggle window is supported from tray icon | ||||||
|  |   - regular click is still ignored, see [this issue](https://github.com/electron/electron/issues/6773) | ||||||
|  | - Fixed about tab not showing | ||||||
|  | - Fixed playback, mpris and API issues | ||||||
|  |  | ||||||
|  | ## 4.0.1 | ||||||
|  |  | ||||||
|  | - Updated build config to make use of a base file that doesn't build anything. | ||||||
|  |   - This fixes the issue of unwanted extra build targets that were introduced with the electron-builder update | ||||||
|  |  | ||||||
|  | ## 4.0.0 | ||||||
|  |  | ||||||
|  | - Updated to Electron 19.0.5 | ||||||
|  |  | ||||||
|  | ## 3.1.1 | ||||||
|  |  | ||||||
|  | - Media update timeout set to 500 instead of 200 | ||||||
|  | - Updated property name for duration because of a tidal update | ||||||
|  | - flag for "disable hardware media keys" now working again | ||||||
|  |  | ||||||
|  | ## 3.1.0 | ||||||
|  |  | ||||||
|  | - Added a separate advanced options settings panel with flags | ||||||
|  |   - Added gpu-rasterization flag | ||||||
|  | - config setting `disableHardwareMediaKeys` moved to `flags.disableHardwareMediaKeys`, it will be migrated automatically | ||||||
|  |  | ||||||
|  | ## 3.0.0 | ||||||
|  |  | ||||||
|  | - Updated to Electron 15 | ||||||
|  | - Fixed the develop "build-unpacked" command | ||||||
|  | - Added setting to disable multiple tidal-hifi windows (defaults to true) | ||||||
|  | - Added setting to disable HardwareMediaKeyHandling (defaults to false) | ||||||
|  |  | ||||||
|  | ## 2.8.2 | ||||||
|  |  | ||||||
|  | - Updated dependencies | ||||||
|  | - Downgraded packaged version of electron to 8.5.2, doesn't seem to like a newer build | ||||||
|  | - Fixed the annoying (and useless) terminal warning about `allowRendererProcessReuse` | ||||||
|  |  | ||||||
|  | ## 2.8.1 | ||||||
|  |  | ||||||
|  | - Mar0xy fixed some build issues (thanks!) | ||||||
|  | - vincens2005 fixed the quit button in the menubar | ||||||
|  |  | ||||||
|  | ## 2.8.0 | ||||||
|  |  | ||||||
|  | - Added the ability to mute artists automatically | ||||||
|  | - Added better error handling for discord rpc | ||||||
|  |  | ||||||
|  | ## 2.7.2 | ||||||
|  |  | ||||||
|  | - Disabled sandboxing to fix a display compositor issue on Linux. | ||||||
|  |  | ||||||
|  | ## 2.7.1 | ||||||
|  |  | ||||||
|  | - Fixed bug: Triggering full screen from the Tidal web app would cause the menubar to be visible even if it was disabled in the settings | ||||||
|  |  | ||||||
|  | ## 2.7.0 | ||||||
|  |  | ||||||
|  | - Switched to the native Notifier (removed node-notifier) | ||||||
|  | - Album art now also has a name, based on [best effort](https://github.com/Mastermindzh/tidal-hifi/pull/88#pullrequestreview-840814847) | ||||||
|  |  | ||||||
|  | ## 2.6.0 | ||||||
|  |  | ||||||
|  | - Add album images to media info and discord | ||||||
|  |  | ||||||
|  | ## 2.5.0 | ||||||
|  |  | ||||||
|  | - Notify-send now correctly shows "Tidal HiFi" as the program name | ||||||
|  | - Updated dependencies (including electron itself) | ||||||
|  |  | ||||||
|  | ### known issues | ||||||
|  |  | ||||||
|  | - Requires older version of nodejs due to electron-builder (use lts/gallium) | ||||||
|  |  | ||||||
|  | ### builds | ||||||
|  |  | ||||||
|  | updated to nodejs 16 in actions | ||||||
|  |  | ||||||
|  | ## 2.4.0 | ||||||
|  |  | ||||||
|  | - Added more MPRIS settings | ||||||
|  | - Added instruction for rescrobbler to get last.fm working without sandbox mode | ||||||
|  |  | ||||||
|  | ## 2.3.0 | ||||||
|  |  | ||||||
|  | - Added a setting to minimize to tray on app close (off by default) | ||||||
|  | - Added the main menu to the tray icon | ||||||
|  |  | ||||||
|  | ## 2.2.1 | ||||||
|  |  | ||||||
|  | - artists is now gotten specifically from the footer. This fixes the [unknown artists bug](https://github.com/Mastermindzh/tidal-hifi/issues/45). | ||||||
|  | - the discord module will check whether the artists is empty and if so substitute it with a default message. This is to prevent sending an empty state to Discord (which it doesn't support). fixes [#45](https://github.com/Mastermindzh/tidal-hifi/issues/54) | ||||||
|  |  | ||||||
|  | ### removed arch build details from source control | ||||||
|  |  | ||||||
|  | moved to: [https://github.com/Mastermindzh/tidal-hifi-aur](https://github.com/Mastermindzh/tidal-hifi-aur) | ||||||
|  |  | ||||||
|  | ## 2.2.0 | ||||||
|  |  | ||||||
|  | - The discord integration now adds a time remaining field based on the song duration | ||||||
|  | - All fields (current, remaining, and url are also available in the API\*) | ||||||
|  | - the artist field is now correctly identified | ||||||
|  |  | ||||||
|  | \* current time only updates on play/pause. | ||||||
|  |  | ||||||
|  | ## 2.1.1 | ||||||
|  |  | ||||||
|  | - The discord integration now doesn't send an update every 15 seconds it sends an update whenever the media info changes | ||||||
|  | - consolidated updating the media info changes with the status changes into a single global event | ||||||
|  |  | ||||||
| ## 2.1.0 | ## 2.1.0 | ||||||
|  |  | ||||||
| - [Mar0xy](https://github.com/Mar0xy) added Discord integration. | - [Mar0xy](https://github.com/Mar0xy) added Discord integration. | ||||||
| @@ -22,7 +202,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 | |||||||
|  |  | ||||||
| ## 1.3.0 | ## 1.3.0 | ||||||
|  |  | ||||||
| -- re-enabled mpris-service wit the electron downloader fixes | -- re-enabled MPRIS-service wit the electron downloader fixes | ||||||
|  |  | ||||||
| ## 1.2.0 | ## 1.2.0 | ||||||
|  |  | ||||||
| @@ -37,7 +217,7 @@ Bugfixes: | |||||||
| ## 1.1.0 | ## 1.1.0 | ||||||
|  |  | ||||||
| - updated to electron 8.0.0 | - updated to electron 8.0.0 | ||||||
| - Added a beta-version of the mpris service | - Added a beta-version of the MPRIS service | ||||||
|  |  | ||||||
| - Bugfixes: | - Bugfixes: | ||||||
|   - icon on gnome not showing in launcher |   - icon on gnome not showing in launcher | ||||||
|   | |||||||
							
								
								
									
										88
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,51 +1,58 @@ | |||||||
| <h1> | # Tidal-hifi<img src = "./build/icon.png" height="40" align="right"/> | ||||||
| Tidal-hifi |  | ||||||
| <img src = "./build/icon.png" height="40" align="right" /> |  | ||||||
| </h1> |  | ||||||
|  |  | ||||||
| The web version of [listen.tidal.com](listen.tidal.com) running in electron with hifi support thanks to widevine. |  | ||||||
|  |  | ||||||
|  | The web version of [listen.tidal.com](https://listen.tidal.com) running in electron with hifi support thanks to widevine. | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Table of contents | ## Table of Contents | ||||||
|  |  | ||||||
| <!-- toc --> | <!-- toc --> | ||||||
|  |  | ||||||
| - [Installation](#installation) | - [Installation](#installation) | ||||||
|  |   - [Dependencies](#dependencies) | ||||||
|   - [Using releases](#using-releases) |   - [Using releases](#using-releases) | ||||||
|     - [Snap install](#snap-install) |   - [Snap](#snap) | ||||||
|   - [Arch Linux](#arch-linux) |   - [Arch Linux](#arch-linux) | ||||||
|  |   - [Flatpak](#flatpak) | ||||||
|  |   - [Nix](#nix) | ||||||
|   - [Using source](#using-source) |   - [Using source](#using-source) | ||||||
| - [features](#features) | - [Features](#features) | ||||||
| - [Integrations](#integrations) | - [Integrations](#integrations) | ||||||
|   - [Known bugs](#known-bugs) |   - [Known bugs](#known-bugs) | ||||||
|  |     - [last.fm doesn't work out of the box. Use rescrobbler as a workaround](#lastfm-doesnt-work-out-of-the-box-use-rescrobbler-as-a-workaround) | ||||||
| - [Why](#why) | - [Why](#why) | ||||||
| - [Why not extend existing projects?](#why-not-extend-existing-projects) | - [Why not extend existing projects?](#why-not-extend-existing-projects) | ||||||
| - [Special thanks to...](#special-thanks-to) | - [Special thanks to](#special-thanks-to) | ||||||
| - [Buy me a coffee? Please don't](#buy-me-a-coffee-please-dont) | - [Buy me a coffee? Please don't](#buy-me-a-coffee-please-dont) | ||||||
| - [Images](#images) | - [Images](#images) | ||||||
|   - [settings window](#settings-window) |   - [Settings window](#settings-window) | ||||||
|   - [user setups](#user-setups) |   - [User setups](#user-setups) | ||||||
|  |  | ||||||
| <!-- tocstop --> | <!-- tocstop --> | ||||||
|  |  | ||||||
| ## Installation | ## Installation | ||||||
|  |  | ||||||
|  | ### Dependencies | ||||||
|  |  | ||||||
|  | Note that you **need** a notification library such as [libnotify](https://github.com/GNOME/libnotify) or [dunst](https://github.com/dunst-project/dunst) in order for the software to work properly. | ||||||
|  |  | ||||||
| ### Using releases | ### Using releases | ||||||
|  |  | ||||||
| Various packaged versions of the software are available on the [releases](https://github.com/Mastermindzh/tidal-hifi/releases) tab. | Various packaged versions of the software are available on the [releases](https://github.com/Mastermindzh/tidal-hifi/releases) tab. | ||||||
|  |  | ||||||
| #### Snap install | ### Snap | ||||||
|  |  | ||||||
| To install with `snap` you need to download the pre-packaged snap-package from this repository, found under releases: | To install with `snap` you need to download the pre-packaged snap-package from this repository, found under releases: | ||||||
|  |  | ||||||
| 1. Download: | 1. Download | ||||||
|  |  | ||||||
| ```sh | ```sh | ||||||
| wget <URI> #for instance: https://github.com/Mastermindzh/tidal-hifi/releases/download/1.0/tidal-hifi_1.0.0_amd64.snap | wget <URI> #for instance: https://github.com/Mastermindzh/tidal-hifi/releases/download/1.0/tidal-hifi_1.0.0_amd64.snap | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| 2. Install: | 2. Install | ||||||
|  |  | ||||||
| ```sh | ```sh | ||||||
| snap install --dangerous <path> #for instance: tidal-hifi_1.0.0_amd64.snap | snap install --dangerous <path> #for instance: tidal-hifi_1.0.0_amd64.snap | ||||||
| @@ -56,33 +63,68 @@ snap install --dangerous <path> #for instance: tidal-hifi_1.0.0_amd64.snap | |||||||
| Arch Linux users can use the AUR to install tidal-hifi: | Arch Linux users can use the AUR to install tidal-hifi: | ||||||
|  |  | ||||||
| ```sh | ```sh | ||||||
| trizen tidal-hifi | trizen tidal-hifi-bin | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Flatpak | ||||||
|  |  | ||||||
|  | To install via [Flatpak](https://flathub.org/apps/details/com.mastermindzh.tidal-hifi) run the following command: | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | flatpak install flathub com.mastermindzh.tidal-hifi | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Nix | ||||||
|  |  | ||||||
|  | To install with Nix run the following command: | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | nix-env -iA nixpkgs.tidal-hifi | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Using source | ### Using source | ||||||
|  |  | ||||||
| To install and work with the code on this project follow these steps: | To install and work with the code on this project follow these steps: | ||||||
|  |  | ||||||
| - git clone https://github.com/Mastermindzh/tidal-hifi.git | - git clone [https://github.com/Mastermindzh/tidal-hifi.git](https://github.com/Mastermindzh/tidal-hifi.git) | ||||||
| - cd tidal-hifi | - cd tidal-hifi | ||||||
| - npm install | - npm install | ||||||
| - npm start | - npm start | ||||||
|  |  | ||||||
| ## features | ## Features | ||||||
|  |  | ||||||
| - HiFi playback | - HiFi playback | ||||||
| - Notifications | - Notifications | ||||||
| - Custom hotkeys ([source](https://defkey.com/tidal-desktop-shortcuts)) | - Custom hotkeys ([source](https://defkey.com/tidal-desktop-shortcuts)) | ||||||
| - API for status and playback | - API for status and playback | ||||||
| - [Settings feature](./docs/settings.png) to disable certain functionality. (`ctrl+=`) | - Disabled audio & visual ads, unlocked lyrics, suggested track, track info, and unlimited skips thanks to uBlockOrigin custom filters ([source](https://github.com/uBlockOrigin/uAssets/issues/17495)) | ||||||
|  | - Custom [integrations](#integrations) | ||||||
|  | - [Settings feature](./docs/settings.png) to disable certain functionality. (`ctrl+=` or `ctrl+0`) | ||||||
|  | - AlbumArt in integrations ([best-effort](https://github.com/Mastermindzh/tidal-hifi/pull/88#pullrequestreview-840814847)) | ||||||
|  |  | ||||||
| ## Integrations | ## Integrations | ||||||
|  |  | ||||||
|  | Tidal-hifi comes with several integrations out of the box. | ||||||
|  | You can find these in the settings menu (`ctrl + =` by default) under the "integrations" tab. | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | It currently includes: | ||||||
|  |  | ||||||
|  | - MPRIS - MPRIS media player controls/status | ||||||
|  | - Discord - Shows what you're listening to on Discord. | ||||||
|  |  | ||||||
|  | Not included: | ||||||
|  |  | ||||||
| - [i3 blocks config](https://github.com/Mastermindzh/dotfiles/commit/9714b2fa1d670108ce811d5511fd3b7a43180647) - My dotfiles where I use this app to fetch currently playing music (direct commit) | - [i3 blocks config](https://github.com/Mastermindzh/dotfiles/commit/9714b2fa1d670108ce811d5511fd3b7a43180647) - My dotfiles where I use this app to fetch currently playing music (direct commit) | ||||||
|  |  | ||||||
| ### Known bugs | ### Known bugs | ||||||
|  |  | ||||||
| - [Last.fm login doesn't work](https://github.com/Mastermindzh/tidal-hifi/issues/4). | #### last.fm doesn't work out of the box. Use rescrobbler as a workaround | ||||||
|  |  | ||||||
|  | The last.fm login doesn't work, as is evident from the following issue: [Last.fm login doesn't work](https://github.com/Mastermindzh/tidal-hifi/issues/4). | ||||||
|  | However, in that same issue you can read about a workaround using [rescrobbler](https://github.com/InputUsername/rescrobbled). | ||||||
|  | For now that will be the default workaround. | ||||||
|  |  | ||||||
| ## Why | ## Why | ||||||
|  |  | ||||||
| @@ -101,23 +143,23 @@ Whilst there are a handful of projects attempting to run Tidal on Electron they | |||||||
|  |  | ||||||
| Sometimes it's just easier to start over, cover my own needs and then making it available to the public :) | Sometimes it's just easier to start over, cover my own needs and then making it available to the public :) | ||||||
|  |  | ||||||
| ## Special thanks to... | ## Special thanks to | ||||||
|  |  | ||||||
| - [Castlabs](https://castlabs.com/) | - [Castlabs](https://castlabs.com/) | ||||||
|   For maintaining Electron with Widevine CDM installation, Verified Media Path (VMP), and persistent licenses (StorageID) |   For maintaining Electron with Widevine CDM installation, Verified Media Path (VMP), and persistent licenses (StorageID) | ||||||
|  |  | ||||||
| ## Buy me a coffee? Please don't | ## Buy me a coffee? Please don't | ||||||
|  |  | ||||||
| Instead spend some money on a charity I care for: [kwf.nl](secure.kwf.nl/donation). | Instead spend some money on a charity I care for: [kwf.nl](https://www.kwf.nl/donatie/donation). | ||||||
| Inspired by [haydenjames' issue](https://github.com/Mastermindzh/tidal-hifi/issues/27#issuecomment-704198429) | Inspired by [haydenjames' issue](https://github.com/Mastermindzh/tidal-hifi/issues/27#issuecomment-704198429) | ||||||
|  |  | ||||||
| ## Images | ## Images | ||||||
|  |  | ||||||
| ### settings window | ### Settings window | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ### user setups | ### User setups | ||||||
|  |  | ||||||
| Some of our users are kind enough to share their usage pictures. | Some of our users are kind enough to share their usage pictures. | ||||||
| If you want to see them or possibly even add one please do so in the following issue: [#3 - image thread](https://github.com/Mastermindzh/tidal-hifi/issues/3). | If you want to see them or possibly even add one please do so in the following issue: [#3 - image thread](https://github.com/Mastermindzh/tidal-hifi/issues/3). | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								assets/icons/128x128.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 16 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/16x16.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 5.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/22x22.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 9.8 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/24x24.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 9.4 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/256x256.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 29 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/32x32.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 6.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/384x384.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 46 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/48x48.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 11 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/64x64.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 11 KiB | 
							
								
								
									
										40
									
								
								build/electron-builder.base.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,40 @@ | |||||||
|  | appId: com.rickvanlieshout.tidal-hifi | ||||||
|  | electronVersion: 24.1.2 | ||||||
|  | electronDownload: | ||||||
|  |   version: 24.1.2+wvcus | ||||||
|  |   mirror: https://github.com/castlabs/electron-releases/releases/download/v | ||||||
|  | snap: | ||||||
|  |   plugs: | ||||||
|  |     - default | ||||||
|  |     - screen-inhibit-control | ||||||
|  | linux: | ||||||
|  |   category: AudioVideo | ||||||
|  |   icon: assets/icons | ||||||
|  |   target: | ||||||
|  |     - dir | ||||||
|  |   executableName: tidal-hifi | ||||||
|  |   desktop: | ||||||
|  |     Encoding: UTF-8 | ||||||
|  |     Name: TIDAL Hi-Fi | ||||||
|  |     GenericName: TIDAL Hi-Fi | ||||||
|  |     Comment: The web version of listen.tidal.com running in electron with hifi support thanks to widevine. | ||||||
|  |     Icon: tidal-hifi | ||||||
|  |     StartupNotify: true | ||||||
|  |     Terminal: false | ||||||
|  |     Type: Application | ||||||
|  |     Categories: Network;Application;AudioVideo;Audio;Video | ||||||
|  |     StartupWMClass: tidal-hifi | ||||||
|  |     X-PulseAudio-Properties: media.role=music | ||||||
|  |     MimeType: x-scheme-handler/tidal; | ||||||
|  |  | ||||||
|  | mac: | ||||||
|  |   category: public.app-category.entertainment | ||||||
|  | win: | ||||||
|  |   icon: icon.png | ||||||
|  |   artifactName: "tidalhifi" | ||||||
|  |   appId: com.rickvanlieshout.tidalhifi | ||||||
|  |   executableName: tidalhifi | ||||||
|  | protocols: | ||||||
|  |   name: "tidal" | ||||||
|  |   role: "Viewer" | ||||||
|  |   schemes: ["tidal"] | ||||||
| @@ -1,6 +1,4 @@ | |||||||
| extends: ./build/electron-builder.yml | extends: ./build/electron-builder.base.yml | ||||||
| linux: | linux: | ||||||
|   category: Audio |  | ||||||
|   icon: ./assets/icon.png |  | ||||||
|   target: |   target: | ||||||
|     - deb |     - deb | ||||||
|   | |||||||
| @@ -1,6 +1,4 @@ | |||||||
| extends: ./build/electron-builder.yml | extends: ./build/electron-builder.base.yml | ||||||
| linux: | linux: | ||||||
|   category: Audio |  | ||||||
|   icon: ./assets/icon.png |  | ||||||
|   target: |   target: | ||||||
|     - pacman |     - pacman | ||||||
|   | |||||||
| @@ -1,6 +1,4 @@ | |||||||
| extends: ./build/electron-builder.yml | extends: ./build/electron-builder.base.yml | ||||||
| linux: | linux: | ||||||
|     category: Audio |   target: | ||||||
|     icon: ./assets/TIDAL.icns |     - rpm | ||||||
|     target: |  | ||||||
|         - rpm |  | ||||||
|   | |||||||
| @@ -1,6 +1,4 @@ | |||||||
| extends: ./build/electron-builder.yml | extends: ./build/electron-builder.base.yml | ||||||
| linux: | linux: | ||||||
|   category: Audio |  | ||||||
|   icon: ./assets/icon.png |  | ||||||
|   target: |   target: | ||||||
|     - snap |     - snap | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								build/electron-builder.unpacked.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | |||||||
|  | extends: ./build/electron-builder.base.yml | ||||||
|  | linux: | ||||||
|  |   target: | ||||||
|  |     - dir | ||||||
| @@ -1,36 +1,16 @@ | |||||||
| appId: com.rickvanlieshout.tidal-hifi | extends: ./build/electron-builder.base.yml | ||||||
| electronVersion: 8.5.2 |  | ||||||
| electronDownload: |  | ||||||
|   version: 8.5.2-wvvmp |  | ||||||
|   mirror: https://github.com/castlabs/electron-releases/releases/download/v |  | ||||||
| snap: |  | ||||||
|   plugs: |  | ||||||
|     - default |  | ||||||
|     - screen-inhibit-control |  | ||||||
| linux: | linux: | ||||||
|   category: Audio |  | ||||||
|   target: |   target: | ||||||
|     # - pacman |     - pacman | ||||||
|     - tar.gz |     - tar.gz | ||||||
|     - deb |     - deb | ||||||
|     - rpm |     - rpm | ||||||
|     - AppImage |     - AppImage | ||||||
|     - snap |     - snap | ||||||
|     - freebsd |     - freebsd | ||||||
|   executableName: tidal-hifi |  | ||||||
|   desktop: |  | ||||||
|     Encoding: UTF-8 |  | ||||||
|     Name: tidal-hifi |  | ||||||
|     GenericName: tidal-hifi |  | ||||||
|     Comment: The web version of listen.tidal.com running in electron with hifi support thanks to widevine. |  | ||||||
|     Icon: assets/icon.png |  | ||||||
|     StartupNotify: true |  | ||||||
|     Terminal: false |  | ||||||
|     Type: Application |  | ||||||
|     Categories: Network;Application;Audio;Video |  | ||||||
|     StartupWMClass: tidal-hifi |  | ||||||
|     X-PulseAudio-Properties: media.role=music |  | ||||||
| mac: |  | ||||||
|   category: public.app-category.entertainment |  | ||||||
| win: | win: | ||||||
|   target: msi |   target: msi | ||||||
|  |   icon: icon.png | ||||||
|  |   artifactName: "tidalhifi" | ||||||
|  |   appId: com.rickvanlieshout.tidalhifi | ||||||
|  |   executableName: tidalhifi | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								build/icon-inverted.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 102 KiB | 
							
								
								
									
										
											BIN
										
									
								
								build/icon.icns
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -1,20 +0,0 @@ | |||||||
| pkgbase = tidal-hifi-git |  | ||||||
| 	pkgdesc = The web version of listen.tidal.com running in electron with hifi support thanks to widevine. |  | ||||||
| 	pkgver = 2.0.0 |  | ||||||
| 	pkgrel = 1 |  | ||||||
| 	url = https://github.com/Mastermindzh/tidal-hifi |  | ||||||
| 	arch = x86_64 |  | ||||||
| 	license = custom:MIT |  | ||||||
| 	makedepends = npm |  | ||||||
| 	makedepends = git |  | ||||||
| 	depends = libxss |  | ||||||
| 	depends = nss |  | ||||||
| 	depends = gtk3 |  | ||||||
| 	provides = tidal-hifi |  | ||||||
| 	source = https://github.com/Mastermindzh/tidal-hifi/archive/2.0.0.zip |  | ||||||
| 	source = tidal-hifi.desktop |  | ||||||
| 	sha512sums = 68fce020128b637fd37383a5afae71d2e4f3eec2d6b4bfcd7f410dcec18c91327d163deb21750e2251967a31f83b2b16092d11d3ff500a1fecd8c595448f380e |  | ||||||
| 	sha512sums = 35f38ac308b871c1822d7f6f760f2fb54c3748cf769822cb0f0dfb90f0f5754ba9316da5e903a0d2e9839de3a43ec76f238f3f2e44021956fa1da19142081349 |  | ||||||
|  |  | ||||||
| pkgname = tidal-hifi-git |  | ||||||
|  |  | ||||||
| @@ -1,58 +0,0 @@ | |||||||
| # Maintainer: Rick van Lieshout <info@rickvanlieshout.com> |  | ||||||
|  |  | ||||||
| _pkgname=tidal-hifi |  | ||||||
| pkgname="$_pkgname-git" |  | ||||||
| pkgver=2.0.0 |  | ||||||
| pkgrel=1 |  | ||||||
| pkgdesc="The web version of listen.tidal.com running in electron with hifi support thanks to widevine." |  | ||||||
| arch=("x86_64") |  | ||||||
| url="https://github.com/Mastermindzh/tidal-hifi" |  | ||||||
| license=("custom:MIT") |  | ||||||
|  |  | ||||||
| depends=("libxss" "nss" "gtk3") |  | ||||||
| makedepends=("npm" "git") |  | ||||||
| provides=("$_pkgname") |  | ||||||
|  |  | ||||||
| source=("https://github.com/Mastermindzh/tidal-hifi/archive/$pkgver.zip" |  | ||||||
|         "${_pkgname}.desktop") |  | ||||||
| sha512sums=('68fce020128b637fd37383a5afae71d2e4f3eec2d6b4bfcd7f410dcec18c91327d163deb21750e2251967a31f83b2b16092d11d3ff500a1fecd8c595448f380e' |  | ||||||
|             '35f38ac308b871c1822d7f6f760f2fb54c3748cf769822cb0f0dfb90f0f5754ba9316da5e903a0d2e9839de3a43ec76f238f3f2e44021956fa1da19142081349') |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| cdToPkg(){ |  | ||||||
|     cd "tidal-hifi-$pkgver" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| prepare() { |  | ||||||
|     cdToPkg |  | ||||||
|  |  | ||||||
|     # install build dependencies |  | ||||||
|     npm install |  | ||||||
| } |  | ||||||
|  |  | ||||||
| build() { |  | ||||||
|     cdToPkg |  | ||||||
|  |  | ||||||
|     # We are not using the systems Electron as we need castlab's Electron. |  | ||||||
|     npm run build-arch |  | ||||||
| } |  | ||||||
|  |  | ||||||
| package() { |  | ||||||
|     cdToPkg |  | ||||||
|  |  | ||||||
|     install -d "${pkgdir}/opt/${_pkgname}/" "${pkgdir}/usr/bin" "${pkgdir}/usr/share/doc" "${pkgdir}/usr/share/licenses" |  | ||||||
|  |  | ||||||
|     cp -r dist/linux-unpacked/* "${pkgdir}/opt/${_pkgname}/" |  | ||||||
|     chmod +x "${pkgdir}/opt/${_pkgname}/${_pkgname}" |  | ||||||
|  |  | ||||||
|     ln -s "/opt/${_pkgname}/${_pkgname}" "${pkgdir}/usr/bin/${_pkgname}" |  | ||||||
|  |  | ||||||
|     install -Dm 644 "build/icon.png" "${pkgdir}/usr/share/pixmaps/${_pkgname}.png" |  | ||||||
|     install -Dm 644 "${srcdir}/${_pkgname}.desktop" "${pkgdir}/usr/share/applications/${_pkgname}.desktop" |  | ||||||
|  |  | ||||||
|     install -Dm 644 "README.md" "${pkgdir}/usr/share/doc/${pkgname}/README.md" |  | ||||||
|     install -Dm 644 "LICENSE" "${pkgdir}/usr/share/licenses/${pkgname}/LICENSE" |  | ||||||
|     ln -s "/opt/${_pkgname}/LICENSE.electron.txt" "${pkgdir}/usr/share/licenses/${pkgname}/LICENSE.electron.txt" |  | ||||||
|     ln -s "/opt/${_pkgname}/LICENSES.chromium.html" "${pkgdir}/usr/share/licenses/${pkgname}/LICENSES.chromium.html" |  | ||||||
| } |  | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| #!/bin/bash |  | ||||||
| # Will generate a correctly formatted SRCINFO file |  | ||||||
|  |  | ||||||
| SCRIPT_DIST=".SRCINFO" |  | ||||||
|  |  | ||||||
| # generate SRCINFO |  | ||||||
| makepkg --printsrcinfo > $SCRIPT_DIST |  | ||||||
|  |  | ||||||
| # replace pkgbase with tidal-hifi-git |  | ||||||
| pkgName="tidal-hifi-git" |  | ||||||
| sed -i "1s/.*/pkgbase = $pkgName/" $SCRIPT_DIST |  | ||||||
|  |  | ||||||
| # replace pkgbase with tidal-hifi-git |  | ||||||
| sed -i '/^pkgname/ d' $SCRIPT_DIST |  | ||||||
| echo "pkgname = $pkgName" >> $SCRIPT_DIST |  | ||||||
|  |  | ||||||
| # remove double line breaks and replace with single line breaks |  | ||||||
| sed -i '/^$/N;/^\n$/D' $SCRIPT_DIST |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| [Desktop Entry] |  | ||||||
| Encoding=UTF-8 |  | ||||||
| Name=tidal-hifi |  | ||||||
| GenericName=tidal-hifi |  | ||||||
| Comment=The web version of listen.tidal.com running in electron with hifi support thanks to widevine. |  | ||||||
| Exec=tidal-hifi %u |  | ||||||
| Icon=tidal-hifi.png |  | ||||||
| StartupNotify=true |  | ||||||
| Terminal=false |  | ||||||
| Type=Application |  | ||||||
| Categories=Network;Application;Audio;Video |  | ||||||
| StartupWMClass=tidal-hifi |  | ||||||
| X-PulseAudio-Properties=media.role=music |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								docs/integrations.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 47 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/no-dutch-music.mp4
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										10545
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
							
								
								
									
										67
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @@ -1,17 +1,28 @@ | |||||||
| { | { | ||||||
|   "name": "tidal-hifi", |   "name": "tidal-hifi", | ||||||
|   "version": "2.1.0", |   "version": "5.2.0", | ||||||
|   "description": "Tidal on Electron with widevine(hifi) support", |   "description": "Tidal on Electron with widevine(hifi) support", | ||||||
|   "main": "src/main.js", |   "main": "ts-dist/main.js", | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "start": "electron .", |     "start": "electron .", | ||||||
|     "build": "electron-builder --publish=never -c ./build/electron-builder.yml", |     "compile": "tsc && npm run sass-and-copy", | ||||||
|     "build-deb": "electron-builder --publish=never -c ./build/electron-builder.deb.yml", |     "watch": "tsc-watch --onSuccess \"npm run sass-and-copy\"", | ||||||
|     "build-rpm": "electron-builder --publish=never -c ./build/electron-builder.rpm.yml", |     "copy-files": "copyfiles -u 1 --exclude './src/**/*.ts' --exclude './src/**/*.scss' \"./src/**/*\" ts-dist", | ||||||
|     "build-snap": "electron-builder --publish=never -c ./build/electron-builder.snap.yml", |     "sass-and-copy": "npm run sass && npm run copy-files", | ||||||
|     "build-arch": "electron-builder --publish=never -c ./build/electron-builder.pacman.yml", |     "build": "npm run builder -- -c ./build/electron-builder.yml", | ||||||
|     "build-wl": "electron-builder --publish=never -c ./build/electron-builder.yml -wl", |     "build-deb": "npm run builder -- -c ./build/electron-builder.deb.yml", | ||||||
|     "build-mac": "electron-builder --publish=never -c ./build/electron-builder.yml -m" |     "build-unpacked": "npm run builder -- -c ./build/electron-builder.unpacked.yml", | ||||||
|  |     "build-rpm": "npm run builder -- -c ./build/electron-builder.rpm.yml", | ||||||
|  |     "build-snap": "npm run builder -- -c ./build/electron-builder.snap.yml", | ||||||
|  |     "build-arch": "npm run builder -- -c ./build/electron-builder.pacman.yml", | ||||||
|  |     "build-wl": "npm run builder -- -c ./build/electron-builder.yml -wl", | ||||||
|  |     "build-mac": "npm run builder -- -c ./build/electron-builder.yml -m", | ||||||
|  |     "build-base": "npm run builder -- -c ./build/electron-builder.base.yml", | ||||||
|  |     "prebuilder": "npm run compile", | ||||||
|  |     "builder": "electron-builder --publish=never", | ||||||
|  |     "sass": "sass ./src/pages/settings/settings.scss ./src/pages/settings/settings.css", | ||||||
|  |     "style-lint": "npx stylelint **/*.scss", | ||||||
|  |     "style-lint-fix": "npx stylelint --fix **/*.scss" | ||||||
|   }, |   }, | ||||||
|   "keywords": [ |   "keywords": [ | ||||||
|     "electron", |     "electron", | ||||||
| @@ -23,21 +34,35 @@ | |||||||
|   "homepage": "https://github.com/Mastermindzh/tidal-hifi", |   "homepage": "https://github.com/Mastermindzh/tidal-hifi", | ||||||
|   "license": "MIT", |   "license": "MIT", | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "discord-rpc": "^3.2.0", |     "@electron/remote": "^2.0.9", | ||||||
|     "electron-store": "^5.1.1", |     "discord-rpc": "^4.0.1", | ||||||
|     "express": "^4.17.1", |     "electron-store": "^8.1.0", | ||||||
|     "hotkeys-js": "^3.7.6", |     "express": "^4.18.2", | ||||||
|     "mpris-service": "^2.1.0", |     "hotkeys-js": "^3.10.2", | ||||||
|     "node-notifier": "^9.0.1", |     "mpris-service": "^2.1.2", | ||||||
|     "request": "^2.88.2" |     "request": "^2.88.2", | ||||||
|  |     "sass": "^1.62.0" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@mastermindzh/prettier-config": "^1.0.0", |     "@mastermindzh/prettier-config": "^1.0.0", | ||||||
|     "electron": "git+https://github.com/castlabs/electron-releases.git#v8.5.2-wvvmp", |     "@types/discord-rpc": "^4.0.4", | ||||||
|     "electron-builder": "^21.2.0", |     "@types/express": "^4.17.17", | ||||||
|     "electron-reload": "^1.5.0", |     "@types/request": "^2.48.8", | ||||||
|     "prettier": "^2.0.4", |     "@typescript-eslint/eslint-plugin": "^5.59.1", | ||||||
|     "dot-prop": ">=4.2.1" |     "@typescript-eslint/parser": "^5.59.1", | ||||||
|  |     "copyfiles": "^2.4.1", | ||||||
|  |     "electron": "git+https://github.com/castlabs/electron-releases.git#v24.1.2+wvcus", | ||||||
|  |     "electron-builder": "^24.2.1", | ||||||
|  |     "eslint": "^8.39.0", | ||||||
|  |     "js-yaml": "^4.1.0", | ||||||
|  |     "markdown-toc": "^1.2.0", | ||||||
|  |     "prettier": "^2.8.8", | ||||||
|  |     "stylelint": "^15.6.0", | ||||||
|  |     "stylelint-config-standard": "^33.0.0", | ||||||
|  |     "stylelint-config-standard-scss": "^9.0.0", | ||||||
|  |     "stylelint-prettier": "^3.0.0", | ||||||
|  |     "tsc-watch": "^6.0.4", | ||||||
|  |     "typescript": "^5.0.4" | ||||||
|   }, |   }, | ||||||
|   "prettier": "@mastermindzh/prettier-config" |   "prettier": "@mastermindzh/prettier-config" | ||||||
| } | } | ||||||
							
								
								
									
										4
									
								
								src/constants/flags.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | |||||||
|  | export const flags: { [key: string]: { flag: string; value?: any }[] } = { | ||||||
|  |   gpuRasterization: [{ flag: "enable-gpu-rasterization", value: undefined }], | ||||||
|  |   disableHardwareMediaKeys: [{ flag: "disable-features", value: "HardwareMediaKeyHandling" }], | ||||||
|  | }; | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| const globalEvents = { | export const globalEvents = { | ||||||
|   play: "play", |   play: "play", | ||||||
|   pause: "pause", |   pause: "pause", | ||||||
|   playPause: "playPause", |   playPause: "playPause", | ||||||
| @@ -6,10 +6,8 @@ const globalEvents = { | |||||||
|   previous: "previous", |   previous: "previous", | ||||||
|   updateInfo: "update-info", |   updateInfo: "update-info", | ||||||
|   hideSettings: "hideSettings", |   hideSettings: "hideSettings", | ||||||
|  |   refreshMenuBar: "refreshMenubar", | ||||||
|   showSettings: "showSettings", |   showSettings: "showSettings", | ||||||
|   updateStatus: "update-status", |  | ||||||
|   storeChanged: "storeChanged", |   storeChanged: "storeChanged", | ||||||
|   error: "error", |   error: "error", | ||||||
| }; | }; | ||||||
| 
 |  | ||||||
| module.exports = globalEvents; |  | ||||||
| @@ -1,9 +1,7 @@ | |||||||
| const globalEvents = require("./globalEvents"); | import { globalEvents } from "./globalEvents"; | ||||||
| 
 | 
 | ||||||
| const mediaKeys = { | export const mediaKeys = { | ||||||
|   MediaPlayPause: globalEvents.playPause, |   MediaPlayPause: globalEvents.playPause, | ||||||
|   MediaNextTrack: globalEvents.next, |   MediaNextTrack: globalEvents.next, | ||||||
|   MediaPreviousTrack: globalEvents.previous, |   MediaPreviousTrack: globalEvents.previous, | ||||||
| }; | }; | ||||||
| 
 |  | ||||||
| module.exports = mediaKeys; |  | ||||||
| @@ -8,24 +8,36 @@ | |||||||
|  *    }, |  *    }, | ||||||
|  *    windowBounds: { width: 800, height: 600 }, |  *    windowBounds: { width: 800, height: 600 }, | ||||||
|  */ |  */ | ||||||
| const settings = { | export const settings = { | ||||||
|   notifications: "notifications", |   adBlock: "adBlock", | ||||||
|   api: "api", |   api: "api", | ||||||
|   menuBar: "menuBar", |  | ||||||
|   playBackControl: "playBackControl", |  | ||||||
|   apiSettings: { |   apiSettings: { | ||||||
|     root: "apiSettings", |     root: "apiSettings", | ||||||
|     port: "apiSettings.port", |     port: "apiSettings.port", | ||||||
|   }, |   }, | ||||||
|   mpris: "mpris", |   customCSS: "customCSS", | ||||||
|  |   disableBackgroundThrottle: "disableBackgroundThrottle", | ||||||
|  |   disableHardwareMediaKeys: "disableHardwareMediaKeys", | ||||||
|   enableCustomHotkeys: "enableCustomHotkeys", |   enableCustomHotkeys: "enableCustomHotkeys", | ||||||
|   trayIcon: "trayIcon", |  | ||||||
|   enableDiscord: "enableDiscord", |   enableDiscord: "enableDiscord", | ||||||
|  |   flags: { | ||||||
|  |     root: "flags", | ||||||
|  |     disableHardwareMediaKeys: "flags.disableHardwareMediaKeys", | ||||||
|  |     gpuRasterization: "flags.gpuRasterization", | ||||||
|  |   }, | ||||||
|  |   menuBar: "menuBar", | ||||||
|  |   minimizeOnClose: "minimizeOnClose", | ||||||
|  |   mpris: "mpris", | ||||||
|  |   notifications: "notifications", | ||||||
|  |   playBackControl: "playBackControl", | ||||||
|  |   singleInstance: "singleInstance", | ||||||
|  |   skipArtists: "skipArtists", | ||||||
|  |   skippedArtists: "skippedArtists", | ||||||
|  |   trayIcon: "trayIcon", | ||||||
|  |   updateFrequency: "updateFrequency", | ||||||
|   windowBounds: { |   windowBounds: { | ||||||
|     root: "windowBounds", |     root: "windowBounds", | ||||||
|     width: "windowBounds.width", |     width: "windowBounds.width", | ||||||
|     height: "windowBounds.height", |     height: "windowBounds.height", | ||||||
|   }, |   }, | ||||||
| }; | }; | ||||||
| 
 |  | ||||||
| module.exports = settings; |  | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| module.exports = { | export const statuses = { | ||||||
|   playing: "playing", |   playing: "playing", | ||||||
|   paused: "paused", |   paused: "paused", | ||||||
| }; | }; | ||||||
							
								
								
									
										3
									
								
								src/constants/values.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | export default { | ||||||
|  |   name: "tidal-hifi", | ||||||
|  | }; | ||||||
							
								
								
									
										1
									
								
								src/declarations.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | declare module "mpris-service"; | ||||||
							
								
								
									
										128
									
								
								src/main.js
									
									
									
									
									
								
							
							
						
						| @@ -1,128 +0,0 @@ | |||||||
| const { app, BrowserWindow, globalShortcut, ipcMain } = require("electron"); |  | ||||||
| const { |  | ||||||
|   settings, |  | ||||||
|   store, |  | ||||||
|   createSettingsWindow, |  | ||||||
|   showSettingsWindow, |  | ||||||
|   closeSettingsWindow, |  | ||||||
|   hideSettingsWindow, |  | ||||||
| } = require("./scripts/settings"); |  | ||||||
| const { addTray, refreshTray } = require("./scripts/tray"); |  | ||||||
| const { addMenu } = require("./scripts/menu"); |  | ||||||
| const path = require("path"); |  | ||||||
| const tidalUrl = "https://listen.tidal.com"; |  | ||||||
| const expressModule = require("./scripts/express"); |  | ||||||
| const mediaKeys = require("./constants/mediaKeys"); |  | ||||||
| const mediaInfoModule = require("./scripts/mediaInfo"); |  | ||||||
| const discordModule = require("./scripts/discord"); |  | ||||||
| const globalEvents = require("./constants/globalEvents"); |  | ||||||
|  |  | ||||||
|  |  | ||||||
| let mainWindow; |  | ||||||
| let icon = path.join(__dirname, "../assets/icon.png"); |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Enable live reload in development builds |  | ||||||
|  */ |  | ||||||
| if (!app.isPackaged) { |  | ||||||
|   require("electron-reload")(`${__dirname}`, { |  | ||||||
|     electron: require(`${__dirname}/../node_modules/electron`), |  | ||||||
|   }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function createWindow(options = {}) { |  | ||||||
|   // Create the browser window. |  | ||||||
|   mainWindow = new BrowserWindow({ |  | ||||||
|     x: options.x, |  | ||||||
|     y: options.y, |  | ||||||
|     width: store && store.get(settings.windowBounds.width), |  | ||||||
|     height: store && store.get(settings.windowBounds.height), |  | ||||||
|     icon, |  | ||||||
|     tray: true, |  | ||||||
|     backgroundColor: options.backgroundColor, |  | ||||||
|     webPreferences: { |  | ||||||
|       affinity: "window", |  | ||||||
|       preload: path.join(__dirname, "preload.js"), |  | ||||||
|       plugins: true, |  | ||||||
|       devTools: true, // I like tinkering, others might too |  | ||||||
|     }, |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   mainWindow.setMenuBarVisibility(store.get(settings.menuBar)); |  | ||||||
|  |  | ||||||
|   // load the Tidal website |  | ||||||
|   mainWindow.loadURL(tidalUrl); |  | ||||||
|  |  | ||||||
|   // run stuff after first load |  | ||||||
|   mainWindow.webContents.once("did-finish-load", () => {}); |  | ||||||
|  |  | ||||||
|   // Emitted when the window is closed. |  | ||||||
|   mainWindow.on("closed", function () { |  | ||||||
|     closeSettingsWindow(); |  | ||||||
|     app.quit(); |  | ||||||
|   }); |  | ||||||
|   mainWindow.on("resize", () => { |  | ||||||
|     let { width, height } = mainWindow.getBounds(); |  | ||||||
|  |  | ||||||
|     store.set(settings.windowBounds.root, { width, height }); |  | ||||||
|   }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function addGlobalShortcuts() { |  | ||||||
|   Object.keys(mediaKeys).forEach((key) => { |  | ||||||
|     globalShortcut.register(`${key}`, () => { |  | ||||||
|       mainWindow.webContents.send("globalEvent", `${mediaKeys[key]}`); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // This method will be called when Electron has finished |  | ||||||
| // initialization and is ready to create browser windows. |  | ||||||
| // Some APIs can only be used after this event occurs. |  | ||||||
| app.on("ready", () => { |  | ||||||
|   createWindow(); |  | ||||||
|   addMenu(); |  | ||||||
|   createSettingsWindow(); |  | ||||||
|   addGlobalShortcuts(); |  | ||||||
|   store.get(settings.trayIcon) && addTray({ icon }) && refreshTray(); |  | ||||||
|   store.get(settings.api) && expressModule.run(mainWindow); |  | ||||||
|   store.get(settings.enableDiscord) && discordModule.initRPC(); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| app.on("activate", function () { |  | ||||||
|   // On OS X it's common to re-create a window in the app when the |  | ||||||
|   // dock icon is clicked and there are no other windows open. |  | ||||||
|   if (mainWindow === null) { |  | ||||||
|     createWindow(); |  | ||||||
|   } |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| // IPC |  | ||||||
|  |  | ||||||
| ipcMain.on(globalEvents.updateInfo, (event, arg) => { |  | ||||||
|   mediaInfoModule.update(arg); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| ipcMain.on(globalEvents.hideSettings, (event, arg) => { |  | ||||||
|   hideSettingsWindow(); |  | ||||||
| }); |  | ||||||
| ipcMain.on(globalEvents.showSettings, (event, arg) => { |  | ||||||
|   showSettingsWindow(); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| ipcMain.on(globalEvents.updateStatus, (event, arg) => { |  | ||||||
|   mediaInfoModule.updateStatus(arg); |  | ||||||
| }); |  | ||||||
| ipcMain.on(globalEvents.storeChanged, (event, arg) => { |  | ||||||
|   mainWindow.setMenuBarVisibility(store.get(settings.menuBar)); |  | ||||||
|  |  | ||||||
|   if(store.get(settings.enableDiscord) && !discordModule.rpc) { |  | ||||||
|     discordModule.initRPC(); |  | ||||||
|   } else if(!store.get(settings.enableDiscord) && discordModule.rpc) { |  | ||||||
|     discordModule.unRPC(); |  | ||||||
|   } |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| ipcMain.on(globalEvents.error, (event, arg) => { |  | ||||||
|   console.log(event); |  | ||||||
| }); |  | ||||||
							
								
								
									
										222
									
								
								src/main.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,222 @@ | |||||||
|  | import { enable, initialize } from "@electron/remote/main"; | ||||||
|  | import { | ||||||
|  |   BrowserWindow, | ||||||
|  |   app, | ||||||
|  |   components, | ||||||
|  |   globalShortcut, | ||||||
|  |   ipcMain, | ||||||
|  |   protocol, | ||||||
|  |   session, | ||||||
|  | } from "electron"; | ||||||
|  | import path from "path"; | ||||||
|  | import { flags } from "./constants/flags"; | ||||||
|  | import { globalEvents } from "./constants/globalEvents"; | ||||||
|  | import { mediaKeys } from "./constants/mediaKeys"; | ||||||
|  | import { initRPC, rpc, unRPC } from "./scripts/discord"; | ||||||
|  | import { startExpress } from "./scripts/express"; | ||||||
|  | import { updateMediaInfo } from "./scripts/mediaInfo"; | ||||||
|  | import { addMenu } from "./scripts/menu"; | ||||||
|  | import { | ||||||
|  |   closeSettingsWindow, | ||||||
|  |   createSettingsWindow, | ||||||
|  |   hideSettingsWindow, | ||||||
|  |   showSettingsWindow, | ||||||
|  |   settingsStore, | ||||||
|  | } from "./scripts/settings"; | ||||||
|  | import { settings } from "./constants/settings"; | ||||||
|  | import { addTray, refreshTray } from "./scripts/tray"; | ||||||
|  | import { MediaInfo } from "./models/mediaInfo"; | ||||||
|  | const tidalUrl = "https://listen.tidal.com"; | ||||||
|  |  | ||||||
|  | initialize(); | ||||||
|  |  | ||||||
|  | let mainWindow: BrowserWindow; | ||||||
|  | const icon = path.join(__dirname, "../assets/icon.png"); | ||||||
|  | const PROTOCOL_PREFIX = "tidal"; | ||||||
|  |  | ||||||
|  | setFlags(); | ||||||
|  |  | ||||||
|  | function setFlags() { | ||||||
|  |   const flagsFromSettings = settingsStore.get(settings.flags.root); | ||||||
|  |   if (flagsFromSettings) { | ||||||
|  |     for (const [key, value] of Object.entries(flags)) { | ||||||
|  |       if (value) { | ||||||
|  |         flags[key].forEach((flag) => { | ||||||
|  |           console.log(`enabling command line switch ${flag.flag} with value ${flag.value}`); | ||||||
|  |           app.commandLine.appendSwitch(flag.flag, flag.value); | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Fix Display Compositor issue. | ||||||
|  |    */ | ||||||
|  |   app.commandLine.appendSwitch("disable-seccomp-filter-sandbox"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Update the menuBarVisbility according to the store value | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | function syncMenuBarWithStore() { | ||||||
|  |   const fixedMenuBar = !!settingsStore.get(settings.menuBar); | ||||||
|  |  | ||||||
|  |   mainWindow.autoHideMenuBar = !fixedMenuBar; | ||||||
|  |   mainWindow.setMenuBarVisibility(fixedMenuBar); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Determine whether the current window is the main window | ||||||
|  |  * if singleInstance is requested. | ||||||
|  |  * If singleInstance isn't requested simply return true | ||||||
|  |  * @returns true if singInstance is not requested, otherwise true/false based on whether the current window is the main window | ||||||
|  |  */ | ||||||
|  | function isMainInstanceOrMultipleInstancesAllowed() { | ||||||
|  |   if (settingsStore.get(settings.singleInstance)) { | ||||||
|  |     const gotTheLock = app.requestSingleInstanceLock(); | ||||||
|  |  | ||||||
|  |     if (!gotTheLock) { | ||||||
|  |       return false; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function createWindow(options = { x: 0, y: 0, backgroundColor: "white" }) { | ||||||
|  |   // Create the browser window. | ||||||
|  |   mainWindow = new BrowserWindow({ | ||||||
|  |     x: options.x, | ||||||
|  |     y: options.y, | ||||||
|  |     width: settingsStore && settingsStore.get(settings.windowBounds.width), | ||||||
|  |     height: settingsStore && settingsStore.get(settings.windowBounds.height), | ||||||
|  |     icon, | ||||||
|  |     backgroundColor: options.backgroundColor, | ||||||
|  |     autoHideMenuBar: true, | ||||||
|  |     webPreferences: { | ||||||
|  |       sandbox: false, | ||||||
|  |       preload: path.join(__dirname, "preload.js"), | ||||||
|  |       plugins: true, | ||||||
|  |       devTools: true, // I like tinkering, others might too | ||||||
|  |     }, | ||||||
|  |   }); | ||||||
|  |   enable(mainWindow.webContents); | ||||||
|  |   registerHttpProtocols(); | ||||||
|  |   syncMenuBarWithStore(); | ||||||
|  |  | ||||||
|  |   // load the Tidal website | ||||||
|  |   mainWindow.loadURL(tidalUrl); | ||||||
|  |  | ||||||
|  |   if (settingsStore.get(settings.disableBackgroundThrottle)) { | ||||||
|  |     // prevent setInterval lag | ||||||
|  |     mainWindow.webContents.setBackgroundThrottling(false); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   mainWindow.on("close", function (event: CloseEvent) { | ||||||
|  |     if (settingsStore.get(settings.minimizeOnClose)) { | ||||||
|  |       event.preventDefault(); | ||||||
|  |       mainWindow.hide(); | ||||||
|  |       refreshTray(mainWindow); | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  |   }); | ||||||
|  |   // Emitted when the window is closed. | ||||||
|  |   mainWindow.on("closed", function () { | ||||||
|  |     closeSettingsWindow(); | ||||||
|  |     app.quit(); | ||||||
|  |   }); | ||||||
|  |   mainWindow.on("resize", () => { | ||||||
|  |     const { width, height } = mainWindow.getBounds(); | ||||||
|  |     settingsStore.set(settings.windowBounds.root, { width, height }); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function registerHttpProtocols() { | ||||||
|  |   protocol.registerHttpProtocol(PROTOCOL_PREFIX, (request) => { | ||||||
|  |     mainWindow.loadURL(`${tidalUrl}/${request.url.substring(PROTOCOL_PREFIX.length + 3)}`); | ||||||
|  |   }); | ||||||
|  |   if (!app.isDefaultProtocolClient(PROTOCOL_PREFIX)) { | ||||||
|  |     app.setAsDefaultProtocolClient(PROTOCOL_PREFIX); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function addGlobalShortcuts() { | ||||||
|  |   Object.keys(mediaKeys).forEach((key) => { | ||||||
|  |     globalShortcut.register(`${key}`, () => { | ||||||
|  |       mainWindow.webContents.send("globalEvent", `${(mediaKeys as any)[key]}`); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // This method will be called when Electron has finished | ||||||
|  | // initialization and is ready to create browser windows. | ||||||
|  | // Some APIs can only be used after this event occurs. | ||||||
|  | app.on("ready", async () => { | ||||||
|  |   if (isMainInstanceOrMultipleInstancesAllowed()) { | ||||||
|  |     await components.whenReady(); | ||||||
|  |  | ||||||
|  |     // Adblock | ||||||
|  |     if (settingsStore.get(settings.adBlock)) { | ||||||
|  |       const filter = { urls: ["https://listen.tidal.com/*"] }; | ||||||
|  |       session.defaultSession.webRequest.onBeforeRequest(filter, (details, callback) => { | ||||||
|  |         if (details.url.match(/\/users\/.*\d\?country/)) callback({ cancel: true }); | ||||||
|  |         else callback({ cancel: false }); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     createWindow(); | ||||||
|  |     addMenu(mainWindow); | ||||||
|  |     createSettingsWindow(); | ||||||
|  |     addGlobalShortcuts(); | ||||||
|  |     if (settingsStore.get(settings.trayIcon)) { | ||||||
|  |       addTray(mainWindow, { icon }); | ||||||
|  |       refreshTray(mainWindow); | ||||||
|  |     } | ||||||
|  |     settingsStore.get(settings.api) && startExpress(mainWindow); | ||||||
|  |     settingsStore.get(settings.enableDiscord) && initRPC(); | ||||||
|  |   } else { | ||||||
|  |     app.quit(); | ||||||
|  |   } | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | app.on("activate", function () { | ||||||
|  |   // On OS X it's common to re-create a window in the app when the | ||||||
|  |   // dock icon is clicked and there are no other windows open. | ||||||
|  |   if (mainWindow === null) { | ||||||
|  |     createWindow(); | ||||||
|  |   } | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | app.on("browser-window-created", (_, window) => { | ||||||
|  |   enable(window.webContents); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // IPC | ||||||
|  | ipcMain.on(globalEvents.updateInfo, (_event, arg: MediaInfo) => { | ||||||
|  |   updateMediaInfo(arg); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | ipcMain.on(globalEvents.hideSettings, () => { | ||||||
|  |   hideSettingsWindow(); | ||||||
|  | }); | ||||||
|  | ipcMain.on(globalEvents.showSettings, () => { | ||||||
|  |   showSettingsWindow(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | ipcMain.on(globalEvents.refreshMenuBar, () => { | ||||||
|  |   syncMenuBarWithStore(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | ipcMain.on(globalEvents.storeChanged, () => { | ||||||
|  |   syncMenuBarWithStore(); | ||||||
|  |  | ||||||
|  |   if (settingsStore.get(settings.enableDiscord) && !rpc) { | ||||||
|  |     initRPC(); | ||||||
|  |   } else if (!settingsStore.get(settings.enableDiscord) && rpc) { | ||||||
|  |     unRPC(); | ||||||
|  |   } | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | ipcMain.on(globalEvents.error, (event) => { | ||||||
|  |   console.log(event); | ||||||
|  | }); | ||||||
							
								
								
									
										13
									
								
								src/models/mediaInfo.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,13 @@ | |||||||
|  | import { MediaStatus } from "./mediaStatus"; | ||||||
|  |  | ||||||
|  | export interface MediaInfo { | ||||||
|  |   title: string; | ||||||
|  |   artists: string; | ||||||
|  |   album: string; | ||||||
|  |   icon: string; | ||||||
|  |   status: MediaStatus; | ||||||
|  |   url: string; | ||||||
|  |   current: string; | ||||||
|  |   duration: string; | ||||||
|  |   image: string; | ||||||
|  | } | ||||||
							
								
								
									
										4
									
								
								src/models/mediaStatus.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | |||||||
|  | export enum MediaStatus { | ||||||
|  |   playing = "playing", | ||||||
|  |   paused = "paused", | ||||||
|  | } | ||||||
							
								
								
									
										12
									
								
								src/models/options.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | |||||||
|  | export interface Options { | ||||||
|  |   title: string; | ||||||
|  |   artists: string; | ||||||
|  |   album: string; | ||||||
|  |   status: string; | ||||||
|  |   url: string; | ||||||
|  |   current: string; | ||||||
|  |   duration: string; | ||||||
|  |   "app-name": string; | ||||||
|  |   image: string; | ||||||
|  |   icon: string; | ||||||
|  | } | ||||||
							
								
								
									
										
											BIN
										
									
								
								src/pages/settings/fonts/NotoSans-Bold.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								src/pages/settings/fonts/NotoSans-Light.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								src/pages/settings/fonts/NotoSans-Regular.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								src/pages/settings/fonts/NotoSans-SemiBold.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -1,98 +0,0 @@ | |||||||
| let notifications; |  | ||||||
| let playBackControl; |  | ||||||
| let api; |  | ||||||
| let port; |  | ||||||
| let menuBar; |  | ||||||
|  |  | ||||||
| const { store, settings } = require("./../../scripts/settings"); |  | ||||||
| const { ipcRenderer } = require("electron"); |  | ||||||
| const globalEvents = require("./../../constants/globalEvents"); |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Sync the UI forms with the current settings |  | ||||||
|  */ |  | ||||||
| function refreshSettings() { |  | ||||||
|   notifications.checked = store.get(settings.notifications); |  | ||||||
|   playBackControl.checked = store.get(settings.playBackControl); |  | ||||||
|   api.checked = store.get(settings.api); |  | ||||||
|   port.value = store.get(settings.apiSettings.port); |  | ||||||
|   menuBar.checked = store.get(settings.menuBar); |  | ||||||
|   trayIcon.checked = store.get(settings.trayIcon); |  | ||||||
|   mpris.checked = store.get(settings.mpris); |  | ||||||
|   enableCustomHotkeys.checked = store.get(settings.enableCustomHotkeys); |  | ||||||
|   enableDiscord.checked = store.get(settings.enableDiscord); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Open an url in the default browsers |  | ||||||
|  */ |  | ||||||
| window.openExternal = function (url) { |  | ||||||
|   const { shell } = require("electron"); |  | ||||||
|   shell.openExternal(url); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * hide the settings window |  | ||||||
|  */ |  | ||||||
| window.hide = function () { |  | ||||||
|   ipcRenderer.send(globalEvents.hideSettings); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Restart tidal-hifi after changes |  | ||||||
|  */ |  | ||||||
| window.restart = function () { |  | ||||||
|   const remote = require("electron").remote; |  | ||||||
|   remote.app.relaunch(); |  | ||||||
|   remote.app.exit(0); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Bind UI components to functions after DOMContentLoaded |  | ||||||
|  */ |  | ||||||
| window.addEventListener("DOMContentLoaded", () => { |  | ||||||
|   function get(id) { |  | ||||||
|     return document.getElementById(id); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   function addInputListener(source, key) { |  | ||||||
|     source.addEventListener("input", function (event, data) { |  | ||||||
|       if (this.value === "on") { |  | ||||||
|         store.set(key, source.checked); |  | ||||||
|       } else { |  | ||||||
|         store.set(key, this.value); |  | ||||||
|       } |  | ||||||
|       ipcRenderer.send(globalEvents.storeChanged); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   ipcRenderer.on("refreshData", () => { |  | ||||||
|     refreshSettings(); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   ipcRenderer.on("goToTab", (_, tab) => { |  | ||||||
|     document.getElementById(tab).click(); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   notifications = get("notifications"); |  | ||||||
|   playBackControl = get("playBackControl"); |  | ||||||
|   api = get("apiCheckbox"); |  | ||||||
|   port = get("port"); |  | ||||||
|   menuBar = get("menuBar"); |  | ||||||
|   trayIcon = get("trayIcon"); |  | ||||||
|   mpris = get("mprisCheckbox"); |  | ||||||
|   enableCustomHotkeys = get("enableCustomHotkeys"); |  | ||||||
|   enableDiscord = get("enableDiscord"); |  | ||||||
|  |  | ||||||
|   refreshSettings(); |  | ||||||
|  |  | ||||||
|   addInputListener(notifications, settings.notifications); |  | ||||||
|   addInputListener(playBackControl, settings.playBackControl); |  | ||||||
|   addInputListener(api, settings.api); |  | ||||||
|   addInputListener(port, settings.apiSettings.port); |  | ||||||
|   addInputListener(menuBar, settings.menuBar); |  | ||||||
|   addInputListener(trayIcon, settings.trayIcon); |  | ||||||
|   addInputListener(mpris, settings.mpris); |  | ||||||
|   addInputListener(enableCustomHotkeys, settings.enableCustomHotkeys); |  | ||||||
|   addInputListener(enableDiscord, settings.enableDiscord); |  | ||||||
| }); |  | ||||||
							
								
								
									
										157
									
								
								src/pages/settings/preload.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,157 @@ | |||||||
|  | import remote from "@electron/remote"; | ||||||
|  | import { ipcRenderer, shell } from "electron"; | ||||||
|  | import { globalEvents } from "../../constants/globalEvents"; | ||||||
|  | import { settings } from "../../constants/settings"; | ||||||
|  | import { settingsStore } from "./../../scripts/settings"; | ||||||
|  |  | ||||||
|  | let adBlock: HTMLInputElement, | ||||||
|  |   api: HTMLInputElement, | ||||||
|  |   customCSS: HTMLInputElement, | ||||||
|  |   disableBackgroundThrottle: HTMLInputElement, | ||||||
|  |   disableHardwareMediaKeys: HTMLInputElement, | ||||||
|  |   enableCustomHotkeys: HTMLInputElement, | ||||||
|  |   enableDiscord: HTMLInputElement, | ||||||
|  |   gpuRasterization: HTMLInputElement, | ||||||
|  |   menuBar: HTMLInputElement, | ||||||
|  |   minimizeOnClose: HTMLInputElement, | ||||||
|  |   mpris: HTMLInputElement, | ||||||
|  |   notifications: HTMLInputElement, | ||||||
|  |   playBackControl: HTMLInputElement, | ||||||
|  |   port: HTMLInputElement, | ||||||
|  |   singleInstance: HTMLInputElement, | ||||||
|  |   skipArtists: HTMLInputElement, | ||||||
|  |   skippedArtists: HTMLInputElement, | ||||||
|  |   trayIcon: HTMLInputElement, | ||||||
|  |   updateFrequency: HTMLInputElement; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Sync the UI forms with the current settings | ||||||
|  |  */ | ||||||
|  | function refreshSettings() { | ||||||
|  |   adBlock.checked = settingsStore.get(settings.adBlock); | ||||||
|  |   api.checked = settingsStore.get(settings.api); | ||||||
|  |   customCSS.value = settingsStore.get(settings.customCSS); | ||||||
|  |   disableBackgroundThrottle.checked = settingsStore.get(settings.disableBackgroundThrottle); | ||||||
|  |   disableHardwareMediaKeys.checked = settingsStore.get(settings.flags.disableHardwareMediaKeys); | ||||||
|  |   enableCustomHotkeys.checked = settingsStore.get(settings.enableCustomHotkeys); | ||||||
|  |   enableDiscord.checked = settingsStore.get(settings.enableDiscord); | ||||||
|  |   gpuRasterization.checked = settingsStore.get(settings.flags.gpuRasterization); | ||||||
|  |   menuBar.checked = settingsStore.get(settings.menuBar); | ||||||
|  |   minimizeOnClose.checked = settingsStore.get(settings.minimizeOnClose); | ||||||
|  |   mpris.checked = settingsStore.get(settings.mpris); | ||||||
|  |   notifications.checked = settingsStore.get(settings.notifications); | ||||||
|  |   playBackControl.checked = settingsStore.get(settings.playBackControl); | ||||||
|  |   port.value = settingsStore.get(settings.apiSettings.port); | ||||||
|  |   singleInstance.checked = settingsStore.get(settings.singleInstance); | ||||||
|  |   skipArtists.checked = settingsStore.get(settings.skipArtists); | ||||||
|  |   skippedArtists.value = settingsStore.get<string, string[]>(settings.skippedArtists).join("\n"); | ||||||
|  |   trayIcon.checked = settingsStore.get(settings.trayIcon); | ||||||
|  |   updateFrequency.value = settingsStore.get(settings.updateFrequency); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Open an url in the default browsers | ||||||
|  |  */ | ||||||
|  | function openExternal(url: string) { | ||||||
|  |   shell.openExternal(url); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * hide the settings window | ||||||
|  |  */ | ||||||
|  | function hide() { | ||||||
|  |   ipcRenderer.send(globalEvents.hideSettings); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Restart tidal-hifi after changes | ||||||
|  |  */ | ||||||
|  | function restart() { | ||||||
|  |   remote.app.relaunch(); | ||||||
|  |   remote.app.exit(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Bind UI components to functions after DOMContentLoaded | ||||||
|  |  */ | ||||||
|  | window.addEventListener("DOMContentLoaded", () => { | ||||||
|  |   function get(id: string): HTMLInputElement { | ||||||
|  |     return document.getElementById(id) as HTMLInputElement; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   document.getElementById("close").addEventListener("click", hide); | ||||||
|  |   document.getElementById("restart").addEventListener("click", restart); | ||||||
|  |   document.querySelectorAll(".external-link").forEach((elem) => | ||||||
|  |     elem.addEventListener("click", function (event) { | ||||||
|  |       openExternal((event.target as HTMLElement).getAttribute("data-url")); | ||||||
|  |     }) | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   function addInputListener(source: HTMLInputElement, key: string) { | ||||||
|  |     source.addEventListener("input", () => { | ||||||
|  |       if (source.value === "on") { | ||||||
|  |         settingsStore.set(key, source.checked); | ||||||
|  |       } else { | ||||||
|  |         settingsStore.set(key, source.value); | ||||||
|  |       } | ||||||
|  |       ipcRenderer.send(globalEvents.storeChanged); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function addTextAreaListener(source: HTMLInputElement, key: string) { | ||||||
|  |     source.addEventListener("input", () => { | ||||||
|  |       settingsStore.set(key, source.value.split("\n")); | ||||||
|  |       ipcRenderer.send(globalEvents.storeChanged); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   ipcRenderer.on("refreshData", () => { | ||||||
|  |     refreshSettings(); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   ipcRenderer.on("goToTab", (_, tab) => { | ||||||
|  |     document.getElementById(tab).click(); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   adBlock = get("adBlock"); | ||||||
|  |   api = get("apiCheckbox"); | ||||||
|  |   customCSS = get("customCSS"); | ||||||
|  |   disableBackgroundThrottle = get("disableBackgroundThrottle"); | ||||||
|  |   disableHardwareMediaKeys = get("disableHardwareMediaKeys"); | ||||||
|  |   enableCustomHotkeys = get("enableCustomHotkeys"); | ||||||
|  |   enableDiscord = get("enableDiscord"); | ||||||
|  |   gpuRasterization = get("gpuRasterization"); | ||||||
|  |   menuBar = get("menuBar"); | ||||||
|  |   minimizeOnClose = get("minimizeOnClose"); | ||||||
|  |   mpris = get("mprisCheckbox"); | ||||||
|  |   notifications = get("notifications"); | ||||||
|  |   playBackControl = get("playBackControl"); | ||||||
|  |   port = get("port"); | ||||||
|  |   trayIcon = get("trayIcon"); | ||||||
|  |   skipArtists = get("skipArtists"); | ||||||
|  |   skippedArtists = get("skippedArtists"); | ||||||
|  |   singleInstance = get("singleInstance"); | ||||||
|  |   updateFrequency = get("updateFrequency"); | ||||||
|  |  | ||||||
|  |   refreshSettings(); | ||||||
|  |  | ||||||
|  |   addInputListener(adBlock, settings.adBlock); | ||||||
|  |   addInputListener(api, settings.api); | ||||||
|  |   addTextAreaListener(customCSS, settings.customCSS); | ||||||
|  |   addInputListener(disableBackgroundThrottle, settings.disableBackgroundThrottle); | ||||||
|  |   addInputListener(disableHardwareMediaKeys, settings.flags.disableHardwareMediaKeys); | ||||||
|  |   addInputListener(enableCustomHotkeys, settings.enableCustomHotkeys); | ||||||
|  |   addInputListener(enableDiscord, settings.enableDiscord); | ||||||
|  |   addInputListener(gpuRasterization, settings.flags.gpuRasterization); | ||||||
|  |   addInputListener(menuBar, settings.menuBar); | ||||||
|  |   addInputListener(minimizeOnClose, settings.minimizeOnClose); | ||||||
|  |   addInputListener(mpris, settings.mpris); | ||||||
|  |   addInputListener(notifications, settings.notifications); | ||||||
|  |   addInputListener(playBackControl, settings.playBackControl); | ||||||
|  |   addInputListener(port, settings.apiSettings.port); | ||||||
|  |   addInputListener(skipArtists, settings.skipArtists); | ||||||
|  |   addTextAreaListener(skippedArtists, settings.skippedArtists); | ||||||
|  |   addInputListener(singleInstance, settings.singleInstance); | ||||||
|  |   addInputListener(trayIcon, settings.trayIcon); | ||||||
|  |   addInputListener(updateFrequency, settings.updateFrequency); | ||||||
|  | }); | ||||||
| @@ -2,462 +2,302 @@ | |||||||
| <html lang="en"> | <html lang="en"> | ||||||
|  |  | ||||||
| <head> | <head> | ||||||
|     <meta charset="UTF-8" /> |   <title>Tidal Hi-Fi settings</title> | ||||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0" /> |   <meta charset="UTF-8" /> | ||||||
|     <meta http-equiv="X-UA-Compatible" content="ie=edge" /> |   <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||||
|  |   <meta http-equiv="X-UA-Compatible" content="ie=edge" /> | ||||||
|  |   <link rel="stylesheet" href="./settings.css" /> | ||||||
| </head> | </head> | ||||||
|  |  | ||||||
| <body> | <body class="settings-window"> | ||||||
|     <div class="header"> |   <div class="settings-window__wrapper"> | ||||||
|         <h1 class="title">Settings</h1> |     <div class="settings-window__drag-area"></div> | ||||||
|         <a href="javascript:hide();" class="exitWindow"> |     <a id="close" class="settings-window__close-button" title="Close settings"> | ||||||
|  |       <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 348.333 348.334" class="settings-window__svg-icon"> | ||||||
|  |         <path fill="white" d="M336.559,68.611L231.016,174.165l105.543,105.549c15.699,15.705,15.699,41.145,0,56.85 | ||||||
|  |             c-7.844,7.844-18.128,11.769-28.407,11.769c-10.296,0-20.581-3.919-28.419-11.769L174.167,231.003L68.609,336.563 | ||||||
|  |             c-7.843,7.844-18.128,11.769-28.416,11.769c-10.285,0-20.563-3.919-28.413-11.769c-15.699-15.698-15.699-41.139,0-56.85 | ||||||
|  |             l105.54-105.549L11.774,68.611c-15.699-15.699-15.699-41.145,0-56.844c15.696-15.687,41.127-15.687,56.829,0l105.563,105.554 | ||||||
|  |             L279.721,11.767c15.705-15.687,41.139-15.687,56.832,0C352.258,27.466,352.258,52.912,336.559,68.611z" /> | ||||||
|  |       </svg> | ||||||
|  |     </a> | ||||||
|  |  | ||||||
|             <svg version="1.1" xmlns="http://www.w3.org/2000/svg" width="18px" height="18px" viewBox="0 0 348.333 348.334"> |     <main class="settings"> | ||||||
|                 <g> |       <input type="radio" name="tab" id="general" checked /> | ||||||
|                     <path fill="white" d="M336.559,68.611L231.016,174.165l105.543,105.549c15.699,15.705,15.699,41.145,0,56.85 |       <label for="general">General</label> | ||||||
|               c-7.844,7.844-18.128,11.769-28.407,11.769c-10.296,0-20.581-3.919-28.419-11.769L174.167,231.003L68.609,336.563 |  | ||||||
|               c-7.843,7.844-18.128,11.769-28.416,11.769c-10.285,0-20.563-3.919-28.413-11.769c-15.699-15.698-15.699-41.139,0-56.85 |  | ||||||
|               l105.54-105.549L11.774,68.611c-15.699-15.699-15.699-41.145,0-56.844c15.696-15.687,41.127-15.687,56.829,0l105.563,105.554 |  | ||||||
|               L279.721,11.767c15.705-15.687,41.139-15.687,56.832,0C352.258,27.466,352.258,52.912,336.559,68.611z" /> |  | ||||||
|             </svg> |  | ||||||
|  |  | ||||||
|         </a> |       <input type="radio" name="tab" id="api" /> | ||||||
|     </div> |       <label for="api">API</label> | ||||||
|     <div class="body"> |  | ||||||
|         <div class="tabset"> |  | ||||||
|             <!-- Tab 1 --> |  | ||||||
|             <input type="radio" name="tabset" id="general" checked /> |  | ||||||
|             <label for="general">General</label> |  | ||||||
|             <!-- Tab 2 --> |  | ||||||
|             <input type="radio" name="tabset" id="api" /> |  | ||||||
|             <label for="api">Api</label> |  | ||||||
|  |  | ||||||
|             <!-- Integrations tab --> |       <input type="radio" name="tab" id="integrations" /> | ||||||
|             <input type="radio" name="tabset" id="integrations" /> |       <label for="integrations">Integrations</label> | ||||||
|             <label for="integrations">Integrations</label> |  | ||||||
|  |  | ||||||
|             <!-- about tab --> |       <input type="radio" name="tab" id="advanced" /> | ||||||
|             <input type="radio" name="tabset" id="about" /> |       <label for="advanced">Advanced</label> | ||||||
|             <label for="about">About</label> |  | ||||||
|  |  | ||||||
|             <div class="tab-panels"> |       <input type="radio" name="tab" id="about" /> | ||||||
|                 <section id="general" class="tab-panel"> |       <label for="about">About</label> | ||||||
|                     <div class="section"> |  | ||||||
|                         <h3>Playback</h3> |  | ||||||
|                         <div class="option"> |  | ||||||
|                             <h4>Notifications</h4> |  | ||||||
|                             <p> |  | ||||||
|                                 Whether to show a notification when a new song starts. |  | ||||||
|                             </p> |  | ||||||
|                             <label class="switch"> |  | ||||||
|                                 <input id="notifications" type="checkbox"> |  | ||||||
|                                 <span class="slider round"></span> |  | ||||||
|                             </label> |  | ||||||
|                         </div> |  | ||||||
|                     </div> |  | ||||||
|                     <div class="section"> |  | ||||||
|                         <h3>UI</h3> |  | ||||||
|                         <div class="option"> |  | ||||||
|                             <h4>Menubar</h4> |  | ||||||
|                             <p> |  | ||||||
|                                 Show Tidal-hifi's menu bar |  | ||||||
|                             </p> |  | ||||||
|                             <label class="switch"> |  | ||||||
|                                 <input id="menuBar" type="checkbox"> |  | ||||||
|                                 <span class="slider round"></span> |  | ||||||
|                             </label> |  | ||||||
|                         </div> |  | ||||||
|                     </div> |  | ||||||
|                     <div class="section"> |  | ||||||
|                       <h3>System</h3> |  | ||||||
|                       <div class="option"> |  | ||||||
|                           <h4>Tray icon</h4> |  | ||||||
|                           <p> |  | ||||||
|                               Show Tidal-hifi's tray icon<br /> |  | ||||||
|                           </p> |  | ||||||
|                           <label class="switch"> |  | ||||||
|                               <input id="trayIcon" type="checkbox"> |  | ||||||
|                               <span class="slider round"></span> |  | ||||||
|                           </label> |  | ||||||
|                       </div> |  | ||||||
|                       <div class="option"> |  | ||||||
|                         <h4>Hotkeys</h4> |  | ||||||
|                         <p> |  | ||||||
|                             Enables extra hotkeys to achieve feature parity with the <a href = "javascript:openExternal('https://defkey.com/tidal-desktop-shortcuts')">desktop apps</a><br /> |  | ||||||
|                         </p> |  | ||||||
|                         <label class="switch"> |  | ||||||
|                             <input id="enableCustomHotkeys" type="checkbox"> |  | ||||||
|                             <span class="slider round"></span> |  | ||||||
|                         </label> |  | ||||||
|                     </div> |  | ||||||
|                   </div> |  | ||||||
|                 </section> |  | ||||||
|                 <section id="api" class="tab-panel"> |  | ||||||
|                     <div class="section"> |  | ||||||
|                         <h3>Api</h3> |  | ||||||
|                         <p style="margin-bottom: 15px;"> |  | ||||||
|                             Tidal-hifi has a web api built in to allow users to get current song information. You can optionally enable playback control as well. |  | ||||||
|                         </p> |  | ||||||
|  |  | ||||||
|                         <div class="option"> |       <div class="tabs"> | ||||||
|                             <h4>Web API</h4> |         <section id="general-section" class="tabs__section"> | ||||||
|                             <p> |           <div class="group"> | ||||||
|                                 Whether to enable the Tidal-hifi web api |             <p class="group__title">Playback</p> | ||||||
|                             </p> |             <div class="group__option"> | ||||||
|                             <label class="switch"> |               <div class="group__description"> | ||||||
|                                 <input id="apiCheckbox" type="checkbox"> |                 <h4>Notifications</h4> | ||||||
|                                 <span class="slider round"></span> |                 <p>Show a notification when a new song starts.</p> | ||||||
|                             </label> |               </div> | ||||||
|                         </div> |               <label class="switch"> | ||||||
|                         <div class="option"> |                 <input id="notifications" type="checkbox" /> | ||||||
|                             <h4 style="margin-bottom: 5px;">API port</h4> |                 <span class="switch__slider"></span> | ||||||
|                             <input id="port" type="text" class="freeTextInput" name="port"> |               </label> | ||||||
|                         </div> |  | ||||||
|                         <div class="option"> |  | ||||||
|                             <h4>Playback control</h4> |  | ||||||
|                             <p> |  | ||||||
|                                 Whether to enable playback control from the api |  | ||||||
|                             </p> |  | ||||||
|                             <label class="switch"> |  | ||||||
|                                 <input id="playBackControl" type="checkbox"> |  | ||||||
|                                 <span class="slider round"></span> |  | ||||||
|                             </label> |  | ||||||
|                         </div> |  | ||||||
|                     </div> |  | ||||||
|                 </section> |  | ||||||
|                 <section id="integrations" class="tab-panel"> |  | ||||||
|                   <div class="section"> |  | ||||||
|                       <h3>integrations</h3> |  | ||||||
|                       <p style="margin-bottom: 15px;"> |  | ||||||
|                           Tidal-hifi is extensible trough the use of integrations. You can enable or disable integrations here |  | ||||||
|                       </p> |  | ||||||
|                       <div class="option"> |  | ||||||
|                           <h4>mpris-player</h4> |  | ||||||
|                           <p> |  | ||||||
|                               Whether to enable the mpris media player controls for Linux systems |  | ||||||
|                           </p> |  | ||||||
|                           <label class="switch"> |  | ||||||
|                               <input id="mprisCheckbox" type="checkbox"> |  | ||||||
|                               <span class="slider round"></span> |  | ||||||
|                           </label> |  | ||||||
|                       </div> |  | ||||||
|                       <div class="option"> |  | ||||||
|                         <h4>Discord RPC</h4> |  | ||||||
|                         <p> |  | ||||||
|                             Show what you're listening to on Discord |  | ||||||
|                         </p> |  | ||||||
|                         <label class="switch"> |  | ||||||
|                             <input id="enableDiscord" type="checkbox"> |  | ||||||
|                             <span class="slider round"></span> |  | ||||||
|                         </label> |  | ||||||
|                     </div> |  | ||||||
|                   </div> |  | ||||||
|               </section> |  | ||||||
|                 <section id="about" class="tab-panel"> |  | ||||||
|                   <div class="section"> |  | ||||||
|                     <img style="width: 100px; height: auto; display: block; margin: 0 auto; margin-bottom: 20px; margin-top: 20px;" src = "./icon.png"> |  | ||||||
|                     <p style="max-width: 350px; display:block; margin: 0 auto; text-align: center;"> |  | ||||||
|                       <a href ="javascript:openExternal('https://github.com/Mastermindzh/tidal-hifi');">Tidal-hifi</a> is made by <a href ="javascript:openExternal('https://www.rickvanlieshout.com')">Rick van Lieshout</a>.<br /> |  | ||||||
|                       It uses <a href="javascript:openExternal('https://castlabs.com/');">castlabs</a> versions of Electron for widevine support. |  | ||||||
|                     </p> |  | ||||||
|  |  | ||||||
|                   </div> |  | ||||||
|                 </section> |  | ||||||
|  |  | ||||||
|               <small>Some settings require a restart of Tidal-hifi. To do so click the button below:</small> |  | ||||||
|               <button onClick="restart()" style ="width: 100%">Restart Tidal-hifi</button> |  | ||||||
|             </div> |             </div> | ||||||
|         </div> |             <div class="group__option"> | ||||||
|     </div> |               <div class="group__description"> | ||||||
|  |                 <h4>Skip Artists automatically</h4> | ||||||
|  |                 <p>The following list of artists (1 per line) will be skipped automatically.</p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="skipArtists" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |             <textarea id="skippedArtists" class="textarea" cols="40" rows="5" spellcheck="false"></textarea> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Block ads</h4> | ||||||
|  |                 <p> | ||||||
|  |                   Disabled audio & visual ads, unlocked lyrics, suggested track, track info, | ||||||
|  |                   unlimited skips | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="adBlock" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |           <div class="group"> | ||||||
|  |             <p class="group__title">UI</p> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Fixed menubar</h4> | ||||||
|  |                 <p>Always show TIDAL Hi-Fi's menu bar.</p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="menuBar" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |           <div class="group"> | ||||||
|  |             <p class="group__title">System</p> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Tray icon</h4> | ||||||
|  |                 <p>Show TIDAL Hi-Fi's tray icon.</p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="trayIcon" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Minimize on Close</h4> | ||||||
|  |                 <p>Minimize window on close instead.</p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="minimizeOnClose" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Hotkeys</h4> | ||||||
|  |                 <p> | ||||||
|  |                   Enable extra hotkeys to achieve feature parity with the | ||||||
|  |                   <a class="external-link" data-url="https://defkey.com/tidal-desktop-shortcuts">desktop apps</a>. | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="enableCustomHotkeys" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Single instance</h4> | ||||||
|  |                 <p>Prevent opening multiple TIDAL Hi-Fi's instances.</p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="singleInstance" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </section> | ||||||
|  |  | ||||||
|  |         <section id="api-section" class="tabs__section"> | ||||||
|  |           <div class="group"> | ||||||
|  |             <p class="group__title">API</p> | ||||||
|  |             <div class="group__description"> | ||||||
|  |               <p> | ||||||
|  |                 TIDAL Hi-Fi has a built-in web API to allow users to get current song information. | ||||||
|  |                 You can optionally enable playback control as well. | ||||||
|  |               </p> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Web API</h4> | ||||||
|  |                 <p>Enable the TIDAL Hi-Fi web API.</p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="apiCheckbox" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <label for="port">API port</label> | ||||||
|  |                 <input id="port" type="number" class="text-input" name="port" /> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Playback control</h4> | ||||||
|  |                 <p>Enable playback control from the web API.</p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="playBackControl" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </section> | ||||||
|  |  | ||||||
|  |         <section id="integrations-section" class="tabs__section"> | ||||||
|  |           <div class="group"> | ||||||
|  |             <p class="group__title">Integrations</p> | ||||||
|  |             <div class="group__description"> | ||||||
|  |               <p> | ||||||
|  |                 TIDAL Hi-Fi is extensible through the use of integrations. You can enable or | ||||||
|  |                 disable them here. | ||||||
|  |               </p> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>MPRIS</h4> | ||||||
|  |                 <p> | ||||||
|  |                   Enable MPRIS interface which provides a mechanism for discovery, querying and | ||||||
|  |                   basic playback control on Linux systems. | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="mprisCheckbox" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Discord RPC</h4> | ||||||
|  |                 <p>Show what you're listening to on Discord.</p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="enableDiscord" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </section> | ||||||
|  |  | ||||||
|  |         <section id="advanced-section" class="tabs__section"> | ||||||
|  |           <div class="group"> | ||||||
|  |             <p class="group__title">Settings</p> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Update frequency</h4> | ||||||
|  |                 <p> | ||||||
|  |                   The amount of time, in milliseconds, that tidal-hifi will refresh its playback info by scraping the | ||||||
|  |                   website. | ||||||
|  |                   The default of 500 seems to work in more cases but if you are fine with a bit more resource usage you | ||||||
|  |                   can decrease it as well. | ||||||
|  |                 </p> | ||||||
|  |                 <input id="updateFrequency" type="number" class="text-input" name="updateFrequency" /> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Custom CSS</h4> | ||||||
|  |                 <p> | ||||||
|  |                   The css that you put in here will be injected into a style tag in the head of the document. | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |           <textarea id="customCSS" class="textarea" cols="40" rows="8" spellcheck="false"></textarea> | ||||||
|  |  | ||||||
|  |           <div class="group"> | ||||||
|  |             <p class="group__title">Flags</p> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Disable hardware built-in media keys</h4> | ||||||
|  |                 <p> | ||||||
|  |                   Also prevents certain desktop environments from recognizing the chrome MPRIS | ||||||
|  |                   client separately from the custom MPRIS client. | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="disableHardwareMediaKeys" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Enable GPU rasterization</h4> | ||||||
|  |                 <p>Move a part of the rendering to the GPU for increased performance.</p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="gpuRasterization" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |             <div class="group__option"> | ||||||
|  |               <div class="group__description"> | ||||||
|  |                 <h4>Disable Background Throttling</h4> | ||||||
|  |                 <p> | ||||||
|  |                   Makes app more responsive while in the background, at the cost of performance. | ||||||
|  |                 </p> | ||||||
|  |               </div> | ||||||
|  |               <label class="switch"> | ||||||
|  |                 <input id="disableBackgroundThrottle" type="checkbox" /> | ||||||
|  |                 <span class="switch__slider"></span> | ||||||
|  |               </label> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </section> | ||||||
|  |  | ||||||
|  |         <section id="about-section" class="tabs__section about-section"> | ||||||
|  |           <img alt="tidal icon" class="about-section__icon" src="./icon.png" /> | ||||||
|  |           <p class="about-section__text"> | ||||||
|  |             <a class="external-link" data-url="https://github.com/Mastermindzh/tidal-hifi">TIDAL Hi-Fi</a> | ||||||
|  |             is made by | ||||||
|  |             <a class="external-link" data-url="https://www.rickvanlieshout.com"> | ||||||
|  |               Rick van Lieshout</a>. <br />It uses | ||||||
|  |             <a class="external-link" data-url="https://castlabs.com/">Castlabs'</a> | ||||||
|  |             version of Electron for widevine support. | ||||||
|  |           </p> | ||||||
|  |         </section> | ||||||
|  |  | ||||||
|  |         <footer class="footer"> | ||||||
|  |           <p class="footer__note"> | ||||||
|  |             Some settings may require a restart of TIDAL Hi-Fi. To do so, click the button below: | ||||||
|  |           </p> | ||||||
|  |           <button class="footer__button" id="restart">Restart TIDAL Hi-Fi</button> | ||||||
|  |         </footer> | ||||||
|  |       </div> | ||||||
|  |     </main> | ||||||
|  |   </div> | ||||||
| </body> | </body> | ||||||
|  |  | ||||||
| <style> |  | ||||||
|     .header { |  | ||||||
|         -webkit-user-select: none; |  | ||||||
|         -webkit-app-region: drag; |  | ||||||
|     } |  | ||||||
|     .header a { |  | ||||||
|       -webkit-app-region: no-drag; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     * { |  | ||||||
|         margin: 0%; |  | ||||||
|         padding: 0%; |  | ||||||
|         color: #ffffff; |  | ||||||
|         font-weight: 400; |  | ||||||
|         font-stretch: normal; |  | ||||||
|         -webkit-font-smoothing: antialiased; |  | ||||||
|         font-family: nationale, nationale-regular, Helvetica, sans-serif; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     html, |  | ||||||
|     body { |  | ||||||
|         height: 100%; |  | ||||||
|         background-color: black; |  | ||||||
|         display: flex; |  | ||||||
|         flex-direction: column; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     h2 { |  | ||||||
|         font-size: 1.2rem; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     small { |  | ||||||
|         font-style: italic; |  | ||||||
|         color: #72777f; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .header { |  | ||||||
|         background-color: #242528; |  | ||||||
|         border-bottom: 1px solid #5a5a5a; |  | ||||||
|         height: 50px; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .title { |  | ||||||
|         float: left; |  | ||||||
|         line-height: 50px; |  | ||||||
|         margin-left: 15px; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .accent { |  | ||||||
|         color: #0ff; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .exitWindow { |  | ||||||
|         border: none; |  | ||||||
|         outline: none; |  | ||||||
|         text-decoration: none; |  | ||||||
|         font-size: 1.4rem; |  | ||||||
|         float: right; |  | ||||||
|         margin-right: 15px; |  | ||||||
|         height: 50px; |  | ||||||
|         line-height: 50px; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .exitWindow:focus { |  | ||||||
|       border: none; |  | ||||||
|       outline: none; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .exitWindow svg { |  | ||||||
|         height: 50px; |  | ||||||
|         color: white; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .section { |  | ||||||
|         padding-top: 10px; |  | ||||||
|         padding-bottom: 10px; |  | ||||||
|         border-bottom: 1px solid rgba(246, 245, 255, .1); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .section .option { |  | ||||||
|         margin-bottom: 15px; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .section .option p { |  | ||||||
|         max-width: 75%; |  | ||||||
|         float: left |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .section .option label { |  | ||||||
|         float: right; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .section:after, |  | ||||||
|     .section .option:after { |  | ||||||
|         content: ""; |  | ||||||
|         display: table; |  | ||||||
|         clear: both; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .section h3 { |  | ||||||
|         margin-bottom: 15px; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .section h4 { |  | ||||||
|         font-size: 0.9rem; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .section p { |  | ||||||
|         color: #72777f; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .bottom-border { |  | ||||||
|         border-bottom: 1px solid #0ff; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .body { |  | ||||||
|         padding: 15px; |  | ||||||
|         flex: 1 1 auto; |  | ||||||
|         position: relative; |  | ||||||
|         overflow-y: auto; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .body::-webkit-scrollbar-track { |  | ||||||
|         -webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.3); |  | ||||||
|         box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.3); |  | ||||||
|         border-radius: 5px; |  | ||||||
|         background-color: 2a2a2a; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .body::-webkit-scrollbar { |  | ||||||
|         width: 10px; |  | ||||||
|         background-color: #2a2a2a; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .body::-webkit-scrollbar-thumb { |  | ||||||
|         border-radius: 10px; |  | ||||||
|         -webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, .3); |  | ||||||
|         box-shadow: inset 0 0 6px rgba(0, 0, 0, .3); |  | ||||||
|         background-color: #5a5a5a; |  | ||||||
|     } |  | ||||||
|     /* Tabs */ |  | ||||||
|  |  | ||||||
|     .tabset > input[type="radio"] { |  | ||||||
|         position: absolute; |  | ||||||
|         left: -200vw; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .tabset .tab-panel { |  | ||||||
|         display: none; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .tabset > input:first-child:checked ~ .tab-panels > .tab-panel:first-child, |  | ||||||
|     .tabset > input:nth-child(3):checked ~ .tab-panels > .tab-panel:nth-child(2), |  | ||||||
|     .tabset > input:nth-child(5):checked ~ .tab-panels > .tab-panel:nth-child(3), |  | ||||||
|     .tabset > input:nth-child(7):checked ~ .tab-panels > .tab-panel:nth-child(4) |  | ||||||
|      { |  | ||||||
|         display: block; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .tabset > label { |  | ||||||
|         position: relative; |  | ||||||
|         display: inline-block; |  | ||||||
|         padding: 15px 0px 10px; |  | ||||||
|         border-bottom: 0; |  | ||||||
|         cursor: pointer; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .tabset > input + label { |  | ||||||
|         color: #e0e0e0; |  | ||||||
|         margin-right: 30px; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .tabset > input:checked + label { |  | ||||||
|         color: #0ff; |  | ||||||
|         border-bottom: 2px solid #0ff; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .tab-panel { |  | ||||||
|         padding: 10px 0; |  | ||||||
|     } |  | ||||||
|     /* switches */ |  | ||||||
|     /* The switch - the box around the slider */ |  | ||||||
|  |  | ||||||
|     .switch { |  | ||||||
|         position: relative; |  | ||||||
|         display: inline-block; |  | ||||||
|         width: 50px; |  | ||||||
|         height: 28px; |  | ||||||
|     } |  | ||||||
|     /* Hide default HTML checkbox */ |  | ||||||
|  |  | ||||||
|     .switch input { |  | ||||||
|         opacity: 0; |  | ||||||
|         width: 0; |  | ||||||
|         height: 0; |  | ||||||
|     } |  | ||||||
|     /* The slider */ |  | ||||||
|  |  | ||||||
|     .slider { |  | ||||||
|         position: absolute; |  | ||||||
|         cursor: pointer; |  | ||||||
|         top: 0; |  | ||||||
|         left: 0; |  | ||||||
|         right: 0; |  | ||||||
|         bottom: 0; |  | ||||||
|         background-color: rgba(246, 245, 255, .1); |  | ||||||
|         -webkit-transition: .4s; |  | ||||||
|         transition: .4s; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .slider:before { |  | ||||||
|         position: absolute; |  | ||||||
|         content: ""; |  | ||||||
|         height: 24px; |  | ||||||
|         width: 24px; |  | ||||||
|         left: 2px; |  | ||||||
|         bottom: 2px; |  | ||||||
|         background-color: white; |  | ||||||
|         -webkit-transition: .4s; |  | ||||||
|         transition: .4s; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     input:checked + .slider { |  | ||||||
|         background-color: #0ff; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     input:focus + .slider { |  | ||||||
|         box-shadow: 0 0 1px #0ff; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     input:checked + .slider:before { |  | ||||||
|         -webkit-transform: translateX(22px); |  | ||||||
|         -ms-transform: translateX(22px); |  | ||||||
|         transform: translateX(22px); |  | ||||||
|     } |  | ||||||
|     /* Rounded sliders */ |  | ||||||
|  |  | ||||||
|     .slider.round { |  | ||||||
|         border-radius: 34px; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .slider.round:before { |  | ||||||
|         border-radius: 50%; |  | ||||||
|     } |  | ||||||
|     /* input field */ |  | ||||||
|  |  | ||||||
|     input { |  | ||||||
|         background: transparent; |  | ||||||
|         border: 0; |  | ||||||
|         border-bottom: 1px solid rgba(246, 245, 255, .1); |  | ||||||
|         color: rgba(229, 238, 255, .6); |  | ||||||
|         width: 100%; |  | ||||||
|         display: block; |  | ||||||
|         padding: 0 0 12px; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .freeTextInput:focus { |  | ||||||
|         outline: none; |  | ||||||
|         border-bottom: 1px solid #0ff; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /* buttons */ |  | ||||||
|     button{ |  | ||||||
|       border:none; |  | ||||||
|       background:none; |  | ||||||
|       align-items: center; |  | ||||||
|       background-color: rgba(229,238,255,.2); |  | ||||||
|       display: inline-flex; |  | ||||||
|       justify-content: center; |  | ||||||
|       border-radius: 12px; |  | ||||||
|       height: 48px; |  | ||||||
|       line-height: 49px; |  | ||||||
|       padding: 0 24px; |  | ||||||
|       text-align: center; |  | ||||||
|       overflow: hidden; |  | ||||||
|       text-overflow: ellipsis; |  | ||||||
|       white-space: nowrap; |  | ||||||
|       transition: background .35s ease; |  | ||||||
|       min-height: 0; |  | ||||||
|       min-width: 0; |  | ||||||
|       font-size: 1.14286rem; |  | ||||||
|       font-family: nationale,nationale-regular,Helvetica,sans-serif; |  | ||||||
|       margin-top: 10px; |  | ||||||
|       cursor: pointer; |  | ||||||
|     } |  | ||||||
|     button:hover{ |  | ||||||
|       background-color: rgba(229,238,255,.3); |  | ||||||
|     } |  | ||||||
| </style> |  | ||||||
| </html> | </html> | ||||||
							
								
								
									
										363
									
								
								src/pages/settings/settings.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,363 @@ | |||||||
|  | // --- Variables --- | ||||||
|  |  | ||||||
|  | $black: #17171a; | ||||||
|  | $grey-333: #333; | ||||||
|  | $white: #f9f9f9; | ||||||
|  | $tidal-blue: #0ff; | ||||||
|  | $tidal-grey: #72777f; | ||||||
|  | $tidal-grey-darker: #404248; | ||||||
|  | $tidal-grey-darker-focus: #55585f; | ||||||
|  | $tidal-grey-darkest: #242528; | ||||||
|  |  | ||||||
|  | // --- Fonts --- | ||||||
|  |  | ||||||
|  | @font-face { | ||||||
|  |   font-family: "Noto Sans"; | ||||||
|  |   font-weight: 300; | ||||||
|  |   src: url("fonts/NotoSans-Light.ttf") format("truetype"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @font-face { | ||||||
|  |   font-family: "Noto Sans"; | ||||||
|  |   font-weight: normal; | ||||||
|  |   src: url("fonts/NotoSans-Regular.ttf") format("truetype"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @font-face { | ||||||
|  |   font-family: "Noto Sans"; | ||||||
|  |   font-weight: 600; | ||||||
|  |   src: url("fonts/NotoSans-SemiBold.ttf") format("truetype"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @font-face { | ||||||
|  |   font-family: "Noto Sans"; | ||||||
|  |   font-weight: bold; | ||||||
|  |   src: url("fonts/NotoSans-Bold.ttf") format("truetype"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | $font1: "Noto Sans", helvetica, sans-serif; | ||||||
|  |  | ||||||
|  | // --- Mixins --- | ||||||
|  |  | ||||||
|  | @mixin drag($enabled: true) { | ||||||
|  |   @if $enabled { | ||||||
|  |     -webkit-app-region: drag; | ||||||
|  |   } @else { | ||||||
|  |     -webkit-app-region: no-drag; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | button { | ||||||
|  |   cursor: pointer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // --- Settings window --- | ||||||
|  |  | ||||||
|  | html { | ||||||
|  |   height: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .external-link { | ||||||
|  |   @extend button; | ||||||
|  |  | ||||||
|  |   text-decoration: underline; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .settings-window { | ||||||
|  |   height: 100%; | ||||||
|  |   margin: 0; | ||||||
|  |   color: $white; | ||||||
|  |   font-family: $font1; | ||||||
|  |  | ||||||
|  |   &__wrapper { | ||||||
|  |     height: 100%; | ||||||
|  |     background: $black; | ||||||
|  |     box-shadow: inset 0 0 2px 0 $tidal-grey; | ||||||
|  |     overflow: hidden; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   &__drag-area { | ||||||
|  |     @include drag; | ||||||
|  |  | ||||||
|  |     position: absolute; | ||||||
|  |     width: 100%; | ||||||
|  |     height: 50px; | ||||||
|  |     z-index: 0; | ||||||
|  |     user-select: none; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   &__close-button { | ||||||
|  |     @extend button; | ||||||
|  |     @include drag(false); | ||||||
|  |  | ||||||
|  |     position: absolute; | ||||||
|  |     top: 12px; | ||||||
|  |     right: 10px; | ||||||
|  |     padding: 10px; | ||||||
|  |     border-radius: 100%; | ||||||
|  |     z-index: 1; | ||||||
|  |  | ||||||
|  |     &:hover { | ||||||
|  |       background: $grey-333; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   &__svg-icon { | ||||||
|  |     display: block; | ||||||
|  |     width: 18px; | ||||||
|  |     height: 18px; | ||||||
|  |     opacity: 0.7; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // --- Settings tabs --- | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .settings { | ||||||
|  |   height: 100%; | ||||||
|  |   margin: 20px 0; | ||||||
|  |   padding-left: 15px; | ||||||
|  |   font-size: 0; | ||||||
|  |  | ||||||
|  |   input { | ||||||
|  |     &[type="radio"] { | ||||||
|  |       margin-right: -18px; | ||||||
|  |       transform: scale(0); | ||||||
|  |       outline: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     & + label { | ||||||
|  |       @include drag(false); | ||||||
|  |  | ||||||
|  |       display: inline-block; | ||||||
|  |       position: relative; | ||||||
|  |       margin-right: 35px; | ||||||
|  |       padding-bottom: 8px; | ||||||
|  |       border-bottom: 0; | ||||||
|  |       font-size: 16px; | ||||||
|  |       cursor: pointer; | ||||||
|  |       z-index: 1; | ||||||
|  |       user-select: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     &:checked + label { | ||||||
|  |       border-bottom: 2px solid $tidal-blue; | ||||||
|  |       color: $tidal-blue; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .tabs { | ||||||
|  |   height: calc(100% - 70px); | ||||||
|  |   padding-right: 15px; | ||||||
|  |   font-size: 16px; | ||||||
|  |   overflow: auto; | ||||||
|  |  | ||||||
|  |   &__section { | ||||||
|  |     display: none; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @for $i from 1 to 6 { | ||||||
|  |     .settings > input:nth-child(#{$i * 2 - 1}):checked ~ & > .tabs__section:nth-child(#{$i}) { | ||||||
|  |       display: block; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   &::-webkit-scrollbar { | ||||||
|  |     width: 10px; | ||||||
|  |  | ||||||
|  |     &-thumb { | ||||||
|  |       border-radius: 10px; | ||||||
|  |       background-color: $tidal-grey-darker; | ||||||
|  |       box-shadow: inset 0 0 10px 2px $tidal-grey-darkest; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // --- Settings group --- | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .group { | ||||||
|  |   padding: 10px 0; | ||||||
|  |   border-bottom: 1px solid $grey-333; | ||||||
|  |  | ||||||
|  |   &:last-child { | ||||||
|  |     border: 0; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   &__title { | ||||||
|  |     margin-bottom: 10px; | ||||||
|  |     font-size: 16px; | ||||||
|  |     font-weight: bold; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   &__option { | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   &__description { | ||||||
|  |     flex-grow: 1; | ||||||
|  |  | ||||||
|  |     h4, | ||||||
|  |     label { | ||||||
|  |       display: block; | ||||||
|  |       margin-top: 10px; | ||||||
|  |       margin-bottom: 0; | ||||||
|  |       font-size: 14px; | ||||||
|  |       font-weight: 600; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     p { | ||||||
|  |       margin-top: 5px; | ||||||
|  |       margin-bottom: 8px; | ||||||
|  |       color: $tidal-grey; | ||||||
|  |       font-size: 14px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .text-input { | ||||||
|  |       display: block; | ||||||
|  |       width: 100%; | ||||||
|  |       margin-bottom: 10px; | ||||||
|  |       padding: 5px 0; | ||||||
|  |       transition: 0.2s; | ||||||
|  |       border: 0; | ||||||
|  |       border-bottom: solid 1px $grey-333; | ||||||
|  |       outline: none; | ||||||
|  |       background: transparent; | ||||||
|  |       color: $tidal-grey; | ||||||
|  |       font-size: 14px; | ||||||
|  |  | ||||||
|  |       &:focus { | ||||||
|  |         border-color: $tidal-blue; | ||||||
|  |         color: $white; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // --- Switch slider component --- | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .switch { | ||||||
|  |   $this: &; | ||||||
|  |  | ||||||
|  |   position: relative; | ||||||
|  |   min-width: 50px; | ||||||
|  |   height: 28px; | ||||||
|  |   margin-left: 10px; | ||||||
|  |  | ||||||
|  |   input { | ||||||
|  |     transform: scale(0); | ||||||
|  |     outline: none; | ||||||
|  |  | ||||||
|  |     &:checked + #{$this}__slider { | ||||||
|  |       background-color: $tidal-blue; | ||||||
|  |  | ||||||
|  |       &::before { | ||||||
|  |         transform: translateX(22px); | ||||||
|  |         background-color: $white; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     &:focus + #{$this}__slider { | ||||||
|  |       box-shadow: inset 0 0 0 1px $tidal-blue; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   &__slider { | ||||||
|  |     @extend button; | ||||||
|  |  | ||||||
|  |     position: absolute; | ||||||
|  |     inset: 0; | ||||||
|  |     transition: 0.4s; | ||||||
|  |     border-radius: 40px; | ||||||
|  |     background-color: $tidal-grey-darkest; | ||||||
|  |  | ||||||
|  |     &::before { | ||||||
|  |       position: absolute; | ||||||
|  |       bottom: 2px; | ||||||
|  |       left: 2px; | ||||||
|  |       width: 24px; | ||||||
|  |       height: 24px; | ||||||
|  |       transition: 0.4s; | ||||||
|  |       border-radius: 50%; | ||||||
|  |       background-color: $white; | ||||||
|  |       content: ""; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // --- Textarea component | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .textarea { | ||||||
|  |   min-width: 100%; | ||||||
|  |   max-width: 100%; | ||||||
|  |   min-height: 50px; | ||||||
|  |   max-height: 100px; | ||||||
|  |   padding: 8px; | ||||||
|  |   transition: 0.2s; | ||||||
|  |   border: 0; | ||||||
|  |   border-bottom: 1px solid transparent; | ||||||
|  |   background: $tidal-grey-darkest; | ||||||
|  |   color: $tidal-grey; | ||||||
|  |   font-size: 13px; | ||||||
|  |   box-sizing: border-box; | ||||||
|  |  | ||||||
|  |   &:focus { | ||||||
|  |     border-color: $tidal-blue; | ||||||
|  |     outline: none; | ||||||
|  |     color: $white; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // --- About section --- | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .about-section { | ||||||
|  |   padding-top: 120px; | ||||||
|  |   text-align: center; | ||||||
|  |  | ||||||
|  |   &__icon { | ||||||
|  |     display: inline-block; | ||||||
|  |     width: 100px; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   &__text { | ||||||
|  |     display: block; | ||||||
|  |     max-width: 350px; | ||||||
|  |     margin: 20px auto 0; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // --- Footer --- | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .footer { | ||||||
|  |   position: sticky; | ||||||
|  |   top: calc(100% - 120px); | ||||||
|  |   height: 100px; | ||||||
|  |   padding-top: 20px; | ||||||
|  |   text-align: center; | ||||||
|  |  | ||||||
|  |   &__note { | ||||||
|  |     max-width: 300px; | ||||||
|  |     margin: 0 auto 15px; | ||||||
|  |     color: $tidal-grey; | ||||||
|  |     font-size: 12px; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   &__button { | ||||||
|  |     @extend button; | ||||||
|  |  | ||||||
|  |     display: block; | ||||||
|  |     height: 48px; | ||||||
|  |     margin: auto; | ||||||
|  |     padding: 0 24px; | ||||||
|  |     transition: 0.2s; | ||||||
|  |     border: 0; | ||||||
|  |     border-radius: 12px; | ||||||
|  |     background: $tidal-grey-darker; | ||||||
|  |     color: $white; | ||||||
|  |     font-size: 16px; | ||||||
|  |  | ||||||
|  |     &:hover { | ||||||
|  |       background: $tidal-grey-darker-focus; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										336
									
								
								src/preload.js
									
									
									
									
									
								
							
							
						
						| @@ -1,336 +0,0 @@ | |||||||
| const { setTitle, getTitle } = require("./scripts/window-functions"); |  | ||||||
| const { dialog, process } = require("electron").remote; |  | ||||||
| const { store, settings } = require("./scripts/settings"); |  | ||||||
| const { ipcRenderer } = require("electron"); |  | ||||||
| const { app } = require("electron").remote; |  | ||||||
| const { downloadFile } = require("./scripts/download"); |  | ||||||
| const statuses = require("./constants/statuses"); |  | ||||||
| const hotkeys = require("./scripts/hotkeys"); |  | ||||||
| const globalEvents = require("./constants/globalEvents"); |  | ||||||
| const notifier = require("node-notifier"); |  | ||||||
| const notificationPath = `${app.getPath("userData")}/notification.jpg`; |  | ||||||
| let currentSong = ""; |  | ||||||
| let player; |  | ||||||
|  |  | ||||||
| const elements = { |  | ||||||
|   play: '*[data-test="play"]', |  | ||||||
|   pause: '*[data-test="pause"]', |  | ||||||
|   next: '*[data-test="next"]', |  | ||||||
|   previous: 'button[data-test="previous"]', |  | ||||||
|   title: '*[data-test^="footer-track-title"]', |  | ||||||
|   artists: '*[class^="css-14o5h2y"]', |  | ||||||
|   home: '*[data-test="menu--home"]', |  | ||||||
|   back: '[class^="backwardButton"]', |  | ||||||
|   forward: '[class^="forwardButton"]', |  | ||||||
|   search: '[class^="searchField"]', |  | ||||||
|   shuffle: '*[data-test="shuffle"]', |  | ||||||
|   repeat: '*[data-test="repeat"]', |  | ||||||
|   block: '[class="blockButton"]', |  | ||||||
|   account: '*[data-test^="profile-image-button"]', |  | ||||||
|   settings: '*[data-test^="open-settings"]', |  | ||||||
|   media: '*[data-test="current-media-imagery"]', |  | ||||||
|   image: "img", |  | ||||||
|   url: 'a[href*="/track/"]', |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Get an element from the dom |  | ||||||
|    * @param {*} key key in elements object to fetch |  | ||||||
|    */ |  | ||||||
|   get: function (key) { |  | ||||||
|     return window.document.querySelector(this[key.toLowerCase()]); |  | ||||||
|   }, |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Get the icon of the current song |  | ||||||
|    */ |  | ||||||
|   getSongIcon: function () { |  | ||||||
|     const figure = this.get("media"); |  | ||||||
|  |  | ||||||
|     if (figure) { |  | ||||||
|       const mediaElement = figure.querySelector(this["image"]); |  | ||||||
|       if (mediaElement) { |  | ||||||
|         return mediaElement.src; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return ""; |  | ||||||
|   }, |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Shorthand function to get the text of a dom element |  | ||||||
|    * @param {*} key key in elements object to fetch |  | ||||||
|    */ |  | ||||||
|   getText: function (key) { |  | ||||||
|     const element = this.get(key); |  | ||||||
|     return element ? element.textContent : ""; |  | ||||||
|   }, |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Shorthand function to click a dom element |  | ||||||
|    * @param {*} key key in elements object to fetch |  | ||||||
|    */ |  | ||||||
|   click: function (key) { |  | ||||||
|     this.get(key).click(); |  | ||||||
|     return this; |  | ||||||
|   }, |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Shorthand function to focus a dom element |  | ||||||
|    * @param {*} key key in elements object to fetch |  | ||||||
|    */ |  | ||||||
|   focus: function (key) { |  | ||||||
|     return this.get(key).focus(); |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Play or pause the current song |  | ||||||
|  */ |  | ||||||
| function playPause() { |  | ||||||
|   const play = elements.get("play"); |  | ||||||
|  |  | ||||||
|   if (play) { |  | ||||||
|     elements.click("play"); |  | ||||||
|   } else { |  | ||||||
|     elements.click("pause"); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Add hotkeys for when tidal is focused |  | ||||||
|  * Reflects the desktop hotkeys found on: |  | ||||||
|  * https://defkey.com/tidal-desktop-shortcuts |  | ||||||
|  */ |  | ||||||
| function addHotKeys() { |  | ||||||
|   if (store.get(settings.enableCustomHotkeys)) { |  | ||||||
|     hotkeys.add("Control+p", function () { |  | ||||||
|       elements.click("account").click("settings"); |  | ||||||
|     }); |  | ||||||
|     hotkeys.add("Control+l", function () { |  | ||||||
|       handleLogout(); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     hotkeys.add("Control+h", function () { |  | ||||||
|       elements.click("home"); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     hotkeys.add("backspace", function () { |  | ||||||
|       elements.click("back"); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     hotkeys.add("shift+backspace", function () { |  | ||||||
|       elements.click("forward"); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     hotkeys.add("control+u", function () { |  | ||||||
|       // reloading window without cache should show the update bar if applicable |  | ||||||
|       window.location.reload(true); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     hotkeys.add("control+r", function () { |  | ||||||
|       elements.click("repeat"); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // always add the hotkey for the settings window |  | ||||||
|   hotkeys.add("control+=", function () { |  | ||||||
|     ipcRenderer.send(globalEvents.showSettings); |  | ||||||
|   }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * This function will ask the user whether he/she wants to log out. |  | ||||||
|  * It will log the user out if he/she selects "yes" |  | ||||||
|  */ |  | ||||||
| function handleLogout() { |  | ||||||
|   const logoutOptions = ["Cancel", "Yes, please", "No, thanks"]; |  | ||||||
|  |  | ||||||
|   dialog.showMessageBox( |  | ||||||
|     null, |  | ||||||
|     { |  | ||||||
|       type: "question", |  | ||||||
|       title: "Logging out", |  | ||||||
|       message: "Are you sure you want to log out?", |  | ||||||
|       buttons: logoutOptions, |  | ||||||
|       defaultId: 2, |  | ||||||
|     }, |  | ||||||
|     function (response) { |  | ||||||
|       if (logoutOptions.indexOf("Yes, please") == response) { |  | ||||||
|         for (let i = 0; i < window.localStorage.length; i++) { |  | ||||||
|           const key = window.localStorage.key(i); |  | ||||||
|           if (key.startsWith("_TIDAL_activeSession")) { |  | ||||||
|             window.localStorage.removeItem(key); |  | ||||||
|             i = window.localStorage.length + 1; |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|         window.location.reload(); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   ); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Add ipc event listeners. |  | ||||||
|  * Some actions triggered outside of the site need info from the site. |  | ||||||
|  */ |  | ||||||
| function addIPCEventListeners() { |  | ||||||
|   window.addEventListener("DOMContentLoaded", () => { |  | ||||||
|     ipcRenderer.on("globalEvent", (event, args) => { |  | ||||||
|       switch (args) { |  | ||||||
|         case globalEvents.playPause: |  | ||||||
|           playPause(); |  | ||||||
|           break; |  | ||||||
|         case globalEvents.next: |  | ||||||
|           elements.click("next"); |  | ||||||
|           break; |  | ||||||
|         case globalEvents.previous: |  | ||||||
|           elements.click("previous"); |  | ||||||
|           break; |  | ||||||
|         case globalEvents.play: |  | ||||||
|           elements.click("play"); |  | ||||||
|           break; |  | ||||||
|         case globalEvents.pause: |  | ||||||
|           elements.click("pause"); |  | ||||||
|           break; |  | ||||||
|       } |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Update the current status of tidal (e.g playing or paused) |  | ||||||
|  */ |  | ||||||
| function updateStatus() { |  | ||||||
|   let pause = elements.get("pause"); |  | ||||||
|   let status; |  | ||||||
|  |  | ||||||
|   // if pause button is visible tidal is playing |  | ||||||
|   if (pause) { |  | ||||||
|     status = statuses.playing; |  | ||||||
|   } else { |  | ||||||
|     status = statuses.paused; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (status) { |  | ||||||
|     ipcRenderer.send(globalEvents.updateStatus, status); |  | ||||||
|     if (player) { |  | ||||||
|       player.playbackStatus = status == statuses.paused ? "Paused" : "Playing"; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Watch for song changes and update title + notify |  | ||||||
|  */ |  | ||||||
| setInterval(function () { |  | ||||||
|   const title = elements.getText("title"); |  | ||||||
|   const url = elements.get("url").href.replace(/[^0-9]/g, ""); |  | ||||||
|   const artists = elements.getText("artists"); |  | ||||||
|   const songDashArtistTitle = `${title} - ${artists}`; |  | ||||||
|  |  | ||||||
|   updateStatus(); |  | ||||||
|  |  | ||||||
|   if (getTitle() !== songDashArtistTitle) { |  | ||||||
|     setTitle(songDashArtistTitle); |  | ||||||
|  |  | ||||||
|     if (currentSong !== songDashArtistTitle) { |  | ||||||
|       currentSong = songDashArtistTitle; |  | ||||||
|       const image = elements.getSongIcon(); |  | ||||||
|  |  | ||||||
|       const options = { |  | ||||||
|         title, |  | ||||||
|         message: artists, |  | ||||||
|         url: `https://tidal.com/browse/track/${url}`, |  | ||||||
|       }; |  | ||||||
|       new Promise((resolve, reject) => { |  | ||||||
|         if (image.startsWith("http")) { |  | ||||||
|           downloadFile(image, notificationPath).then( |  | ||||||
|             () => { |  | ||||||
|               options.icon = notificationPath; |  | ||||||
|               resolve(); |  | ||||||
|             }, |  | ||||||
|             () => { |  | ||||||
|               // if the image can't be downloaded then continue without it |  | ||||||
|               resolve(); |  | ||||||
|             } |  | ||||||
|           ); |  | ||||||
|         } else { |  | ||||||
|           // if the image can't be found on the page continue without it |  | ||||||
|           resolve(); |  | ||||||
|         } |  | ||||||
|       }).then( |  | ||||||
|         () => { |  | ||||||
|           ipcRenderer.send(globalEvents.updateInfo, options); |  | ||||||
|           store.get(settings.notifications) && notifier.notify(options); |  | ||||||
|  |  | ||||||
|           if (player) { |  | ||||||
|             player.metadata = { |  | ||||||
|               ...player.metadata, |  | ||||||
|               ...{ |  | ||||||
|                 "xesam:title": title, |  | ||||||
|                 "xesam:artist": [artists], |  | ||||||
|                 "mpris:artUrl": image, |  | ||||||
|               }, |  | ||||||
|             }; |  | ||||||
|           } |  | ||||||
|         }, |  | ||||||
|         () => {} |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| }, 200); |  | ||||||
|  |  | ||||||
| if (process.platform === "linux" && store.get(settings.mpris)) { |  | ||||||
|   try { |  | ||||||
|     const Player = require("mpris-service"); |  | ||||||
|     player = Player({ |  | ||||||
|       name: "tidal-hifi", |  | ||||||
|       identity: "tidal-hifi", |  | ||||||
|       supportedUriSchemes: ["file"], |  | ||||||
|       supportedMimeTypes: [ |  | ||||||
|         "audio/mpeg", |  | ||||||
|         "audio/flac", |  | ||||||
|         "audio/x-flac", |  | ||||||
|         "application/ogg", |  | ||||||
|         "audio/wav", |  | ||||||
|       ], |  | ||||||
|       supportedInterfaces: ["player"], |  | ||||||
|       desktopEntry: "tidal-hifi", |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     // Events |  | ||||||
|     var events = { |  | ||||||
|       next: "next", |  | ||||||
|       previous: "previous", |  | ||||||
|       pause: "pause", |  | ||||||
|       playpause: "playpause", |  | ||||||
|       stop: "stop", |  | ||||||
|       play: "play", |  | ||||||
|       loopStatus: "repeat", |  | ||||||
|       shuffle: "shuffle", |  | ||||||
|       seek: "seek", |  | ||||||
|     }; |  | ||||||
|     Object.keys(events).forEach(function (eventName) { |  | ||||||
|       player.on(eventName, function () { |  | ||||||
|         const eventValue = events[eventName]; |  | ||||||
|         switch (events[eventValue]) { |  | ||||||
|           case events.playpause: |  | ||||||
|             playPause(); |  | ||||||
|             break; |  | ||||||
|  |  | ||||||
|           default: |  | ||||||
|             elements.click(eventValue); |  | ||||||
|         } |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     player.on("quit", function () { |  | ||||||
|       app.quit(); |  | ||||||
|     }); |  | ||||||
|   } catch (exception) { |  | ||||||
|     console.log("player api not working"); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| addHotKeys(); |  | ||||||
| addIPCEventListeners(); |  | ||||||
							
								
								
									
										518
									
								
								src/preload.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,518 @@ | |||||||
|  | import { Notification, app, dialog } from "@electron/remote"; | ||||||
|  | import { ipcRenderer } from "electron"; | ||||||
|  | import Player from "mpris-service"; | ||||||
|  | import { globalEvents } from "./constants/globalEvents"; | ||||||
|  | import { settings } from "./constants/settings"; | ||||||
|  | import { statuses } from "./constants/statuses"; | ||||||
|  | import { Options } from "./models/options"; | ||||||
|  | import { downloadFile } from "./scripts/download"; | ||||||
|  | import { addHotkey } from "./scripts/hotkeys"; | ||||||
|  |  | ||||||
|  | import { settingsStore } from "./scripts/settings"; | ||||||
|  | import { setTitle } from "./scripts/window-functions"; | ||||||
|  | const notificationPath = `${app.getPath("userData")}/notification.jpg`; | ||||||
|  | const appName = "Tidal Hifi"; | ||||||
|  | let currentSong = ""; | ||||||
|  | let player: any; | ||||||
|  | let currentPlayStatus = statuses.paused; | ||||||
|  |  | ||||||
|  | const elements = { | ||||||
|  |   play: '*[data-test="play"]', | ||||||
|  |   pause: '*[data-test="pause"]', | ||||||
|  |   next: '*[data-test="next"]', | ||||||
|  |   previous: 'button[data-test="previous"]', | ||||||
|  |   title: '*[data-test^="footer-track-title"]', | ||||||
|  |   artists: '*[data-test^="grid-item-detail-text-title-artist"]', | ||||||
|  |   home: '*[data-test="menu--home"]', | ||||||
|  |   back: '[class^="backwardButton"]', | ||||||
|  |   forward: '[class^="forwardButton"]', | ||||||
|  |   search: '[class^="searchField"]', | ||||||
|  |   shuffle: '*[data-test="shuffle"]', | ||||||
|  |   repeat: '*[data-test="repeat"]', | ||||||
|  |   block: '[class="blockButton"]', | ||||||
|  |   account: '*[data-test^="profile-image-button"]', | ||||||
|  |   settings: '*[data-test^="open-settings"]', | ||||||
|  |   media: '*[data-test="current-media-imagery"]', | ||||||
|  |   image: "img", | ||||||
|  |   current: '*[data-test="current-time"]', | ||||||
|  |   duration: '*[data-test="duration"]', | ||||||
|  |   bar: '*[data-test="progress-bar"]', | ||||||
|  |   footer: "#footerPlayer", | ||||||
|  |   album_header_title: '.header-details [data-test="title"]', | ||||||
|  |   playing_title: 'span[data-test="table-cell-title"].css-geqnfr', | ||||||
|  |   album_name_cell: '[data-test="table-cell-album"]', | ||||||
|  |   tracklist_row: '[data-test="tracklist-row"]', | ||||||
|  |   volume: '*[data-test="volume"]', | ||||||
|  |   /** | ||||||
|  |    * Get an element from the dom | ||||||
|  |    * @param {*} key key in elements object to fetch | ||||||
|  |    */ | ||||||
|  |   get: function (key: string) { | ||||||
|  |     return window.document.querySelector(this[key.toLowerCase()]); | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Get the icon of the current song | ||||||
|  |    */ | ||||||
|  |   getSongIcon: function () { | ||||||
|  |     const figure = this.get("media"); | ||||||
|  |  | ||||||
|  |     if (figure) { | ||||||
|  |       const mediaElement = figure.querySelector(this["image"]); | ||||||
|  |       if (mediaElement) { | ||||||
|  |         return mediaElement.src.replace("80x80", "640x640"); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return ""; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * returns an array of all artists in the current song | ||||||
|  |    * @returns {Array} artists | ||||||
|  |    */ | ||||||
|  |   getArtistsArray: function () { | ||||||
|  |     const footer = this.get("footer"); | ||||||
|  |  | ||||||
|  |     if (footer) { | ||||||
|  |       const artists = footer.querySelectorAll(this.artists); | ||||||
|  |       if (artists) return Array.from(artists).map((artist) => (artist as HTMLElement).textContent); | ||||||
|  |     } | ||||||
|  |     return []; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * unify the artists array into a string separated by commas | ||||||
|  |    * @param {Array} artistsArray | ||||||
|  |    * @returns {String} artists | ||||||
|  |    */ | ||||||
|  |   getArtistsString: function (artistsArray: string[]) { | ||||||
|  |     if (artistsArray.length > 0) return artistsArray.join(", "); | ||||||
|  |     return "unknown artist(s)"; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   getAlbumName: function () { | ||||||
|  |     //If listening to an album, get its name from the header title | ||||||
|  |     if (window.location.href.includes("/album/")) { | ||||||
|  |       const albumName = window.document.querySelector(this.album_header_title); | ||||||
|  |       if (albumName) { | ||||||
|  |         return albumName.textContent; | ||||||
|  |       } | ||||||
|  |       //If listening to a playlist or a mix, get album name from the list | ||||||
|  |     } else if ( | ||||||
|  |       window.location.href.includes("/playlist/") || | ||||||
|  |       window.location.href.includes("/mix/") | ||||||
|  |     ) { | ||||||
|  |       if (currentPlayStatus === statuses.playing) { | ||||||
|  |         const row = window.document.querySelector(this.playing_title).closest(this.tracklist_row); | ||||||
|  |         if (row) { | ||||||
|  |           return row.querySelector(this.album_name_cell).textContent; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return ""; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   isMuted: function () { | ||||||
|  |     return this.get("volume").getAttribute("aria-checked") === "false"; // it's muted if aria-checked is false | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Shorthand function to get the text of a dom element | ||||||
|  |    * @param {*} key key in elements object to fetch | ||||||
|  |    */ | ||||||
|  |   getText: function (key: string) { | ||||||
|  |     const element = this.get(key); | ||||||
|  |     return element ? element.textContent : ""; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Shorthand function to click a dom element | ||||||
|  |    * @param {*} key key in elements object to fetch | ||||||
|  |    */ | ||||||
|  |   click: function (key: string) { | ||||||
|  |     this.get(key).click(); | ||||||
|  |     return this; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Shorthand function to focus a dom element | ||||||
|  |    * @param {*} key key in elements object to fetch | ||||||
|  |    */ | ||||||
|  |   focus: function (key: string) { | ||||||
|  |     return this.get(key).focus(); | ||||||
|  |   }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | function addCustomCss() { | ||||||
|  |   window.addEventListener("DOMContentLoaded", () => { | ||||||
|  |     const style = document.createElement("style"); | ||||||
|  |     style.innerHTML = settingsStore.get(settings.customCSS); | ||||||
|  |     document.head.appendChild(style); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Get the update frequency from the store | ||||||
|  |  * make sure it returns a number, if not use the default | ||||||
|  |  */ | ||||||
|  | function getUpdateFrequency() { | ||||||
|  |   const storeValue = settingsStore.get(settings.updateFrequency) as number; | ||||||
|  |   const defaultValue = 500; | ||||||
|  |  | ||||||
|  |   if (!isNaN(storeValue)) { | ||||||
|  |     return storeValue; | ||||||
|  |   } else { | ||||||
|  |     return defaultValue; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Play or pause the current song | ||||||
|  |  */ | ||||||
|  | function playPause() { | ||||||
|  |   const play = elements.get("play"); | ||||||
|  |  | ||||||
|  |   if (play) { | ||||||
|  |     elements.click("play"); | ||||||
|  |   } else { | ||||||
|  |     elements.click("pause"); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Add hotkeys for when tidal is focused | ||||||
|  |  * Reflects the desktop hotkeys found on: | ||||||
|  |  * https://defkey.com/tidal-desktop-shortcuts | ||||||
|  |  */ | ||||||
|  | function addHotKeys() { | ||||||
|  |   if (settingsStore.get(settings.enableCustomHotkeys)) { | ||||||
|  |     addHotkey("Control+p", function () { | ||||||
|  |       elements.click("account").click("settings"); | ||||||
|  |     }); | ||||||
|  |     addHotkey("Control+l", function () { | ||||||
|  |       handleLogout(); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     addHotkey("Control+h", function () { | ||||||
|  |       elements.click("home"); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     addHotkey("backspace", function () { | ||||||
|  |       elements.click("back"); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     addHotkey("shift+backspace", function () { | ||||||
|  |       elements.click("forward"); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     addHotkey("control+u", function () { | ||||||
|  |       // reloading window without cache should show the update bar if applicable | ||||||
|  |       window.location.reload(); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     addHotkey("control+r", function () { | ||||||
|  |       elements.click("repeat"); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // always add the hotkey for the settings window | ||||||
|  |   addHotkey("control+=", function () { | ||||||
|  |     ipcRenderer.send(globalEvents.showSettings); | ||||||
|  |   }); | ||||||
|  |   addHotkey("control+0", function () { | ||||||
|  |     ipcRenderer.send(globalEvents.showSettings); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * This function will ask the user whether he/she wants to log out. | ||||||
|  |  * It will log the user out if he/she selects "yes" | ||||||
|  |  */ | ||||||
|  | function handleLogout() { | ||||||
|  |   const logoutOptions = ["Cancel", "Yes, please", "No, thanks"]; | ||||||
|  |  | ||||||
|  |   dialog | ||||||
|  |     .showMessageBox(null, { | ||||||
|  |       type: "question", | ||||||
|  |       title: "Logging out", | ||||||
|  |       message: "Are you sure you want to log out?", | ||||||
|  |       buttons: logoutOptions, | ||||||
|  |       defaultId: 2, | ||||||
|  |     }) | ||||||
|  |     .then((result: { response: number }) => { | ||||||
|  |       if (logoutOptions.indexOf("Yes, please") == result.response) { | ||||||
|  |         for (let i = 0; i < window.localStorage.length; i++) { | ||||||
|  |           const key = window.localStorage.key(i); | ||||||
|  |           if (key.startsWith("_TIDAL_activeSession")) { | ||||||
|  |             window.localStorage.removeItem(key); | ||||||
|  |             break; | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |         window.location.reload(); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function addFullScreenListeners() { | ||||||
|  |   window.document.addEventListener("fullscreenchange", () => { | ||||||
|  |     ipcRenderer.send(globalEvents.refreshMenuBar); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Add ipc event listeners. | ||||||
|  |  * Some actions triggered outside of the site need info from the site. | ||||||
|  |  */ | ||||||
|  | function addIPCEventListeners() { | ||||||
|  |   window.addEventListener("DOMContentLoaded", () => { | ||||||
|  |     ipcRenderer.on("globalEvent", (_event, args) => { | ||||||
|  |       switch (args) { | ||||||
|  |         case globalEvents.playPause: | ||||||
|  |           playPause(); | ||||||
|  |           break; | ||||||
|  |         case globalEvents.next: | ||||||
|  |           elements.click("next"); | ||||||
|  |           break; | ||||||
|  |         case globalEvents.previous: | ||||||
|  |           elements.click("previous"); | ||||||
|  |           break; | ||||||
|  |         case globalEvents.play: | ||||||
|  |           elements.click("play"); | ||||||
|  |           break; | ||||||
|  |         case globalEvents.pause: | ||||||
|  |           elements.click("pause"); | ||||||
|  |           break; | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Update the current status of tidal (e.g playing or paused) | ||||||
|  |  */ | ||||||
|  | function getCurrentlyPlayingStatus() { | ||||||
|  |   const pause = elements.get("pause"); | ||||||
|  |   let status = undefined; | ||||||
|  |  | ||||||
|  |   // if pause button is visible tidal is playing | ||||||
|  |   if (pause) { | ||||||
|  |     status = statuses.playing; | ||||||
|  |   } else { | ||||||
|  |     status = statuses.paused; | ||||||
|  |   } | ||||||
|  |   return status; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Convert the duration from MM:SS to seconds | ||||||
|  |  * @param {*} duration | ||||||
|  |  */ | ||||||
|  | function convertDuration(duration: string) { | ||||||
|  |   const parts = duration.split(":"); | ||||||
|  |   return parseInt(parts[1]) + 60 * parseInt(parts[0]); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Update Tidal-hifi's media info | ||||||
|  |  * | ||||||
|  |  * @param {*} options | ||||||
|  |  */ | ||||||
|  | function updateMediaInfo(options: Options, notify: boolean) { | ||||||
|  |   if (options) { | ||||||
|  |     ipcRenderer.send(globalEvents.updateInfo, options); | ||||||
|  |     if (settingsStore.get(settings.notifications) && notify) { | ||||||
|  |       new Notification({ title: options.title, body: options.artists, icon: options.icon }).show(); | ||||||
|  |     } | ||||||
|  |     if (player) { | ||||||
|  |       player.metadata = { | ||||||
|  |         ...player.metadata, | ||||||
|  |         ...{ | ||||||
|  |           "xesam:title": options.title, | ||||||
|  |           "xesam:artist": [options.artists], | ||||||
|  |           "xesam:album": options.album, | ||||||
|  |           "mpris:artUrl": options.image, | ||||||
|  |           "mpris:length": convertDuration(options.duration) * 1000 * 1000, | ||||||
|  |           "mpris:trackid": "/org/mpris/MediaPlayer2/track/" + getTrackID(), | ||||||
|  |         }, | ||||||
|  |       }; | ||||||
|  |       player.playbackStatus = options.status == statuses.paused ? "Paused" : "Playing"; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Checks if Tidal is playing a video or song by grabbing the "a" element from the title. | ||||||
|  |  * If it's a song it returns the track URL, if not it will return undefined | ||||||
|  |  */ | ||||||
|  | function getTrackURL() { | ||||||
|  |   const id = getTrackID(); | ||||||
|  |   return `https://tidal.com/browse/track/${id}`; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function getTrackID() { | ||||||
|  |   const URLelement = elements.get("title").querySelector("a"); | ||||||
|  |   if (URLelement !== null) { | ||||||
|  |     const id = URLelement.href.replace(/\D/g, ""); | ||||||
|  |     return id; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return window.location; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function updateMediaSession(options: Options) { | ||||||
|  |   if ("mediaSession" in navigator) { | ||||||
|  |     navigator.mediaSession.metadata = new MediaMetadata({ | ||||||
|  |       title: options.title, | ||||||
|  |       artist: options.artists, | ||||||
|  |       album: options.album, | ||||||
|  |       artwork: [ | ||||||
|  |         { | ||||||
|  |           src: options.icon, | ||||||
|  |           sizes: "640x640", | ||||||
|  |           type: "image/png", | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Watch for song changes and update title + notify | ||||||
|  |  */ | ||||||
|  | setInterval(function () { | ||||||
|  |   const title = elements.getText("title"); | ||||||
|  |   const artistsArray = elements.getArtistsArray(); | ||||||
|  |   const artistsString = elements.getArtistsString(artistsArray); | ||||||
|  |   skipArtistsIfFoundInSkippedArtistsList(artistsArray); | ||||||
|  |  | ||||||
|  |   const album = elements.getAlbumName(); | ||||||
|  |   const current = elements.getText("current"); | ||||||
|  |   const duration = elements.getText("duration"); | ||||||
|  |   const songDashArtistTitle = `${title} - ${artistsString}`; | ||||||
|  |   const currentStatus = getCurrentlyPlayingStatus(); | ||||||
|  |   const options = { | ||||||
|  |     title, | ||||||
|  |     artists: artistsString, | ||||||
|  |     album: album, | ||||||
|  |     status: currentStatus, | ||||||
|  |     url: getTrackURL(), | ||||||
|  |     current, | ||||||
|  |     duration, | ||||||
|  |     "app-name": appName, | ||||||
|  |     image: "", | ||||||
|  |     icon: "", | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const titleOrArtistsChanged = currentSong !== songDashArtistTitle; | ||||||
|  |  | ||||||
|  |   // update title, url and play info with new info | ||||||
|  |   setTitle(songDashArtistTitle); | ||||||
|  |   getTrackURL(); | ||||||
|  |   currentSong = songDashArtistTitle; | ||||||
|  |   currentPlayStatus = currentStatus; | ||||||
|  |  | ||||||
|  |   const image = elements.getSongIcon(); | ||||||
|  |  | ||||||
|  |   new Promise<void>((resolve) => { | ||||||
|  |     if (image.startsWith("http")) { | ||||||
|  |       options.image = image; | ||||||
|  |       downloadFile(image, notificationPath).then( | ||||||
|  |         () => { | ||||||
|  |           options.icon = notificationPath; | ||||||
|  |           resolve(); | ||||||
|  |         }, | ||||||
|  |         () => { | ||||||
|  |           // if the image can't be downloaded then continue without it | ||||||
|  |           resolve(); | ||||||
|  |         } | ||||||
|  |       ); | ||||||
|  |     } else { | ||||||
|  |       // if the image can't be found on the page continue without it | ||||||
|  |       resolve(); | ||||||
|  |     } | ||||||
|  |   }).then(() => { | ||||||
|  |     updateMediaInfo(options, titleOrArtistsChanged); | ||||||
|  |     if (titleOrArtistsChanged) { | ||||||
|  |       updateMediaSession(options); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * automatically skip a song if the artists are found in the list of artists to skip | ||||||
|  |    * @param {*} artists array of artists | ||||||
|  |    */ | ||||||
|  |   function skipArtistsIfFoundInSkippedArtistsList(artists: string[]) { | ||||||
|  |     if (settingsStore.get(settings.skipArtists)) { | ||||||
|  |       const skippedArtists = settingsStore.get<string, string[]>(settings.skippedArtists); | ||||||
|  |       if (skippedArtists.length > 0) { | ||||||
|  |         const artistsToSkip = skippedArtists.map((artist) => artist); | ||||||
|  |         const artistNames = Object.values(artists).map((artist) => artist); | ||||||
|  |         const foundArtist = artistNames.some((artist) => artistsToSkip.includes(artist)); | ||||||
|  |         if (foundArtist) { | ||||||
|  |           elements.click("next"); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }, getUpdateFrequency()); | ||||||
|  |  | ||||||
|  | if (process.platform === "linux" && settingsStore.get(settings.mpris)) { | ||||||
|  |   try { | ||||||
|  |     player = Player({ | ||||||
|  |       name: "tidal-hifi", | ||||||
|  |       identity: "tidal-hifi", | ||||||
|  |       supportedUriSchemes: ["file"], | ||||||
|  |       supportedMimeTypes: [ | ||||||
|  |         "audio/mpeg", | ||||||
|  |         "audio/flac", | ||||||
|  |         "audio/x-flac", | ||||||
|  |         "application/ogg", | ||||||
|  |         "audio/wav", | ||||||
|  |       ], | ||||||
|  |       supportedInterfaces: ["player"], | ||||||
|  |       desktopEntry: "tidal-hifi", | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     // Events | ||||||
|  |     const events = { | ||||||
|  |       next: "next", | ||||||
|  |       previous: "previous", | ||||||
|  |       pause: "pause", | ||||||
|  |       playpause: "playpause", | ||||||
|  |       stop: "stop", | ||||||
|  |       play: "play", | ||||||
|  |       loopStatus: "repeat", | ||||||
|  |       shuffle: "shuffle", | ||||||
|  |       seek: "seek", | ||||||
|  |     } as { [key: string]: string }; | ||||||
|  |     Object.keys(events).forEach(function (eventName) { | ||||||
|  |       player.on(eventName, function () { | ||||||
|  |         const eventValue = events[eventName]; | ||||||
|  |         switch (events[eventValue]) { | ||||||
|  |           case events.playpause: | ||||||
|  |             playPause(); | ||||||
|  |             break; | ||||||
|  |  | ||||||
|  |           default: | ||||||
|  |             elements.click(eventValue); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |     // Override get position function | ||||||
|  |     player.getPosition = function () { | ||||||
|  |       return convertDuration(elements.getText("current")) * 1000 * 1000; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     player.on("quit", function () { | ||||||
|  |       app.quit(); | ||||||
|  |     }); | ||||||
|  |   } catch (exception) { | ||||||
|  |     console.log("player api not working"); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | addCustomCss(); | ||||||
|  | addHotKeys(); | ||||||
|  | addIPCEventListeners(); | ||||||
|  | addFullScreenListeners(); | ||||||
| @@ -1,52 +0,0 @@ | |||||||
| const discordrpc = require("discord-rpc"); |  | ||||||
| const clientId = '833617820704440341'; |  | ||||||
| const mediaInfoModule = require("./mediaInfo"); |  | ||||||
|  |  | ||||||
| const discordModule = []; |  | ||||||
|  |  | ||||||
| let discord; |  | ||||||
| let rpc; |  | ||||||
|  |  | ||||||
| const idleStatus = { |  | ||||||
|     details: `Browsing Tidal`, |  | ||||||
|     largeImageKey: 'tidal-hifi-icon', |  | ||||||
|     largeImageText: 'Tidal HiFi 2.0.0', |  | ||||||
|     instance: false, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| discordModule.initRPC = function () { |  | ||||||
|     rpc = new discordrpc.Client({ transport: 'ipc' }); |  | ||||||
|     rpc.login({ clientId }).catch(console.error); |  | ||||||
|     discordModule.rpc = rpc; |  | ||||||
|  |  | ||||||
|     rpc.on('ready', () => { |  | ||||||
|         rpc.setActivity(idleStatus); |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     discord = setInterval(() => { |  | ||||||
|         if (mediaInfoModule.mediaInfo.status == 'paused' && rpc) { |  | ||||||
|             rpc.setActivity(idleStatus); |  | ||||||
|         } else if (rpc) { |  | ||||||
|             rpc.setActivity({ |  | ||||||
|                 details: `Listening to ${mediaInfoModule.mediaInfo.title}`, |  | ||||||
|                 state: mediaInfoModule.mediaInfo.artist, |  | ||||||
|                 largeImageKey: 'tidal-hifi-icon', |  | ||||||
|                 largeImageText: 'Tidal HiFi 2.0.0', |  | ||||||
|                 buttons: [ |  | ||||||
|                     { label: "Play on Tidal", url: mediaInfoModule.mediaInfo.url } |  | ||||||
|                 ], |  | ||||||
|                 instance: false, |  | ||||||
|             }); |  | ||||||
|         } |  | ||||||
|     }, 15e3); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| discordModule.unRPC = function () { |  | ||||||
|     clearInterval(discord); |  | ||||||
|     rpc.clearActivity(); |  | ||||||
|     rpc.destroy(); |  | ||||||
|     rpc = false; |  | ||||||
|     discordModule.rpc = rpc; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| module.exports = discordModule; |  | ||||||
							
								
								
									
										88
									
								
								src/scripts/discord.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,88 @@ | |||||||
|  | import { Client } from "discord-rpc"; | ||||||
|  | import { app, ipcMain } from "electron"; | ||||||
|  | import { globalEvents } from "../constants/globalEvents"; | ||||||
|  | import { MediaStatus } from "../models/mediaStatus"; | ||||||
|  | import { mediaInfo } from "./mediaInfo"; | ||||||
|  |  | ||||||
|  | const clientId = "833617820704440341"; | ||||||
|  |  | ||||||
|  | function timeToSeconds(timeArray: string[]) { | ||||||
|  |   const minutes = parseInt(timeArray[0]) * 1; | ||||||
|  |   const seconds = minutes * 60 + parseInt(timeArray[1]) * 1; | ||||||
|  |   return seconds; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export let rpc: Client; | ||||||
|  |  | ||||||
|  | const observer = () => { | ||||||
|  |   if (mediaInfo.status == MediaStatus.paused && rpc) { | ||||||
|  |     rpc.setActivity(idleStatus); | ||||||
|  |   } else if (rpc) { | ||||||
|  |     const currentSeconds = timeToSeconds(mediaInfo.current.split(":")); | ||||||
|  |     const durationSeconds = timeToSeconds(mediaInfo.duration.split(":")); | ||||||
|  |     const date = new Date(); | ||||||
|  |     const now = (date.getTime() / 1000) | 0; | ||||||
|  |     const remaining = date.setSeconds(date.getSeconds() + (durationSeconds - currentSeconds)); | ||||||
|  |     if (mediaInfo.url) { | ||||||
|  |       rpc.setActivity({ | ||||||
|  |         ...idleStatus, | ||||||
|  |         ...{ | ||||||
|  |           details: `Listening to ${mediaInfo.title}`, | ||||||
|  |           state: mediaInfo.artists ? mediaInfo.artists : "unknown artist(s)", | ||||||
|  |           startTimestamp: now, | ||||||
|  |           endTimestamp: remaining, | ||||||
|  |           largeImageKey: mediaInfo.image, | ||||||
|  |           largeImageText: mediaInfo.album ? mediaInfo.album : `${idleStatus.largeImageText}`, | ||||||
|  |           buttons: [{ label: "Play on Tidal", url: mediaInfo.url }], | ||||||
|  |         }, | ||||||
|  |       }); | ||||||
|  |     } else { | ||||||
|  |       rpc.setActivity({ | ||||||
|  |         ...idleStatus, | ||||||
|  |         ...{ | ||||||
|  |           details: `Watching ${mediaInfo.title}`, | ||||||
|  |           state: mediaInfo.artists, | ||||||
|  |           startTimestamp: now, | ||||||
|  |           endTimestamp: remaining, | ||||||
|  |         }, | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const idleStatus = { | ||||||
|  |   details: `Browsing Tidal`, | ||||||
|  |   largeImageKey: "tidal-hifi-icon", | ||||||
|  |   largeImageText: `Tidal HiFi ${app.getVersion()}`, | ||||||
|  |   instance: false, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Set up the discord rpc and listen on globalEvents.updateInfo | ||||||
|  |  */ | ||||||
|  | export const initRPC = () => { | ||||||
|  |   rpc = new Client({ transport: "ipc" }); | ||||||
|  |   rpc.login({ clientId }).then( | ||||||
|  |     () => { | ||||||
|  |       rpc.on("ready", () => { | ||||||
|  |         rpc.setActivity(idleStatus); | ||||||
|  |       }); | ||||||
|  |       ipcMain.on(globalEvents.updateInfo, observer); | ||||||
|  |     }, | ||||||
|  |     () => { | ||||||
|  |       console.error("Can't connect to Discord, is it running?"); | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Remove any RPC connection with discord and remove the event listener on globalEvents.updateInfo | ||||||
|  |  */ | ||||||
|  | export const unRPC = () => { | ||||||
|  |   if (rpc) { | ||||||
|  |     rpc.clearActivity(); | ||||||
|  |     rpc.destroy(); | ||||||
|  |     rpc = null; | ||||||
|  |     ipcMain.removeListener(globalEvents.updateInfo, observer); | ||||||
|  |   } | ||||||
|  | }; | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| const download = {}; |  | ||||||
| const request = require("request"); |  | ||||||
| const fs = require("fs"); |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * download and save a file |  | ||||||
|  * @param {*} fileUrl url to download |  | ||||||
|  * @param {*} targetPath path to save it at |  | ||||||
|  */ |  | ||||||
| download.downloadFile = function(fileUrl, targetPath) { |  | ||||||
|   return new Promise((resolve, reject) => { |  | ||||||
|     var req = request({ |  | ||||||
|       method: "GET", |  | ||||||
|       uri: fileUrl, |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     var out = fs.createWriteStream(targetPath); |  | ||||||
|     req.pipe(out); |  | ||||||
|  |  | ||||||
|     req.on("end", resolve); |  | ||||||
|  |  | ||||||
|     req.on("error", reject); |  | ||||||
|   }); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| module.exports = download; |  | ||||||
							
								
								
									
										23
									
								
								src/scripts/download.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | |||||||
|  | import fs from "fs"; | ||||||
|  | import request from "request"; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * download and save a file | ||||||
|  |  * @param {string} fileUrl url to download | ||||||
|  |  * @param {string} targetPath path to save it at | ||||||
|  |  */ | ||||||
|  | export const downloadFile = function (fileUrl: string, targetPath: string) { | ||||||
|  |   return new Promise((resolve, reject) => { | ||||||
|  |     const req = request({ | ||||||
|  |       method: "GET", | ||||||
|  |       uri: fileUrl, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const out = fs.createWriteStream(targetPath); | ||||||
|  |     req.pipe(out); | ||||||
|  |  | ||||||
|  |     req.on("end", resolve); | ||||||
|  |  | ||||||
|  |     req.on("error", reject); | ||||||
|  |   }); | ||||||
|  | }; | ||||||
| @@ -1,70 +0,0 @@ | |||||||
| const express = require("express"); |  | ||||||
| const { mediaInfo } = require("./mediaInfo"); |  | ||||||
| const { store, settings } = require("./settings"); |  | ||||||
| const globalEvents = require("./../constants/globalEvents"); |  | ||||||
| const statuses = require("./../constants/statuses"); |  | ||||||
| const expressModule = {}; |  | ||||||
| const fs = require("fs"); |  | ||||||
|  |  | ||||||
| let expressInstance; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Function to enable tidal-hifi's express api |  | ||||||
|  */ |  | ||||||
| expressModule.run = function(mainWindow) { |  | ||||||
|   /** |  | ||||||
|    * Shorthand to handle a fire and forget global event |  | ||||||
|    * @param {*} res |  | ||||||
|    * @param {*} action |  | ||||||
|    */ |  | ||||||
|   function handleGlobalEvent(res, action) { |  | ||||||
|     mainWindow.webContents.send("globalEvent", action); |  | ||||||
|     res.sendStatus(200); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const expressApp = express(); |  | ||||||
|   expressApp.get("/", (req, res) => res.send("Hello World!")); |  | ||||||
|   expressApp.get("/current", (req, res) => res.json(mediaInfo)); |  | ||||||
|   expressApp.get("/image", (req, res) => { |  | ||||||
|     var stream = fs.createReadStream(mediaInfo.icon); |  | ||||||
|     stream.on("open", function() { |  | ||||||
|       res.set("Content-Type", "image/png"); |  | ||||||
|       stream.pipe(res); |  | ||||||
|     }); |  | ||||||
|     stream.on("error", function() { |  | ||||||
|       res.set("Content-Type", "text/plain"); |  | ||||||
|       res.status(404).end("Not found"); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   if (store.get(settings.playBackControl)) { |  | ||||||
|     expressApp.get("/play", (req, res) => handleGlobalEvent(res, globalEvents.play)); |  | ||||||
|     expressApp.get("/pause", (req, res) => handleGlobalEvent(res, globalEvents.pause)); |  | ||||||
|     expressApp.get("/next", (req, res) => handleGlobalEvent(res, globalEvents.next)); |  | ||||||
|     expressApp.get("/previous", (req, res) => handleGlobalEvent(res, globalEvents.previous)); |  | ||||||
|     expressApp.get("/playpause", (req, res) => { |  | ||||||
|       if (mediaInfo.status == statuses.playing) { |  | ||||||
|         handleGlobalEvent(res, globalEvents.pause); |  | ||||||
|       } else { |  | ||||||
|         handleGlobalEvent(res, globalEvents.play); |  | ||||||
|       } |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|   if (store.get(settings.api)) { |  | ||||||
|     let port = store.get(settings.apiSettings.port); |  | ||||||
|  |  | ||||||
|     expressInstance = expressApp.listen(port, "127.0.0.1", () => {}); |  | ||||||
|     expressInstance.on("error", function(e) { |  | ||||||
|       let message = e.code; |  | ||||||
|       if (e.code === "EADDRINUSE") { |  | ||||||
|         message = `Port ${port} in use.`; |  | ||||||
|       } |  | ||||||
|       const { dialog } = require("electron"); |  | ||||||
|       dialog.showErrorBox("Api failed to start.", message); |  | ||||||
|     }); |  | ||||||
|   } else { |  | ||||||
|     expressInstance = undefined; |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| module.exports = expressModule; |  | ||||||
							
								
								
									
										66
									
								
								src/scripts/express.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,66 @@ | |||||||
|  | import { BrowserWindow, dialog } from "electron"; | ||||||
|  | import express, { Response } from "express"; | ||||||
|  | import fs from "fs"; | ||||||
|  | import { globalEvents } from "./../constants/globalEvents"; | ||||||
|  | import { statuses } from "./../constants/statuses"; | ||||||
|  | import { mediaInfo } from "./mediaInfo"; | ||||||
|  | import { settingsStore } from "./settings"; | ||||||
|  | import { settings } from "../constants/settings"; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Function to enable tidal-hifi's express api | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | // expressModule.run = function (mainWindow) | ||||||
|  | export const startExpress = (mainWindow: BrowserWindow) => { | ||||||
|  |   /** | ||||||
|  |    * Shorthand to handle a fire and forget global event | ||||||
|  |    * @param {*} res | ||||||
|  |    * @param {*} action | ||||||
|  |    */ | ||||||
|  |   function handleGlobalEvent(res: Response, action: any) { | ||||||
|  |     mainWindow.webContents.send("globalEvent", action); | ||||||
|  |     res.sendStatus(200); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const expressApp = express(); | ||||||
|  |   expressApp.get("/", (req, res) => res.send("Hello World!")); | ||||||
|  |   expressApp.get("/current", (req, res) => res.json({ ...mediaInfo, artist: mediaInfo.artists })); | ||||||
|  |   expressApp.get("/image", (req, res) => { | ||||||
|  |     const stream = fs.createReadStream(mediaInfo.icon); | ||||||
|  |     stream.on("open", function () { | ||||||
|  |       res.set("Content-Type", "image/png"); | ||||||
|  |       stream.pipe(res); | ||||||
|  |     }); | ||||||
|  |     stream.on("error", function () { | ||||||
|  |       res.set("Content-Type", "text/plain"); | ||||||
|  |       res.status(404).end("Not found"); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   if (settingsStore.get(settings.playBackControl)) { | ||||||
|  |     expressApp.get("/play", (req, res) => handleGlobalEvent(res, globalEvents.play)); | ||||||
|  |     expressApp.get("/pause", (req, res) => handleGlobalEvent(res, globalEvents.pause)); | ||||||
|  |     expressApp.get("/next", (req, res) => handleGlobalEvent(res, globalEvents.next)); | ||||||
|  |     expressApp.get("/previous", (req, res) => handleGlobalEvent(res, globalEvents.previous)); | ||||||
|  |     expressApp.get("/playpause", (req, res) => { | ||||||
|  |       if (mediaInfo.status == statuses.playing) { | ||||||
|  |         handleGlobalEvent(res, globalEvents.pause); | ||||||
|  |       } else { | ||||||
|  |         handleGlobalEvent(res, globalEvents.play); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const port = settingsStore.get<string, number>(settings.apiSettings.port); | ||||||
|  |  | ||||||
|  |   const expressInstance = expressApp.listen(port, "127.0.0.1"); | ||||||
|  |   expressInstance.on("error", function (e: { code: string }) { | ||||||
|  |     let message = e.code; | ||||||
|  |     if (e.code === "EADDRINUSE") { | ||||||
|  |       message = `Port ${port} in use.`; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     dialog.showErrorBox("Api failed to start.", message); | ||||||
|  |   }); | ||||||
|  | }; | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| const hotkeyjs = require("hotkeys-js"); |  | ||||||
| const hotkeys = {}; |  | ||||||
|  |  | ||||||
| hotkeys.add = function(keys, func) { |  | ||||||
|   hotkeyjs(keys, function(event, args) { |  | ||||||
|     event.preventDefault(); |  | ||||||
|     func(event, args); |  | ||||||
|   }); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| module.exports = hotkeys; |  | ||||||
							
								
								
									
										11
									
								
								src/scripts/hotkeys.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | |||||||
|  | import hotkeyjs, { HotkeysEvent } from "hotkeys-js"; | ||||||
|  |  | ||||||
|  | export const addHotkey = function ( | ||||||
|  |   keys: string, | ||||||
|  |   func: (event?: KeyboardEvent, args?: HotkeysEvent) => void | ||||||
|  | ) { | ||||||
|  |   hotkeyjs(keys, function (event, args) { | ||||||
|  |     event.preventDefault(); | ||||||
|  |     func(event, args); | ||||||
|  |   }); | ||||||
|  | }; | ||||||
| @@ -1,42 +0,0 @@ | |||||||
| const statuses = require("./../constants/statuses"); |  | ||||||
|  |  | ||||||
| const mediaInfo = { |  | ||||||
|   title: "", |  | ||||||
|   artist: "", |  | ||||||
|   icon: "", |  | ||||||
|   status: statuses.paused, |  | ||||||
|   url: "" |  | ||||||
| }; |  | ||||||
| const mediaInfoModule = { |  | ||||||
|   mediaInfo, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Update artist and song info in the mediaInfo constant |  | ||||||
|  */ |  | ||||||
| mediaInfoModule.update = function(arg) { |  | ||||||
|   mediaInfo.title = propOrDefault(arg.title); |  | ||||||
|   mediaInfo.artist = propOrDefault(arg.message); |  | ||||||
|   mediaInfo.icon = propOrDefault(arg.icon); |  | ||||||
|   mediaInfo.url = propOrDefault(arg.url); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Update tidal's status in the mediaInfo constant |  | ||||||
|  */ |  | ||||||
| mediaInfoModule.updateStatus = function(status) { |  | ||||||
|   if (Object.values(statuses).includes(status)) { |  | ||||||
|     mediaInfo.status = status; |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Return the property or a default value |  | ||||||
|  * @param {*} prop property to check |  | ||||||
|  * @param {*} defaultValue defaults to "" |  | ||||||
|  */ |  | ||||||
| function propOrDefault(prop, defaultValue = "") { |  | ||||||
|   return prop ? prop : defaultValue; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| module.exports = mediaInfoModule; |  | ||||||
							
								
								
									
										35
									
								
								src/scripts/mediaInfo.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,35 @@ | |||||||
|  | import { MediaInfo } from "../models/mediaInfo"; | ||||||
|  | import { statuses } from "./../constants/statuses"; | ||||||
|  |  | ||||||
|  | export const mediaInfo = { | ||||||
|  |   title: "", | ||||||
|  |   artists: "", | ||||||
|  |   album: "", | ||||||
|  |   icon: "", | ||||||
|  |   status: statuses.paused, | ||||||
|  |   url: "", | ||||||
|  |   current: "", | ||||||
|  |   duration: "", | ||||||
|  |   image: "tidal-hifi-icon", | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const updateMediaInfo = (arg: MediaInfo) => { | ||||||
|  |   mediaInfo.title = propOrDefault(arg.title); | ||||||
|  |   mediaInfo.artists = propOrDefault(arg.artists); | ||||||
|  |   mediaInfo.album = propOrDefault(arg.album); | ||||||
|  |   mediaInfo.icon = propOrDefault(arg.icon); | ||||||
|  |   mediaInfo.url = propOrDefault(arg.url); | ||||||
|  |   mediaInfo.status = propOrDefault(arg.status); | ||||||
|  |   mediaInfo.current = propOrDefault(arg.current); | ||||||
|  |   mediaInfo.duration = propOrDefault(arg.duration); | ||||||
|  |   mediaInfo.image = propOrDefault(arg.image); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Return the property or a default value | ||||||
|  |  * @param {*} prop property to check | ||||||
|  |  * @param {*} defaultValue defaults to "" | ||||||
|  |  */ | ||||||
|  | function propOrDefault(prop: string, defaultValue = "") { | ||||||
|  |   return prop ? prop : defaultValue; | ||||||
|  | } | ||||||
| @@ -1,107 +0,0 @@ | |||||||
| const { Menu } = require("electron"); |  | ||||||
| const { showSettingsWindow } = require("./settings"); |  | ||||||
| const isMac = process.platform === "darwin"; |  | ||||||
|  |  | ||||||
| const settingsMenuEntry = { |  | ||||||
|   label: "Settings", |  | ||||||
|   click() { |  | ||||||
|     showSettingsWindow(); |  | ||||||
|   }, |  | ||||||
|   accelerator: "Control+/", |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const mainMenu = [ |  | ||||||
|   ...(isMac |  | ||||||
|     ? [ |  | ||||||
|         { |  | ||||||
|           label: app.name, |  | ||||||
|           submenu: [ |  | ||||||
|             { role: "about" }, |  | ||||||
|             settingsMenuEntry, |  | ||||||
|             { type: "separator" }, |  | ||||||
|             { role: "services" }, |  | ||||||
|             { type: "separator" }, |  | ||||||
|             { role: "hide" }, |  | ||||||
|             { role: "hideothers" }, |  | ||||||
|             { role: "unhide" }, |  | ||||||
|             { type: "separator" }, |  | ||||||
|             { role: "quit" }, |  | ||||||
|           ], |  | ||||||
|         }, |  | ||||||
|       ] |  | ||||||
|     : []), |  | ||||||
|   // { role: 'fileMenu' } |  | ||||||
|   { |  | ||||||
|     label: "File", |  | ||||||
|     submenu: [settingsMenuEntry, isMac ? { role: "close" } : { role: "quit" }], |  | ||||||
|   }, |  | ||||||
|   // { role: 'editMenu' } |  | ||||||
|   { |  | ||||||
|     label: "Edit", |  | ||||||
|     submenu: [ |  | ||||||
|       { role: "undo" }, |  | ||||||
|       { role: "redo" }, |  | ||||||
|       { type: "separator" }, |  | ||||||
|       { role: "cut" }, |  | ||||||
|       { role: "copy" }, |  | ||||||
|       { role: "paste" }, |  | ||||||
|       ...(isMac |  | ||||||
|         ? [ |  | ||||||
|             { role: "pasteAndMatchStyle" }, |  | ||||||
|             { role: "delete" }, |  | ||||||
|             { role: "selectAll" }, |  | ||||||
|             { type: "separator" }, |  | ||||||
|             { |  | ||||||
|               label: "Speech", |  | ||||||
|               submenu: [{ role: "startspeaking" }, { role: "stopspeaking" }], |  | ||||||
|             }, |  | ||||||
|           ] |  | ||||||
|         : [{ role: "delete" }, { type: "separator" }, { role: "selectAll" }]), |  | ||||||
|       { type: "separator" }, |  | ||||||
|       settingsMenuEntry, |  | ||||||
|     ], |  | ||||||
|   }, |  | ||||||
|   // { role: 'viewMenu' } |  | ||||||
|   { |  | ||||||
|     label: "View", |  | ||||||
|     submenu: [ |  | ||||||
|       { role: "reload" }, |  | ||||||
|       { role: "forcereload" }, |  | ||||||
|       { type: "separator" }, |  | ||||||
|       { role: "resetzoom" }, |  | ||||||
|       { role: "zoomin" }, |  | ||||||
|       { role: "zoomout" }, |  | ||||||
|       { type: "separator" }, |  | ||||||
|       { role: "togglefullscreen" }, |  | ||||||
|     ], |  | ||||||
|   }, |  | ||||||
|   // { role: 'windowMenu' } |  | ||||||
|   { |  | ||||||
|     label: "Window", |  | ||||||
|     submenu: [ |  | ||||||
|       { role: "minimize" }, |  | ||||||
|       ...(isMac |  | ||||||
|         ? [{ type: "separator" }, { role: "front" }, { type: "separator" }, { role: "window" }] |  | ||||||
|         : [{ role: "close" }]), |  | ||||||
|     ], |  | ||||||
|   }, |  | ||||||
|   settingsMenuEntry, |  | ||||||
|   { |  | ||||||
|     label: "About", |  | ||||||
|     click() { |  | ||||||
|       showSettingsWindow("about"); |  | ||||||
|     }, |  | ||||||
|   }, |  | ||||||
| ]; |  | ||||||
|  |  | ||||||
| const menuModule = { mainMenu }; |  | ||||||
|  |  | ||||||
| menuModule.getMenu = function () { |  | ||||||
|   return Menu.buildFromTemplate(mainMenu); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| menuModule.addMenu = function () { |  | ||||||
|   Menu.setApplicationMenu(menuModule.getMenu()); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| module.exports = menuModule; |  | ||||||
							
								
								
									
										119
									
								
								src/scripts/menu.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,119 @@ | |||||||
|  | import { BrowserWindow, Menu, app } from "electron"; | ||||||
|  | import { showSettingsWindow } from "./settings"; | ||||||
|  | const isMac = process.platform === "darwin"; | ||||||
|  | import name from "./../constants/values"; | ||||||
|  |  | ||||||
|  | const settingsMenuEntry = { | ||||||
|  |   label: "Settings", | ||||||
|  |   click() { | ||||||
|  |     showSettingsWindow(); | ||||||
|  |   }, | ||||||
|  |   accelerator: "Control+=", | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const quitMenuEntry = { | ||||||
|  |   label: "Quit", | ||||||
|  |   click() { | ||||||
|  |     app.exit(0); | ||||||
|  |   }, | ||||||
|  |   accelerator: "Control+Q", | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const getMenu = function (mainWindow: BrowserWindow) { | ||||||
|  |   const toggleWindow = { | ||||||
|  |     label: "Toggle Window", | ||||||
|  |     click: function () { | ||||||
|  |       mainWindow.isVisible() ? mainWindow.hide() : mainWindow.show(); | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   const mainMenu = [ | ||||||
|  |     ...(isMac | ||||||
|  |       ? [ | ||||||
|  |           { | ||||||
|  |             label: name, | ||||||
|  |             submenu: [ | ||||||
|  |               { role: "about" }, | ||||||
|  |               settingsMenuEntry, | ||||||
|  |               { type: "separator" }, | ||||||
|  |               { role: "services" }, | ||||||
|  |               { type: "separator" }, | ||||||
|  |               { role: "hide" }, | ||||||
|  |               { role: "hideothers" }, | ||||||
|  |               { role: "unhide" }, | ||||||
|  |               { type: "separator" }, | ||||||
|  |               quitMenuEntry, | ||||||
|  |             ], | ||||||
|  |           }, | ||||||
|  |         ] | ||||||
|  |       : []), | ||||||
|  |     { | ||||||
|  |       label: "File", | ||||||
|  |       submenu: [settingsMenuEntry, isMac ? { role: "close" } : quitMenuEntry], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: "Edit", | ||||||
|  |       submenu: [ | ||||||
|  |         { role: "undo" }, | ||||||
|  |         { role: "redo" }, | ||||||
|  |         { type: "separator" }, | ||||||
|  |         { role: "cut" }, | ||||||
|  |         { role: "copy" }, | ||||||
|  |         { role: "paste" }, | ||||||
|  |         ...(isMac | ||||||
|  |           ? [ | ||||||
|  |               { role: "pasteAndMatchStyle" }, | ||||||
|  |               { role: "delete" }, | ||||||
|  |               { role: "selectAll" }, | ||||||
|  |               { type: "separator" }, | ||||||
|  |               { | ||||||
|  |                 label: "Speech", | ||||||
|  |                 submenu: [{ role: "startspeaking" }, { role: "stopspeaking" }], | ||||||
|  |               }, | ||||||
|  |             ] | ||||||
|  |           : [{ role: "delete" }, { type: "separator" }, { role: "selectAll" }]), | ||||||
|  |         { type: "separator" }, | ||||||
|  |         settingsMenuEntry, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: "View", | ||||||
|  |       submenu: [ | ||||||
|  |         { role: "reload" }, | ||||||
|  |         { role: "forcereload" }, | ||||||
|  |         { type: "separator" }, | ||||||
|  |         { role: "resetzoom" }, | ||||||
|  |         { role: "zoomin" }, | ||||||
|  |         { role: "zoomout" }, | ||||||
|  |         { type: "separator" }, | ||||||
|  |         { role: "togglefullscreen" }, | ||||||
|  |         { role: "toggledevtools" }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       label: "Window", | ||||||
|  |       submenu: [ | ||||||
|  |         { role: "minimize" }, | ||||||
|  |         toggleWindow, | ||||||
|  |         ...(isMac | ||||||
|  |           ? [{ type: "separator" }, { role: "front" }, { type: "separator" }, { role: "window" }] | ||||||
|  |           : [{ role: "close" }]), | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     settingsMenuEntry, | ||||||
|  |     { | ||||||
|  |       label: "About", | ||||||
|  |       click() { | ||||||
|  |         showSettingsWindow("about"); | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     toggleWindow, | ||||||
|  |     quitMenuEntry, | ||||||
|  |   ]; | ||||||
|  |  | ||||||
|  |   return Menu.buildFromTemplate(mainMenu as any); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const addMenu = function (mainWindow: BrowserWindow) { | ||||||
|  |   Menu.setApplicationMenu(getMenu(mainWindow)); | ||||||
|  | }; | ||||||
| @@ -1,73 +0,0 @@ | |||||||
| const Store = require("electron-store"); |  | ||||||
| const settings = require("./../constants/settings"); |  | ||||||
| const path = require("path"); |  | ||||||
| const { BrowserWindow } = require("electron"); |  | ||||||
|  |  | ||||||
| let settingsWindow; |  | ||||||
|  |  | ||||||
| const store = new Store({ |  | ||||||
|   defaults: { |  | ||||||
|     notifications: true, |  | ||||||
|     api: true, |  | ||||||
|     playBackControl: true, |  | ||||||
|     menuBar: true, |  | ||||||
|     apiSettings: { |  | ||||||
|       port: 47836, |  | ||||||
|     }, |  | ||||||
|     trayIcon: true, |  | ||||||
|     mpris: false, |  | ||||||
|     enableCustomHotkeys: false, |  | ||||||
|     enableDiscord: false, |  | ||||||
|     windowBounds: { width: 800, height: 600 }, |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| const settingsModule = { |  | ||||||
|   store, |  | ||||||
|   settings, |  | ||||||
|   settingsWindow, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| settingsModule.createSettingsWindow = function () { |  | ||||||
|   settingsWindow = new BrowserWindow({ |  | ||||||
|     width: 500, |  | ||||||
|     height: 600, |  | ||||||
|     show: false, |  | ||||||
|     frame: false, |  | ||||||
|     title: "Tidal-hifi - settings", |  | ||||||
|     webPreferences: { |  | ||||||
|       affinity: "window", |  | ||||||
|       preload: path.join(__dirname, "../pages/settings/preload.js"), |  | ||||||
|       plugins: true, |  | ||||||
|       nodeIntegration: true, |  | ||||||
|     }, |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   settingsWindow.on("close", (event) => { |  | ||||||
|     if (settingsWindow != null) { |  | ||||||
|       event.preventDefault(); |  | ||||||
|       settingsWindow.hide(); |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   settingsWindow.loadURL(`file://${__dirname}/../pages/settings/settings.html`); |  | ||||||
|  |  | ||||||
|   settingsModule.settingsWindow = settingsWindow; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| settingsModule.showSettingsWindow = function (tab = "general") { |  | ||||||
|   settingsWindow.webContents.send("goToTab", tab); |  | ||||||
|  |  | ||||||
|   // refresh data just before showing the window |  | ||||||
|   settingsWindow.webContents.send("refreshData"); |  | ||||||
|   settingsWindow.show(); |  | ||||||
| }; |  | ||||||
| settingsModule.hideSettingsWindow = function () { |  | ||||||
|   settingsWindow.hide(); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| settingsModule.closeSettingsWindow = function () { |  | ||||||
|   settingsWindow = null; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| module.exports = settingsModule; |  | ||||||
							
								
								
									
										94
									
								
								src/scripts/settings.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,94 @@ | |||||||
|  | import Store from "electron-store"; | ||||||
|  |  | ||||||
|  | import { settings } from "../constants/settings"; | ||||||
|  | import path from "path"; | ||||||
|  | import { BrowserWindow } from "electron"; | ||||||
|  |  | ||||||
|  | let settingsWindow: BrowserWindow; | ||||||
|  |  | ||||||
|  | export const settingsStore = new Store({ | ||||||
|  |   defaults: { | ||||||
|  |     adBlock: false, | ||||||
|  |     api: true, | ||||||
|  |     apiSettings: { | ||||||
|  |       port: 47836, | ||||||
|  |     }, | ||||||
|  |     customCSS: "", | ||||||
|  |     disableBackgroundThrottle: true, | ||||||
|  |     disableHardwareMediaKeys: false, | ||||||
|  |     enableCustomHotkeys: false, | ||||||
|  |     enableDiscord: false, | ||||||
|  |     flags: { | ||||||
|  |       gpuRasterization: true, | ||||||
|  |       disableHardwareMediaKeys: false, | ||||||
|  |     }, | ||||||
|  |     menuBar: true, | ||||||
|  |     minimizeOnClose: false, | ||||||
|  |     mpris: false, | ||||||
|  |     notifications: true, | ||||||
|  |     playBackControl: true, | ||||||
|  |     singleInstance: true, | ||||||
|  |     skipArtists: false, | ||||||
|  |     skippedArtists: [""], | ||||||
|  |     trayIcon: true, | ||||||
|  |     updateFrequency: 500, | ||||||
|  |     windowBounds: { width: 800, height: 600 }, | ||||||
|  |   }, | ||||||
|  |   migrations: { | ||||||
|  |     "3.1.0": (migrationStore) => { | ||||||
|  |       console.log("running migrations for 3.1.0"); | ||||||
|  |       migrationStore.set( | ||||||
|  |         settings.flags.disableHardwareMediaKeys, | ||||||
|  |         migrationStore.get("disableHardwareMediaKeys") ?? false | ||||||
|  |       ); | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | const settingsModule = { | ||||||
|  |   // settings, | ||||||
|  |   settingsWindow, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const createSettingsWindow = function () { | ||||||
|  |   settingsWindow = new BrowserWindow({ | ||||||
|  |     width: 700, | ||||||
|  |     height: 600, | ||||||
|  |     resizable: false, | ||||||
|  |     show: false, | ||||||
|  |     transparent: true, | ||||||
|  |     frame: false, | ||||||
|  |     title: "TIDAL Hi-Fi settings", | ||||||
|  |     webPreferences: { | ||||||
|  |       preload: path.join(__dirname, "../pages/settings/preload.js"), | ||||||
|  |       plugins: true, | ||||||
|  |       nodeIntegration: true, | ||||||
|  |     }, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   settingsWindow.on("close", (event: any) => { | ||||||
|  |     if (settingsWindow != null) { | ||||||
|  |       event.preventDefault(); | ||||||
|  |       settingsWindow.hide(); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   settingsWindow.loadURL(`file://${__dirname}/../pages/settings/settings.html`); | ||||||
|  |  | ||||||
|  |   settingsModule.settingsWindow = settingsWindow; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const showSettingsWindow = function (tab = "general") { | ||||||
|  |   settingsWindow.webContents.send("goToTab", tab); | ||||||
|  |  | ||||||
|  |   // refresh data just before showing the window | ||||||
|  |   settingsWindow.webContents.send("refreshData"); | ||||||
|  |   settingsWindow.show(); | ||||||
|  | }; | ||||||
|  | export const hideSettingsWindow = function () { | ||||||
|  |   settingsWindow.hide(); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const closeSettingsWindow = function () { | ||||||
|  |   settingsWindow = null; | ||||||
|  | }; | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| const { Tray } = require("electron"); |  | ||||||
| const { getMenu } = require("./menu"); |  | ||||||
| const trayModule = {}; |  | ||||||
| let tray; |  | ||||||
|  |  | ||||||
| trayModule.addTray = function (options = { icon: "" }) { |  | ||||||
|   tray = new Tray(options.icon); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| trayModule.refreshTray = function () { |  | ||||||
|   if (!tray) { |  | ||||||
|     trayModule.addTray(); |  | ||||||
|   } |  | ||||||
|   tray.on("click", function (e) { |  | ||||||
|     // do nothing on click |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   tray.setToolTip("Tidal-hifi"); |  | ||||||
|   tray.setContextMenu(getMenu()); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| module.exports = trayModule; |  | ||||||
							
								
								
									
										32
									
								
								src/scripts/tray.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,32 @@ | |||||||
|  | import { BrowserWindow, Tray } from "electron"; | ||||||
|  | import { getMenu } from "./menu"; | ||||||
|  |  | ||||||
|  | let tray: Tray; | ||||||
|  |  | ||||||
|  | export const addTray = function (mainWindow: BrowserWindow, options = { icon: "" }) { | ||||||
|  |   tray = new Tray(options.icon); | ||||||
|  |   tray.setIgnoreDoubleClickEvents(true); | ||||||
|  |   tray.setToolTip("Tidal-hifi"); | ||||||
|  |  | ||||||
|  |   const menu = getMenu(mainWindow); | ||||||
|  |  | ||||||
|  |   tray.setContextMenu(menu); | ||||||
|  |  | ||||||
|  |   tray.on("click", function () { | ||||||
|  |     if (mainWindow.isVisible()) { | ||||||
|  |       if (!mainWindow.isFocused()) { | ||||||
|  |         mainWindow.focus(); | ||||||
|  |       } else { | ||||||
|  |         mainWindow.hide(); | ||||||
|  |       } | ||||||
|  |     } else { | ||||||
|  |       mainWindow.show(); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const refreshTray = function (mainWindow: BrowserWindow) { | ||||||
|  |   if (!tray) { | ||||||
|  |     addTray(mainWindow); | ||||||
|  |   } | ||||||
|  | }; | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| const windowFunctions = {}; |  | ||||||
|  |  | ||||||
| windowFunctions.setTitle = function(title) { |  | ||||||
|   window.document.title = title; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| windowFunctions.getTitle = function() { |  | ||||||
|   return window.document.title; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| module.exports = windowFunctions; |  | ||||||
							
								
								
									
										7
									
								
								src/scripts/window-functions.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | |||||||
|  | export const setTitle = function (title: string) { | ||||||
|  |   window.document.title = title; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const getTitle = function () { | ||||||
|  |   return window.document.title; | ||||||
|  | }; | ||||||
							
								
								
									
										16
									
								
								tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,16 @@ | |||||||
|  | { | ||||||
|  |   "compilerOptions": { | ||||||
|  |     "module": "commonjs", | ||||||
|  |     "target": "ES6", | ||||||
|  |     "noImplicitAny": true, | ||||||
|  |     "sourceMap": true, | ||||||
|  |     "allowJs": true, | ||||||
|  |     "outDir": "ts-dist", | ||||||
|  |     "esModuleInterop": true, | ||||||
|  |     "baseUrl": ".", | ||||||
|  |     "paths": { | ||||||
|  |       "*": ["node_modules/*"] | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |   "include": ["src/**/*"] | ||||||
|  | } | ||||||