mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-04-30 01:15:24 +02:00
Add multi select dropdown with query from server
This commit is contained in:
parent
2a30cc428f
commit
f2be3bc95e
@ -14,7 +14,8 @@
|
|||||||
<div class="flex mt-2 -mx-1">
|
<div class="flex mt-2 -mx-1">
|
||||||
<div class="w-3/4 px-1">
|
<div class="w-3/4 px-1">
|
||||||
<!-- <ui-text-input-with-label v-model="details.authors" label="Author" /> -->
|
<!-- <ui-text-input-with-label v-model="details.authors" label="Author" /> -->
|
||||||
<p>Authors placeholder</p>
|
<!-- <p>Authors placeholder</p> -->
|
||||||
|
<ui-multi-select-query-input ref="authorsSelect" v-model="authorNames" label="Authors" endpoint="authors/search" />
|
||||||
</div>
|
</div>
|
||||||
<div class="flex-grow px-1">
|
<div class="flex-grow px-1">
|
||||||
<ui-text-input-with-label v-model="details.publishYear" type="number" label="Publish Year" />
|
<ui-text-input-with-label v-model="details.publishYear" type="number" label="Publish Year" />
|
||||||
@ -115,6 +116,7 @@ export default {
|
|||||||
genres: []
|
genres: []
|
||||||
},
|
},
|
||||||
newTags: [],
|
newTags: [],
|
||||||
|
authorNames: [],
|
||||||
resettingProgress: false,
|
resettingProgress: false,
|
||||||
isScrollable: false,
|
isScrollable: false,
|
||||||
savingMetadata: false,
|
savingMetadata: false,
|
||||||
@ -278,7 +280,7 @@ export default {
|
|||||||
this.details.title = this.mediaMetadata.title
|
this.details.title = this.mediaMetadata.title
|
||||||
this.details.subtitle = this.mediaMetadata.subtitle
|
this.details.subtitle = this.mediaMetadata.subtitle
|
||||||
this.details.description = this.mediaMetadata.description
|
this.details.description = this.mediaMetadata.description
|
||||||
this.details.authors = this.mediaMetadata.authors
|
this.details.authors = this.mediaMetadata.authors || []
|
||||||
this.details.narrator = this.mediaMetadata.narrator
|
this.details.narrator = this.mediaMetadata.narrator
|
||||||
this.details.genres = this.mediaMetadata.genres || []
|
this.details.genres = this.mediaMetadata.genres || []
|
||||||
this.details.series = this.mediaMetadata.series
|
this.details.series = this.mediaMetadata.series
|
||||||
@ -289,6 +291,7 @@ export default {
|
|||||||
this.details.asin = this.mediaMetadata.asin || null
|
this.details.asin = this.mediaMetadata.asin || null
|
||||||
|
|
||||||
this.newTags = this.media.tags || []
|
this.newTags = this.media.tags || []
|
||||||
|
this.authorNames = this.details.authors.map((au) => au.name)
|
||||||
},
|
},
|
||||||
removeItem() {
|
removeItem() {
|
||||||
if (confirm(`Are you sure you want to remove this item?\n\n*Does not delete your files, only removes the item from audiobookshelf`)) {
|
if (confirm(`Are you sure you want to remove this item?\n\n*Does not delete your files, only removes the item from audiobookshelf`)) {
|
||||||
|
252
client/components/ui/MultiSelectQueryInput.vue
Normal file
252
client/components/ui/MultiSelectQueryInput.vue
Normal file
@ -0,0 +1,252 @@
|
|||||||
|
<template>
|
||||||
|
<div class="w-full">
|
||||||
|
<p class="px-1 text-sm font-semibold" :class="disabled ? 'text-gray-400' : ''">{{ label }}</p>
|
||||||
|
<div ref="wrapper" class="relative">
|
||||||
|
<form @submit.prevent="submitForm">
|
||||||
|
<div ref="inputWrapper" style="min-height: 40px" class="flex-wrap relative w-full shadow-sm flex items-center border border-gray-600 rounded-md px-2 py-1 cursor-text" :class="disabled ? 'bg-black-300' : 'bg-primary'" @click.stop.prevent="clickWrapper" @mouseup.stop.prevent @mousedown.prevent>
|
||||||
|
<div v-for="item in selected" :key="item" class="rounded-full px-2 py-1 ma-0.5 text-xs bg-bg flex flex-nowrap whitespace-nowrap items-center relative">
|
||||||
|
<div class="w-full h-full rounded-full absolute top-0 left-0 opacity-0 hover:opacity-100 px-1 bg-bg bg-opacity-75 flex items-center justify-end cursor-pointer">
|
||||||
|
<span class="material-icons text-white hover:text-error" style="font-size: 1.1rem" @click.stop="removeItem(item)">close</span>
|
||||||
|
</div>
|
||||||
|
{{ item }}
|
||||||
|
</div>
|
||||||
|
<input ref="input" v-model="textInput" :disabled="disabled" style="min-width: 40px; width: 40px" class="h-full bg-primary focus:outline-none px-1" @keydown="keydownInput" @focus="inputFocus" @blur="inputBlur" />
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<ul ref="menu" v-show="showMenu" class="absolute z-50 mt-1 w-full bg-bg border border-black-200 shadow-lg max-h-56 rounded-md py-1 text-base ring-1 ring-black ring-opacity-5 overflow-auto focus:outline-none sm:text-sm" role="listbox" aria-labelledby="listbox-label">
|
||||||
|
<template v-for="item in itemsToShow">
|
||||||
|
<li :key="item.id" class="text-gray-50 select-none relative py-2 pr-9 cursor-pointer hover:bg-black-400" role="option" @click="clickedOption($event, item.name)" @mouseup.stop.prevent @mousedown.prevent>
|
||||||
|
<div class="flex items-center">
|
||||||
|
<span class="font-normal ml-3 block truncate">{{ item.name }}</span>
|
||||||
|
</div>
|
||||||
|
<!-- <span v-if="selected.includes(item)" class="text-yellow-400 absolute inset-y-0 right-0 flex items-center pr-4">
|
||||||
|
<span class="material-icons text-xl">checkmark</span>
|
||||||
|
</span> -->
|
||||||
|
</li>
|
||||||
|
</template>
|
||||||
|
<li v-if="!itemsToShow.length" class="text-gray-50 select-none relative py-2 pr-9" role="option">
|
||||||
|
<div class="flex items-center justify-center">
|
||||||
|
<span class="font-normal">No items</span>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
value: {
|
||||||
|
type: Array,
|
||||||
|
default: () => []
|
||||||
|
},
|
||||||
|
endpoint: String,
|
||||||
|
label: String,
|
||||||
|
disabled: Boolean
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
textInput: null,
|
||||||
|
currentSearch: null,
|
||||||
|
searching: false,
|
||||||
|
typingTimeout: null,
|
||||||
|
isFocused: false,
|
||||||
|
menu: null,
|
||||||
|
items: []
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
showMenu(newVal) {
|
||||||
|
if (newVal) this.setListener()
|
||||||
|
else this.removeListener()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
selected: {
|
||||||
|
get() {
|
||||||
|
return this.value
|
||||||
|
},
|
||||||
|
set(val) {
|
||||||
|
this.$emit('input', val)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
showMenu() {
|
||||||
|
return this.isFocused
|
||||||
|
},
|
||||||
|
itemsToShow() {
|
||||||
|
return this.items
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
async search() {
|
||||||
|
if (this.searching) return
|
||||||
|
this.currentSearch = this.textInput
|
||||||
|
this.searching = true
|
||||||
|
var results = await this.$axios.$get(`/api/${this.endpoint}?q=${this.currentSearch}&limit=15`).catch((error) => {
|
||||||
|
console.error('Failed to get search results', error)
|
||||||
|
return []
|
||||||
|
})
|
||||||
|
console.log('Search results', results)
|
||||||
|
this.items = results || []
|
||||||
|
this.searching = false
|
||||||
|
},
|
||||||
|
keydownInput() {
|
||||||
|
clearTimeout(this.typingTimeout)
|
||||||
|
this.typingTimeout = setTimeout(() => {
|
||||||
|
this.search()
|
||||||
|
}, 500)
|
||||||
|
this.setInputWidth()
|
||||||
|
},
|
||||||
|
setInputWidth() {
|
||||||
|
setTimeout(() => {
|
||||||
|
var value = this.$refs.input.value
|
||||||
|
var len = value.length * 7 + 24
|
||||||
|
this.$refs.input.style.width = len + 'px'
|
||||||
|
this.recalcMenuPos()
|
||||||
|
}, 50)
|
||||||
|
},
|
||||||
|
recalcMenuPos() {
|
||||||
|
if (!this.menu) return
|
||||||
|
var boundingBox = this.$refs.inputWrapper.getBoundingClientRect()
|
||||||
|
if (boundingBox.y > window.innerHeight - 8) {
|
||||||
|
// Input is off the page
|
||||||
|
return this.forceBlur()
|
||||||
|
}
|
||||||
|
var menuHeight = this.menu.clientHeight
|
||||||
|
var top = boundingBox.y + boundingBox.height - 4
|
||||||
|
if (top + menuHeight > window.innerHeight - 20) {
|
||||||
|
// Reverse menu to open upwards
|
||||||
|
top = boundingBox.y - menuHeight - 4
|
||||||
|
}
|
||||||
|
|
||||||
|
this.menu.style.top = top + 'px'
|
||||||
|
this.menu.style.left = boundingBox.x + 'px'
|
||||||
|
this.menu.style.width = boundingBox.width + 'px'
|
||||||
|
},
|
||||||
|
unmountMountMenu() {
|
||||||
|
if (!this.$refs.menu) return
|
||||||
|
this.menu = this.$refs.menu
|
||||||
|
|
||||||
|
var boundingBox = this.$refs.inputWrapper.getBoundingClientRect()
|
||||||
|
this.menu.remove()
|
||||||
|
document.body.appendChild(this.menu)
|
||||||
|
this.menu.style.top = boundingBox.y + boundingBox.height - 4 + 'px'
|
||||||
|
this.menu.style.left = boundingBox.x + 'px'
|
||||||
|
this.menu.style.width = boundingBox.width + 'px'
|
||||||
|
},
|
||||||
|
inputFocus() {
|
||||||
|
if (!this.menu) {
|
||||||
|
this.unmountMountMenu()
|
||||||
|
}
|
||||||
|
this.isFocused = true
|
||||||
|
this.$nextTick(this.recalcMenuPos)
|
||||||
|
},
|
||||||
|
inputBlur() {
|
||||||
|
if (!this.isFocused) return
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
if (document.activeElement === this.$refs.input) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
this.isFocused = false
|
||||||
|
if (this.textInput) this.submitForm()
|
||||||
|
}, 50)
|
||||||
|
},
|
||||||
|
focus() {
|
||||||
|
if (this.$refs.input) this.$refs.input.focus()
|
||||||
|
},
|
||||||
|
blur() {
|
||||||
|
if (this.$refs.input) this.$refs.input.blur()
|
||||||
|
},
|
||||||
|
forceBlur() {
|
||||||
|
this.isFocused = false
|
||||||
|
if (this.textInput) this.submitForm()
|
||||||
|
if (this.$refs.input) this.$refs.input.blur()
|
||||||
|
},
|
||||||
|
clickedOption(e, itemValue) {
|
||||||
|
if (e) {
|
||||||
|
e.stopPropagation()
|
||||||
|
e.preventDefault()
|
||||||
|
}
|
||||||
|
if (this.$refs.input) this.$refs.input.focus()
|
||||||
|
|
||||||
|
var newSelected = null
|
||||||
|
if (this.selected.includes(itemValue)) {
|
||||||
|
newSelected = this.selected.filter((s) => s !== itemValue)
|
||||||
|
this.$emit('removedItem', itemValue)
|
||||||
|
} else {
|
||||||
|
newSelected = this.selected.concat([itemValue])
|
||||||
|
}
|
||||||
|
this.textInput = null
|
||||||
|
this.currentSearch = null
|
||||||
|
this.$emit('input', newSelected)
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.recalcMenuPos()
|
||||||
|
})
|
||||||
|
},
|
||||||
|
clickWrapper() {
|
||||||
|
if (this.disabled) return
|
||||||
|
if (this.showMenu) {
|
||||||
|
return this.blur()
|
||||||
|
}
|
||||||
|
this.focus()
|
||||||
|
},
|
||||||
|
removeItem(item) {
|
||||||
|
var remaining = this.selected.filter((i) => i !== item)
|
||||||
|
this.$emit('input', remaining)
|
||||||
|
this.$emit('removedItem', item)
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.recalcMenuPos()
|
||||||
|
})
|
||||||
|
},
|
||||||
|
insertNewItem(item) {
|
||||||
|
this.selected.push(item)
|
||||||
|
this.$emit('input', this.selected)
|
||||||
|
this.$emit('newItem', item)
|
||||||
|
this.textInput = null
|
||||||
|
this.currentSearch = null
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.blur()
|
||||||
|
})
|
||||||
|
},
|
||||||
|
submitForm() {
|
||||||
|
if (!this.textInput) return
|
||||||
|
|
||||||
|
var cleaned = this.textInput.trim()
|
||||||
|
var matchesItem = this.items.find((i) => {
|
||||||
|
return i === cleaned
|
||||||
|
})
|
||||||
|
if (matchesItem) {
|
||||||
|
this.clickedOption(null, matchesItem)
|
||||||
|
} else {
|
||||||
|
this.insertNewItem(this.textInput)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
scroll() {
|
||||||
|
this.recalcMenuPos()
|
||||||
|
},
|
||||||
|
setListener() {
|
||||||
|
document.addEventListener('scroll', this.scroll, true)
|
||||||
|
},
|
||||||
|
removeListener() {
|
||||||
|
document.removeEventListener('scroll', this.scroll, true)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {},
|
||||||
|
beforeDestroy() {
|
||||||
|
if (this.menu) this.menu.remove()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
input {
|
||||||
|
border-style: inherit !important;
|
||||||
|
}
|
||||||
|
input:read-only {
|
||||||
|
color: #aaa;
|
||||||
|
background-color: #444;
|
||||||
|
}
|
||||||
|
</style>
|
@ -22,12 +22,11 @@ const PodcastFinder = require('./finders/PodcastFinder')
|
|||||||
const FileSystemController = require('./controllers/FileSystemController')
|
const FileSystemController = require('./controllers/FileSystemController')
|
||||||
|
|
||||||
class ApiController {
|
class ApiController {
|
||||||
constructor(db, auth, scanner, streamManager, rssFeeds, downloadManager, coverController, backupManager, watcher, cacheManager, emitter, clientEmitter) {
|
constructor(db, auth, scanner, streamManager, downloadManager, coverController, backupManager, watcher, cacheManager, emitter, clientEmitter) {
|
||||||
this.db = db
|
this.db = db
|
||||||
this.auth = auth
|
this.auth = auth
|
||||||
this.scanner = scanner
|
this.scanner = scanner
|
||||||
this.streamManager = streamManager
|
this.streamManager = streamManager
|
||||||
this.rssFeeds = rssFeeds
|
|
||||||
this.downloadManager = downloadManager
|
this.downloadManager = downloadManager
|
||||||
this.backupManager = backupManager
|
this.backupManager = backupManager
|
||||||
this.coverController = coverController
|
this.coverController = coverController
|
||||||
@ -145,6 +144,7 @@ class ApiController {
|
|||||||
this.router.get('/search/covers', this.findCovers.bind(this))
|
this.router.get('/search/covers', this.findCovers.bind(this))
|
||||||
this.router.get('/search/books', this.findBooks.bind(this))
|
this.router.get('/search/books', this.findBooks.bind(this))
|
||||||
this.router.get('/search/podcast', this.findPodcasts.bind(this))
|
this.router.get('/search/podcast', this.findPodcasts.bind(this))
|
||||||
|
this.router.get('/search/authors', this.findAuthor.bind(this))
|
||||||
|
|
||||||
//
|
//
|
||||||
// File System Routes
|
// File System Routes
|
||||||
@ -155,7 +155,7 @@ class ApiController {
|
|||||||
// Others
|
// Others
|
||||||
//
|
//
|
||||||
this.router.get('/authors', this.getAuthors.bind(this))
|
this.router.get('/authors', this.getAuthors.bind(this))
|
||||||
this.router.get('/authors/search', this.searchAuthor.bind(this))
|
this.router.get('/authors/search', this.searchAuthors.bind(this))
|
||||||
this.router.get('/authors/:id', this.getAuthor.bind(this))
|
this.router.get('/authors/:id', this.getAuthor.bind(this))
|
||||||
this.router.post('/authors', this.createAuthor.bind(this))
|
this.router.post('/authors', this.createAuthor.bind(this))
|
||||||
this.router.patch('/authors/:id', this.updateAuthor.bind(this))
|
this.router.patch('/authors/:id', this.updateAuthor.bind(this))
|
||||||
@ -165,8 +165,6 @@ class ApiController {
|
|||||||
|
|
||||||
this.router.post('/authorize', this.authorize.bind(this))
|
this.router.post('/authorize', this.authorize.bind(this))
|
||||||
|
|
||||||
this.router.post('/feed', this.openRssFeed.bind(this))
|
|
||||||
|
|
||||||
this.router.get('/download/:id', this.download.bind(this))
|
this.router.get('/download/:id', this.download.bind(this))
|
||||||
|
|
||||||
this.router.post('/syncUserAudiobookData', this.syncUserAudiobookData.bind(this))
|
this.router.post('/syncUserAudiobookData', this.syncUserAudiobookData.bind(this))
|
||||||
@ -201,6 +199,12 @@ class ApiController {
|
|||||||
res.json(results)
|
res.json(results)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async findAuthor(req, res) {
|
||||||
|
var query = req.query.q
|
||||||
|
var author = await this.authorFinder.findAuthorByName(query)
|
||||||
|
res.json(author)
|
||||||
|
}
|
||||||
|
|
||||||
authorize(req, res) {
|
authorize(req, res) {
|
||||||
if (!req.user) {
|
if (!req.user) {
|
||||||
Logger.error('Invalid user in authorize')
|
Logger.error('Invalid user in authorize')
|
||||||
@ -209,20 +213,19 @@ class ApiController {
|
|||||||
res.json({ user: req.user })
|
res.json({ user: req.user })
|
||||||
}
|
}
|
||||||
|
|
||||||
async openRssFeed(req, res) {
|
|
||||||
var audiobookId = req.body.audiobookId
|
|
||||||
var audiobook = this.db.audiobooks.find(ab => ab.id === audiobookId)
|
|
||||||
if (!audiobook) return res.sendStatus(404)
|
|
||||||
var feed = await this.rssFeeds.openFeed(audiobook)
|
|
||||||
console.log('Feed open', feed)
|
|
||||||
res.json(feed)
|
|
||||||
}
|
|
||||||
|
|
||||||
async getAuthors(req, res) {
|
async getAuthors(req, res) {
|
||||||
var authors = this.db.authors.filter(p => p.isAuthor)
|
var authors = this.db.authors.filter(p => p.isAuthor)
|
||||||
res.json(authors)
|
res.json(authors)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
searchAuthors(req, res) {
|
||||||
|
var query = req.query.q || ''
|
||||||
|
var limit = req.query.limit && !isNaN(req.query.limit) ? Number(req.query.limit) : 100
|
||||||
|
var authors = this.db.authors.filter(au => au.name.toLowerCase().includes(query.toLowerCase()))
|
||||||
|
authors = authors.slice(0, limit)
|
||||||
|
res.json(authors)
|
||||||
|
}
|
||||||
|
|
||||||
async getAuthor(req, res) {
|
async getAuthor(req, res) {
|
||||||
var author = this.db.authors.find(p => p.id === req.params.id)
|
var author = this.db.authors.find(p => p.id === req.params.id)
|
||||||
if (!author) {
|
if (!author) {
|
||||||
@ -231,12 +234,6 @@ class ApiController {
|
|||||||
res.json(author.toJSON())
|
res.json(author.toJSON())
|
||||||
}
|
}
|
||||||
|
|
||||||
async searchAuthor(req, res) {
|
|
||||||
var query = req.query.q
|
|
||||||
var author = await this.authorFinder.findAuthorByName(query)
|
|
||||||
res.json(author)
|
|
||||||
}
|
|
||||||
|
|
||||||
async createAuthor(req, res) {
|
async createAuthor(req, res) {
|
||||||
var author = await this.authorFinder.createAuthor(req.body)
|
var author = await this.authorFinder.createAuthor(req.body)
|
||||||
if (!author) {
|
if (!author) {
|
||||||
|
@ -25,7 +25,6 @@ const LogManager = require('./LogManager')
|
|||||||
const ApiController = require('./ApiController')
|
const ApiController = require('./ApiController')
|
||||||
const HlsController = require('./HlsController')
|
const HlsController = require('./HlsController')
|
||||||
const StreamManager = require('./StreamManager')
|
const StreamManager = require('./StreamManager')
|
||||||
const RssFeeds = require('./RssFeeds')
|
|
||||||
const DownloadManager = require('./DownloadManager')
|
const DownloadManager = require('./DownloadManager')
|
||||||
const CoverController = require('./CoverController')
|
const CoverController = require('./CoverController')
|
||||||
const CacheManager = require('./CacheManager')
|
const CacheManager = require('./CacheManager')
|
||||||
@ -62,9 +61,8 @@ class Server {
|
|||||||
this.scanner = new Scanner(this.db, this.coverController, this.emitter.bind(this))
|
this.scanner = new Scanner(this.db, this.coverController, this.emitter.bind(this))
|
||||||
|
|
||||||
this.streamManager = new StreamManager(this.db, this.emitter.bind(this), this.clientEmitter.bind(this))
|
this.streamManager = new StreamManager(this.db, this.emitter.bind(this), this.clientEmitter.bind(this))
|
||||||
this.rssFeeds = new RssFeeds(this.Port, this.db)
|
|
||||||
this.downloadManager = new DownloadManager(this.db, this.Uid, this.Gid)
|
this.downloadManager = new DownloadManager(this.db, this.Uid, this.Gid)
|
||||||
this.apiController = new ApiController(this.db, this.auth, this.scanner, this.streamManager, this.rssFeeds, this.downloadManager, this.coverController, this.backupManager, this.watcher, this.cacheManager, this.emitter.bind(this), this.clientEmitter.bind(this))
|
this.apiController = new ApiController(this.db, this.auth, this.scanner, this.streamManager, this.downloadManager, this.coverController, this.backupManager, this.watcher, this.cacheManager, this.emitter.bind(this), this.clientEmitter.bind(this))
|
||||||
this.hlsController = new HlsController(this.db, this.auth, this.streamManager, this.emitter.bind(this), this.streamManager.StreamsPath)
|
this.hlsController = new HlsController(this.db, this.auth, this.streamManager, this.emitter.bind(this), this.streamManager.StreamsPath)
|
||||||
|
|
||||||
Logger.logManager = this.logManager
|
Logger.logManager = this.logManager
|
||||||
|
Loading…
Reference in New Issue
Block a user