2015-12-13 04:42:28 +00:00
|
|
|
<template>
|
2016-06-25 16:05:24 +00:00
|
|
|
<div id="app">
|
|
|
|
<div id="main" tabindex="0" v-show="authenticated"
|
|
|
|
@keydown.space="togglePlayback"
|
|
|
|
@keydown.j = "playNext"
|
|
|
|
@keydown.k = "playPrev"
|
|
|
|
@keydown.f = "search"
|
2016-07-08 08:24:41 +00:00
|
|
|
@keydown.mediaPrev = "playPrev"
|
|
|
|
@keydown.mediaNext = "playNext"
|
|
|
|
@keydown.mediaToggle = "togglePlayback"
|
2016-06-25 16:05:24 +00:00
|
|
|
>
|
2016-10-31 04:28:12 +00:00
|
|
|
<site-header/>
|
|
|
|
<main-wrapper/>
|
|
|
|
<site-footer/>
|
|
|
|
<overlay ref="overlay"/>
|
|
|
|
<edit-songs-form ref="editSongsForm"/>
|
2015-12-29 01:16:36 +00:00
|
|
|
</div>
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
<div class="login-wrapper" v-if="!authenticated">
|
2016-10-31 04:28:12 +00:00
|
|
|
<login-form/>
|
2016-06-25 16:05:24 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
2015-12-13 04:42:28 +00:00
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
2016-11-26 03:25:35 +00:00
|
|
|
import Vue from 'vue'
|
|
|
|
|
|
|
|
import siteHeader from './components/site-header/index.vue'
|
|
|
|
import siteFooter from './components/site-footer/index.vue'
|
|
|
|
import mainWrapper from './components/main-wrapper/index.vue'
|
|
|
|
import overlay from './components/shared/overlay.vue'
|
|
|
|
import loginForm from './components/auth/login-form.vue'
|
|
|
|
import editSongsForm from './components/modals/edit-songs-form.vue'
|
|
|
|
|
2016-12-20 15:44:47 +00:00
|
|
|
import { event, showOverlay, hideOverlay, forceReloadWindow, $ } from './utils'
|
2016-11-26 03:25:35 +00:00
|
|
|
import { sharedStore, userStore, preferenceStore as preferences } from './stores'
|
|
|
|
import { playback, ls } from './services'
|
|
|
|
import { focusDirective, clickawayDirective } from './directives'
|
|
|
|
import router from './router'
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
export default {
|
|
|
|
components: { siteHeader, siteFooter, mainWrapper, overlay, loginForm, editSongsForm },
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
data () {
|
2016-06-25 16:05:24 +00:00
|
|
|
return {
|
2016-11-26 03:25:35 +00:00
|
|
|
authenticated: false
|
|
|
|
}
|
2016-06-25 16:05:24 +00:00
|
|
|
},
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
mounted () {
|
2016-06-25 16:05:24 +00:00
|
|
|
// The app has just been initialized, check if we can get the user data with an already existing token
|
2016-11-26 03:25:35 +00:00
|
|
|
const token = ls.get('jwt-token')
|
2016-06-25 16:05:24 +00:00
|
|
|
if (token) {
|
2016-11-26 03:25:35 +00:00
|
|
|
this.authenticated = true
|
|
|
|
this.init()
|
2016-06-25 16:05:24 +00:00
|
|
|
}
|
2015-12-13 04:42:28 +00:00
|
|
|
|
2016-06-25 16:05:24 +00:00
|
|
|
// Create the element to be the ghost drag image.
|
2016-12-20 15:44:47 +00:00
|
|
|
const dragGhost = document.createElement('div')
|
|
|
|
dragGhost.id = 'dragGhost'
|
|
|
|
document.body.appendChild(dragGhost)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
2016-07-07 13:54:20 +00:00
|
|
|
// And the textarea to copy stuff
|
2016-12-20 15:44:47 +00:00
|
|
|
const copyArea = document.createElement('textarea')
|
|
|
|
copyArea.id = 'copyArea'
|
|
|
|
document.body.appendChild(copyArea)
|
2016-07-07 13:54:20 +00:00
|
|
|
|
2016-06-25 16:05:24 +00:00
|
|
|
// Add an ugly mac/non-mac class for OS-targeting styles.
|
|
|
|
// I'm crying inside.
|
2016-12-20 15:44:47 +00:00
|
|
|
$.addClass(document.documentElement, navigator.userAgent.indexOf('Mac') !== -1 ? 'mac' : 'non-mac')
|
2016-06-25 16:05:24 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
methods: {
|
2016-11-26 03:25:35 +00:00
|
|
|
init () {
|
|
|
|
showOverlay()
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
// Make the most important HTTP request to get all necessary data from the server.
|
|
|
|
// Afterwards, init all mandatory stores and services.
|
2016-06-27 06:11:35 +00:00
|
|
|
sharedStore.init().then(() => {
|
2016-11-26 03:25:35 +00:00
|
|
|
playback.init()
|
|
|
|
hideOverlay()
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
// Ask for user's notification permission.
|
2016-11-26 03:25:35 +00:00
|
|
|
this.requestNotifPermission()
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
// To confirm or not to confirm closing, it's a question.
|
|
|
|
window.onbeforeunload = e => {
|
|
|
|
if (!preferences.confirmClosing) {
|
2016-11-26 03:25:35 +00:00
|
|
|
return
|
2016-06-25 16:05:24 +00:00
|
|
|
}
|
|
|
|
|
2016-10-11 11:50:03 +00:00
|
|
|
// Notice that a custom message like this has ceased to be supported
|
|
|
|
// starting from Chrome 51.
|
2016-11-26 03:25:35 +00:00
|
|
|
return 'You asked Koel to confirm before closing, so here it is.'
|
|
|
|
}
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
// Let all other components know we're ready.
|
2016-11-26 03:25:35 +00:00
|
|
|
event.emit('koel:ready')
|
|
|
|
}).catch(() => {
|
|
|
|
this.authenticated = false
|
|
|
|
})
|
2016-06-25 16:05:24 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Toggle playback when user presses Space key.
|
|
|
|
*
|
|
|
|
* @param {Object} e The keydown event
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
togglePlayback (e) {
|
2016-12-20 15:44:47 +00:00
|
|
|
if ($.is(e.target, 'input,textarea,button,select')) {
|
2016-11-26 03:25:35 +00:00
|
|
|
return true
|
2016-06-25 16:05:24 +00:00
|
|
|
}
|
|
|
|
|
2016-12-20 15:44:47 +00:00
|
|
|
// Whatever play/pause control is there, we blindly click it.
|
|
|
|
const play = document.querySelector('#mainFooter .play')
|
|
|
|
play ? play.click() : document.querySelector('#mainFooter .pause').click()
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
e.preventDefault()
|
2016-06-25 16:05:24 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Play the previous song when user presses K.
|
|
|
|
*
|
|
|
|
* @param {Object} e The keydown event
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
playPrev (e) {
|
2016-12-20 15:44:47 +00:00
|
|
|
if ($.is(e.target, 'input,textarea')) {
|
2016-11-26 03:25:35 +00:00
|
|
|
return true
|
2016-06-25 16:05:24 +00:00
|
|
|
}
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
playback.playPrev()
|
|
|
|
e.preventDefault()
|
2016-06-25 16:05:24 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Play the next song when user presses J.
|
|
|
|
*
|
|
|
|
* @param {Object} e The keydown event
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
playNext (e) {
|
2016-12-20 15:44:47 +00:00
|
|
|
if ($.is(e.target, 'input,textarea')) {
|
2016-11-26 03:25:35 +00:00
|
|
|
return true
|
2016-06-25 16:05:24 +00:00
|
|
|
}
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
playback.playNext()
|
|
|
|
e.preventDefault()
|
2016-06-25 16:05:24 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Put focus into the search field when user presses F.
|
|
|
|
*
|
|
|
|
* @param {Object} e The keydown event
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
search (e) {
|
2016-12-20 15:44:47 +00:00
|
|
|
if ($.is(e.target, 'input,textarea') || e.metaKey || e.ctrlKey) {
|
2016-11-26 03:25:35 +00:00
|
|
|
return true
|
2016-06-25 16:05:24 +00:00
|
|
|
}
|
|
|
|
|
2016-12-20 15:44:47 +00:00
|
|
|
const selectBox = document.querySelector('#searchForm input[type="search"]')
|
|
|
|
selectBox.focus()
|
|
|
|
selectBox.select()
|
2016-11-26 03:25:35 +00:00
|
|
|
e.preventDefault()
|
2016-06-25 16:05:24 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Request for notification permission if it's not provided and the user is OK with notifs.
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
requestNotifPermission () {
|
|
|
|
if (window.Notification && preferences.notify && window.Notification.permission !== 'granted') {
|
|
|
|
window.Notification.requestPermission(result => {
|
2016-06-25 16:05:24 +00:00
|
|
|
if (result === 'denied') {
|
2016-11-26 03:25:35 +00:00
|
|
|
preferences.notify = false
|
2016-06-25 16:05:24 +00:00
|
|
|
}
|
2016-11-26 03:25:35 +00:00
|
|
|
})
|
2016-06-25 16:05:24 +00:00
|
|
|
}
|
2016-11-26 03:25:35 +00:00
|
|
|
}
|
2016-06-25 16:05:24 +00:00
|
|
|
},
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
created () {
|
2016-06-25 16:05:24 +00:00
|
|
|
event.on({
|
|
|
|
/**
|
|
|
|
* When the user logs in, set the whole app to be "authenticated" and initialize it.
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
'user:loggedin': () => {
|
|
|
|
this.authenticated = true
|
|
|
|
this.init()
|
2016-06-25 16:05:24 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shows the "Edit Song" form.
|
|
|
|
*
|
|
|
|
* @param {Array.<Object>} An array of songs to edit
|
|
|
|
*/
|
|
|
|
'songs:edit': songs => this.$refs.editSongsForm.open(songs),
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Log the current user out and reset the application state.
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
logout () {
|
2016-06-27 06:11:35 +00:00
|
|
|
userStore.logout().then((r) => {
|
2016-11-26 03:25:35 +00:00
|
|
|
ls.remove('jwt-token')
|
|
|
|
forceReloadWindow()
|
|
|
|
})
|
2016-06-25 16:05:24 +00:00
|
|
|
},
|
2016-07-07 13:54:20 +00:00
|
|
|
|
|
|
|
/**
|
2016-07-11 01:28:15 +00:00
|
|
|
* Init our basic, custom router on ready to determine app state.
|
2016-07-07 13:54:20 +00:00
|
|
|
*/
|
|
|
|
'koel:ready': () => {
|
2016-11-26 03:25:35 +00:00
|
|
|
router.init()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2016-06-25 16:05:24 +00:00
|
|
|
|
2016-07-08 08:24:41 +00:00
|
|
|
// Register our custom key codes
|
|
|
|
Vue.config.keyCodes = {
|
2016-07-08 08:25:34 +00:00
|
|
|
a: 65,
|
2016-07-08 08:24:41 +00:00
|
|
|
j: 74,
|
|
|
|
k: 75,
|
|
|
|
f: 70,
|
|
|
|
mediaNext: 176,
|
|
|
|
mediaPrev: 177,
|
|
|
|
mediaToggle: 179
|
2016-11-26 03:25:35 +00:00
|
|
|
}
|
2016-07-08 08:24:41 +00:00
|
|
|
|
2016-06-25 16:05:24 +00:00
|
|
|
// …and the global directives
|
2016-11-26 03:25:35 +00:00
|
|
|
Vue.directive('koel-focus', focusDirective)
|
|
|
|
Vue.directive('koel-clickaway', clickawayDirective)
|
2015-12-13 04:42:28 +00:00
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="sass">
|
2016-06-25 16:05:24 +00:00
|
|
|
@import "resources/assets/sass/partials/_vars.scss";
|
|
|
|
@import "resources/assets/sass/partials/_mixins.scss";
|
|
|
|
@import "resources/assets/sass/partials/_shared.scss";
|
|
|
|
|
|
|
|
#dragGhost {
|
2016-11-28 09:55:32 +00:00
|
|
|
position: absolute;
|
2016-06-25 16:05:24 +00:00
|
|
|
display: inline-block;
|
|
|
|
background: $colorGreen;
|
|
|
|
padding: .8rem;
|
|
|
|
border-radius: .2rem;
|
|
|
|
color: #fff;
|
|
|
|
font-family: $fontFamily;
|
|
|
|
font-size: 1rem;
|
|
|
|
font-weight: $fontWeight_Thin;
|
2016-11-28 09:55:32 +00:00
|
|
|
top: -100px;
|
|
|
|
left: 0px;
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* We can totally hide this element on touch devices, because there's
|
|
|
|
* no drag and drop support there anyway.
|
|
|
|
*/
|
|
|
|
html.touchevents & {
|
|
|
|
display: none;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-07 13:54:20 +00:00
|
|
|
#copyArea {
|
|
|
|
position: absolute;
|
|
|
|
left: -9999px;
|
|
|
|
width: 1px;
|
|
|
|
height: 1px;
|
2016-11-28 09:55:32 +00:00
|
|
|
bottom: 1px;
|
2016-07-07 13:54:20 +00:00
|
|
|
|
|
|
|
html.touchevents & {
|
|
|
|
display: none;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-25 16:05:24 +00:00
|
|
|
#main, .login-wrapper {
|
|
|
|
display: flex;
|
|
|
|
min-height: 100vh;
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
|
|
background: $colorMainBgr;
|
|
|
|
color: $colorMainText;
|
|
|
|
|
|
|
|
font-family: $fontFamily;
|
|
|
|
font-size: 1rem;
|
|
|
|
line-height: 1.5rem;
|
|
|
|
font-weight: $fontWeight_Thin;
|
|
|
|
|
|
|
|
padding-bottom: $footerHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
.login-wrapper {
|
|
|
|
@include vertical-center();
|
|
|
|
|
|
|
|
padding-bottom: 0;
|
|
|
|
}
|
2015-12-13 04:42:28 +00:00
|
|
|
</style>
|