mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-03 00:06:46 +01:00
Fix: Setting root socket error, Change: collection books table ordering and icons #151
This commit is contained in:
parent
d115382abe
commit
0980b6d5d5
@ -14,6 +14,9 @@
|
|||||||
#librariesTable .item {
|
#librariesTable .item {
|
||||||
cursor: n-resize;
|
cursor: n-resize;
|
||||||
}
|
}
|
||||||
|
.drag-handle {
|
||||||
|
cursor: n-resize;
|
||||||
|
}
|
||||||
.list-group-item:not(.exclude) {
|
.list-group-item:not(.exclude) {
|
||||||
cursor: n-resize;
|
cursor: n-resize;
|
||||||
}
|
}
|
||||||
|
@ -175,6 +175,7 @@ export default {
|
|||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
goPrevBook() {
|
goPrevBook() {
|
||||||
|
console.log('GO PREV', this.currentBookshelfIndex)
|
||||||
if (this.currentBookshelfIndex - 1 < 0) return
|
if (this.currentBookshelfIndex - 1 < 0) return
|
||||||
var prevBookId = this.bookshelfBookIds[this.currentBookshelfIndex - 1]
|
var prevBookId = this.bookshelfBookIds[this.currentBookshelfIndex - 1]
|
||||||
var prevBook = this.$store.getters['audiobooks/getAudiobook'](prevBookId)
|
var prevBook = this.$store.getters['audiobooks/getAudiobook'](prevBookId)
|
||||||
@ -186,6 +187,7 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
goNextBook() {
|
goNextBook() {
|
||||||
|
console.log('GO NEXT', this.currentBookshelfIndex)
|
||||||
if (this.currentBookshelfIndex >= this.bookshelfBookIds.length - 1) return
|
if (this.currentBookshelfIndex >= this.bookshelfBookIds.length - 1) return
|
||||||
|
|
||||||
var nextBookId = this.bookshelfBookIds[this.currentBookshelfIndex + 1]
|
var nextBookId = this.bookshelfBookIds[this.currentBookshelfIndex + 1]
|
||||||
@ -224,6 +226,7 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
hotkey(action) {
|
hotkey(action) {
|
||||||
|
console.log('HOTKEY', action)
|
||||||
if (action === this.$hotkeys.Modal.NEXT_PAGE) {
|
if (action === this.$hotkeys.Modal.NEXT_PAGE) {
|
||||||
this.goNextBook()
|
this.goNextBook()
|
||||||
} else if (action === this.$hotkeys.Modal.PREV_PAGE) {
|
} else if (action === this.$hotkeys.Modal.PREV_PAGE) {
|
||||||
|
@ -150,8 +150,6 @@ export default {
|
|||||||
<style>
|
<style>
|
||||||
.list-complete-item {
|
.list-complete-item {
|
||||||
transition: all 0.8s ease;
|
transition: all 0.8s ease;
|
||||||
/* display: block;
|
|
||||||
margin-right: 10px; */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.list-complete-enter-from,
|
.list-complete-enter-from,
|
||||||
|
@ -8,22 +8,39 @@
|
|||||||
<div class="flex-grow" />
|
<div class="flex-grow" />
|
||||||
<p v-if="totalDuration">{{ totalDurationPretty }}</p>
|
<p v-if="totalDuration">{{ totalDurationPretty }}</p>
|
||||||
</div>
|
</div>
|
||||||
|
<draggable v-model="books" v-bind="dragOptions" class="list-group" handle=".drag-handle" draggable=".item" tag="div" @start="drag = true" @end="drag = false" @update="draggableUpdate">
|
||||||
|
<transition-group type="transition" :name="!drag ? 'list-complete' : null">
|
||||||
<template v-for="book in books">
|
<template v-for="book in books">
|
||||||
<tables-collection-book-table-row :key="book.id" :book="book" />
|
<tables-collection-book-table-row :key="book.id" :book="book" :collection-id="collectionId" class="item list-complete-item" @edit="editBook" />
|
||||||
</template>
|
</template>
|
||||||
|
</transition-group>
|
||||||
|
</draggable>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
import draggable from 'vuedraggable'
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
components: {
|
||||||
|
draggable
|
||||||
|
},
|
||||||
props: {
|
props: {
|
||||||
|
collectionId: String,
|
||||||
books: {
|
books: {
|
||||||
type: Array,
|
type: Array,
|
||||||
default: () => []
|
default: () => []
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {}
|
return {
|
||||||
|
drag: false,
|
||||||
|
dragOptions: {
|
||||||
|
animation: 200,
|
||||||
|
group: 'description',
|
||||||
|
ghostClass: 'ghost'
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
totalDuration() {
|
totalDuration() {
|
||||||
@ -37,7 +54,30 @@ export default {
|
|||||||
return this.$elapsedPretty(this.totalDuration)
|
return this.$elapsedPretty(this.totalDuration)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {},
|
methods: {
|
||||||
|
draggableUpdate() {},
|
||||||
|
editBook(book) {
|
||||||
|
var bookIds = this.books.map((b) => b.id)
|
||||||
|
this.$store.commit('setBookshelfBookIds', bookIds)
|
||||||
|
this.$store.commit('showEditModal', book)
|
||||||
|
}
|
||||||
|
},
|
||||||
mounted() {}
|
mounted() {}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.list-complete-item {
|
||||||
|
transition: all 0.8s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-complete-enter-from,
|
||||||
|
.list-complete-leave-to {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(30px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-complete-leave-active {
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
</style>
|
@ -1,6 +1,11 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="w-full px-6 py-2" @mouseover="mouseover" @mouseleave="mouseleave" :class="isHovering ? 'bg-white bg-opacity-5' : ''">
|
<div class="w-full px-2 py-2 overflow-hidden relative" @mouseover="mouseover" @mouseleave="mouseleave" :class="isHovering ? 'bg-white bg-opacity-5' : ''">
|
||||||
<div v-if="book" class="flex h-20">
|
<div v-if="book" class="flex h-20">
|
||||||
|
<div class="w-16 max-w-16 h-full">
|
||||||
|
<div class="flex h-full items-center justify-center">
|
||||||
|
<span class="material-icons drag-handle text-xl">menu</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
<covers-book-cover :audiobook="book" :width="50" />
|
<covers-book-cover :audiobook="book" :width="50" />
|
||||||
<div class="w-80 h-full px-2 flex items-center">
|
<div class="w-80 h-full px-2 flex items-center">
|
||||||
<div>
|
<div>
|
||||||
@ -9,18 +14,38 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex-grow flex items-center">
|
<div class="flex-grow flex items-center">
|
||||||
<p>{{ bookDuration }}</p>
|
<p class="font-mono text-sm">{{ bookDuration }}</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- <div class="w-12 flex items-center justify-center">
|
<!-- <div class="w-12 flex items-center justify-center">
|
||||||
<span class="material-icons text-lg text-white text-opacity-70 hover:text-opacity-100 cursor-pointer">radio_button_unchecked</span>
|
<span class="material-icons text-lg text-white text-opacity-70 hover:text-opacity-100 cursor-pointer">radio_button_unchecked</span>
|
||||||
</div> -->
|
</div> -->
|
||||||
</div>
|
</div>
|
||||||
|
<!-- <div class="absolute top-0 left-0 z-40 bg-red-500 w-full h-full">
|
||||||
|
<div class="w-24 h-full absolute top-0 -right-24 transform transition-transform" :class="isHovering ? 'translate-x-0' : '-translate-x-24'">
|
||||||
|
<span class="material-icons">edit</span>
|
||||||
|
</div>
|
||||||
|
</div> -->
|
||||||
|
<div class="w-40 absolute top-0 -right-40 h-full transform transition-transform" :class="!isHovering ? 'translate-x-0' : '-translate-x-40'">
|
||||||
|
<div class="flex h-full items-center">
|
||||||
|
<ui-tooltip :text="isRead ? 'Mark as Not Read' : 'Mark as Read'" direction="top">
|
||||||
|
<ui-read-icon-btn :disabled="isProcessingReadUpdate" :is-read="isRead" borderless class="mx-1 mt-0.5" @click="toggleRead" />
|
||||||
|
</ui-tooltip>
|
||||||
|
<div class="mx-1">
|
||||||
|
<ui-icon-btn icon="edit" borderless @click="clickEdit" />
|
||||||
|
</div>
|
||||||
|
<div class="mx-1">
|
||||||
|
<ui-icon-btn icon="close" borderless @click="removeClick" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
export default {
|
export default {
|
||||||
props: {
|
props: {
|
||||||
|
collectionId: String,
|
||||||
book: {
|
book: {
|
||||||
type: Object,
|
type: Object,
|
||||||
default: () => {}
|
default: () => {}
|
||||||
@ -28,9 +53,19 @@ export default {
|
|||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
|
isProcessingReadUpdate: false,
|
||||||
|
processingRemove: false,
|
||||||
isHovering: false
|
isHovering: false
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
watch: {
|
||||||
|
userIsRead: {
|
||||||
|
immediate: true,
|
||||||
|
handler(newVal) {
|
||||||
|
this.isRead = newVal
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
computed: {
|
computed: {
|
||||||
_book() {
|
_book() {
|
||||||
return this.book.book || {}
|
return this.book.book || {}
|
||||||
@ -43,6 +78,15 @@ export default {
|
|||||||
},
|
},
|
||||||
bookDuration() {
|
bookDuration() {
|
||||||
return this.$secondsToTimestamp(this.book.duration)
|
return this.$secondsToTimestamp(this.book.duration)
|
||||||
|
},
|
||||||
|
userAudiobooks() {
|
||||||
|
return this.$store.state.user.user ? this.$store.state.user.user.audiobooks || {} : {}
|
||||||
|
},
|
||||||
|
userAudiobook() {
|
||||||
|
return this.userAudiobooks[this.book.id] || null
|
||||||
|
},
|
||||||
|
userIsRead() {
|
||||||
|
return this.userAudiobook ? !!this.userAudiobook.isRead : false
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
@ -51,6 +95,43 @@ export default {
|
|||||||
},
|
},
|
||||||
mouseleave() {
|
mouseleave() {
|
||||||
this.isHovering = false
|
this.isHovering = false
|
||||||
|
},
|
||||||
|
clickRemove() {},
|
||||||
|
clickEdit() {
|
||||||
|
this.$emit('edit', this.book)
|
||||||
|
},
|
||||||
|
toggleRead() {
|
||||||
|
var updatePayload = {
|
||||||
|
isRead: !this.isRead
|
||||||
|
}
|
||||||
|
this.isProcessingReadUpdate = true
|
||||||
|
this.$axios
|
||||||
|
.$patch(`/api/user/audiobook/${this.book.id}`, updatePayload)
|
||||||
|
.then(() => {
|
||||||
|
this.isProcessingReadUpdate = false
|
||||||
|
this.$toast.success(`"${this.title}" Marked as ${updatePayload.isRead ? 'Read' : 'Not Read'}`)
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.error('Failed', error)
|
||||||
|
this.isProcessingReadUpdate = false
|
||||||
|
this.$toast.error(`Failed to mark as ${updatePayload.isRead ? 'Read' : 'Not Read'}`)
|
||||||
|
})
|
||||||
|
},
|
||||||
|
removeClick() {
|
||||||
|
this.processingRemove = true
|
||||||
|
|
||||||
|
this.$axios
|
||||||
|
.$delete(`/api/collection/${this.collectionId}/book/${this.book.id}`)
|
||||||
|
.then((updatedCollection) => {
|
||||||
|
console.log(`Book removed from collection`, updatedCollection)
|
||||||
|
this.$toast.success('Book removed from collection')
|
||||||
|
this.processingRemove = false
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.error('Failed to remove book from collection', error)
|
||||||
|
this.$toast.error('Failed to remove book from collection')
|
||||||
|
this.processingRemove = false
|
||||||
|
})
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {}
|
mounted() {}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<button class="icon-btn rounded-md border border-gray-600 flex items-center justify-center h-9 w-9 relative" :disabled="disabled" :class="className" @click="clickBtn">
|
<button class="icon-btn rounded-md flex items-center justify-center h-9 w-9 relative" @mousedown.prevent :disabled="disabled" :class="className" @click="clickBtn">
|
||||||
<span :class="outlined ? 'material-icons-outlined' : 'material-icons'" :style="{ fontSize }">{{ icon }}</span>
|
<span :class="outlined ? 'material-icons-outlined' : 'material-icons'" :style="{ fontSize }">{{ icon }}</span>
|
||||||
</button>
|
</button>
|
||||||
</template>
|
</template>
|
||||||
@ -13,7 +13,8 @@ export default {
|
|||||||
type: String,
|
type: String,
|
||||||
default: 'primary'
|
default: 'primary'
|
||||||
},
|
},
|
||||||
outlined: Boolean
|
outlined: Boolean,
|
||||||
|
borderless: Boolean
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {}
|
return {}
|
||||||
@ -21,7 +22,9 @@ export default {
|
|||||||
computed: {
|
computed: {
|
||||||
className() {
|
className() {
|
||||||
var classes = []
|
var classes = []
|
||||||
classes.push(`bg-${this.bgColor}`)
|
if (!this.borderless) {
|
||||||
|
classes.push(`bg-${this.bgColor} border border-gray-600`)
|
||||||
|
}
|
||||||
return classes.join(' ')
|
return classes.join(' ')
|
||||||
},
|
},
|
||||||
fontSize() {
|
fontSize() {
|
||||||
@ -35,6 +38,7 @@ export default {
|
|||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
e.preventDefault()
|
||||||
this.$emit('click')
|
this.$emit('click')
|
||||||
e.stopPropagation()
|
e.stopPropagation()
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<button class="icon-btn rounded-md bg-primary border border-gray-600 flex items-center justify-center h-9 w-9 relative" @click="clickBtn">
|
<button class="icon-btn rounded-md flex items-center justify-center h-9 w-9 relative" :class="borderless ? '' : 'bg-primary border border-gray-600'" @click="clickBtn">
|
||||||
<div class="w-5 h-5 text-white relative">
|
<div class="w-5 h-5 text-white relative">
|
||||||
<svg v-if="isRead" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="rgb(63, 181, 68)">
|
<svg v-if="isRead" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="rgb(63, 181, 68)">
|
||||||
<path d="M19 1H5c-1.1 0-1.99.9-1.99 2L3 15.93c0 .69.35 1.3.88 1.66L12 23l8.11-5.41c.53-.36.88-.97.88-1.66L21 3c0-1.1-.9-2-2-2zm-9 15l-5-5 1.41-1.41L10 13.17l7.59-7.59L19 7l-9 9z" />
|
<path d="M19 1H5c-1.1 0-1.99.9-1.99 2L3 15.93c0 .69.35 1.3.88 1.66L12 23l8.11-5.41c.53-.36.88-.97.88-1.66L21 3c0-1.1-.9-2-2-2zm-9 15l-5-5 1.41-1.41L10 13.17l7.59-7.59L19 7l-9 9z" />
|
||||||
@ -15,7 +15,8 @@
|
|||||||
export default {
|
export default {
|
||||||
props: {
|
props: {
|
||||||
isRead: Boolean,
|
isRead: Boolean,
|
||||||
disabled: Boolean
|
disabled: Boolean,
|
||||||
|
borderless: Boolean
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {}
|
return {}
|
||||||
|
@ -309,7 +309,7 @@ export default {
|
|||||||
upgrade: false,
|
upgrade: false,
|
||||||
reconnection: true
|
reconnection: true
|
||||||
})
|
})
|
||||||
// this.$root.socket = this.socket
|
this.$root.socket = this.socket
|
||||||
|
|
||||||
this.socket.on('connect', this.connect)
|
this.socket.on('connect', this.connect)
|
||||||
this.socket.on('connect_error', this.connectError)
|
this.socket.on('connect_error', this.connectError)
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "audiobookshelf-client",
|
"name": "audiobookshelf-client",
|
||||||
"version": "1.6.9",
|
"version": "1.6.10",
|
||||||
"description": "Audiobook manager and player",
|
"description": "Audiobook manager and player",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
@ -23,7 +23,7 @@
|
|||||||
<p class="text-base text-gray-100">{{ description }}</p>
|
<p class="text-base text-gray-100">{{ description }}</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<tables-collection-books-table :books="bookItems" />
|
<tables-collection-books-table :books="bookItems" :collection-id="collection.id" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -47,13 +47,17 @@ export default {
|
|||||||
return redirect('/')
|
return redirect('/')
|
||||||
}
|
}
|
||||||
store.commit('user/addUpdateCollection', collection)
|
store.commit('user/addUpdateCollection', collection)
|
||||||
|
collection.books.forEach((book) => {
|
||||||
|
store.commit('audiobooks/addUpdate', book)
|
||||||
|
})
|
||||||
return {
|
return {
|
||||||
collection
|
collectionId: collection.id
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
processingRemove: false
|
processingRemove: false,
|
||||||
|
collectionCopy: {}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@ -68,6 +72,9 @@ export default {
|
|||||||
},
|
},
|
||||||
description() {
|
description() {
|
||||||
return this.collection.description || ''
|
return this.collection.description || ''
|
||||||
|
},
|
||||||
|
collection() {
|
||||||
|
return this.$store.getters['user/getCollection'](this.collectionId)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
@ -90,8 +97,14 @@ export default {
|
|||||||
this.$toast.error(`Failed to remove collection`)
|
this.$toast.error(`Failed to remove collection`)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
collectionsUpdated() {
|
||||||
|
// this.collectionCopy = { ...this.collection }
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {}
|
mounted() {
|
||||||
|
// this.$store.commit('user/addCollectionsListener', { meth: this.collectionsUpdated, key: 'collection-page' })
|
||||||
|
},
|
||||||
|
beforeDestroy() {}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
@ -51,6 +51,9 @@ export const getters = {
|
|||||||
if (!state.user) return false
|
if (!state.user) return false
|
||||||
if (getters.getUserCanAccessAllLibraries) return true
|
if (getters.getUserCanAccessAllLibraries) return true
|
||||||
return getters.getLibrariesAccessible.includes(libraryId)
|
return getters.getLibrariesAccessible.includes(libraryId)
|
||||||
|
},
|
||||||
|
getCollection: state => id => {
|
||||||
|
return state.collections.find(c => c.id === id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "audiobookshelf",
|
"name": "audiobookshelf",
|
||||||
"version": "1.6.9",
|
"version": "1.6.10",
|
||||||
"description": "Self-hosted audiobook server for managing and playing audiobooks",
|
"description": "Self-hosted audiobook server for managing and playing audiobooks",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
Loading…
Reference in New Issue
Block a user