Compare commits

..

19 Commits

Author SHA1 Message Date
4941aae950 Bugfix/4.1.1 (#161)
* - Fixed `cannot read property of undefined` error because of not passing mainWindow around.
- vincens2005, fixed inconsistent auto muting

* Fix inconsistent auto-muting (#159)

* fix muting sometimes not working

* fix inconsistent unmuting

* fix bad code in inconsistent muting fig

Co-authored-by: Cukmekerb <cukmekerb@gmail.com>
2022-08-23 21:20:46 +02:00
1439a11969 Feature/4.1.0 (#156)
* added protocol handler

* Switched icon strategies to fix bugs with icons

* fixed tray icon issues

* fixed about :)

* Fixed playback, mpris and API issues
2022-08-07 16:05:48 +02:00
Tomasz Hołubowicz
3a3e0e1a2d Add instruction for installation via Nix (#153) 2022-07-05 16:41:12 +02:00
fa9ab22867 - 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
2022-06-25 21:45:30 +02:00
207a61d199 4.0.0 with electron 19.0.5 release 2022-06-23 17:06:17 +02:00
Tomasz Hołubowicz
7b18322e17 Bump electron version (#152) 2022-06-23 17:00:35 +02:00
8f47756244 fixed hardware media flag upadting gpu rasterization options 2022-06-18 10:40:31 +02:00
Marie
cdcf9431bf Fix bugs related to media info (#150)
* Fix duration time element

* change interval time from 200 to 500
2022-06-18 10:26:21 +02:00
374f3da740 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
2022-05-07 18:13:36 +02:00
3965ada0a2 Add link to flathub 2022-04-23 23:18:08 +02:00
79ff02d06c Electron 15, single-instance-lock and setting to disable hardwareMediaKeyHandling (#134)
* Update to Electron 15 and add Flatpak to README (#131)

* Update functional

* Change to 15

* Replace window.hide, window.restart with counterparts

* Fix openexternal by doing default JS

* Change mouse to pointer for <a> elements

* turn mouse to pointer for exit button

* Fix deprecation of Audio for AudioVideo

* Made a small mistake

* Add Flatpak to readme

* 3.0.0 prep

* Added setting to disable multiple tidal-hifi windows (defaults to true)

fixes #121

* Added setting to disable HardwareMediaKeyHandling (defaults to false)

fixes #133

Co-authored-by: Marie <marie@kaifa.ch>
2022-04-23 22:59:32 +02:00
7b2afd2290 added separate build-unpacked config 2022-04-22 00:21:05 +02:00
5fde20ace1 hotfix, downgrade packaged version to 8.5.2 2022-04-21 23:58:48 +02:00
7f5f5e7f62 Merge branch 'master' of github.com:Mastermindzh/tidal-hifi 2022-04-21 17:52:50 +02:00
6a1a1efe74 add apt-get update 2022-04-21 17:52:31 +02:00
94e1bb1780 add apt-get update 2022-04-21 17:51:21 +02:00
d66dd8cc9e 2.8.1 release 2022-04-21 17:49:11 +02:00
Cukmekerb
de97ac8a00 make quit button actually quit app (#123) 2022-04-21 17:46:49 +02:00
Marie
82ac5edf22 Update build version to fit with package.json electron version (#128)
* Update build version to fit with package.json electron version

* Update electron-builder.yml

* Use wvcus.2 for additional fixes

* Also use wvcus.2 for package.json
2022-04-21 17:46:05 +02:00
25 changed files with 2699 additions and 2729 deletions

View File

@@ -9,6 +9,8 @@ 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 - name: Install libarchive-tools
run: sudo apt-get install -y libarchive-tools run: sudo apt-get install -y libarchive-tools
- uses: actions/checkout@master - uses: actions/checkout@master

View File

@@ -9,6 +9,8 @@ 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 - name: Install libarchive-tools
run: sudo apt-get install -y libarchive-tools run: sudo apt-get install -y libarchive-tools
- uses: actions/checkout@master - uses: actions/checkout@master

View File

@@ -4,6 +4,61 @@ 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).
## 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 ## 2.8.0
- Added the ability to mute artists automatically - Added the ability to mute artists automatically

View File

@@ -12,22 +12,23 @@ The web version of [listen.tidal.com](https://listen.tidal.com) running in elect
<!-- toc --> <!-- toc -->
- [Installation](#installation) - [Installation](#installation)
- [Using releases](#using-releases) * [Using releases](#using-releases)
- [Snap install](#snap-install) * [Snap](#snap)
- [Arch Linux](#arch-linux) * [Arch Linux](#arch-linux)
- [Using source](#using-source) * [Flatpak](#flatpak)
- [features](#features) * [Nix](#nix)
* [Using source](#using-source)
- [Features](#features)
- [Integrations](#integrations) - [Integrations](#integrations)
- [not included](#not-included) * [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)
- [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 -->
@@ -37,7 +38,7 @@ The web version of [listen.tidal.com](https://listen.tidal.com) running in elect
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:
@@ -61,6 +62,22 @@ Arch Linux users can use the AUR to install tidal-hifi:
trizen tidal-hifi-bin 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:
@@ -70,7 +87,7 @@ To install and work with the code on this project follow these steps:
- npm install - npm install
- npm start - npm start
## features ## Features
- HiFi playback - HiFi playback
- Notifications - Notifications
@@ -93,7 +110,7 @@ It currently includes:
- MPRIS - MPRIS media player controls/status - MPRIS - MPRIS media player controls/status
- Discord - Shows what you're listening to on Discord. - Discord - Shows what you're listening to on Discord.
### not included 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)
@@ -134,11 +151,11 @@ Inspired by [haydenjames' issue](https://github.com/Mastermindzh/tidal-hifi/issu
## Images ## Images
### settings window ### Settings window
![settings window](./docs/settings-preview.png) ![settings window](./docs/settings-preview.png)
### 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).

View File

@@ -0,0 +1,39 @@
appId: com.rickvanlieshout.tidal-hifi
electronVersion: 19.0.5
electronDownload:
version: 19.0.5+wvcus
mirror: https://github.com/castlabs/electron-releases/releases/download/v
snap:
plugs:
- default
- screen-inhibit-control
linux:
category: Audio
target:
- dir
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: icon.png
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"]

View File

@@ -1,6 +1,6 @@
extends: ./build/electron-builder.yml extends: ./build/electron-builder.base.yml
linux: linux:
category: Audio category: Audio
icon: ./assets/icon.png icon: icon.png
target: target:
- deb - deb

View File

@@ -1,6 +1,6 @@
extends: ./build/electron-builder.yml extends: ./build/electron-builder.base.yml
linux: linux:
category: Audio category: Audio
icon: ./assets/icon.png icon: icon.png
target: target:
- pacman - pacman

View File

@@ -1,6 +1,6 @@
extends: ./build/electron-builder.yml extends: ./build/electron-builder.base.yml
linux: linux:
category: Audio category: Audio
icon: ./assets/TIDAL.icns icon: icon.png
target: target:
- rpm - rpm

View File

@@ -1,6 +1,6 @@
extends: ./build/electron-builder.yml extends: ./build/electron-builder.base.yml
linux: linux:
category: Audio category: Audio
icon: ./assets/icon.png icon: icon.png
target: target:
- snap - snap

View File

@@ -0,0 +1,4 @@
extends: ./build/electron-builder.base.yml
linux:
target:
- dir

View File

@@ -1,12 +1,7 @@
appId: com.rickvanlieshout.tidal-hifi electronVersion: 19.0.5
electronVersion: 8.5.2
electronDownload: electronDownload:
version: 8.5.2-wvvmp version: 19.0.5+wvcus
mirror: https://github.com/castlabs/electron-releases/releases/download/v mirror: https://github.com/castlabs/electron-releases/releases/download/v
snap:
plugs:
- default
- screen-inhibit-control
linux: linux:
category: Audio category: Audio
target: target:
@@ -17,24 +12,9 @@ linux:
- 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: build/icon.png icon: icon.png
artifactName: "tidalhifi" artifactName: "tidalhifi"
appId: com.rickvanlieshout.tidalhifi appId: com.rickvanlieshout.tidalhifi
executableName: tidalhifi executableName: tidalhifi

BIN
build/icon-inverted.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 102 KiB

BIN
build/icon.icns Executable file

Binary file not shown.

3730
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -1,17 +1,19 @@
{ {
"name": "tidal-hifi", "name": "tidal-hifi",
"version": "2.8.0", "version": "4.1.1",
"description": "Tidal on Electron with widevine(hifi) support", "description": "Tidal on Electron with widevine(hifi) support",
"main": "src/main.js", "main": "src/main.js",
"scripts": { "scripts": {
"start": "electron .", "start": "electron .",
"build": "electron-builder --publish=never -c ./build/electron-builder.yml", "build": "electron-builder --publish=never -c ./build/electron-builder.yml",
"build-deb": "electron-builder --publish=never -c ./build/electron-builder.deb.yml", "build-deb": "electron-builder --publish=never -c ./build/electron-builder.deb.yml",
"build-unpacked": "electron-builder --publish=never -c ./build/electron-builder.unpacked.yml",
"build-rpm": "electron-builder --publish=never -c ./build/electron-builder.rpm.yml", "build-rpm": "electron-builder --publish=never -c ./build/electron-builder.rpm.yml",
"build-snap": "electron-builder --publish=never -c ./build/electron-builder.snap.yml", "build-snap": "electron-builder --publish=never -c ./build/electron-builder.snap.yml",
"build-arch": "electron-builder --publish=never -c ./build/electron-builder.pacman.yml", "build-arch": "electron-builder --publish=never -c ./build/electron-builder.pacman.yml",
"build-wl": "electron-builder --publish=never -c ./build/electron-builder.yml -wl", "build-wl": "electron-builder --publish=never -c ./build/electron-builder.yml -wl",
"build-mac": "electron-builder --publish=never -c ./build/electron-builder.yml -m" "build-mac": "electron-builder --publish=never -c ./build/electron-builder.yml -m",
"build-base": "electron-builder --publish=never -c ./build/electron-builder.base.yml"
}, },
"keywords": [ "keywords": [
"electron", "electron",
@@ -23,19 +25,19 @@
"homepage": "https://github.com/Mastermindzh/tidal-hifi", "homepage": "https://github.com/Mastermindzh/tidal-hifi",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@electron/remote": "^2.0.8",
"discord-rpc": "^4.0.1", "discord-rpc": "^4.0.1",
"electron-store": "^8.0.1", "electron-store": "^8.1.0",
"express": "^4.17.1", "express": "^4.18.1",
"hotkeys-js": "^3.8.7", "hotkeys-js": "^3.9.4",
"mpris-service": "^2.1.2", "mpris-service": "^2.1.2",
"request": "^2.88.2" "request": "^2.88.2"
}, },
"devDependencies": { "devDependencies": {
"@mastermindzh/prettier-config": "^1.0.0", "@mastermindzh/prettier-config": "^1.0.0",
"electron": "git+https://github.com/castlabs/electron-releases.git#v16.0.4+wvcus", "electron": "git+https://github.com/castlabs/electron-releases.git#v19.0.5+wvcus",
"electron-builder": "^22.14.5", "electron-builder": "^23.3.3",
"electron-reload": "^1.5.0", "prettier": "^2.7.1"
"prettier": "^2.5.0"
}, },
"prettier": "@mastermindzh/prettier-config" "prettier": "@mastermindzh/prettier-config"
} }

6
src/constants/flags.js Normal file
View File

@@ -0,0 +1,6 @@
const flags = {
gpuRasterization: [{ flag: "enable-gpu-rasterization", value: undefined }],
disableHardwareMediaKeys: [{ flag: "disable-features", value: "HardwareMediaKeyHandling" }],
};
module.exports = flags;

View File

@@ -19,6 +19,12 @@ const settings = {
root: "apiSettings", root: "apiSettings",
port: "apiSettings.port", port: "apiSettings.port",
}, },
singleInstance: "singleInstance",
disableHardwareMediaKeys: "disableHardwareMediaKeys",
flags: {
disableHardwareMediaKeys: "flags.disableHardwareMediaKeys",
gpuRasterization: "flags.gpuRasterization",
},
mpris: "mpris", mpris: "mpris",
enableCustomHotkeys: "enableCustomHotkeys", enableCustomHotkeys: "enableCustomHotkeys",
trayIcon: "trayIcon", trayIcon: "trayIcon",

3
src/constants/values.js Normal file
View File

@@ -0,0 +1,3 @@
module.exports = {
name: "tidal-hifi",
};

View File

@@ -1,4 +1,5 @@
const { app, BrowserWindow, globalShortcut, ipcMain } = require("electron"); require("@electron/remote/main").initialize();
const { app, BrowserWindow, components, globalShortcut, ipcMain, protocol } = require("electron");
const { const {
settings, settings,
store, store,
@@ -16,23 +17,32 @@ const mediaKeys = require("./constants/mediaKeys");
const mediaInfoModule = require("./scripts/mediaInfo"); const mediaInfoModule = require("./scripts/mediaInfo");
const discordModule = require("./scripts/discord"); const discordModule = require("./scripts/discord");
const globalEvents = require("./constants/globalEvents"); const globalEvents = require("./constants/globalEvents");
const flagValues = require("./constants/flags");
let mainWindow; let mainWindow;
let icon = path.join(__dirname, "../assets/icon.png"); let icon = path.join(__dirname, "../assets/icon.png");
const PROTOCOL_PREFIX = "tidal";
/** setFlags();
* Enable live reload in development builds
*/ function setFlags() {
if (!app.isPackaged) { const flags = store.get().flags;
require("electron-reload")(`${__dirname}`, { if (flags) {
electron: require(`${__dirname}/../node_modules/electron`), for (const [key, value] of Object.entries(flags)) {
if (value) {
flagValues[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. * Fix Display Compositor issue.
*/ */
app.commandLine.appendSwitch('disable-seccomp-filter-sandbox'); app.commandLine.appendSwitch("disable-seccomp-filter-sandbox");
}
/** /**
* Update the menuBarVisbility according to the store value * Update the menuBarVisbility according to the store value
@@ -42,6 +52,23 @@ function syncMenuBarWithStore() {
mainWindow.setMenuBarVisibility(store.get(settings.menuBar)); mainWindow.setMenuBarVisibility(store.get(settings.menuBar));
} }
/**
* 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 (store.get(settings.singleInstance)) {
const gotTheLock = app.requestSingleInstanceLock();
if (!gotTheLock) {
return false;
}
}
return true;
}
function createWindow(options = {}) { function createWindow(options = {}) {
// Create the browser window. // Create the browser window.
mainWindow = new BrowserWindow({ mainWindow = new BrowserWindow({
@@ -50,17 +77,15 @@ function createWindow(options = {}) {
width: store && store.get(settings.windowBounds.width), width: store && store.get(settings.windowBounds.width),
height: store && store.get(settings.windowBounds.height), height: store && store.get(settings.windowBounds.height),
icon, icon,
tray: true,
backgroundColor: options.backgroundColor, backgroundColor: options.backgroundColor,
webPreferences: { webPreferences: {
affinity: "window",
preload: path.join(__dirname, "preload.js"), preload: path.join(__dirname, "preload.js"),
plugins: true, plugins: true,
devTools: true, // I like tinkering, others might too devTools: true, // I like tinkering, others might too
enableRemoteModule: true,
}, },
}); });
require("@electron/remote/main").enable(mainWindow.webContents);
registerHttpProtocols();
syncMenuBarWithStore(); syncMenuBarWithStore();
// load the Tidal website // load the Tidal website
@@ -89,6 +114,13 @@ function createWindow(options = {}) {
}); });
} }
function registerHttpProtocols() {
protocol.registerHttpProtocol(PROTOCOL_PREFIX, (request, _callback) => {
mainWindow.loadURL(`${tidalUrl}/${request.url.substring(PROTOCOL_PREFIX.length + 3)}`);
});
app.setAsDefaultProtocolClient(PROTOCOL_PREFIX);
}
function addGlobalShortcuts() { function addGlobalShortcuts() {
Object.keys(mediaKeys).forEach((key) => { Object.keys(mediaKeys).forEach((key) => {
globalShortcut.register(`${key}`, () => { globalShortcut.register(`${key}`, () => {
@@ -100,15 +132,20 @@ function addGlobalShortcuts() {
// This method will be called when Electron has finished // This method will be called when Electron has finished
// initialization and is ready to create browser windows. // initialization and is ready to create browser windows.
// Some APIs can only be used after this event occurs. // Some APIs can only be used after this event occurs.
app.on("ready", () => { app.on("ready", async () => {
if (isMainInstanceOrMultipleInstancesAllowed()) {
await components.whenReady();
createWindow(); createWindow();
addMenu(); addMenu(mainWindow);
createSettingsWindow(); createSettingsWindow();
addGlobalShortcuts(); addGlobalShortcuts();
store.get(settings.trayIcon) && addTray({ icon }) && refreshTray(); store.get(settings.trayIcon) && addTray(mainWindow, { icon }) && refreshTray();
store.get(settings.api) && expressModule.run(mainWindow); store.get(settings.api) && expressModule.run(mainWindow);
store.get(settings.enableDiscord) && discordModule.initRPC(); store.get(settings.enableDiscord) && discordModule.initRPC();
// mainWindow.webContents.openDevTools(); // mainWindow.webContents.openDevTools();
} else {
app.quit();
}
}); });
app.on("activate", function () { app.on("activate", function () {
@@ -119,23 +156,27 @@ app.on("activate", function () {
} }
}); });
app.on("browser-window-created", (_, window) => {
require("@electron/remote/main").enable(window.webContents);
});
// IPC // IPC
ipcMain.on(globalEvents.updateInfo, (event, arg) => { ipcMain.on(globalEvents.updateInfo, (_event, arg) => {
mediaInfoModule.update(arg); mediaInfoModule.update(arg);
}); });
ipcMain.on(globalEvents.hideSettings, (event, arg) => { ipcMain.on(globalEvents.hideSettings, (_event, _arg) => {
hideSettingsWindow(); hideSettingsWindow();
}); });
ipcMain.on(globalEvents.showSettings, (event, arg) => { ipcMain.on(globalEvents.showSettings, (_event, _arg) => {
showSettingsWindow(); showSettingsWindow();
}); });
ipcMain.on(globalEvents.refreshMenuBar, (event, arg) => { ipcMain.on(globalEvents.refreshMenuBar, (_event, _arg) => {
syncMenuBarWithStore(); syncMenuBarWithStore();
}); });
ipcMain.on(globalEvents.storeChanged, (event, arg) => { ipcMain.on(globalEvents.storeChanged, (_event, _arg) => {
syncMenuBarWithStore(); syncMenuBarWithStore();
if (store.get(settings.enableDiscord) && !discordModule.rpc) { if (store.get(settings.enableDiscord) && !discordModule.rpc) {
@@ -145,6 +186,6 @@ ipcMain.on(globalEvents.storeChanged, (event, arg) => {
} }
}); });
ipcMain.on(globalEvents.error, (event, arg) => { ipcMain.on(globalEvents.error, (event, _arg) => {
console.log(event); console.log(event);
}); });

View File

@@ -9,12 +9,16 @@ let trayIcon,
api, api,
port, port,
menuBar, menuBar,
mutedArtists; mutedArtists,
singleInstance,
disableHardwareMediaKeys,
gpuRasterization;
const { store, settings } = require("./../../scripts/settings"); const { store, settings } = require("./../../scripts/settings");
const { ipcRenderer } = require("electron"); const { ipcRenderer } = require("electron");
const globalEvents = require("./../../constants/globalEvents"); const globalEvents = require("./../../constants/globalEvents");
const remote = require("@electron/remote");
const { app } = remote;
/** /**
* Sync the UI forms with the current settings * Sync the UI forms with the current settings
*/ */
@@ -31,31 +35,33 @@ function refreshSettings() {
minimizeOnClose.checked = store.get(settings.minimizeOnClose); minimizeOnClose.checked = store.get(settings.minimizeOnClose);
muteArtists.checked = store.get(settings.muteArtists); muteArtists.checked = store.get(settings.muteArtists);
mutedArtists.value = store.get(settings.mutedArtists).join("\n"); mutedArtists.value = store.get(settings.mutedArtists).join("\n");
singleInstance.checked = store.get(settings.singleInstance);
disableHardwareMediaKeys.checked = store.get(settings.flags.disableHardwareMediaKeys);
gpuRasterization.checked = store.get(settings.flags.gpuRasterization);
} }
/** /**
* Open an url in the default browsers * Open an url in the default browsers
*/ */
window.openExternal = function (url) { function openExternal(url) {
const { shell } = require("electron"); const { shell } = require("electron");
shell.openExternal(url); shell.openExternal(url);
}; }
/** /**
* hide the settings window * hide the settings window
*/ */
window.hide = function () { function hide() {
ipcRenderer.send(globalEvents.hideSettings); ipcRenderer.send(globalEvents.hideSettings);
}; }
/** /**
* Restart tidal-hifi after changes * Restart tidal-hifi after changes
*/ */
window.restart = function () { function restart() {
const remote = require("electron").remote; app.relaunch();
remote.app.relaunch(); app.quit();
remote.app.exit(0); }
};
/** /**
* Bind UI components to functions after DOMContentLoaded * Bind UI components to functions after DOMContentLoaded
@@ -65,8 +71,16 @@ window.addEventListener("DOMContentLoaded", () => {
return document.getElementById(id); return document.getElementById(id);
} }
document.getElementById("close").addEventListener("click", hide);
document.getElementById("restart").addEventListener("click", restart);
document.querySelectorAll("#openExternal").forEach((elem) =>
elem.addEventListener("click", function (event) {
openExternal(event.target.getAttribute("data-url"));
})
);
function addInputListener(source, key) { function addInputListener(source, key) {
source.addEventListener("input", function (event, data) { source.addEventListener("input", function (_event, _data) {
if (this.value === "on") { if (this.value === "on") {
store.set(key, source.checked); store.set(key, source.checked);
} else { } else {
@@ -77,7 +91,7 @@ window.addEventListener("DOMContentLoaded", () => {
} }
function addTextAreaListener(source, key) { function addTextAreaListener(source, key) {
source.addEventListener("input", function (event, data) { source.addEventListener("input", function (_event, _data) {
store.set(key, source.value.split("\n")); store.set(key, source.value.split("\n"));
ipcRenderer.send(globalEvents.storeChanged); ipcRenderer.send(globalEvents.storeChanged);
}); });
@@ -103,6 +117,9 @@ window.addEventListener("DOMContentLoaded", () => {
enableDiscord = get("enableDiscord"); enableDiscord = get("enableDiscord");
muteArtists = get("muteArtists"); muteArtists = get("muteArtists");
mutedArtists = get("mutedArtists"); mutedArtists = get("mutedArtists");
singleInstance = get("singleInstance");
disableHardwareMediaKeys = get("disableHardwareMediaKeys");
gpuRasterization = get("gpuRasterization");
refreshSettings(); refreshSettings();
@@ -118,4 +135,7 @@ window.addEventListener("DOMContentLoaded", () => {
addInputListener(minimizeOnClose, settings.minimizeOnClose); addInputListener(minimizeOnClose, settings.minimizeOnClose);
addInputListener(muteArtists, settings.muteArtists); addInputListener(muteArtists, settings.muteArtists);
addTextAreaListener(mutedArtists, settings.mutedArtists); addTextAreaListener(mutedArtists, settings.mutedArtists);
addInputListener(singleInstance, settings.singleInstance);
addInputListener(disableHardwareMediaKeys, settings.flags.disableHardwareMediaKeys);
addInputListener(gpuRasterization, settings.flags.gpuRasterization);
}); });

View File

@@ -11,7 +11,7 @@
<body> <body>
<div class="header"> <div class="header">
<h1 class="title">Settings</h1> <h1 class="title">Settings</h1>
<a href="javascript:hide();" class="exitWindow"> <a id="close" style="cursor: pointer;" class="exitWindow">
<svg version="1.1" xmlns="http://www.w3.org/2000/svg" width="18px" height="18px" viewBox="0 0 348.333 348.334"> <svg version="1.1" xmlns="http://www.w3.org/2000/svg" width="18px" height="18px" viewBox="0 0 348.333 348.334">
<g> <g>
@@ -26,10 +26,10 @@
</div> </div>
<div class="body"> <div class="body">
<div class="tabset"> <div class="tabset">
<!-- Tab 1 -->
<input type="radio" name="tabset" id="general" checked /> <input type="radio" name="tabset" id="general" checked />
<label for="general">General</label> <label for="general">General</label>
<!-- Tab 2 -->
<input type="radio" name="tabset" id="api" /> <input type="radio" name="tabset" id="api" />
<label for="api">Api</label> <label for="api">Api</label>
@@ -37,6 +37,9 @@
<input type="radio" name="tabset" id="integrations" /> <input type="radio" name="tabset" id="integrations" />
<label for="integrations">Integrations</label> <label for="integrations">Integrations</label>
<!-- advanced tab -->
<input type="radio" name="tabset" id="advanced" />
<label for="advanced">Advanced</label>
<!-- about tab --> <!-- about tab -->
<input type="radio" name="tabset" id="about" /> <input type="radio" name="tabset" id="about" />
<label for="about">About</label> <label for="about">About</label>
@@ -62,7 +65,7 @@
</p> </p>
<label class="switch" style="margin-bottom:10px"> <label class="switch" style="margin-bottom:10px">
<input id="muteArtists" type="checkbox"> <input id="muteArtists" type="checkbox">
<span class="slider round" ></span> <span class="slider round"></span>
</label> </label>
<textarea id="mutedArtists" cols="40" rows="5"></textarea> <textarea id="mutedArtists" cols="40" rows="5"></textarea>
</div> </div>
@@ -105,20 +108,33 @@
<div class="option"> <div class="option">
<h4>Hotkeys</h4> <h4>Hotkeys</h4>
<p> <p>
Enables extra hotkeys to achieve feature parity with the <a href = "javascript:openExternal('https://defkey.com/tidal-desktop-shortcuts')">desktop apps</a><br /> Enables extra hotkeys to achieve feature parity with the <a id="openExternal"
style="text-decoration: underline; cursor: pointer;"
data-url="https://defkey.com/tidal-desktop-shortcuts">desktop apps</a><br />
</p> </p>
<label class="switch"> <label class="switch">
<input id="enableCustomHotkeys" type="checkbox"> <input id="enableCustomHotkeys" type="checkbox">
<span class="slider round"></span> <span class="slider round"></span>
</label> </label>
</div> </div>
<div class="option">
<h4>Single instance</h4>
<p>
Prevent opening multiple tidal-hifi instances
</p>
<label class="switch">
<input id="singleInstance" type="checkbox">
<span class="slider round"></span>
</label>
</div>
</div> </div>
</section> </section>
<section id="api" class="tab-panel"> <section id="api" class="tab-panel">
<div class="section"> <div class="section">
<h3>Api</h3> <h3>Api</h3>
<p style="margin-bottom: 15px;"> <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. 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> </p>
<div class="option"> <div class="option">
@@ -175,19 +191,51 @@
</div> </div>
</div> </div>
</section> </section>
<section id="advanced" class="tab-panel">
<div class="section">
<h3>Flags</h3>
<div class="option">
<h4>Disable hardware media keys</h4>
<p>
Disable built-in media keys. <br />
Also prevents certain desktop environments from recognizing the chrome MPRIS client separetely from the
custom MPRIS client.
</p>
<label class="switch">
<input id="disableHardwareMediaKeys" type="checkbox">
<span class="slider round"></span>
</label>
</div>
<div class="option">
<h4>Enable GPU rasterization</h4>
<p>
Move a part of the rendering to the GPU for increased performance.
</p>
<label class="switch">
<input id="gpuRasterization" type="checkbox">
<span class="slider round"></span>
</label>
</div>
</div>
</section>
<section id="about" class="tab-panel"> <section id="about" class="tab-panel">
<div class="section"> <div class="section">
<img alt="tidal icon" style="width: 100px; height: auto; display: block; margin: 0 auto; margin-bottom: 20px; margin-top: 20px;" src = "./icon.png"> <img alt="tidal icon"
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;"> <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 /> <a id="openExternal" style="text-decoration: underline; cursor: pointer;"
It uses <a href="javascript:openExternal('https://castlabs.com/');">Castlabs'</a> versions of Electron for widevine support. data-url="https://github.com/Mastermindzh/tidal-hifi">Tidal-hifi</a> is made by <a id="openExternal"
data-url="https://www.rickvanlieshout.com" style="text-decoration: underline; cursor: pointer;">Rick van
Lieshout</a>.<br />
It uses <a style="text-decoration: underline; cursor: pointer;" id="openExternal"
data-url="https://castlabs.com/">Castlabs'</a> versions of Electron for widevine support.
</p> </p>
</div> </div>
</section> </section>
<small>Some settings require a restart of Tidal-hifi. To do so, click the button below:</small> <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> <button id="restart" style="width: 100%">Restart Tidal-hifi</button>
</div> </div>
</div> </div>
</div> </div>
@@ -198,6 +246,7 @@
-webkit-user-select: none; -webkit-user-select: none;
-webkit-app-region: drag; -webkit-app-region: drag;
} }
.header a { .header a {
-webkit-app-region: no-drag; -webkit-app-region: no-drag;
} }
@@ -333,9 +382,10 @@
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; background-color: #5a5a5a;
} }
/* Tabs */ /* Tabs */
.tabset > input[type="radio"] { .tabset>input[type="radio"] {
position: absolute; position: absolute;
left: -200vw; left: -200vw;
} }
@@ -344,15 +394,15 @@
display: none; display: none;
} }
.tabset > input:first-child:checked ~ .tab-panels > .tab-panel:first-child, .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(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(5):checked~.tab-panels>.tab-panel:nth-child(3),
.tabset > input:nth-child(7):checked ~ .tab-panels > .tab-panel:nth-child(4) .tabset>input:nth-child(7):checked~.tab-panels>.tab-panel:nth-child(4),
{ .tabset>input:nth-child(9):checked~.tab-panels>.tab-panel:nth-child(5) {
display: block; display: block;
} }
.tabset > label { .tabset>label {
position: relative; position: relative;
display: inline-block; display: inline-block;
padding: 15px 0px 10px; padding: 15px 0px 10px;
@@ -360,12 +410,12 @@
cursor: pointer; cursor: pointer;
} }
.tabset > input + label { .tabset>input+label {
color: #e0e0e0; color: #e0e0e0;
margin-right: 30px; margin-right: 30px;
} }
.tabset > input:checked + label { .tabset>input:checked+label {
color: #0ff; color: #0ff;
border-bottom: 2px solid #0ff; border-bottom: 2px solid #0ff;
} }
@@ -373,6 +423,7 @@
.tab-panel { .tab-panel {
padding: 10px 0; padding: 10px 0;
} }
/* switches */ /* switches */
/* The switch - the box around the slider */ /* The switch - the box around the slider */
@@ -382,6 +433,7 @@
width: 50px; width: 50px;
height: 28px; height: 28px;
} }
/* Hide default HTML checkbox */ /* Hide default HTML checkbox */
.switch input { .switch input {
@@ -389,6 +441,7 @@
width: 0; width: 0;
height: 0; height: 0;
} }
/* The slider */ /* The slider */
.slider { .slider {
@@ -415,19 +468,20 @@
transition: .4s; transition: .4s;
} }
input:checked + .slider { input:checked+.slider {
background-color: #0ff; background-color: #0ff;
} }
input:focus + .slider { input:focus+.slider {
box-shadow: 0 0 1px #0ff; box-shadow: 0 0 1px #0ff;
} }
input:checked + .slider:before { input:checked+.slider:before {
-webkit-transform: translateX(22px); -webkit-transform: translateX(22px);
-ms-transform: translateX(22px); -ms-transform: translateX(22px);
transform: translateX(22px); transform: translateX(22px);
} }
/* Rounded sliders */ /* Rounded sliders */
.slider.round { .slider.round {
@@ -437,6 +491,7 @@
.slider.round:before { .slider.round:before {
border-radius: 50%; border-radius: 50%;
} }
/* input field */ /* input field */
input { input {
@@ -455,11 +510,11 @@
} }
/* buttons */ /* buttons */
button{ button {
border:none; border: none;
background:none; background: none;
align-items: center; align-items: center;
background-color: rgba(229,238,255,.2); background-color: rgba(229, 238, 255, .2);
display: inline-flex; display: inline-flex;
justify-content: center; justify-content: center;
border-radius: 12px; border-radius: 12px;
@@ -474,13 +529,15 @@
min-height: 0; min-height: 0;
min-width: 0; min-width: 0;
font-size: 1.14286rem; font-size: 1.14286rem;
font-family: nationale,nationale-regular,Helvetica,sans-serif; font-family: nationale, nationale-regular, Helvetica, sans-serif;
margin-top: 10px; margin-top: 10px;
cursor: pointer; cursor: pointer;
} }
button:hover{
background-color: rgba(229,238,255,.3); button:hover {
background-color: rgba(229, 238, 255, .3);
} }
textarea { textarea {
color: #72777f; color: #72777f;
background: #242528; background: #242528;
@@ -488,12 +545,14 @@
width: 100%; width: 100%;
color: rgba(229, 238, 255, .6); color: rgba(229, 238, 255, .6);
padding: 0 0 12px; padding: 0 0 12px;
display:block; display: block;
} }
textarea:focus { textarea:focus {
outline: none; outline: none;
border: 0; border: 0;
border-bottom: 1px solid #0ff; border-bottom: 1px solid #0ff;
} }
</style> </style>
</html> </html>

View File

@@ -1,21 +1,18 @@
const { setTitle } = require("./scripts/window-functions"); const { setTitle } = require("./scripts/window-functions");
const { dialog, process, Notification } = require("electron").remote; const { dialog, process, Notification } = require("@electron/remote");
const { store, settings } = require("./scripts/settings"); const { store, settings } = require("./scripts/settings");
const { ipcRenderer } = require("electron"); const { ipcRenderer } = require("electron");
const { app } = require("electron").remote; const { app } = require("@electron/remote");
const { downloadFile } = require("./scripts/download"); const { downloadFile } = require("./scripts/download");
const statuses = require("./constants/statuses"); const statuses = require("./constants/statuses");
const hotkeys = require("./scripts/hotkeys"); const hotkeys = require("./scripts/hotkeys");
const globalEvents = require("./constants/globalEvents"); const globalEvents = require("./constants/globalEvents");
const notificationPath = `${app.getPath("userData")}/notification.jpg`; const notificationPath = `${app.getPath("userData")}/notification.jpg`;
const appName = "Tidal Hifi";
let currentSong = ""; let currentSong = "";
let player; let player;
let currentPlayStatus = statuses.paused; let currentPlayStatus = statuses.paused;
let progressBarTime; let isMutedArtist = true;
let currentTimeChanged = false;
let currentTime;
let currentURL = undefined;
let isMutedArtist = false;
const elements = { const elements = {
play: '*[data-test="play"]', play: '*[data-test="play"]',
@@ -36,7 +33,7 @@ const elements = {
media: '*[data-test="current-media-imagery"]', media: '*[data-test="current-media-imagery"]',
image: "img", image: "img",
current: '*[data-test="current-time"]', current: '*[data-test="current-time"]',
duration: '*[data-test="duration-time"]', duration: '*[data-test="duration"]',
bar: '*[data-test="progress-bar"]', bar: '*[data-test="progress-bar"]',
footer: "#footerPlayer", footer: "#footerPlayer",
album_header_title: '.header-details [data-test="title"]', album_header_title: '.header-details [data-test="title"]',
@@ -225,7 +222,7 @@ function handleLogout() {
} }
function addFullScreenListeners() { function addFullScreenListeners() {
window.document.addEventListener("fullscreenchange", (event) => { window.document.addEventListener("fullscreenchange", () => {
ipcRenderer.send(globalEvents.refreshMenuBar); ipcRenderer.send(globalEvents.refreshMenuBar);
}); });
} }
@@ -236,7 +233,7 @@ function addFullScreenListeners() {
*/ */
function addIPCEventListeners() { function addIPCEventListeners() {
window.addEventListener("DOMContentLoaded", () => { window.addEventListener("DOMContentLoaded", () => {
ipcRenderer.on("globalEvent", (event, args) => { ipcRenderer.on("globalEvent", (_event, args) => {
switch (args) { switch (args) {
case globalEvents.playPause: case globalEvents.playPause:
playPause(); playPause();
@@ -312,19 +309,16 @@ function updateMediaInfo(options, notify) {
/** /**
* Checks if Tidal is playing a video or song by grabbing the "a" element from the title. * Checks if Tidal is playing a video or song by grabbing the "a" element from the title.
* If it's a song it sets the track URL as currentURL, If it's a video it will set currentURL to undefined. * If it's a song it returns the track URL, if not it will return undefined
*/ */
function updateURL() { function getTrackURL() {
const URLelement = elements.get("title").querySelector("a"); const URLelement = elements.get("title").querySelector("a");
switch (URLelement) { if (URLelement !== null) {
case null:
currentURL = undefined;
break;
default:
const id = URLelement.href.replace(/[^0-9]/g, ""); const id = URLelement.href.replace(/[^0-9]/g, "");
currentURL = `https://tidal.com/browse/track/${id}`; return `https://tidal.com/browse/track/${id}`;
break;
} }
return window.location;
} }
/** /**
@@ -333,11 +327,11 @@ function updateURL() {
setInterval(function () { setInterval(function () {
const title = elements.getText("title"); const title = elements.getText("title");
const artists = elements.getArtists(); const artists = elements.getArtists();
muteArtistIfFoundInMutedArtistsList(); // doing this here so that nothing can possibly fail before we call this function
const album = elements.getAlbumName(); const album = elements.getAlbumName();
const current = elements.getText("current"); const current = elements.getText("current");
const duration = elements.getText("duration"); const duration = elements.getText("duration");
const appName = "Tidal Hifi";
const progressBarcurrentTime = elements.get("bar").getAttribute("aria-valuenow");
const songDashArtistTitle = `${title} - ${artists}`; const songDashArtistTitle = `${title} - ${artists}`;
const currentStatus = getCurrentlyPlayingStatus(); const currentStatus = getCurrentlyPlayingStatus();
const options = { const options = {
@@ -345,46 +339,23 @@ setInterval(function () {
message: artists, message: artists,
album: album, album: album,
status: currentStatus, status: currentStatus,
url: currentURL, url: getTrackURL(),
current: current, current,
duration: duration, duration,
"app-name": appName, "app-name": appName,
}; };
const playStatusChanged = currentStatus !== currentPlayStatus;
const progressBarTimeChanged = progressBarcurrentTime !== progressBarTime;
const titleOrArtistChanged = currentSong !== songDashArtistTitle; const titleOrArtistChanged = currentSong !== songDashArtistTitle;
muteArtistIfFoundInMutedArtistsList();
if (titleOrArtistChanged || playStatusChanged || progressBarTimeChanged || currentTimeChanged) {
// update title, url and play info with new info // update title, url and play info with new info
setTitle(songDashArtistTitle); setTitle(songDashArtistTitle);
updateURL(); getTrackURL();
currentSong = songDashArtistTitle; currentSong = songDashArtistTitle;
currentPlayStatus = currentStatus; currentPlayStatus = currentStatus;
// check progress bar value and make sure current stays up to date after switch
if (progressBarTime != progressBarcurrentTime && !titleOrArtistChanged) {
progressBarTime = progressBarcurrentTime;
currentTime = options.current;
options.duration = duration;
currentTimeChanged = true;
}
if (currentTimeChanged) {
if (options.current == currentTime && currentStatus != "paused") return;
currentTime = options.current;
currentTimeChanged = false;
}
// make sure current is set to 0 if title changes
if (titleOrArtistChanged) {
options.current = "0:00";
currentTime = options.current;
progressBarTime = progressBarcurrentTime;
}
const image = elements.getSongIcon(); const image = elements.getSongIcon();
new Promise((resolve) => { new Promise((resolve) => {
@@ -410,7 +381,6 @@ setInterval(function () {
}, },
() => {} () => {}
); );
}
/** /**
* Checks whether the current artist is included in the "muted artists" list and if so it will automatically mute the player * Checks whether the current artist is included in the "muted artists" list and if so it will automatically mute the player
@@ -423,13 +393,13 @@ setInterval(function () {
isMutedArtist = true; isMutedArtist = true;
elements.click("volume"); elements.click("volume");
} }
} else if (currentStatus === statuses.playing && isMutedArtist && elements.isMuted()) { } else if (isMutedArtist && elements.isMuted()) {
elements.click("volume"); elements.click("volume");
isMutedArtist = false; isMutedArtist = false;
} }
} }
} }
}, 200); }, 1000);
if (process.platform === "linux" && store.get(settings.mpris)) { if (process.platform === "linux" && store.get(settings.mpris)) {
try { try {

View File

@@ -1,6 +1,7 @@
const { Menu } = require("electron"); const { Menu, app } = require("electron");
const { showSettingsWindow } = require("./settings"); const { showSettingsWindow } = require("./settings");
const isMac = process.platform === "darwin"; const isMac = process.platform === "darwin";
const { name } = require("./../constants/values");
const settingsMenuEntry = { const settingsMenuEntry = {
label: "Settings", label: "Settings",
@@ -10,11 +11,29 @@ const settingsMenuEntry = {
accelerator: "Control+/", accelerator: "Control+/",
}; };
const mainMenu = [ const quitMenuEntry = {
label: "Quit",
click() {
app.exit(0);
},
accelerator: "Control+Q",
};
const menuModule = {};
menuModule.getMenu = function (mainWindow) {
const toggleWindow = {
label: "Toggle Window",
click: function () {
mainWindow.isVisible() ? mainWindow.hide() : mainWindow.show();
},
};
const mainMenu = [
...(isMac ...(isMac
? [ ? [
{ {
label: app.name, label: name,
submenu: [ submenu: [
{ role: "about" }, { role: "about" },
settingsMenuEntry, settingsMenuEntry,
@@ -25,17 +44,15 @@ const mainMenu = [
{ role: "hideothers" }, { role: "hideothers" },
{ role: "unhide" }, { role: "unhide" },
{ type: "separator" }, { type: "separator" },
{ role: "quit" }, quitMenuEntry,
], ],
}, },
] ]
: []), : []),
// { role: 'fileMenu' }
{ {
label: "File", label: "File",
submenu: [settingsMenuEntry, isMac ? { role: "close" } : { role: "quit" }], submenu: [settingsMenuEntry, isMac ? { role: "close" } : quitMenuEntry],
}, },
// { role: 'editMenu' }
{ {
label: "Edit", label: "Edit",
submenu: [ submenu: [
@@ -61,7 +78,6 @@ const mainMenu = [
settingsMenuEntry, settingsMenuEntry,
], ],
}, },
// { role: 'viewMenu' }
{ {
label: "View", label: "View",
submenu: [ submenu: [
@@ -73,13 +89,14 @@ const mainMenu = [
{ role: "zoomout" }, { role: "zoomout" },
{ type: "separator" }, { type: "separator" },
{ role: "togglefullscreen" }, { role: "togglefullscreen" },
{ role: "toggledevtools" },
], ],
}, },
// { role: 'windowMenu' }
{ {
label: "Window", label: "Window",
submenu: [ submenu: [
{ role: "minimize" }, { role: "minimize" },
toggleWindow,
...(isMac ...(isMac
? [{ type: "separator" }, { role: "front" }, { type: "separator" }, { role: "window" }] ? [{ type: "separator" }, { role: "front" }, { type: "separator" }, { role: "window" }]
: [{ role: "close" }]), : [{ role: "close" }]),
@@ -92,16 +109,14 @@ const mainMenu = [
showSettingsWindow("about"); showSettingsWindow("about");
}, },
}, },
]; toggleWindow,
];
const menuModule = { mainMenu };
menuModule.getMenu = function () {
return Menu.buildFromTemplate(mainMenu); return Menu.buildFromTemplate(mainMenu);
}; };
menuModule.addMenu = function () { menuModule.addMenu = function (mainWindow) {
Menu.setApplicationMenu(menuModule.getMenu()); Menu.setApplicationMenu(menuModule.getMenu(mainWindow));
}; };
module.exports = menuModule; module.exports = menuModule;

View File

@@ -16,12 +16,27 @@ const store = new Store({
apiSettings: { apiSettings: {
port: 47836, port: 47836,
}, },
singleInstance: true,
disableHardwareMediaKeys: false,
trayIcon: true, trayIcon: true,
minimizeOnClose: false, minimizeOnClose: false,
mpris: false, mpris: false,
enableCustomHotkeys: false, enableCustomHotkeys: false,
enableDiscord: false, enableDiscord: false,
windowBounds: { width: 800, height: 600 }, windowBounds: { width: 800, height: 600 },
flags: {
gpuRasterization: true,
disableHardwareMediaKeys: false,
},
},
migrations: {
"3.1.0": (migrationStore) => {
console.log("running migrations for 3.1.0");
migrationStore.set(
settings.flags.disableHardwareMediaKeys,
migrationStore.get("disableHardwareMediaKeys") ?? false
);
},
}, },
}); });
@@ -39,7 +54,6 @@ settingsModule.createSettingsWindow = function () {
frame: false, frame: false,
title: "Tidal-hifi - settings", title: "Tidal-hifi - settings",
webPreferences: { webPreferences: {
affinity: "window",
preload: path.join(__dirname, "../pages/settings/preload.js"), preload: path.join(__dirname, "../pages/settings/preload.js"),
plugins: true, plugins: true,
nodeIntegration: true, nodeIntegration: true,

View File

@@ -1,48 +1,21 @@
const { Tray, app } = require("electron"); const { Tray } = require("electron");
const { Menu } = require("electron"); const { getMenu } = require("./menu");
const { getMenu, mainMenu } = require("./menu");
const { store, settings } = require("./settings");
const trayModule = {}; const trayModule = {};
let tray; let tray;
trayModule.addTray = function (options = { icon: "" }) { trayModule.addTray = function (mainWindow, options = { icon: "" }) {
tray = new Tray(options.icon); tray = new Tray(options.icon);
tray.setIgnoreDoubleClickEvents(true);
tray.setToolTip("Tidal-hifi");
const menu = getMenu(mainWindow);
tray.setContextMenu(menu);
}; };
trayModule.refreshTray = function (mainWindow) { trayModule.refreshTray = function (mainWindow) {
if (!tray) { if (!tray) {
trayModule.addTray(); trayModule.addTray(mainWindow);
}
tray.on("click", function (e) {
if (mainWindow) {
mainWindow.show();
}
});
tray.setToolTip("Tidal-hifi");
if (mainWindow && store.get(settings.minimizeOnClose)) {
tray.setContextMenu(
Menu.buildFromTemplate([
{
label: "Toggle Window",
click: function () {
mainWindow.isVisible() ? mainWindow.hide() : mainWindow.show();
},
},
{
label: "Quit",
click: function () {
mainWindow.destroy();
app.quit();
},
},
...mainMenu, //we add menu items from the other context
])
);
} else {
tray.setContextMenu(getMenu());
} }
}; };