Merge branch 'md' into dev
commit
a244c6d5e3
|
@ -578,3 +578,5 @@
|
||||||
@radius-progress-border: 5px;
|
@radius-progress-border: 5px;
|
||||||
|
|
||||||
@transition-theme: .4s ease;
|
@transition-theme: .4s ease;
|
||||||
|
|
||||||
|
@form-radius: 3px;
|
||||||
|
|
|
@ -153,6 +153,9 @@ export default {
|
||||||
'setting.player.togglePlayMethod'(n) {
|
'setting.player.togglePlayMethod'(n) {
|
||||||
this.audio.loop = n === 'singleLoop'
|
this.audio.loop = n === 'singleLoop'
|
||||||
},
|
},
|
||||||
|
'setting.player.mediaDeviceName'(n) {
|
||||||
|
this.setMediaDevice()
|
||||||
|
},
|
||||||
list(n, o) {
|
list(n, o) {
|
||||||
if (n === o) {
|
if (n === o) {
|
||||||
let index = this.listId == 'download'
|
let index = this.listId == 'download'
|
||||||
|
@ -192,8 +195,10 @@ export default {
|
||||||
]),
|
]),
|
||||||
...mapMutations(['setVolume']),
|
...mapMutations(['setVolume']),
|
||||||
...mapMutations('list', ['updateMusicInfo']),
|
...mapMutations('list', ['updateMusicInfo']),
|
||||||
|
...mapMutations(['setMediaDeviceId']),
|
||||||
init() {
|
init() {
|
||||||
this.audio = document.createElement('audio')
|
this.audio = document.createElement('audio')
|
||||||
|
this.setMediaDevice()
|
||||||
this.volume = this.audio.volume = this.setting.player.volume
|
this.volume = this.audio.volume = this.setting.player.volume
|
||||||
this.audio.controls = false
|
this.audio.controls = false
|
||||||
this.audio.autoplay = true
|
this.audio.autoplay = true
|
||||||
|
@ -572,6 +577,18 @@ export default {
|
||||||
this.mediaBuffer.timeout = null
|
this.mediaBuffer.timeout = null
|
||||||
this.mediaBuffer.playTime = 0
|
this.mediaBuffer.playTime = 0
|
||||||
},
|
},
|
||||||
|
async setMediaDevice() {
|
||||||
|
let mediaDeviceName = this.setting.player.mediaDeviceName
|
||||||
|
if (!mediaDeviceName) return
|
||||||
|
const devices = await navigator.mediaDevices.enumerateDevices()
|
||||||
|
let device = devices.find(device => device.label === mediaDeviceName)
|
||||||
|
if (!device) return this.setMediaDeviceId('default')
|
||||||
|
console.log(device)
|
||||||
|
this.audio.setSinkId(device.deviceId).catch((err) => {
|
||||||
|
console.log(err)
|
||||||
|
this.setMediaDeviceId('default')
|
||||||
|
})
|
||||||
|
},
|
||||||
handleSetTransition() {
|
handleSetTransition() {
|
||||||
this.isActiveTransition = true
|
this.isActiveTransition = true
|
||||||
// console.log('active transition')
|
// console.log('active transition')
|
||||||
|
|
|
@ -24,7 +24,7 @@ export default {
|
||||||
.btn {
|
.btn {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
border: none;
|
border: none;
|
||||||
border-radius: 3px;
|
border-radius: @form-radius;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
padding: 8px 15px;
|
padding: 8px 15px;
|
||||||
color: @color-btn;
|
color: @color-btn;
|
||||||
|
|
|
@ -78,7 +78,7 @@ export default {
|
||||||
button {
|
button {
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
border: none;
|
border: none;
|
||||||
border-radius: 3px;
|
border-radius: @form-radius;
|
||||||
margin-right: 5px;
|
margin-right: 5px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
padding: 4px 7px;
|
padding: 4px 7px;
|
||||||
|
|
|
@ -49,7 +49,7 @@ export default {
|
||||||
.input {
|
.input {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
border: none;
|
border: none;
|
||||||
border-radius: 3px;
|
border-radius: @form-radius;
|
||||||
padding: 7px 8px;
|
padding: 7px 8px;
|
||||||
color: @color-btn;
|
color: @color-btn;
|
||||||
outline: none;
|
outline: none;
|
||||||
|
|
|
@ -91,7 +91,7 @@ export default {
|
||||||
button {
|
button {
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
border: none;
|
border: none;
|
||||||
border-radius: 3px;
|
border-radius: @form-radius;
|
||||||
margin-right: 5px;
|
margin-right: 5px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
padding: 4px 7px;
|
padding: 4px 7px;
|
||||||
|
|
|
@ -133,7 +133,7 @@ export default {
|
||||||
@import '../../assets/styles/layout.less';
|
@import '../../assets/styles/layout.less';
|
||||||
|
|
||||||
.search {
|
.search {
|
||||||
border-radius: 3px;
|
border-radius: @form-radius;
|
||||||
transition: box-shadow .4s ease, background-color @transition-theme;
|
transition: box-shadow .4s ease, background-color @transition-theme;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-flow: column nowrap;
|
flex-flow: column nowrap;
|
||||||
|
|
|
@ -0,0 +1,183 @@
|
||||||
|
<template lang="pug">
|
||||||
|
div(:class="[$style.select, show ? $style.active : '']")
|
||||||
|
div(:class="$style.label" ref="dom_btn" @click="handleShow") {{label}}
|
||||||
|
ul(:class="$style.list")
|
||||||
|
li(v-for="item in list" :class="(itemKey ? item[itemKey] : item) == value ? $style.active : null" @click="handleClick(item)" :title="itemName ? item[itemName] : item") {{itemName ? item[itemName] : item}}
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// import { isChildren } from '../../utils'
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
list: {
|
||||||
|
type: Array,
|
||||||
|
default() {
|
||||||
|
return []
|
||||||
|
},
|
||||||
|
},
|
||||||
|
value: {
|
||||||
|
type: [String, Number],
|
||||||
|
},
|
||||||
|
itemName: {
|
||||||
|
type: String,
|
||||||
|
},
|
||||||
|
itemKey: {
|
||||||
|
type: String,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
show: false,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
document.addEventListener('click', this.handleHide)
|
||||||
|
},
|
||||||
|
beforeDestroy() {
|
||||||
|
document.removeEventListener('click', this.handleHide)
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
label() {
|
||||||
|
if (!this.value) return ''
|
||||||
|
if (!this.itemName) return this.value
|
||||||
|
const item = this.list.find(l => l[this.itemKey] == this.value)
|
||||||
|
if (!item) return ''
|
||||||
|
return item[this.itemName]
|
||||||
|
},
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
handleHide(e) {
|
||||||
|
// if (e && e.target.parentNode != this.$refs.dom_list && this.show) return this.show = false
|
||||||
|
if (e && e.target == this.$refs.dom_btn) return
|
||||||
|
setTimeout(() => {
|
||||||
|
this.show = false
|
||||||
|
}, 50)
|
||||||
|
},
|
||||||
|
handleClick(item) {
|
||||||
|
console.log(this.value)
|
||||||
|
if (item === this.value) return
|
||||||
|
this.$emit('input', this.itemKey ? item[this.itemKey] : item)
|
||||||
|
this.$emit('change', item)
|
||||||
|
},
|
||||||
|
handleShow() {
|
||||||
|
this.show = !this.show
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
|
||||||
|
<style lang="less" module>
|
||||||
|
@import '../../assets/styles/layout.less';
|
||||||
|
|
||||||
|
@selection-height: 28px;
|
||||||
|
|
||||||
|
.select {
|
||||||
|
font-size: 12px;
|
||||||
|
position: relative;
|
||||||
|
width: 300px;
|
||||||
|
|
||||||
|
&.active {
|
||||||
|
.label {
|
||||||
|
background-color: @color-btn-background;
|
||||||
|
}
|
||||||
|
.list {
|
||||||
|
transform: scaleY(1);
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.label {
|
||||||
|
background-color: @color-btn-background;
|
||||||
|
padding: 0 10px;
|
||||||
|
transition: background-color @transition-theme;
|
||||||
|
height: @selection-height;
|
||||||
|
line-height: 27px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
color: @color-btn;
|
||||||
|
border-radius: @form-radius;
|
||||||
|
cursor: pointer;
|
||||||
|
.mixin-ellipsis-1;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: @color-theme_2-hover;
|
||||||
|
}
|
||||||
|
&:active {
|
||||||
|
background-color: @color-theme_2-active;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.list {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
background-color: @color-theme_2-background_2;
|
||||||
|
opacity: 0;
|
||||||
|
transform: scaleY(0);
|
||||||
|
transform-origin: 0 @selection-height / 2 0;
|
||||||
|
transition: .25s ease;
|
||||||
|
transition-property: transform, opacity;
|
||||||
|
z-index: 10;
|
||||||
|
border-radius: @form-radius;
|
||||||
|
overflow: hidden;
|
||||||
|
box-shadow: 0 1px 4px rgba(0, 0, 0, .12);
|
||||||
|
|
||||||
|
li {
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0 10px;
|
||||||
|
line-height: @selection-height;
|
||||||
|
// color: @color-btn;
|
||||||
|
outline: none;
|
||||||
|
transition: background-color @transition-theme;
|
||||||
|
background-color: @color-btn-background;
|
||||||
|
box-sizing: border-box;
|
||||||
|
.mixin-ellipsis-1;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: @color-theme_2-hover;
|
||||||
|
}
|
||||||
|
&:active {
|
||||||
|
background-color: @color-theme_2-active;
|
||||||
|
}
|
||||||
|
&.active {
|
||||||
|
color: @color-btn;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
each(@themes, {
|
||||||
|
:global(#container.@{value}) {
|
||||||
|
.label {
|
||||||
|
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}';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.list {
|
||||||
|
background-color: ~'@{color-@{value}-theme_2-background_2}';
|
||||||
|
li {
|
||||||
|
// 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}';
|
||||||
|
}
|
||||||
|
&.active {
|
||||||
|
color: ~'@{color-@{value}-btn}';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
</style>
|
|
@ -26,6 +26,8 @@
|
||||||
"play_quality": "优先播放高品质音乐",
|
"play_quality": "优先播放高品质音乐",
|
||||||
"play_task_bar_title": "在任务栏上显示当前歌曲播放进度",
|
"play_task_bar_title": "在任务栏上显示当前歌曲播放进度",
|
||||||
"play_task_bar": "任务栏播放进度条",
|
"play_task_bar": "任务栏播放进度条",
|
||||||
|
"play_mediaDevice_title": "选择声音输出的媒体设备",
|
||||||
|
"play_mediaDevice": "音频输出",
|
||||||
|
|
||||||
"list": "列表设置",
|
"list": "列表设置",
|
||||||
"list_source_title": "是否显示歌曲源",
|
"list_source_title": "是否显示歌曲源",
|
||||||
|
|
|
@ -25,6 +25,8 @@
|
||||||
"play_quality": "優先播放高品質音樂",
|
"play_quality": "優先播放高品質音樂",
|
||||||
"play_task_bar_title": "在任務欄上顯示當前歌曲播放進度",
|
"play_task_bar_title": "在任務欄上顯示當前歌曲播放進度",
|
||||||
"play_task_bar": "任務欄播放進度條",
|
"play_task_bar": "任務欄播放進度條",
|
||||||
|
"play_mediaDevice_title": "選擇聲音輸出的媒體設備",
|
||||||
|
"play_mediaDevice": "音頻輸出",
|
||||||
"list": "列表設置",
|
"list": "列表設置",
|
||||||
"list_source_title": "是否顯示歌曲源",
|
"list_source_title": "是否顯示歌曲源",
|
||||||
"list_source": "是否顯示歌曲源(僅對我的音樂分類有效)",
|
"list_source": "是否顯示歌曲源(僅對我的音樂分類有效)",
|
||||||
|
|
|
@ -26,6 +26,8 @@
|
||||||
"play_quality": "Prioritize high-quality music",
|
"play_quality": "Prioritize high-quality music",
|
||||||
"play_task_bar_title": "Show current song playback progress on taskbar",
|
"play_task_bar_title": "Show current song playback progress on taskbar",
|
||||||
"play_task_bar": "Taskbar playback progress bar",
|
"play_task_bar": "Taskbar playback progress bar",
|
||||||
|
"play_mediaDevice_title": "Select the media device for sound output",
|
||||||
|
"play_mediaDevice": "Audio output",
|
||||||
|
|
||||||
"list": "List settings",
|
"list": "List settings",
|
||||||
"list_source_title": "Whether to show song sources",
|
"list_source_title": "Whether to show song sources",
|
||||||
|
|
|
@ -42,4 +42,7 @@ export default {
|
||||||
setVolume(state, val) {
|
setVolume(state, val) {
|
||||||
state.setting.player.volume = val
|
state.setting.player.volume = val
|
||||||
},
|
},
|
||||||
|
setMediaDeviceId(state, val) {
|
||||||
|
state.setting.player.mediaDeviceId = val
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
|
@ -177,7 +177,7 @@ export const isChildren = (parent, children) => {
|
||||||
* @param {*} setting
|
* @param {*} setting
|
||||||
*/
|
*/
|
||||||
export const updateSetting = (setting, version) => {
|
export const updateSetting = (setting, version) => {
|
||||||
const defaultVersion = '1.0.16'
|
const defaultVersion = '1.0.17'
|
||||||
if (!version) {
|
if (!version) {
|
||||||
if (setting) {
|
if (setting) {
|
||||||
version = setting.version
|
version = setting.version
|
||||||
|
@ -190,6 +190,7 @@ export const updateSetting = (setting, version) => {
|
||||||
highQuality: false,
|
highQuality: false,
|
||||||
isShowTaskProgess: true,
|
isShowTaskProgess: true,
|
||||||
volume: 1,
|
volume: 1,
|
||||||
|
mediaDeviceId: 'default',
|
||||||
},
|
},
|
||||||
list: {
|
list: {
|
||||||
isShowAlbumName: true,
|
isShowAlbumName: true,
|
||||||
|
|
|
@ -0,0 +1,30 @@
|
||||||
|
import { httpFetch } from '../../request'
|
||||||
|
|
||||||
|
|
||||||
|
export default {
|
||||||
|
regExps: {
|
||||||
|
relWord: /RELWORD=(.+)/,
|
||||||
|
},
|
||||||
|
requestObj: null,
|
||||||
|
tempSearch(str) {
|
||||||
|
this.cancelTempSearch()
|
||||||
|
this.requestObj = httpFetch(`https://c.y.qq.com/splcloud/fcgi-bin/smartbox_new.fcg?is_xml=0&format=json&key=${encodeURIComponent(str)}&loginUin=0&hostUin=0&format=json&inCharset=utf8&outCharset=utf-8¬ice=0&platform=yqq&needNewCode=0`, {
|
||||||
|
headers: {
|
||||||
|
Referer: 'https://y.qq.com/portal/player.html',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
return this.requestObj.promise.then(({ statusCode, body }) => {
|
||||||
|
if (statusCode != 200 || body.code != 0) return Promise.reject(new Error('请求失败'))
|
||||||
|
return body.data
|
||||||
|
})
|
||||||
|
},
|
||||||
|
handleResult(rawData) {
|
||||||
|
return rawData.map(info => `${info.name} - ${info.singer}`)
|
||||||
|
},
|
||||||
|
cancelTempSearch() {
|
||||||
|
if (this.requestObj && this.requestObj.cancelHttp) this.requestObj.cancelHttp()
|
||||||
|
},
|
||||||
|
async search(str) {
|
||||||
|
return this.tempSearch(str).then(result => this.handleResult(result.song.itemlist))
|
||||||
|
},
|
||||||
|
}
|
|
@ -49,6 +49,11 @@ div.scroll(:class="$style.setting")
|
||||||
h3 {{$t('view.setting.play_task_bar')}}
|
h3 {{$t('view.setting.play_task_bar')}}
|
||||||
div
|
div
|
||||||
material-checkbox(id="setting_player_showTaskProgess" v-model="current_setting.player.isShowTaskProgess" :label="$t('view.setting.is_enable')")
|
material-checkbox(id="setting_player_showTaskProgess" v-model="current_setting.player.isShowTaskProgess" :label="$t('view.setting.is_enable')")
|
||||||
|
dd(:title="$t('view.setting.play_mediaDevice_title')")
|
||||||
|
h3 {{$t('view.setting.play_mediaDevice')}}
|
||||||
|
div
|
||||||
|
material-selection(:list="mediaDevices" @change="handleMediaDeviceChange" v-model="current_setting.player.mediaDeviceId" item-key="deviceId" item-name="label")
|
||||||
|
|
||||||
dt {{$t('view.setting.list')}}
|
dt {{$t('view.setting.list')}}
|
||||||
dd(:title="$t('view.setting.list_source_title')")
|
dd(:title="$t('view.setting.list_source_title')")
|
||||||
h3 {{$t('view.setting.list_source')}}
|
h3 {{$t('view.setting.list_source')}}
|
||||||
|
@ -283,6 +288,8 @@ export default {
|
||||||
togglePlayMethod: 'random',
|
togglePlayMethod: 'random',
|
||||||
highQuality: false,
|
highQuality: false,
|
||||||
isShowTaskProgess: true,
|
isShowTaskProgess: true,
|
||||||
|
volume: 1,
|
||||||
|
mediaDeviceId: 'default',
|
||||||
},
|
},
|
||||||
list: {
|
list: {
|
||||||
isShowAlbumName: true,
|
isShowAlbumName: true,
|
||||||
|
@ -320,6 +327,7 @@ export default {
|
||||||
},
|
},
|
||||||
languageList,
|
languageList,
|
||||||
cacheSize: '0 B',
|
cacheSize: '0 B',
|
||||||
|
mediaDevices: [],
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
|
@ -346,10 +354,12 @@ export default {
|
||||||
methods: {
|
methods: {
|
||||||
...mapMutations(['setSetting', 'setSettingVersion', 'setVersionModalVisible']),
|
...mapMutations(['setSetting', 'setSettingVersion', 'setVersionModalVisible']),
|
||||||
...mapMutations('list', ['setList']),
|
...mapMutations('list', ['setList']),
|
||||||
|
...mapMutations(['setMediaDeviceId']),
|
||||||
init() {
|
init() {
|
||||||
this.current_setting = JSON.parse(JSON.stringify(this.setting))
|
this.current_setting = JSON.parse(JSON.stringify(this.setting))
|
||||||
if (!window.currentWindowSizeId) window.currentWindowSizeId = this.setting.windowSizeId
|
if (!window.currentWindowSizeId) window.currentWindowSizeId = this.setting.windowSizeId
|
||||||
this.getCacheSize()
|
this.getCacheSize()
|
||||||
|
this.getMediaDevice()
|
||||||
},
|
},
|
||||||
handleChangeSavePath() {
|
handleChangeSavePath() {
|
||||||
selectDir({
|
selectDir({
|
||||||
|
@ -552,6 +562,15 @@ export default {
|
||||||
handleLangChange(id) {
|
handleLangChange(id) {
|
||||||
this.$i18n.locale = id
|
this.$i18n.locale = id
|
||||||
},
|
},
|
||||||
|
async getMediaDevice() {
|
||||||
|
const devices = await navigator.mediaDevices.enumerateDevices()
|
||||||
|
const audioDevices = devices.filter(device => device.kind === 'audiooutput')
|
||||||
|
this.mediaDevices = audioDevices
|
||||||
|
console.log(this.mediaDevices)
|
||||||
|
},
|
||||||
|
handleMediaDeviceChange(audioDevice) {
|
||||||
|
this.setMediaDeviceId(audioDevice.deviceId)
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
Loading…
Reference in New Issue