This repository has been archived on 2023-08-30. You can view files and clone it, but cannot push or open issues/pull-requests.
appcat
/
sonist
Archived
1
0
Fork 0
sonist/src/js/app.js

538 lines
12 KiB
JavaScript

/**
* {sonist app}
* @author yutent<yutent@doui.cc>
* @date 2018/12/16 17:15:57
*/
import '/lib/anot.js'
import '/lib/layer/index.js'
import '/lib/store/index.js'
import AudioPlayer from '/lib/audio/index.js'
import Lyrics from '/lib/lyrics/index.js'
import Api from '/js/api.js'
import Artist from '/js/modules/artist.js'
import Local from '/js/modules/local.js'
import Profile from '/js/modules/profile.js'
import Search from '/js/modules/search.js'
import KTV from '/js/modules/ktv.js'
import PLAYCTRL from '/js/modules/play-ctrl.js'
const log = console.log
const fs = require('iofs')
const path = require('path')
const { remote, ipcRenderer, screen } = require('electron')
const { createDesktopLrcWindow, createMiniWindow } = remote.require(
'./tools/windows'
)
const MAIN_SCREEN = screen.getPrimaryDisplay()
const WIN = remote.getCurrentWindow()
const __LRC__ = createDesktopLrcWindow(MAIN_SCREEN)
const __MINI__ = createMiniWindow(MAIN_SCREEN)
const PLAY_MODE = {
0: 'all',
1: 'single',
2: 'random'
}
// window.onblur = function() {
// location.reload()
// }
// 本地音乐和试用音乐列表
window.LS = store.collection('local')
window.TS = store.collection('temp')
// 音乐播放器
window.SONIST = new AudioPlayer()
window.LYRICS = new Lyrics()
let appInit = ipcRenderer.sendSync('sonist', { type: 'get-init' })
Anot.ss('app-init', appInit)
SONIST.target = 'local' //播放目标是本地音乐
Anot({
$id: 'app',
state: {
theme: appInit.theme || 1, // 1:macos, 2: deepin
winFocus: false,
winShow: true,
mod: 'local',
searchTxt: '',
playMode: Anot.ls('play-mode') >>> 0, // 0:all | 1:single | 2:random
ktvMode: 0,
isPlaying: false,
optBoxShow: false,
volumeCtrlShow: false,
volume: Anot.ls('volume') || 70,
muted: false,
curr: {
id: '',
cover: '',
title: '',
artist: '',
album: '',
time: 0,
duration: 0
},
ctrlLrc: '暂无歌词...',
...KTV.data,
loading: false,
progress: 0
},
skip: ['winShow'],
computed: {
views() {
if (!this.mod) {
return
}
return '/views/' + this.mod + '.htm'
},
coverBG() {
if (this.curr.cover) {
return `url(${this.curr.cover})`
} else {
return 'none'
}
}
},
watch: {
mod(val) {
this.activeModule(val)
}
},
mounted() {
let canvas = this.$refs.player
// 画布放大4倍, 以解决模糊的问题
this.__WIDTH__ = canvas.clientWidth * 4
this.__HEIGHT__ = canvas.clientHeight * 4
canvas.width = this.__WIDTH__
canvas.height = this.__HEIGHT__
this.__CTX__ = canvas.getContext('2d')
this.draw(true)
// 修改歌曲进度
canvas.addEventListener(
'click',
ev => {
if (!this.curr.id) {
return
}
let rect = canvas.getBoundingClientRect()
let aw = rect.width
let ax = ev.pageX - rect.left
let ay = ev.pageY - rect.top
if (ax < 80) {
this.ktvMode = this.ktvMode ^ 1
if (!this.isPlaying) {
this.draw()
}
return
}
if (ax > 124 && ay > 55 && ay < 64) {
let pp = (ax - 124) / (aw - 124)
this.curr.time = pp * this.curr.duration
SONIST.seek(this.curr.time)
LYRICS.seek(this.curr.time)
if (!this.isPlaying) {
this.draw()
}
}
},
false
)
// 设置循环模式
SONIST.mode = PLAY_MODE[this.playMode]
SONIST.volume = this.volume
SONIST.on('play', time => {
this.curr.time = time
LYRICS.update(time)
})
SONIST.on('end', time => {
this.nextSong(1)
})
// 控制条的单行歌词
LYRICS.on('ctrl-lrc', lrc => {
this.ctrlLrc = lrc
})
// ktv模式的歌词
LYRICS.on('ktv-lrc', lrc => {
this.lrc = lrc
__LRC__.emit('ktv-lrc', lrc)
})
// ktv模式的歌词
LYRICS.on('view-all', lrc => {
this.allLrc = lrc
})
this.activeModule(this.mod)
remote.app.on('browser-window-focus', _ => {
this.winFocus = true
})
remote.app.on('browser-window-blur', _ => {
this.winFocus = false
})
/**
* 响应mini窗口的事件
*/
WIN.on('mini-ctrl', ev => {
switch (ev) {
case 'prev':
this.nextSong(-1)
break
case 'play':
this.play()
break
case 'next':
this.nextSong(1)
break
case 'desktoplrc':
this.toggleDesktopLrc()
break
default:
if (ev.name === 'play-mode') {
this.playMode = ev.value
SONIST.mode = PLAY_MODE[ev.value]
Anot.ls('play-mode', ev.value)
}
}
})
/**
* 响应 全局快捷键的事件
*/
WIN.on('gs-ctrl', ev => {
switch (ev) {
case 'prev':
this.nextSong(-1)
break
case 'play':
this.play()
break
case 'next':
this.nextSong(1)
break
case 'stop':
this.isPlaying = false
this.curr.time = 0
SONIST.seek(0)
LYRICS.seek(0)
SONIST.pause()
this.draw()
break
case 'vu':
this.volume += 5
if (this.volume >= 100) {
this.volume = 100
}
SONIST.volume = this.volume
break
case 'vd':
this.volume -= 5
if (this.volume <= 0) {
this.volume = 0
}
SONIST.volume = this.volume
break
case 'lrc':
this.toggleDesktopLrc()
break
case 'mini':
this.change2mini()
break
default:
break
}
})
// 迷你模式开启时, 不响应托盘和dock栏的点击事件
ipcRenderer.on('dock-click', () => {
if (!__MINI__.isVisible()) {
WIN.show()
}
})
Anot(document).bind('keydown', ev => {
if (ev.target === document.body) {
switch (ev.keyCode) {
case 32: // 空格
this.play()
break
case 37: // 向左 (prev)
if (ev.metaKey) {
this.nextSong(-1)
}
break
case 39: // 向右 (next)
if (ev.metaKey) {
this.nextSong(1)
}
break
case 38: // 向上 (音量+
if (ev.metaKey) {
this.volume += 5
if (this.volume >= 100) {
this.volume = 100
}
SONIST.volume = this.volume
}
break
case 40: // 向下 (音量-
if (ev.metaKey) {
this.volume -= 5
if (this.volume <= 0) {
this.volume = 0
}
SONIST.volume = this.volume
}
break
case 77: // M (迷你模式)
if (ev.metaKey && ev.altKey) {
this.change2mini()
}
break
case 82: // R (桌面歌词)
if (ev.metaKey) {
this.toggleDesktopLrc()
}
break
}
}
})
WIN.on('show', ev => {
this.winShow = true
this.draw()
})
WIN.on('hide', ev => {
this.winShow = false
this.draw()
})
},
methods: {
quit(force) {
if (force) {
remote.app.exit()
} else {
if (appInit.allowPlayOnBack) {
WIN.hide()
} else {
remote.app.exit()
}
}
},
minimize() {
WIN.minimize()
},
change2mini() {
this.optBoxShow = false
WIN.hide()
__MINI__.show()
let song = this.curr.$model
if (!this.isPlaying) {
delete song.id
}
__MINI__.emit('mini-init', song)
},
activeModule(mod) {
switch (mod) {
case 'artist':
Artist.__init__()
break
case 'local':
Local.__init__()
break
case 'profile':
Profile.__init__()
break
case 'search':
Search.__init__()
break
default:
break
}
},
toggleOptBox() {
this.optBoxShow = !this.optBoxShow
},
toggleDesktopLrc() {
if (__LRC__.isVisible()) {
__LRC__.hide()
} else {
__LRC__.showInactive()
}
},
toggleModule(mod) {
if ('mv' === mod) {
return
}
this.optBoxShow = false
this.__last__ = this.mod
this.mod = mod
},
// 设置保存 回调
onProfileSaved() {
this.toggleModule(this.__last__)
appInit = JSON.parse(Anot.ss('app-init'))
},
// 切换循环模式
togglePlayMode() {
let mod = this.playMode
mod++
if (mod > 2) {
mod = 0
}
this.playMode = mod
SONIST.mode = PLAY_MODE[mod]
Anot.ls('play-mode', mod)
},
// 修改音量
changeValume(ev) {
let volume = 575 - ev.pageY
if (volume < 0) {
volume = 0
}
if (volume > 100) {
volume = 100
}
this.volume = volume
SONIST.volume = volume
Anot.ls('volume', volume)
},
searchMusic(ev) {
if (ev.keyCode === 13) {
let txt = this.searchTxt.trim()
if (!txt) {
return
}
if (txt === ':debug:') {
log('----- 调试模式 -----')
this.searchTxt = ''
WIN.openDevTools()
return
}
if (this.mod !== 'search') {
this.toggleModule('search')
}
Search.search(txt)
this.searchTxt = ''
// layer.toast('搜索功能还未开放')
}
},
/**
* 播放按钮的事件
*/
handleCtrl(ev) {
let key = ev.target.dataset.key
switch (key) {
case 'prev':
this.nextSong(-1)
break
case 'play':
this.play()
break
case 'next':
this.nextSong(1)
break
default:
break
}
},
nextSong(step) {
let _p = null
if (step > 0) {
_p = SONIST.next()
} else {
_p = SONIST.prev()
}
this.isPlaying = false
_p.then(it => {
if (this.mod === 'local') {
Local.__updateSong__(it)
}
// 通知子模块歌曲已经改变
this.$fire('child!curr', it.id)
this.play(it)
})
},
updateCurr(obj) {
let old = this.curr.$model
this.curr = Object.assign(old, obj)
__MINI__.emit('mini-init', this.curr.$model)
},
play(song) {
// 有参数的,说明是播放回调通知
// 此时仅更新播放控制条的信息即可
if (song) {
song.time = 0
this.ctrlLrc = '暂无歌词...'
this.updateCurr(song)
this.isPlaying = true
this.draw(true)
LYRICS.__init__(song.id)
} else {
if (SONIST.stat === 'ready') {
let played = this.isPlaying
this.isPlaying = !this.isPlaying
if (this.curr.id) {
if (played) {
SONIST.pause()
} else {
SONIST.play()
}
this.draw()
} else {
let lastPlay = Anot.ls('last-play') || 0
SONIST.play(lastPlay).then(it => {
it.time = 0
this.ctrlLrc = '暂无歌词...'
this.updateCurr(it)
this.draw(true)
// this.ktvMode = 1
LYRICS.__init__(it.id)
})
}
if (!this.winShow) {
__MINI__.emit('mini-ctrl', this.isPlaying ? 'play' : 'pause')
}
}
}
},
...PLAYCTRL.methods,
...KTV.methods
}
})
一个音乐播放器, 主打本地音乐播放。支持 自动歌词/自动封面/均衡器等常见功能。
JavaScript 60.1%
SCSS 19.2%
HTML 16.9%
CSS 3.8%