koel/resources/assets/js/app.vue

283 lines
6.8 KiB
Vue
Raw Normal View History

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 $ from 'jquery'
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'
import { event, showOverlay, hideOverlay, forceReloadWindow } from './utils'
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-11-26 03:25:35 +00:00
$('<div id="dragGhost"></div>').appendTo('body')
2016-06-25 16:05:24 +00:00
2016-07-07 13:54:20 +00:00
// And the textarea to copy stuff
2016-11-26 03:25:35 +00:00
$('<textarea id="copyArea"></textarea>').appendTo('body')
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-11-26 03:25:35 +00:00
$('html').addClass(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
}
// 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-06-25 16:05:24 +00:00
if ($(e.target).is('input,textarea,button,select')) {
2016-11-26 03:25:35 +00:00
return true
2016-06-25 16:05:24 +00:00
}
// Ah... Good ol' jQuery. Whatever play/pause control is there, we blindly click it.
2016-11-26 03:25:35 +00:00
$('#mainFooter .play:visible, #mainFooter .pause:visible').click()
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-06-25 16:05:24 +00:00
if ($(e.target).is('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-06-25 16:05:24 +00:00
if ($(e.target).is('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-06-25 16:05:24 +00:00
if ($(e.target).is('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-11-26 03:25:35 +00:00
$('#searchForm input[type="search"]').focus().select()
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 {
position: relative;
display: inline-block;
background: $colorGreen;
padding: .8rem;
border-radius: .2rem;
color: #fff;
font-family: $fontFamily;
font-size: 1rem;
font-weight: $fontWeight_Thin;
/**
* 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;
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>