mirror of
https://github.com/Mastermindzh/tidal-hifi.git
synced 2025-09-13 07:09:06 +02:00
Compare commits
19 Commits
Author | SHA1 | Date | |
---|---|---|---|
d34ddfeb75 | |||
b7f163c1a1 | |||
|
791a92a446 | ||
|
10c1e57680 | ||
|
c65d1a56c8 | ||
|
d8f2dbd0c2 | ||
|
43ce85bb28 | ||
|
8201e23e4b | ||
|
3cc288e014 | ||
|
08ec7fadac | ||
|
7a30b125ec | ||
|
cac5db123f | ||
|
59f8b2d0b5 | ||
|
81a536bbdb | ||
|
5e952e3899 | ||
|
df887b8628 | ||
|
31d90a342c | ||
|
8607337580 | ||
4fe42a3671 |
@@ -4,6 +4,14 @@ 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).
|
||||||
|
|
||||||
|
## 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
|
## 2.1.1
|
||||||
|
|
||||||
- The discord integration now doesn't send an update every 15 seconds it sends an update whenever the media info changes
|
- The discord integration now doesn't send an update every 15 seconds it sends an update whenever the media info changes
|
||||||
|
20
README.md
20
README.md
@@ -18,6 +18,7 @@ The web version of [listen.tidal.com](listen.tidal.com) running in electron with
|
|||||||
- [Using source](#using-source)
|
- [Using source](#using-source)
|
||||||
- [features](#features)
|
- [features](#features)
|
||||||
- [Integrations](#integrations)
|
- [Integrations](#integrations)
|
||||||
|
- [not included](#not-included)
|
||||||
- [Known bugs](#known-bugs)
|
- [Known bugs](#known-bugs)
|
||||||
- [Why](#why)
|
- [Why](#why)
|
||||||
- [Why not extend existing projects?](#why-not-extend-existing-projects)
|
- [Why not extend existing projects?](#why-not-extend-existing-projects)
|
||||||
@@ -39,13 +40,13 @@ Various packaged versions of the software are available on the [releases](https:
|
|||||||
|
|
||||||
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
|
||||||
@@ -63,7 +64,7 @@ trizen tidal-hifi
|
|||||||
|
|
||||||
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
|
||||||
@@ -74,10 +75,23 @@ To install and work with the code on this project follow these steps:
|
|||||||
- 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
|
||||||
|
- Custom [integrations](#integrations)
|
||||||
- [Settings feature](./docs/settings.png) to disable certain functionality. (`ctrl+=`)
|
- [Settings feature](./docs/settings.png) to disable certain functionality. (`ctrl+=`)
|
||||||
|
|
||||||
## 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
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
pkgbase = tidal-hifi-git
|
pkgbase = tidal-hifi-git
|
||||||
pkgdesc = The web version of listen.tidal.com running in electron with hifi support thanks to widevine.
|
pkgdesc = The web version of listen.tidal.com running in electron with hifi support thanks to widevine.
|
||||||
pkgver = 2.1.0
|
pkgver = 2.1.1
|
||||||
pkgrel = 1
|
pkgrel = 1
|
||||||
url = https://github.com/Mastermindzh/tidal-hifi
|
url = https://github.com/Mastermindzh/tidal-hifi
|
||||||
arch = x86_64
|
arch = x86_64
|
||||||
@@ -11,9 +11,9 @@ pkgbase = tidal-hifi-git
|
|||||||
depends = nss
|
depends = nss
|
||||||
depends = gtk3
|
depends = gtk3
|
||||||
provides = tidal-hifi
|
provides = tidal-hifi
|
||||||
source = https://github.com/Mastermindzh/tidal-hifi/archive/2.1.0.zip
|
source = https://github.com/Mastermindzh/tidal-hifi/archive/2.1.1.zip
|
||||||
source = tidal-hifi.desktop
|
source = tidal-hifi.desktop
|
||||||
sha512sums = 30b50243e7033219351f7d7f318757e4590e5c78a7031762cf846ae3652b424ab3f54388cba44209e49434d8534edb198e1c65d0b53450b4367636ef0f4653e0
|
sha512sums = 510c94d9db8573103ee900dcd75b0416ea06ea62af7ed21818d2a5a77ab009a2737c852e7e20b97f846e14b07754a9e5a36ac5b9d0cd0a2a8d0aff3bbc75e622
|
||||||
sha512sums = 35f38ac308b871c1822d7f6f760f2fb54c3748cf769822cb0f0dfb90f0f5754ba9316da5e903a0d2e9839de3a43ec76f238f3f2e44021956fa1da19142081349
|
sha512sums = 35f38ac308b871c1822d7f6f760f2fb54c3748cf769822cb0f0dfb90f0f5754ba9316da5e903a0d2e9839de3a43ec76f238f3f2e44021956fa1da19142081349
|
||||||
|
|
||||||
pkgname = tidal-hifi-git
|
pkgname = tidal-hifi-git
|
||||||
|
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
_pkgname=tidal-hifi
|
_pkgname=tidal-hifi
|
||||||
pkgname="$_pkgname-git"
|
pkgname="$_pkgname-git"
|
||||||
pkgver=2.1.0
|
pkgver=2.1.1
|
||||||
pkgrel=1
|
pkgrel=1
|
||||||
pkgdesc="The web version of listen.tidal.com running in electron with hifi support thanks to widevine."
|
pkgdesc="The web version of listen.tidal.com running in electron with hifi support thanks to widevine."
|
||||||
arch=("x86_64")
|
arch=("x86_64")
|
||||||
@@ -15,7 +15,7 @@ provides=("$_pkgname")
|
|||||||
|
|
||||||
source=("https://github.com/Mastermindzh/tidal-hifi/archive/$pkgver.zip"
|
source=("https://github.com/Mastermindzh/tidal-hifi/archive/$pkgver.zip"
|
||||||
"${_pkgname}.desktop")
|
"${_pkgname}.desktop")
|
||||||
sha512sums=('30b50243e7033219351f7d7f318757e4590e5c78a7031762cf846ae3652b424ab3f54388cba44209e49434d8534edb198e1c65d0b53450b4367636ef0f4653e0'
|
sha512sums=('510c94d9db8573103ee900dcd75b0416ea06ea62af7ed21818d2a5a77ab009a2737c852e7e20b97f846e14b07754a9e5a36ac5b9d0cd0a2a8d0aff3bbc75e622'
|
||||||
'35f38ac308b871c1822d7f6f760f2fb54c3748cf769822cb0f0dfb90f0f5754ba9316da5e903a0d2e9839de3a43ec76f238f3f2e44021956fa1da19142081349')
|
'35f38ac308b871c1822d7f6f760f2fb54c3748cf769822cb0f0dfb90f0f5754ba9316da5e903a0d2e9839de3a43ec76f238f3f2e44021956fa1da19142081349')
|
||||||
|
|
||||||
cdToPkg(){
|
cdToPkg(){
|
||||||
|
BIN
docs/integrations.png
Normal file
BIN
docs/integrations.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 47 KiB |
4365
package-lock.json
generated
4365
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "tidal-hifi",
|
"name": "tidal-hifi",
|
||||||
"version": "2.1.1",
|
"version": "2.2.0",
|
||||||
"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": {
|
||||||
@@ -33,11 +33,11 @@
|
|||||||
},
|
},
|
||||||
"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",
|
"dot-prop": ">=4.2.1",
|
||||||
|
"electron": "git+https://github.com/castlabs/electron-releases.git#v10.4.3-wvvmp",
|
||||||
"electron-builder": "^21.2.0",
|
"electron-builder": "^21.2.0",
|
||||||
"electron-reload": "^1.5.0",
|
"electron-reload": "^1.5.0",
|
||||||
"prettier": "^2.0.4",
|
"prettier": "^2.0.4"
|
||||||
"dot-prop": ">=4.2.1"
|
|
||||||
},
|
},
|
||||||
"prettier": "@mastermindzh/prettier-config"
|
"prettier": "@mastermindzh/prettier-config"
|
||||||
}
|
}
|
||||||
|
@@ -44,6 +44,7 @@ function createWindow(options = {}) {
|
|||||||
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,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@@ -12,6 +12,10 @@ const notificationPath = `${app.getPath("userData")}/notification.jpg`;
|
|||||||
let currentSong = "";
|
let currentSong = "";
|
||||||
let player;
|
let player;
|
||||||
let currentPlayStatus = statuses.paused;
|
let currentPlayStatus = statuses.paused;
|
||||||
|
let progressBarTime;
|
||||||
|
let currentTimeChanged = false;
|
||||||
|
let currentTime;
|
||||||
|
let currentURL = undefined;
|
||||||
|
|
||||||
const elements = {
|
const elements = {
|
||||||
play: '*[data-test="play"]',
|
play: '*[data-test="play"]',
|
||||||
@@ -19,7 +23,7 @@ const elements = {
|
|||||||
next: '*[data-test="next"]',
|
next: '*[data-test="next"]',
|
||||||
previous: 'button[data-test="previous"]',
|
previous: 'button[data-test="previous"]',
|
||||||
title: '*[data-test^="footer-track-title"]',
|
title: '*[data-test^="footer-track-title"]',
|
||||||
artists: '*[data-test^="grid-item-detail-text-title-artist"]',
|
artists: '*[class^="elemental__text elemental__text css-oxcos"]',
|
||||||
home: '*[data-test="menu--home"]',
|
home: '*[data-test="menu--home"]',
|
||||||
back: '[class^="backwardButton"]',
|
back: '[class^="backwardButton"]',
|
||||||
forward: '[class^="forwardButton"]',
|
forward: '[class^="forwardButton"]',
|
||||||
@@ -31,7 +35,9 @@ const elements = {
|
|||||||
settings: '*[data-test^="open-settings"]',
|
settings: '*[data-test^="open-settings"]',
|
||||||
media: '*[data-test="current-media-imagery"]',
|
media: '*[data-test="current-media-imagery"]',
|
||||||
image: "img",
|
image: "img",
|
||||||
url: 'a[href*="/track/"]',
|
current: '*[data-test="current-time"]',
|
||||||
|
duration: '*[data-test="duration-time"]',
|
||||||
|
bar: '*[data-test="progress-bar"]',
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get an element from the dom
|
* Get an element from the dom
|
||||||
@@ -238,31 +244,75 @@ function updateMediaInfo(options, notify) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
function updateURL() {
|
||||||
|
const URLelement = elements.get("title").querySelector("a");
|
||||||
|
switch (URLelement) {
|
||||||
|
case null:
|
||||||
|
currentURL = undefined;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
const id = URLelement.href.replace(/[^0-9]/g, "");
|
||||||
|
currentURL = `https://tidal.com/browse/track/${id}`;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Watch for song changes and update title + notify
|
* Watch for song changes and update title + notify
|
||||||
*/
|
*/
|
||||||
setInterval(function () {
|
setInterval(function () {
|
||||||
const title = elements.getText("title");
|
const title = elements.getText("title");
|
||||||
const url = elements.get("url").href.replace(/[^0-9]/g, "");
|
|
||||||
const artists = elements.getText("artists");
|
const artists = elements.getText("artists");
|
||||||
|
const current = elements.getText("current");
|
||||||
|
const duration = elements.getText("duration");
|
||||||
|
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 = {
|
||||||
title,
|
title,
|
||||||
message: artists,
|
message: artists,
|
||||||
status: currentStatus,
|
status: currentStatus,
|
||||||
url: `https://tidal.com/browse/track/${url}`,
|
url: currentURL,
|
||||||
|
current: current,
|
||||||
|
duration: duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
const playStatusChanged = currentStatus !== currentPlayStatus;
|
const playStatusChanged = currentStatus !== currentPlayStatus;
|
||||||
|
const progressBarTimeChanged = progressBarcurrentTime !== progressBarTime;
|
||||||
const titleOrArtistChanged = currentSong !== songDashArtistTitle;
|
const titleOrArtistChanged = currentSong !== songDashArtistTitle;
|
||||||
|
|
||||||
if (titleOrArtistChanged || playStatusChanged) {
|
if (titleOrArtistChanged || playStatusChanged || progressBarTimeChanged || currentTimeChanged) {
|
||||||
// update title and play info with new info
|
// update title, url and play info with new info
|
||||||
setTitle(songDashArtistTitle);
|
setTitle(songDashArtistTitle);
|
||||||
|
updateURL();
|
||||||
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) => {
|
||||||
|
@@ -1,31 +1,55 @@
|
|||||||
const discordrpc = require("discord-rpc");
|
const discordrpc = require("discord-rpc");
|
||||||
const { ipcMain } = require("electron");
|
const { app, ipcMain } = require("electron");
|
||||||
const electron = require("electron");
|
|
||||||
const globalEvents = require("../constants/globalEvents");
|
const globalEvents = require("../constants/globalEvents");
|
||||||
const clientId = "833617820704440341";
|
const clientId = "833617820704440341";
|
||||||
const mediaInfoModule = require("./mediaInfo");
|
const mediaInfoModule = require("./mediaInfo");
|
||||||
const discordModule = [];
|
const discordModule = [];
|
||||||
|
|
||||||
|
function timeToSeconds(timeArray) {
|
||||||
|
let minutes = (timeArray[0] * 1);
|
||||||
|
let seconds = (minutes * 60) + (timeArray[1] * 1);
|
||||||
|
return seconds;
|
||||||
|
}
|
||||||
|
|
||||||
let rpc;
|
let rpc;
|
||||||
const observer = (event, arg) => {
|
const observer = (event, arg) => {
|
||||||
if (mediaInfoModule.mediaInfo.status == "paused" && rpc) {
|
if (mediaInfoModule.mediaInfo.status == "paused" && rpc) {
|
||||||
rpc.setActivity(idleStatus);
|
rpc.setActivity(idleStatus);
|
||||||
} else if (rpc) {
|
} else if (rpc) {
|
||||||
rpc.setActivity({
|
const currentSeconds = timeToSeconds(mediaInfoModule.mediaInfo.current.split(":"));
|
||||||
...idleStatus,
|
const durationSeconds = timeToSeconds(mediaInfoModule.mediaInfo.duration.split(":"));
|
||||||
...{
|
const date = new Date();
|
||||||
details: `Listening to ${mediaInfoModule.mediaInfo.title}`,
|
const now = date.getTime() / 1000 | 0;
|
||||||
state: mediaInfoModule.mediaInfo.artist,
|
const remaining = date.setSeconds(date.getSeconds() + (durationSeconds - currentSeconds));
|
||||||
buttons: [{ label: "Play on Tidal", url: mediaInfoModule.mediaInfo.url }],
|
if (mediaInfoModule.mediaInfo.url) {
|
||||||
},
|
rpc.setActivity({
|
||||||
});
|
...idleStatus,
|
||||||
|
...{
|
||||||
|
details: `Listening to ${mediaInfoModule.mediaInfo.title}`,
|
||||||
|
state: mediaInfoModule.mediaInfo.artist,
|
||||||
|
startTimestamp: parseInt(now),
|
||||||
|
endTimestamp: parseInt(remaining),
|
||||||
|
buttons: [{ label: "Play on Tidal", url: mediaInfoModule.mediaInfo.url }],
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
rpc.setActivity({
|
||||||
|
...idleStatus,
|
||||||
|
...{
|
||||||
|
details: `Watching ${mediaInfoModule.mediaInfo.title}`,
|
||||||
|
state: mediaInfoModule.mediaInfo.artist,
|
||||||
|
startTimestamp: parseInt(now),
|
||||||
|
endTimestamp: parseInt(remaining),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const idleStatus = {
|
const idleStatus = {
|
||||||
details: `Browsing Tidal`,
|
details: `Browsing Tidal`,
|
||||||
largeImageKey: "tidal-hifi-icon",
|
largeImageKey: "tidal-hifi-icon",
|
||||||
largeImageText: `Tidal HiFi ${electron.app.getVersion()}`,
|
largeImageText: `Tidal HiFi ${app.getVersion()}`,
|
||||||
instance: false,
|
instance: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@@ -6,6 +6,8 @@ const mediaInfo = {
|
|||||||
icon: "",
|
icon: "",
|
||||||
status: statuses.paused,
|
status: statuses.paused,
|
||||||
url: "",
|
url: "",
|
||||||
|
current: "",
|
||||||
|
duration: ""
|
||||||
};
|
};
|
||||||
const mediaInfoModule = {
|
const mediaInfoModule = {
|
||||||
mediaInfo,
|
mediaInfo,
|
||||||
@@ -20,6 +22,8 @@ mediaInfoModule.update = function (arg) {
|
|||||||
mediaInfo.icon = propOrDefault(arg.icon);
|
mediaInfo.icon = propOrDefault(arg.icon);
|
||||||
mediaInfo.url = propOrDefault(arg.url);
|
mediaInfo.url = propOrDefault(arg.url);
|
||||||
mediaInfo.status = propOrDefault(arg.status);
|
mediaInfo.status = propOrDefault(arg.status);
|
||||||
|
mediaInfo.current = propOrDefault(arg.current);
|
||||||
|
mediaInfo.duration = propOrDefault(arg.duration);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Reference in New Issue
Block a user