commit
b3d13c3a90
33
CHANGELOG.md
33
CHANGELOG.md
|
@ -6,6 +6,39 @@ Project versioning adheres to [Semantic Versioning](http://semver.org/).
|
||||||
Commit convention is based on [Conventional Commits](http://conventionalcommits.org).
|
Commit convention is based on [Conventional Commits](http://conventionalcommits.org).
|
||||||
Change log format is based on [Keep a Changelog](http://keepachangelog.com/).
|
Change log format is based on [Keep a Changelog](http://keepachangelog.com/).
|
||||||
|
|
||||||
|
## [1.1.0](https://github.com/lyswhut/lx-music-desktop/compare/v1.0.1...v1.1.0) - 2020-09-18
|
||||||
|
|
||||||
|
### 新增
|
||||||
|
|
||||||
|
- 在歌单详情界面新增播放当前歌单按钮、收藏歌单按钮,注:播放歌单不会将歌曲添加到试听列表
|
||||||
|
- 新增`不允许将歌词窗口拖出主屏幕之外`的设置项,默认开启,在连接多个屏幕时想要拖动到其他屏幕时可关闭此设置
|
||||||
|
- 新增大部分平台的歌词翻译,感谢 @InoriHimea 提供的[krc解码算法](https://github.com/lyswhut/lx-music-desktop/issues/296#issuecomment-683285784)
|
||||||
|
- 新增`显示歌词翻译`设置,默认开启,仅支持某些平台,注:无论该设置是否开启,嵌入或下载歌词时都不会带上翻译
|
||||||
|
- 新增`显示切换动画`设置,默认开启,关闭时将基本禁用软件内的所有切换动画
|
||||||
|
- 播放状态栏新增桌面歌词的开关、播放模式的切换、歌曲的收藏按钮,Thanks to @andylow for the [icon](https://github.com/lyswhut/lx-music-desktop/pull/309)!
|
||||||
|
|
||||||
|
### 修复
|
||||||
|
|
||||||
|
- 修复使用全局快捷键还原窗口时,窗口没有获取焦点的问题
|
||||||
|
- 修复我的列表搜索对最后一个字符的匹配问题
|
||||||
|
- 修复窗口在`较小`模式下最小化/关闭按钮不居中的问题
|
||||||
|
|
||||||
|
### 优化
|
||||||
|
|
||||||
|
- 桌面歌词当前播放行改为上下居中
|
||||||
|
- 为区分静音状态,静音时音量条会变淡,调整音量条时将会取消静音
|
||||||
|
- 优化随机播放机制,现在通过`下一曲`切换歌曲时,直到播放完整个列表之前将不会再随机到之前播放过的歌曲,并且通过`上一曲`可以正确播放上一首歌曲
|
||||||
|
- 当下载目录没有写入权限时将显示没有写入权限的提示
|
||||||
|
|
||||||
|
### 移除
|
||||||
|
|
||||||
|
- 移除默认的全局声音媒体快捷键接管
|
||||||
|
- 移除对百度音乐的支持,因百度音乐原有的大部分API失效,而且该平台相对其他平台来说音乐太少了,可有可无,以后再看情况恢复
|
||||||
|
|
||||||
|
### 其他
|
||||||
|
|
||||||
|
- 更新electron到 10.1.2
|
||||||
|
|
||||||
## [1.0.1](https://github.com/lyswhut/lx-music-desktop/compare/v1.0.0...v1.0.1) - 2020-07-25
|
## [1.0.1](https://github.com/lyswhut/lx-music-desktop/compare/v1.0.0...v1.0.1) - 2020-07-25
|
||||||
|
|
||||||
### 优化
|
### 优化
|
||||||
|
|
5
FAQ.md
5
FAQ.md
|
@ -35,7 +35,7 @@
|
||||||
|
|
||||||
## 播放整个歌单或排行榜
|
## 播放整个歌单或排行榜
|
||||||
|
|
||||||
播放在线列表内的歌曲需要将它们都添加到我的列表才能播放,你可以全选列表内的歌曲然后添加到现有列表或者新创建的列表,然后去播放改列表内的歌曲。
|
播放在线列表内的歌曲需要将它们都添加到我的列表才能播放,你可以全选列表内的歌曲然后添加到现有列表或者新创建的列表,然后去播放该列表内的歌曲。
|
||||||
|
|
||||||
## 桌面歌词显示异常
|
## 桌面歌词显示异常
|
||||||
|
|
||||||
|
@ -108,6 +108,7 @@ Windows 7 未开启 Aero 效果时桌面歌词会有问题,详情看下面的*
|
||||||
|
|
||||||
- 若你之前可以安装成功,但现在安装失败,就去**控制面板-程序和功能**或用第三方卸载工具看下有没有之前的版本残留,若同时在不同路径下安装了多个版本就可能会出现该问题,这种情况卸载掉所有版本重新安装即可
|
- 若你之前可以安装成功,但现在安装失败,就去**控制面板-程序和功能**或用第三方卸载工具看下有没有之前的版本残留,若同时在不同路径下安装了多个版本就可能会出现该问题,这种情况卸载掉所有版本重新安装即可
|
||||||
- 清理安装路径下的残留文件
|
- 清理安装路径下的残留文件
|
||||||
|
- 清理注册表(建议用清理工具清理)
|
||||||
|
|
||||||
## 缺少`xxx.dll`
|
## 缺少`xxx.dll`
|
||||||
|
|
||||||
|
@ -118,7 +119,7 @@ Windows 7 未开启 Aero 效果时桌面歌词会有问题,详情看下面的*
|
||||||
|
|
||||||
## 杀毒软件提示有病毒或恶意行为
|
## 杀毒软件提示有病毒或恶意行为
|
||||||
|
|
||||||
本人只能保证我写的代码不包含任何**恶意代码**、**收集用户信息**的行为,并且软件代码已开源,请自行查阅,软件安装包也是由CI拉取源代码构建,构建日志:[windows包](https://ci.appveyor.com/project/lyswhut/lx-music-desktop)、[Mac/Linux包](https://travis-ci.org/lyswhut/lx-music-desktop)<br>
|
本人只能保证我写的代码不包含任何**恶意代码**、**收集用户信息**的行为,并且软件代码已开源,请自行查阅,软件安装包也是由CI拉取源代码构建,构建日志:[windows包](https://ci.appveyor.com/project/lyswhut/lx-music-desktop)、[Mac/Linux包](https://travis-ci.com/github/lyswhut/lx-music-desktop)<br>
|
||||||
尽管如此,但这不意味着软件是100%安全的,由于软件使用了第三方依赖,当这些依赖存在恶意行为时([供应链攻击](https://docs.microsoft.com/zh-cn/windows/security/threat-protection/intelligence/supply-chain-malware)),软件也将会受到牵连,所以我只能尽量选择使用较多人用、信任度较高的依赖。<br>
|
尽管如此,但这不意味着软件是100%安全的,由于软件使用了第三方依赖,当这些依赖存在恶意行为时([供应链攻击](https://docs.microsoft.com/zh-cn/windows/security/threat-protection/intelligence/supply-chain-malware)),软件也将会受到牵连,所以我只能尽量选择使用较多人用、信任度较高的依赖。<br>
|
||||||
当然,以上说明建立的前提是在你所用的安装包是从**本项目主页上写的链接**下载的,或者有相关能力者还可以下载源代码自己构建安装包。
|
当然,以上说明建立的前提是在你所用的安装包是从**本项目主页上写的链接**下载的,或者有相关能力者还可以下载源代码自己构建安装包。
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<a href="https://github.com/lyswhut/lx-music-desktop/releases"><img src="https://img.shields.io/github/release/lyswhut/lx-music-desktop" alt="Release version"></a>
|
<a href="https://github.com/lyswhut/lx-music-desktop/releases"><img src="https://img.shields.io/github/release/lyswhut/lx-music-desktop" alt="Release version"></a>
|
||||||
<a href="https://ci.appveyor.com/project/lyswhut/lx-music-desktop"><img src="https://ci.appveyor.com/api/projects/status/flrsqd5ymp8fnte5?svg=true" alt="Build status"></a>
|
<a href="https://ci.appveyor.com/project/lyswhut/lx-music-desktop"><img src="https://ci.appveyor.com/api/projects/status/flrsqd5ymp8fnte5?svg=true" alt="Build status"></a>
|
||||||
<a href="https://travis-ci.org/lyswhut/lx-music-desktop"><img src="https://travis-ci.org/lyswhut/lx-music-desktop.svg?branch=master" alt="Build status"></a>
|
<a href="https://travis-ci.com/lyswhut/lx-music-desktop"><img src="https://travis-ci.com/lyswhut/lx-music-desktop.svg?branch=master" alt="Build status"></a>
|
||||||
<a href="https://electronjs.org/releases/stable"><img src="https://img.shields.io/github/package-json/dependency-version/lyswhut/lx-music-desktop/dev/electron/master" alt="Electron version"></a>
|
<a href="https://electronjs.org/releases/stable"><img src="https://img.shields.io/github/package-json/dependency-version/lyswhut/lx-music-desktop/dev/electron/master" alt="Electron version"></a>
|
||||||
<!-- <a href="https://github.com/lyswhut/lx-music-desktop/releases"><img src="https://img.shields.io/github/downloads/lyswhut/lx-music-desktop/latest/total" alt="Downloads"></a> -->
|
<!-- <a href="https://github.com/lyswhut/lx-music-desktop/releases"><img src="https://img.shields.io/github/downloads/lyswhut/lx-music-desktop/latest/total" alt="Downloads"></a> -->
|
||||||
<a href="https://github.com/lyswhut/lx-music-desktop/tree/dev"><img src="https://img.shields.io/github/package-json/v/lyswhut/lx-music-desktop/dev" alt="Dev branch version"></a>
|
<a href="https://github.com/lyswhut/lx-music-desktop/tree/dev"><img src="https://img.shields.io/github/package-json/v/lyswhut/lx-music-desktop/dev" alt="Dev branch version"></a>
|
||||||
|
@ -36,7 +36,7 @@
|
||||||
|
|
||||||
所用技术栈:
|
所用技术栈:
|
||||||
|
|
||||||
- Electron 9
|
- Electron 10
|
||||||
- Vue 2
|
- Vue 2
|
||||||
|
|
||||||
已支持的平台:
|
已支持的平台:
|
||||||
|
@ -47,7 +47,7 @@
|
||||||
|
|
||||||
软件变化请查看:[更新日志](https://github.com/lyswhut/lx-music-desktop/blob/master/CHANGELOG.md)<br>
|
软件变化请查看:[更新日志](https://github.com/lyswhut/lx-music-desktop/blob/master/CHANGELOG.md)<br>
|
||||||
软件下载请转到:[发布页面](https://github.com/lyswhut/lx-music-desktop/releases)<br>
|
软件下载请转到:[发布页面](https://github.com/lyswhut/lx-music-desktop/releases)<br>
|
||||||
或者到网盘下载(网盘内有MAC、windows版):`https://t-s.lanzous.com/b0bf2cfa/` 密码:`glqw`<br>
|
或者到网盘下载(网盘内有MAC、windows版):`https://www.lanzoux.com/b0bf2cfa/` 密码:`glqw`<br>
|
||||||
使用常见问题请转至:[常见问题](https://github.com/lyswhut/lx-music-desktop/blob/master/FAQ.md)
|
使用常见问题请转至:[常见问题](https://github.com/lyswhut/lx-music-desktop/blob/master/FAQ.md)
|
||||||
|
|
||||||
### 源码使用方法
|
### 源码使用方法
|
||||||
|
|
|
@ -3,6 +3,6 @@ const isDev = process.env.NODE_ENV === 'development'
|
||||||
module.exports = {
|
module.exports = {
|
||||||
modules: {
|
modules: {
|
||||||
localIdentName: isDev ? '[folder]-[name]--[local]--[hash:base64:5]' : '[hash:base64:5]',
|
localIdentName: isDev ? '[folder]-[name]--[local]--[hash:base64:5]' : '[hash:base64:5]',
|
||||||
|
exportLocalsConvention: 'camelCase',
|
||||||
},
|
},
|
||||||
localsConvention: 'camelCase',
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,7 +14,7 @@ const okayLog = chalk.bgGreen.white(' OKAY ') + ' '
|
||||||
|
|
||||||
|
|
||||||
function build() {
|
function build() {
|
||||||
del.sync(['dist/electron', 'build'])
|
del.sync(['dist/electron/**', 'build/**'])
|
||||||
|
|
||||||
const spinners = new Spinnies({ color: 'blue' })
|
const spinners = new Spinnies({ color: 'blue' })
|
||||||
spinners.add('main', { text: 'main building' })
|
spinners.add('main', { text: 'main building' })
|
||||||
|
|
|
@ -1,9 +1,12 @@
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
const VueLoaderPlugin = require('vue-loader/lib/plugin')
|
const VueLoaderPlugin = require('vue-loader/lib/plugin')
|
||||||
const HTMLPlugin = require('html-webpack-plugin')
|
const HTMLPlugin = require('html-webpack-plugin')
|
||||||
|
const MiniCssExtractPlugin = require('mini-css-extract-plugin')
|
||||||
|
|
||||||
const vueLoaderConfig = require('../vue-loader.config')
|
const vueLoaderConfig = require('../vue-loader.config')
|
||||||
|
const { mergeCSSLoader } = require('../utils')
|
||||||
|
|
||||||
|
const isDev = process.env.NODE_ENV === 'development'
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
target: 'electron-renderer',
|
target: 'electron-renderer',
|
||||||
|
@ -46,6 +49,28 @@ module.exports = {
|
||||||
loader: 'babel-loader',
|
loader: 'babel-loader',
|
||||||
exclude: /node_modules/,
|
exclude: /node_modules/,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
test: /\.css$/,
|
||||||
|
oneOf: mergeCSSLoader(),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
test: /\.less$/,
|
||||||
|
oneOf: mergeCSSLoader({
|
||||||
|
loader: 'less-loader',
|
||||||
|
options: {
|
||||||
|
sourceMap: true,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
test: /\.styl(:?us)?$/,
|
||||||
|
oneOf: mergeCSSLoader({
|
||||||
|
loader: 'stylus-loader',
|
||||||
|
options: {
|
||||||
|
sourceMap: true,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
},
|
||||||
{
|
{
|
||||||
test: /\.pug$/,
|
test: /\.pug$/,
|
||||||
oneOf: [
|
oneOf: [
|
||||||
|
@ -98,5 +123,11 @@ module.exports = {
|
||||||
__dirname,
|
__dirname,
|
||||||
}),
|
}),
|
||||||
new VueLoaderPlugin(),
|
new VueLoaderPlugin(),
|
||||||
|
new MiniCssExtractPlugin({
|
||||||
|
// Options similar to the same options in webpackOptions.output
|
||||||
|
// both options are optional
|
||||||
|
filename: isDev ? '[name].css' : '[name].[contenthash:8].css',
|
||||||
|
chunkFilename: isDev ? '[id].css' : '[id].[contenthash:8].css',
|
||||||
|
}),
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,37 +6,9 @@ const { merge } = require('webpack-merge')
|
||||||
|
|
||||||
const baseConfig = require('./webpack.config.base')
|
const baseConfig = require('./webpack.config.base')
|
||||||
|
|
||||||
const { mergeCSSLoaderDev } = require('../utils')
|
|
||||||
|
|
||||||
module.exports = merge(baseConfig, {
|
module.exports = merge(baseConfig, {
|
||||||
mode: 'development',
|
mode: 'development',
|
||||||
devtool: 'eval-source-map',
|
devtool: 'eval-source-map',
|
||||||
module: {
|
|
||||||
rules: [
|
|
||||||
{
|
|
||||||
test: /\.css$/,
|
|
||||||
oneOf: mergeCSSLoaderDev(),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.less$/,
|
|
||||||
oneOf: mergeCSSLoaderDev({
|
|
||||||
loader: 'less-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.styl(:?us)?$/,
|
|
||||||
oneOf: mergeCSSLoaderDev({
|
|
||||||
loader: 'stylus-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
plugins: [
|
plugins: [
|
||||||
new webpack.HotModuleReplacementPlugin(),
|
new webpack.HotModuleReplacementPlugin(),
|
||||||
new webpack.NoEmitOnErrorsPlugin(),
|
new webpack.NoEmitOnErrorsPlugin(),
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
const webpack = require('webpack')
|
const webpack = require('webpack')
|
||||||
const MiniCssExtractPlugin = require('mini-css-extract-plugin')
|
|
||||||
const OptimizeCSSAssetsPlugin = require('optimize-css-assets-webpack-plugin')
|
const OptimizeCSSAssetsPlugin = require('optimize-css-assets-webpack-plugin')
|
||||||
const TerserPlugin = require('terser-webpack-plugin')
|
const TerserPlugin = require('terser-webpack-plugin')
|
||||||
const CopyWebpackPlugin = require('copy-webpack-plugin')
|
const CopyWebpackPlugin = require('copy-webpack-plugin')
|
||||||
|
@ -8,7 +7,6 @@ const { merge } = require('webpack-merge')
|
||||||
|
|
||||||
const baseConfig = require('./webpack.config.base')
|
const baseConfig = require('./webpack.config.base')
|
||||||
|
|
||||||
const { mergeCSSLoaderProd } = require('../utils')
|
|
||||||
const { dependencies } = require('../../package.json')
|
const { dependencies } = require('../../package.json')
|
||||||
|
|
||||||
let whiteListedModules = ['vue']
|
let whiteListedModules = ['vue']
|
||||||
|
@ -20,32 +18,6 @@ module.exports = merge(baseConfig, {
|
||||||
externals: [
|
externals: [
|
||||||
...Object.keys(dependencies || {}).filter(d => !whiteListedModules.includes(d)),
|
...Object.keys(dependencies || {}).filter(d => !whiteListedModules.includes(d)),
|
||||||
],
|
],
|
||||||
module: {
|
|
||||||
rules: [
|
|
||||||
{
|
|
||||||
test: /\.css$/,
|
|
||||||
oneOf: mergeCSSLoaderProd(),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.less$/,
|
|
||||||
oneOf: mergeCSSLoaderProd({
|
|
||||||
loader: 'less-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.styl(:?us)?$/,
|
|
||||||
oneOf: mergeCSSLoaderProd({
|
|
||||||
loader: 'stylus-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
plugins: [
|
plugins: [
|
||||||
new CopyWebpackPlugin({
|
new CopyWebpackPlugin({
|
||||||
patterns: [
|
patterns: [
|
||||||
|
@ -60,9 +32,6 @@ module.exports = merge(baseConfig, {
|
||||||
NODE_ENV: '"production"',
|
NODE_ENV: '"production"',
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
new MiniCssExtractPlugin({
|
|
||||||
filename: '[name].css',
|
|
||||||
}),
|
|
||||||
new webpack.NamedChunksPlugin(),
|
new webpack.NamedChunksPlugin(),
|
||||||
],
|
],
|
||||||
optimization: {
|
optimization: {
|
||||||
|
|
|
@ -1,9 +1,12 @@
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
const VueLoaderPlugin = require('vue-loader/lib/plugin')
|
const VueLoaderPlugin = require('vue-loader/lib/plugin')
|
||||||
const HTMLPlugin = require('html-webpack-plugin')
|
const HTMLPlugin = require('html-webpack-plugin')
|
||||||
|
const MiniCssExtractPlugin = require('mini-css-extract-plugin')
|
||||||
|
|
||||||
const vueLoaderConfig = require('../vue-loader.config')
|
const vueLoaderConfig = require('../vue-loader.config')
|
||||||
|
const { mergeCSSLoader } = require('../utils')
|
||||||
|
|
||||||
|
const isDev = process.env.NODE_ENV === 'development'
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
target: 'electron-renderer',
|
target: 'electron-renderer',
|
||||||
|
@ -46,6 +49,28 @@ module.exports = {
|
||||||
loader: 'babel-loader',
|
loader: 'babel-loader',
|
||||||
exclude: /node_modules/,
|
exclude: /node_modules/,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
test: /\.css$/,
|
||||||
|
oneOf: mergeCSSLoader(),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
test: /\.less$/,
|
||||||
|
oneOf: mergeCSSLoader({
|
||||||
|
loader: 'less-loader',
|
||||||
|
options: {
|
||||||
|
sourceMap: true,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
test: /\.styl(:?us)?$/,
|
||||||
|
oneOf: mergeCSSLoader({
|
||||||
|
loader: 'stylus-loader',
|
||||||
|
options: {
|
||||||
|
sourceMap: true,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
},
|
||||||
{
|
{
|
||||||
test: /\.pug$/,
|
test: /\.pug$/,
|
||||||
oneOf: [
|
oneOf: [
|
||||||
|
@ -98,5 +123,11 @@ module.exports = {
|
||||||
__dirname,
|
__dirname,
|
||||||
}),
|
}),
|
||||||
new VueLoaderPlugin(),
|
new VueLoaderPlugin(),
|
||||||
|
new MiniCssExtractPlugin({
|
||||||
|
// Options similar to the same options in webpackOptions.output
|
||||||
|
// both options are optional
|
||||||
|
filename: isDev ? '[name].css' : '[name].[contenthash:8].css',
|
||||||
|
chunkFilename: isDev ? '[id].css' : '[id].[contenthash:8].css',
|
||||||
|
}),
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,37 +6,9 @@ const { merge } = require('webpack-merge')
|
||||||
|
|
||||||
const baseConfig = require('./webpack.config.base')
|
const baseConfig = require('./webpack.config.base')
|
||||||
|
|
||||||
const { mergeCSSLoaderDev } = require('../utils')
|
|
||||||
|
|
||||||
module.exports = merge(baseConfig, {
|
module.exports = merge(baseConfig, {
|
||||||
mode: 'development',
|
mode: 'development',
|
||||||
devtool: 'eval-source-map',
|
devtool: 'eval-source-map',
|
||||||
module: {
|
|
||||||
rules: [
|
|
||||||
{
|
|
||||||
test: /\.css$/,
|
|
||||||
oneOf: mergeCSSLoaderDev(),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.less$/,
|
|
||||||
oneOf: mergeCSSLoaderDev({
|
|
||||||
loader: 'less-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.styl(:?us)?$/,
|
|
||||||
oneOf: mergeCSSLoaderDev({
|
|
||||||
loader: 'stylus-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
plugins: [
|
plugins: [
|
||||||
new webpack.HotModuleReplacementPlugin(),
|
new webpack.HotModuleReplacementPlugin(),
|
||||||
new webpack.NoEmitOnErrorsPlugin(),
|
new webpack.NoEmitOnErrorsPlugin(),
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
const webpack = require('webpack')
|
const webpack = require('webpack')
|
||||||
const MiniCssExtractPlugin = require('mini-css-extract-plugin')
|
|
||||||
const OptimizeCSSAssetsPlugin = require('optimize-css-assets-webpack-plugin')
|
const OptimizeCSSAssetsPlugin = require('optimize-css-assets-webpack-plugin')
|
||||||
const TerserPlugin = require('terser-webpack-plugin')
|
const TerserPlugin = require('terser-webpack-plugin')
|
||||||
const CopyWebpackPlugin = require('copy-webpack-plugin')
|
const CopyWebpackPlugin = require('copy-webpack-plugin')
|
||||||
|
@ -8,7 +7,6 @@ const { merge } = require('webpack-merge')
|
||||||
|
|
||||||
const baseConfig = require('./webpack.config.base')
|
const baseConfig = require('./webpack.config.base')
|
||||||
|
|
||||||
const { mergeCSSLoaderProd } = require('../utils')
|
|
||||||
const { dependencies } = require('../../package.json')
|
const { dependencies } = require('../../package.json')
|
||||||
|
|
||||||
let whiteListedModules = ['vue']
|
let whiteListedModules = ['vue']
|
||||||
|
@ -20,32 +18,6 @@ module.exports = merge(baseConfig, {
|
||||||
externals: [
|
externals: [
|
||||||
...Object.keys(dependencies || {}).filter(d => !whiteListedModules.includes(d)),
|
...Object.keys(dependencies || {}).filter(d => !whiteListedModules.includes(d)),
|
||||||
],
|
],
|
||||||
module: {
|
|
||||||
rules: [
|
|
||||||
{
|
|
||||||
test: /\.css$/,
|
|
||||||
oneOf: mergeCSSLoaderProd(),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.less$/,
|
|
||||||
oneOf: mergeCSSLoaderProd({
|
|
||||||
loader: 'less-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
test: /\.styl(:?us)?$/,
|
|
||||||
oneOf: mergeCSSLoaderProd({
|
|
||||||
loader: 'stylus-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
plugins: [
|
plugins: [
|
||||||
new CopyWebpackPlugin({
|
new CopyWebpackPlugin({
|
||||||
patterns: [
|
patterns: [
|
||||||
|
@ -60,9 +32,6 @@ module.exports = merge(baseConfig, {
|
||||||
NODE_ENV: '"production"',
|
NODE_ENV: '"production"',
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
new MiniCssExtractPlugin({
|
|
||||||
filename: '[name].css',
|
|
||||||
}),
|
|
||||||
new webpack.NamedChunksPlugin(),
|
new webpack.NamedChunksPlugin(),
|
||||||
],
|
],
|
||||||
optimization: {
|
optimization: {
|
||||||
|
|
|
@ -2,62 +2,21 @@ const MiniCssExtractPlugin = require('mini-css-extract-plugin')
|
||||||
const cssLoaderConfig = require('./css-loader.config')
|
const cssLoaderConfig = require('./css-loader.config')
|
||||||
const chalk = require('chalk')
|
const chalk = require('chalk')
|
||||||
|
|
||||||
// merge css-loader in development
|
const isDev = process.env.NODE_ENV === 'development'
|
||||||
exports.mergeCSSLoaderDev = beforeLoader => {
|
|
||||||
const loader = [
|
|
||||||
// 这里匹配 `<style module>`
|
|
||||||
{
|
|
||||||
resourceQuery: /module/,
|
|
||||||
use: [
|
|
||||||
'vue-style-loader',
|
|
||||||
{
|
|
||||||
loader: 'css-loader',
|
|
||||||
options: Object.assign({
|
|
||||||
sourceMap: true,
|
|
||||||
}, cssLoaderConfig),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
loader: 'postcss-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
// 这里匹配普通的 `<style>` 或 `<style scoped>`
|
|
||||||
{
|
|
||||||
use: [
|
|
||||||
'vue-style-loader',
|
|
||||||
{
|
|
||||||
loader: 'css-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
loader: 'postcss-loader',
|
|
||||||
options: {
|
|
||||||
sourceMap: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
]
|
|
||||||
if (beforeLoader) {
|
|
||||||
loader[0].use.push(beforeLoader)
|
|
||||||
loader[1].use.push(beforeLoader)
|
|
||||||
}
|
|
||||||
return loader
|
|
||||||
}
|
|
||||||
|
|
||||||
// merge css-loader in production
|
// merge css-loader
|
||||||
exports.mergeCSSLoaderProd = beforeLoader => {
|
exports.mergeCSSLoader = beforeLoader => {
|
||||||
const loader = [
|
const loader = [
|
||||||
// 这里匹配 `<style module>`
|
// 这里匹配 `<style module>`
|
||||||
{
|
{
|
||||||
resourceQuery: /module/,
|
resourceQuery: /module/,
|
||||||
use: [
|
use: [
|
||||||
MiniCssExtractPlugin.loader,
|
{
|
||||||
|
loader: MiniCssExtractPlugin.loader,
|
||||||
|
options: {
|
||||||
|
hmr: isDev,
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
loader: 'css-loader',
|
loader: 'css-loader',
|
||||||
options: cssLoaderConfig,
|
options: cssLoaderConfig,
|
||||||
|
@ -68,7 +27,12 @@ exports.mergeCSSLoaderProd = beforeLoader => {
|
||||||
// 这里匹配普通的 `<style>` 或 `<style scoped>`
|
// 这里匹配普通的 `<style>` 或 `<style scoped>`
|
||||||
{
|
{
|
||||||
use: [
|
use: [
|
||||||
MiniCssExtractPlugin.loader,
|
{
|
||||||
|
loader: MiniCssExtractPlugin.loader,
|
||||||
|
options: {
|
||||||
|
hmr: isDev,
|
||||||
|
},
|
||||||
|
},
|
||||||
'css-loader',
|
'css-loader',
|
||||||
'postcss-loader',
|
'postcss-loader',
|
||||||
],
|
],
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
70
package.json
70
package.json
|
@ -1,11 +1,10 @@
|
||||||
{
|
{
|
||||||
"name": "lx-music-desktop",
|
"name": "lx-music-desktop",
|
||||||
"version": "1.0.1",
|
"version": "1.1.0",
|
||||||
"description": "一个免费的音乐下载助手",
|
"description": "一个免费的音乐下载助手",
|
||||||
"main": "./dist/electron/main.js",
|
"main": "./dist/electron/main.js",
|
||||||
"productName": "lx-music-desktop",
|
"productName": "lx-music-desktop",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"php": "babel file.php -o file.js",
|
|
||||||
"pack": "node build-config/pack.js && npm run pack:win",
|
"pack": "node build-config/pack.js && npm run pack:win",
|
||||||
"pack:win": "npm run pack:win:setup:x86_64 && npm run pack:win:7z",
|
"pack:win": "npm run pack:win:setup:x86_64 && npm run pack:win:7z",
|
||||||
"pack:win:setup:x86_64": "cross-env TARGET=win_安装版 ARCH=x86_64 electron-builder -w=nsis --x64 --ia32",
|
"pack:win:setup:x86_64": "cross-env TARGET=win_安装版 ARCH=x86_64 electron-builder -w=nsis --x64 --ia32",
|
||||||
|
@ -51,7 +50,6 @@
|
||||||
"pack:dir": "node build-config/pack.js && electron-builder --dir",
|
"pack:dir": "node build-config/pack.js && electron-builder --dir",
|
||||||
"dev": "node build-config/runner-dev.js",
|
"dev": "node build-config/runner-dev.js",
|
||||||
"clean:electron": "rimraf dist/electron",
|
"clean:electron": "rimraf dist/electron",
|
||||||
"clean:web": "rimraf dist/web",
|
|
||||||
"clean": "rimraf dist && rimraf build",
|
"clean": "rimraf dist && rimraf build",
|
||||||
"build:main": "cross-env NODE_ENV=production webpack --config build-config/main/webpack.config.prod.js --progress --hide-modules",
|
"build:main": "cross-env NODE_ENV=production webpack --config build-config/main/webpack.config.prod.js --progress --hide-modules",
|
||||||
"build:renderer": "cross-env NODE_ENV=production webpack --config build-config/renderer/webpack.config.prod.js --progress --hide-modules",
|
"build:renderer": "cross-env NODE_ENV=production webpack --config build-config/renderer/webpack.config.prod.js --progress --hide-modules",
|
||||||
|
@ -61,7 +59,7 @@
|
||||||
"lint:fix": "eslint --ext .js,.vue -f ./node_modules/eslint-formatter-friendly --fix src"
|
"lint:fix": "eslint --ext .js,.vue -f ./node_modules/eslint-formatter-friendly --fix src"
|
||||||
},
|
},
|
||||||
"browserslist": [
|
"browserslist": [
|
||||||
"Electron 9.1.1"
|
"Electron 10.1.2"
|
||||||
],
|
],
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">= 12"
|
"node": ">= 12"
|
||||||
|
@ -101,7 +99,7 @@
|
||||||
"allowToChangeInstallationDirectory": true,
|
"allowToChangeInstallationDirectory": true,
|
||||||
"differentialPackage": true,
|
"differentialPackage": true,
|
||||||
"license": "./licenses/license.rtf",
|
"license": "./licenses/license.rtf",
|
||||||
"shortcutName": "lx-music"
|
"shortcutName": "LX Music"
|
||||||
},
|
},
|
||||||
"dmg": {
|
"dmg": {
|
||||||
"window": {
|
"window": {
|
||||||
|
@ -112,7 +110,7 @@
|
||||||
{
|
{
|
||||||
"x": 106,
|
"x": 106,
|
||||||
"y": 252,
|
"y": 252,
|
||||||
"name": "lx-music"
|
"name": "LX Music"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"x": 490,
|
"x": 490,
|
||||||
|
@ -160,47 +158,46 @@
|
||||||
},
|
},
|
||||||
"homepage": "https://github.com/lyswhut/lx-music-desktop#readme",
|
"homepage": "https://github.com/lyswhut/lx-music-desktop#readme",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "^7.10.5",
|
"@babel/core": "^7.11.6",
|
||||||
"@babel/plugin-syntax-dynamic-import": "^7.8.3",
|
"@babel/plugin-syntax-dynamic-import": "^7.8.3",
|
||||||
"@babel/plugin-transform-modules-umd": "^7.10.4",
|
"@babel/plugin-transform-modules-umd": "^7.10.4",
|
||||||
"@babel/plugin-transform-runtime": "^7.10.5",
|
"@babel/plugin-transform-runtime": "^7.11.5",
|
||||||
"@babel/polyfill": "^7.10.4",
|
"@babel/polyfill": "^7.11.5",
|
||||||
"@babel/preset-env": "^7.10.4",
|
"@babel/preset-env": "^7.11.5",
|
||||||
"babel-eslint": "^10.1.0",
|
"babel-eslint": "^10.1.0",
|
||||||
"babel-loader": "^8.1.0",
|
"babel-loader": "^8.1.0",
|
||||||
"babel-minify-webpack-plugin": "^0.3.1",
|
"babel-minify-webpack-plugin": "^0.3.1",
|
||||||
"babel-preset-minify": "^0.5.1",
|
"babel-preset-minify": "^0.5.1",
|
||||||
"browserslist": "^4.13.0",
|
|
||||||
"cfonts": "^2.8.6",
|
"cfonts": "^2.8.6",
|
||||||
"chalk": "^4.1.0",
|
"chalk": "^4.1.0",
|
||||||
"changelog-parser": "^2.8.0",
|
"changelog-parser": "^2.8.0",
|
||||||
"copy-webpack-plugin": "^6.0.3",
|
"copy-webpack-plugin": "^6.1.0",
|
||||||
"core-js": "^3.6.5",
|
"core-js": "^3.6.5",
|
||||||
"cross-env": "^7.0.2",
|
"cross-env": "^7.0.2",
|
||||||
"css-loader": "^3.6.0",
|
"css-loader": "^4.3.0",
|
||||||
"del": "^5.1.0",
|
"del": "^5.1.0",
|
||||||
"electron": "^9.1.1",
|
"electron": "^10.1.2",
|
||||||
"electron-builder": "^22.7.0",
|
"electron-builder": "^22.8.1",
|
||||||
"electron-debug": "^3.1.0",
|
"electron-debug": "^3.1.0",
|
||||||
"electron-devtools-installer": "^3.1.1",
|
"electron-devtools-installer": "^3.1.1",
|
||||||
"eslint": "^7.5.0",
|
"eslint": "^7.9.0",
|
||||||
"eslint-config-standard": "^14.1.1",
|
"eslint-config-standard": "^14.1.1",
|
||||||
"eslint-formatter-friendly": "^7.0.0",
|
"eslint-formatter-friendly": "^7.0.0",
|
||||||
"eslint-loader": "^4.0.2",
|
"eslint-loader": "^4.0.2",
|
||||||
"eslint-plugin-html": "^6.0.2",
|
"eslint-plugin-html": "^6.1.0",
|
||||||
"eslint-plugin-import": "^2.22.0",
|
"eslint-plugin-import": "^2.22.0",
|
||||||
"eslint-plugin-node": "^11.1.0",
|
"eslint-plugin-node": "^11.1.0",
|
||||||
"eslint-plugin-promise": "^4.2.1",
|
"eslint-plugin-promise": "^4.2.1",
|
||||||
"eslint-plugin-standard": "^4.0.1",
|
"eslint-plugin-standard": "^4.0.1",
|
||||||
"file-loader": "^6.0.0",
|
"file-loader": "^6.1.0",
|
||||||
"friendly-errors-webpack-plugin": "^1.7.0",
|
"friendly-errors-webpack-plugin": "^1.7.0",
|
||||||
"html-webpack-plugin": "^4.3.0",
|
"html-webpack-plugin": "^4.4.1",
|
||||||
"less": "^3.12.2",
|
"less": "^3.12.2",
|
||||||
"less-loader": "^6.2.0",
|
"less-loader": "^7.0.1",
|
||||||
"markdown-it": "^11.0.0",
|
"markdown-it": "^11.0.1",
|
||||||
"mini-css-extract-plugin": "^0.9.0",
|
"mini-css-extract-plugin": "^0.11.2",
|
||||||
"optimize-css-assets-webpack-plugin": "^5.0.3",
|
"optimize-css-assets-webpack-plugin": "^5.0.4",
|
||||||
"postcss-loader": "^3.0.0",
|
"postcss-loader": "^4.0.2",
|
||||||
"postcss-pxtorem": "^5.1.1",
|
"postcss-pxtorem": "^5.1.1",
|
||||||
"pug": "^3.0.0",
|
"pug": "^3.0.0",
|
||||||
"pug-loader": "^2.4.0",
|
"pug-loader": "^2.4.0",
|
||||||
|
@ -210,33 +207,32 @@
|
||||||
"spinnies": "^0.5.1",
|
"spinnies": "^0.5.1",
|
||||||
"stylus": "^0.54.8",
|
"stylus": "^0.54.8",
|
||||||
"stylus-loader": "^3.0.2",
|
"stylus-loader": "^3.0.2",
|
||||||
"terser-webpack-plugin": "^3.0.7",
|
"terser-webpack-plugin": "^4.2.1",
|
||||||
"url-loader": "^4.1.0",
|
"url-loader": "^4.1.0",
|
||||||
"vue-loader": "^15.9.3",
|
"vue-loader": "^15.9.3",
|
||||||
"vue-style-loader": "^4.1.2",
|
"vue-template-compiler": "^2.6.12",
|
||||||
"vue-template-compiler": "^2.6.11",
|
"webpack": "^4.44.2",
|
||||||
"webpack": "^4.43.0",
|
|
||||||
"webpack-cli": "^3.3.12",
|
"webpack-cli": "^3.3.12",
|
||||||
"webpack-dev-server": "^3.11.0",
|
"webpack-dev-server": "^3.11.0",
|
||||||
"webpack-hot-middleware": "^2.25.0",
|
"webpack-hot-middleware": "^2.25.0",
|
||||||
"webpack-merge": "^5.0.9"
|
"webpack-merge": "^5.1.4"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"crypto-js": "^4.0.0",
|
"crypto-js": "^4.0.0",
|
||||||
"dnscache": "^1.0.2",
|
"dnscache": "^1.0.2",
|
||||||
"electron-log": "^4.2.2",
|
"electron-log": "^4.2.4",
|
||||||
"electron-store": "^6.0.0",
|
"electron-store": "^6.0.0",
|
||||||
"electron-updater": "^4.3.1",
|
"electron-updater": "^4.3.5",
|
||||||
"iconv-lite": "^0.6.2",
|
"iconv-lite": "^0.6.2",
|
||||||
"image-size": "^0.8.3",
|
"image-size": "^0.9.1",
|
||||||
"js-htmlencode": "^0.3.0",
|
"js-htmlencode": "^0.3.0",
|
||||||
"lrc-file-parser": "^1.0.5",
|
"lrc-file-parser": "^1.0.5",
|
||||||
"needle": "^2.5.0",
|
"needle": "^2.5.2",
|
||||||
"node-id3": "^0.1.17",
|
"node-id3": "^0.1.18",
|
||||||
"request": "^2.88.2",
|
"request": "^2.88.2",
|
||||||
"vue": "^2.6.11",
|
"vue": "^2.6.12",
|
||||||
"vue-i18n": "^8.18.2",
|
"vue-i18n": "^8.21.1",
|
||||||
"vue-router": "^3.3.4",
|
"vue-router": "^3.4.3",
|
||||||
"vuex": "^3.5.1",
|
"vuex": "^3.5.1",
|
||||||
"vuex-router-sync": "^5.0.0"
|
"vuex-router-sync": "^5.0.0"
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,30 @@
|
||||||
### 优化
|
### 新增
|
||||||
|
|
||||||
- 对我的列表歌曲搜索结果进行相似度排序
|
- 在歌单详情界面新增播放当前歌单按钮、收藏歌单按钮,注:播放歌单不会将歌曲添加到试听列表
|
||||||
|
- 新增`不允许将歌词窗口拖出主屏幕之外`的设置项,默认开启,在连接多个屏幕时想要拖动到其他屏幕时可关闭此设置
|
||||||
|
- 新增大部分平台的歌词翻译,感谢 @InoriHimea 提供的[krc解码算法](https://github.com/lyswhut/lx-music-desktop/issues/296#issuecomment-683285784)
|
||||||
|
- 新增`显示歌词翻译`设置,默认开启,仅支持某些平台,注:无论该设置是否开启,嵌入或下载歌词时都不会带上翻译
|
||||||
|
- 新增`显示切换动画`设置,默认开启,关闭时将基本禁用软件内的所有切换动画
|
||||||
|
- 播放状态栏新增桌面歌词的开关、播放模式的切换、歌曲的收藏按钮,Thanks to @andylow for the [icon](https://github.com/lyswhut/lx-music-desktop/pull/309)!
|
||||||
|
|
||||||
### 修复
|
### 修复
|
||||||
|
|
||||||
- 修复在 Windows 系统下缩放比非100%时,拖动桌面歌词会自动加大桌面歌词窗口的问题
|
- 修复使用全局快捷键还原窗口时,窗口没有获取焦点的问题
|
||||||
|
- 修复我的列表搜索对最后一个字符的匹配问题
|
||||||
|
- 修复窗口在`较小`模式下最小化/关闭按钮不居中的问题
|
||||||
|
|
||||||
|
### 优化
|
||||||
|
|
||||||
|
- 桌面歌词当前播放行改为上下居中
|
||||||
|
- 为区分静音状态,静音时音量条会变淡,调整音量条时将会取消静音
|
||||||
|
- 优化随机播放机制,现在通过`下一曲`切换歌曲时,直到播放完整个列表之前将不会再随机到之前播放过的歌曲,并且通过`上一曲`可以正确播放上一首歌曲
|
||||||
|
- 当下载目录没有写入权限时将显示没有写入权限的提示
|
||||||
|
|
||||||
|
### 移除
|
||||||
|
|
||||||
|
- 移除默认的全局声音媒体快捷键接管
|
||||||
|
- 移除对百度音乐的支持,因百度音乐原有的大部分API失效,而且该平台相对其他平台来说音乐太少了,可有可无,以后再看情况恢复
|
||||||
|
|
||||||
|
### 其他
|
||||||
|
|
||||||
|
- 更新electron到 10.1.2
|
||||||
|
|
File diff suppressed because one or more lines are too long
|
@ -44,21 +44,6 @@ module.exports = {
|
||||||
name: null,
|
name: null,
|
||||||
action: hotKeyPlayer.next.action,
|
action: hotKeyPlayer.next.action,
|
||||||
},
|
},
|
||||||
VolumeUp: {
|
|
||||||
type: hotKeyPlayer.volume_up.type,
|
|
||||||
name: null,
|
|
||||||
action: hotKeyPlayer.volume_up.action,
|
|
||||||
},
|
|
||||||
VolumeDown: {
|
|
||||||
type: hotKeyPlayer.volume_down.type,
|
|
||||||
name: null,
|
|
||||||
action: hotKeyPlayer.volume_down.action,
|
|
||||||
},
|
|
||||||
VolumeMute: {
|
|
||||||
type: hotKeyPlayer.volume_mute.type,
|
|
||||||
name: null,
|
|
||||||
action: hotKeyPlayer.volume_mute.action,
|
|
||||||
},
|
|
||||||
'mod+alt+f5': {
|
'mod+alt+f5': {
|
||||||
type: hotKeyPlayer.toggle_play.type,
|
type: hotKeyPlayer.toggle_play.type,
|
||||||
name: hotKeyPlayer.toggle_play.name,
|
name: hotKeyPlayer.toggle_play.name,
|
||||||
|
|
|
@ -3,7 +3,7 @@ const os = require('os')
|
||||||
const { isMac } = require('./utils')
|
const { isMac } = require('./utils')
|
||||||
|
|
||||||
const defaultSetting = {
|
const defaultSetting = {
|
||||||
version: '1.0.34',
|
version: '1.0.38',
|
||||||
player: {
|
player: {
|
||||||
togglePlayMethod: 'listLoop',
|
togglePlayMethod: 'listLoop',
|
||||||
highQuality: false,
|
highQuality: false,
|
||||||
|
@ -12,6 +12,7 @@ const defaultSetting = {
|
||||||
isMute: false,
|
isMute: false,
|
||||||
mediaDeviceId: 'default',
|
mediaDeviceId: 'default',
|
||||||
isMediaDeviceRemovedStopPlay: false,
|
isMediaDeviceRemovedStopPlay: false,
|
||||||
|
isShowLyricTransition: true,
|
||||||
},
|
},
|
||||||
desktopLyric: {
|
desktopLyric: {
|
||||||
enable: false,
|
enable: false,
|
||||||
|
@ -22,6 +23,7 @@ const defaultSetting = {
|
||||||
x: null,
|
x: null,
|
||||||
y: null,
|
y: null,
|
||||||
theme: 0,
|
theme: 0,
|
||||||
|
isLockScreen: true,
|
||||||
style: {
|
style: {
|
||||||
fontSize: 120,
|
fontSize: 120,
|
||||||
opacity: 95,
|
opacity: 95,
|
||||||
|
@ -86,6 +88,7 @@ const defaultSetting = {
|
||||||
sourceId: 'kw',
|
sourceId: 'kw',
|
||||||
apiSource: 'temp',
|
apiSource: 'temp',
|
||||||
sourceNameType: 'alias',
|
sourceNameType: 'alias',
|
||||||
|
isShowAnimation: true,
|
||||||
randomAnimate: true,
|
randomAnimate: true,
|
||||||
ignoreVersion: null,
|
ignoreVersion: null,
|
||||||
isAgreePact: false,
|
isAgreePact: false,
|
||||||
|
|
|
@ -43,6 +43,14 @@ const names = {
|
||||||
hide_toggle: 'hide_toggle',
|
hide_toggle: 'hide_toggle',
|
||||||
get_data_path: 'get_data_path',
|
get_data_path: 'get_data_path',
|
||||||
show_dialog: 'show_dialog',
|
show_dialog: 'show_dialog',
|
||||||
|
|
||||||
|
get_setting: 'get_setting',
|
||||||
|
get_playlist: 'get_playlist',
|
||||||
|
save_playlist: 'save_playlist',
|
||||||
|
get_data: 'get_data',
|
||||||
|
set_data: 'set_data',
|
||||||
|
save_data: 'save_data',
|
||||||
|
get_hot_key: 'get_hot_key',
|
||||||
},
|
},
|
||||||
winLyric: {
|
winLyric: {
|
||||||
close: 'close',
|
close: 'close',
|
||||||
|
|
|
@ -151,8 +151,11 @@ exports.initSetting = () => {
|
||||||
name: 'config',
|
name: 'config',
|
||||||
})
|
})
|
||||||
let setting = electronStore_config.get('setting')
|
let setting = electronStore_config.get('setting')
|
||||||
if (!electronStore_config.get('version') && setting) { // 迁移配置
|
if (setting) {
|
||||||
electronStore_config.set('version', electronStore_config.get('setting.version'))
|
let version = electronStore_config.get('version')
|
||||||
|
if (!version) { // 迁移配置
|
||||||
|
version = electronStore_config.get('setting.version')
|
||||||
|
electronStore_config.set('version', version)
|
||||||
electronStore_config.delete('setting.version')
|
electronStore_config.delete('setting.version')
|
||||||
const list = electronStore_config.get('list')
|
const list = electronStore_config.get('list')
|
||||||
if (list) {
|
if (list) {
|
||||||
|
@ -168,13 +171,14 @@ exports.initSetting = () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
// 迁移列表滚动位置设置 ~0.18.3
|
// 迁移列表滚动位置设置 ~0.18.3
|
||||||
if (setting && setting.list.scroll) {
|
if (setting.list.scroll) {
|
||||||
let scroll = setting.list.scroll
|
let scroll = setting.list.scroll
|
||||||
electronStore_list.set('defaultList.location', scroll.locations.defaultList || 0)
|
electronStore_list.set('defaultList.location', scroll.locations.defaultList || 0)
|
||||||
electronStore_list.set('loveList.location', scroll.locations.loveList || 0)
|
electronStore_list.set('loveList.location', scroll.locations.loveList || 0)
|
||||||
electronStore_config.delete('setting.list.scroll')
|
electronStore_config.delete('setting.list.scroll')
|
||||||
electronStore_config.set('setting.list.isSaveScrollLocation', scroll.enable)
|
electronStore_config.set('setting.list.isSaveScrollLocation', scroll.enable)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const { version: settingVersion, setting: newSetting } = exports.mergeSetting(setting, electronStore_config.get('version'))
|
const { version: settingVersion, setting: newSetting } = exports.mergeSetting(setting, electronStore_config.get('version'))
|
||||||
|
|
||||||
|
@ -184,7 +188,7 @@ exports.initSetting = () => {
|
||||||
// newSetting.controlBtnPosition = 'right'
|
// newSetting.controlBtnPosition = 'right'
|
||||||
electronStore_config.set('version', settingVersion)
|
electronStore_config.set('version', settingVersion)
|
||||||
electronStore_config.set('setting', newSetting)
|
electronStore_config.set('setting', newSetting)
|
||||||
return newSetting
|
return { version: settingVersion, setting: newSetting }
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -202,6 +206,15 @@ exports.initHotKey = () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
let globalConfig = electronStore_hotKey.get('global')
|
let globalConfig = electronStore_hotKey.get('global')
|
||||||
|
|
||||||
|
// 移除v1.0.1及之前设置的全局声音媒体快捷键接管
|
||||||
|
if (globalConfig && globalConfig.keys.VolumeUp) {
|
||||||
|
delete globalConfig.keys.VolumeUp
|
||||||
|
delete globalConfig.keys.VolumeDown
|
||||||
|
delete globalConfig.keys.VolumeMute
|
||||||
|
electronStore_hotKey.set('global', globalConfig)
|
||||||
|
}
|
||||||
|
|
||||||
if (!globalConfig) {
|
if (!globalConfig) {
|
||||||
globalConfig = defaultHotKey.global
|
globalConfig = defaultHotKey.global
|
||||||
electronStore_hotKey.set('global', globalConfig)
|
electronStore_hotKey.set('global', globalConfig)
|
||||||
|
|
|
@ -22,6 +22,7 @@ global.lx_event.mainWindow.on(MAIN_WINDOW_EVENT_NAME.toggle_minimize, () => {
|
||||||
global.modules.mainWindow.show()
|
global.modules.mainWindow.show()
|
||||||
}
|
}
|
||||||
global.modules.mainWindow.restore()
|
global.modules.mainWindow.restore()
|
||||||
|
global.modules.mainWindow.focus()
|
||||||
} else {
|
} else {
|
||||||
global.modules.mainWindow.minimize()
|
global.modules.mainWindow.minimize()
|
||||||
}
|
}
|
||||||
|
|
|
@ -124,7 +124,9 @@ global.appHotKey = {
|
||||||
}
|
}
|
||||||
|
|
||||||
function init() {
|
function init() {
|
||||||
global.appSetting = initSetting()
|
const info = initSetting()
|
||||||
|
global.appSetting = info.setting
|
||||||
|
global.appSettingVersion = info.version
|
||||||
global.appHotKey.config = initHotKey()
|
global.appHotKey.config = initHotKey()
|
||||||
global.lx_event.common.initSetting()
|
global.lx_event.common.initSetting()
|
||||||
global.lx_event.hotKey.init()
|
global.lx_event.hotKey.init()
|
||||||
|
|
|
@ -1,16 +1,19 @@
|
||||||
const { common: COMMON_EVENT_NAME, winLyric: WIN_LYRIC_EVENT_NAME, hotKey: HOT_KEY_EVENT_NAME, mainWindow: MAIN_WINDOW_EVENT_NAME } = require('../../events/_name')
|
const { common: COMMON_EVENT_NAME, winLyric: WIN_LYRIC_EVENT_NAME, hotKey: HOT_KEY_EVENT_NAME, mainWindow: MAIN_WINDOW_EVENT_NAME } = require('../../events/_name')
|
||||||
const { mainSend, NAMES: { winLyric: ipcWinLyricNames } } = require('../../../common/ipc')
|
const { mainSend, NAMES: { winLyric: ipcWinLyricNames } } = require('../../../common/ipc')
|
||||||
const { desktop_lyric } = require('../../../common/hotKey')
|
const { desktop_lyric } = require('../../../common/hotKey')
|
||||||
|
const { setLyricWindow } = require('./utils')
|
||||||
|
|
||||||
let isLock = null
|
let isLock = null
|
||||||
let isEnable = null
|
let isEnable = null
|
||||||
let isAlwaysOnTop = null
|
let isAlwaysOnTop = null
|
||||||
|
let isLockScreen = null
|
||||||
const setLrcConfig = () => {
|
const setLrcConfig = () => {
|
||||||
let desktopLyric = global.appSetting.desktopLyric
|
let desktopLyric = global.appSetting.desktopLyric
|
||||||
if (global.modules.lyricWindow) {
|
if (global.modules.lyricWindow) {
|
||||||
mainSend(global.modules.lyricWindow, ipcWinLyricNames.set_lyric_config, {
|
mainSend(global.modules.lyricWindow, ipcWinLyricNames.set_lyric_config, {
|
||||||
config: desktopLyric,
|
config: desktopLyric,
|
||||||
languageId: global.appSetting.langId,
|
languageId: global.appSetting.langId,
|
||||||
|
isShowLyricTransition: global.appSetting.player.isShowLyricTransition,
|
||||||
})
|
})
|
||||||
if (isLock != desktopLyric.isLock) {
|
if (isLock != desktopLyric.isLock) {
|
||||||
isLock = desktopLyric.isLock
|
isLock = desktopLyric.isLock
|
||||||
|
@ -33,10 +36,21 @@ const setLrcConfig = () => {
|
||||||
global.lx_event.winLyric.close()
|
global.lx_event.winLyric.close()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (isLockScreen != desktopLyric.isLockScreen) {
|
||||||
|
isLockScreen = desktopLyric.isLockScreen
|
||||||
|
if (desktopLyric.isLockScreen) {
|
||||||
|
setLyricWindow({
|
||||||
|
x: desktopLyric.x,
|
||||||
|
y: desktopLyric.y,
|
||||||
|
w: desktopLyric.width,
|
||||||
|
h: desktopLyric.height,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
global.lx_event.common.on(COMMON_EVENT_NAME.config, name => {
|
global.lx_event.common.on(COMMON_EVENT_NAME.config, name => {
|
||||||
if (WIN_LYRIC_EVENT_NAME.name === name) return
|
if (WIN_LYRIC_EVENT_NAME.name === name) return
|
||||||
setLrcConfig(false)
|
setLrcConfig()
|
||||||
})
|
})
|
||||||
|
|
||||||
global.lx_event.mainWindow.on(MAIN_WINDOW_EVENT_NAME.setLyricInfo, info => {
|
global.lx_event.mainWindow.on(MAIN_WINDOW_EVENT_NAME.setLyricInfo, info => {
|
||||||
|
|
|
@ -80,14 +80,18 @@ const createWindow = () => {
|
||||||
x = screenWidth - width - offset
|
x = screenWidth - width - offset
|
||||||
y = screenHeight - height - offset
|
y = screenHeight - height - offset
|
||||||
}
|
}
|
||||||
|
if (global.appSetting.desktopLyric.isLockScreen) {
|
||||||
|
x = Math.max(-offset, screenWidth < (width + x) ? screenWidth - width : x)
|
||||||
|
y = Math.max(-offset, screenHeight < (height + y) ? screenHeight - height : y)
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* Initial window options
|
* Initial window options
|
||||||
*/
|
*/
|
||||||
global.modules.lyricWindow = new BrowserWindow({
|
global.modules.lyricWindow = new BrowserWindow({
|
||||||
height: Math.max(height > screenHeight ? screenHeight : height, 80),
|
height: Math.max(height > screenHeight ? screenHeight : height, 80),
|
||||||
width: Math.max(width > screenWidth ? screenWidth : width, 380),
|
width: Math.max(width > screenWidth ? screenWidth : width, 380),
|
||||||
x: Math.max(-offset, screenWidth < (width + x) ? screenWidth - width : x),
|
x,
|
||||||
y: Math.max(-offset, screenHeight < (height + y) ? screenHeight - height : y),
|
y,
|
||||||
minWidth: 380,
|
minWidth: 380,
|
||||||
minHeight: 80,
|
minHeight: 80,
|
||||||
useContentSize: true,
|
useContentSize: true,
|
||||||
|
|
|
@ -8,6 +8,7 @@ const {
|
||||||
},
|
},
|
||||||
} = require('../../../common/ipc')
|
} = require('../../../common/ipc')
|
||||||
const { winLyric: WIN_LYRIC_EVENT_NAME } = require('../../events/_name')
|
const { winLyric: WIN_LYRIC_EVENT_NAME } = require('../../events/_name')
|
||||||
|
const { setLyricWindow } = require('./utils')
|
||||||
|
|
||||||
mainOn(ipcWinLyricNames.get_lyric_info, (event, action) => {
|
mainOn(ipcWinLyricNames.get_lyric_info, (event, action) => {
|
||||||
if (!global.modules.mainWindow) return
|
if (!global.modules.mainWindow) return
|
||||||
|
@ -23,54 +24,9 @@ mainOn(ipcWinLyricNames.set_lyric_config, (event, config) => {
|
||||||
})
|
})
|
||||||
|
|
||||||
mainHandle(ipcWinLyricNames.get_lyric_config, async() => {
|
mainHandle(ipcWinLyricNames.get_lyric_config, async() => {
|
||||||
return { config: global.appSetting.desktopLyric, languageId: global.appSetting.langId }
|
return { config: global.appSetting.desktopLyric, languageId: global.appSetting.langId, isShowLyricTransition: global.appSetting.player.isShowLyricTransition }
|
||||||
})
|
})
|
||||||
|
|
||||||
let bounds
|
mainOn(ipcWinLyricNames.set_win_bounds, (event, options) => {
|
||||||
let winX
|
setLyricWindow(options)
|
||||||
let winY
|
|
||||||
let wasW
|
|
||||||
let wasY
|
|
||||||
let offset = 8
|
|
||||||
mainOn(ipcWinLyricNames.set_win_bounds, (event, { x = 0, y = 0, w = 0, h = 0 }) => {
|
|
||||||
if (!global.modules.lyricWindow) return
|
|
||||||
bounds = global.modules.lyricWindow.getBounds()
|
|
||||||
wasW = global.envParams.workAreaSize.width
|
|
||||||
wasY = global.envParams.workAreaSize.height + offset
|
|
||||||
|
|
||||||
bounds.width = w
|
|
||||||
bounds.height = h
|
|
||||||
if (bounds.width > wasW - offset) {
|
|
||||||
bounds.width = wasW - offset
|
|
||||||
} else if (bounds.width < 380) {
|
|
||||||
bounds.width = 380
|
|
||||||
}
|
|
||||||
if (bounds.height > wasY) {
|
|
||||||
bounds.height = wasY + offset
|
|
||||||
} else if (bounds.height < 80) {
|
|
||||||
bounds.height = 80
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (x != 0) {
|
|
||||||
winX = bounds.x + x
|
|
||||||
if (winX > wasW - bounds.width + offset) {
|
|
||||||
winX = wasW - bounds.width + offset
|
|
||||||
} else if (winX < -offset) {
|
|
||||||
winX = -offset
|
|
||||||
}
|
|
||||||
bounds.x = winX
|
|
||||||
}
|
|
||||||
if (y != 0) {
|
|
||||||
winY = bounds.y + y
|
|
||||||
if (winY > wasY - bounds.height) {
|
|
||||||
winY = wasY - bounds.height
|
|
||||||
} else if (winY < -offset) {
|
|
||||||
winY = -offset
|
|
||||||
}
|
|
||||||
bounds.y = winY
|
|
||||||
}
|
|
||||||
|
|
||||||
// console.log(bounds, x, y, w, h)
|
|
||||||
global.modules.lyricWindow.setBounds(bounds)
|
|
||||||
})
|
})
|
||||||
|
|
|
@ -0,0 +1,56 @@
|
||||||
|
// 设置窗口位置、大小
|
||||||
|
let bounds
|
||||||
|
let winX
|
||||||
|
let winY
|
||||||
|
let wasW
|
||||||
|
let wasY
|
||||||
|
let offset = 8
|
||||||
|
exports.setLyricWindow = ({ x = 0, y = 0, w = 0, h = 0 }) => {
|
||||||
|
if (!global.modules.lyricWindow) return
|
||||||
|
bounds = global.modules.lyricWindow.getBounds()
|
||||||
|
wasW = global.envParams.workAreaSize.width
|
||||||
|
wasY = global.envParams.workAreaSize.height + offset
|
||||||
|
|
||||||
|
bounds.width = w
|
||||||
|
bounds.height = h
|
||||||
|
if (bounds.width > wasW - offset) {
|
||||||
|
bounds.width = wasW - offset
|
||||||
|
} else if (bounds.width < 380) {
|
||||||
|
bounds.width = 380
|
||||||
|
}
|
||||||
|
if (bounds.height > wasY) {
|
||||||
|
bounds.height = wasY + offset
|
||||||
|
} else if (bounds.height < 80) {
|
||||||
|
bounds.height = 80
|
||||||
|
}
|
||||||
|
|
||||||
|
if (global.appSetting.desktopLyric.isLockScreen) {
|
||||||
|
if (x != 0) {
|
||||||
|
winX = bounds.x + x
|
||||||
|
if (winX > wasW - bounds.width + offset) {
|
||||||
|
winX = wasW - bounds.width + offset
|
||||||
|
} else if (winX < -offset) {
|
||||||
|
winX = -offset
|
||||||
|
}
|
||||||
|
bounds.x = winX
|
||||||
|
}
|
||||||
|
if (y != 0) {
|
||||||
|
winY = bounds.y + y
|
||||||
|
if (winY > wasY - bounds.height) {
|
||||||
|
winY = wasY - bounds.height
|
||||||
|
} else if (winY < -offset) {
|
||||||
|
winY = -offset
|
||||||
|
}
|
||||||
|
bounds.y = winY
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (x != 0) {
|
||||||
|
bounds.x = bounds.x + x
|
||||||
|
}
|
||||||
|
if (y != 0) {
|
||||||
|
bounds.y = bounds.y + y
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// console.log(bounds, x, y, w, h)
|
||||||
|
global.modules.lyricWindow.setBounds(bounds)
|
||||||
|
}
|
|
@ -1,7 +1,9 @@
|
||||||
const { mainWindow: MAIN_WINDOW_EVENT_NAME } = require('../events/_name')
|
const { mainWindow: MAIN_WINDOW_EVENT_NAME } = require('../events/_name')
|
||||||
const { mainOn, NAMES: { mainWindow: ipcMainWindowNames } } = require('../../common/ipc')
|
const { mainOn, NAMES: { mainWindow: ipcMainWindowNames }, mainHandle } = require('../../common/ipc')
|
||||||
|
|
||||||
mainOn(ipcMainWindowNames.set_app_setting, (event, config) => {
|
mainOn(ipcMainWindowNames.set_app_setting, (event, config) => {
|
||||||
if (!config) return
|
if (!config) return
|
||||||
global.lx_event.common.setAppConfig(config, MAIN_WINDOW_EVENT_NAME.name)
|
global.lx_event.common.setAppConfig(config, MAIN_WINDOW_EVENT_NAME.name)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
mainHandle(ipcMainWindowNames.get_setting, async() => ({ setting: global.appSetting, version: global.appSettingVersion }))
|
|
@ -1,7 +1,11 @@
|
||||||
const { mainHandle, NAMES: { mainWindow: ipcMainWindowNames } } = require('../../common/ipc')
|
const { mainHandle, NAMES: { mainWindow: ipcMainWindowNames } } = require('../../common/ipc')
|
||||||
|
|
||||||
|
mainHandle(ipcMainWindowNames.clear_cache, async(event, options) => {
|
||||||
|
if (!global.modules.mainWindow) throw new Error('mainWindow is undefined')
|
||||||
|
return global.modules.mainWindow.webContents.session.clearCache()
|
||||||
|
})
|
||||||
|
|
||||||
mainHandle(ipcMainWindowNames.get_cache_size, async(event, options) => {
|
mainHandle(ipcMainWindowNames.get_cache_size, async(event, options) => {
|
||||||
if (!global.modules.mainWindow) throw new Error('mainWindow is undefined')
|
if (!global.modules.mainWindow) throw new Error('mainWindow is undefined')
|
||||||
return global.modules.mainWindow.webContents.session.getCacheSize()
|
return global.modules.mainWindow.webContents.session.getCacheSize()
|
||||||
})
|
})
|
||||||
|
|
|
@ -1,8 +0,0 @@
|
||||||
const { mainHandle, NAMES: { mainWindow: ipcMainWindowNames } } = require('../../common/ipc')
|
|
||||||
|
|
||||||
mainHandle(ipcMainWindowNames.clear_cache, async(event, options) => {
|
|
||||||
if (!global.modules.mainWindow) throw new Error('mainWindow is undefined')
|
|
||||||
return global.modules.mainWindow.webContents.session.clearCache()
|
|
||||||
})
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,12 @@
|
||||||
|
const Store = require('electron-store')
|
||||||
|
const { mainOn, NAMES: { mainWindow: ipcMainWindowNames }, mainHandle } = require('../../common/ipc')
|
||||||
|
|
||||||
|
|
||||||
|
const electronStore_data = new Store({
|
||||||
|
name: 'data',
|
||||||
|
})
|
||||||
|
|
||||||
|
mainHandle(ipcMainWindowNames.get_data, async(event, path) => electronStore_data.get(path))
|
||||||
|
|
||||||
|
|
||||||
|
mainOn(ipcMainWindowNames.save_data, (event, { path, data }) => electronStore_data.get(path, data))
|
|
@ -1,5 +1,10 @@
|
||||||
const { mainSend, NAMES: { mainWindow: ipcMainWindowNames }, mainOn } = require('../../common/ipc')
|
const Store = require('electron-store')
|
||||||
|
const { mainSend, NAMES: { mainWindow: ipcMainWindowNames }, mainOn, mainHandle } = require('../../common/ipc')
|
||||||
const { mainWindow: MAIN_WINDOW_EVENT_NAME, hotKey: HOT_KEY_EVENT_NAME } = require('../events/_name')
|
const { mainWindow: MAIN_WINDOW_EVENT_NAME, hotKey: HOT_KEY_EVENT_NAME } = require('../events/_name')
|
||||||
|
|
||||||
|
const electronStore_hotKey = new Store({
|
||||||
|
name: 'hotKey',
|
||||||
|
})
|
||||||
// const { registerHotkey, unRegisterHotkey } = require('../modules/hotKey/utils')
|
// const { registerHotkey, unRegisterHotkey } = require('../modules/hotKey/utils')
|
||||||
|
|
||||||
// mainHandle(ipcMainWindowNames.set_hot_key_config, async(event, { action, data }) => {
|
// mainHandle(ipcMainWindowNames.set_hot_key_config, async(event, { action, data }) => {
|
||||||
|
@ -14,6 +19,11 @@ const { mainWindow: MAIN_WINDOW_EVENT_NAME, hotKey: HOT_KEY_EVENT_NAME } = requi
|
||||||
// }
|
// }
|
||||||
// })
|
// })
|
||||||
|
|
||||||
|
mainHandle(ipcMainWindowNames.get_hot_key, async() => ({
|
||||||
|
local: electronStore_hotKey.get('local'),
|
||||||
|
global: electronStore_hotKey.get('global'),
|
||||||
|
}))
|
||||||
|
|
||||||
mainOn(ipcMainWindowNames.quit, () => global.lx_event.mainWindow.quit())
|
mainOn(ipcMainWindowNames.quit, () => global.lx_event.mainWindow.quit())
|
||||||
mainOn(ipcMainWindowNames.min_toggle, () => global.lx_event.mainWindow.toggleMinimize())
|
mainOn(ipcMainWindowNames.min_toggle, () => global.lx_event.mainWindow.toggleMinimize())
|
||||||
mainOn(ipcMainWindowNames.hide_toggle, () => global.lx_event.mainWindow.toggleHide())
|
mainOn(ipcMainWindowNames.hide_toggle, () => global.lx_event.mainWindow.toggleHide())
|
||||||
|
|
|
@ -7,15 +7,16 @@ require('./musicMeta')
|
||||||
require('./selectDir')
|
require('./selectDir')
|
||||||
require('./setWindowSize')
|
require('./setWindowSize')
|
||||||
require('./showSaveDialog')
|
require('./showSaveDialog')
|
||||||
require('./clearCache')
|
require('./cache')
|
||||||
require('./getCacheSize')
|
|
||||||
require('./setIgnoreMouseEvent')
|
require('./setIgnoreMouseEvent')
|
||||||
require('./getEnvParams')
|
require('./getEnvParams')
|
||||||
require('./setAppSetting')
|
require('./appSetting')
|
||||||
require('./setLyricInfo')
|
require('./setLyricInfo')
|
||||||
require('./hotKey')
|
require('./hotKey')
|
||||||
require('./getDataPath')
|
require('./getDataPath')
|
||||||
require('./showDialog')
|
require('./showDialog')
|
||||||
|
require('./playList')
|
||||||
|
require('./data')
|
||||||
|
|
||||||
require('./xm_verify')
|
require('./xm_verify')
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,23 @@
|
||||||
|
const Store = require('electron-store')
|
||||||
|
const { mainOn, NAMES: { mainWindow: ipcMainWindowNames }, mainHandle } = require('../../common/ipc')
|
||||||
|
|
||||||
|
|
||||||
|
const electronStore_list = new Store({
|
||||||
|
name: 'playList',
|
||||||
|
})
|
||||||
|
|
||||||
|
mainHandle(ipcMainWindowNames.get_playlist, async(event, isIgnoredError = false) => {
|
||||||
|
let electronStore_list = new Store({
|
||||||
|
name: 'playList',
|
||||||
|
clearInvalidConfig: !isIgnoredError,
|
||||||
|
})
|
||||||
|
|
||||||
|
return {
|
||||||
|
defaultList: electronStore_list.get('defaultList'),
|
||||||
|
loveList: electronStore_list.get('loveList'),
|
||||||
|
userList: electronStore_list.get('userList'),
|
||||||
|
downloadList: electronStore_list.get('downloadList'),
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
mainOn(ipcMainWindowNames.save_playlist, (event, { type, data }) => electronStore_list.set(type, data))
|
|
@ -4,7 +4,7 @@
|
||||||
transition(enter-active-class="animated-fast fadeIn" leave-active-class="animated-fast fadeOut")
|
transition(enter-active-class="animated-fast fadeIn" leave-active-class="animated-fast fadeOut")
|
||||||
.control-bar(v-show="!lrcConfig.isLock")
|
.control-bar(v-show="!lrcConfig.isLock")
|
||||||
core-control-bar(:lrcConfig="lrcConfig" :themes="themeList")
|
core-control-bar(:lrcConfig="lrcConfig" :themes="themeList")
|
||||||
core-lyric(:lrcConfig="lrcConfig")
|
core-lyric(:lrcConfig="lrcConfig" :isShowLyricTransition="isShowLyricTransition")
|
||||||
div.resize-left(@mousedown.self="handleMouseDown('left', $event)")
|
div.resize-left(@mousedown.self="handleMouseDown('left', $event)")
|
||||||
div.resize-top(@mousedown.self="handleMouseDown('top', $event)")
|
div.resize-top(@mousedown.self="handleMouseDown('top', $event)")
|
||||||
div.resize-right(@mousedown.self="handleMouseDown('right', $event)")
|
div.resize-right(@mousedown.self="handleMouseDown('right', $event)")
|
||||||
|
@ -44,6 +44,7 @@ export default {
|
||||||
isZoomActiveLrc: true,
|
isZoomActiveLrc: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
isShowLyricTransition: true,
|
||||||
themeList: [
|
themeList: [
|
||||||
{
|
{
|
||||||
id: 0,
|
id: 0,
|
||||||
|
@ -117,8 +118,9 @@ export default {
|
||||||
document.removeEventListener('mouseup', this.handleMouseUp)
|
document.removeEventListener('mouseup', this.handleMouseUp)
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
handleUpdateConfig({ config, languageId }) {
|
handleUpdateConfig({ config, languageId, isShowLyricTransition }) {
|
||||||
this.lrcConfig = config
|
this.lrcConfig = config
|
||||||
|
this.isShowLyricTransition = isShowLyricTransition
|
||||||
if (this.$i18n.locale !== languageId && languageId != null) this.$i18n.locale = languageId
|
if (this.$i18n.locale !== languageId && languageId != null) this.$i18n.locale = languageId
|
||||||
},
|
},
|
||||||
handleMouseDown(origin, event) {
|
handleMouseDown(origin, event) {
|
||||||
|
|
|
@ -27,6 +27,10 @@ export default {
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
isShowLyricTransition: {
|
||||||
|
type: Boolean,
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
|
@ -58,6 +62,10 @@ export default {
|
||||||
lyricLines: [],
|
lyricLines: [],
|
||||||
isSetedLines: false,
|
isSetedLines: false,
|
||||||
isPlay: false,
|
isPlay: false,
|
||||||
|
lyrics: {
|
||||||
|
lyric: '',
|
||||||
|
tlyric: '',
|
||||||
|
},
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
|
@ -112,6 +120,11 @@ export default {
|
||||||
},
|
},
|
||||||
immediate: true,
|
immediate: true,
|
||||||
},
|
},
|
||||||
|
isShowLyricTransition(n) {
|
||||||
|
console.log(n)
|
||||||
|
this.setLyric()
|
||||||
|
rendererSend(NAMES.winLyric.get_lyric_info, 'status')
|
||||||
|
},
|
||||||
},
|
},
|
||||||
created() {
|
created() {
|
||||||
rendererOn(NAMES.winLyric.set_lyric_info, (event, data) => this.handleSetInfo(data))
|
rendererOn(NAMES.winLyric.set_lyric_info, (event, data) => this.handleSetInfo(data))
|
||||||
|
@ -144,7 +157,9 @@ export default {
|
||||||
// console.log(type, data)
|
// console.log(type, data)
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case 'lyric':
|
case 'lyric':
|
||||||
window.lrc.setLyric(data)
|
this.lyrics.lyric = data.lrc
|
||||||
|
this.lyrics.tlyric = data.tlrc
|
||||||
|
this.setLyric()
|
||||||
break
|
break
|
||||||
case 'play':
|
case 'play':
|
||||||
this.isPlay = true
|
this.isPlay = true
|
||||||
|
@ -156,7 +171,9 @@ export default {
|
||||||
break
|
break
|
||||||
case 'info':
|
case 'info':
|
||||||
// console.log('info', data)
|
// console.log('info', data)
|
||||||
window.lrc.setLyric(data.lyric)
|
this.lyrics.lyric = data.lyric
|
||||||
|
this.lyrics.tlyric = data.tlyric
|
||||||
|
this.setLyric()
|
||||||
this.$nextTick(() => {
|
this.$nextTick(() => {
|
||||||
this.lyric.line = data.line
|
this.lyric.line = data.line
|
||||||
rendererSend(NAMES.winLyric.get_lyric_info, 'status')
|
rendererSend(NAMES.winLyric.get_lyric_info, 'status')
|
||||||
|
@ -191,7 +208,7 @@ export default {
|
||||||
}
|
}
|
||||||
if (this.lyricEvent.isStopScroll) return
|
if (this.lyricEvent.isStopScroll) return
|
||||||
let dom_p = this.dom_lines[this.lyric.line]
|
let dom_p = this.dom_lines[this.lyric.line]
|
||||||
cancelScrollFn = scrollTo(this.$refs.dom_lyric, dom_p ? (dom_p.offsetTop - this.$refs.dom_lyric.clientHeight * 0.38) : 0)
|
cancelScrollFn = scrollTo(this.$refs.dom_lyric, dom_p ? (dom_p.offsetTop - this.$refs.dom_lyric.clientHeight * 0.5 + dom_p.clientHeight / 2) : 0)
|
||||||
},
|
},
|
||||||
handleLyricMouseDown(e) {
|
handleLyricMouseDown(e) {
|
||||||
if (e.target.classList.contains(this.$style.lrcLine)) {
|
if (e.target.classList.contains(this.$style.lrcLine)) {
|
||||||
|
@ -261,6 +278,9 @@ export default {
|
||||||
close() {
|
close() {
|
||||||
rendererSend(NAMES.winLyric.close)
|
rendererSend(NAMES.winLyric.close)
|
||||||
},
|
},
|
||||||
|
setLyric() {
|
||||||
|
window.lrc.setLyric((this.isShowLyricTransition && this.lyrics.tlyric ? this.lyrics.tlyric + '\n' : '') + this.lyrics.lyric)
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -27,7 +27,7 @@ import { mapMutations, mapGetters, mapActions } from 'vuex'
|
||||||
import { rendererOn, rendererSend, rendererInvoke, NAMES } from '../common/ipc'
|
import { rendererOn, rendererSend, rendererInvoke, NAMES } from '../common/ipc'
|
||||||
import { isLinux } from '../common/utils'
|
import { isLinux } from '../common/utils'
|
||||||
import music from './utils/music'
|
import music from './utils/music'
|
||||||
import { throttle, openUrl, compareVer } from './utils'
|
import { throttle, openUrl, compareVer, getPlayList } from './utils'
|
||||||
import { base as eventBaseName } from './event/names'
|
import { base as eventBaseName } from './event/names'
|
||||||
|
|
||||||
window.ELECTRON_DISABLE_SECURITY_WARNINGS = process.env.ELECTRON_DISABLE_SECURITY_WARNINGS
|
window.ELECTRON_DISABLE_SECURITY_WARNINGS = process.env.ELECTRON_DISABLE_SECURITY_WARNINGS
|
||||||
|
@ -70,19 +70,34 @@ export default {
|
||||||
},
|
},
|
||||||
created() {
|
created() {
|
||||||
this.saveDefaultList = throttle(n => {
|
this.saveDefaultList = throttle(n => {
|
||||||
window.electronStore_list.set('defaultList', n)
|
rendererSend(NAMES.mainWindow.save_playlist, {
|
||||||
|
type: 'defaultList',
|
||||||
|
data: n,
|
||||||
|
})
|
||||||
}, 500)
|
}, 500)
|
||||||
this.saveLoveList = throttle(n => {
|
this.saveLoveList = throttle(n => {
|
||||||
window.electronStore_list.set('loveList', n)
|
rendererSend(NAMES.mainWindow.save_playlist, {
|
||||||
|
type: 'loveList',
|
||||||
|
data: n,
|
||||||
|
})
|
||||||
}, 500)
|
}, 500)
|
||||||
this.saveUserList = throttle(n => {
|
this.saveUserList = throttle(n => {
|
||||||
window.electronStore_list.set('userList', n)
|
rendererSend(NAMES.mainWindow.save_playlist, {
|
||||||
|
type: 'userList',
|
||||||
|
data: n,
|
||||||
|
})
|
||||||
}, 500)
|
}, 500)
|
||||||
this.saveDownloadList = throttle(n => {
|
this.saveDownloadList = throttle(n => {
|
||||||
window.electronStore_list.set('downloadList', n)
|
rendererSend(NAMES.mainWindow.save_playlist, {
|
||||||
|
type: 'downloadList',
|
||||||
|
data: n,
|
||||||
|
})
|
||||||
}, 1000)
|
}, 1000)
|
||||||
this.saveSearchHistoryList = throttle(n => {
|
this.saveSearchHistoryList = throttle(n => {
|
||||||
window.electronStore_data.set('searchHistoryList', n)
|
rendererSend(NAMES.mainWindow.set_data, {
|
||||||
|
path: 'searchHistoryList',
|
||||||
|
data: n,
|
||||||
|
})
|
||||||
}, 500)
|
}, 500)
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
|
@ -135,12 +150,29 @@ export default {
|
||||||
'windowSizeActive.fontSize'(n) {
|
'windowSizeActive.fontSize'(n) {
|
||||||
document.documentElement.style.fontSize = n
|
document.documentElement.style.fontSize = n
|
||||||
},
|
},
|
||||||
|
'setting.isShowAnimation': {
|
||||||
|
handler(n) {
|
||||||
|
if (n) {
|
||||||
|
if (document.body.classList.contains('disableAnimation')) {
|
||||||
|
document.body.classList.remove('disableAnimation')
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (!document.body.classList.contains('disableAnimation')) {
|
||||||
|
document.body.classList.add('disableAnimation')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
immediate: true,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
...mapActions(['getVersionInfo']),
|
...mapActions(['getVersionInfo']),
|
||||||
...mapMutations(['setNewVersion', 'setVersionModalVisible', 'setDownloadProgress', 'setSetting', 'setDesktopLyricConfig']),
|
...mapMutations(['setNewVersion', 'setVersionModalVisible', 'setDownloadProgress', 'setSetting', 'setDesktopLyricConfig']),
|
||||||
...mapMutations('list', ['initList']),
|
...mapMutations('list', ['initList']),
|
||||||
...mapMutations('download', ['updateDownloadList']),
|
...mapMutations('download', ['updateDownloadList']),
|
||||||
|
...mapMutations('search', {
|
||||||
|
setSearchHistoryList: 'setHistory',
|
||||||
|
}),
|
||||||
init() {
|
init() {
|
||||||
document.documentElement.style.fontSize = this.windowSizeActive.fontSize
|
document.documentElement.style.fontSize = this.windowSizeActive.fontSize
|
||||||
|
|
||||||
|
@ -229,19 +261,23 @@ export default {
|
||||||
},
|
},
|
||||||
|
|
||||||
initData() { // 初始化数据
|
initData() { // 初始化数据
|
||||||
this.initPlayList() // 初始化播放列表
|
this.initLocalList() // 初始化播放列表
|
||||||
this.initDownloadList() // 初始化下载列表
|
// this.initDownloadList() // 初始化下载列表
|
||||||
|
this.initSearchHistoryList() // 初始化搜索历史列表
|
||||||
},
|
},
|
||||||
initPlayList() {
|
initLocalList() {
|
||||||
let defaultList = window.electronStore_list.get('defaultList') || this.defaultList
|
getPlayList().then(({ defaultList, loveList, userList, downloadList }) => {
|
||||||
let loveList = window.electronStore_list.get('loveList') || this.loveList
|
if (!defaultList) defaultList = this.defaultList
|
||||||
let userList = window.electronStore_list.get('userList') || this.userList
|
if (!loveList) loveList = this.loveList
|
||||||
|
if (!userList) userList = this.userList
|
||||||
|
|
||||||
if (!defaultList.list) defaultList.list = []
|
if (!defaultList.list) defaultList.list = []
|
||||||
if (!loveList.list) loveList.list = []
|
if (!loveList.list) loveList.list = []
|
||||||
this.initList({ defaultList, loveList, userList })
|
this.initList({ defaultList, loveList, userList })
|
||||||
|
this.initDownloadList(downloadList) // 初始化下载列表
|
||||||
|
})
|
||||||
},
|
},
|
||||||
initDownloadList() {
|
initDownloadList(downloadList) {
|
||||||
let downloadList = window.electronStore_list.get('downloadList')
|
|
||||||
if (downloadList) {
|
if (downloadList) {
|
||||||
downloadList.forEach(item => {
|
downloadList.forEach(item => {
|
||||||
if (item.status == this.downloadStatus.RUN || item.status == this.downloadStatus.WAITING) {
|
if (item.status == this.downloadStatus.RUN || item.status == this.downloadStatus.WAITING) {
|
||||||
|
@ -252,6 +288,16 @@ export default {
|
||||||
this.updateDownloadList(downloadList)
|
this.updateDownloadList(downloadList)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
initSearchHistoryList() {
|
||||||
|
rendererInvoke(NAMES.mainWindow.get_data, 'searchHistoryList').then(historyList => {
|
||||||
|
if (historyList == null) {
|
||||||
|
historyList = []
|
||||||
|
rendererInvoke(NAMES.mainWindow.set_data, { path: 'searchHistoryList', data: historyList })
|
||||||
|
} else {
|
||||||
|
this.setSearchHistoryList(historyList)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
showUpdateModal() {
|
showUpdateModal() {
|
||||||
(this.version.newVersion && this.version.newVersion.history
|
(this.version.newVersion && this.version.newVersion.history
|
||||||
? Promise.resolve(this.version.newVersion)
|
? Promise.resolve(this.version.newVersion)
|
||||||
|
@ -354,6 +400,11 @@ body {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.disableAnimation * {
|
||||||
|
transition: none !important;
|
||||||
|
animation: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
.transparent {
|
.transparent {
|
||||||
padding: @shadow-app;
|
padding: @shadow-app;
|
||||||
#container {
|
#container {
|
||||||
|
|
|
@ -147,7 +147,7 @@ export default {
|
||||||
height: @control-btn-width;
|
height: @control-btn-width;
|
||||||
background: none;
|
background: none;
|
||||||
border: none;
|
border: none;
|
||||||
// display: flex;
|
display: flex;
|
||||||
// justify-content: center;
|
// justify-content: center;
|
||||||
// align-items: center;
|
// align-items: center;
|
||||||
outline: none;
|
outline: none;
|
||||||
|
|
|
@ -165,5 +165,46 @@ svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/19
|
||||||
g#icon-window-close(fill='currentColor')
|
g#icon-window-close(fill='currentColor')
|
||||||
//- 0 0 24 24
|
//- 0 0 24 24
|
||||||
path(d="M20 6.91L17.09 4L12 9.09L6.91 4L4 6.91L9.09 12L4 17.09L6.91 20L12 14.91L17.09 20L20 17.09L14.91 12L20 6.91Z")
|
path(d="M20 6.91L17.09 4L12 9.09L6.91 4L4 6.91L9.09 12L4 17.09L6.91 20L12 14.91L17.09 20L20 17.09L14.91 12L20 6.91Z")
|
||||||
|
|
||||||
|
|
||||||
|
g#icon-list-loop(fill='currentColor')
|
||||||
|
// 0 0 24 24
|
||||||
|
path(d="M0 0h24v24H0z" fill="none")
|
||||||
|
path(d='M7 7h10v3l4-4-4-4v3H5v6h2V7zm10 10H7v-3l-4 4 4 4v-3h12v-6h-2v4z')
|
||||||
|
|
||||||
|
g#icon-single-loop(fill='currentColor')
|
||||||
|
// 0 0 24 24
|
||||||
|
path(d="M0 0h24v24H0z" fill="none")
|
||||||
|
path(d="M7 7h10v3l4-4-4-4v3H5v6h2V7zm10 10H7v-3l-4 4 4 4v-3h12v-6h-2v4zm-4-2V9h-1l-2 1v1h1.5v4H13z")
|
||||||
|
|
||||||
|
g#icon-list-random(fill='currentColor')
|
||||||
|
// 0 0 24 24
|
||||||
|
path(d="M0 0h24v24H0z" fill="none")
|
||||||
|
path(d="M10.59 9.17L5.41 4 4 5.41l5.17 5.17 1.42-1.41zM14.5 4l2.04 2.04L4 18.59 5.41 20 17.96 7.46 20 9.5V4h-5.5zm.33 9.41l-1.41 1.41 3.13 3.13L14.5 20H20v-5.5l-2.04 2.04-3.13-3.13z")
|
||||||
|
|
||||||
|
g#icon-list-order(fill='currentColor')
|
||||||
|
// 0 0 24 24
|
||||||
|
path(d="M0 0h24v24H0z" fill="none")
|
||||||
|
path(d="M4 10h12v2H4zm0-4h12v2H4zm0 8h8v2H4zm10 0v6l5-3z")
|
||||||
|
|
||||||
|
g#icon-single(fill='currentColor')
|
||||||
|
//- 0 0 24 24
|
||||||
|
path(d='M18,6.8v9.9h-2.1v-5.3c0-0.8,0-1.2,0-1.4c0-0.2-0.1-0.3-0.3-0.4c-0.1-0.1-0.5-0.1-1-0.1h-0.2V8.4c1-0.3,1.8-0.8,2.3-1.6H18z')
|
||||||
|
polygon(points='6,18 14.5,12 6,6 ')
|
||||||
|
|
||||||
|
g#icon-desktop-lyric-on(fill='currentColor')
|
||||||
|
//- 0 0 512 512
|
||||||
|
path(d='M162.64,335.554H81.017V152.1H55.143V359.7h107.5V335.554ZM288.807,291.4a117.012,117.012,0,0,0-11.222-20.092,29.568,29.568,0,0,0-13.365-10.562,54.323,54.323,0,0,0,17.225-7.7,50.6,50.6,0,0,0,12.579-12.23,52.253,52.253,0,0,0,7.648-15.724,64.034,64.034,0,0,0,2.573-18.345,67.706,67.706,0,0,0-3.5-22.316,42.831,42.831,0,0,0-11.007-17.233,51.262,51.262,0,0,0-19.227-11.118q-11.722-3.969-28.018-3.971H192.459V359.7H217.9v-92.44h12.008a37.092,37.092,0,0,1,10.078,1.271,24.136,24.136,0,0,1,8.291,4.209,32.591,32.591,0,0,1,7.076,7.941A76.14,76.14,0,0,1,261.79,293.3L289.664,359.7H318.4Zm-13.938-67.106a30.684,30.684,0,0,1-7.862,11.118,33.47,33.47,0,0,1-12.293,6.83,53.247,53.247,0,0,1-16.225,2.3H217.9V175.928H241.92q17.01,0,26.374,8.259t9.363,24.937A38.647,38.647,0,0,1,274.869,224.292ZM456.714,325.229a100.649,100.649,0,0,1-21.156,8.657,84.575,84.575,0,0,1-23.015,3.1q-28.448,0-43.17-20.489t-14.724-60.833a133.161,133.161,0,0,1,4-34.228q4-15.009,11.436-25.413a50.462,50.462,0,0,1,18.083-15.883,51.244,51.244,0,0,1,23.8-5.48,82.427,82.427,0,0,1,23.73,3.256,89.743,89.743,0,0,1,21.013,9.451v-27.8a94.1,94.1,0,0,0-21.442-7.544,112.019,112.019,0,0,0-24.158-2.462q-19.158,0-34.594,7.624a74.755,74.755,0,0,0-26.3,21.68q-10.865,14.056-16.724,34.228T327.632,258.2q0,51.461,21.227,77.748t60.825,26.286a111.234,111.234,0,0,0,47.03-10.324V325.229Z')
|
||||||
|
|
||||||
|
g#icon-desktop-lyric-off(fill='currentColor')
|
||||||
|
//- 0 0 512 512
|
||||||
|
path(d='M162.64,335.554H81.017V152.1H55.143V359.7h107.5V335.554ZM288.807,291.4a117.012,117.012,0,0,0-11.222-20.092,29.568,29.568,0,0,0-13.365-10.562,54.323,54.323,0,0,0,17.225-7.7,50.6,50.6,0,0,0,12.579-12.23,52.253,52.253,0,0,0,7.648-15.724,64.034,64.034,0,0,0,2.573-18.345,67.706,67.706,0,0,0-3.5-22.316,42.831,42.831,0,0,0-11.007-17.233,51.262,51.262,0,0,0-19.227-11.118q-11.722-3.969-28.018-3.971H192.459V359.7H217.9v-92.44h12.008a37.092,37.092,0,0,1,10.078,1.271,24.136,24.136,0,0,1,8.291,4.209,32.591,32.591,0,0,1,7.076,7.941A76.14,76.14,0,0,1,261.79,293.3L289.664,359.7H318.4Zm-13.938-67.106a30.684,30.684,0,0,1-7.862,11.118,33.47,33.47,0,0,1-12.293,6.83,53.247,53.247,0,0,1-16.225,2.3H217.9V175.928H241.92q17.01,0,26.374,8.259t9.363,24.937A38.647,38.647,0,0,1,274.869,224.292ZM456.714,325.229a100.649,100.649,0,0,1-21.156,8.657,84.575,84.575,0,0,1-23.015,3.1q-28.448,0-43.17-20.489t-14.724-60.833a133.161,133.161,0,0,1,4-34.228q4-15.009,11.436-25.413a50.462,50.462,0,0,1,18.083-15.883,51.244,51.244,0,0,1,23.8-5.48,82.427,82.427,0,0,1,23.73,3.256,89.743,89.743,0,0,1,21.013,9.451v-27.8a94.1,94.1,0,0,0-21.442-7.544,112.019,112.019,0,0,0-24.158-2.462q-19.158,0-34.594,7.624a74.755,74.755,0,0,0-26.3,21.68q-10.865,14.056-16.724,34.228T327.632,258.2q0,51.461,21.227,77.748t60.825,26.286a111.234,111.234,0,0,0,47.03-10.324V325.229Z')
|
||||||
|
path(d='M69.148,97.826l17.7-17.651,333,334-17.7,17.652Z')
|
||||||
|
|
||||||
|
g#icon-add-2(fill='currentColor')
|
||||||
|
//- 0 0 1024 1024
|
||||||
|
path(d='M256,170s-62.469-76.808-141-24C44.762,222.824,84.909,325.08,256,415c21.339-8.361,44-17,44-17,19-6.392,28.155,20.742,16,26-27.589,11.935,5.974-4.141-60,28C-35.524,313.85,43.993,149.031,95,117c86.8-65.89,162,10,162,10s58.158-60.523,140-23c104.032,58.528,64,161.9,45,196-9.152,15.154-39.559-4.159-32-16,20.34-37.888,45.522-107.349-25-150C314.919,103.92,256,170,256,170Z')
|
||||||
|
path(d='M383,368c-8.1.01-24.77-.155-40,0-15.713.16-15.282,34.964,0,35,15.1,0.035,40,0,40,0s-0.068,42.8,0,48c0.208,15.961,32.261,15.791,32-1-0.072-4.649,0-47,0-47s38.008-.031,43,0c15.732,0.046,14.947-33.98-1-34-4.884.093-42,0-42,0s-0.053-28.341,0-46c0.046-15.189-32.028-15.512-32,0C383.027,337.74,382.782,365.139,383,368Z')
|
||||||
|
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|
|
@ -9,11 +9,31 @@ div(:class="$style.player")
|
||||||
div(:class="$style.column1")
|
div(:class="$style.column1")
|
||||||
div(:class="$style.container")
|
div(:class="$style.container")
|
||||||
div(:class="$style.title" @click="handleCopy(title)" :title="title + $t('core.player.copy_title')") {{title}}
|
div(:class="$style.title" @click="handleCopy(title)" :title="title + $t('core.player.copy_title')") {{title}}
|
||||||
|
div(:class="$style.controlBtn")
|
||||||
|
|
||||||
div(:class="$style.volumeContent")
|
div(:class="$style.volumeContent")
|
||||||
div(:class="$style.volume")
|
div(:class="[$style.volume, setting.player.isMute ? $style.muted : null]")
|
||||||
div(:class="$style.volumeBar" :style="{ transform: `scaleX(${volume || 0})` }")
|
div(:class="$style.volumeBar" :style="{ transform: `scaleX(${volume || 0})` }")
|
||||||
div(:class="$style.volumeMask" @mousedown="handleVolumeMsDown" ref="dom_volumeMask" :title="`${$t('core.player.volume')}${parseInt(volume * 100)}%`")
|
div(:class="$style.volumeMask" @mousedown="handleVolumeMsDown" ref="dom_volumeMask" :title="`${$t('core.player.volume')}${parseInt(volume * 100)}%`")
|
||||||
|
div(:class="$style.titleBtn" @click='toggleDesktopLyric' :title="setting.desktopLyric.enable ? $t('core.player.desktop_lyric_off') : $t('core.player.desktop_lyric_on')")
|
||||||
|
svg(v-if="setting.desktopLyric.enable" version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 512 512' space='preserve')
|
||||||
|
use(xlink:href='#icon-desktop-lyric-off')
|
||||||
|
svg(v-else version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 512 512' space='preserve')
|
||||||
|
use(xlink:href='#icon-desktop-lyric-on')
|
||||||
|
div(:class="$style.titleBtn" @click='toggleNextPlayMode' :title="nextTogglePlayName")
|
||||||
|
svg(v-if="setting.player.togglePlayMethod == 'listLoop'" version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='80%' viewBox='0 0 24 24' space='preserve')
|
||||||
|
use(xlink:href='#icon-list-loop')
|
||||||
|
svg(v-else-if="setting.player.togglePlayMethod == 'random'" version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' width='100%' viewBox='0 0 24 24' space='preserve')
|
||||||
|
use(xlink:href='#icon-list-random')
|
||||||
|
svg(v-else-if="setting.player.togglePlayMethod == 'list'" version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' width='120%' viewBox='0 0 24 24' space='preserve')
|
||||||
|
use(xlink:href='#icon-list-order')
|
||||||
|
svg(v-else-if="setting.player.togglePlayMethod == 'singleLoop'" version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' width='100%' viewBox='0 0 24 24' space='preserve')
|
||||||
|
use(xlink:href='#icon-single-loop')
|
||||||
|
svg(v-else version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' width='120%' viewBox='0 0 24 24' space='preserve')
|
||||||
|
use(xlink:href='#icon-single')
|
||||||
|
div(:class="$style.titleBtn" @click='addMusicTo' :title="$t('core.player.add_music_to')")
|
||||||
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='80%' viewBox='0 0 512 512' space='preserve')
|
||||||
|
use(xlink:href='#icon-add-2')
|
||||||
//- div(:class="$style.playBtn" @click='handleNext' title="音量")
|
//- div(:class="$style.playBtn" @click='handleNext' title="音量")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 291.063 291.064' space='preserve')
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 291.063 291.064' space='preserve')
|
||||||
use(xlink:href='#icon-sound')
|
use(xlink:href='#icon-sound')
|
||||||
|
@ -47,6 +67,8 @@ div(:class="$style.player")
|
||||||
:lyric="lyric" :list="list" :listId="listId"
|
:lyric="lyric" :list="list" :listId="listId"
|
||||||
:playInfo="{ nowPlayTimeStr, maxPlayTimeStr, progress, nowPlayTime, status }"
|
:playInfo="{ nowPlayTimeStr, maxPlayTimeStr, progress, nowPlayTime, status }"
|
||||||
:isPlay="isPlay" @action="handlePlayDetailAction")
|
:isPlay="isPlay" @action="handlePlayDetailAction")
|
||||||
|
|
||||||
|
material-list-add-modal(:show="isShowAddMusicTo" :musicInfo="listId == 'download' ? targetSong.musicInfo : targetSong" @close="isShowAddMusicTo = false")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' style="display: none;")
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' style="display: none;")
|
||||||
defs
|
defs
|
||||||
g(:id="$style.iconPic")
|
g(:id="$style.iconPic")
|
||||||
|
@ -72,6 +94,13 @@ import path from 'path'
|
||||||
|
|
||||||
let audio
|
let audio
|
||||||
|
|
||||||
|
const playNextModes = [
|
||||||
|
'listLoop',
|
||||||
|
'random',
|
||||||
|
'list',
|
||||||
|
'singleLoop',
|
||||||
|
]
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
|
@ -112,11 +141,12 @@ export default {
|
||||||
timeout: null,
|
timeout: null,
|
||||||
playTime: 0,
|
playTime: 0,
|
||||||
},
|
},
|
||||||
|
isShowAddMusicTo: false,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapGetters(['setting']),
|
...mapGetters(['setting']),
|
||||||
...mapGetters('player', ['list', 'playIndex', 'changePlay', 'listId', 'isShowPlayerDetail']),
|
...mapGetters('player', ['list', 'playIndex', 'changePlay', 'listId', 'isShowPlayerDetail', 'playedList']),
|
||||||
// pic() {
|
// pic() {
|
||||||
// return this.musicInfo.img ? this.musicInfo.img : ''
|
// return this.musicInfo.img ? this.musicInfo.img : ''
|
||||||
// },
|
// },
|
||||||
|
@ -134,6 +164,15 @@ export default {
|
||||||
progress() {
|
progress() {
|
||||||
return this.nowPlayTime / this.maxPlayTime || 0
|
return this.nowPlayTime / this.maxPlayTime || 0
|
||||||
},
|
},
|
||||||
|
nextTogglePlayName() {
|
||||||
|
switch (this.setting.player.togglePlayMethod) {
|
||||||
|
case 'listLoop': return this.$t('core.player.play_toggle_mode_list_loop')
|
||||||
|
case 'random': return this.$t('core.player.play_toggle_mode_random')
|
||||||
|
case 'singleLoop': return this.$t('core.player.play_toggle_mode_single_loop')
|
||||||
|
case 'list': return this.$t('core.player.play_toggle_mode_list')
|
||||||
|
default: return this.$t('core.player.play_toggle_mode_off')
|
||||||
|
}
|
||||||
|
},
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
this.init()
|
this.init()
|
||||||
|
@ -153,6 +192,7 @@ export default {
|
||||||
name: this.musicInfo.name,
|
name: this.musicInfo.name,
|
||||||
album: this.musicInfo.album,
|
album: this.musicInfo.album,
|
||||||
lyric: this.musicInfo.lrc,
|
lyric: this.musicInfo.lrc,
|
||||||
|
tlyric: this.musicInfo.tlrc,
|
||||||
isPlay: this.isPlay,
|
isPlay: this.isPlay,
|
||||||
line: this.lyric.line,
|
line: this.lyric.line,
|
||||||
played_time: audio.currentTime * 1000,
|
played_time: audio.currentTime * 1000,
|
||||||
|
@ -194,6 +234,8 @@ export default {
|
||||||
},
|
},
|
||||||
'setting.player.togglePlayMethod'(n) {
|
'setting.player.togglePlayMethod'(n) {
|
||||||
audio.loop = n === 'singleLoop'
|
audio.loop = n === 'singleLoop'
|
||||||
|
if (this.playedList.length) this.clearPlayedList()
|
||||||
|
if (n == 'random' && this.playIndex > -1) this.setPlayedList(this.list[this.playIndex])
|
||||||
},
|
},
|
||||||
'setting.player.isMute'(n) {
|
'setting.player.isMute'(n) {
|
||||||
audio.muted = n
|
audio.muted = n
|
||||||
|
@ -201,6 +243,9 @@ export default {
|
||||||
'setting.player.mediaDeviceId'(n) {
|
'setting.player.mediaDeviceId'(n) {
|
||||||
this.setMediaDevice()
|
this.setMediaDevice()
|
||||||
},
|
},
|
||||||
|
'setting.player.isShowLyricTransition'() {
|
||||||
|
this.setLyric()
|
||||||
|
},
|
||||||
async list(n, o) {
|
async list(n, o) {
|
||||||
if (n === o && this.musicInfo.songmid) {
|
if (n === o && this.musicInfo.songmid) {
|
||||||
let index = this.listId == 'download'
|
let index = this.listId == 'download'
|
||||||
|
@ -238,8 +283,11 @@ export default {
|
||||||
'fixPlayIndex',
|
'fixPlayIndex',
|
||||||
'resetChangePlay',
|
'resetChangePlay',
|
||||||
'visiblePlayerDetail',
|
'visiblePlayerDetail',
|
||||||
|
'clearPlayedList',
|
||||||
|
'setPlayedList',
|
||||||
|
'removePlayedList',
|
||||||
]),
|
]),
|
||||||
...mapMutations(['setVolume']),
|
...mapMutations(['setVolume', 'setPlayNextMode', 'setVisibleDesktopLyric']),
|
||||||
...mapMutations('list', ['updateMusicInfo']),
|
...mapMutations('list', ['updateMusicInfo']),
|
||||||
...mapMutations(['setMediaDeviceId']),
|
...mapMutations(['setMediaDeviceId']),
|
||||||
handleRegisterEvent(action) {
|
handleRegisterEvent(action) {
|
||||||
|
@ -360,7 +408,7 @@ export default {
|
||||||
this.lyric.lines = lines
|
this.lyric.lines = lines
|
||||||
this.lyric.line = 0
|
this.lyric.line = 0
|
||||||
},
|
},
|
||||||
offset: 100,
|
offset: 80,
|
||||||
})
|
})
|
||||||
|
|
||||||
this.handleRegisterEvent('on')
|
this.handleRegisterEvent('on')
|
||||||
|
@ -369,6 +417,7 @@ export default {
|
||||||
console.log('play', this.playIndex)
|
console.log('play', this.playIndex)
|
||||||
this.checkDelayNextTimeout()
|
this.checkDelayNextTimeout()
|
||||||
let targetSong = this.targetSong = this.list[this.playIndex]
|
let targetSong = this.targetSong = this.list[this.playIndex]
|
||||||
|
if (this.setting.player.togglePlayMethod == 'random') this.setPlayedList(targetSong)
|
||||||
this.retryNum = 0
|
this.retryNum = 0
|
||||||
this.audioErrorTime = 0
|
this.audioErrorTime = 0
|
||||||
|
|
||||||
|
@ -420,20 +469,51 @@ export default {
|
||||||
async filterList() {
|
async filterList() {
|
||||||
// if (this.list.listName === null) return
|
// if (this.list.listName === null) return
|
||||||
let list
|
let list
|
||||||
|
let playedList = [...this.playedList]
|
||||||
if (this.listId == 'download') {
|
if (this.listId == 'download') {
|
||||||
list = []
|
list = []
|
||||||
for (const item of this.list) {
|
for (const item of this.list) {
|
||||||
const filePath = path.join(this.setting.download.savePath, item.fileName)
|
const filePath = path.join(this.setting.download.savePath, item.fileName)
|
||||||
if (!await checkPath(filePath) || !item.isComplate || /\.ape$/.test(filePath)) continue
|
if (!await checkPath(filePath) || !item.isComplate || /\.ape$/.test(filePath)) continue
|
||||||
|
|
||||||
|
let index = playedList.indexOf(item)
|
||||||
|
if (index > -1) {
|
||||||
|
playedList.splice(index, 1)
|
||||||
|
continue
|
||||||
|
}
|
||||||
list.push(item)
|
list.push(item)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
list = this.list.filter(s => this.assertApiSupport(s.source))
|
list = this.list.filter(s => {
|
||||||
|
let index = playedList.indexOf(s)
|
||||||
|
if (index > -1) {
|
||||||
|
playedList.splice(index, 1)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return this.assertApiSupport(s.source)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
if (!list.length && this.playedList.length) {
|
||||||
|
this.clearPlayedList()
|
||||||
|
return this.filterList()
|
||||||
}
|
}
|
||||||
return list
|
return list
|
||||||
},
|
},
|
||||||
async handlePrev() {
|
async handlePrev() {
|
||||||
// console.log(playIndex)
|
// console.log(playIndex)
|
||||||
|
if (this.setting.player.togglePlayMethod == 'random' && this.playedList.length) {
|
||||||
|
let index = this.playedList.indexOf(this.targetSong)
|
||||||
|
index -= 1
|
||||||
|
if (index > -1) {
|
||||||
|
let listIndex = this.list.indexOf(this.playedList[index])
|
||||||
|
if (listIndex < 0) {
|
||||||
|
this.removePlayedList(index)
|
||||||
|
return this.handlePrev()
|
||||||
|
}
|
||||||
|
this.setPlayIndex(listIndex)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
let list = await this.filterList()
|
let list = await this.filterList()
|
||||||
if (!list.length) return this.setPlayIndex(-1)
|
if (!list.length) return this.setPlayIndex(-1)
|
||||||
let playIndex = list.indexOf(this.list[this.playIndex])
|
let playIndex = list.indexOf(this.list[this.playIndex])
|
||||||
|
@ -455,6 +535,20 @@ export default {
|
||||||
},
|
},
|
||||||
async handleNext() {
|
async handleNext() {
|
||||||
// if (this.list.listName === null) return
|
// if (this.list.listName === null) return
|
||||||
|
// eslint-disable-next-line no-debugger
|
||||||
|
if (this.setting.player.togglePlayMethod == 'random' && this.playedList.length) {
|
||||||
|
let index = this.playedList.indexOf(this.targetSong)
|
||||||
|
index += 1
|
||||||
|
if (index < this.playedList.length) {
|
||||||
|
let listIndex = this.list.indexOf(this.playedList[index])
|
||||||
|
if (listIndex < 0) {
|
||||||
|
this.removePlayedList(index)
|
||||||
|
return this.handleNext()
|
||||||
|
}
|
||||||
|
this.setPlayIndex(listIndex)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
let list = await this.filterList()
|
let list = await this.filterList()
|
||||||
if (!list.length) return this.setPlayIndex(-1)
|
if (!list.length) return this.setPlayIndex(-1)
|
||||||
let playIndex = list.indexOf(this.list[this.playIndex])
|
let playIndex = list.indexOf(this.list[this.playIndex])
|
||||||
|
@ -561,25 +655,21 @@ export default {
|
||||||
},
|
},
|
||||||
setLrc(targetSong) {
|
setLrc(targetSong) {
|
||||||
this.musicInfo.lrc = targetSong.lrc
|
this.musicInfo.lrc = targetSong.lrc
|
||||||
|
this.musicInfo.tlrc = targetSong.tlrc
|
||||||
|
|
||||||
let lrcP = this.musicInfo.lrc
|
let lrcP = this.musicInfo.lrc && this.musicInfo.tlrc != null
|
||||||
? Promise.resolve()
|
? Promise.resolve()
|
||||||
: this.getLrc(targetSong).then(() => {
|
: this.getLrc(targetSong).then(() => {
|
||||||
this.musicInfo.lrc = targetSong.lrc
|
this.musicInfo.lrc = targetSong.lrc
|
||||||
|
this.musicInfo.tlrc = targetSong.tlrc
|
||||||
})
|
})
|
||||||
|
|
||||||
lrcP
|
lrcP
|
||||||
.then(() => {
|
|
||||||
window.lrc.setLyric(this.musicInfo.lrc)
|
|
||||||
this.handleUpdateWinLyricInfo('lyric', this.musicInfo.lrc)
|
|
||||||
if (this.isPlay && (this.musicInfo.url || this.listId == 'download')) {
|
|
||||||
window.lrc.play(audio.currentTime * 1000)
|
|
||||||
this.handleUpdateWinLyricInfo('play', audio.currentTime * 1000)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.catch(() => {
|
.catch(() => {
|
||||||
this.handleUpdateWinLyricInfo('lyric', this.musicInfo.lrc)
|
|
||||||
this.status = this.statusText = this.$t('core.player.lyric_error')
|
this.status = this.statusText = this.$t('core.player.lyric_error')
|
||||||
|
}).finally(() => {
|
||||||
|
this.setLyric()
|
||||||
|
this.handleUpdateWinLyricInfo('lyric', { lrc: this.musicInfo.lrc, tlrc: this.musicInfo.tlrc })
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
handleRemoveMusic() {
|
handleRemoveMusic() {
|
||||||
|
@ -623,8 +713,9 @@ export default {
|
||||||
|
|
||||||
this.volume = val
|
this.volume = val
|
||||||
this.volumeEvent.msDownVolume = val
|
this.volumeEvent.msDownVolume = val
|
||||||
// console.log(val)
|
|
||||||
if (audio) audio.volume = this.volume
|
if (audio) audio.volume = this.volume
|
||||||
|
|
||||||
|
if (this.setting.player.isMute) this.setVolume(false)
|
||||||
},
|
},
|
||||||
handleSetVolumeMute() {
|
handleSetVolumeMute() {
|
||||||
audio.muted = !audio.muted
|
audio.muted = !audio.muted
|
||||||
|
@ -654,7 +745,7 @@ export default {
|
||||||
this.setProgressWidth()
|
this.setProgressWidth()
|
||||||
},
|
},
|
||||||
handleToMusicLocation() {
|
handleToMusicLocation() {
|
||||||
if (this.listId == 'download') return
|
if (!this.listId || this.listId == 'download') return
|
||||||
if (this.playIndex == -1) return
|
if (this.playIndex == -1) return
|
||||||
this.$router.push({
|
this.$router.push({
|
||||||
path: 'list',
|
path: 'list',
|
||||||
|
@ -766,6 +857,25 @@ export default {
|
||||||
info,
|
info,
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
setLyric() {
|
||||||
|
window.lrc.setLyric((this.setting.player.isShowLyricTransition && this.musicInfo.tlrc ? this.musicInfo.tlrc + '\n' : '') + this.musicInfo.lrc)
|
||||||
|
if (this.isPlay && (this.musicInfo.url || this.listId == 'download')) {
|
||||||
|
window.lrc.play(audio.currentTime * 1000)
|
||||||
|
this.handleUpdateWinLyricInfo('play', audio.currentTime * 1000)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
toggleDesktopLyric() {
|
||||||
|
this.setVisibleDesktopLyric(!this.setting.desktopLyric.enable)
|
||||||
|
},
|
||||||
|
toggleNextPlayMode() {
|
||||||
|
let index = playNextModes.indexOf(this.setting.player.togglePlayMethod)
|
||||||
|
if (++index >= playNextModes.length) index = -1
|
||||||
|
this.setPlayNextMode(playNextModes[index] || '')
|
||||||
|
},
|
||||||
|
addMusicTo() {
|
||||||
|
if (!this.musicInfo.songmid) return
|
||||||
|
this.isShowAddMusicTo = true
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
@ -855,17 +965,26 @@ export default {
|
||||||
}
|
}
|
||||||
|
|
||||||
.title {
|
.title {
|
||||||
flex: 1 1 0;
|
flex: 0 1 auto;
|
||||||
width: 0;
|
min-width: 0;
|
||||||
padding-right: 5px;
|
padding-right: 5px;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
line-height: 18px;
|
line-height: 18px;
|
||||||
.mixin-ellipsis-1;
|
.mixin-ellipsis-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.controlBtn {
|
||||||
|
flex: none;
|
||||||
|
display: flex;
|
||||||
|
flex-flow: row nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
.volume-content {
|
.volume-content {
|
||||||
|
flex: none;
|
||||||
position: relative;
|
position: relative;
|
||||||
width: 80px;
|
width: 80px;
|
||||||
|
margin-right: 10px;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
opacity: .5;
|
opacity: .5;
|
||||||
|
@ -882,13 +1001,17 @@ export default {
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
// overflow: hidden;
|
// overflow: hidden;
|
||||||
transition: @transition-theme;
|
transition: @transition-theme;
|
||||||
transition-property: background-color;
|
transition-property: background-color, opacity;
|
||||||
background-color: @color-player-progress-bar1;
|
background-color: @color-player-progress-bar1;
|
||||||
// background-color: #f5f5f5;
|
// background-color: #f5f5f5;
|
||||||
position: relative;
|
position: relative;
|
||||||
border-radius: @radius-progress-border;
|
border-radius: @radius-progress-border;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.muted {
|
||||||
|
opacity: .5;
|
||||||
|
}
|
||||||
|
|
||||||
.volume-bar {
|
.volume-bar {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
left: 0;
|
left: 0;
|
||||||
|
@ -905,6 +1028,7 @@ export default {
|
||||||
box-shadow: 0 0 2px rgba(0, 0, 0, 0.2);
|
box-shadow: 0 0 2px rgba(0, 0, 0, 0.2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
.volume-mask {
|
.volume-mask {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 0;
|
top: 0;
|
||||||
|
@ -913,6 +1037,32 @@ export default {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.titleBtn {
|
||||||
|
flex: none;
|
||||||
|
margin-left: 5px;
|
||||||
|
height: 100%;
|
||||||
|
width: 20px;
|
||||||
|
color: @color-theme;
|
||||||
|
display: flex;
|
||||||
|
flex-flow: column nowrap;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
transition: opacity 0.2s ease;
|
||||||
|
opacity: .5;
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
svg {
|
||||||
|
filter: drop-shadow(0 0 1px rgba(0, 0, 0, 0.2));
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
&:active {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.play-btn {
|
.play-btn {
|
||||||
+ .play-btn {
|
+ .play-btn {
|
||||||
margin-left: 15px;
|
margin-left: 15px;
|
||||||
|
@ -1041,6 +1191,9 @@ each(@themes, {
|
||||||
// border-color: ~'@{color-@{value}-theme_2-background_1}';
|
// border-color: ~'@{color-@{value}-theme_2-background_1}';
|
||||||
// }
|
// }
|
||||||
}
|
}
|
||||||
|
.titleBtn {
|
||||||
|
color: ~'@{color-@{value}-theme}';
|
||||||
|
}
|
||||||
.play-btn {
|
.play-btn {
|
||||||
color: ~'@{color-@{value}-theme}';
|
color: ~'@{color-@{value}-theme}';
|
||||||
svg {
|
svg {
|
||||||
|
|
|
@ -526,7 +526,7 @@ export default {
|
||||||
padding: 8px 0;
|
padding: 8px 0;
|
||||||
line-height: 1.2;
|
line-height: 1.2;
|
||||||
overflow-wrap: break-word;
|
overflow-wrap: break-word;
|
||||||
transition: @transition-theme;
|
transition: @transition-theme !important;
|
||||||
transition-property: color, font-size;
|
transition-property: color, font-size;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -183,6 +183,7 @@ export default {
|
||||||
}
|
}
|
||||||
|
|
||||||
button {
|
button {
|
||||||
|
display: flex;
|
||||||
position: relative;
|
position: relative;
|
||||||
width: @control-btn-width;
|
width: @control-btn-width;
|
||||||
height: @control-btn-width;
|
height: @control-btn-width;
|
||||||
|
|
|
@ -1,123 +0,0 @@
|
||||||
<template lang="pug">
|
|
||||||
transition(enter-active-class="animated-fast zoomIn" leave-active-class="animated zoomOut")
|
|
||||||
div(:class="$style.btns" v-show="show")
|
|
||||||
button(type="button" v-if="playBtn" :title="$t('material.flow_btn.play')" @click.stop="handleClick('play')")
|
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 287.386 287.386' space='preserve')
|
|
||||||
use(xlink:href='#icon-testPlay')
|
|
||||||
button(type="button" v-if="addBtn" :title="$t('material.flow_btn.add')" @click.stop="handleClick('add')")
|
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 42 42' space='preserve')
|
|
||||||
use(xlink:href='#icon-addTo')
|
|
||||||
button(type="button" v-if="downloadBtn" :title="$t('material.flow_btn.download')" @click.stop="handleClick('download')")
|
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 475.078 475.077' space='preserve')
|
|
||||||
use(xlink:href='#icon-download')
|
|
||||||
button(type="button" v-if="startBtn" :title="$t('material.flow_btn.start')" @click.stop="handleClick('start')")
|
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 170 170' space='preserve')
|
|
||||||
use(xlink:href='#icon-play')
|
|
||||||
button(type="button" v-if="pauseBtn" :title="$t('material.flow_btn.pause')" @click.stop="handleClick('pause')")
|
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 277.338 277.338' space='preserve')
|
|
||||||
use(xlink:href='#icon-pause')
|
|
||||||
button(type="button" v-if="removeBtn" :title="$t('material.flow_btn.remove')" @click.stop="handleClick('remove')")
|
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 212.982 212.982' space='preserve')
|
|
||||||
use(xlink:href='#icon-delete')
|
|
||||||
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script>
|
|
||||||
|
|
||||||
export default {
|
|
||||||
props: {
|
|
||||||
removeBtn: {
|
|
||||||
type: Boolean,
|
|
||||||
default: true,
|
|
||||||
},
|
|
||||||
startBtn: {
|
|
||||||
type: Boolean,
|
|
||||||
default: false,
|
|
||||||
},
|
|
||||||
pauseBtn: {
|
|
||||||
type: Boolean,
|
|
||||||
default: false,
|
|
||||||
},
|
|
||||||
downloadBtn: {
|
|
||||||
type: Boolean,
|
|
||||||
default: true,
|
|
||||||
},
|
|
||||||
addBtn: {
|
|
||||||
type: Boolean,
|
|
||||||
default: true,
|
|
||||||
},
|
|
||||||
playBtn: {
|
|
||||||
type: Boolean,
|
|
||||||
default: true,
|
|
||||||
},
|
|
||||||
show: {
|
|
||||||
type: Boolean,
|
|
||||||
default: false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
methods: {
|
|
||||||
handleClick(action) {
|
|
||||||
this.$emit('btn-click', action)
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
|
|
||||||
<style lang="less" module>
|
|
||||||
@import '../../assets/styles/layout.less';
|
|
||||||
|
|
||||||
.btns {
|
|
||||||
position: fixed;
|
|
||||||
bottom: 80px;
|
|
||||||
right: 30px;
|
|
||||||
// transform: translateX(-25%);
|
|
||||||
background-color: @color-theme_2-background_2;
|
|
||||||
border-radius: 5px;
|
|
||||||
// padding: 3px 5px;
|
|
||||||
box-shadow: 0 1px 8px 0 rgba(0,0,0,.2);
|
|
||||||
button {
|
|
||||||
background-color: transparent;
|
|
||||||
border: none;
|
|
||||||
border-radius: @form-radius;
|
|
||||||
margin-right: 2px;
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 6px 10px;
|
|
||||||
color: @color-btn;
|
|
||||||
outline: none;
|
|
||||||
transition: background-color 0.2s ease;
|
|
||||||
line-height: 0;
|
|
||||||
&:last-child {
|
|
||||||
margin-right: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
svg {
|
|
||||||
width: 18px;
|
|
||||||
}
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background-color: @color-theme_2-hover;
|
|
||||||
}
|
|
||||||
&:active {
|
|
||||||
background-color: @color-theme_2-active;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
each(@themes, {
|
|
||||||
:global(#container.@{value}) {
|
|
||||||
.btns {
|
|
||||||
background-color: ~'@{color-@{value}-theme_2-background_2}';
|
|
||||||
button {
|
|
||||||
color: ~'@{color-@{value}-btn}';
|
|
||||||
&:hover {
|
|
||||||
background-color: ~'@{color-@{value}-theme_2-hover}';
|
|
||||||
}
|
|
||||||
&:active {
|
|
||||||
background-color: ~'@{color-@{value}-theme_2-active}';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
</style>
|
|
|
@ -1,80 +0,0 @@
|
||||||
<template lang="pug">
|
|
||||||
input(type="range" :class="[$style.range, min ? $style.min : '']" :disabled="disabled" v-model="val" input="handleInput" @change="handleChange")
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script>
|
|
||||||
export default {
|
|
||||||
props: {
|
|
||||||
min: {
|
|
||||||
type: Boolean,
|
|
||||||
},
|
|
||||||
disabled: {
|
|
||||||
type: Boolean,
|
|
||||||
default: false,
|
|
||||||
},
|
|
||||||
value: {
|
|
||||||
type: Number,
|
|
||||||
default: 0,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
data() {
|
|
||||||
return {
|
|
||||||
val: 0,
|
|
||||||
}
|
|
||||||
},
|
|
||||||
mounted() {
|
|
||||||
this.val = this.value
|
|
||||||
},
|
|
||||||
methods: {
|
|
||||||
handleChange(e) {
|
|
||||||
this.$emit('input', this.val)
|
|
||||||
this.$emit('change', e)
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
|
|
||||||
<style lang="less" module>
|
|
||||||
@import '../../assets/styles/layout.less';
|
|
||||||
|
|
||||||
.range {
|
|
||||||
display: inline-block;
|
|
||||||
border-radius: .25em;
|
|
||||||
cursor: pointer;
|
|
||||||
color: @color-btn;
|
|
||||||
outline: none;
|
|
||||||
background: transparent;
|
|
||||||
// background-color: @color-btn-background;
|
|
||||||
&[disabled] {
|
|
||||||
opacity: .4;
|
|
||||||
}
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background-color: @color-theme_2-hover;
|
|
||||||
}
|
|
||||||
&:active {
|
|
||||||
background-color: @color-theme_2-active;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.min {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// each(@themes, {
|
|
||||||
// :global(#container.@{value}) {
|
|
||||||
// .btn {
|
|
||||||
// color: ~'@{color-@{value}-btn}';
|
|
||||||
// background-color: ~'@{color-@{value}-btn-background}';
|
|
||||||
// &:hover {
|
|
||||||
// background-color: ~'@{color-@{value}-theme_2-hover}';
|
|
||||||
// }
|
|
||||||
// &:active {
|
|
||||||
// background-color: ~'@{color-@{value}-theme_2-active}';
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// })
|
|
||||||
|
|
||||||
</style>
|
|
|
@ -1,31 +1,31 @@
|
||||||
<template lang="pug">
|
<template lang="pug">
|
||||||
div(:class="$style.btns")
|
div(:class="$style.btns")
|
||||||
button(type="button" v-if="playBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.play')" @click.stop="handleClick('play')")
|
button(type="button" v-if="playBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.play')" @click.stop="handleClick('play')")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 287.386 287.386' space='preserve')
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 287.386 287.386' space='preserve' v-once)
|
||||||
use(xlink:href='#icon-testPlay')
|
use(xlink:href='#icon-testPlay')
|
||||||
button(type="button" v-if="listAddBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.add_to')" @click.stop="handleClick('listAdd')")
|
button(type="button" v-if="listAddBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.add_to')" @click.stop="handleClick('listAdd')")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 42 42' space='preserve')
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 42 42' space='preserve' v-once)
|
||||||
use(xlink:href='#icon-addTo')
|
use(xlink:href='#icon-addTo')
|
||||||
button(type="button" v-if="downloadBtn && setting.download.enable" @contextmenu.capture.stop :title="$t('material.list_buttons.download')" @click.stop="handleClick('download')")
|
button(type="button" v-if="downloadBtn && setting.download.enable" @contextmenu.capture.stop :title="$t('material.list_buttons.download')" @click.stop="handleClick('download')")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 475.078 475.077' space='preserve')
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 475.078 475.077' space='preserve' v-once)
|
||||||
use(xlink:href='#icon-download')
|
use(xlink:href='#icon-download')
|
||||||
//- button(type="button" :title="$t('material.list_buttons.add')" v-if="userInfo" @click.stop="handleClick('add')")
|
//- button(type="button" :title="$t('material.list_buttons.add')" v-if="userInfo" @click.stop="handleClick('add')")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 42 42' space='preserve')
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 42 42' space='preserve')
|
||||||
use(xlink:href='#icon-addTo')
|
use(xlink:href='#icon-addTo')
|
||||||
button(type="button" v-if="startBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.start')" @click.stop="handleClick('start')")
|
button(type="button" v-if="startBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.start')" @click.stop="handleClick('start')")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 170 170' space='preserve')
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 170 170' space='preserve' v-once)
|
||||||
use(xlink:href='#icon-play')
|
use(xlink:href='#icon-play')
|
||||||
button(type="button" v-if="pauseBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.pause')" @click.stop="handleClick('pause')")
|
button(type="button" v-if="pauseBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.pause')" @click.stop="handleClick('pause')")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 277.338 277.338' space='preserve')
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 277.338 277.338' space='preserve' v-once)
|
||||||
use(xlink:href='#icon-pause')
|
use(xlink:href='#icon-pause')
|
||||||
button(type="button" v-if="fileBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.file')" @click.stop="handleClick('file')")
|
button(type="button" v-if="fileBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.file')" @click.stop="handleClick('file')")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='-61 0 512 512' space='preserve')
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='-61 0 512 512' space='preserve' v-once)
|
||||||
use(xlink:href='#icon-musicFile')
|
use(xlink:href='#icon-musicFile')
|
||||||
button(type="button" v-if="searchBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.search')" @click.stop="handleClick('search')")
|
button(type="button" v-if="searchBtn" @contextmenu.capture.stop :title="$t('material.list_buttons.search')" @click.stop="handleClick('search')")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 30.239 30.239' space='preserve')
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 30.239 30.239' space='preserve' v-once)
|
||||||
use(xlink:href='#icon-search')
|
use(xlink:href='#icon-search')
|
||||||
button(type="button" v-if="removeBtn" :title="$t('material.list_buttons.remove')" @click.stop="handleClick('remove')")
|
button(type="button" v-if="removeBtn" :title="$t('material.list_buttons.remove')" @click.stop="handleClick('remove')")
|
||||||
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 212.982 212.982' space='preserve')
|
svg(version='1.1' xmlns='http://www.w3.org/2000/svg' xlink='http://www.w3.org/1999/xlink' height='100%' viewBox='0 0 212.982 212.982' space='preserve' v-once)
|
||||||
use(xlink:href='#icon-delete')
|
use(xlink:href='#icon-delete')
|
||||||
|
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -239,7 +239,7 @@ export default {
|
||||||
handleSearch() {
|
handleSearch() {
|
||||||
if (!this.text.length) return this.resultList = []
|
if (!this.text.length) return this.resultList = []
|
||||||
let list = []
|
let list = []
|
||||||
let rxp = new RegExp(this.text.split('').join('.*'), 'i')
|
let rxp = new RegExp(this.text.split('').join('.*') + '.*', 'i')
|
||||||
for (const item of this.list) {
|
for (const item of this.list) {
|
||||||
if (rxp.test(`${item.name}${item.singer}${item.albumName ? item.albumName : ''}`)) list.push(item)
|
if (rxp.test(`${item.name}${item.singer}${item.albumName ? item.albumName : ''}`)) list.push(item)
|
||||||
}
|
}
|
||||||
|
|
|
@ -46,7 +46,7 @@ material-modal(:show="version.showModal" @close="handleClose" v-if="version.newV
|
||||||
| 你可以去
|
| 你可以去
|
||||||
strong.hover.underline(@click="handleOpenUrl('https://github.com/lyswhut/lx-music-desktop/releases')" title="点击打开") 软件发布页
|
strong.hover.underline(@click="handleOpenUrl('https://github.com/lyswhut/lx-music-desktop/releases')" title="点击打开") 软件发布页
|
||||||
| 或
|
| 或
|
||||||
strong.hover.underline(@click="handleOpenUrl('https://www.lanzous.com/b906260/')" title="点击打开") 网盘
|
strong.hover.underline(@click="handleOpenUrl('https://www.lanzoux.com/b0bf2cfa/')" title="点击打开") 网盘
|
||||||
| (密码:
|
| (密码:
|
||||||
strong.hover(@click="handleCopy('glqw')" title="点击复制") glqw
|
strong.hover(@click="handleCopy('glqw')" title="点击复制") glqw
|
||||||
| ) 下载新版本,
|
| ) 下载新版本,
|
||||||
|
@ -63,7 +63,7 @@ material-modal(:show="version.showModal" @close="handleClose" v-if="version.newV
|
||||||
| 你可以去
|
| 你可以去
|
||||||
material-btn(min @click="handleOpenUrl('https://github.com/lyswhut/lx-music-desktop/releases')" title="点击打开") 软件发布页
|
material-btn(min @click="handleOpenUrl('https://github.com/lyswhut/lx-music-desktop/releases')" title="点击打开") 软件发布页
|
||||||
| 或
|
| 或
|
||||||
material-btn(min @click="handleOpenUrl('https://www.lanzous.com/b906260/')" title="点击打开") 网盘
|
material-btn(min @click="handleOpenUrl('https://www.lanzoux.com/b0bf2cfa/')" title="点击打开") 网盘
|
||||||
| (密码:
|
| (密码:
|
||||||
strong.hover(@click="handleCopy('glqw')" title="点击复制") glqw
|
strong.hover(@click="handleCopy('glqw')" title="点击复制") glqw
|
||||||
| )下载新版本,
|
| )下载新版本,
|
||||||
|
@ -84,7 +84,7 @@ material-modal(:show="version.showModal" @close="handleClose" v-if="version.newV
|
||||||
| 检查方法:打开
|
| 检查方法:打开
|
||||||
material-btn(min @click="handleOpenUrl('https://github.com/lyswhut/lx-music-desktop/releases')" title="点击打开") 软件发布页
|
material-btn(min @click="handleOpenUrl('https://github.com/lyswhut/lx-music-desktop/releases')" title="点击打开") 软件发布页
|
||||||
| 或
|
| 或
|
||||||
material-btn(min @click="handleOpenUrl('https://www.lanzous.com/b906260/')" title="点击打开") 网盘
|
material-btn(min @click="handleOpenUrl('https://www.lanzoux.com/b0bf2cfa/')" title="点击打开") 网盘
|
||||||
| (密码:
|
| (密码:
|
||||||
strong.hover(@click="handleCopy('glqw')" title="点击复制") glqw
|
strong.hover(@click="handleCopy('glqw')" title="点击复制") glqw
|
||||||
| )查看它们的
|
| )查看它们的
|
||||||
|
@ -117,7 +117,7 @@ material-modal(:show="version.showModal" @close="handleClose" v-if="version.newV
|
||||||
| 手动更新可以去
|
| 手动更新可以去
|
||||||
strong.hover.underline(@click="handleOpenUrl('https://github.com/lyswhut/lx-music-desktop/releases')" title="点击打开") 软件发布页
|
strong.hover.underline(@click="handleOpenUrl('https://github.com/lyswhut/lx-music-desktop/releases')" title="点击打开") 软件发布页
|
||||||
| 或
|
| 或
|
||||||
strong.hover.underline(@click="handleOpenUrl('https://www.lanzous.com/b906260/')" title="点击打开") 网盘
|
strong.hover.underline(@click="handleOpenUrl('https://www.lanzoux.com/b0bf2cfa/')" title="点击打开") 网盘
|
||||||
| (密码:
|
| (密码:
|
||||||
strong.hover(@click="handleCopy('glqw')" title="点击复制") glqw
|
strong.hover(@click="handleCopy('glqw')" title="点击复制") glqw
|
||||||
| ) 下载,
|
| ) 下载,
|
||||||
|
|
|
@ -61,7 +61,7 @@ export default {
|
||||||
this.defaultList,
|
this.defaultList,
|
||||||
this.loveList,
|
this.loveList,
|
||||||
...this.userList,
|
...this.userList,
|
||||||
].filter(l => l.id != this.excludeListId)
|
].filter(l => l.id != this.excludeListId.includes(l.id))
|
||||||
},
|
},
|
||||||
spaceNum() {
|
spaceNum() {
|
||||||
return this.lists.length < 2 ? 0 : (3 - this.lists.length % 3 - 1)
|
return this.lists.length < 2 ? 0 : (3 - this.lists.length % 3 - 1)
|
||||||
|
@ -91,7 +91,7 @@ export default {
|
||||||
this.newListName = event.target.value = ''
|
this.newListName = event.target.value = ''
|
||||||
this.isEditing = false
|
this.isEditing = false
|
||||||
if (!name) return
|
if (!name) return
|
||||||
this.createUserList(name)
|
this.createUserList({ name })
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
|
@ -62,7 +62,7 @@ export default {
|
||||||
this.defaultList,
|
this.defaultList,
|
||||||
this.loveList,
|
this.loveList,
|
||||||
...this.userList,
|
...this.userList,
|
||||||
].filter(l => l.id != this.excludeListId)
|
].filter(l => l.id != this.excludeListId.includes(l.id))
|
||||||
},
|
},
|
||||||
spaceNum() {
|
spaceNum() {
|
||||||
return this.lists.length < 2 ? 0 : (3 - this.lists.length % 3 - 1)
|
return this.lists.length < 2 ? 0 : (3 - this.lists.length % 3 - 1)
|
||||||
|
@ -92,7 +92,7 @@ export default {
|
||||||
this.newListName = event.target.value = ''
|
this.newListName = event.target.value = ''
|
||||||
this.isEditing = false
|
this.isEditing = false
|
||||||
if (!name) return
|
if (!name) return
|
||||||
this.createUserList(name)
|
this.createUserList({ name })
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,21 +1,22 @@
|
||||||
import Vue from 'vue'
|
import Vue from 'vue'
|
||||||
import keyBind from '../utils/keyBind'
|
import keyBind from '../utils/keyBind'
|
||||||
import { rendererOn, rendererSend, NAMES } from '../../common/ipc'
|
import { rendererOn, rendererSend, NAMES, rendererInvoke } from '../../common/ipc'
|
||||||
import { base as baseName } from './names'
|
import { base as baseName } from './names'
|
||||||
import Store from 'electron-store'
|
|
||||||
import { common as hotKeyNamesCommon } from '../../common/hotKey'
|
import { common as hotKeyNamesCommon } from '../../common/hotKey'
|
||||||
|
|
||||||
const eventHub = window.eventHub = new Vue()
|
const eventHub = window.eventHub = new Vue()
|
||||||
|
|
||||||
const electronStore_hotKey = window.electronStore_hotKey = new Store({
|
|
||||||
name: 'hotKey',
|
|
||||||
})
|
|
||||||
|
|
||||||
window.isEditingHotKey = false
|
window.isEditingHotKey = false
|
||||||
const appHotKeyConfig = window.appHotKeyConfig = {
|
let appHotKeyConfig = {
|
||||||
local: electronStore_hotKey.get('local'),
|
local: {},
|
||||||
global: electronStore_hotKey.get('global'),
|
global: {},
|
||||||
}
|
}
|
||||||
|
rendererInvoke(NAMES.mainWindow.get_hot_key).then(({ local, global }) => {
|
||||||
|
appHotKeyConfig = window.appHotKeyConfig = {
|
||||||
|
local,
|
||||||
|
global,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
eventHub.$on(baseName.bindKey, () => {
|
eventHub.$on(baseName.bindKey, () => {
|
||||||
keyBind.bindKey((key, type, event, keys) => {
|
keyBind.bindKey((key, type, event, keys) => {
|
||||||
|
|
|
@ -17,5 +17,13 @@
|
||||||
"hide_detail": "Hide detail page",
|
"hide_detail": "Hide detail page",
|
||||||
"name": "Name: ",
|
"name": "Name: ",
|
||||||
"singer": "Artist: ",
|
"singer": "Artist: ",
|
||||||
"album": "Album: "
|
"album": "Album: ",
|
||||||
|
"add_music_to": "Add the current song to...",
|
||||||
|
"desktop_lyric_on": "Open Desktop Lyrics",
|
||||||
|
"desktop_lyric_off": "Close Desktop Lyrics",
|
||||||
|
"play_toggle_mode_list_loop": "List Loop",
|
||||||
|
"play_toggle_mode_random": "List Random",
|
||||||
|
"play_toggle_mode_list": "Play in order",
|
||||||
|
"play_toggle_mode_single_loop": "Single Loop",
|
||||||
|
"play_toggle_mode_off": "Disable"
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
{
|
{
|
||||||
"basic": "General",
|
"basic": "General",
|
||||||
"basic_theme": "Theme",
|
"basic_theme": "Theme",
|
||||||
|
"basic_show_animation": "Show switching animation",
|
||||||
"basic_animation_title": "Animation effect of the pop-up layer",
|
"basic_animation_title": "Animation effect of the pop-up layer",
|
||||||
"basic_animation": "Random pop-up animation",
|
"basic_animation": "Random pop-up animation",
|
||||||
"basic_source_title": "Choose a music source",
|
"basic_source_title": "Choose a music source",
|
||||||
|
@ -35,6 +36,7 @@
|
||||||
"play_toggle_random": "Playlist shuffle",
|
"play_toggle_random": "Playlist shuffle",
|
||||||
"play_toggle_list": "Play in order",
|
"play_toggle_list": "Play in order",
|
||||||
"play_toggle_single_loop": "Single repeat",
|
"play_toggle_single_loop": "Single repeat",
|
||||||
|
"play_lyric_transition": "Show lyrics translation",
|
||||||
"play_quality_title": "The 320k quality is preferred for playing",
|
"play_quality_title": "The 320k quality is preferred for playing",
|
||||||
"play_quality": "Prefer High Quality 320k",
|
"play_quality": "Prefer High Quality 320k",
|
||||||
"play_task_bar_title": "Show playing progress on the taskbar",
|
"play_task_bar_title": "Show playing progress on the taskbar",
|
||||||
|
@ -48,6 +50,7 @@
|
||||||
"desktop_lyric_enable": "Display lyrics",
|
"desktop_lyric_enable": "Display lyrics",
|
||||||
"desktop_lyric_lock": "Lock lyrics",
|
"desktop_lyric_lock": "Lock lyrics",
|
||||||
"desktop_lyric_always_on_top": "Make the lyrics always above other windows",
|
"desktop_lyric_always_on_top": "Make the lyrics always above other windows",
|
||||||
|
"desktop_lyric_lock_screen": "It is not allowed to drag the lyrics window out of the main screen",
|
||||||
|
|
||||||
"search": "Search",
|
"search": "Search",
|
||||||
"search_hot_title": "Select whether to show popular searches",
|
"search_hot_title": "Select whether to show popular searches",
|
||||||
|
|
|
@ -6,5 +6,7 @@
|
||||||
"input_text": "Enter songlist link or songlist ID",
|
"input_text": "Enter songlist link or songlist ID",
|
||||||
"tip_1": "Cross-source playlists are not supported, please confirm whether the playlist to be opened corresponds to the current playlist source",
|
"tip_1": "Cross-source playlists are not supported, please confirm whether the playlist to be opened corresponds to the current playlist source",
|
||||||
"tip_2": "If you encounter a link to a playlist that cannot be opened, welcome feedback",
|
"tip_2": "If you encounter a link to a playlist that cannot be opened, welcome feedback",
|
||||||
"tip_3": "Kugou source does not support opening with playlist ID, but supports Kugou code opening"
|
"tip_3": "Kugou source does not support opening with playlist ID, but supports Kugou code opening",
|
||||||
|
"play_all": "Play",
|
||||||
|
"add_all": "Collect"
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,5 +17,13 @@
|
||||||
"hide_detail": "隐藏详情页",
|
"hide_detail": "隐藏详情页",
|
||||||
"name": "歌曲名:",
|
"name": "歌曲名:",
|
||||||
"singer": "艺术家:",
|
"singer": "艺术家:",
|
||||||
"album": "专辑名:"
|
"album": "专辑名:",
|
||||||
|
"add_music_to": "添加当前歌曲到...",
|
||||||
|
"desktop_lyric_on": "开启桌面歌词",
|
||||||
|
"desktop_lyric_off": "关闭桌面歌词",
|
||||||
|
"play_toggle_mode_list_loop": "列表循环",
|
||||||
|
"play_toggle_mode_random": "列表随机",
|
||||||
|
"play_toggle_mode_list": "顺序播放",
|
||||||
|
"play_toggle_mode_single_loop": "单曲循环",
|
||||||
|
"play_toggle_mode_off": "禁用"
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,6 +3,7 @@
|
||||||
"basic_theme": "主题颜色",
|
"basic_theme": "主题颜色",
|
||||||
"basic_animation_title": "弹出层的动画效果",
|
"basic_animation_title": "弹出层的动画效果",
|
||||||
"basic_animation": "弹出层随机动画",
|
"basic_animation": "弹出层随机动画",
|
||||||
|
"basic_show_animation": "显示切换动画",
|
||||||
"basic_source_title": "选择音乐来源",
|
"basic_source_title": "选择音乐来源",
|
||||||
"basic_source_test": "测试接口(几乎软件的所有功能都可用)",
|
"basic_source_test": "测试接口(几乎软件的所有功能都可用)",
|
||||||
"basic_source_temp": "临时接口(软件的某些功能不可用,建议测试接口不可用再使用本接口)",
|
"basic_source_temp": "临时接口(软件的某些功能不可用,建议测试接口不可用再使用本接口)",
|
||||||
|
@ -35,6 +36,7 @@
|
||||||
"play_toggle_random": "列表随机",
|
"play_toggle_random": "列表随机",
|
||||||
"play_toggle_list": "顺序播放",
|
"play_toggle_list": "顺序播放",
|
||||||
"play_toggle_single_loop": "单曲循环",
|
"play_toggle_single_loop": "单曲循环",
|
||||||
|
"play_lyric_transition": "显示歌词翻译",
|
||||||
"play_quality_title": "启用时将优先播放320K品质的歌曲",
|
"play_quality_title": "启用时将优先播放320K品质的歌曲",
|
||||||
"play_quality": "优先播放高品质音乐",
|
"play_quality": "优先播放高品质音乐",
|
||||||
"play_task_bar_title": "在任务栏上显示当前歌曲播放进度",
|
"play_task_bar_title": "在任务栏上显示当前歌曲播放进度",
|
||||||
|
@ -48,6 +50,7 @@
|
||||||
"desktop_lyric_enable": "显示歌词",
|
"desktop_lyric_enable": "显示歌词",
|
||||||
"desktop_lyric_lock": "锁定歌词",
|
"desktop_lyric_lock": "锁定歌词",
|
||||||
"desktop_lyric_always_on_top": "使歌词总是在其他窗口之上",
|
"desktop_lyric_always_on_top": "使歌词总是在其他窗口之上",
|
||||||
|
"desktop_lyric_lock_screen": "不允许歌词窗口拖出主屏幕之外",
|
||||||
|
|
||||||
"search": "搜索设置",
|
"search": "搜索设置",
|
||||||
"search_hot_title": "是否显示热门搜索",
|
"search_hot_title": "是否显示热门搜索",
|
||||||
|
|
|
@ -6,5 +6,7 @@
|
||||||
"input_text": "输入歌单链接或歌单ID",
|
"input_text": "输入歌单链接或歌单ID",
|
||||||
"tip_1": "不支持跨源打开歌单,请确认要打开的歌单与当前歌单源是否对应",
|
"tip_1": "不支持跨源打开歌单,请确认要打开的歌单与当前歌单源是否对应",
|
||||||
"tip_2": "若遇到无法打开的歌单链接,欢迎反馈",
|
"tip_2": "若遇到无法打开的歌单链接,欢迎反馈",
|
||||||
"tip_3": "酷狗源不支持用歌单ID打开,但支持酷狗码打开"
|
"tip_3": "酷狗源不支持用歌单ID打开,但支持酷狗码打开",
|
||||||
|
"play_all": "播放",
|
||||||
|
"add_all": "收藏"
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,5 +17,13 @@
|
||||||
"hide_detail": "隱藏詳情頁",
|
"hide_detail": "隱藏詳情頁",
|
||||||
"name": "歌曲名:",
|
"name": "歌曲名:",
|
||||||
"singer": "藝術家:",
|
"singer": "藝術家:",
|
||||||
"album": "專輯名:"
|
"album": "專輯名:",
|
||||||
|
"add_music_to": "添加當前歌曲到...",
|
||||||
|
"desktop_lyric_on": "開啟桌面歌詞",
|
||||||
|
"desktop_lyric_off": "關閉桌面歌詞",
|
||||||
|
"play_toggle_mode_list_loop": "列表循環",
|
||||||
|
"play_toggle_mode_random": "列表隨機",
|
||||||
|
"play_toggle_mode_list": "順序播放",
|
||||||
|
"play_toggle_mode_single_loop": "單曲循環",
|
||||||
|
"play_toggle_mode_off": "禁用"
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,6 +3,7 @@
|
||||||
"basic_theme": "主題顏色",
|
"basic_theme": "主題顏色",
|
||||||
"basic_animation_title": "彈出層的動畫效果",
|
"basic_animation_title": "彈出層的動畫效果",
|
||||||
"basic_animation": "彈出層隨機動畫",
|
"basic_animation": "彈出層隨機動畫",
|
||||||
|
"basic_show_animation": "顯示切換動畫",
|
||||||
"basic_source_title": "選擇音樂來源",
|
"basic_source_title": "選擇音樂來源",
|
||||||
"basic_source_test": "測試接口(幾乎軟件的所有功能都可用)",
|
"basic_source_test": "測試接口(幾乎軟件的所有功能都可用)",
|
||||||
"basic_source_temp": "臨時接口(軟件的某些功能不可用,建議測試接口不可用再使用本接口)",
|
"basic_source_temp": "臨時接口(軟件的某些功能不可用,建議測試接口不可用再使用本接口)",
|
||||||
|
@ -34,6 +35,7 @@
|
||||||
"play_toggle_random": "列表隨機",
|
"play_toggle_random": "列表隨機",
|
||||||
"play_toggle_list": "順序播放",
|
"play_toggle_list": "順序播放",
|
||||||
"play_toggle_single_loop": "單曲循環",
|
"play_toggle_single_loop": "單曲循環",
|
||||||
|
"play_lyric_transition": "顯示歌詞翻譯",
|
||||||
"play_quality_title": "啟用時將優先播放320K品質的歌曲",
|
"play_quality_title": "啟用時將優先播放320K品質的歌曲",
|
||||||
"play_quality": "優先播放高品質音樂",
|
"play_quality": "優先播放高品質音樂",
|
||||||
"play_task_bar_title": "在任務欄上顯示當前歌曲播放進度",
|
"play_task_bar_title": "在任務欄上顯示當前歌曲播放進度",
|
||||||
|
@ -46,6 +48,7 @@
|
||||||
"desktop_lyric_enable": "顯示歌詞",
|
"desktop_lyric_enable": "顯示歌詞",
|
||||||
"desktop_lyric_lock": "鎖定歌詞",
|
"desktop_lyric_lock": "鎖定歌詞",
|
||||||
"desktop_lyric_always_on_top": "使歌詞總是在其他窗口之上",
|
"desktop_lyric_always_on_top": "使歌詞總是在其他窗口之上",
|
||||||
|
"desktop_lyric_lock_screen": "不允許歌詞窗口拖出主屏幕之外",
|
||||||
"search": "搜索設置",
|
"search": "搜索設置",
|
||||||
"search_hot_title": "是否顯示熱門搜索",
|
"search_hot_title": "是否顯示熱門搜索",
|
||||||
"search_hot": "熱門搜索",
|
"search_hot": "熱門搜索",
|
||||||
|
|
|
@ -6,5 +6,7 @@
|
||||||
"input_text": "輸入歌單鏈接或歌單ID",
|
"input_text": "輸入歌單鏈接或歌單ID",
|
||||||
"tip_1": "不支持跨源打開歌單,請確認要打開的歌單與當前歌單源是否對應",
|
"tip_1": "不支持跨源打開歌單,請確認要打開的歌單與當前歌單源是否對應",
|
||||||
"tip_2": "若遇到無法打開的歌單鏈接,歡迎反饋",
|
"tip_2": "若遇到無法打開的歌單鏈接,歡迎反饋",
|
||||||
"tip_3": "酷狗源不支持用歌單ID打開,但支持酷狗碼打開"
|
"tip_3": "酷狗源不支持用歌單ID打開,但支持酷狗碼打開",
|
||||||
|
"play_all": "播放",
|
||||||
|
"add_all": "收藏"
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,10 +16,39 @@ import store from './store'
|
||||||
|
|
||||||
import '../common/error'
|
import '../common/error'
|
||||||
|
|
||||||
|
import { getSetting } from './utils'
|
||||||
|
import languageList from '@/lang/languages.json'
|
||||||
|
import { rendererSend, NAMES } from '../common/ipc'
|
||||||
|
|
||||||
sync(store, router)
|
sync(store, router)
|
||||||
|
|
||||||
Vue.config.productionTip = false
|
Vue.config.productionTip = false
|
||||||
|
|
||||||
|
|
||||||
|
getSetting().then(({ setting, version }) => {
|
||||||
|
// Set language automatically
|
||||||
|
if (!window.i18n.availableLocales.includes(setting.langId)) {
|
||||||
|
let langId = null
|
||||||
|
let locale = window.navigator.language.toLocaleLowerCase()
|
||||||
|
if (window.i18n.availableLocales.includes(locale)) {
|
||||||
|
langId = locale
|
||||||
|
} else {
|
||||||
|
for (const lang of languageList) {
|
||||||
|
if (lang.alternate == locale) {
|
||||||
|
langId = lang.locale
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (langId == null) langId = 'en-us'
|
||||||
|
}
|
||||||
|
setting.langId = langId
|
||||||
|
rendererSend(NAMES.mainWindow.set_app_setting, setting)
|
||||||
|
console.log('Set lang', setting.langId)
|
||||||
|
}
|
||||||
|
window.i18n.locale = setting.langId
|
||||||
|
store.commit('setSetting', setting)
|
||||||
|
store.commit('setSettingVersion', version)
|
||||||
|
|
||||||
new Vue({
|
new Vue({
|
||||||
router,
|
router,
|
||||||
store,
|
store,
|
||||||
|
@ -27,3 +56,5 @@ new Vue({
|
||||||
el: '#root',
|
el: '#root',
|
||||||
render: h => h(App),
|
render: h => h(App),
|
||||||
})
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
|
|
@ -177,7 +177,7 @@ const saveMeta = (downloadInfo, filePath, isEmbedPic, isEmbedLyric) => {
|
||||||
: Promise.resolve(),
|
: Promise.resolve(),
|
||||||
isEmbedLyric
|
isEmbedLyric
|
||||||
? downloadInfo.musicInfo.lrc
|
? downloadInfo.musicInfo.lrc
|
||||||
? Promise.resolve(downloadInfo.musicInfo.lrc)
|
? Promise.resolve({ lyric: downloadInfo.musicInfo.lrc, tlyric: downloadInfo.musicInfo.tlrc || '' })
|
||||||
: music[downloadInfo.musicInfo.source].getLyric(downloadInfo.musicInfo).promise.catch(err => {
|
: music[downloadInfo.musicInfo.source].getLyric(downloadInfo.musicInfo).promise.catch(err => {
|
||||||
console.log(err)
|
console.log(err)
|
||||||
return null
|
return null
|
||||||
|
@ -190,7 +190,7 @@ const saveMeta = (downloadInfo, filePath, isEmbedPic, isEmbedLyric) => {
|
||||||
artist: downloadInfo.musicInfo.singer,
|
artist: downloadInfo.musicInfo.singer,
|
||||||
album: downloadInfo.musicInfo.albumName,
|
album: downloadInfo.musicInfo.albumName,
|
||||||
APIC: imgUrl,
|
APIC: imgUrl,
|
||||||
lyrics,
|
lyrics: lyrics.lyric,
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -202,10 +202,10 @@ const saveMeta = (downloadInfo, filePath, isEmbedPic, isEmbedLyric) => {
|
||||||
*/
|
*/
|
||||||
const downloadLyric = (downloadInfo, filePath) => {
|
const downloadLyric = (downloadInfo, filePath) => {
|
||||||
const promise = downloadInfo.musicInfo.lrc
|
const promise = downloadInfo.musicInfo.lrc
|
||||||
? Promise.resolve(downloadInfo.musicInfo.lrc)
|
? Promise.resolve({ lyric: downloadInfo.musicInfo.lrc, tlyric: downloadInfo.musicInfo.tlrc || '' })
|
||||||
: music[downloadInfo.musicInfo.source].getLyric(downloadInfo.musicInfo).promise
|
: music[downloadInfo.musicInfo.source].getLyric(downloadInfo.musicInfo).promise
|
||||||
promise.then(lrc => {
|
promise.then(lrcs => {
|
||||||
if (lrc) saveLrc(filePath.replace(/(mp3|flac|ape|wav)$/, 'lrc'), lrc)
|
if (lrcs.lyric) saveLrc(filePath.replace(/(mp3|flac|ape|wav)$/, 'lrc'), lrcs.lyric)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -218,13 +218,13 @@ const refreshUrl = function(commit, downloadInfo) {
|
||||||
if (!dl) return
|
if (!dl) return
|
||||||
dl.refreshUrl(result.url)
|
dl.refreshUrl(result.url)
|
||||||
dl.start().catch(err => {
|
dl.start().catch(err => {
|
||||||
commit('onError', downloadInfo)
|
commit('onError', { downloadInfo, errorMsg: err.message })
|
||||||
commit('setStatusText', { downloadInfo, text: err.message })
|
commit('setStatusText', { downloadInfo, text: err.message })
|
||||||
this.dispatch('download/startTask')
|
this.dispatch('download/startTask')
|
||||||
})
|
})
|
||||||
}).catch(err => {
|
}).catch(err => {
|
||||||
// console.log(err)
|
// console.log(err)
|
||||||
commit('onError', downloadInfo)
|
commit('onError', { downloadInfo, errorMsg: err.message })
|
||||||
commit('setStatusText', { downloadInfo, text: err.message })
|
commit('setStatusText', { downloadInfo, text: err.message })
|
||||||
this.dispatch('download/startTask')
|
this.dispatch('download/startTask')
|
||||||
})
|
})
|
||||||
|
@ -299,7 +299,7 @@ const actions = {
|
||||||
try {
|
try {
|
||||||
await checkPath(rootState.setting.download.savePath)
|
await checkPath(rootState.setting.download.savePath)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
commit('onError', downloadInfo)
|
commit('onError', { downloadInfo, errorMsg: error.message })
|
||||||
commit('setStatusText', '检查下载目录出错: ' + error.message)
|
commit('setStatusText', '检查下载目录出错: ' + error.message)
|
||||||
await dispatch('startTask')
|
await dispatch('startTask')
|
||||||
return
|
return
|
||||||
|
@ -324,9 +324,14 @@ const actions = {
|
||||||
console.log('on complate')
|
console.log('on complate')
|
||||||
},
|
},
|
||||||
onError(err) {
|
onError(err) {
|
||||||
|
// console.log(err)
|
||||||
|
if (err.code == 'EPERM') {
|
||||||
|
commit('onError', { downloadInfo, errorMsg: '歌曲下载目录没有写入权限,请尝试更改歌曲保存路径' })
|
||||||
|
return
|
||||||
|
}
|
||||||
// console.log(tryNum[downloadInfo.key])
|
// console.log(tryNum[downloadInfo.key])
|
||||||
if (++tryNum[downloadInfo.key] > 2) {
|
if (++tryNum[downloadInfo.key] > 2) {
|
||||||
commit('onError', downloadInfo)
|
commit('onError', { downloadInfo, errorMsg: err.message })
|
||||||
dispatch('startTask')
|
dispatch('startTask')
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -375,7 +380,7 @@ const actions = {
|
||||||
dls[downloadInfo.key] = download(options)
|
dls[downloadInfo.key] = download(options)
|
||||||
}).catch(err => {
|
}).catch(err => {
|
||||||
// console.log(err.message)
|
// console.log(err.message)
|
||||||
commit('onError', downloadInfo)
|
commit('onError', { downloadInfo, errorMsg: err.message })
|
||||||
commit('setStatusText', { downloadInfo, text: err.message })
|
commit('setStatusText', { downloadInfo, text: err.message })
|
||||||
dispatch('startTask')
|
dispatch('startTask')
|
||||||
})
|
})
|
||||||
|
@ -437,7 +442,7 @@ const actions = {
|
||||||
try {
|
try {
|
||||||
await dl.start()
|
await dl.start()
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
commit('onError', downloadInfo)
|
commit('onError', { downloadInfo, errorMsg: error.message })
|
||||||
commit('setStatusText', error.message)
|
commit('setStatusText', error.message)
|
||||||
await dispatch('startTask')
|
await dispatch('startTask')
|
||||||
}
|
}
|
||||||
|
@ -448,7 +453,7 @@ const actions = {
|
||||||
startTasks(store, list) {
|
startTasks(store, list) {
|
||||||
if (isRuningActionTask) return
|
if (isRuningActionTask) return
|
||||||
isRuningActionTask = true
|
isRuningActionTask = true
|
||||||
return startTasks(store, [...list]).finally(() => {
|
return startTasks(store, list.filter(item => !(item.isComplate || item.status == state.downloadStatus.RUN || item.status == state.downloadStatus.WAITING))).finally(() => {
|
||||||
isRuningActionTask = false
|
isRuningActionTask = false
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
@ -522,9 +527,9 @@ const mutations = {
|
||||||
downloadInfo.status = state.downloadStatus.COMPLETED
|
downloadInfo.status = state.downloadStatus.COMPLETED
|
||||||
downloadInfo.statusText = '下载完成'
|
downloadInfo.statusText = '下载完成'
|
||||||
},
|
},
|
||||||
onError(state, downloadInfo) {
|
onError(state, { downloadInfo, errorMsg }) {
|
||||||
downloadInfo.status = state.downloadStatus.ERROR
|
downloadInfo.status = state.downloadStatus.ERROR
|
||||||
downloadInfo.statusText = '任务出错'
|
downloadInfo.statusText = errorMsg || '任务出错'
|
||||||
},
|
},
|
||||||
onStart(state, downloadInfo) {
|
onStart(state, downloadInfo) {
|
||||||
downloadInfo.status = state.downloadStatus.RUN
|
downloadInfo.status = state.downloadStatus.RUN
|
||||||
|
|
|
@ -28,6 +28,12 @@ const state = {
|
||||||
list: [],
|
list: [],
|
||||||
location: 0,
|
location: 0,
|
||||||
},
|
},
|
||||||
|
tempList: {
|
||||||
|
id: 'temp',
|
||||||
|
name: '临时列表',
|
||||||
|
list: [],
|
||||||
|
location: 0,
|
||||||
|
},
|
||||||
userList: [],
|
userList: [],
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -140,15 +146,19 @@ const mutations = {
|
||||||
if (!targetList) return
|
if (!targetList) return
|
||||||
Object.assign(targetList.list[index], data)
|
Object.assign(targetList.list[index], data)
|
||||||
},
|
},
|
||||||
createUserList(state, name) {
|
createUserList(state, { name, id = `userlist_${Date.now()}`, list = [] }) {
|
||||||
let newList = {
|
let newList = state.userList.find(item => item.id === id)
|
||||||
|
if (!newList) {
|
||||||
|
newList = {
|
||||||
name,
|
name,
|
||||||
id: `userlist_${Date.now()}`,
|
id,
|
||||||
list: [],
|
list: [],
|
||||||
location: 0,
|
location: 0,
|
||||||
}
|
}
|
||||||
state.userList.push(newList)
|
state.userList.push(newList)
|
||||||
allListUpdate(newList)
|
allListUpdate(newList)
|
||||||
|
}
|
||||||
|
this.commit('list/listAddMultiple', { id, list })
|
||||||
},
|
},
|
||||||
removeUserList(state, index) {
|
removeUserList(state, index) {
|
||||||
let list = state.userList.splice(index, 1)[0]
|
let list = state.userList.splice(index, 1)[0]
|
||||||
|
|
|
@ -9,6 +9,7 @@ const state = {
|
||||||
playIndex: -1,
|
playIndex: -1,
|
||||||
changePlay: false,
|
changePlay: false,
|
||||||
isShowPlayerDetail: false,
|
isShowPlayerDetail: false,
|
||||||
|
playedList: [],
|
||||||
}
|
}
|
||||||
|
|
||||||
let urlRequest
|
let urlRequest
|
||||||
|
@ -22,6 +23,7 @@ const getters = {
|
||||||
changePlay: satte => satte.changePlay,
|
changePlay: satte => satte.changePlay,
|
||||||
playIndex: state => state.playIndex,
|
playIndex: state => state.playIndex,
|
||||||
isShowPlayerDetail: state => state.isShowPlayerDetail,
|
isShowPlayerDetail: state => state.isShowPlayerDetail,
|
||||||
|
playedList: state => state.playedList,
|
||||||
}
|
}
|
||||||
|
|
||||||
// actions
|
// actions
|
||||||
|
@ -58,9 +60,9 @@ const actions = {
|
||||||
getLrc({ commit, state }, musicInfo) {
|
getLrc({ commit, state }, musicInfo) {
|
||||||
if (lrcRequest && lrcRequest.cancelHttp) lrcRequest.cancelHttp()
|
if (lrcRequest && lrcRequest.cancelHttp) lrcRequest.cancelHttp()
|
||||||
lrcRequest = music[musicInfo.source].getLyric(musicInfo)
|
lrcRequest = music[musicInfo.source].getLyric(musicInfo)
|
||||||
return lrcRequest.promise.then(lrc => {
|
return lrcRequest.promise.then(({ lyric, tlyric }) => {
|
||||||
lrcRequest = null
|
lrcRequest = null
|
||||||
commit('setLrc', { musicInfo, lrc })
|
commit('setLrc', { musicInfo, lyric, tlyric })
|
||||||
}).catch(err => {
|
}).catch(err => {
|
||||||
lrcRequest = null
|
lrcRequest = null
|
||||||
return Promise.reject(err)
|
return Promise.reject(err)
|
||||||
|
@ -78,12 +80,14 @@ const mutations = {
|
||||||
datas.musicInfo.img = datas.url
|
datas.musicInfo.img = datas.url
|
||||||
},
|
},
|
||||||
setLrc(state, datas) {
|
setLrc(state, datas) {
|
||||||
datas.musicInfo.lrc = datas.lrc
|
datas.musicInfo.lrc = datas.lyric
|
||||||
|
datas.musicInfo.tlrc = datas.tlyric
|
||||||
},
|
},
|
||||||
setList(state, { list, index }) {
|
setList(state, { list, index }) {
|
||||||
state.listInfo = list
|
state.listInfo = list
|
||||||
state.playIndex = index
|
state.playIndex = index
|
||||||
state.changePlay = true
|
state.changePlay = true
|
||||||
|
if (state.playedList.length) this.commit('player/clearPlayedList')
|
||||||
},
|
},
|
||||||
setPlayIndex(state, index) {
|
setPlayIndex(state, index) {
|
||||||
state.playIndex = index
|
state.playIndex = index
|
||||||
|
@ -96,6 +100,16 @@ const mutations = {
|
||||||
resetChangePlay(state) {
|
resetChangePlay(state) {
|
||||||
state.changePlay = false
|
state.changePlay = false
|
||||||
},
|
},
|
||||||
|
setPlayedList(state, item) {
|
||||||
|
if (state.playedList.includes(item)) return
|
||||||
|
state.playedList.push(item)
|
||||||
|
},
|
||||||
|
removePlayedList(state, index) {
|
||||||
|
state.playedList.splice(index, 1)
|
||||||
|
},
|
||||||
|
clearPlayedList(state) {
|
||||||
|
state.playedList = []
|
||||||
|
},
|
||||||
visiblePlayerDetail(state, visible) {
|
visiblePlayerDetail(state, visible) {
|
||||||
state.isShowPlayerDetail = visible
|
state.isShowPlayerDetail = visible
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,15 +1,5 @@
|
||||||
import Store from 'electron-store'
|
|
||||||
import music from '../../utils/music'
|
import music from '../../utils/music'
|
||||||
|
|
||||||
const electronStore_data = window.electronStore_data = new Store({
|
|
||||||
name: 'data',
|
|
||||||
})
|
|
||||||
let historyList = electronStore_data.get('searchHistoryList')
|
|
||||||
if (historyList == null) {
|
|
||||||
historyList = []
|
|
||||||
electronStore_data.set('searchHistoryList', historyList)
|
|
||||||
}
|
|
||||||
|
|
||||||
const sources = []
|
const sources = []
|
||||||
const sourceList = {}
|
const sourceList = {}
|
||||||
const sourceMaxPage = {}
|
const sourceMaxPage = {}
|
||||||
|
@ -103,7 +93,7 @@ const state = {
|
||||||
allPage: 1,
|
allPage: 1,
|
||||||
total: 0,
|
total: 0,
|
||||||
sourceMaxPage,
|
sourceMaxPage,
|
||||||
historyList,
|
historyList: [],
|
||||||
}
|
}
|
||||||
|
|
||||||
// getters
|
// getters
|
||||||
|
@ -211,6 +201,9 @@ const mutations = {
|
||||||
clearHistory(state) {
|
clearHistory(state) {
|
||||||
state.historyList = []
|
state.historyList = []
|
||||||
},
|
},
|
||||||
|
setHistory(state, list) {
|
||||||
|
state.historyList = list
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
|
|
@ -79,37 +79,29 @@ const actions = {
|
||||||
cache.has(key)
|
cache.has(key)
|
||||||
? Promise.resolve(cache.get(key))
|
? Promise.resolve(cache.get(key))
|
||||||
: music[source].songList.getListDetail(id, page)
|
: music[source].songList.getListDetail(id, page)
|
||||||
).then(result => commit('setListDetail', { result, key, page }))
|
).then(result => commit('setListDetail', { result, key, source, id, page }))
|
||||||
},
|
},
|
||||||
/* getListDetailAll({ state, rootState }, id) {
|
getListDetailAll({ state, rootState }, id) {
|
||||||
let source = rootState.setting.songList.source
|
let source = rootState.setting.songList.source
|
||||||
let key = `sdetail__${source}__${id}__all`
|
const loadData = (id, page) => {
|
||||||
if (cache.has(key)) return Promise.resolve(cache.get(key))
|
let key = `sdetail__${source}__${id}__${page}`
|
||||||
music[source].songList.getListDetail(id, 1).then(result => {
|
return cache.has(key) ? Promise.resolve(cache.get(key)) : music[source].songList.getListDetail(id, page).then(result => {
|
||||||
let data = { list: result.list, id }
|
cache.set(key, result)
|
||||||
if (result.total <= result.limit) {
|
return result
|
||||||
data = { list: result.list, id }
|
})
|
||||||
cache.set(key, data)
|
|
||||||
return data
|
|
||||||
}
|
}
|
||||||
|
return loadData(id, 1).then(result => {
|
||||||
|
if (result.total <= result.limit) return result.list
|
||||||
|
|
||||||
let maxPage = Math.ceil(result.total / result.limit)
|
let maxPage = Math.ceil(result.total / result.limit)
|
||||||
const loadDetail = (loadPage = 1) => {
|
const loadDetail = (loadPage = 1) => {
|
||||||
let task = []
|
|
||||||
let loadNum = 0
|
|
||||||
while (loadPage <= maxPage && loadNum < 3) {
|
|
||||||
task.push(music[source].songList.getListDetail(id, ++loadPage))
|
|
||||||
loadNum++
|
|
||||||
}
|
|
||||||
return loadPage == maxPage
|
return loadPage == maxPage
|
||||||
? Promise.all(task)
|
? loadData(id, ++loadPage).then(result => result.list)
|
||||||
: Promise.all(task).then(result => loadDetail(loadPage).then(result2 => [...result, ...result2]))
|
: loadData(id, ++loadPage).then(result1 => loadDetail(loadPage).then(result2 => [...result1.list, ...result2]))
|
||||||
}
|
}
|
||||||
return loadDetail().then(result2 => {
|
return loadDetail().then(result2 => [...result.list, ...result2])
|
||||||
console.log(result2)
|
|
||||||
})
|
})
|
||||||
})
|
},
|
||||||
}, */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// mitations
|
// mitations
|
||||||
|
@ -129,8 +121,10 @@ const mutations = {
|
||||||
state.list.key = key
|
state.list.key = key
|
||||||
cache.set(key, result)
|
cache.set(key, result)
|
||||||
},
|
},
|
||||||
setListDetail(state, { result, key, page }) {
|
setListDetail(state, { result, key, source, id, page }) {
|
||||||
state.listDetail.list = result.list
|
state.listDetail.list = result.list
|
||||||
|
state.listDetail.id = id
|
||||||
|
state.listDetail.source = source
|
||||||
state.listDetail.total = result.total
|
state.listDetail.total = result.total
|
||||||
state.listDetail.limit = result.limit
|
state.listDetail.limit = result.limit
|
||||||
state.listDetail.page = page
|
state.listDetail.page = page
|
||||||
|
@ -153,6 +147,8 @@ const mutations = {
|
||||||
},
|
},
|
||||||
clearListDetail(state) {
|
clearListDetail(state) {
|
||||||
state.listDetail = {
|
state.listDetail = {
|
||||||
|
id: null,
|
||||||
|
source: null,
|
||||||
list: [],
|
list: [],
|
||||||
desc: null,
|
desc: null,
|
||||||
total: 0,
|
total: 0,
|
||||||
|
|
|
@ -9,12 +9,12 @@ export default {
|
||||||
setSetting(state, val) {
|
setSetting(state, val) {
|
||||||
state.setting = val
|
state.setting = val
|
||||||
},
|
},
|
||||||
setAgreePact(state) {
|
|
||||||
state.setting.isAgreePact = true
|
|
||||||
},
|
|
||||||
setSettingVersion(state, val) {
|
setSettingVersion(state, val) {
|
||||||
state.settingVersion = val
|
state.settingVersion = val
|
||||||
},
|
},
|
||||||
|
setAgreePact(state) {
|
||||||
|
state.setting.isAgreePact = true
|
||||||
|
},
|
||||||
setLeaderboard(state, { tabId, source }) {
|
setLeaderboard(state, { tabId, source }) {
|
||||||
if (tabId != null) state.setting.leaderboard.tabId = tabId
|
if (tabId != null) state.setting.leaderboard.tabId = tabId
|
||||||
if (source != null) state.setting.leaderboard.source = source
|
if (source != null) state.setting.leaderboard.source = source
|
||||||
|
@ -49,6 +49,12 @@ export default {
|
||||||
state.setting.player.volume = val
|
state.setting.player.volume = val
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
setPlayNextMode(state, val) {
|
||||||
|
state.setting.player.togglePlayMethod = val
|
||||||
|
},
|
||||||
|
setVisibleDesktopLyric(state, val) {
|
||||||
|
state.setting.desktopLyric.enable = val
|
||||||
|
},
|
||||||
setMediaDeviceId(state, val) {
|
setMediaDeviceId(state, val) {
|
||||||
state.setting.player.mediaDeviceId = val
|
state.setting.player.mediaDeviceId = val
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,67 +1,9 @@
|
||||||
|
|
||||||
// const isDev = process.env.NODE_ENV === 'development'
|
// const isDev = process.env.NODE_ENV === 'development'
|
||||||
import Store from 'electron-store'
|
|
||||||
import { windowSizeList } from '../../common/config'
|
import { windowSizeList } from '../../common/config'
|
||||||
import { version } from '../../../package.json'
|
import { version } from '../../../package.json'
|
||||||
import { rendererSend, rendererInvoke, NAMES } from '../../common/ipc'
|
|
||||||
import languageList from '@/lang/languages.json'
|
|
||||||
import path from 'path'
|
|
||||||
import { openDirInExplorer } from '../utils'
|
|
||||||
|
|
||||||
|
|
||||||
const electronStore_config = window.electronStore_config = new Store({
|
|
||||||
name: 'config',
|
|
||||||
})
|
|
||||||
let setting = electronStore_config.get('setting')
|
|
||||||
let settingVersion = electronStore_config.get('version')
|
|
||||||
|
|
||||||
process.versions.app = version
|
process.versions.app = version
|
||||||
|
|
||||||
// Set language automatically
|
|
||||||
if (!window.i18n.availableLocales.includes(setting.langId)) {
|
|
||||||
let langId = null
|
|
||||||
let locale = window.navigator.language.toLocaleLowerCase()
|
|
||||||
if (window.i18n.availableLocales.includes(locale)) {
|
|
||||||
langId = locale
|
|
||||||
} else {
|
|
||||||
for (const lang of languageList) {
|
|
||||||
if (lang.alternate == locale) {
|
|
||||||
langId = lang.locale
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (langId == null) langId = 'en-us'
|
|
||||||
}
|
|
||||||
setting.langId = langId
|
|
||||||
electronStore_config.set('setting', setting)
|
|
||||||
rendererSend(NAMES.mainWindow.set_app_setting, setting)
|
|
||||||
console.log('Set lang', setting.langId)
|
|
||||||
}
|
|
||||||
|
|
||||||
window.i18n.locale = setting.langId
|
|
||||||
|
|
||||||
try {
|
|
||||||
window.electronStore_list = new Store({
|
|
||||||
name: 'playList',
|
|
||||||
clearInvalidConfig: false,
|
|
||||||
})
|
|
||||||
} catch (error) {
|
|
||||||
rendererInvoke(NAMES.mainWindow.get_data_path).then(dataPath => {
|
|
||||||
let filePath = path.join(dataPath, 'playList.json.bak')
|
|
||||||
rendererInvoke(NAMES.mainWindow.show_dialog, {
|
|
||||||
type: 'error',
|
|
||||||
message: window.i18n.t('store.state.load_list_file_error_title'),
|
|
||||||
detail: window.i18n.t('store.state.load_list_file_error_detail', {
|
|
||||||
path: filePath,
|
|
||||||
detail: error.message,
|
|
||||||
}),
|
|
||||||
}).then(() => openDirInExplorer(filePath))
|
|
||||||
})
|
|
||||||
window.electronStore_list = new Store({
|
|
||||||
name: 'playList',
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
themes: [
|
themes: [
|
||||||
|
@ -144,8 +86,8 @@ export default {
|
||||||
downloadProgress: null,
|
downloadProgress: null,
|
||||||
},
|
},
|
||||||
userInfo: null,
|
userInfo: null,
|
||||||
setting,
|
setting: null,
|
||||||
settingVersion,
|
settingVersion: null,
|
||||||
|
|
||||||
windowSizeList,
|
windowSizeList,
|
||||||
}
|
}
|
||||||
|
|
|
@ -172,7 +172,7 @@ class Task extends EventEmitter {
|
||||||
this.chunkInfo.startByte = 0
|
this.chunkInfo.startByte = 0
|
||||||
this.resumeLastChunk = null
|
this.resumeLastChunk = null
|
||||||
this.progress.downloaded = 0
|
this.progress.downloaded = 0
|
||||||
if (unlinkErr) this.__handleError(unlinkErr)
|
if (unlinkErr && unlinkErr.code !== 'ENOENT') this.__handleError(unlinkErr)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import fs from 'fs'
|
import fs from 'fs'
|
||||||
|
import path from 'path'
|
||||||
import { shell, clipboard } from 'electron'
|
import { shell, clipboard } from 'electron'
|
||||||
import crypto from 'crypto'
|
import crypto from 'crypto'
|
||||||
import { rendererSend, rendererInvoke, NAMES } from '../../common/ipc'
|
import { rendererSend, rendererInvoke, NAMES } from '../../common/ipc'
|
||||||
|
@ -341,3 +342,22 @@ export const getProxyInfo = () => window.globalObj.proxy.enable
|
||||||
|
|
||||||
|
|
||||||
export const assertApiSupport = source => window.globalObj.qualityList[source] != undefined
|
export const assertApiSupport = source => window.globalObj.qualityList[source] != undefined
|
||||||
|
|
||||||
|
export const getSetting = () => rendererInvoke(NAMES.mainWindow.get_setting)
|
||||||
|
export const saveSetting = () => rendererInvoke(NAMES.mainWindow.set_app_setting)
|
||||||
|
|
||||||
|
export const getPlayList = () => rendererInvoke(NAMES.mainWindow.get_playlist).catch(error => {
|
||||||
|
rendererInvoke(NAMES.mainWindow.get_data_path).then(dataPath => {
|
||||||
|
let filePath = path.join(dataPath, 'playList.json.bak')
|
||||||
|
rendererInvoke(NAMES.mainWindow.show_dialog, {
|
||||||
|
type: 'error',
|
||||||
|
message: window.i18n.t('store.state.load_list_file_error_title'),
|
||||||
|
detail: window.i18n.t('store.state.load_list_file_error_detail', {
|
||||||
|
path: filePath,
|
||||||
|
detail: error.message,
|
||||||
|
}),
|
||||||
|
}).then(() => openDirInExplorer(filePath))
|
||||||
|
})
|
||||||
|
return rendererInvoke(NAMES.mainWindow.get_playlist, true)
|
||||||
|
})
|
||||||
|
|
||||||
|
|
|
@ -12,7 +12,7 @@ module.exports = [
|
||||||
wy: ['128k'],
|
wy: ['128k'],
|
||||||
mg: ['128k'],
|
mg: ['128k'],
|
||||||
xm: ['128k'],
|
xm: ['128k'],
|
||||||
bd: ['128k'],
|
// bd: ['128k'],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|
|
@ -21,7 +21,7 @@ const bd = {
|
||||||
},
|
},
|
||||||
getLyric(songInfo) {
|
getLyric(songInfo) {
|
||||||
const requestObj = this.getMusicInfo(songInfo)
|
const requestObj = this.getMusicInfo(songInfo)
|
||||||
requestObj.promise = requestObj.promise.then(info => httpFetch(info.lrclink).promise.then(resp => resp.body))
|
requestObj.promise = requestObj.promise.then(info => httpFetch(info.lrclink).promise.then(resp => ({ lyric: resp.body, tlyric: '' })))
|
||||||
return requestObj
|
return requestObj
|
||||||
},
|
},
|
||||||
// getLyric(songInfo) {
|
// getLyric(songInfo) {
|
||||||
|
|
|
@ -34,10 +34,10 @@ const sources = {
|
||||||
name: '虾米音乐',
|
name: '虾米音乐',
|
||||||
id: 'xm',
|
id: 'xm',
|
||||||
},
|
},
|
||||||
{
|
// {
|
||||||
name: '百度音乐',
|
// name: '百度音乐',
|
||||||
id: 'bd',
|
// id: 'bd',
|
||||||
},
|
// },
|
||||||
],
|
],
|
||||||
kw,
|
kw,
|
||||||
kg,
|
kg,
|
||||||
|
|
|
@ -1,4 +1,41 @@
|
||||||
import { httpFetch } from '../../request'
|
import { httpFetch } from '../../request'
|
||||||
|
import { decodeLyric } from './util'
|
||||||
|
|
||||||
|
const parseLyric = str => {
|
||||||
|
str = str.replace(/(?:<\d+,\d+,\d+>|\r)/g, '')
|
||||||
|
let trans = str.match(/\[language:([\w=\\/+]+)\]/)
|
||||||
|
let tlyric
|
||||||
|
if (trans) {
|
||||||
|
str = str.replace(/\[language:[\w=\\/+]+\]\n/, '')
|
||||||
|
let json = JSON.parse(Buffer.from(trans[1], 'base64').toString())
|
||||||
|
for (const item of json.content) {
|
||||||
|
if (item.type == 1) {
|
||||||
|
tlyric = item.lyricContent
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let i = 0
|
||||||
|
let lyric = str.replace(/\[((\d+),\d+)\].*/g, str => {
|
||||||
|
let result = str.match(/\[((\d+),\d+)\].*/)
|
||||||
|
let time = parseInt(result[2])
|
||||||
|
let ms = time % 1000
|
||||||
|
time /= 1000
|
||||||
|
let h = parseInt(time / 3600).toString().padStart(2, '0')
|
||||||
|
time %= 3600
|
||||||
|
let m = parseInt(time / 60).toString().padStart(2, '0')
|
||||||
|
time %= 60
|
||||||
|
let s = parseInt(time).toString().padStart(2, '0')
|
||||||
|
time = `${h}:${m}:${s}.${ms}`
|
||||||
|
if (tlyric) tlyric[i] = `[${time}]${tlyric[i++][0]}`
|
||||||
|
return str.replace(result[1], time)
|
||||||
|
})
|
||||||
|
tlyric = tlyric ? tlyric.join('\n') : ''
|
||||||
|
return {
|
||||||
|
lyric,
|
||||||
|
tlyric,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
getIntv(interval) {
|
getIntv(interval) {
|
||||||
|
@ -11,8 +48,30 @@ export default {
|
||||||
}
|
}
|
||||||
return parseInt(intv)
|
return parseInt(intv)
|
||||||
},
|
},
|
||||||
getLyric(songInfo, tryNum = 0) {
|
// getLyric(songInfo, tryNum = 0) {
|
||||||
let requestObj = httpFetch(`http://m.kugou.com/app/i/krc.php?cmd=100&keyword=${encodeURIComponent(songInfo.name)}&hash=${songInfo.hash}&timelength=${songInfo._interval || this.getIntv(songInfo.interval)}&d=0.38664927426725626`, {
|
// let requestObj = httpFetch(`http://m.kugou.com/app/i/krc.php?cmd=100&keyword=${encodeURIComponent(songInfo.name)}&hash=${songInfo.hash}&timelength=${songInfo._interval || this.getIntv(songInfo.interval)}&d=0.38664927426725626`, {
|
||||||
|
// headers: {
|
||||||
|
// 'KG-RC': 1,
|
||||||
|
// 'KG-THash': 'expand_search_manager.cpp:852736169:451',
|
||||||
|
// 'User-Agent': 'KuGou2012-9020-ExpandSearchManager',
|
||||||
|
// },
|
||||||
|
// })
|
||||||
|
// requestObj.promise = requestObj.promise.then(({ body, statusCode }) => {
|
||||||
|
// if (statusCode !== 200) {
|
||||||
|
// if (tryNum > 5) return Promise.reject('歌词获取失败')
|
||||||
|
// let tryRequestObj = this.getLyric(songInfo, ++tryNum)
|
||||||
|
// requestObj.cancelHttp = tryRequestObj.cancelHttp.bind(tryRequestObj)
|
||||||
|
// return tryRequestObj.promise
|
||||||
|
// }
|
||||||
|
// return {
|
||||||
|
// lyric: body,
|
||||||
|
// tlyric: '',
|
||||||
|
// }
|
||||||
|
// })
|
||||||
|
// return requestObj
|
||||||
|
// },
|
||||||
|
searchLyric(name, hash, time, tryNum = 0) {
|
||||||
|
let requestObj = httpFetch(`http://lyrics.kugou.com/search?ver=1&man=yes&client=pc&keyword=${encodeURIComponent(name)}&hash=${hash}&timelength=${time}`, {
|
||||||
headers: {
|
headers: {
|
||||||
'KG-RC': 1,
|
'KG-RC': 1,
|
||||||
'KG-THash': 'expand_search_manager.cpp:852736169:451',
|
'KG-THash': 'expand_search_manager.cpp:852736169:451',
|
||||||
|
@ -22,11 +81,49 @@ export default {
|
||||||
requestObj.promise = requestObj.promise.then(({ body, statusCode }) => {
|
requestObj.promise = requestObj.promise.then(({ body, statusCode }) => {
|
||||||
if (statusCode !== 200) {
|
if (statusCode !== 200) {
|
||||||
if (tryNum > 5) return Promise.reject('歌词获取失败')
|
if (tryNum > 5) return Promise.reject('歌词获取失败')
|
||||||
let tryRequestObj = this.getLyric(songInfo, ++tryNum)
|
let tryRequestObj = this.searchLyric(name, hash, time, ++tryNum)
|
||||||
requestObj.cancelHttp = tryRequestObj.cancelHttp.bind(tryRequestObj)
|
requestObj.cancelHttp = tryRequestObj.cancelHttp.bind(tryRequestObj)
|
||||||
return tryRequestObj.promise
|
return tryRequestObj.promise
|
||||||
}
|
}
|
||||||
return body
|
if (body.candidates.length) {
|
||||||
|
let info = body.candidates[0]
|
||||||
|
return { id: info.id, accessKey: info.accesskey }
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
})
|
||||||
|
return requestObj
|
||||||
|
},
|
||||||
|
getLyricDownload(id, accessKey, tryNum = 0) {
|
||||||
|
let requestObj = httpFetch(`http://lyrics.kugou.com/download?ver=1&client=pc&id=${id}&accesskey=${accessKey}&fmt=krc&charset=utf8`, {
|
||||||
|
headers: {
|
||||||
|
'KG-RC': 1,
|
||||||
|
'KG-THash': 'expand_search_manager.cpp:852736169:451',
|
||||||
|
'User-Agent': 'KuGou2012-9020-ExpandSearchManager',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
requestObj.promise = requestObj.promise.then(({ body, statusCode }) => {
|
||||||
|
if (statusCode !== 200) {
|
||||||
|
if (tryNum > 5) return Promise.reject('歌词获取失败')
|
||||||
|
let tryRequestObj = this.getLyric(id, accessKey, ++tryNum)
|
||||||
|
requestObj.cancelHttp = tryRequestObj.cancelHttp.bind(tryRequestObj)
|
||||||
|
return tryRequestObj.promise
|
||||||
|
}
|
||||||
|
|
||||||
|
return decodeLyric(body.content).then(result => parseLyric(result))
|
||||||
|
})
|
||||||
|
return requestObj
|
||||||
|
},
|
||||||
|
getLyric(songInfo, tryNum = 0) {
|
||||||
|
let requestObj = this.searchLyric(songInfo.name, songInfo.hash, songInfo._interval || this.getIntv(songInfo.interval))
|
||||||
|
|
||||||
|
requestObj.promise = requestObj.promise.then(result => {
|
||||||
|
if (!result) return { lyric: '', tlyric: '' }
|
||||||
|
|
||||||
|
let requestObj2 = this.getLyricDownload(result.id, result.accessKey)
|
||||||
|
|
||||||
|
requestObj.cancelHttp = requestObj2.cancelHttp.bind(requestObj2)
|
||||||
|
|
||||||
|
return requestObj2.promise
|
||||||
})
|
})
|
||||||
return requestObj
|
return requestObj
|
||||||
},
|
},
|
||||||
|
|
|
@ -0,0 +1,19 @@
|
||||||
|
import { inflate } from 'zlib'
|
||||||
|
|
||||||
|
// https://github.com/lyswhut/lx-music-desktop/issues/296#issuecomment-683285784
|
||||||
|
const enc_key = Buffer.from([0x40, 0x47, 0x61, 0x77, 0x5e, 0x32, 0x74, 0x47, 0x51, 0x36, 0x31, 0x2d, 0xce, 0xd2, 0x6e, 0x69], 'binary')
|
||||||
|
export const decodeLyric = str => new Promise((resolve, reject) => {
|
||||||
|
if (!str.length) return
|
||||||
|
const buf_str = Buffer.from(str, 'base64').slice(4)
|
||||||
|
for (let i = 0, len = buf_str.length; i < len; i++) {
|
||||||
|
buf_str[i] = buf_str[i] ^ enc_key[i % 16]
|
||||||
|
}
|
||||||
|
inflate(buf_str, (err, result) => {
|
||||||
|
if (err) return reject(err)
|
||||||
|
resolve(result.toString())
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
// s.content[0].lyricContent.forEach(([str]) => {
|
||||||
|
// console.log(str)
|
||||||
|
// })
|
|
@ -17,7 +17,10 @@ export default {
|
||||||
return requestObj.promise.then(({ statusCode, body, raw }) => {
|
return requestObj.promise.then(({ statusCode, body, raw }) => {
|
||||||
if (statusCode != 200) return Promise.reject(new Error(JSON.stringify(body)))
|
if (statusCode != 200) return Promise.reject(new Error(JSON.stringify(body)))
|
||||||
return decodeLyric({ lrcBase64: raw.toString('base64'), isGetLyricx }).then(base64Data => {
|
return decodeLyric({ lrcBase64: raw.toString('base64'), isGetLyricx }).then(base64Data => {
|
||||||
return Buffer.from(base64Data, 'base64').toString()
|
return {
|
||||||
|
lyric: Buffer.from(base64Data, 'base64').toString(),
|
||||||
|
tlyric: '',
|
||||||
|
}
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
|
@ -12,7 +12,10 @@ export default {
|
||||||
requestObj.cancelHttp = tryRequestObj.cancelHttp.bind(tryRequestObj)
|
requestObj.cancelHttp = tryRequestObj.cancelHttp.bind(tryRequestObj)
|
||||||
return tryRequestObj.promise
|
return tryRequestObj.promise
|
||||||
}
|
}
|
||||||
return body
|
return {
|
||||||
|
lyric: body,
|
||||||
|
tlyric: '',
|
||||||
|
}
|
||||||
})
|
})
|
||||||
return requestObj
|
return requestObj
|
||||||
} else {
|
} else {
|
||||||
|
@ -28,7 +31,10 @@ export default {
|
||||||
requestObj.cancelHttp = tryRequestObj.cancelHttp.bind(tryRequestObj)
|
requestObj.cancelHttp = tryRequestObj.cancelHttp.bind(tryRequestObj)
|
||||||
return tryRequestObj.promise
|
return tryRequestObj.promise
|
||||||
}
|
}
|
||||||
return body.lyric
|
return {
|
||||||
|
lyric: body.lyric,
|
||||||
|
tlyric: '',
|
||||||
|
}
|
||||||
})
|
})
|
||||||
return requestObj
|
return requestObj
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,13 +6,17 @@ export default {
|
||||||
matchLrc: /.+"lyric":"([\w=+/]*)".+/,
|
matchLrc: /.+"lyric":"([\w=+/]*)".+/,
|
||||||
},
|
},
|
||||||
getLyric(songmid) {
|
getLyric(songmid) {
|
||||||
const requestObj = httpFetch(`https://c.y.qq.com/lyric/fcgi-bin/fcg_query_lyric_new.fcg?songmid=${songmid}&g_tk=2001461048&loginUin=0&hostUin=0&format=jsonp&inCharset=utf8&outCharset=utf-8&platform=yqq`, {
|
const requestObj = httpFetch(`https://c.y.qq.com/lyric/fcgi-bin/fcg_query_lyric_new.fcg?songmid=${songmid}&g_tk=5381&loginUin=0&hostUin=0&format=json&inCharset=utf8&outCharset=utf-8&platform=yqq`, {
|
||||||
headers: {
|
headers: {
|
||||||
Referer: 'https://y.qq.com/portal/player.html',
|
Referer: 'https://y.qq.com/portal/player.html',
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
requestObj.promise = requestObj.promise.then(({ body }) => {
|
requestObj.promise = requestObj.promise.then(({ body }) => {
|
||||||
return decodeName(b64DecodeUnicode(body.replace(this.regexps.matchLrc, '$1')))
|
if (body.code != 0) return Promise.reject(new Error('获取歌词失败'))
|
||||||
|
return {
|
||||||
|
lyric: decodeName(b64DecodeUnicode(body.lyric)),
|
||||||
|
tlyric: decodeName(b64DecodeUnicode(body.trans)),
|
||||||
|
}
|
||||||
})
|
})
|
||||||
return requestObj
|
return requestObj
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,20 +1,27 @@
|
||||||
import { httpFetch } from '../../request'
|
import { httpFetch } from '../../request'
|
||||||
import { weapi } from './utils/crypto'
|
import { linuxapi } from './utils/crypto'
|
||||||
|
|
||||||
export default songmid => {
|
export default songmid => {
|
||||||
const requestObj = httpFetch('http://music.163.com/weapi/song/lyric?csrf_token=', {
|
const requestObj = httpFetch('https://music.163.com/api/linux/forward', {
|
||||||
method: 'post',
|
method: 'post',
|
||||||
headers: {
|
|
||||||
'User-Agent': 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36',
|
'User-Agent': 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36',
|
||||||
Referer: 'https://music.163.com/song?id=' + songmid,
|
form: linuxapi({
|
||||||
origin: 'https://music.163.com',
|
method: 'POST',
|
||||||
|
url: 'https://music.163.com/api/song/lyric',
|
||||||
|
params: {
|
||||||
|
id: songmid,
|
||||||
|
lv: -1,
|
||||||
|
kv: -1,
|
||||||
|
tv: -1,
|
||||||
},
|
},
|
||||||
form: weapi({ id: songmid, lv: -1, tv: -1, csrf_token: '' }),
|
}),
|
||||||
})
|
})
|
||||||
requestObj.promise = requestObj.promise.then(({ body }) => {
|
requestObj.promise = requestObj.promise.then(({ body }) => {
|
||||||
// console.log(body)
|
|
||||||
if (body.code !== 200) return Promise.reject('获取歌词失败')
|
if (body.code !== 200) return Promise.reject('获取歌词失败')
|
||||||
return body.lrc.lyric
|
return {
|
||||||
|
lyric: body.lrc.lyric,
|
||||||
|
tlyric: body.tlyric.lyric,
|
||||||
|
}
|
||||||
})
|
})
|
||||||
return requestObj
|
return requestObj
|
||||||
}
|
}
|
||||||
|
|
|
@ -78,7 +78,7 @@ export default {
|
||||||
return arr.join('、')
|
return arr.join('、')
|
||||||
},
|
},
|
||||||
filterData(rawList) {
|
filterData(rawList) {
|
||||||
console.log(rawList)
|
// console.log(rawList)
|
||||||
let ids = new Set()
|
let ids = new Set()
|
||||||
const list = []
|
const list = []
|
||||||
rawList.forEach(songData => {
|
rawList.forEach(songData => {
|
||||||
|
|
|
@ -1,6 +1,23 @@
|
||||||
import { httpGet, httpFetch } from '../../request'
|
import { httpGet, httpFetch } from '../../request'
|
||||||
import { xmRequest } from './util'
|
import { xmRequest } from './util'
|
||||||
|
|
||||||
|
const parseLyric = str => {
|
||||||
|
str = str.replace(/(?:<\d+>|\r)/g, '')
|
||||||
|
let tlyric = []
|
||||||
|
let lyric = str.replace(/\[[\d:.]+\].*?\n\[x-trans\].*/g, s => {
|
||||||
|
// console.log(s)
|
||||||
|
let [lrc, tlrc] = s.split('\n')
|
||||||
|
tlrc = tlrc.replace('[x-trans]', lrc.replace(/^(\[[\d:.]+\]).*$/, '$1'))
|
||||||
|
tlyric.push(tlrc)
|
||||||
|
return lrc
|
||||||
|
})
|
||||||
|
tlyric = tlyric.join('\n')
|
||||||
|
return {
|
||||||
|
lyric,
|
||||||
|
tlyric,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
failTime: 0,
|
failTime: 0,
|
||||||
expireTime: 60 * 1000 * 1000,
|
expireTime: 60 * 1000 * 1000,
|
||||||
|
@ -13,7 +30,10 @@ export default {
|
||||||
requestObj.cancelHttp = tryRequestObj.cancelHttp.bind(tryRequestObj)
|
requestObj.cancelHttp = tryRequestObj.cancelHttp.bind(tryRequestObj)
|
||||||
return tryRequestObj.promise
|
return tryRequestObj.promise
|
||||||
}
|
}
|
||||||
return body
|
return url.endsWith('.xtrc') ? parseLyric(body) : {
|
||||||
|
lyric: body,
|
||||||
|
tlyric: '',
|
||||||
|
}
|
||||||
})
|
})
|
||||||
return requestObj
|
return requestObj
|
||||||
},
|
},
|
||||||
|
@ -27,7 +47,10 @@ export default {
|
||||||
},
|
},
|
||||||
}, function(err, resp, body) {
|
}, function(err, resp, body) {
|
||||||
if (err || resp.statusCode !== 200) return this.getLyricFile(url, ++retryNum).then(resolve).catch(reject)
|
if (err || resp.statusCode !== 200) return this.getLyricFile(url, ++retryNum).then(resolve).catch(reject)
|
||||||
return resolve(body)
|
return resolve(url.endsWith('.xtrc') ? parseLyric(body) : {
|
||||||
|
lyric: body,
|
||||||
|
tlyric: '',
|
||||||
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
@ -47,8 +70,12 @@ export default {
|
||||||
return tryRequestObj.promise
|
return tryRequestObj.promise
|
||||||
}
|
}
|
||||||
if (!body.result.data.lyrics.length) return Promise.reject(new Error('未找到歌词'))
|
if (!body.result.data.lyrics.length) return Promise.reject(new Error('未找到歌词'))
|
||||||
let lrc = body.result.data.lyrics.find(lyric => /\.lrc$/.test(lyric.lyricUrl))
|
let lrc = body.result.data.lyrics.find(lyric => /\.(trc|lrc)$/.test(lyric.lyricUrl))
|
||||||
return lrc ? lrc.content : Promise.reject(new Error('未找到歌词'))
|
return lrc
|
||||||
|
? lrc.lyricUrl.endsWith('.trc')
|
||||||
|
? parseLyric(lrc.content)
|
||||||
|
: { lyric: lrc.content, tlyric: '' }
|
||||||
|
: Promise.reject(new Error('未找到歌词'))
|
||||||
})
|
})
|
||||||
return requestObj
|
return requestObj
|
||||||
},
|
},
|
||||||
|
@ -74,7 +101,7 @@ export default {
|
||||||
return requestObj
|
return requestObj
|
||||||
},
|
},
|
||||||
getLyric(songInfo) {
|
getLyric(songInfo) {
|
||||||
if (songInfo.lrcUrl && /\.lrc$/.test(songInfo.lrcUrl)) return this.getLyricFile_1(songInfo.lrcUrl)
|
if (songInfo.lrcUrl && /\.(xtrc|lrc)$/.test(songInfo.lrcUrl)) return this.getLyricFile_1(songInfo.lrcUrl)
|
||||||
return Date.now() - this.failTime > this.expireTime ? this.getLyricUrl_1(songInfo) : this.getLyricUrl_2(songInfo)
|
return Date.now() - this.failTime > this.expireTime ? this.getLyricUrl_1(songInfo) : this.getLyricUrl_2(songInfo)
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,7 +29,6 @@ div(:class="$style.download")
|
||||||
:start-btn="!item.isComplate && item.status != downloadStatus.WAITING && (item.status != downloadStatus.RUN)"
|
:start-btn="!item.isComplate && item.status != downloadStatus.WAITING && (item.status != downloadStatus.RUN)"
|
||||||
:pause-btn="!item.isComplate && (item.status == downloadStatus.RUN || item.status == downloadStatus.WAITING)" :list-add-btn="false"
|
:pause-btn="!item.isComplate && (item.status == downloadStatus.RUN || item.status == downloadStatus.WAITING)" :list-add-btn="false"
|
||||||
:play-btn="item.status == downloadStatus.COMPLETED" :search-btn="item.status == downloadStatus.ERROR" @btn-click="handleListBtnClick")
|
:play-btn="item.status == downloadStatus.COMPLETED" :search-btn="item.status == downloadStatus.ERROR" @btn-click="handleListBtnClick")
|
||||||
//- material-flow-btn(:show="isShowEditBtn" :play-btn="false" :download-btn="false" :add-btn="false" :start-btn="true" :pause-btn="true" @btn-click="handleFlowBtnClick")
|
|
||||||
material-menu(:menus="listItemMenu" :location="listMenu.menuLocation" item-name="name" :isShow="listMenu.isShowItemMenu" @menu-click="handleListItemMenuClick")
|
material-menu(:menus="listItemMenu" :location="listMenu.menuLocation" item-name="name" :isShow="listMenu.isShowItemMenu" @menu-click="handleListItemMenuClick")
|
||||||
div(:class="$style.noItem" v-else)
|
div(:class="$style.noItem" v-else)
|
||||||
</template>
|
</template>
|
||||||
|
@ -47,7 +46,6 @@ export default {
|
||||||
clickTime: window.performance.now(),
|
clickTime: window.performance.now(),
|
||||||
clickIndex: -1,
|
clickIndex: -1,
|
||||||
selectdData: [],
|
selectdData: [],
|
||||||
// isShowEditBtn: false,
|
|
||||||
isShowDownloadMultiple: false,
|
isShowDownloadMultiple: false,
|
||||||
tabId: 'all',
|
tabId: 'all',
|
||||||
keyEvent: {
|
keyEvent: {
|
||||||
|
@ -166,14 +164,6 @@ export default {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
selectdData(n) {
|
|
||||||
const len = n.length
|
|
||||||
if (len) {
|
|
||||||
this.isShowEditBtn = true
|
|
||||||
} else {
|
|
||||||
this.isShowEditBtn = false
|
|
||||||
}
|
|
||||||
},
|
|
||||||
list() {
|
list() {
|
||||||
this.removeAllSelect()
|
this.removeAllSelect()
|
||||||
},
|
},
|
||||||
|
|
|
@ -643,7 +643,7 @@ export default {
|
||||||
this.listsData.isShowNewList = false
|
this.listsData.isShowNewList = false
|
||||||
})
|
})
|
||||||
|
|
||||||
this.createUserList(name)
|
this.createUserList({ name })
|
||||||
},
|
},
|
||||||
handleShowNewList() {
|
handleShowNewList() {
|
||||||
this.listsData.isShowNewList = true
|
this.listsData.isShowNewList = true
|
||||||
|
|
|
@ -10,6 +10,11 @@ div.scroll(:class="$style.setting")
|
||||||
span
|
span
|
||||||
label {{$t('store.state.theme_' + theme.class)}}
|
label {{$t('store.state.theme_' + theme.class)}}
|
||||||
|
|
||||||
|
dd
|
||||||
|
h3 {{$t('view.setting.basic_show_animation')}}
|
||||||
|
div
|
||||||
|
material-checkbox(id="setting_show_animate" v-model="current_setting.isShowAnimation" :label="$t('view.setting.is_show')")
|
||||||
|
|
||||||
dd(:title="$t('view.setting.basic_animation_title')")
|
dd(:title="$t('view.setting.basic_animation_title')")
|
||||||
h3 {{$t('view.setting.basic_animation')}}
|
h3 {{$t('view.setting.basic_animation')}}
|
||||||
div
|
div
|
||||||
|
@ -58,6 +63,10 @@ div.scroll(:class="$style.setting")
|
||||||
div
|
div
|
||||||
material-checkbox(:id="`setting_player_togglePlay_${item.value}`" :class="$style.gapLeft" :value="item.value" :key="item.value"
|
material-checkbox(:id="`setting_player_togglePlay_${item.value}`" :class="$style.gapLeft" :value="item.value" :key="item.value"
|
||||||
v-model="current_setting.player.togglePlayMethod" v-for="item in togglePlayMethods" :label="item.name")
|
v-model="current_setting.player.togglePlayMethod" v-for="item in togglePlayMethods" :label="item.name")
|
||||||
|
dd
|
||||||
|
h3 {{$t('view.setting.play_lyric_transition')}}
|
||||||
|
div
|
||||||
|
material-checkbox(id="setting_player_lyric_transition" v-model="current_setting.player.isShowLyricTransition" :label="$t('view.setting.is_show')")
|
||||||
dd(:title="$t('view.setting.play_quality_title')")
|
dd(:title="$t('view.setting.play_quality_title')")
|
||||||
h3 {{$t('view.setting.play_quality')}}
|
h3 {{$t('view.setting.play_quality')}}
|
||||||
div
|
div
|
||||||
|
@ -82,6 +91,8 @@ div.scroll(:class="$style.setting")
|
||||||
material-checkbox(id="setting_desktop_lyric_lock" v-model="current_setting.desktopLyric.isLock" :label="$t('view.setting.desktop_lyric_lock')")
|
material-checkbox(id="setting_desktop_lyric_lock" v-model="current_setting.desktopLyric.isLock" :label="$t('view.setting.desktop_lyric_lock')")
|
||||||
div(:class="$style.gapTop")
|
div(:class="$style.gapTop")
|
||||||
material-checkbox(id="setting_desktop_lyric_alwaysOnTop" v-model="current_setting.desktopLyric.isAlwaysOnTop" :label="$t('view.setting.desktop_lyric_always_on_top')")
|
material-checkbox(id="setting_desktop_lyric_alwaysOnTop" v-model="current_setting.desktopLyric.isAlwaysOnTop" :label="$t('view.setting.desktop_lyric_always_on_top')")
|
||||||
|
div(:class="$style.gapTop")
|
||||||
|
material-checkbox(id="setting_desktop_lyric_lockScreen" v-model="current_setting.desktopLyric.isLockScreen" :label="$t('view.setting.desktop_lyric_lock_screen')")
|
||||||
dt {{$t('view.setting.search')}}
|
dt {{$t('view.setting.search')}}
|
||||||
dd(:title="$t('view.setting.search_hot_title')")
|
dd(:title="$t('view.setting.search_hot_title')")
|
||||||
h3 {{$t('view.setting.search_hot')}}
|
h3 {{$t('view.setting.search_hot')}}
|
||||||
|
@ -228,7 +239,7 @@ div.scroll(:class="$style.setting")
|
||||||
span.hover.underline(:title="$t('view.setting.click_open')" @click="handleOpenUrl('https://github.com/lyswhut/lx-music-desktop#readme')") https://github.com/lyswhut/lx-music-desktop
|
span.hover.underline(:title="$t('view.setting.click_open')" @click="handleOpenUrl('https://github.com/lyswhut/lx-music-desktop#readme')") https://github.com/lyswhut/lx-music-desktop
|
||||||
p.small
|
p.small
|
||||||
| 最新版网盘下载地址(网盘内有Windows、MAC版):
|
| 最新版网盘下载地址(网盘内有Windows、MAC版):
|
||||||
span.hover.underline(:title="$t('view.setting.click_open')" @click="handleOpenUrl('https://t-s.lanzous.com/b0bf2cfa')") 网盘地址
|
span.hover.underline(:title="$t('view.setting.click_open')" @click="handleOpenUrl('https://www.lanzoux.com/b0bf2cfa/')") 网盘地址
|
||||||
| 密码:
|
| 密码:
|
||||||
span.hover(:title="$t('view.setting.click_copy')" @click="clipboardWriteText('glqw')") glqw
|
span.hover(:title="$t('view.setting.click_copy')" @click="clipboardWriteText('glqw')") glqw
|
||||||
p.small
|
p.small
|
||||||
|
@ -448,6 +459,7 @@ export default {
|
||||||
langId: 'cns',
|
langId: 'cns',
|
||||||
themeId: 0,
|
themeId: 0,
|
||||||
sourceId: 0,
|
sourceId: 0,
|
||||||
|
isShowAnimation: true,
|
||||||
randomAnimate: true,
|
randomAnimate: true,
|
||||||
isAgreePact: false,
|
isAgreePact: false,
|
||||||
controlBtnPosition: 'left',
|
controlBtnPosition: 'left',
|
||||||
|
@ -587,6 +599,15 @@ export default {
|
||||||
'setting.player.mediaDeviceId'(n) {
|
'setting.player.mediaDeviceId'(n) {
|
||||||
this.current_setting.player.mediaDeviceId = n
|
this.current_setting.player.mediaDeviceId = n
|
||||||
},
|
},
|
||||||
|
'setting.player.isMute'(n) {
|
||||||
|
this.current_setting.player.isMute = n
|
||||||
|
},
|
||||||
|
'setting.desktopLyric.enable'(n) {
|
||||||
|
this.current_setting.desktopLyric.enable = n
|
||||||
|
},
|
||||||
|
'setting.player.togglePlayMethod'(n) {
|
||||||
|
this.current_setting.player.togglePlayMethod = n
|
||||||
|
},
|
||||||
'current_setting.player.isShowTaskProgess'(n) {
|
'current_setting.player.isShowTaskProgess'(n) {
|
||||||
if (n) return
|
if (n) return
|
||||||
this.$nextTick(() => {
|
this.$nextTick(() => {
|
||||||
|
|
|
@ -10,11 +10,9 @@
|
||||||
h3(:title="listDetail.info.name || selectListInfo.name") {{listDetail.info.name || selectListInfo.name}}
|
h3(:title="listDetail.info.name || selectListInfo.name") {{listDetail.info.name || selectListInfo.name}}
|
||||||
p(:title="listDetail.info.desc || selectListInfo.desc") {{listDetail.info.desc || selectListInfo.desc}}
|
p(:title="listDetail.info.desc || selectListInfo.desc") {{listDetail.info.desc || selectListInfo.desc}}
|
||||||
div(:class="$style.songListHeaderRight")
|
div(:class="$style.songListHeaderRight")
|
||||||
//- material-btn(:class="$style.closeDetailButton" :disabled="detailLoading" @click="addSongListDetail") 添加
|
material-btn(:class="$style.headerRightBtn" :disabled="detailLoading" @click="playSongListDetail") {{$t('view.song_list.play_all')}}
|
||||||
//- |
|
material-btn(:class="$style.headerRightBtn" :disabled="detailLoading" @click="addSongListDetail") {{$t('view.song_list.add_all')}}
|
||||||
//- material-btn(:class="$style.closeDetailButton" :disabled="detailLoading" @click="playSongListDetail") 播放
|
material-btn(:class="$style.headerRightBtn" @click="hideListDetail") {{$t('view.song_list.back')}}
|
||||||
//- |
|
|
||||||
material-btn(:class="$style.closeDetailButton" @click="hideListDetail") {{$t('view.song_list.back')}}
|
|
||||||
material-song-list(v-model="selectedData" @action="handleSongListAction" :source="source" :page="listDetail.page" :limit="listDetail.limit"
|
material-song-list(v-model="selectedData" @action="handleSongListAction" :source="source" :page="listDetail.page" :limit="listDetail.limit"
|
||||||
:total="listDetail.total" :noItem="isGetDetailFailed ? $t('view.song_list.loding_list_fail') : $t('view.song_list.loding_list')" :list="listDetail.list")
|
:total="listDetail.total" :noItem="isGetDetailFailed ? $t('view.song_list.loding_list_fail') : $t('view.song_list.loding_list')" :list="listDetail.list")
|
||||||
transition(enter-active-class="animated-fast fadeIn" leave-active-class="animated-fast fadeOut")
|
transition(enter-active-class="animated-fast fadeIn" leave-active-class="animated-fast fadeOut")
|
||||||
|
@ -84,7 +82,7 @@ export default {
|
||||||
listWidth: 645,
|
listWidth: 645,
|
||||||
isGetDetailFailed: false,
|
isGetDetailFailed: false,
|
||||||
isInitedTagListWidth: false,
|
isInitedTagListWidth: false,
|
||||||
// detailLoading: true,
|
detailLoading: false,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
|
@ -97,7 +95,7 @@ export default {
|
||||||
switch (this.source) {
|
switch (this.source) {
|
||||||
case 'wy':
|
case 'wy':
|
||||||
case 'kw':
|
case 'kw':
|
||||||
case 'bd':
|
// case 'bd':
|
||||||
case 'tx':
|
case 'tx':
|
||||||
case 'mg':
|
case 'mg':
|
||||||
case 'kg':
|
case 'kg':
|
||||||
|
@ -179,11 +177,13 @@ export default {
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
...mapMutations(['setSongList']),
|
...mapMutations(['setSongList']),
|
||||||
...mapActions('songList', ['getTags', 'getList', 'getListDetail']),
|
...mapActions('songList', ['getTags', 'getList', 'getListDetail', 'getListDetailAll']),
|
||||||
...mapMutations('songList', ['setVisibleListDetail', 'setSelectListInfo']),
|
...mapMutations('songList', ['setVisibleListDetail', 'setSelectListInfo']),
|
||||||
...mapActions('download', ['createDownload', 'createDownloadMultiple']),
|
...mapActions('download', ['createDownload', 'createDownloadMultiple']),
|
||||||
...mapMutations('list', ['listAdd', 'listAddMultiple']),
|
...mapMutations('list', ['listAdd', 'listAddMultiple', 'createUserList']),
|
||||||
...mapMutations('player', ['setList']),
|
...mapMutations('player', {
|
||||||
|
setPlayList: 'setList',
|
||||||
|
}),
|
||||||
listenEvent() {
|
listenEvent() {
|
||||||
window.eventHub.$on('key_backspace_down', this.handle_key_backspace_down)
|
window.eventHub.$on('key_backspace_down', this.handle_key_backspace_down)
|
||||||
},
|
},
|
||||||
|
@ -273,7 +273,7 @@ export default {
|
||||||
s => s.songmid === targetSong.songmid,
|
s => s.songmid === targetSong.songmid,
|
||||||
)
|
)
|
||||||
if (targetIndex > -1) {
|
if (targetIndex > -1) {
|
||||||
this.setList({
|
this.setPlayList({
|
||||||
list: this.defaultList,
|
list: this.defaultList,
|
||||||
index: targetIndex,
|
index: targetIndex,
|
||||||
})
|
})
|
||||||
|
@ -402,15 +402,28 @@ export default {
|
||||||
assertApiSupport(source) {
|
assertApiSupport(source) {
|
||||||
return assertApiSupport(source)
|
return assertApiSupport(source)
|
||||||
},
|
},
|
||||||
/* addSongListDetail() {
|
async fetchList() {
|
||||||
// this.detailLoading = true
|
this.detailLoading = true
|
||||||
// this.getListDetailAll(this.selectListInfo.id).then(() => {
|
const list = await this.getListDetailAll(this.selectListInfo.id)
|
||||||
// this.detailLoading = false
|
this.detailLoading = false
|
||||||
// })
|
return list
|
||||||
|
},
|
||||||
|
async addSongListDetail() {
|
||||||
|
if (!this.listDetail.info.name) return
|
||||||
|
const list = await this.fetchList()
|
||||||
|
this.createUserList({ name: this.listDetail.info.name, id: `${this.listDetail.source}__${this.listDetail.id}`, list })
|
||||||
|
},
|
||||||
|
async playSongListDetail() {
|
||||||
|
if (!this.listDetail.info.name) return
|
||||||
|
const list = await this.fetchList()
|
||||||
|
this.setPlayList({
|
||||||
|
list: {
|
||||||
|
list,
|
||||||
|
id: null,
|
||||||
|
},
|
||||||
|
index: 0,
|
||||||
|
})
|
||||||
},
|
},
|
||||||
playSongListDetail() {
|
|
||||||
|
|
||||||
}, */
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
@ -517,6 +530,17 @@ export default {
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding-right: 15px;
|
padding-right: 15px;
|
||||||
}
|
}
|
||||||
|
.header-right-btn {
|
||||||
|
border-radius: 0;
|
||||||
|
&:first-child {
|
||||||
|
border-top-left-radius: 4px;
|
||||||
|
border-bottom-left-radius: 4px;
|
||||||
|
}
|
||||||
|
&:last-child {
|
||||||
|
border-top-right-radius: 4px;
|
||||||
|
border-bottom-right-radius: 4px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.song-list-detail-content {
|
.song-list-detail-content {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
|
|
Loading…
Reference in New Issue