mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2026-06-05 18:22:44 +02:00
Compare commits
41 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 06554811e2 | |||
| 9935bd2ffa | |||
| 4260903bbe | |||
| 64ae8ef849 | |||
| be15f2f5a0 | |||
| 341de6a196 | |||
| 740f6966ba | |||
| 2e889ff9fe | |||
| d123abd4cd | |||
| cc84349d6d | |||
| b5e83d8866 | |||
| ca9521ac9a | |||
| 078f404fe4 | |||
| bc47dfa343 | |||
| 03f39d71e3 | |||
| 69cd6aa4d0 | |||
| e161f70710 | |||
| 3ef0173226 | |||
| 87a308f749 | |||
| f45a1c4161 | |||
| 5e15695f64 | |||
| b35997e8be | |||
| fcd664c16e | |||
| 94741598af | |||
| 6cb418a871 | |||
| d234fdea11 | |||
| 13ac5f1d2a | |||
| f4d6e65380 | |||
| baccbaf82a | |||
| d6969e0b85 | |||
| b3ad9c95ce | |||
| 2f6417dec2 | |||
| f54d48270e | |||
| 57321084af | |||
| 1d97422011 | |||
| 2f2a64b89e | |||
| b2e129eec7 | |||
| cb79e48685 | |||
| e735ef7869 | |||
| 8f1152762a | |||
| 587adb3773 |
@@ -11,3 +11,4 @@ dev.js
|
|||||||
test/
|
test/
|
||||||
/client/.nuxt/
|
/client/.nuxt/
|
||||||
/client/dist/
|
/client/dist/
|
||||||
|
/dist/
|
||||||
@@ -8,3 +8,4 @@ node_modules/
|
|||||||
test/
|
test/
|
||||||
/client/.nuxt/
|
/client/.nuxt/
|
||||||
/client/dist/
|
/client/dist/
|
||||||
|
/dist/
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
Package: audiobookshelf
|
||||||
|
Version: 1.2.1
|
||||||
|
Section: base
|
||||||
|
Priority: optional
|
||||||
|
Architecture: amd64
|
||||||
|
Depends:
|
||||||
|
Maintainer: advplyr
|
||||||
|
Description: Self-hosted audiobook server for managing and playing audiobooks
|
||||||
@@ -0,0 +1,85 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
set -o pipefail
|
||||||
|
|
||||||
|
declare -r init_type='auto'
|
||||||
|
declare -ri no_rebuild='0'
|
||||||
|
|
||||||
|
add_user() {
|
||||||
|
: "${1:?'User was not defined'}"
|
||||||
|
declare -r user="$1"
|
||||||
|
declare -r uid="$2"
|
||||||
|
|
||||||
|
if [ -z "$uid" ]; then
|
||||||
|
declare -r uid_flags=""
|
||||||
|
else
|
||||||
|
declare -r uid_flags="--uid $uid"
|
||||||
|
fi
|
||||||
|
|
||||||
|
declare -r group="${3:-$user}"
|
||||||
|
declare -r descr="${4:-No description}"
|
||||||
|
declare -r shell="${5:-/bin/false}"
|
||||||
|
|
||||||
|
if ! getent passwd | grep -q "^$user:"; then
|
||||||
|
echo "Creating system user: $user in $group with $descr and shell $shell"
|
||||||
|
useradd $uid_flags --gid $group --no-create-home --system --shell $shell -c "$descr" $user
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
add_group() {
|
||||||
|
: "${1:?'Group was not defined'}"
|
||||||
|
declare -r group="$1"
|
||||||
|
declare -r gid="$2"
|
||||||
|
|
||||||
|
if [ -z "$gid" ]; then
|
||||||
|
declare -r gid_flags=""
|
||||||
|
else
|
||||||
|
declare -r gid_flags="--gid $gid"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! getent group | grep -q "^$group:" ; then
|
||||||
|
echo "Creating system group: $group"
|
||||||
|
groupadd $gid_flags --system $group
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
start_service () {
|
||||||
|
: "${1:?'Service name was not defined'}"
|
||||||
|
declare -r service_name="$1"
|
||||||
|
|
||||||
|
if hash systemctl 2> /dev/null; then
|
||||||
|
if [[ "$init_type" == 'auto' || "$init_type" == 'systemd' ]]; then
|
||||||
|
{
|
||||||
|
systemctl enable "$service_name.service" && \
|
||||||
|
systemctl start "$service_name.service"
|
||||||
|
} || echo "$service_name could not be registered or started"
|
||||||
|
fi
|
||||||
|
elif hash service 2> /dev/null; then
|
||||||
|
if [[ "$init_type" == 'auto' || "$init_type" == 'upstart' || "$init_type" == 'sysv' ]]; then
|
||||||
|
service "$service_name" start || echo "$service_name could not be registered or started"
|
||||||
|
fi
|
||||||
|
elif hash start 2> /dev/null; then
|
||||||
|
if [[ "$init_type" == 'auto' || "$init_type" == 'upstart' ]]; then
|
||||||
|
start "$service_name" || echo "$service_name could not be registered or started"
|
||||||
|
fi
|
||||||
|
elif hash update-rc.d 2> /dev/null; then
|
||||||
|
if [[ "$init_type" == 'auto' || "$init_type" == 'sysv' ]]; then
|
||||||
|
{
|
||||||
|
update-rc.d "$service_name" defaults && \
|
||||||
|
"/etc/init.d/$service_name" start
|
||||||
|
} || echo "$service_name could not be registered or started"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
echo 'Your system does not appear to use systemd, Upstart, or System V, so the service could not be started'
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
add_group 'audiobookshelf' ''
|
||||||
|
add_user 'audiobookshelf' '' 'audiobookshelf' 'audiobookshelf user-daemon' '/bin/false'
|
||||||
|
|
||||||
|
mkdir -p '/var/log/audiobookshelf'
|
||||||
|
chown -R 'audiobookshelf:audiobookshelf' '/var/log/audiobookshelf'
|
||||||
|
chown -R 'audiobookshelf:audiobookshelf' '/usr/share/audiobookshelf'
|
||||||
|
|
||||||
|
start_service 'audiobookshelf'
|
||||||
@@ -0,0 +1,99 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
set -o pipefail
|
||||||
|
|
||||||
|
DEFAULT_AUDIOBOOK_PATH="/usr/share/audiobookshelf/audiobooks"
|
||||||
|
DEFAULT_DATA_PATH="/usr/share/audiobookshelf"
|
||||||
|
DEFAULT_PORT=7331
|
||||||
|
|
||||||
|
CONFIG_PATH="/etc/default/audiobookshelf"
|
||||||
|
|
||||||
|
install_ffmpeg() {
|
||||||
|
echo "Starting FFMPEG Install"
|
||||||
|
|
||||||
|
WGET="wget https://johnvansickle.com/ffmpeg/builds/ffmpeg-git-amd64-static.tar.xz"
|
||||||
|
TARGET_DIR="/usr/lib/audiobookshelf-ffmpeg/"
|
||||||
|
|
||||||
|
if ! cd "$TARGET_DIR"; then
|
||||||
|
echo "WARNING: can't access working directory ($TARGET_DIR) creating it" >&2
|
||||||
|
mkdir "$TARGET_DIR"
|
||||||
|
cd "$TARGET_DIR"
|
||||||
|
fi
|
||||||
|
|
||||||
|
$WGET
|
||||||
|
tar xvf ffmpeg-git-amd64-static.tar.xz --strip-components=1
|
||||||
|
rm ffmpeg-git-amd64-static.tar.xz
|
||||||
|
|
||||||
|
chown -R 'audiobookshelf:audiobookshelf' "$TARGET_DIR"
|
||||||
|
|
||||||
|
echo "Good to go on Ffmpeg... hopefully"
|
||||||
|
}
|
||||||
|
|
||||||
|
should_build_config() {
|
||||||
|
if [ -f "$CONFIG_PATH" ]; then
|
||||||
|
echo "You already have a config file. Do you want to use it?"
|
||||||
|
|
||||||
|
options=("Yes" "No")
|
||||||
|
select yn in "${options[@]}"
|
||||||
|
do
|
||||||
|
case $yn in
|
||||||
|
"Yes")
|
||||||
|
false; return
|
||||||
|
;;
|
||||||
|
"No")
|
||||||
|
true; return
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
else
|
||||||
|
echo "No existing config found in $CONFIG_PATH"
|
||||||
|
true; return
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
setup_config() {
|
||||||
|
if should_build_config; then
|
||||||
|
echo "Okay, let's setup a new config."
|
||||||
|
|
||||||
|
AUDIOBOOK_PATH=""
|
||||||
|
read -p "
|
||||||
|
Enter path for your audiobooks [Default: $DEFAULT_AUDIOBOOK_PATH]:" AUDIOBOOK_PATH
|
||||||
|
|
||||||
|
if [[ -z "$AUDIOBOOK_PATH" ]]; then
|
||||||
|
AUDIOBOOK_PATH="$DEFAULT_AUDIOBOOK_PATH"
|
||||||
|
fi
|
||||||
|
|
||||||
|
DATA_PATH=""
|
||||||
|
read -p "
|
||||||
|
Enter path for data files, i.e. streams, downloads, database [Default: $DEFAULT_DATA_PATH]:" DATA_PATH
|
||||||
|
|
||||||
|
if [[ -z "$DATA_PATH" ]]; then
|
||||||
|
DATA_PATH="$DEFAULT_DATA_PATH"
|
||||||
|
fi
|
||||||
|
|
||||||
|
PORT=""
|
||||||
|
read -p "
|
||||||
|
Port for the web ui [Default: $DEFAULT_PORT]:" PORT
|
||||||
|
|
||||||
|
if [[ -z "$PORT" ]]; then
|
||||||
|
PORT="$DEFAULT_PORT"
|
||||||
|
fi
|
||||||
|
|
||||||
|
config_text="AUDIOBOOK_PATH=$AUDIOBOOK_PATH
|
||||||
|
METADATA_PATH=$DATA_PATH/metadata
|
||||||
|
CONFIG_PATH=$DATA_PATH/config
|
||||||
|
FFMPEG_PATH=/usr/lib/audiobookshelf-ffmpeg/ffmpeg
|
||||||
|
PORT=$PORT"
|
||||||
|
|
||||||
|
echo "$config_text"
|
||||||
|
|
||||||
|
echo "$config_text" > /etc/default/audiobookshelf;
|
||||||
|
|
||||||
|
echo "Config created"
|
||||||
|
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
setup_config
|
||||||
|
|
||||||
|
install_ffmpeg
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
set -o pipefail
|
||||||
|
|
||||||
|
declare -r init_type='auto'
|
||||||
|
declare -r service_name='audiobookshelf'
|
||||||
|
|
||||||
|
if [[ "$init_type" == 'auto' || "$init_type" == 'systemd' || "$init_type" == 'upstart' || "$init_type" == 'sysv' ]]; then
|
||||||
|
if hash systemctl 2> /dev/null; then
|
||||||
|
systemctl disable "$service_name.service" && \
|
||||||
|
systemctl stop "$service_name.service" || \
|
||||||
|
echo "$service_name wasn't even running!"
|
||||||
|
elif hash service 2> /dev/null; then
|
||||||
|
service "$service_name" stop || echo "$service_name wasn't even running!"
|
||||||
|
elif hash stop 2> /dev/null; then
|
||||||
|
stop "$service_name" || echo "$service_name wasn't even running!"
|
||||||
|
elif hash update-rc.d 2> /dev/null; then
|
||||||
|
{
|
||||||
|
update-rc.d "$service_name" remove && \
|
||||||
|
"/etc/init.d/$service_name" stop
|
||||||
|
} || "$service_name wasn't even running!"
|
||||||
|
else
|
||||||
|
echo "Your system does not appear to use upstart, systemd or sysv, so $service_name could not be stopped"
|
||||||
|
echo 'Unless these systems were removed since install, no processes have been left running'
|
||||||
|
fi
|
||||||
|
fi
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=Self-hosted audiobook server for managing and playing audiobooks
|
||||||
|
Requires=network.target
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=simple
|
||||||
|
EnvironmentFile=/etc/default/audiobookshelf
|
||||||
|
WorkingDirectory=/usr/share/audiobookshelf
|
||||||
|
ExecStart=/usr/share/audiobookshelf/audiobookshelf
|
||||||
|
ExecReload=/bin/kill -HUP $MAINPID
|
||||||
|
Restart=always
|
||||||
|
User=audiobookshelf
|
||||||
|
PermissionsStartOnly=true
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
Executable
+57
@@ -0,0 +1,57 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
set -o pipefail
|
||||||
|
|
||||||
|
SCRIPT_DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" &> /dev/null && pwd )"
|
||||||
|
|
||||||
|
cd "$SCRIPT_DIR/.."
|
||||||
|
|
||||||
|
# Get package version without double quotes
|
||||||
|
VERSION="$( eval echo $( jq '.version' package.json) )"
|
||||||
|
DESCRIPTION="$( eval echo $( jq '.description' package.json) )"
|
||||||
|
OUTPUT_FILE="audiobookshelf_${VERSION}_amd64.deb"
|
||||||
|
|
||||||
|
echo ">>> Building Client"
|
||||||
|
echo "--------------------"
|
||||||
|
|
||||||
|
cd client
|
||||||
|
rm -rf node_modules
|
||||||
|
npm ci --unsafe-perm=true --allow-root
|
||||||
|
npm run generate
|
||||||
|
cd ..
|
||||||
|
|
||||||
|
echo ">>> Building Server"
|
||||||
|
echo "--------------------"
|
||||||
|
|
||||||
|
rm -rf node_modules
|
||||||
|
npm ci --unsafe-perm=true --allow-root
|
||||||
|
|
||||||
|
echo ">>> Packaging"
|
||||||
|
echo "--------------------"
|
||||||
|
|
||||||
|
# Create debian control file
|
||||||
|
|
||||||
|
mkdir -p dist
|
||||||
|
cp -R build/debian dist/debian
|
||||||
|
chmod -R 775 dist/debian
|
||||||
|
|
||||||
|
controlfile="Package: audiobookshelf
|
||||||
|
Version: $VERSION
|
||||||
|
Section: base
|
||||||
|
Priority: optional
|
||||||
|
Architecture: amd64
|
||||||
|
Depends:
|
||||||
|
Maintainer: advplyr
|
||||||
|
Description: $DESCRIPTION"
|
||||||
|
|
||||||
|
echo "$controlfile" > dist/debian/DEBIAN/control;
|
||||||
|
|
||||||
|
# Package debian
|
||||||
|
pkg -t node12-linux-x64 -o dist/debian/usr/share/audiobookshelf/audiobookshelf .
|
||||||
|
|
||||||
|
fakeroot dpkg-deb --build dist/debian
|
||||||
|
|
||||||
|
mv dist/debian.deb "dist/$OUTPUT_FILE"
|
||||||
|
chmod +x "dist/$OUTPUT_FILE"
|
||||||
|
|
||||||
|
echo "Finished! Filename: $OUTPUT_FILE"
|
||||||
@@ -102,3 +102,11 @@
|
|||||||
.box-shadow-book {
|
.box-shadow-book {
|
||||||
box-shadow: 4px 1px 8px #11111166, -4px 1px 8px #11111166, 1px -4px 8px #11111166;
|
box-shadow: 4px 1px 8px #11111166, -4px 1px 8px #11111166, 1px -4px 8px #11111166;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.box-shadow-book3d {
|
||||||
|
box-shadow: 4px 1px 8px #11111166, 1px -4px 8px #11111166;
|
||||||
|
}
|
||||||
|
|
||||||
|
.box-shadow-side {
|
||||||
|
box-shadow: 5px 0px 5px #11111166;
|
||||||
|
}
|
||||||
|
|||||||
@@ -12,9 +12,6 @@
|
|||||||
<div v-if="chapters.length" class="cursor-pointer flex items-center justify-center text-gray-300" @mousedown.prevent @mouseup.prevent @click.stop="showChapters">
|
<div v-if="chapters.length" class="cursor-pointer flex items-center justify-center text-gray-300" @mousedown.prevent @mouseup.prevent @click.stop="showChapters">
|
||||||
<span class="material-icons text-3xl">format_list_bulleted</span>
|
<span class="material-icons text-3xl">format_list_bulleted</span>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="flex items-center justify-center text-gray-500">
|
|
||||||
<span class="material-icons text-3xl">format_list_bulleted</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="absolute right-32 top-0 bottom-0">
|
<div class="absolute right-32 top-0 bottom-0">
|
||||||
@@ -56,10 +53,17 @@
|
|||||||
<div ref="trackCursor" class="h-full w-0.5 bg-gray-50 absolute top-0 left-0 opacity-0 pointer-events-none" />
|
<div ref="trackCursor" class="h-full w-0.5 bg-gray-50 absolute top-0 left-0 opacity-0 pointer-events-none" />
|
||||||
<div v-if="loading" class="h-full w-1/4 absolute left-0 top-0 loadingTrack pointer-events-none bg-white bg-opacity-25" />
|
<div v-if="loading" class="h-full w-1/4 absolute left-0 top-0 loadingTrack pointer-events-none bg-white bg-opacity-25" />
|
||||||
</div>
|
</div>
|
||||||
|
<div ref="track" class="w-full h-2 relative overflow-hidden">
|
||||||
|
<template v-for="(tick, index) in chapterTicks">
|
||||||
|
<div :key="index" :style="{ left: tick.left + 'px' }" class="absolute top-0 w-px bg-white bg-opacity-50 h-1 pointer-events-none" />
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
|
||||||
<!-- Hover timestamp -->
|
<!-- Hover timestamp -->
|
||||||
<div ref="hoverTimestamp" class="absolute -top-8 left-0 bg-white text-black rounded-full opacity-0 pointer-events-none">
|
<div ref="hoverTimestamp" class="absolute -top-8 left-0 bg-white text-black rounded-full opacity-0 pointer-events-none">
|
||||||
<p ref="hoverTimestampText" class="text-xs font-mono text-center px-2 py-0.5">00:00</p>
|
<p ref="hoverTimestampText" class="text-xs font-mono text-center px-2 py-0.5 truncate whitespace-nowrap">00:00</p>
|
||||||
|
</div>
|
||||||
|
<div ref="hoverTimestampArrow" class="absolute -top-3 left-0 bg-white text-black rounded-full opacity-0 pointer-events-none">
|
||||||
<div class="absolute -bottom-1.5 left-0 right-0 w-full flex justify-center">
|
<div class="absolute -bottom-1.5 left-0 right-0 w-full flex justify-center">
|
||||||
<div class="arrow-down" />
|
<div class="arrow-down" />
|
||||||
</div>
|
</div>
|
||||||
@@ -68,7 +72,7 @@
|
|||||||
|
|
||||||
<audio ref="audio" @pause="paused" @playing="playing" @progress="progress" @timeupdate="timeupdate" @loadeddata="audioLoadedData" />
|
<audio ref="audio" @pause="paused" @playing="playing" @progress="progress" @timeupdate="timeupdate" @loadeddata="audioLoadedData" />
|
||||||
|
|
||||||
<modals-chapters-modal v-model="showChaptersModal" :chapters="chapters" @select="selectChapter" />
|
<modals-chapters-modal v-model="showChaptersModal" :current-chapter="currentChapter" :chapters="chapters" @select="selectChapter" />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -100,7 +104,9 @@ export default {
|
|||||||
totalDuration: 0,
|
totalDuration: 0,
|
||||||
seekedTime: 0,
|
seekedTime: 0,
|
||||||
seekLoading: false,
|
seekLoading: false,
|
||||||
showChaptersModal: false
|
showChaptersModal: false,
|
||||||
|
currentTime: 0,
|
||||||
|
trackOffsetLeft: 16 // Track is 16px from edge
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@@ -109,6 +115,18 @@ export default {
|
|||||||
},
|
},
|
||||||
totalDurationPretty() {
|
totalDurationPretty() {
|
||||||
return this.$secondsToTimestamp(this.totalDuration)
|
return this.$secondsToTimestamp(this.totalDuration)
|
||||||
|
},
|
||||||
|
chapterTicks() {
|
||||||
|
return this.chapters.map((chap) => {
|
||||||
|
var perc = chap.start / this.totalDuration
|
||||||
|
return {
|
||||||
|
title: chap.title,
|
||||||
|
left: perc * this.trackWidth
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
currentChapter() {
|
||||||
|
return this.chapters.find((chapter) => chapter.start <= this.currentTime && this.currentTime < chapter.end)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
@@ -172,10 +190,29 @@ export default {
|
|||||||
if (this.$refs.hoverTimestamp) {
|
if (this.$refs.hoverTimestamp) {
|
||||||
var width = this.$refs.hoverTimestamp.clientWidth
|
var width = this.$refs.hoverTimestamp.clientWidth
|
||||||
this.$refs.hoverTimestamp.style.opacity = 1
|
this.$refs.hoverTimestamp.style.opacity = 1
|
||||||
this.$refs.hoverTimestamp.style.left = offsetX - width / 2 + 'px'
|
var posLeft = offsetX - width / 2
|
||||||
|
if (posLeft + width + this.trackOffsetLeft > window.innerWidth) {
|
||||||
|
posLeft = window.innerWidth - width - this.trackOffsetLeft
|
||||||
|
} else if (posLeft < -this.trackOffsetLeft) {
|
||||||
|
posLeft = -this.trackOffsetLeft
|
||||||
|
}
|
||||||
|
this.$refs.hoverTimestamp.style.left = posLeft + 'px'
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.$refs.hoverTimestampArrow) {
|
||||||
|
var width = this.$refs.hoverTimestampArrow.clientWidth
|
||||||
|
var posLeft = offsetX - width / 2
|
||||||
|
this.$refs.hoverTimestampArrow.style.opacity = 1
|
||||||
|
this.$refs.hoverTimestampArrow.style.left = posLeft + 'px'
|
||||||
}
|
}
|
||||||
if (this.$refs.hoverTimestampText) {
|
if (this.$refs.hoverTimestampText) {
|
||||||
this.$refs.hoverTimestampText.innerText = this.$secondsToTimestamp(time)
|
var hoverText = this.$secondsToTimestamp(time)
|
||||||
|
|
||||||
|
var chapter = this.chapters.find((chapter) => chapter.start <= time && time < chapter.end)
|
||||||
|
if (chapter && chapter.title) {
|
||||||
|
hoverText += ` - ${chapter.title}`
|
||||||
|
}
|
||||||
|
this.$refs.hoverTimestampText.innerText = hoverText
|
||||||
}
|
}
|
||||||
if (this.$refs.trackCursor) {
|
if (this.$refs.trackCursor) {
|
||||||
this.$refs.trackCursor.style.opacity = 1
|
this.$refs.trackCursor.style.opacity = 1
|
||||||
@@ -186,6 +223,9 @@ export default {
|
|||||||
if (this.$refs.hoverTimestamp) {
|
if (this.$refs.hoverTimestamp) {
|
||||||
this.$refs.hoverTimestamp.style.opacity = 0
|
this.$refs.hoverTimestamp.style.opacity = 0
|
||||||
}
|
}
|
||||||
|
if (this.$refs.hoverTimestampArrow) {
|
||||||
|
this.$refs.hoverTimestampArrow.style.opacity = 0
|
||||||
|
}
|
||||||
if (this.$refs.trackCursor) {
|
if (this.$refs.trackCursor) {
|
||||||
this.$refs.trackCursor.style.opacity = 0
|
this.$refs.trackCursor.style.opacity = 0
|
||||||
}
|
}
|
||||||
@@ -289,7 +329,6 @@ export default {
|
|||||||
end: end + offset
|
end: end + offset
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
return ranges
|
return ranges
|
||||||
},
|
},
|
||||||
getLastBufferedTime() {
|
getLastBufferedTime() {
|
||||||
@@ -315,7 +354,6 @@ export default {
|
|||||||
this.bufferTrackWidth = bufferlen
|
this.bufferTrackWidth = bufferlen
|
||||||
},
|
},
|
||||||
timeupdate() {
|
timeupdate() {
|
||||||
// console.log('Time update', this.audioEl.currentTime)
|
|
||||||
if (!this.$refs.playedTrack) {
|
if (!this.$refs.playedTrack) {
|
||||||
console.error('Invalid no played track ref')
|
console.error('Invalid no played track ref')
|
||||||
return
|
return
|
||||||
@@ -335,6 +373,8 @@ export default {
|
|||||||
|
|
||||||
this.updateTimestamp()
|
this.updateTimestamp()
|
||||||
|
|
||||||
|
this.currentTime = this.audioEl.currentTime
|
||||||
|
|
||||||
var perc = this.audioEl.currentTime / this.audioEl.duration
|
var perc = this.audioEl.currentTime / this.audioEl.duration
|
||||||
var ptWidth = Math.round(perc * this.trackWidth)
|
var ptWidth = Math.round(perc * this.trackWidth)
|
||||||
if (this.playedTrackWidth === ptWidth) {
|
if (this.playedTrackWidth === ptWidth) {
|
||||||
|
|||||||
@@ -7,20 +7,21 @@
|
|||||||
<span class="material-icons text-4xl text-white">arrow_back</span>
|
<span class="material-icons text-4xl text-white">arrow_back</span>
|
||||||
</a>
|
</a>
|
||||||
<h1 class="text-2xl font-book mr-6">AudioBookshelf</h1>
|
<h1 class="text-2xl font-book mr-6">AudioBookshelf</h1>
|
||||||
|
<!-- <div class="-mb-2">
|
||||||
|
<h1 class="text-lg font-book leading-3 mr-6 px-1">AudioBookshelf</h1>
|
||||||
|
<div class="bg-black bg-opacity-20 rounded-sm py-1.5 px-2 mt-1.5 flex items-center justify-between border border-bg">
|
||||||
|
<p class="text-sm text-gray-400 leading-3">My Library</p>
|
||||||
|
<span class="material-icons text-sm leading-3 text-gray-400">expand_more</span>
|
||||||
|
</div>
|
||||||
|
</div> -->
|
||||||
<controls-global-search />
|
<controls-global-search />
|
||||||
<div class="flex-grow" />
|
<div class="flex-grow" />
|
||||||
|
|
||||||
<!-- <a v-if="isUpdateAvailable" :href="githubTagUrl" target="_blank" class="flex items-center rounded-full bg-warning p-2 text-sm">
|
<nuxt-link v-if="userCanUpload" to="/upload" class="outline-none hover:text-gray-200 cursor-pointer w-8 h-8 flex items-center justify-center">
|
||||||
<span class="material-icons">notification_important</span>
|
|
||||||
<span class="pl-2">Update is available! Check release notes for v{{ latestVersion }}</span>
|
|
||||||
</a> -->
|
|
||||||
|
|
||||||
<nuxt-link v-if="isRootUser" to="/upload" class="outline-none hover:text-gray-200 cursor-pointer w-8 h-8 flex items-center justify-center mr-4">
|
|
||||||
<span class="material-icons">upload</span>
|
<span class="material-icons">upload</span>
|
||||||
</nuxt-link>
|
</nuxt-link>
|
||||||
|
|
||||||
<nuxt-link v-if="isRootUser" to="/config" class="outline-none hover:text-gray-200 cursor-pointer w-8 h-8 flex items-center justify-center">
|
<nuxt-link v-if="isRootUser" to="/config" class="outline-none hover:text-gray-200 cursor-pointer w-8 h-8 flex items-center justify-center ml-4">
|
||||||
<span class="material-icons">settings</span>
|
<span class="material-icons">settings</span>
|
||||||
</nuxt-link>
|
</nuxt-link>
|
||||||
|
|
||||||
@@ -45,10 +46,8 @@
|
|||||||
</ui-tooltip>
|
</ui-tooltip>
|
||||||
<template v-if="userCanUpdate">
|
<template v-if="userCanUpdate">
|
||||||
<ui-icon-btn v-show="!processingBatchDelete" icon="edit" bg-color="warning" class="mx-1.5" @click="batchEditClick" />
|
<ui-icon-btn v-show="!processingBatchDelete" icon="edit" bg-color="warning" class="mx-1.5" @click="batchEditClick" />
|
||||||
<!-- <ui-btn v-show="!processingBatchDelete" color="warning" small class="mx-2 w-10 h-10" :padding-y="0" :padding-x="0" @click="batchEditClick"><span class="material-icons text-gray-200 text-base">edit</span></ui-btn> -->
|
|
||||||
</template>
|
</template>
|
||||||
<ui-icon-btn v-show="userCanDelete" :disabled="processingBatchDelete" icon="delete" bg-color="error" class="mx-1.5" @click="batchDeleteClick" />
|
<ui-icon-btn v-show="userCanDelete" :disabled="processingBatchDelete" icon="delete" bg-color="error" class="mx-1.5" @click="batchDeleteClick" />
|
||||||
<!-- <ui-btn v-if="userCanDelete" color="error" small class="mx-2" :loading="processingBatchDelete" @click="batchDeleteClick"><span class="material-icons text-gray-200 pt-1">delete</span></ui-btn> -->
|
|
||||||
<span class="material-icons text-4xl px-4 hover:text-gray-100 cursor-pointer" :class="processingBatchDelete ? 'text-gray-400' : ''" @click="cancelSelectionMode">close</span>
|
<span class="material-icons text-4xl px-4 hover:text-gray-100 cursor-pointer" :class="processingBatchDelete ? 'text-gray-400' : ''" @click="cancelSelectionMode">close</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -64,7 +63,7 @@ export default {
|
|||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
showBack() {
|
showBack() {
|
||||||
return this.$route.name !== 'index'
|
return this.$route.name !== 'library-id'
|
||||||
},
|
},
|
||||||
user() {
|
user() {
|
||||||
return this.$store.state.user.user
|
return this.$store.state.user.user
|
||||||
@@ -96,6 +95,9 @@ export default {
|
|||||||
userCanDelete() {
|
userCanDelete() {
|
||||||
return this.$store.getters['user/getUserCanDelete']
|
return this.$store.getters['user/getUserCanDelete']
|
||||||
},
|
},
|
||||||
|
userCanUpload() {
|
||||||
|
return this.$store.getters['user/getUserCanUpload']
|
||||||
|
},
|
||||||
selectedIsRead() {
|
selectedIsRead() {
|
||||||
// Find an audiobook that is not read, if none then all audiobooks read
|
// Find an audiobook that is not read, if none then all audiobooks read
|
||||||
return !this.selectedAudiobooks.find((ab) => {
|
return !this.selectedAudiobooks.find((ab) => {
|
||||||
@@ -112,7 +114,7 @@ export default {
|
|||||||
if (this.$route.name === 'audiobook-id-edit') {
|
if (this.$route.name === 'audiobook-id-edit') {
|
||||||
this.$router.push(`/audiobook/${this.$route.params.id}`)
|
this.$router.push(`/audiobook/${this.$route.params.id}`)
|
||||||
} else {
|
} else {
|
||||||
this.$router.push('/')
|
this.$router.push('/library')
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
cancelSelectionMode() {
|
cancelSelectionMode() {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<div id="bookshelf" ref="wrapper" class="w-full h-full overflow-y-auto relative">
|
<div id="bookshelf" ref="wrapper" class="w-full h-full overflow-y-scroll relative">
|
||||||
<!-- Cover size widget -->
|
<!-- Cover size widget -->
|
||||||
<div v-show="!isSelectionMode" class="fixed bottom-2 right-4 z-20">
|
<div v-show="!isSelectionMode" class="fixed bottom-2 right-4 z-20">
|
||||||
<div class="rounded-full py-1 bg-primary px-2 border border-black-100 text-center flex items-center box-shadow-md" @mousedown.prevent @mouseup.prevent>
|
<div class="rounded-full py-1 bg-primary px-2 border border-black-100 text-center flex items-center box-shadow-md" @mousedown.prevent @mouseup.prevent>
|
||||||
@@ -10,23 +10,30 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="!audiobooks.length" class="w-full flex flex-col items-center justify-center py-12">
|
<div v-if="!audiobooks.length" class="w-full flex flex-col items-center justify-center py-12">
|
||||||
<p class="text-center text-2xl font-book mb-4">Your Audiobookshelf is empty!</p>
|
<p class="text-center text-2xl font-book mb-4 py-4">Your Audiobookshelf is empty!</p>
|
||||||
<ui-btn color="success" @click="scan">Scan your Audiobooks</ui-btn>
|
<div class="flex">
|
||||||
|
<ui-btn to="/config" color="primary" class="w-52 mr-2" @click="scan">Configure Scanner</ui-btn>
|
||||||
|
<ui-btn color="success" class="w-52" @click="scan">Scan Audiobooks</ui-btn>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="w-full flex flex-col items-center">
|
<div v-else id="bookshelf" class="w-full flex flex-col items-center">
|
||||||
<template v-for="(shelf, index) in groupedBooks">
|
<template v-for="(shelf, index) in shelves">
|
||||||
<div :key="index" class="w-full bookshelfRow relative">
|
<div :key="index" class="w-full bookshelfRow relative">
|
||||||
<div class="flex justify-center items-center">
|
<div class="flex justify-center items-center">
|
||||||
<template v-for="audiobook in shelf">
|
<template v-for="entity in shelf">
|
||||||
<cards-book-card :ref="`audiobookCard-${audiobook.id}`" :key="audiobook.id" :width="bookCoverWidth" :user-progress="userAudiobooks[audiobook.id]" :audiobook="audiobook" />
|
<cards-group-card v-if="showGroups" :key="entity.id" :width="bookCoverWidth" :group="entity" @click="clickGroup" />
|
||||||
|
<!-- <cards-book-3d :key="entity.id" v-else :width="100" :src="$store.getters['audiobooks/getBookCoverSrc'](entity.book)" /> -->
|
||||||
|
<cards-book-card v-else :key="entity.id" :show-volume-number="selectedSeries" :width="bookCoverWidth" :user-progress="userAudiobooks[entity.id]" :audiobook="entity" />
|
||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
<div class="bookshelfDivider h-4 w-full absolute bottom-0 left-0 right-0 z-10" />
|
<div class="bookshelfDivider h-4 w-full absolute bottom-0 left-0 right-0 z-10" />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<div v-show="!groupedBooks.length" class="w-full py-16 text-center text-xl">
|
<div v-show="!shelves.length" class="w-full py-16 text-center text-xl">
|
||||||
<div class="py-4">No Audiobooks</div>
|
<div v-if="page === 'search'" class="py-4 mb-6"><p class="text-2xl">No Results</p></div>
|
||||||
<ui-btn v-if="filterBy !== 'all' || keywordFilter" @click="clearFilter">Clear Filter</ui-btn>
|
<div v-else class="py-4">No {{ showGroups ? 'Series' : 'Audiobooks' }}</div>
|
||||||
|
<ui-btn v-if="!showGroups && (filterBy !== 'all' || keywordFilter)" @click="clearFilter">Clear Filter</ui-btn>
|
||||||
|
<ui-btn v-else-if="page === 'search'" to="/library">Back to Library</ui-btn>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -34,21 +41,40 @@
|
|||||||
|
|
||||||
<script>
|
<script>
|
||||||
export default {
|
export default {
|
||||||
|
props: {
|
||||||
|
page: String,
|
||||||
|
selectedSeries: String,
|
||||||
|
searchResults: {
|
||||||
|
type: Array,
|
||||||
|
default: () => []
|
||||||
|
},
|
||||||
|
searchQuery: String
|
||||||
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
width: 0,
|
shelves: [],
|
||||||
booksPerRow: 0,
|
|
||||||
groupedBooks: [],
|
|
||||||
currFilterOrderKey: null,
|
currFilterOrderKey: null,
|
||||||
availableSizes: [60, 80, 100, 120, 140, 160, 180, 200, 220],
|
availableSizes: [60, 80, 100, 120, 140, 160, 180, 200, 220],
|
||||||
selectedSizeIndex: 3,
|
selectedSizeIndex: 3,
|
||||||
rowPaddingX: 40,
|
rowPaddingX: 40,
|
||||||
keywordFilterTimeout: null
|
keywordFilterTimeout: null,
|
||||||
|
scannerParseSubtitle: false,
|
||||||
|
wrapperClientWidth: 0
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
keywordFilter() {
|
keywordFilter() {
|
||||||
this.checkKeywordFilter()
|
this.checkKeywordFilter()
|
||||||
|
},
|
||||||
|
selectedSeries() {
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.setBookshelfEntities()
|
||||||
|
})
|
||||||
|
},
|
||||||
|
searchResults() {
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.setBookshelfEntities()
|
||||||
|
})
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@@ -81,9 +107,32 @@ export default {
|
|||||||
},
|
},
|
||||||
filterBy() {
|
filterBy() {
|
||||||
return this.$store.getters['user/getUserSetting']('filterBy')
|
return this.$store.getters['user/getUserSetting']('filterBy')
|
||||||
|
},
|
||||||
|
showGroups() {
|
||||||
|
return this.page !== '' && this.page !== 'search' && !this.selectedSeries
|
||||||
|
},
|
||||||
|
entities() {
|
||||||
|
if (this.page === '') {
|
||||||
|
return this.$store.getters['audiobooks/getFilteredAndSorted']()
|
||||||
|
} else if (this.page === 'search') {
|
||||||
|
return this.searchResults || []
|
||||||
|
} else {
|
||||||
|
var seriesGroups = this.$store.getters['audiobooks/getSeriesGroups']()
|
||||||
|
if (this.selectedSeries) {
|
||||||
|
var group = seriesGroups.find((group) => group.name === this.selectedSeries)
|
||||||
|
return group.books
|
||||||
|
}
|
||||||
|
return seriesGroups
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
clickGroup(group) {
|
||||||
|
this.$emit('update:selectedSeries', group.name)
|
||||||
|
},
|
||||||
|
changeRotation() {
|
||||||
|
this.rotation = 'show-right'
|
||||||
|
},
|
||||||
clearFilter() {
|
clearFilter() {
|
||||||
this.$store.commit('audiobooks/setKeywordFilter', null)
|
this.$store.commit('audiobooks/setKeywordFilter', null)
|
||||||
if (this.filterBy !== 'all') {
|
if (this.filterBy !== 'all') {
|
||||||
@@ -91,13 +140,13 @@ export default {
|
|||||||
filterBy: 'all'
|
filterBy: 'all'
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
this.setGroupedBooks()
|
this.setBookshelfEntities()
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
checkKeywordFilter() {
|
checkKeywordFilter() {
|
||||||
clearTimeout(this.keywordFilterTimeout)
|
clearTimeout(this.keywordFilterTimeout)
|
||||||
this.keywordFilterTimeout = setTimeout(() => {
|
this.keywordFilterTimeout = setTimeout(() => {
|
||||||
this.setGroupedBooks()
|
this.setBookshelfEntities()
|
||||||
}, 500)
|
}, 500)
|
||||||
},
|
},
|
||||||
increaseSize() {
|
increaseSize() {
|
||||||
@@ -110,59 +159,52 @@ export default {
|
|||||||
this.resize()
|
this.resize()
|
||||||
this.$store.dispatch('user/updateUserSettings', { bookshelfCoverSize: this.bookCoverWidth })
|
this.$store.dispatch('user/updateUserSettings', { bookshelfCoverSize: this.bookCoverWidth })
|
||||||
},
|
},
|
||||||
setGroupedBooks() {
|
setBookshelfEntities() {
|
||||||
|
this.wrapperClientWidth = this.$refs.wrapper.clientWidth
|
||||||
|
var width = Math.max(0, this.wrapperClientWidth - this.rowPaddingX * 2)
|
||||||
|
var booksPerRow = Math.floor(width / this.bookWidth)
|
||||||
|
|
||||||
|
var entities = this.entities
|
||||||
var groups = []
|
var groups = []
|
||||||
var currentRow = 0
|
var currentRow = 0
|
||||||
var currentGroup = []
|
var currentGroup = []
|
||||||
|
|
||||||
var audiobooksSorted = this.$store.getters['audiobooks/getFilteredAndSorted']()
|
for (let i = 0; i < entities.length; i++) {
|
||||||
this.currFilterOrderKey = this.filterOrderKey
|
var row = Math.floor(i / booksPerRow)
|
||||||
|
|
||||||
for (let i = 0; i < audiobooksSorted.length; i++) {
|
|
||||||
var row = Math.floor(i / this.booksPerRow)
|
|
||||||
if (row > currentRow) {
|
if (row > currentRow) {
|
||||||
groups.push([...currentGroup])
|
groups.push([...currentGroup])
|
||||||
currentRow = row
|
currentRow = row
|
||||||
currentGroup = []
|
currentGroup = []
|
||||||
}
|
}
|
||||||
currentGroup.push(audiobooksSorted[i])
|
currentGroup.push(entities[i])
|
||||||
}
|
}
|
||||||
if (currentGroup.length) {
|
if (currentGroup.length) {
|
||||||
groups.push([...currentGroup])
|
groups.push([...currentGroup])
|
||||||
}
|
}
|
||||||
this.groupedBooks = groups
|
this.shelves = groups
|
||||||
},
|
},
|
||||||
calculateBookshelf() {
|
async init() {
|
||||||
this.width = this.$refs.wrapper.clientWidth
|
this.wrapperClientWidth = this.$refs.wrapper ? this.$refs.wrapper.clientWidth : 0
|
||||||
this.width = Math.max(0, this.width - this.rowPaddingX * 2)
|
|
||||||
var booksPerRow = Math.floor(this.width / this.bookWidth)
|
|
||||||
this.booksPerRow = booksPerRow
|
|
||||||
},
|
|
||||||
getAudiobookCard(id) {
|
|
||||||
if (this.$refs[`audiobookCard-${id}`] && this.$refs[`audiobookCard-${id}`].length) {
|
|
||||||
return this.$refs[`audiobookCard-${id}`][0]
|
|
||||||
}
|
|
||||||
return null
|
|
||||||
},
|
|
||||||
init() {
|
|
||||||
var bookshelfCoverSize = this.$store.getters['user/getUserSetting']('bookshelfCoverSize')
|
var bookshelfCoverSize = this.$store.getters['user/getUserSetting']('bookshelfCoverSize')
|
||||||
var sizeIndex = this.availableSizes.findIndex((s) => s === bookshelfCoverSize)
|
var sizeIndex = this.availableSizes.findIndex((s) => s === bookshelfCoverSize)
|
||||||
if (!isNaN(sizeIndex)) this.selectedSizeIndex = sizeIndex
|
if (!isNaN(sizeIndex)) this.selectedSizeIndex = sizeIndex
|
||||||
this.calculateBookshelf()
|
|
||||||
|
var isLoading = await this.$store.dispatch('audiobooks/load')
|
||||||
|
if (!isLoading) {
|
||||||
|
this.setBookshelfEntities()
|
||||||
|
}
|
||||||
},
|
},
|
||||||
resize() {
|
resize() {
|
||||||
this.$nextTick(() => {
|
this.$nextTick(this.setBookshelfEntities)
|
||||||
this.calculateBookshelf()
|
|
||||||
this.setGroupedBooks()
|
|
||||||
})
|
|
||||||
},
|
},
|
||||||
audiobooksUpdated() {
|
audiobooksUpdated() {
|
||||||
console.log('[AudioBookshelf] Audiobooks Updated')
|
console.log('[AudioBookshelf] Audiobooks Updated')
|
||||||
this.setGroupedBooks()
|
this.setBookshelfEntities()
|
||||||
},
|
},
|
||||||
settingsUpdated(settings) {
|
settingsUpdated(settings) {
|
||||||
if (this.currFilterOrderKey !== this.filterOrderKey) {
|
if (this.currFilterOrderKey !== this.filterOrderKey) {
|
||||||
this.setGroupedBooks()
|
this.setBookshelfEntities()
|
||||||
}
|
}
|
||||||
if (settings.bookshelfCoverSize !== this.bookCoverWidth && settings.bookshelfCoverSize !== undefined) {
|
if (settings.bookshelfCoverSize !== this.bookCoverWidth && settings.bookshelfCoverSize !== undefined) {
|
||||||
var index = this.availableSizes.indexOf(settings.bookshelfCoverSize)
|
var index = this.availableSizes.indexOf(settings.bookshelfCoverSize)
|
||||||
@@ -176,18 +218,24 @@ export default {
|
|||||||
this.$root.socket.emit('scan')
|
this.$root.socket.emit('scan')
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
updated() {
|
||||||
|
if (this.$refs.wrapper) {
|
||||||
|
if (this.wrapperClientWidth !== this.$refs.wrapper.clientWidth) {
|
||||||
|
this.$nextTick(this.setBookshelfEntities)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
|
window.addEventListener('resize', this.resize)
|
||||||
this.$store.commit('audiobooks/addListener', { id: 'bookshelf', meth: this.audiobooksUpdated })
|
this.$store.commit('audiobooks/addListener', { id: 'bookshelf', meth: this.audiobooksUpdated })
|
||||||
this.$store.commit('user/addSettingsListener', { id: 'bookshelf', meth: this.settingsUpdated })
|
this.$store.commit('user/addSettingsListener', { id: 'bookshelf', meth: this.settingsUpdated })
|
||||||
|
|
||||||
this.$store.dispatch('audiobooks/load')
|
|
||||||
this.init()
|
this.init()
|
||||||
window.addEventListener('resize', this.resize)
|
|
||||||
},
|
},
|
||||||
beforeDestroy() {
|
beforeDestroy() {
|
||||||
|
window.removeEventListener('resize', this.resize)
|
||||||
this.$store.commit('audiobooks/removeListener', 'bookshelf')
|
this.$store.commit('audiobooks/removeListener', 'bookshelf')
|
||||||
this.$store.commit('user/removeSettingsListener', 'bookshelf')
|
this.$store.commit('user/removeSettingsListener', 'bookshelf')
|
||||||
window.removeEventListener('resize', this.resize)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,20 +1,47 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="w-full h-10 relative">
|
<div class="w-full h-10 relative">
|
||||||
<div id="toolbar" class="absolute top-0 left-0 w-full h-full z-20 flex items-center px-8">
|
<div id="toolbar" class="absolute top-0 left-0 w-full h-full z-20 flex items-center px-8">
|
||||||
<p class="font-book">{{ numShowing }} Audiobooks</p>
|
<template v-if="page !== 'search'">
|
||||||
<div class="flex-grow" />
|
<p v-if="!selectedSeries" class="font-book">{{ numShowing }} {{ entityName }}</p>
|
||||||
|
<div v-else class="flex items-center">
|
||||||
|
<div @click="seriesBackArrow" class="rounded-full h-10 w-10 flex items-center justify-center hover:bg-white hover:bg-opacity-10 cursor-pointer">
|
||||||
|
<span class="material-icons text-3xl text-white">west</span>
|
||||||
|
</div>
|
||||||
|
<!-- <span class="material-icons text-2xl cursor-pointer" @click="seriesBackArrow">west</span> -->
|
||||||
|
<p class="pl-4 font-book text-lg">
|
||||||
|
{{ selectedSeries }} <span class="ml-3 font-mono text-lg bg-black bg-opacity-30 rounded-lg px-1 py-0.5">{{ numShowing }}</span>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div class="flex-grow" />
|
||||||
|
|
||||||
<ui-text-input v-model="_keywordFilter" placeholder="Keyword Filter" :padding-y="1.5" class="text-xs w-40" />
|
<ui-text-input v-show="!selectedSeries" v-model="_keywordFilter" placeholder="Keyword Filter" :padding-y="1.5" class="text-xs w-40" />
|
||||||
|
<controls-filter-select v-show="showSortFilters" v-model="settings.filterBy" class="w-48 h-7.5 ml-4" @change="updateFilter" />
|
||||||
<controls-filter-select v-model="settings.filterBy" class="w-48 h-7.5 ml-4" @change="updateFilter" />
|
<controls-order-select v-show="showSortFilters" v-model="settings.orderBy" :descending.sync="settings.orderDesc" class="w-48 h-7.5 ml-4" @change="updateOrder" />
|
||||||
|
</template>
|
||||||
<controls-order-select v-model="settings.orderBy" :descending.sync="settings.orderDesc" class="w-48 h-7.5 ml-4" @change="updateOrder" />
|
<template v-else>
|
||||||
|
<div @click="searchBackArrow" class="rounded-full h-10 w-10 flex items-center justify-center hover:bg-white hover:bg-opacity-10 cursor-pointer">
|
||||||
|
<span class="material-icons text-3xl text-white">west</span>
|
||||||
|
</div>
|
||||||
|
<!-- <p class="font-book pl-4">{{ numShowing }} showing</p> -->
|
||||||
|
<div class="flex-grow" />
|
||||||
|
<p>Search results for "{{ searchQuery }}"</p>
|
||||||
|
<div class="flex-grow" />
|
||||||
|
</template>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
export default {
|
export default {
|
||||||
|
props: {
|
||||||
|
page: String,
|
||||||
|
selectedSeries: String,
|
||||||
|
searchResults: {
|
||||||
|
type: Array,
|
||||||
|
default: () => []
|
||||||
|
},
|
||||||
|
searchQuery: String
|
||||||
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
settings: {},
|
settings: {},
|
||||||
@@ -22,8 +49,29 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
|
showSortFilters() {
|
||||||
|
return this.page === ''
|
||||||
|
},
|
||||||
numShowing() {
|
numShowing() {
|
||||||
return this.$store.getters['audiobooks/getFiltered']().length
|
if (this.page === '') {
|
||||||
|
return this.$store.getters['audiobooks/getFiltered']().length
|
||||||
|
} else if (this.page === 'search') {
|
||||||
|
return (this.searchResults || []).length
|
||||||
|
} else {
|
||||||
|
var groups = this.$store.getters['audiobooks/getSeriesGroups']()
|
||||||
|
if (this.selectedSeries) {
|
||||||
|
var group = groups.find((g) => g.name === this.selectedSeries)
|
||||||
|
if (group) return group.books.length
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
return groups.length
|
||||||
|
}
|
||||||
|
},
|
||||||
|
entityName() {
|
||||||
|
if (!this.page) return 'Audiobooks'
|
||||||
|
if (this.page === 'series') return 'Series'
|
||||||
|
if (this.page === 'collections') return 'Collections'
|
||||||
|
return ''
|
||||||
},
|
},
|
||||||
_keywordFilter: {
|
_keywordFilter: {
|
||||||
get() {
|
get() {
|
||||||
@@ -35,6 +83,13 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
searchBackArrow() {
|
||||||
|
this.$router.replace('/library')
|
||||||
|
},
|
||||||
|
seriesBackArrow() {
|
||||||
|
this.$router.replace('/library/series')
|
||||||
|
this.$emit('update:selectedSeries', null)
|
||||||
|
},
|
||||||
updateOrder() {
|
updateOrder() {
|
||||||
this.saveSettings()
|
this.saveSettings()
|
||||||
},
|
},
|
||||||
@@ -42,7 +97,6 @@ export default {
|
|||||||
this.saveSettings()
|
this.saveSettings()
|
||||||
},
|
},
|
||||||
saveSettings() {
|
saveSettings() {
|
||||||
this.$store.commit('user/setSettings', this.settings) // Immediate update
|
|
||||||
this.$store.dispatch('user/updateUserSettings', this.settings)
|
this.$store.dispatch('user/updateUserSettings', this.settings)
|
||||||
},
|
},
|
||||||
init() {
|
init() {
|
||||||
@@ -67,6 +121,6 @@ export default {
|
|||||||
|
|
||||||
<style>
|
<style>
|
||||||
#toolbar {
|
#toolbar {
|
||||||
box-shadow: 0px 8px 8px #111111aa;
|
box-shadow: 0px 8px 6px #111111aa;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
<template>
|
||||||
|
<div class="w-20 bg-bg h-full relative box-shadow-side z-20" style="min-width: 80px">
|
||||||
|
<div class="absolute top-0 -right-4 w-4 bg-bg h-10 pointer-events-none" />
|
||||||
|
<nuxt-link to="/library" class="w-full h-20 flex flex-col items-center justify-center text-white border-b border-primary border-opacity-70 hover:bg-primary cursor-pointer relative" :class="paramId === '' ? 'bg-primary bg-opacity-80' : 'bg-bg bg-opacity-60'">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="h-6 w-6" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 6.253v13m0-13C10.832 5.477 9.246 5 7.5 5S4.168 5.477 3 6.253v13C4.168 18.477 5.754 18 7.5 18s3.332.477 4.5 1.253m0-13C13.168 5.477 14.754 5 16.5 5c1.747 0 3.332.477 4.5 1.253v13C19.832 18.477 18.247 18 16.5 18c-1.746 0-3.332.477-4.5 1.253" />
|
||||||
|
</svg>
|
||||||
|
|
||||||
|
<p class="font-book pt-1.5" style="font-size: 1rem">Library</p>
|
||||||
|
|
||||||
|
<div v-show="paramId === ''" class="h-full w-0.5 bg-yellow-400 absolute top-0 left-0" />
|
||||||
|
</nuxt-link>
|
||||||
|
|
||||||
|
<nuxt-link to="/library/series" class="w-full h-20 flex flex-col items-center justify-center text-white text-opacity-80 border-b border-primary border-opacity-70 hover:bg-primary cursor-pointer relative" :class="paramId === 'series' ? 'bg-primary bg-opacity-80' : 'bg-bg bg-opacity-60'">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="h-6 w-6" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 17V7m0 10a2 2 0 01-2 2H5a2 2 0 01-2-2V7a2 2 0 012-2h2a2 2 0 012 2m0 10a2 2 0 002 2h2a2 2 0 002-2M9 7a2 2 0 012-2h2a2 2 0 012 2m0 10V7m0 10a2 2 0 002 2h2a2 2 0 002-2V7a2 2 0 00-2-2h-2a2 2 0 00-2 2" />
|
||||||
|
</svg>
|
||||||
|
|
||||||
|
<p class="font-book pt-1.5" style="font-size: 1rem">Series</p>
|
||||||
|
|
||||||
|
<div v-show="paramId === 'series'" class="h-full w-0.5 bg-yellow-400 absolute top-0 left-0" />
|
||||||
|
</nuxt-link>
|
||||||
|
|
||||||
|
<!-- <nuxt-link to="/library/collections" class="w-full h-20 flex flex-col items-center justify-center text-white text-opacity-80 border-b border-primary border-opacity-70 hover:bg-primary cursor-pointer relative" :class="paramId === 'collections' ? 'bg-primary bg-opacity-80' : 'bg-bg bg-opacity-60'">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="h-6 w-6" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 11H5m14 0a2 2 0 012 2v6a2 2 0 01-2 2H5a2 2 0 01-2-2v-6a2 2 0 012-2m14 0V9a2 2 0 00-2-2M5 11V9a2 2 0 012-2m0 0V5a2 2 0 012-2h6a2 2 0 012 2v2M7 7h10" />
|
||||||
|
</svg>
|
||||||
|
|
||||||
|
<p class="font-book pt-1.5" style="font-size: 0.8rem">Collections</p>
|
||||||
|
|
||||||
|
<div v-show="paramId === 'collections'" class="h-full w-0.5 bg-yellow-400 absolute top-0 left-0" />
|
||||||
|
</nuxt-link> -->
|
||||||
|
|
||||||
|
<!-- <nuxt-link to="/library/tags" class="w-full h-20 flex flex-col items-center justify-center text-white text-opacity-80 border-b border-primary border-opacity-70 hover:bg-primary cursor-pointer relative" :class="paramId === 'tags' ? 'bg-primary bg-opacity-80' : 'bg-bg bg-opacity-60'">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="h-6 w-6" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M7 7h.01M7 3h5c.512 0 1.024.195 1.414.586l7 7a2 2 0 010 2.828l-7 7a2 2 0 01-2.828 0l-7-7A1.994 1.994 0 013 12V7a4 4 0 014-4z" />
|
||||||
|
</svg>
|
||||||
|
|
||||||
|
<p class="font-book pt-1.5" style="font-size: 0.8rem">Tags</p>
|
||||||
|
|
||||||
|
<div v-show="paramId === 'tags'" class="h-full w-0.5 bg-yellow-400 absolute top-0 left-0" />
|
||||||
|
</nuxt-link> -->
|
||||||
|
|
||||||
|
<!-- <nuxt-link to="/library/authors" class="w-full h-20 flex flex-col items-center justify-center text-white text-opacity-80 border-b border-primary border-opacity-70 hover:bg-primary cursor-pointer relative" :class="paramId === 'authors' ? 'bg-primary bg-opacity-80' : 'bg-bg bg-opacity-60'">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" class="h-6 w-6" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4.354a4 4 0 110 5.292M15 21H3v-1a6 6 0 0112 0v1zm0 0h6v-1a6 6 0 00-9-5.197M13 7a4 4 0 11-8 0 4 4 0 018 0z" />
|
||||||
|
</svg>
|
||||||
|
|
||||||
|
<p class="font-book pt-1.5" style="font-size: 0.8rem">Authors</p>
|
||||||
|
|
||||||
|
<div v-show="paramId === 'authors'" class="h-full w-0.5 bg-yellow-400 absolute top-0 left-0" />
|
||||||
|
</nuxt-link> -->
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
paramId() {
|
||||||
|
return this.$route.params ? this.$route.params.id || '' : ''
|
||||||
|
},
|
||||||
|
selectedClassName() {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {},
|
||||||
|
mounted() {}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -1,14 +1,14 @@
|
|||||||
<template>
|
<template>
|
||||||
<div v-if="streamAudiobook" id="streamContainer" class="w-full fixed bottom-0 left-0 right-0 h-40 z-40 bg-primary p-4">
|
<div v-if="streamAudiobook" id="streamContainer" class="w-full fixed bottom-0 left-0 right-0 h-40 z-40 bg-primary p-4">
|
||||||
<div class="absolute -top-16 left-4">
|
<nuxt-link :to="`/audiobook/${streamAudiobook.id}`" class="absolute -top-16 left-4 cursor-pointer">
|
||||||
<cards-book-cover :audiobook="streamAudiobook" :width="88" />
|
<cards-book-cover :audiobook="streamAudiobook" :width="88" />
|
||||||
</div>
|
</nuxt-link>
|
||||||
<div class="flex items-center pl-24">
|
<div class="flex items-center pl-24">
|
||||||
<div>
|
<div>
|
||||||
<h1>
|
<nuxt-link :to="`/audiobook/${streamAudiobook.id}`" class="hover:underline cursor-pointer">
|
||||||
{{ title }} <span v-if="stream" class="text-xs text-gray-400">({{ stream.id }})</span>
|
{{ title }} <span v-if="stream && $isDev" class="text-xs text-gray-400">({{ stream.id }})</span>
|
||||||
</h1>
|
</nuxt-link>
|
||||||
<p class="text-gray-400 text-sm">by {{ author }}</p>
|
<p class="text-gray-400 text-sm hover:underline cursor-pointer" @click="filterByAuthor">by {{ author }}</p>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex-grow" />
|
<div class="flex-grow" />
|
||||||
<span v-if="stream" class="material-icons px-4 cursor-pointer" @click="cancelStream">close</span>
|
<span v-if="stream" class="material-icons px-4 cursor-pointer" @click="cancelStream">close</span>
|
||||||
@@ -66,6 +66,15 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
filterByAuthor() {
|
||||||
|
if (this.$route.name !== 'index') {
|
||||||
|
this.$router.push('/library')
|
||||||
|
}
|
||||||
|
var settingsUpdate = {
|
||||||
|
filterBy: `authors.${this.$encode(this.author)}`
|
||||||
|
}
|
||||||
|
this.$store.dispatch('user/updateUserSettings', settingsUpdate)
|
||||||
|
},
|
||||||
audioPlayerMounted() {
|
audioPlayerMounted() {
|
||||||
this.audioPlayerReady = true
|
this.audioPlayerReady = true
|
||||||
if (this.stream) {
|
if (this.stream) {
|
||||||
|
|||||||
@@ -0,0 +1,254 @@
|
|||||||
|
<template>
|
||||||
|
<div ref="wrapper" class="relative pointer-events-none" :style="{ width: standardWidth * 0.8 * 1.1 * scale + 'px', height: standardHeight * 1.1 * scale + 'px', marginBottom: 20 + 'px', marginTop: 15 + 'px' }">
|
||||||
|
<div ref="card" class="wrap absolute origin-center transform duration-200" :style="{ transform: `scale(${scale * scaleMultiplier}) translateY(${hover2 ? '-40%' : '-50%'})` }">
|
||||||
|
<div class="perspective">
|
||||||
|
<div class="book-wrap transform duration-100 pointer-events-auto" :class="hover2 ? 'z-80' : 'rotate'" @mouseover="hover = true" @mouseout="hover = false">
|
||||||
|
<div class="book book-1 box-shadow-book3d" ref="front"></div>
|
||||||
|
<div class="title book-1 pointer-events-none" ref="left"></div>
|
||||||
|
<div class="bottom book-1 pointer-events-none" ref="bottom"></div>
|
||||||
|
<div class="book-back book-1 pointer-events-none">
|
||||||
|
<div class="text pointer-events-none">
|
||||||
|
<h3 class="mb-4">Book Back</h3>
|
||||||
|
<p>
|
||||||
|
<span>Lorem ipsum dolor sit amet, consectetur adipisicing elit. Sunt earum doloremque aliquam culpa dolor nostrum consequatur quas dicta? Molestias repellendus minima pariatur libero vel, reiciendis optio magnam rerum, labore corporis.</span>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
src: String,
|
||||||
|
width: {
|
||||||
|
type: Number,
|
||||||
|
default: 200
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
hover: false,
|
||||||
|
hover2: false,
|
||||||
|
standardWidth: 200,
|
||||||
|
standardHeight: 320,
|
||||||
|
isAttached: true,
|
||||||
|
pageX: 0,
|
||||||
|
pageY: 0
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
src(newVal) {
|
||||||
|
this.setCover()
|
||||||
|
},
|
||||||
|
width(newVal) {
|
||||||
|
this.init()
|
||||||
|
},
|
||||||
|
hover(newVal) {
|
||||||
|
if (newVal) {
|
||||||
|
this.unattach()
|
||||||
|
} else {
|
||||||
|
this.attach()
|
||||||
|
}
|
||||||
|
setTimeout(() => {
|
||||||
|
this.hover2 = newVal
|
||||||
|
}, 100)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
scaleMultiplier() {
|
||||||
|
return this.hover2 ? 1.25 : 1
|
||||||
|
},
|
||||||
|
scale() {
|
||||||
|
var scale = this.width / this.standardWidth
|
||||||
|
return scale
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
unattach() {
|
||||||
|
if (this.$refs.card && this.isAttached) {
|
||||||
|
var bookshelf = document.getElementById('bookshelf')
|
||||||
|
if (bookshelf) {
|
||||||
|
var pos = this.$refs.wrapper.getBoundingClientRect()
|
||||||
|
|
||||||
|
this.pageX = pos.x
|
||||||
|
this.pageY = pos.y
|
||||||
|
document.body.appendChild(this.$refs.card)
|
||||||
|
this.$refs.card.style.left = this.pageX + 'px'
|
||||||
|
this.$refs.card.style.top = this.pageY + 'px'
|
||||||
|
this.$refs.card.style.zIndex = 50
|
||||||
|
this.isAttached = false
|
||||||
|
} else if (bookshelf) {
|
||||||
|
console.log(this.pageX, this.pageY)
|
||||||
|
this.isAttached = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
attach() {
|
||||||
|
if (this.$refs.card && !this.isAttached) {
|
||||||
|
if (this.$refs.wrapper) {
|
||||||
|
this.isAttached = true
|
||||||
|
|
||||||
|
this.$refs.wrapper.appendChild(this.$refs.card)
|
||||||
|
this.$refs.card.style.left = '0px'
|
||||||
|
this.$refs.card.style.top = '0px'
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log('Is attached already', this.isAttached)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
init() {
|
||||||
|
var standardWidth = this.standardWidth
|
||||||
|
document.documentElement.style.setProperty('--book-w', standardWidth + 'px')
|
||||||
|
document.documentElement.style.setProperty('--book-wx', standardWidth + 1 + 'px')
|
||||||
|
document.documentElement.style.setProperty('--book-h', standardWidth * 1.6 + 'px')
|
||||||
|
document.documentElement.style.setProperty('--book-d', 40 + 'px')
|
||||||
|
},
|
||||||
|
setElBg(el) {
|
||||||
|
el.style.backgroundImage = `url("${this.src}")`
|
||||||
|
el.style.backgroundSize = 'cover'
|
||||||
|
el.style.backgroundPosition = 'center center'
|
||||||
|
el.style.backgroundRepeat = 'no-repeat'
|
||||||
|
},
|
||||||
|
setCover() {
|
||||||
|
if (this.$refs.front) {
|
||||||
|
this.setElBg(this.$refs.front)
|
||||||
|
}
|
||||||
|
if (this.$refs.bottom) {
|
||||||
|
this.setElBg(this.$refs.bottom)
|
||||||
|
this.$refs.bottom.style.backgroundSize = '2000%'
|
||||||
|
this.$refs.bottom.style.filter = 'blur(1px)'
|
||||||
|
}
|
||||||
|
if (this.$refs.left) {
|
||||||
|
this.setElBg(this.$refs.left)
|
||||||
|
this.$refs.left.style.backgroundSize = '2000%'
|
||||||
|
this.$refs.left.style.filter = 'blur(1px)'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.setCover()
|
||||||
|
this.init()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
/* :root {
|
||||||
|
--book-w: 200px;
|
||||||
|
--book-h: 320px;
|
||||||
|
--book-d: 30px;
|
||||||
|
--book-wx: 201px;
|
||||||
|
} */
|
||||||
|
/*
|
||||||
|
.wrap {
|
||||||
|
width: calc(1.1 * var(--book-w));
|
||||||
|
height: calc(1.1 * var(--book-h));
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
.perspective {
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
|
||||||
|
perspective: 600px;
|
||||||
|
transform-style: preserve-3d;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.book-wrap {
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
transform-style: preserve-3d;
|
||||||
|
transition: 'all ease-out 0.6s';
|
||||||
|
}
|
||||||
|
|
||||||
|
.book {
|
||||||
|
width: var(--book-w);
|
||||||
|
height: var(--book-h);
|
||||||
|
background: url(https://covers.openlibrary.org/b/id/8303020-L.jpg) no-repeat center center;
|
||||||
|
background-size: cover;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
margin: auto;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.title {
|
||||||
|
content: '';
|
||||||
|
height: var(--book-h);
|
||||||
|
width: var(--book-d);
|
||||||
|
position: absolute;
|
||||||
|
right: 0;
|
||||||
|
left: calc(var(--book-wx) * -1);
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
margin: auto;
|
||||||
|
background: #444;
|
||||||
|
transform: rotateY(-80deg) translateX(-14px);
|
||||||
|
|
||||||
|
background: url(https://covers.openlibrary.org/b/id/8303020-L.jpg) no-repeat center center;
|
||||||
|
background-size: 5000%;
|
||||||
|
filter: blur(1px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.bottom {
|
||||||
|
content: '';
|
||||||
|
height: var(--book-d);
|
||||||
|
width: var(--book-w);
|
||||||
|
position: absolute;
|
||||||
|
right: 0;
|
||||||
|
bottom: var(--book-h);
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
margin: auto;
|
||||||
|
background: #444;
|
||||||
|
transform: rotateY(0deg) rotateX(90deg) translateY(-15px) translateX(-2.5px) skewX(10deg);
|
||||||
|
|
||||||
|
background: url(https://covers.openlibrary.org/b/id/8303020-L.jpg) no-repeat center center;
|
||||||
|
background-size: 5000%;
|
||||||
|
filter: blur(1px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.book-back {
|
||||||
|
width: var(--book-w);
|
||||||
|
height: var(--book-h);
|
||||||
|
background-color: #444;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
margin: auto;
|
||||||
|
cursor: pointer;
|
||||||
|
transform: rotate(180deg) translateZ(-30px) translateX(5px);
|
||||||
|
}
|
||||||
|
.book-back .text {
|
||||||
|
transform: rotateX(180deg);
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0px;
|
||||||
|
padding: 20px;
|
||||||
|
text-align: left;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
.book-back .text h3 {
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
.book-back .text span {
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.book-wrap.rotate {
|
||||||
|
transform: rotateY(30deg) rotateX(0deg);
|
||||||
|
}
|
||||||
|
.book-wrap.flip {
|
||||||
|
transform: rotateY(180deg);
|
||||||
|
} */
|
||||||
|
</style>
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="relative">
|
<div class="relative">
|
||||||
<!-- New Book Flag -->
|
<!-- New Book Flag -->
|
||||||
<div v-show="isNew" class="absolute top-4 left-0 w-4 h-10 pr-2 bg-darkgreen box-shadow-xl">
|
<div v-show="isNew" class="absolute top-4 left-0 w-4 h-10 pr-2 bg-darkgreen box-shadow-xl z-20">
|
||||||
<div class="absolute top-0 left-0 w-full h-full transform -rotate-90 flex items-center justify-center">
|
<div class="absolute top-0 left-0 w-full h-full transform -rotate-90 flex items-center justify-center">
|
||||||
<p class="text-center text-sm">New</p>
|
<p class="text-center text-sm">New</p>
|
||||||
</div>
|
</div>
|
||||||
@@ -29,7 +29,11 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-show="!isSelectionMode" class="absolute bottom-0 left-0 h-1 shadow-sm" :class="userIsRead ? 'bg-success' : 'bg-yellow-400'" :style="{ width: width * userProgressPercent + 'px' }"></div>
|
<div v-if="volumeNumber && showVolumeNumber && !isHovering" class="absolute rounded-lg bg-black bg-opacity-90 box-shadow-md" :style="{ top: 0.375 * sizeMultiplier + 'rem', right: 0.375 * sizeMultiplier + 'rem', padding: `${0.1 * sizeMultiplier}rem ${0.25 * sizeMultiplier}rem` }">
|
||||||
|
<p :style="{ fontSize: sizeMultiplier * 0.8 + 'rem' }">#{{ volumeNumber }}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-show="!isSelectionMode" class="absolute bottom-0 left-0 h-1 shadow-sm max-w-full" :class="userIsRead ? 'bg-success' : 'bg-yellow-400'" :style="{ width: width * userProgressPercent + 'px' }"></div>
|
||||||
|
|
||||||
<ui-tooltip v-if="showError" :text="errorText" class="absolute bottom-4 left-0">
|
<ui-tooltip v-if="showError" :text="errorText" class="absolute bottom-4 left-0">
|
||||||
<div :style="{ height: 1.5 * sizeMultiplier + 'rem', width: 2.5 * sizeMultiplier + 'rem' }" class="bg-error rounded-r-full shadow-md flex items-center justify-end border-r border-b border-red-300">
|
<div :style="{ height: 1.5 * sizeMultiplier + 'rem', width: 2.5 * sizeMultiplier + 'rem' }" class="bg-error rounded-r-full shadow-md flex items-center justify-end border-r border-b border-red-300">
|
||||||
@@ -56,7 +60,8 @@ export default {
|
|||||||
width: {
|
width: {
|
||||||
type: Number,
|
type: Number,
|
||||||
default: 120
|
default: 120
|
||||||
}
|
},
|
||||||
|
showVolumeNumber: Boolean
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
@@ -65,7 +70,7 @@ export default {
|
|||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
isNew() {
|
isNew() {
|
||||||
return this.tags.includes('new')
|
return this.tags.includes('New')
|
||||||
},
|
},
|
||||||
tags() {
|
tags() {
|
||||||
return this.audiobook.tags || []
|
return this.audiobook.tags || []
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
<div v-if="showCoverBg" class="bg-primary absolute top-0 left-0 w-full h-full">
|
<div v-if="showCoverBg" class="bg-primary absolute top-0 left-0 w-full h-full">
|
||||||
<div class="w-full h-full z-0" ref="coverBg" />
|
<div class="w-full h-full z-0" ref="coverBg" />
|
||||||
</div>
|
</div>
|
||||||
<img ref="cover" :src="cover" @error="imageError" @load="imageLoaded" class="w-full h-full absolute top-0 left-0" :class="showCoverBg ? 'object-contain' : 'object-cover'" />
|
<img ref="cover" :src="fullCoverUrl" @error="imageError" @load="imageLoaded" class="w-full h-full absolute top-0 left-0" :class="showCoverBg ? 'object-contain' : 'object-cover'" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="imageFailed" class="absolute top-0 left-0 right-0 bottom-0 w-full h-full bg-red-100" :style="{ padding: placeholderCoverPadding + 'rem' }">
|
<div v-if="imageFailed" class="absolute top-0 left-0 right-0 bottom-0 w-full h-full bg-red-100" :style="{ padding: placeholderCoverPadding + 'rem' }">
|
||||||
@@ -53,6 +53,9 @@ export default {
|
|||||||
book() {
|
book() {
|
||||||
return this.audiobook.book || {}
|
return this.audiobook.book || {}
|
||||||
},
|
},
|
||||||
|
bookLastUpdate() {
|
||||||
|
return this.book.lastUpdate || Date.now()
|
||||||
|
},
|
||||||
title() {
|
title() {
|
||||||
return this.book.title || 'No Title'
|
return this.book.title || 'No Title'
|
||||||
},
|
},
|
||||||
@@ -76,15 +79,7 @@ export default {
|
|||||||
return '/book_placeholder.jpg'
|
return '/book_placeholder.jpg'
|
||||||
},
|
},
|
||||||
fullCoverUrl() {
|
fullCoverUrl() {
|
||||||
if (!this.cover || this.cover === this.placeholderUrl) return ''
|
return this.$store.getters['audiobooks/getBookCoverSrc'](this.book, this.placeholderUrl)
|
||||||
if (this.cover.startsWith('http:') || this.cover.startsWith('https:')) return this.cover
|
|
||||||
try {
|
|
||||||
var url = new URL(this.cover, document.baseURI)
|
|
||||||
return url.href
|
|
||||||
} catch (err) {
|
|
||||||
console.error(err)
|
|
||||||
return ''
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
cover() {
|
cover() {
|
||||||
return this.book.cover || this.placeholderUrl
|
return this.book.cover || this.placeholderUrl
|
||||||
@@ -106,6 +101,9 @@ export default {
|
|||||||
},
|
},
|
||||||
authorBottom() {
|
authorBottom() {
|
||||||
return 0.75 * this.sizeMultiplier
|
return 0.75 * this.sizeMultiplier
|
||||||
|
},
|
||||||
|
userToken() {
|
||||||
|
return this.$store.getters['user/getToken']
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
|||||||
@@ -0,0 +1,98 @@
|
|||||||
|
<template>
|
||||||
|
<div class="relative">
|
||||||
|
<div class="rounded-sm h-full overflow-hidden relative" :style="{ padding: `16px ${paddingX}px` }" @mouseover="isHovering = true" @mouseleave="isHovering = false" @click="clickCard">
|
||||||
|
<nuxt-link :to="`/library/series?${groupType}=${groupEncode}`" class="cursor-pointer">
|
||||||
|
<div class="w-full relative" :class="isHovering ? 'bg-black-400' : 'bg-primary'" :style="{ height: height + 'px', width: height + 'px' }">
|
||||||
|
<cards-group-cover ref="groupcover" :name="groupName" :book-items="bookItems" :width="height" :height="height" />
|
||||||
|
|
||||||
|
<div v-if="hasValidCovers" class="bg-black bg-opacity-60 absolute top-0 left-0 w-full h-full flex items-center justify-center text-center transition-opacity" :class="isHovering ? '' : 'opacity-0'" :style="{ padding: `${sizeMultiplier}rem` }">
|
||||||
|
<p class="font-book" :style="{ fontSize: sizeMultiplier + 'rem' }">{{ groupName }}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="absolute top-2 right-2 w-7 h-7 rounded-lg bg-black bg-opacity-90 text-gray-300 box-shadow-book flex items-center justify-center border border-white border-opacity-25 pointer-events-none">
|
||||||
|
<p class="font-book text-xl">{{ bookItems.length }}</p>
|
||||||
|
</div>
|
||||||
|
<div class="absolute bottom-0 left-0 w-full h-1 flex flex-nowrap">
|
||||||
|
<div v-for="userProgress in userProgressItems" :key="userProgress.audiobookId" class="h-full w-full" :class="userProgress.isRead ? 'bg-success' : userProgress.progress > 0 ? 'bg-yellow-400' : ''" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</nuxt-link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
group: {
|
||||||
|
type: Object,
|
||||||
|
default: () => null
|
||||||
|
},
|
||||||
|
width: {
|
||||||
|
type: Number,
|
||||||
|
default: 120
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
isHovering: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
width(newVal) {
|
||||||
|
this.$nextTick(() => {
|
||||||
|
if (this.$refs.groupcover) {
|
||||||
|
this.$refs.groupcover.init()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
_group() {
|
||||||
|
return this.group || {}
|
||||||
|
},
|
||||||
|
height() {
|
||||||
|
return this.width * 1.6
|
||||||
|
},
|
||||||
|
sizeMultiplier() {
|
||||||
|
return this.width / 120
|
||||||
|
},
|
||||||
|
paddingX() {
|
||||||
|
return 16 * this.sizeMultiplier
|
||||||
|
},
|
||||||
|
bookItems() {
|
||||||
|
return this._group.books || []
|
||||||
|
},
|
||||||
|
userAudiobooks() {
|
||||||
|
return Object.values(this.$store.state.user.user ? this.$store.state.user.user.audiobooks || {} : {})
|
||||||
|
},
|
||||||
|
userProgressItems() {
|
||||||
|
return this.bookItems.map((item) => {
|
||||||
|
var userAudiobook = this.userAudiobooks.find((ab) => ab.audiobookId === item.id)
|
||||||
|
return userAudiobook || {}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
groupName() {
|
||||||
|
return this._group.name || 'No Name'
|
||||||
|
},
|
||||||
|
groupType() {
|
||||||
|
return this._group.type
|
||||||
|
},
|
||||||
|
groupEncode() {
|
||||||
|
return this.$encode(this.groupName)
|
||||||
|
},
|
||||||
|
filter() {
|
||||||
|
return `${this.groupType}.${this.$encode(this.groupName)}`
|
||||||
|
},
|
||||||
|
hasValidCovers() {
|
||||||
|
var validCovers = this.bookItems.map((bookItem) => bookItem.book.cover)
|
||||||
|
return !!validCovers.length
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
clickCard() {
|
||||||
|
this.$emit('click', this.group)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,139 @@
|
|||||||
|
<template>
|
||||||
|
<div ref="wrapper" :style="{ height: height + 'px', width: width + 'px' }" class="relative">
|
||||||
|
<div v-if="noValidCovers" class="absolute top-0 left-0 w-full h-full flex items-center justify-center box-shadow-book" :style="{ padding: `${sizeMultiplier}rem` }">
|
||||||
|
<p :style="{ fontSize: sizeMultiplier + 'rem' }">{{ name }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
name: String,
|
||||||
|
bookItems: {
|
||||||
|
type: Array,
|
||||||
|
default: () => []
|
||||||
|
},
|
||||||
|
width: Number,
|
||||||
|
height: Number
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
noValidCovers: false,
|
||||||
|
coverDiv: null
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
bookItems: {
|
||||||
|
immediate: true,
|
||||||
|
handler(newVal) {
|
||||||
|
if (newVal) {
|
||||||
|
// ensure wrapper is initialized
|
||||||
|
this.$nextTick(this.init)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
sizeMultiplier() {
|
||||||
|
return this.width / 192
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
getCoverUrl(book) {
|
||||||
|
return this.$store.getters['audiobooks/getBookCoverSrc'](book, '')
|
||||||
|
},
|
||||||
|
async buildCoverImg(src, bgCoverWidth, offsetLeft, forceCoverBg = false) {
|
||||||
|
var showCoverBg =
|
||||||
|
forceCoverBg ||
|
||||||
|
(await new Promise((resolve) => {
|
||||||
|
var image = new Image()
|
||||||
|
|
||||||
|
image.onload = () => {
|
||||||
|
var { naturalWidth, naturalHeight } = image
|
||||||
|
var aspectRatio = naturalHeight / naturalWidth
|
||||||
|
var arDiff = Math.abs(aspectRatio - 1.6)
|
||||||
|
|
||||||
|
// If image aspect ratio is <= 1.45 or >= 1.75 then use cover bg, otherwise stretch to fit
|
||||||
|
if (arDiff > 0.15) {
|
||||||
|
resolve(true)
|
||||||
|
} else {
|
||||||
|
resolve(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
image.onerror = (err) => {
|
||||||
|
console.error(err)
|
||||||
|
resolve(false)
|
||||||
|
}
|
||||||
|
image.src = src
|
||||||
|
}))
|
||||||
|
|
||||||
|
var imgdiv = document.createElement('div')
|
||||||
|
imgdiv.style.height = this.height + 'px'
|
||||||
|
imgdiv.style.width = bgCoverWidth + 'px'
|
||||||
|
imgdiv.style.left = offsetLeft + 'px'
|
||||||
|
imgdiv.className = 'absolute top-0 box-shadow-book'
|
||||||
|
imgdiv.style.boxShadow = '-4px 0px 4px #11111166'
|
||||||
|
// imgdiv.style.transform = 'skew(0deg, 15deg)'
|
||||||
|
|
||||||
|
if (showCoverBg) {
|
||||||
|
var coverbgwrapper = document.createElement('div')
|
||||||
|
coverbgwrapper.className = 'absolute top-0 left-0 w-full h-full bg-primary'
|
||||||
|
|
||||||
|
var coverbg = document.createElement('div')
|
||||||
|
coverbg.className = 'w-full h-full'
|
||||||
|
coverbg.style.backgroundImage = `url("${src}")`
|
||||||
|
coverbg.style.backgroundSize = 'cover'
|
||||||
|
coverbg.style.backgroundPosition = 'center'
|
||||||
|
coverbg.style.opacity = 0.25
|
||||||
|
coverbg.style.filter = 'blur(1px)'
|
||||||
|
|
||||||
|
coverbgwrapper.appendChild(coverbg)
|
||||||
|
imgdiv.appendChild(coverbgwrapper)
|
||||||
|
}
|
||||||
|
|
||||||
|
var img = document.createElement('img')
|
||||||
|
img.src = src
|
||||||
|
img.className = 'absolute top-0 left-0 w-full h-full'
|
||||||
|
img.style.objectFit = showCoverBg ? 'contain' : 'cover'
|
||||||
|
|
||||||
|
imgdiv.appendChild(img)
|
||||||
|
return imgdiv
|
||||||
|
},
|
||||||
|
async init() {
|
||||||
|
if (this.coverDiv) {
|
||||||
|
this.coverDiv.remove()
|
||||||
|
this.coverDiv = null
|
||||||
|
}
|
||||||
|
var validCovers = this.bookItems.map((bookItem) => this.getCoverUrl(bookItem.book)).filter((b) => b !== '')
|
||||||
|
if (!validCovers.length) {
|
||||||
|
this.noValidCovers = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
this.noValidCovers = false
|
||||||
|
|
||||||
|
var coverWidth = this.width
|
||||||
|
var widthPer = this.width
|
||||||
|
if (validCovers.length > 1) {
|
||||||
|
coverWidth = this.height / 1.6
|
||||||
|
widthPer = (this.width - coverWidth) / (validCovers.length - 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
var outerdiv = document.createElement('div')
|
||||||
|
outerdiv.className = 'w-full h-full relative'
|
||||||
|
|
||||||
|
for (let i = 0; i < validCovers.length; i++) {
|
||||||
|
var offsetLeft = widthPer * i
|
||||||
|
var img = await this.buildCoverImg(validCovers[i], coverWidth, offsetLeft, validCovers.length === 1)
|
||||||
|
outerdiv.appendChild(img)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.$refs.wrapper) {
|
||||||
|
this.coverDiv = outerdiv
|
||||||
|
this.$refs.wrapper.appendChild(outerdiv)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,82 @@
|
|||||||
|
<template>
|
||||||
|
<div class="relative rounded-sm overflow-hidden" :style="{ height: width * 1.6 + 'px', width: width + 'px', maxWidth: width + 'px', minWidth: width + 'px' }">
|
||||||
|
<div class="w-full h-full relative">
|
||||||
|
<div v-if="showCoverBg" class="bg-primary absolute top-0 left-0 w-full h-full">
|
||||||
|
<div class="w-full h-full z-0" ref="coverBg" />
|
||||||
|
</div>
|
||||||
|
<img ref="cover" :src="cover" @error="imageError" @load="imageLoaded" class="w-full h-full absolute top-0 left-0" :class="showCoverBg ? 'object-contain' : 'object-cover'" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="imageFailed" class="absolute top-0 left-0 right-0 bottom-0 w-full h-full bg-red-100" :style="{ padding: placeholderCoverPadding + 'rem' }">
|
||||||
|
<div class="w-full h-full border-2 border-error flex flex-col items-center justify-center">
|
||||||
|
<img src="/Logo.png" class="mb-2" :style="{ height: 64 * sizeMultiplier + 'px' }" />
|
||||||
|
<p class="text-center font-book text-error" :style="{ fontSize: sizeMultiplier + 'rem' }">Invalid Cover</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
src: String,
|
||||||
|
width: {
|
||||||
|
type: Number,
|
||||||
|
default: 120
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
imageFailed: false,
|
||||||
|
showCoverBg: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
cover() {
|
||||||
|
this.imageFailed = false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
cover() {
|
||||||
|
return this.src
|
||||||
|
},
|
||||||
|
sizeMultiplier() {
|
||||||
|
return this.width / 120
|
||||||
|
},
|
||||||
|
placeholderCoverPadding() {
|
||||||
|
return 0.8 * this.sizeMultiplier
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
setCoverBg() {
|
||||||
|
if (this.$refs.coverBg) {
|
||||||
|
this.$refs.coverBg.style.backgroundImage = `url("${this.src}")`
|
||||||
|
this.$refs.coverBg.style.backgroundSize = 'cover'
|
||||||
|
this.$refs.coverBg.style.backgroundPosition = 'center'
|
||||||
|
this.$refs.coverBg.style.opacity = 0.25
|
||||||
|
this.$refs.coverBg.style.filter = 'blur(1px)'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
imageLoaded() {
|
||||||
|
if (this.$refs.cover) {
|
||||||
|
var { naturalWidth, naturalHeight } = this.$refs.cover
|
||||||
|
var aspectRatio = naturalHeight / naturalWidth
|
||||||
|
var arDiff = Math.abs(aspectRatio - 1.6)
|
||||||
|
|
||||||
|
// If image aspect ratio is <= 1.45 or >= 1.75 then use cover bg, otherwise stretch to fit
|
||||||
|
if (arDiff > 0.15) {
|
||||||
|
this.showCoverBg = true
|
||||||
|
this.$nextTick(this.setCoverBg)
|
||||||
|
} else {
|
||||||
|
this.showCoverBg = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
imageError(err) {
|
||||||
|
console.error('ImgError', err)
|
||||||
|
this.imageFailed = true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -86,6 +86,11 @@ export default {
|
|||||||
text: 'Authors',
|
text: 'Authors',
|
||||||
value: 'authors',
|
value: 'authors',
|
||||||
sublist: true
|
sublist: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: 'Progress',
|
||||||
|
value: 'progress',
|
||||||
|
sublist: true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
@@ -132,6 +137,9 @@ export default {
|
|||||||
authors() {
|
authors() {
|
||||||
return this.$store.getters['audiobooks/getUniqueAuthors']
|
return this.$store.getters['audiobooks/getUniqueAuthors']
|
||||||
},
|
},
|
||||||
|
progress() {
|
||||||
|
return ['Read', 'Unread', 'In Progress']
|
||||||
|
},
|
||||||
sublistItems() {
|
sublistItems() {
|
||||||
return (this[this.sublist] || []).map((item) => {
|
return (this[this.sublist] || []).map((item) => {
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="w-64 ml-8 relative">
|
<div class="w-64 ml-8 relative">
|
||||||
<ui-text-input v-model="search" placeholder="Search.." @input="inputUpdate" @focus="focussed" @blur="blurred" class="w-full h-8 text-sm" />
|
<form @submit.prevent="submitSearch">
|
||||||
|
<ui-text-input ref="input" v-model="search" placeholder="Search.." @input="inputUpdate" @focus="focussed" @blur="blurred" class="w-full h-8 text-sm" />
|
||||||
|
</form>
|
||||||
<div class="absolute top-0 right-0 bottom-0 h-full flex items-center px-2 text-gray-400 cursor-pointer" @click="clickClear">
|
<div class="absolute top-0 right-0 bottom-0 h-full flex items-center px-2 text-gray-400 cursor-pointer" @click="clickClear">
|
||||||
<span v-if="!search" class="material-icons" style="font-size: 1.2rem">search</span>
|
<span v-if="!search" class="material-icons" style="font-size: 1.2rem">search</span>
|
||||||
<span v-else class="material-icons" style="font-size: 1.2rem">close</span>
|
<span v-else class="material-icons" style="font-size: 1.2rem">close</span>
|
||||||
@@ -18,7 +20,7 @@
|
|||||||
</li>
|
</li>
|
||||||
<template v-else>
|
<template v-else>
|
||||||
<template v-for="item in items">
|
<template v-for="item in items">
|
||||||
<li :key="item.id" class="text-gray-50 select-none relative cursor-pointer hover:bg-black-400" role="option" @click="clickedOption(item)">
|
<li :key="item.id" class="text-gray-50 select-none relative cursor-pointer hover:bg-black-400 py-1" role="option" @click="clickedOption(item)">
|
||||||
<template v-if="item.type === 'audiobook'">
|
<template v-if="item.type === 'audiobook'">
|
||||||
<cards-audiobook-search-card :audiobook="item.data" />
|
<cards-audiobook-search-card :audiobook="item.data" />
|
||||||
</template>
|
</template>
|
||||||
@@ -51,6 +53,19 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
submitSearch() {
|
||||||
|
if (!this.search) return
|
||||||
|
this.$router.push(`/library/search?query=${this.search}`)
|
||||||
|
|
||||||
|
this.search = null
|
||||||
|
this.items = []
|
||||||
|
this.showMenu = false
|
||||||
|
this.$nextTick(() => {
|
||||||
|
if (this.$refs.input) {
|
||||||
|
this.$refs.input.blur()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
focussed() {
|
focussed() {
|
||||||
this.isFocused = true
|
this.isFocused = true
|
||||||
this.showMenu = true
|
this.showMenu = true
|
||||||
@@ -73,6 +88,10 @@ export default {
|
|||||||
return []
|
return []
|
||||||
})
|
})
|
||||||
this.isFetching = false
|
this.isFetching = false
|
||||||
|
if (!this.showMenu) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
this.items = results.map((res) => {
|
this.items = results.map((res) => {
|
||||||
return {
|
return {
|
||||||
id: res.id,
|
id: res.id,
|
||||||
|
|||||||
@@ -74,10 +74,10 @@ export default {
|
|||||||
this.showMenu = false
|
this.showMenu = false
|
||||||
},
|
},
|
||||||
leftArrowClick() {
|
leftArrowClick() {
|
||||||
this.rateIndex = Math.max(0, this.rateIndex - 4)
|
this.rateIndex = Math.max(0, this.rateIndex - 1)
|
||||||
},
|
},
|
||||||
rightArrowClick() {
|
rightArrowClick() {
|
||||||
this.rateIndex = Math.min(this.rates.length - this.numVisible, this.rateIndex + 4)
|
this.rateIndex = Math.min(this.rates.length - this.numVisible, this.rateIndex + 1)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {}
|
mounted() {}
|
||||||
|
|||||||
@@ -27,9 +27,9 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="!isEditingRoot && newUser.permissions" class="w-full border-t border-b border-black-200 py-2 mt-4">
|
<div v-if="!isEditingRoot && newUser.permissions" class="w-full border-t border-b border-black-200 py-2 px-3 mt-4">
|
||||||
<p class="text-lg mb-2">Permissions</p>
|
<p class="text-lg mb-2 font-semibold">Permissions</p>
|
||||||
<div class="flex items-center my-2 max-w-lg">
|
<div class="flex items-center my-2 max-w-md">
|
||||||
<div class="w-1/2">
|
<div class="w-1/2">
|
||||||
<p>Can Download</p>
|
<p>Can Download</p>
|
||||||
</div>
|
</div>
|
||||||
@@ -38,7 +38,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex items-center my-2 max-w-lg">
|
<div class="flex items-center my-2 max-w-md">
|
||||||
<div class="w-1/2">
|
<div class="w-1/2">
|
||||||
<p>Can Update</p>
|
<p>Can Update</p>
|
||||||
</div>
|
</div>
|
||||||
@@ -47,7 +47,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex items-center my-2 max-w-lg">
|
<div class="flex items-center my-2 max-w-md">
|
||||||
<div class="w-1/2">
|
<div class="w-1/2">
|
||||||
<p>Can Delete</p>
|
<p>Can Delete</p>
|
||||||
</div>
|
</div>
|
||||||
@@ -55,6 +55,15 @@
|
|||||||
<ui-toggle-switch v-model="newUser.permissions.delete" />
|
<ui-toggle-switch v-model="newUser.permissions.delete" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="flex items-center my-2 max-w-md">
|
||||||
|
<div class="w-1/2">
|
||||||
|
<p>Can Upload</p>
|
||||||
|
</div>
|
||||||
|
<div class="w-1/2">
|
||||||
|
<ui-toggle-switch v-model="newUser.permissions.upload" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex pt-4">
|
<div class="flex pt-4">
|
||||||
@@ -179,7 +188,8 @@ export default {
|
|||||||
this.newUser.permissions = {
|
this.newUser.permissions = {
|
||||||
download: type !== 'guest',
|
download: type !== 'guest',
|
||||||
update: type === 'admin',
|
update: type === 'admin',
|
||||||
delete: type === 'admin'
|
delete: type === 'admin',
|
||||||
|
upload: type === 'admin'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
init() {
|
init() {
|
||||||
@@ -201,7 +211,8 @@ export default {
|
|||||||
permissions: {
|
permissions: {
|
||||||
download: true,
|
download: true,
|
||||||
update: false,
|
update: false,
|
||||||
delete: false
|
delete: false,
|
||||||
|
upload: false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,13 @@
|
|||||||
<template>
|
<template>
|
||||||
<modals-modal v-model="show" :width="500" :height="'unset'">
|
<modals-modal v-model="show" :width="500" :height="'unset'">
|
||||||
<div class="w-full rounded-lg bg-primary box-shadow-md overflow-y-auto overflow-x-hidden" style="max-height: 500px">
|
<div ref="container" class="w-full rounded-lg bg-primary box-shadow-md overflow-y-auto overflow-x-hidden" style="max-height: 80vh">
|
||||||
<template v-for="chap in chapters">
|
<template v-for="chap in chapters">
|
||||||
<div :key="chap.id" class="flex items-center px-6 py-3 justify-start cursor-pointer hover:bg-bg bg-opacity-20 rounded-lg relative" @click="clickChapter(chap)">
|
<div :key="chap.id" :id="`chapter-row-${chap.id}`" class="flex items-center px-6 py-3 justify-start cursor-pointer hover:bg-bg relative" :class="chap.id === currentChapterId ? 'bg-bg bg-opacity-80' : 'bg-opacity-20'" @click="clickChapter(chap)">
|
||||||
{{ chap.title }}
|
{{ chap.title }}
|
||||||
<span class="flex-grow" />
|
<span class="flex-grow" />
|
||||||
<span class="font-mono text-sm text-gray-300">{{ $secondsToTimestamp(chap.start) }}</span>
|
<span class="font-mono text-sm text-gray-300">{{ $secondsToTimestamp(chap.start) }}</span>
|
||||||
|
|
||||||
|
<div v-show="chap.id === currentChapterId" class="w-0.5 h-full absolute top-0 left-0 bg-yellow-400" />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
@@ -19,11 +21,20 @@ export default {
|
|||||||
chapters: {
|
chapters: {
|
||||||
type: Array,
|
type: Array,
|
||||||
default: () => []
|
default: () => []
|
||||||
|
},
|
||||||
|
currentChapter: {
|
||||||
|
type: Object,
|
||||||
|
default: () => null
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {}
|
return {}
|
||||||
},
|
},
|
||||||
|
watch: {
|
||||||
|
value(newVal) {
|
||||||
|
this.$nextTick(this.scrollToChapter)
|
||||||
|
}
|
||||||
|
},
|
||||||
computed: {
|
computed: {
|
||||||
show: {
|
show: {
|
||||||
get() {
|
get() {
|
||||||
@@ -32,13 +43,28 @@ export default {
|
|||||||
set(val) {
|
set(val) {
|
||||||
this.$emit('input', val)
|
this.$emit('input', val)
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
currentChapterId() {
|
||||||
|
return this.currentChapter ? this.currentChapter.id : null
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
clickChapter(chap) {
|
clickChapter(chap) {
|
||||||
this.$emit('select', chap)
|
this.$emit('select', chap)
|
||||||
|
},
|
||||||
|
scrollToChapter() {
|
||||||
|
if (!this.currentChapterId) return
|
||||||
|
|
||||||
|
var container = this.$refs.container
|
||||||
|
if (container) {
|
||||||
|
var currChapterEl = document.getElementById(`chapter-row-${this.currentChapterId}`)
|
||||||
|
if (currChapterEl) {
|
||||||
|
var offsetTop = currChapterEl.offsetTop
|
||||||
|
var containerHeight = container.clientHeight
|
||||||
|
container.scrollTo({ top: offsetTop - containerHeight / 2 })
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
mounted() {}
|
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="w-full h-full overflow-hidden overflow-y-auto px-4 py-6">
|
<div class="w-full h-full overflow-hidden overflow-y-auto px-4 py-6 relative">
|
||||||
<div class="flex">
|
<div class="flex">
|
||||||
<div class="relative">
|
<div class="relative">
|
||||||
<cards-book-cover :audiobook="audiobook" />
|
<cards-book-cover :audiobook="audiobook" />
|
||||||
@@ -12,12 +12,15 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex-grow pl-6 pr-2">
|
<div class="flex-grow pl-6 pr-2">
|
||||||
<form @submit.prevent="submitForm">
|
<div class="flex items-center">
|
||||||
<div class="flex items-center">
|
<div v-if="userCanUpload" class="w-40 pr-2" style="min-width: 160px">
|
||||||
|
<ui-file-input ref="fileInput" @change="fileUploadSelected" />
|
||||||
|
</div>
|
||||||
|
<form @submit.prevent="submitForm" class="flex flex-grow">
|
||||||
<ui-text-input-with-label v-model="imageUrl" label="Cover Image URL" />
|
<ui-text-input-with-label v-model="imageUrl" label="Cover Image URL" />
|
||||||
<ui-btn color="success" type="submit" :padding-x="4" class="mt-5 ml-3 w-24">Update</ui-btn>
|
<ui-btn color="success" type="submit" :padding-x="4" class="mt-5 ml-3 w-24">Update</ui-btn>
|
||||||
</div>
|
</form>
|
||||||
</form>
|
</div>
|
||||||
|
|
||||||
<div v-if="localCovers.length" class="mb-4 mt-6 border-t border-b border-primary">
|
<div v-if="localCovers.length" class="mb-4 mt-6 border-t border-b border-primary">
|
||||||
<div class="flex items-center justify-center py-2">
|
<div class="flex items-center justify-center py-2">
|
||||||
@@ -63,6 +66,18 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div v-if="previewUpload" class="absolute top-0 left-0 w-full h-full z-10 bg-bg p-8">
|
||||||
|
<p class="text-lg">Preview Cover</p>
|
||||||
|
<span class="absolute top-4 right-4 material-icons text-2xl cursor-pointer" @click="resetCoverPreview">close</span>
|
||||||
|
<div class="flex justify-center py-4">
|
||||||
|
<cards-preview-cover :src="previewUpload" :width="240" />
|
||||||
|
</div>
|
||||||
|
<div class="absolute bottom-0 right-0 flex py-4 px-5">
|
||||||
|
<ui-btn :disabled="processingUpload" class="mx-2" @click="resetCoverPreview">Clear</ui-btn>
|
||||||
|
<ui-btn :loading="processingUpload" color="success" @click="submitCoverUpload">Upload</ui-btn>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -79,12 +94,15 @@ export default {
|
|||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
|
processingUpload: false,
|
||||||
searchTitle: null,
|
searchTitle: null,
|
||||||
searchAuthor: null,
|
searchAuthor: null,
|
||||||
imageUrl: null,
|
imageUrl: null,
|
||||||
coversFound: [],
|
coversFound: [],
|
||||||
hasSearched: false,
|
hasSearched: false,
|
||||||
showLocalCovers: false
|
showLocalCovers: false,
|
||||||
|
previewUpload: null,
|
||||||
|
selectedFile: null
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
@@ -112,6 +130,9 @@ export default {
|
|||||||
otherFiles() {
|
otherFiles() {
|
||||||
return this.audiobook ? this.audiobook.otherFiles || [] : []
|
return this.audiobook ? this.audiobook.otherFiles || [] : []
|
||||||
},
|
},
|
||||||
|
userCanUpload() {
|
||||||
|
return this.$store.getters['user/getUserCanUpload']
|
||||||
|
},
|
||||||
localCovers() {
|
localCovers() {
|
||||||
return this.otherFiles
|
return this.otherFiles
|
||||||
.filter((f) => f.filetype === 'image')
|
.filter((f) => f.filetype === 'image')
|
||||||
@@ -123,6 +144,39 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
submitCoverUpload() {
|
||||||
|
this.processingUpload = true
|
||||||
|
var form = new FormData()
|
||||||
|
form.set('cover', this.selectedFile)
|
||||||
|
|
||||||
|
this.$axios
|
||||||
|
.$post(`/api/audiobook/${this.audiobook.id}/cover`, form)
|
||||||
|
.then((data) => {
|
||||||
|
if (data.error) {
|
||||||
|
this.$toast.error(data.error)
|
||||||
|
} else {
|
||||||
|
this.$toast.success('Cover Uploaded')
|
||||||
|
this.resetCoverPreview()
|
||||||
|
}
|
||||||
|
this.processingUpload = false
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.error('Failed', error)
|
||||||
|
this.$toast.error('Oops, something went wrong...')
|
||||||
|
this.processingUpload = false
|
||||||
|
})
|
||||||
|
},
|
||||||
|
resetCoverPreview() {
|
||||||
|
if (this.$refs.fileInput) {
|
||||||
|
this.$refs.fileInput.reset()
|
||||||
|
}
|
||||||
|
this.previewUpload = null
|
||||||
|
this.selectedFile = null
|
||||||
|
},
|
||||||
|
fileUploadSelected(file) {
|
||||||
|
this.previewUpload = URL.createObjectURL(file)
|
||||||
|
this.selectedFile = file
|
||||||
|
},
|
||||||
init() {
|
init() {
|
||||||
this.showLocalCovers = false
|
this.showLocalCovers = false
|
||||||
if (this.coversFound.length && (this.searchTitle !== this.book.title || this.searchAuthor !== this.book.author)) {
|
if (this.coversFound.length && (this.searchTitle !== this.book.title || this.searchAuthor !== this.book.author)) {
|
||||||
|
|||||||
@@ -1,5 +1,14 @@
|
|||||||
<template>
|
<template>
|
||||||
<button class="btn outline-none rounded-md shadow-md relative border border-gray-600" :disabled="disabled || loading" :type="type" :class="classList" @click="click">
|
<nuxt-link v-if="to" :to="to" class="btn outline-none rounded-md shadow-md relative border border-gray-600" :disabled="disabled || loading" :class="classList">
|
||||||
|
<slot />
|
||||||
|
<div v-if="loading" class="text-white absolute top-0 left-0 w-full h-full flex items-center justify-center text-opacity-100">
|
||||||
|
<!-- <span class="material-icons animate-spin">refresh</span> -->
|
||||||
|
<svg class="animate-spin" style="width: 24px; height: 24px" viewBox="0 0 24 24">
|
||||||
|
<path fill="currentColor" d="M12,4V2A10,10 0 0,0 2,12H4A8,8 0 0,1 12,4Z" />
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
</nuxt-link>
|
||||||
|
<button v-else class="btn outline-none rounded-md shadow-md relative border border-gray-600" :disabled="disabled || loading" :type="type" :class="classList" @click="click">
|
||||||
<slot />
|
<slot />
|
||||||
<div v-if="loading" class="text-white absolute top-0 left-0 w-full h-full flex items-center justify-center text-opacity-100">
|
<div v-if="loading" class="text-white absolute top-0 left-0 w-full h-full flex items-center justify-center text-opacity-100">
|
||||||
<!-- <span class="material-icons animate-spin">refresh</span> -->
|
<!-- <span class="material-icons animate-spin">refresh</span> -->
|
||||||
@@ -13,6 +22,7 @@
|
|||||||
<script>
|
<script>
|
||||||
export default {
|
export default {
|
||||||
props: {
|
props: {
|
||||||
|
to: String,
|
||||||
color: {
|
color: {
|
||||||
type: String,
|
type: String,
|
||||||
default: 'primary'
|
default: 'primary'
|
||||||
|
|||||||
@@ -0,0 +1,39 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<input ref="fileInput" id="hidden-input" type="file" :accept="inputAccept" class="hidden" @change="inputChanged" />
|
||||||
|
<ui-btn @click="clickUpload" color="primary" type="text">Upload Cover</ui-btn>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
inputAccept: 'image/*'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {},
|
||||||
|
methods: {
|
||||||
|
reset() {
|
||||||
|
if (this.$refs.fileInput) {
|
||||||
|
this.$refs.fileInput.value = ''
|
||||||
|
}
|
||||||
|
},
|
||||||
|
clickUpload() {
|
||||||
|
if (this.$refs.fileInput) {
|
||||||
|
this.$refs.fileInput.click()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
inputChanged(e) {
|
||||||
|
if (!e.target || !e.target.files) return
|
||||||
|
var _files = Array.from(e.target.files)
|
||||||
|
if (_files && _files.length) {
|
||||||
|
var file = _files[0]
|
||||||
|
console.log('File', file)
|
||||||
|
this.$emit('change', file)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -113,10 +113,12 @@ export default {
|
|||||||
this.currentSearch = null
|
this.currentSearch = null
|
||||||
},
|
},
|
||||||
clickedOption(e, item) {
|
clickedOption(e, item) {
|
||||||
var newValue = this.input === item ? null : item
|
|
||||||
this.textInput = null
|
this.textInput = null
|
||||||
this.currentSearch = null
|
this.currentSearch = null
|
||||||
this.input = this.textInput ? this.textInput.trim() : null
|
this.input = item
|
||||||
|
|
||||||
|
// this.input = this.textInput ? this.textInput.trim() : null
|
||||||
|
console.log('Clicked option', item)
|
||||||
if (this.$refs.input) this.$refs.input.blur()
|
if (this.$refs.input) this.$refs.input.blur()
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -180,7 +180,7 @@ export default {
|
|||||||
submitForm() {
|
submitForm() {
|
||||||
if (!this.textInput) return
|
if (!this.textInput) return
|
||||||
|
|
||||||
var cleaned = this.textInput.toLowerCase().trim()
|
var cleaned = this.textInput.trim()
|
||||||
var matchesItem = this.items.find((i) => {
|
var matchesItem = this.items.find((i) => {
|
||||||
return i === cleaned
|
return i === cleaned
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<input v-model="inputValue" :type="type" :readonly="readonly" :disabled="disabled" :placeholder="placeholder" class="rounded bg-primary text-gray-200 focus:border-gray-500 focus:outline-none border border-gray-600" :class="classList" @keyup="keyup" @change="change" @focus="focused" @blur="blurred" />
|
<input ref="input" v-model="inputValue" :type="type" :readonly="readonly" :disabled="disabled" :placeholder="placeholder" class="rounded bg-primary text-gray-200 focus:border-gray-500 focus:outline-none border border-gray-600" :class="classList" @keyup="keyup" @change="change" @focus="focused" @blur="blurred" />
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
@@ -53,6 +53,9 @@ export default {
|
|||||||
},
|
},
|
||||||
keyup(e) {
|
keyup(e) {
|
||||||
this.$emit('keyup', e)
|
this.$emit('keyup', e)
|
||||||
|
},
|
||||||
|
blur() {
|
||||||
|
if (this.$refs.input) this.$refs.input.blur()
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {}
|
mounted() {}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<div ref="box" class="tooltip-box" @mouseover="mouseover" @mouseleave="mouseleave">
|
<div ref="box" @mouseover="mouseover" @mouseleave="mouseleave">
|
||||||
<slot />
|
<slot />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -51,8 +51,9 @@ export default {
|
|||||||
createTooltip() {
|
createTooltip() {
|
||||||
if (!this.$refs.box) return
|
if (!this.$refs.box) return
|
||||||
var tooltip = document.createElement('div')
|
var tooltip = document.createElement('div')
|
||||||
tooltip.className = 'absolute px-2 bg-black bg-opacity-90 py-1 text-white pointer-events-none text-xs rounded shadow-lg'
|
tooltip.className = 'absolute px-2 py-1 text-white pointer-events-none text-xs rounded shadow-lg max-w-xs'
|
||||||
tooltip.style.zIndex = 100
|
tooltip.style.zIndex = 100
|
||||||
|
tooltip.style.backgroundColor = 'rgba(0,0,0,0.85)'
|
||||||
tooltip.innerHTML = this.text
|
tooltip.innerHTML = this.text
|
||||||
|
|
||||||
this.setTooltipPosition(tooltip)
|
this.setTooltipPosition(tooltip)
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="text-white max-h-screen h-screen overflow-hidden bg-bg">
|
<div class="text-white max-h-screen h-screen overflow-hidden bg-bg">
|
||||||
<app-appbar />
|
<app-appbar />
|
||||||
|
|
||||||
<Nuxt />
|
<Nuxt />
|
||||||
|
|
||||||
<app-stream-container ref="streamContainer" />
|
<app-stream-container ref="streamContainer" />
|
||||||
<modals-edit-modal />
|
<modals-edit-modal />
|
||||||
<widgets-scan-alert />
|
<widgets-scan-alert />
|
||||||
@@ -24,6 +26,9 @@ export default {
|
|||||||
if (this.$store.state.selectedAudiobooks) {
|
if (this.$store.state.selectedAudiobooks) {
|
||||||
this.$store.commit('setSelectedAudiobooks', [])
|
this.$store.commit('setSelectedAudiobooks', [])
|
||||||
}
|
}
|
||||||
|
if (this.$store.state.audiobooks.keywordFilter) {
|
||||||
|
this.$store.commit('audiobooks/setKeywordFilter', '')
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@@ -84,7 +89,7 @@ export default {
|
|||||||
audiobookRemoved(audiobook) {
|
audiobookRemoved(audiobook) {
|
||||||
if (this.$route.name.startsWith('audiobook')) {
|
if (this.$route.name.startsWith('audiobook')) {
|
||||||
if (this.$route.params.id === audiobook.id) {
|
if (this.$route.params.id === audiobook.id) {
|
||||||
this.$router.replace('/')
|
this.$router.replace('/library')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this.$store.commit('audiobooks/remove', audiobook)
|
this.$store.commit('audiobooks/remove', audiobook)
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
export default function ({ store, redirect, route }) {
|
export default function ({ store, redirect, route, app }) {
|
||||||
// If the user is not authenticated
|
// If the user is not authenticated
|
||||||
if (!store.state.user.user) {
|
if (!store.state.user.user) {
|
||||||
if (route.name === 'batch') return redirect('/login')
|
if (route.name === 'batch') return redirect('/login')
|
||||||
return redirect(`/login?redirect=${route.path}`)
|
return redirect(`/login?redirect=${route.fullPath}`)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -71,7 +71,8 @@ module.exports = {
|
|||||||
|
|
||||||
proxy: {
|
proxy: {
|
||||||
'/dev/': { target: 'http://localhost:3333', pathRewrite: { '^/dev/': '' } },
|
'/dev/': { target: 'http://localhost:3333', pathRewrite: { '^/dev/': '' } },
|
||||||
'/local/': { target: process.env.NODE_ENV !== 'production' ? 'http://localhost:3333' : '/', pathRewrite: { '^/local/': '' } }
|
'/local/': { target: process.env.NODE_ENV !== 'production' ? 'http://localhost:3333' : '/', pathRewrite: { '^/local/': '' } },
|
||||||
|
'/metadata/': { target: process.env.NODE_ENV !== 'production' ? 'http://localhost:3333' : '/' }
|
||||||
},
|
},
|
||||||
|
|
||||||
io: {
|
io: {
|
||||||
|
|||||||
Generated
+1
-1
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "audiobookshelf-client",
|
"name": "audiobookshelf-client",
|
||||||
"version": "1.0.0",
|
"version": "1.1.13",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
|||||||
+2
-2
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "audiobookshelf-client",
|
"name": "audiobookshelf-client",
|
||||||
"version": "1.1.12",
|
"version": "1.2.4",
|
||||||
"description": "Audiobook manager and player",
|
"description": "Audiobook manager and player",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@@ -9,7 +9,7 @@
|
|||||||
"start": "nuxt start",
|
"start": "nuxt start",
|
||||||
"generate": "nuxt generate"
|
"generate": "nuxt generate"
|
||||||
},
|
},
|
||||||
"author": "",
|
"author": "advplyr",
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@nuxtjs/axios": "^5.13.6",
|
"@nuxtjs/axios": "^5.13.6",
|
||||||
|
|||||||
@@ -13,9 +13,11 @@
|
|||||||
<div class="mb-2">
|
<div class="mb-2">
|
||||||
<h1 class="text-2xl font-book leading-7">{{ title }}</h1>
|
<h1 class="text-2xl font-book leading-7">{{ title }}</h1>
|
||||||
<h3 v-if="series" class="font-book text-gray-300 text-lg leading-7">{{ seriesText }}</h3>
|
<h3 v-if="series" class="font-book text-gray-300 text-lg leading-7">{{ seriesText }}</h3>
|
||||||
<ui-tooltip :text="authorTooltipText" direction="bottom">
|
<div class="w-min">
|
||||||
<p class="text-sm text-gray-100 leading-7">by {{ author }}</p>
|
<ui-tooltip :text="authorTooltipText" direction="bottom">
|
||||||
</ui-tooltip>
|
<span class="text-sm text-gray-100 leading-7 whitespace-nowrap">by {{ author }}</span>
|
||||||
|
</ui-tooltip>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex-grow" />
|
<div class="flex-grow" />
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -130,7 +130,7 @@ export default {
|
|||||||
this.isProcessing = false
|
this.isProcessing = false
|
||||||
if (data.updates) {
|
if (data.updates) {
|
||||||
this.$toast.success(`Successfully updated ${data.updates} audiobooks`)
|
this.$toast.success(`Successfully updated ${data.updates} audiobooks`)
|
||||||
this.$router.replace('/')
|
this.$router.replace('/library')
|
||||||
} else {
|
} else {
|
||||||
this.$toast.warning('No updates were necessary')
|
this.$toast.warning('No updates were necessary')
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -49,7 +49,12 @@
|
|||||||
<div class="flex-grow" />
|
<div class="flex-grow" />
|
||||||
<div class="w-40 flex flex-col">
|
<div class="w-40 flex flex-col">
|
||||||
<ui-btn color="success" class="mb-4" :loading="isScanning" :disabled="isScanningCovers" @click="scan">Scan</ui-btn>
|
<ui-btn color="success" class="mb-4" :loading="isScanning" :disabled="isScanningCovers" @click="scan">Scan</ui-btn>
|
||||||
<ui-btn color="primary" small :padding-x="2" :loading="isScanningCovers" :disabled="isScanning" @click="scanCovers">Scan for Covers</ui-btn>
|
|
||||||
|
<div class="w-full">
|
||||||
|
<ui-tooltip direction="bottom" text="Only scans audiobooks without a cover. Covers will be applied if a close match is found." class="w-full">
|
||||||
|
<ui-btn color="primary" class="w-full" small :padding-x="2" :loading="isScanningCovers" :disabled="isScanning" @click="scanCovers">Scan for Covers</ui-btn>
|
||||||
|
</ui-tooltip>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
+10
-2
@@ -1,12 +1,20 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="page" :class="streamAudiobook ? 'streaming' : ''">
|
<div class="page" :class="streamAudiobook ? 'streaming' : ''">
|
||||||
<app-book-shelf-toolbar />
|
<!-- <app-book-shelf-toolbar /> -->
|
||||||
<app-book-shelf />
|
<!-- <div class="flex h-full">
|
||||||
|
<app-side-rail />
|
||||||
|
<div class="flex-grow"> -->
|
||||||
|
<!-- <app-book-shelf /> -->
|
||||||
|
<!-- </div> -->
|
||||||
|
<!-- </div> -->
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
export default {
|
export default {
|
||||||
|
asyncData({ redirect }) {
|
||||||
|
redirect('/library')
|
||||||
|
},
|
||||||
data() {
|
data() {
|
||||||
return {}
|
return {}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -0,0 +1,64 @@
|
|||||||
|
<template>
|
||||||
|
<div class="page" :class="streamAudiobook ? 'streaming' : ''">
|
||||||
|
<div class="flex h-full">
|
||||||
|
<app-side-rail />
|
||||||
|
<div class="flex-grow">
|
||||||
|
<app-book-shelf-toolbar :page="id || ''" :search-results="searchResults" :search-query="searchQuery" :selected-series.sync="selectedSeries" />
|
||||||
|
<app-book-shelf :page="id || ''" :search-results="searchResults" :search-query="searchQuery" :selected-series.sync="selectedSeries" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
async asyncData({ params, query, store, app }) {
|
||||||
|
if (query.filter) {
|
||||||
|
store.dispatch('user/updateUserSettings', { filterBy: query.filter })
|
||||||
|
}
|
||||||
|
var searchResults = []
|
||||||
|
var searchQuery = null
|
||||||
|
if (params.id === 'search' && query.query) {
|
||||||
|
searchQuery = query.query
|
||||||
|
searchResults = await app.$axios.$get(`/api/audiobooks?q=${query.query}`).catch((error) => {
|
||||||
|
console.error('Search error', error)
|
||||||
|
return []
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
id: params.id,
|
||||||
|
searchQuery,
|
||||||
|
searchResults,
|
||||||
|
selectedSeries: query.series ? app.$decode(query.series) : null
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
'$route.query'(newVal) {
|
||||||
|
if (this.id === 'search' && this.$route.query.query) {
|
||||||
|
if (this.$route.query.query !== this.searchQuery) {
|
||||||
|
this.newQuery()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
streamAudiobook() {
|
||||||
|
return this.$store.state.streamAudiobook
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
async newQuery() {
|
||||||
|
var query = this.$route.query.query
|
||||||
|
this.searchResults = await this.$axios.$get(`/api/audiobooks?q=${query}`).catch((error) => {
|
||||||
|
console.error('Search error', error)
|
||||||
|
return []
|
||||||
|
})
|
||||||
|
this.searchQuery = query
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -37,7 +37,7 @@ export default {
|
|||||||
if (this.$route.query.redirect) {
|
if (this.$route.query.redirect) {
|
||||||
this.$router.replace(this.$route.query.redirect)
|
this.$router.replace(this.$route.query.redirect)
|
||||||
} else {
|
} else {
|
||||||
this.$router.replace('/')
|
this.$router.replace('/library')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
<template>
|
<template>
|
||||||
<div id="page-wrapper" class="page p-6" :class="streamAudiobook ? 'streaming' : ''">
|
<div id="page-wrapper" class="page p-6" :class="streamAudiobook ? 'streaming' : ''">
|
||||||
<main class="container mx-auto h-full max-w-screen-lg p-6">
|
<main class="container mx-auto h-full max-w-screen-lg p-6">
|
||||||
<article class="max-h-full overflow-y-auto relative flex flex-col bg-primary shadow-xl rounded-md" @drop="drop" @dragover="dragover" @dragleave="dragleave" @dragenter="dragenter">
|
<article class="max-h-full overflow-y-auto relative flex flex-col rounded-md" @drop="drop" @dragover="dragover" @dragleave="dragleave" @dragenter="dragenter">
|
||||||
<h1 class="text-xl font-book px-4 pt-4 pb-2"><span class="text-error pr-4">(Experimental)</span>Audiobook Uploader</h1>
|
<h1 class="text-xl font-book px-8 pt-4 pb-2">Audiobook Uploader</h1>
|
||||||
|
|
||||||
<div class="flex my-2 px-6">
|
<div class="flex my-2 px-6">
|
||||||
<div class="w-1/2 px-2">
|
<div class="w-1/2 px-2">
|
||||||
@@ -170,19 +170,16 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
drop(evt) {
|
drop(evt) {
|
||||||
console.log('Dropped event', evt)
|
|
||||||
this.isDragOver = false
|
this.isDragOver = false
|
||||||
this.preventDefaults(evt)
|
this.preventDefaults(evt)
|
||||||
const files = [...evt.dataTransfer.files]
|
const files = [...evt.dataTransfer.files]
|
||||||
this.filesChanged(files)
|
this.filesChanged(files)
|
||||||
},
|
},
|
||||||
dragover(evt) {
|
dragover(evt) {
|
||||||
console.log('Dragged over', evt)
|
|
||||||
this.isDragOver = true
|
this.isDragOver = true
|
||||||
this.preventDefaults(evt)
|
this.preventDefaults(evt)
|
||||||
},
|
},
|
||||||
dragleave(evt) {
|
dragleave(evt) {
|
||||||
console.log('Dragged leave', evt)
|
|
||||||
this.isDragOver = false
|
this.isDragOver = false
|
||||||
this.preventDefaults(evt)
|
this.preventDefaults(evt)
|
||||||
},
|
},
|
||||||
@@ -195,7 +192,6 @@ export default {
|
|||||||
e.stopPropagation()
|
e.stopPropagation()
|
||||||
},
|
},
|
||||||
filesChanged(files) {
|
filesChanged(files) {
|
||||||
console.log('FilesChanged', files)
|
|
||||||
this.showUploader = false
|
this.showUploader = false
|
||||||
|
|
||||||
for (let i = 0; i < files.length; i++) {
|
for (let i = 0; i < files.length; i++) {
|
||||||
|
|||||||
@@ -125,3 +125,6 @@ export {
|
|||||||
encode,
|
encode,
|
||||||
decode
|
decode
|
||||||
}
|
}
|
||||||
|
export default ({ app }, inject) => {
|
||||||
|
app.$decode = decode
|
||||||
|
}
|
||||||
@@ -5,7 +5,7 @@ function parseSemver(ver) {
|
|||||||
if (!ver) return null
|
if (!ver) return null
|
||||||
var groups = ver.match(/^v((([0-9]+)\.([0-9]+)\.([0-9]+)(?:-([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)$/)
|
var groups = ver.match(/^v((([0-9]+)\.([0-9]+)\.([0-9]+)(?:-([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)$/)
|
||||||
if (groups && groups.length > 6) {
|
if (groups && groups.length > 6) {
|
||||||
var total = Number(groups[3]) * 100 + Number(groups[4]) * 10 + Number(groups[5])
|
var total = Number(groups[3]) * 10000 + Number(groups[4]) * 100 + Number(groups[5])
|
||||||
if (isNaN(total)) {
|
if (isNaN(total)) {
|
||||||
console.warn('Invalid version total', groups[3], groups[4], groups[5])
|
console.warn('Invalid version total', groups[3], groups[4], groups[5])
|
||||||
return null
|
return null
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ const STANDARD_GENRES = ['Adventure', 'Autobiography', 'Biography', 'Childrens',
|
|||||||
|
|
||||||
export const state = () => ({
|
export const state = () => ({
|
||||||
audiobooks: [],
|
audiobooks: [],
|
||||||
|
lastLoad: 0,
|
||||||
listeners: [],
|
listeners: [],
|
||||||
genres: [...STANDARD_GENRES],
|
genres: [...STANDARD_GENRES],
|
||||||
tags: [],
|
tags: [],
|
||||||
@@ -16,12 +17,12 @@ export const getters = {
|
|||||||
getAudiobook: (state) => id => {
|
getAudiobook: (state) => id => {
|
||||||
return state.audiobooks.find(ab => ab.id === id)
|
return state.audiobooks.find(ab => ab.id === id)
|
||||||
},
|
},
|
||||||
getFiltered: (state, getters, rootState) => () => {
|
getFiltered: (state, getters, rootState, rootGetters) => () => {
|
||||||
var filtered = state.audiobooks
|
var filtered = state.audiobooks
|
||||||
var settings = rootState.user.settings || {}
|
var settings = rootState.user.settings || {}
|
||||||
var filterBy = settings.filterBy || ''
|
var filterBy = settings.filterBy || ''
|
||||||
|
|
||||||
var searchGroups = ['genres', 'tags', 'series', 'authors']
|
var searchGroups = ['genres', 'tags', 'series', 'authors', 'progress']
|
||||||
var group = searchGroups.find(_group => filterBy.startsWith(_group + '.'))
|
var group = searchGroups.find(_group => filterBy.startsWith(_group + '.'))
|
||||||
if (group) {
|
if (group) {
|
||||||
var filter = decode(filterBy.replace(`${group}.`, ''))
|
var filter = decode(filterBy.replace(`${group}.`, ''))
|
||||||
@@ -29,6 +30,16 @@ export const getters = {
|
|||||||
else if (group === 'tags') filtered = filtered.filter(ab => ab.tags.includes(filter))
|
else if (group === 'tags') filtered = filtered.filter(ab => ab.tags.includes(filter))
|
||||||
else if (group === 'series') filtered = filtered.filter(ab => ab.book && ab.book.series === filter)
|
else if (group === 'series') filtered = filtered.filter(ab => ab.book && ab.book.series === filter)
|
||||||
else if (group === 'authors') filtered = filtered.filter(ab => ab.book && ab.book.author === filter)
|
else if (group === 'authors') filtered = filtered.filter(ab => ab.book && ab.book.author === filter)
|
||||||
|
else if (group === 'progress') {
|
||||||
|
filtered = filtered.filter(ab => {
|
||||||
|
var userAudiobook = rootGetters['user/getUserAudiobook'](ab.id)
|
||||||
|
var isRead = userAudiobook && userAudiobook.isRead
|
||||||
|
if (filter === 'Read' && isRead) return true
|
||||||
|
if (filter === 'Unread' && !isRead) return true
|
||||||
|
if (filter === 'In Progress' && (userAudiobook && !userAudiobook.isRead && userAudiobook.progress > 0)) return true
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (state.keywordFilter) {
|
if (state.keywordFilter) {
|
||||||
const keywordFilterKeys = ['title', 'subtitle', 'author', 'series', 'narrarator']
|
const keywordFilterKeys = ['title', 'subtitle', 'author', 'series', 'narrarator']
|
||||||
@@ -53,6 +64,33 @@ export const getters = {
|
|||||||
return value
|
return value
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
getSeriesGroups: (state, getters, rootState) => () => {
|
||||||
|
var series = {}
|
||||||
|
state.audiobooks.forEach((audiobook) => {
|
||||||
|
if (audiobook.book && audiobook.book.series) {
|
||||||
|
if (series[audiobook.book.series]) {
|
||||||
|
series[audiobook.book.series].books.push(audiobook)
|
||||||
|
} else {
|
||||||
|
series[audiobook.book.series] = {
|
||||||
|
type: 'series',
|
||||||
|
name: audiobook.book.series || '',
|
||||||
|
books: [audiobook]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
var seriesArray = Object.values(series).map((_series) => {
|
||||||
|
_series.books = sort(_series.books)['asc']((ab) => {
|
||||||
|
return ab.book && ab.book.volumeNumber && !isNaN(ab.book.volumeNumber) ? Number(ab.book.volumeNumber) : null
|
||||||
|
})
|
||||||
|
return _series
|
||||||
|
})
|
||||||
|
if (state.keywordFilter) {
|
||||||
|
const keywordFilter = state.keywordFilter.toLowerCase()
|
||||||
|
return seriesArray.filter((_series) => _series.name.toLowerCase().includes(keywordFilter))
|
||||||
|
}
|
||||||
|
return seriesArray
|
||||||
|
},
|
||||||
getUniqueAuthors: (state) => {
|
getUniqueAuthors: (state) => {
|
||||||
var _authors = state.audiobooks.filter(ab => !!(ab.book && ab.book.author)).map(ab => ab.book.author)
|
var _authors = state.audiobooks.filter(ab => !!(ab.book && ab.book.author)).map(ab => ab.book.author)
|
||||||
return [...new Set(_authors)].sort((a, b) => a.toLowerCase() < b.toLowerCase() ? -1 : 1)
|
return [...new Set(_authors)].sort((a, b) => a.toLowerCase() < b.toLowerCase() ? -1 : 1)
|
||||||
@@ -61,29 +99,63 @@ export const getters = {
|
|||||||
var _genres = []
|
var _genres = []
|
||||||
state.audiobooks.filter(ab => !!(ab.book && ab.book.genres)).forEach(ab => _genres = _genres.concat(ab.book.genres))
|
state.audiobooks.filter(ab => !!(ab.book && ab.book.genres)).forEach(ab => _genres = _genres.concat(ab.book.genres))
|
||||||
return [...new Set(_genres)].sort((a, b) => a.toLowerCase() < b.toLowerCase() ? -1 : 1)
|
return [...new Set(_genres)].sort((a, b) => a.toLowerCase() < b.toLowerCase() ? -1 : 1)
|
||||||
|
},
|
||||||
|
getBookCoverSrc: (state, getters, rootState, rootGetters) => (book, placeholder = '/book_placeholder.jpg') => {
|
||||||
|
if (!book || !book.cover || book.cover === placeholder) return placeholder
|
||||||
|
var cover = book.cover
|
||||||
|
|
||||||
|
// Absolute URL covers
|
||||||
|
if (cover.startsWith('http:') || cover.startsWith('https:')) return cover
|
||||||
|
|
||||||
|
// Server hosted covers
|
||||||
|
try {
|
||||||
|
// Ensure cover is refreshed if cached
|
||||||
|
var bookLastUpdate = book.lastUpdate || Date.now()
|
||||||
|
var userToken = rootGetters['user/getToken']
|
||||||
|
|
||||||
|
var url = new URL(cover, document.baseURI)
|
||||||
|
return url.href + `?token=${userToken}&ts=${bookLastUpdate}`
|
||||||
|
} catch (err) {
|
||||||
|
console.error(err)
|
||||||
|
return placeholder
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export const actions = {
|
export const actions = {
|
||||||
load({ commit, rootState }) {
|
// Return true if calling load
|
||||||
|
load({ state, commit, rootState }) {
|
||||||
if (!rootState.user || !rootState.user.user) {
|
if (!rootState.user || !rootState.user.user) {
|
||||||
console.error('audiobooks/load - User not set')
|
console.error('audiobooks/load - User not set')
|
||||||
return
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Don't load again if already loaded in the last 5 minutes
|
||||||
|
var lastLoadDiff = Date.now() - state.lastLoad
|
||||||
|
if (lastLoadDiff < 5 * 60 * 1000) {
|
||||||
|
// Already up to date
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
this.$axios
|
this.$axios
|
||||||
.$get(`/api/audiobooks`)
|
.$get(`/api/audiobooks`)
|
||||||
.then((data) => {
|
.then((data) => {
|
||||||
commit('set', data)
|
commit('set', data)
|
||||||
|
commit('setLastLoad')
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.error('Failed', error)
|
console.error('Failed', error)
|
||||||
commit('set', [])
|
commit('set', [])
|
||||||
})
|
})
|
||||||
|
return true
|
||||||
},
|
},
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const mutations = {
|
export const mutations = {
|
||||||
|
setLastLoad(state) {
|
||||||
|
state.lastLoad = Date.now()
|
||||||
|
},
|
||||||
setKeywordFilter(state, val) {
|
setKeywordFilter(state, val) {
|
||||||
state.keywordFilter = val
|
state.keywordFilter = val
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -33,6 +33,9 @@ export const getters = {
|
|||||||
},
|
},
|
||||||
getUserCanDownload: (state) => {
|
getUserCanDownload: (state) => {
|
||||||
return state.user && state.user.permissions ? !!state.user.permissions.download : false
|
return state.user && state.user.permissions ? !!state.user.permissions.download : false
|
||||||
|
},
|
||||||
|
getUserCanUpload: (state) => {
|
||||||
|
return state.user && state.user.permissions ? !!state.user.permissions.upload : false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -41,6 +44,8 @@ export const actions = {
|
|||||||
var updatePayload = {
|
var updatePayload = {
|
||||||
...payload
|
...payload
|
||||||
}
|
}
|
||||||
|
// Immediately update
|
||||||
|
commit('setSettings', updatePayload)
|
||||||
return this.$axios.$patch('/api/user/settings', updatePayload).then((result) => {
|
return this.$axios.$patch('/api/user/settings', updatePayload).then((result) => {
|
||||||
if (result.success) {
|
if (result.success) {
|
||||||
commit('setSettings', result.settings)
|
commit('setSettings', result.settings)
|
||||||
@@ -60,10 +65,8 @@ export const mutations = {
|
|||||||
state.user = user
|
state.user = user
|
||||||
if (user) {
|
if (user) {
|
||||||
if (user.token) localStorage.setItem('token', user.token)
|
if (user.token) localStorage.setItem('token', user.token)
|
||||||
console.log('setUser', user.username)
|
|
||||||
} else {
|
} else {
|
||||||
localStorage.removeItem('token')
|
localStorage.removeItem('token')
|
||||||
console.warn('setUser cleared')
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
setSettings(state, settings) {
|
setSettings(state, settings) {
|
||||||
|
|||||||
Generated
+35
-1
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "audiobookshelf",
|
"name": "audiobookshelf",
|
||||||
"version": "1.1.7",
|
"version": "1.1.13",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -138,6 +138,11 @@
|
|||||||
"is-primitive": "^3.0.1"
|
"is-primitive": "^3.0.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"array-back": {
|
||||||
|
"version": "3.1.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/array-back/-/array-back-3.1.0.tgz",
|
||||||
|
"integrity": "sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q=="
|
||||||
|
},
|
||||||
"array-flatten": {
|
"array-flatten": {
|
||||||
"version": "1.1.1",
|
"version": "1.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
|
||||||
@@ -288,6 +293,17 @@
|
|||||||
"mimic-response": "^1.0.0"
|
"mimic-response": "^1.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"command-line-args": {
|
||||||
|
"version": "5.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/command-line-args/-/command-line-args-5.2.0.tgz",
|
||||||
|
"integrity": "sha512-4zqtU1hYsSJzcJBOcNZIbW5Fbk9BkjCp1pZVhQKoRaWL5J7N4XphDLwo8aWwdQpTugxwu+jf9u2ZhkXiqp5Z6A==",
|
||||||
|
"requires": {
|
||||||
|
"array-back": "^3.1.0",
|
||||||
|
"find-replace": "^3.0.0",
|
||||||
|
"lodash.camelcase": "^4.3.0",
|
||||||
|
"typical": "^4.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"component-emitter": {
|
"component-emitter": {
|
||||||
"version": "1.3.0",
|
"version": "1.3.0",
|
||||||
"resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz",
|
||||||
@@ -566,6 +582,14 @@
|
|||||||
"unpipe": "~1.0.0"
|
"unpipe": "~1.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"find-replace": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/find-replace/-/find-replace-3.0.0.tgz",
|
||||||
|
"integrity": "sha512-6Tb2myMioCAgv5kfvP5/PkZZ/ntTpVK39fHY7WkWBgvbeE+VHd/tZuZ4mrC+bxh4cfOZeYKVPaJIZtZXV7GNCQ==",
|
||||||
|
"requires": {
|
||||||
|
"array-back": "^3.0.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
"fluent-ffmpeg": {
|
"fluent-ffmpeg": {
|
||||||
"version": "2.1.2",
|
"version": "2.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/fluent-ffmpeg/-/fluent-ffmpeg-2.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/fluent-ffmpeg/-/fluent-ffmpeg-2.1.2.tgz",
|
||||||
@@ -830,6 +854,11 @@
|
|||||||
"got": "11.3.x"
|
"got": "11.3.x"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"lodash.camelcase": {
|
||||||
|
"version": "4.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz",
|
||||||
|
"integrity": "sha1-soqmKIorn8ZRA1x3EfZathkDMaY="
|
||||||
|
},
|
||||||
"lodash.defaults": {
|
"lodash.defaults": {
|
||||||
"version": "4.2.0",
|
"version": "4.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz",
|
||||||
@@ -1339,6 +1368,11 @@
|
|||||||
"mime-types": "~2.1.24"
|
"mime-types": "~2.1.24"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"typical": {
|
||||||
|
"version": "4.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/typical/-/typical-4.0.0.tgz",
|
||||||
|
"integrity": "sha512-VAH4IvQ7BDFYglMd7BPRDfLgxZZX4O4TFcRDA6EN5X7erNJJq+McIEp8np9aVtxrCJ6qx4GTYVfOWNjcqwZgRw=="
|
||||||
|
},
|
||||||
"universalify": {
|
"universalify": {
|
||||||
"version": "2.0.0",
|
"version": "2.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz",
|
||||||
|
|||||||
+16
-3
@@ -1,11 +1,23 @@
|
|||||||
{
|
{
|
||||||
"name": "audiobookshelf",
|
"name": "audiobookshelf",
|
||||||
"version": "1.1.12",
|
"version": "1.2.4",
|
||||||
"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": {
|
||||||
"dev": "node index.js",
|
"dev": "node index.js",
|
||||||
"start": "node index.js"
|
"start": "node index.js",
|
||||||
|
"client": "cd client && npm install && npm run generate",
|
||||||
|
"prod": "npm run client && npm install && node prod.js",
|
||||||
|
"build-win": "npm run build-prep && pkg -t node12-win-x64 -o ./dist/win/audiobookshelf .",
|
||||||
|
"build-linux": "build/linuxpackager"
|
||||||
|
},
|
||||||
|
"bin": "prod.js",
|
||||||
|
"pkg": {
|
||||||
|
"assets": "client/dist/**/*",
|
||||||
|
"scripts": [
|
||||||
|
"prod.js",
|
||||||
|
"server/**/*.js"
|
||||||
|
]
|
||||||
},
|
},
|
||||||
"author": "advplyr",
|
"author": "advplyr",
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
@@ -13,6 +25,7 @@
|
|||||||
"archiver": "^5.3.0",
|
"archiver": "^5.3.0",
|
||||||
"axios": "^0.21.1",
|
"axios": "^0.21.1",
|
||||||
"bcryptjs": "^2.4.3",
|
"bcryptjs": "^2.4.3",
|
||||||
|
"command-line-args": "^5.2.0",
|
||||||
"cookie-parser": "^1.4.5",
|
"cookie-parser": "^1.4.5",
|
||||||
"express": "^4.17.1",
|
"express": "^4.17.1",
|
||||||
"express-fileupload": "^1.2.1",
|
"express-fileupload": "^1.2.1",
|
||||||
|
|||||||
@@ -0,0 +1,31 @@
|
|||||||
|
const optionDefinitions = [
|
||||||
|
{ name: 'config', alias: 'c', type: String },
|
||||||
|
{ name: 'audiobooks', alias: 'a', type: String },
|
||||||
|
{ name: 'metadata', alias: 'm', type: String },
|
||||||
|
{ name: 'port', alias: 'p', type: String }
|
||||||
|
]
|
||||||
|
|
||||||
|
const commandLineArgs = require('command-line-args')
|
||||||
|
const options = commandLineArgs(optionDefinitions)
|
||||||
|
|
||||||
|
const Path = require('path')
|
||||||
|
process.env.TOKEN_SECRET = '09f26e402586e2faa8da4c98a35f1b20d6b033c6097befa8be3486a829587fe2f90a832bd3ff9d42710a4da095a2ce285b009f0c3730cd9b8e1af3eb84df6611'
|
||||||
|
process.env.NODE_ENV = 'production'
|
||||||
|
|
||||||
|
const server = require('./server/Server')
|
||||||
|
|
||||||
|
global.appRoot = __dirname
|
||||||
|
|
||||||
|
var inputConfig = options.config ? Path.resolve(options.config) : null
|
||||||
|
var inputAudiobook = options.audiobooks ? Path.resolve(options.audiobooks) : null
|
||||||
|
var inputMetadata = options.metadata ? Path.resolve(options.metadata) : null
|
||||||
|
|
||||||
|
const PORT = options.port || process.env.PORT || 3333
|
||||||
|
const CONFIG_PATH = inputConfig || process.env.CONFIG_PATH || Path.resolve('config')
|
||||||
|
const AUDIOBOOK_PATH = inputAudiobook || process.env.AUDIOBOOK_PATH || Path.resolve('audiobooks')
|
||||||
|
const METADATA_PATH = inputMetadata || process.env.METADATA_PATH || Path.resolve('metadata')
|
||||||
|
|
||||||
|
console.log(process.env.NODE_ENV, 'Config', CONFIG_PATH, METADATA_PATH, AUDIOBOOK_PATH)
|
||||||
|
|
||||||
|
const Server = new server(PORT, CONFIG_PATH, METADATA_PATH, AUDIOBOOK_PATH)
|
||||||
|
Server.start()
|
||||||
@@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
AudioBookshelf is a self-hosted audiobook server for managing and playing your audiobooks.
|
AudioBookshelf is a self-hosted audiobook server for managing and playing your audiobooks.
|
||||||
|
|
||||||
|
See [Install guides](https://audiobookshelf.org/install) and docs coming soon to [audiobookshelf.org](https://audiobookshelf.org)
|
||||||
|
|
||||||
Android app is in beta, try it out on the [Google Play Store](https://play.google.com/store/apps/details?id=com.audiobookshelf.app)
|
Android app is in beta, try it out on the [Google Play Store](https://play.google.com/store/apps/details?id=com.audiobookshelf.app)
|
||||||
|
|
||||||
**Free & open source Android/iOS app is in development**
|
**Free & open source Android/iOS app is in development**
|
||||||
@@ -9,36 +11,137 @@ Android app is in beta, try it out on the [Google Play Store](https://play.googl
|
|||||||
<img alt="Screenshot1" src="https://github.com/advplyr/audiobookshelf/raw/master/images/ss_streaming.png" />
|
<img alt="Screenshot1" src="https://github.com/advplyr/audiobookshelf/raw/master/images/ss_streaming.png" />
|
||||||
|
|
||||||
|
|
||||||
#### Folder Structures Supported:
|
## Directory Structure
|
||||||
|
|
||||||
```bash
|
Author, Series, Volume Number, Title and Publish Year can all be parsed from your folder structure.
|
||||||
/Title/...
|
|
||||||
/Author/Title/...
|
|
||||||
/Author/Series/Title/...
|
|
||||||
|
|
||||||
Title can start with the publish year like so:
|
**Note**: Files in the root directory `/audiobooks` will be ignored, all audiobooks should be in a directory
|
||||||
/1989 - Book Title/...
|
|
||||||
|
|
||||||
(Optional Setting) Subtitle can be seperated to its own field:
|
**1 Folder:** `/Title/...`\
|
||||||
/Book Title - With a Subtitle/...
|
**2 Folders:** `/Author/Title/...`\
|
||||||
/1989 - Book Title - With a Subtitle/...
|
**3 Folders:** `/Author/Series/Title/...`
|
||||||
will store "With a Subtitle" as the subtitle
|
|
||||||
```
|
### Parsing publish year
|
||||||
|
|
||||||
|
`/1984 - Hackers/...`\
|
||||||
|
Will save the publish year as `1984` and the title as `Hackers`
|
||||||
|
|
||||||
|
### Parsing volume number (only for series)
|
||||||
|
|
||||||
|
`/Book 3 - Hackers/...`\
|
||||||
|
Will save the volume number as `3` and the title as `Hackers`
|
||||||
|
|
||||||
|
`Book` `Volume` `Vol` `Vol.` are all supported case insensitive
|
||||||
|
|
||||||
|
These combinations will also work:\
|
||||||
|
`/Hackers - Vol. 3/...`\
|
||||||
|
`/1984 - Volume 3 - Hackers/...`\
|
||||||
|
`/1984 - Hackers Book 3/...`
|
||||||
|
|
||||||
|
|
||||||
#### Features coming soon:
|
### Parsing subtitles (optional in settings)
|
||||||
|
|
||||||
|
Title Folder: `/Hackers - Heroes of the Computer Revolution/...`
|
||||||
|
|
||||||
|
Will save the title as `Hackers` and the subtitle as `Heroes of the Computer Revolution`
|
||||||
|
|
||||||
|
|
||||||
|
### Full example
|
||||||
|
|
||||||
|
`/Steven Levy/The Hacker Series/1984 - Hackers - Heroes of the Computer Revolution - Vol. 1/...`
|
||||||
|
|
||||||
|
**Becomes:**
|
||||||
|
| Key | Value |
|
||||||
|
|---------------|-----------------------------------|
|
||||||
|
| Author | Steven Levy |
|
||||||
|
| Series | The Hacker Series |
|
||||||
|
| Publish Year | 1984 |
|
||||||
|
| Title | Hackers |
|
||||||
|
| Subtitle | Heroes of the Computer Revolution |
|
||||||
|
| Volume Number | 1 |
|
||||||
|
|
||||||
|
|
||||||
|
## Features coming soon
|
||||||
|
|
||||||
* Support different views to see more details of each audiobook
|
* Support different views to see more details of each audiobook
|
||||||
* iOS App (Android is in beta [here](https://play.google.com/store/apps/details?id=com.audiobookshelf.app))
|
* iOS App (Android is in beta [here](https://play.google.com/store/apps/details?id=com.audiobookshelf.app))
|
||||||
|
|
||||||
<img alt="Screenshot2" src="https://github.com/advplyr/audiobookshelf/raw/master/images/ss_audiobook.png" />
|
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
Built to run in Docker for now (also on Unraid server Community Apps)
|
### Docker Install
|
||||||
|
Available in Unraid Community Apps
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
docker run -d -p 1337:80 -v /audiobooks:/audiobooks -v /config:/config -v /metadata:/metadata --name audiobookshelf --rm advplyr/audiobookshelf
|
docker pull advplyr/audiobookshelf
|
||||||
|
|
||||||
|
docker run -d \
|
||||||
|
-p 1337:80 \
|
||||||
|
-v </path/to/audiobooks>:/audiobooks \
|
||||||
|
-v </path/to/config>:/config \
|
||||||
|
-v </path/to/metadata>:/metadata \
|
||||||
|
--name audiobookshelf \
|
||||||
|
--rm advplyr/audiobookshelf
|
||||||
|
```
|
||||||
|
|
||||||
|
### Linux (amd64) Install
|
||||||
|
|
||||||
|
A simple installer is added to setup the initial config. If you already have audiobooks, you can enter the path to your audiobooks during the install. The installer will create a user and group named `audiobookshelf`.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Ubuntu Install via PPA
|
||||||
|
|
||||||
|
A PPA is hosted on [github](https://github.com/advplyr/audiobookshelf-ppa), add and install:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -s --compressed "https://advplyr.github.io/audiobookshelf-ppa/KEY.gpg" | sudo apt-key add -
|
||||||
|
|
||||||
|
sudo curl -s --compressed -o /etc/apt/sources.list.d/audiobookshelf.list "https://advplyr.github.io/audiobookshelf-ppa/audiobookshelf.list"
|
||||||
|
|
||||||
|
sudo apt update
|
||||||
|
|
||||||
|
sudo apt install audiobookshelf
|
||||||
|
```
|
||||||
|
|
||||||
|
or use a single command
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -s --compressed "https://advplyr.github.io/audiobookshelf-ppa/KEY.gpg" | sudo apt-key add - && sudo curl -s --compressed -o /etc/apt/sources.list.d/audiobookshelf.list "https://advplyr.github.io/audiobookshelf-ppa/audiobookshelf.list" && sudo apt update && sudo apt install audiobookshelf
|
||||||
|
```
|
||||||
|
|
||||||
|
### Install via debian package
|
||||||
|
|
||||||
|
Get the `deb` file from the [github repo](https://github.com/advplyr/audiobookshelf-ppa).
|
||||||
|
|
||||||
|
```bash
|
||||||
|
wget https://advplyr.github.io/audiobookshelf-ppa/audiobookshelf_1.2.3_amd64.deb
|
||||||
|
|
||||||
|
sudo apt install ./audiobookshelf_1.2.3_amd64.deb
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
#### File locations
|
||||||
|
|
||||||
|
Project directory: `/usr/share/audiobookshelf/`
|
||||||
|
|
||||||
|
Config file: `/etc/default/audiobookshelf`
|
||||||
|
|
||||||
|
System Service: `/lib/systemd/system/audiobookshelf.service`
|
||||||
|
|
||||||
|
Ffmpeg static build: `/usr/lib/audiobookshelf-ffmpeg/`
|
||||||
|
|
||||||
|
## Run from source
|
||||||
|
|
||||||
|
Note: you will need `npm`, `node12`, and `ffmpeg` to run this project locally
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git clone https://github.com/advplyr/audiobookshelf.git
|
||||||
|
cd audiobookshelf
|
||||||
|
|
||||||
|
# All paths default to root directory. Config path is the database.
|
||||||
|
# Directories will be created if they don't exist
|
||||||
|
# Paths are relative to the root directory, so "../Audiobooks" would be a valid path
|
||||||
|
npm run prod -- -p [PORT] --audiobooks [AUDIOBOOKS_PATH] --config [CONFIG_PATH] --metadata [METADATA_PATH]
|
||||||
```
|
```
|
||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
|
|||||||
+86
-2
@@ -1,10 +1,13 @@
|
|||||||
const express = require('express')
|
const express = require('express')
|
||||||
|
const Path = require('path')
|
||||||
|
const fs = require('fs-extra')
|
||||||
const Logger = require('./Logger')
|
const Logger = require('./Logger')
|
||||||
const User = require('./objects/User')
|
const User = require('./objects/User')
|
||||||
const { isObject } = require('./utils/index')
|
const { isObject, isAcceptableCoverMimeType } = require('./utils/index')
|
||||||
|
const { CoverDestination } = require('./utils/constants')
|
||||||
|
|
||||||
class ApiController {
|
class ApiController {
|
||||||
constructor(db, scanner, auth, streamManager, rssFeeds, downloadManager, emitter, clientEmitter) {
|
constructor(MetadataPath, db, scanner, auth, streamManager, rssFeeds, downloadManager, emitter, clientEmitter) {
|
||||||
this.db = db
|
this.db = db
|
||||||
this.scanner = scanner
|
this.scanner = scanner
|
||||||
this.auth = auth
|
this.auth = auth
|
||||||
@@ -13,6 +16,7 @@ class ApiController {
|
|||||||
this.downloadManager = downloadManager
|
this.downloadManager = downloadManager
|
||||||
this.emitter = emitter
|
this.emitter = emitter
|
||||||
this.clientEmitter = clientEmitter
|
this.clientEmitter = clientEmitter
|
||||||
|
this.MetadataPath = MetadataPath
|
||||||
|
|
||||||
this.router = express()
|
this.router = express()
|
||||||
this.init()
|
this.init()
|
||||||
@@ -30,6 +34,7 @@ class ApiController {
|
|||||||
this.router.get('/audiobook/:id', this.getAudiobook.bind(this))
|
this.router.get('/audiobook/:id', this.getAudiobook.bind(this))
|
||||||
this.router.delete('/audiobook/:id', this.deleteAudiobook.bind(this))
|
this.router.delete('/audiobook/:id', this.deleteAudiobook.bind(this))
|
||||||
this.router.patch('/audiobook/:id/tracks', this.updateAudiobookTracks.bind(this))
|
this.router.patch('/audiobook/:id/tracks', this.updateAudiobookTracks.bind(this))
|
||||||
|
this.router.post('/audiobook/:id/cover', this.uploadAudiobookCover.bind(this))
|
||||||
this.router.patch('/audiobook/:id', this.updateAudiobook.bind(this))
|
this.router.patch('/audiobook/:id', this.updateAudiobook.bind(this))
|
||||||
|
|
||||||
this.router.get('/metadata/:id/:trackIndex', this.getMetadata.bind(this))
|
this.router.get('/metadata/:id/:trackIndex', this.getMetadata.bind(this))
|
||||||
@@ -217,6 +222,85 @@ class ApiController {
|
|||||||
res.json(audiobook.toJSON())
|
res.json(audiobook.toJSON())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async uploadAudiobookCover(req, res) {
|
||||||
|
if (!req.user.canUpload || !req.user.canUpdate) {
|
||||||
|
Logger.warn('User attempted to upload a cover without permission', req.user)
|
||||||
|
return res.sendStatus(403)
|
||||||
|
}
|
||||||
|
if (!req.files || !req.files.cover) {
|
||||||
|
return res.status(400).send('No files were uploaded')
|
||||||
|
}
|
||||||
|
var audiobookId = req.params.id
|
||||||
|
var audiobook = this.db.audiobooks.find(ab => ab.id === audiobookId)
|
||||||
|
if (!audiobook) {
|
||||||
|
return res.status(404).send('Audiobook not found')
|
||||||
|
}
|
||||||
|
|
||||||
|
var coverFile = req.files.cover
|
||||||
|
var mimeType = coverFile.mimetype
|
||||||
|
var extname = Path.extname(coverFile.name.toLowerCase()) || '.jpg'
|
||||||
|
if (!isAcceptableCoverMimeType(mimeType)) {
|
||||||
|
return res.status(400).send('Invalid image file type: ' + mimeType)
|
||||||
|
}
|
||||||
|
|
||||||
|
var coverDestination = this.db.serverSettings ? this.db.serverSettings.coverDestination : CoverDestination.METADATA
|
||||||
|
Logger.info(`[ApiController] Cover Upload destination ${coverDestination}`)
|
||||||
|
|
||||||
|
var coverDirpath = audiobook.fullPath
|
||||||
|
var coverRelDirpath = Path.join('/local', audiobook.path)
|
||||||
|
if (coverDestination === CoverDestination.METADATA) {
|
||||||
|
coverDirpath = Path.join(this.MetadataPath, 'books', audiobookId)
|
||||||
|
coverRelDirpath = Path.join('/metadata', 'books', audiobookId)
|
||||||
|
Logger.debug(`[ApiController] storing in metadata | ${coverDirpath}`)
|
||||||
|
await fs.ensureDir(coverDirpath)
|
||||||
|
} else {
|
||||||
|
Logger.debug(`[ApiController] storing in audiobook | ${coverRelDirpath}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
var coverFilename = `cover${extname}`
|
||||||
|
var coverFullPath = Path.join(coverDirpath, coverFilename)
|
||||||
|
var coverPath = Path.join(coverRelDirpath, coverFilename)
|
||||||
|
|
||||||
|
// If current cover is a metadata cover and does not match replacement, then remove it
|
||||||
|
var currentBookCover = audiobook.book.cover
|
||||||
|
if (currentBookCover && currentBookCover.startsWith(Path.sep + 'metadata')) {
|
||||||
|
Logger.debug(`Current Book Cover is metadata ${currentBookCover}`)
|
||||||
|
if (currentBookCover !== coverPath) {
|
||||||
|
Logger.info(`[ApiController] removing old metadata cover "${currentBookCover}"`)
|
||||||
|
var oldFullBookCoverPath = Path.join(this.MetadataPath, currentBookCover.replace(Path.sep + 'metadata', ''))
|
||||||
|
|
||||||
|
// Metadata path may have changed, check if exists first
|
||||||
|
var exists = await fs.pathExists(oldFullBookCoverPath)
|
||||||
|
if (exists) {
|
||||||
|
try {
|
||||||
|
await fs.remove(oldFullBookCoverPath)
|
||||||
|
} catch (error) {
|
||||||
|
Logger.error(`[ApiController] Failed to remove old metadata book cover ${oldFullBookCoverPath}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var success = await coverFile.mv(coverFullPath).then(() => true).catch((error) => {
|
||||||
|
Logger.error('Failed to move cover file', path, error)
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!success) {
|
||||||
|
return res.status(500).send('Failed to move cover into destination')
|
||||||
|
}
|
||||||
|
|
||||||
|
Logger.info(`[ApiController] Uploaded audiobook cover "${coverPath}" for "${audiobook.title}"`)
|
||||||
|
|
||||||
|
audiobook.updateBookCover(coverPath)
|
||||||
|
await this.db.updateAudiobook(audiobook)
|
||||||
|
this.emitter('audiobook_updated', audiobook.toJSONMinified())
|
||||||
|
res.json({
|
||||||
|
success: true,
|
||||||
|
cover: coverPath
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
async updateAudiobook(req, res) {
|
async updateAudiobook(req, res) {
|
||||||
if (!req.user.canUpdate) {
|
if (!req.user.canUpdate) {
|
||||||
Logger.warn('User attempted to update without permission', req.user)
|
Logger.warn('User attempted to update without permission', req.user)
|
||||||
|
|||||||
+10
-2
@@ -38,8 +38,16 @@ class Auth {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async authMiddleware(req, res, next) {
|
async authMiddleware(req, res, next) {
|
||||||
const authHeader = req.headers['authorization']
|
var token = null
|
||||||
const token = authHeader && authHeader.split(' ')[1]
|
|
||||||
|
// If using a get request, the token can be passed as a query string
|
||||||
|
if (req.method === 'GET' && req.query && req.query.token) {
|
||||||
|
token = req.query.token
|
||||||
|
} else {
|
||||||
|
const authHeader = req.headers['authorization']
|
||||||
|
token = authHeader && authHeader.split(' ')[1]
|
||||||
|
}
|
||||||
|
|
||||||
if (token == null) {
|
if (token == null) {
|
||||||
Logger.error('Api called without a token', req.path)
|
Logger.error('Api called without a token', req.path)
|
||||||
return res.sendStatus(401)
|
return res.sendStatus(401)
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ const Logger = require('./Logger')
|
|||||||
const Download = require('./objects/Download')
|
const Download = require('./objects/Download')
|
||||||
const { writeConcatFile, writeMetadataFile } = require('./utils/ffmpegHelpers')
|
const { writeConcatFile, writeMetadataFile } = require('./utils/ffmpegHelpers')
|
||||||
const { getFileSize } = require('./utils/fileUtils')
|
const { getFileSize } = require('./utils/fileUtils')
|
||||||
|
const TAG = 'DownloadManager'
|
||||||
class DownloadManager {
|
class DownloadManager {
|
||||||
constructor(db, MetadataPath, AudiobookPath, emitter) {
|
constructor(db, MetadataPath, AudiobookPath, emitter) {
|
||||||
this.db = db
|
this.db = db
|
||||||
@@ -260,7 +260,21 @@ class DownloadManager {
|
|||||||
output: download.fullPath,
|
output: download.fullPath,
|
||||||
}
|
}
|
||||||
|
|
||||||
var worker = new workerThreads.Worker('./server/utils/downloadWorker.js', { workerData })
|
var worker = null
|
||||||
|
try {
|
||||||
|
var workerPath = Path.join(global.appRoot, 'server/utils/downloadWorker.js')
|
||||||
|
worker = new workerThreads.Worker(workerPath, { workerData })
|
||||||
|
} catch (error) {
|
||||||
|
Logger.error(`[${TAG}] Start worker thread failed`, error)
|
||||||
|
if (download.socket) {
|
||||||
|
var downloadJson = download.toJSON()
|
||||||
|
download.socket.emit('download_failed', downloadJson)
|
||||||
|
}
|
||||||
|
this.removeDownload(download)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
worker.on('message', (message) => {
|
worker.on('message', (message) => {
|
||||||
if (message != null && typeof message === 'object') {
|
if (message != null && typeof message === 'object') {
|
||||||
if (message.type === 'RESULT') {
|
if (message.type === 'RESULT') {
|
||||||
|
|||||||
@@ -4,13 +4,13 @@ const fs = require('fs-extra')
|
|||||||
const Logger = require('./Logger')
|
const Logger = require('./Logger')
|
||||||
|
|
||||||
class HlsController {
|
class HlsController {
|
||||||
constructor(db, scanner, auth, streamManager, emitter, MetadataPath) {
|
constructor(db, scanner, auth, streamManager, emitter, StreamsPath) {
|
||||||
this.db = db
|
this.db = db
|
||||||
this.scanner = scanner
|
this.scanner = scanner
|
||||||
this.auth = auth
|
this.auth = auth
|
||||||
this.streamManager = streamManager
|
this.streamManager = streamManager
|
||||||
this.emitter = emitter
|
this.emitter = emitter
|
||||||
this.MetadataPath = MetadataPath
|
this.StreamsPath = StreamsPath
|
||||||
|
|
||||||
this.router = express()
|
this.router = express()
|
||||||
this.init()
|
this.init()
|
||||||
@@ -28,7 +28,7 @@ class HlsController {
|
|||||||
|
|
||||||
async streamFileRequest(req, res) {
|
async streamFileRequest(req, res) {
|
||||||
var streamId = req.params.stream
|
var streamId = req.params.stream
|
||||||
var fullFilePath = Path.join(this.MetadataPath, streamId, req.params.file)
|
var fullFilePath = Path.join(this.StreamsPath, streamId, req.params.file)
|
||||||
|
|
||||||
// development test stream - ignore
|
// development test stream - ignore
|
||||||
if (streamId === 'test') {
|
if (streamId === 'test') {
|
||||||
|
|||||||
+6
-1
@@ -100,7 +100,12 @@ class Scanner {
|
|||||||
hasUpdatedAudioFiles = true
|
hasUpdatedAudioFiles = true
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
newAudioFiles.push(file)
|
var audioFileWithMatchingPath = existingAudiobook.getAudioFileByPath(file.fullPath)
|
||||||
|
if (audioFileWithMatchingPath) {
|
||||||
|
Logger.warn(`[Scanner] Audio file with path already exists with different inode, New: "${file.filename}" (${file.ino}) | Existing: ${audioFileWithMatchingPath.filename} (${audioFileWithMatchingPath.ino})`)
|
||||||
|
} else {
|
||||||
|
newAudioFiles.push(file)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
if (newAudioFiles.length) {
|
if (newAudioFiles.length) {
|
||||||
|
|||||||
+54
-35
@@ -35,8 +35,8 @@ class Server {
|
|||||||
this.streamManager = new StreamManager(this.db, this.MetadataPath)
|
this.streamManager = new StreamManager(this.db, this.MetadataPath)
|
||||||
this.rssFeeds = new RssFeeds(this.Port, this.db)
|
this.rssFeeds = new RssFeeds(this.Port, this.db)
|
||||||
this.downloadManager = new DownloadManager(this.db, this.MetadataPath, this.AudiobookPath, this.emitter.bind(this))
|
this.downloadManager = new DownloadManager(this.db, this.MetadataPath, this.AudiobookPath, this.emitter.bind(this))
|
||||||
this.apiController = new ApiController(this.db, this.scanner, this.auth, this.streamManager, this.rssFeeds, this.downloadManager, this.emitter.bind(this), this.clientEmitter.bind(this))
|
this.apiController = new ApiController(this.MetadataPath, this.db, this.scanner, this.auth, this.streamManager, this.rssFeeds, this.downloadManager, this.emitter.bind(this), this.clientEmitter.bind(this))
|
||||||
this.hlsController = new HlsController(this.db, this.scanner, this.auth, this.streamManager, this.emitter.bind(this), this.MetadataPath)
|
this.hlsController = new HlsController(this.db, this.scanner, this.auth, this.streamManager, this.emitter.bind(this), this.streamManager.StreamsPath)
|
||||||
|
|
||||||
this.server = null
|
this.server = null
|
||||||
this.io = null
|
this.io = null
|
||||||
@@ -110,8 +110,10 @@ class Server {
|
|||||||
|
|
||||||
async init() {
|
async init() {
|
||||||
Logger.info('[Server] Init')
|
Logger.info('[Server] Init')
|
||||||
|
await this.streamManager.ensureStreamsDir()
|
||||||
await this.streamManager.removeOrphanStreams()
|
await this.streamManager.removeOrphanStreams()
|
||||||
await this.downloadManager.removeOrphanDownloads()
|
await this.downloadManager.removeOrphanDownloads()
|
||||||
|
|
||||||
await this.db.init()
|
await this.db.init()
|
||||||
this.auth.init()
|
this.auth.init()
|
||||||
|
|
||||||
@@ -123,9 +125,53 @@ class Server {
|
|||||||
this.auth.authMiddleware(req, res, next)
|
this.auth.authMiddleware(req, res, next)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async handleUpload(req, res) {
|
||||||
|
if (!req.user.canUpload) {
|
||||||
|
Logger.warn('User attempted to upload without permission', req.user)
|
||||||
|
return res.sendStatus(403)
|
||||||
|
}
|
||||||
|
var files = Object.values(req.files)
|
||||||
|
var title = req.body.title
|
||||||
|
var author = req.body.author
|
||||||
|
var series = req.body.series
|
||||||
|
|
||||||
|
if (!files.length || !title || !author) {
|
||||||
|
return res.json({
|
||||||
|
error: 'Invalid post data received'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
var outputDirectory = ''
|
||||||
|
if (series && series.length && series !== 'null') {
|
||||||
|
outputDirectory = Path.join(this.AudiobookPath, author, series, title)
|
||||||
|
} else {
|
||||||
|
outputDirectory = Path.join(this.AudiobookPath, author, title)
|
||||||
|
}
|
||||||
|
|
||||||
|
var exists = await fs.pathExists(outputDirectory)
|
||||||
|
if (exists) {
|
||||||
|
Logger.error(`[Server] Upload directory "${outputDirectory}" already exists`)
|
||||||
|
return res.json({
|
||||||
|
error: `Directory "${outputDirectory}" already exists`
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
await fs.ensureDir(outputDirectory)
|
||||||
|
Logger.info(`Uploading ${files.length} files to`, outputDirectory)
|
||||||
|
|
||||||
|
for (let i = 0; i < files.length; i++) {
|
||||||
|
var file = files[i]
|
||||||
|
|
||||||
|
var path = Path.join(outputDirectory, file.name)
|
||||||
|
await file.mv(path).catch((error) => {
|
||||||
|
Logger.error('Failed to move file', path, error)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
res.sendStatus(200)
|
||||||
|
}
|
||||||
|
|
||||||
async start() {
|
async start() {
|
||||||
Logger.info('=== Starting Server ===')
|
Logger.info('=== Starting Server ===')
|
||||||
|
|
||||||
await this.init()
|
await this.init()
|
||||||
|
|
||||||
const app = express()
|
const app = express()
|
||||||
@@ -144,6 +190,8 @@ class Server {
|
|||||||
app.use(express.static(this.AudiobookPath))
|
app.use(express.static(this.AudiobookPath))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
app.use('/metadata', this.authMiddleware.bind(this), express.static(this.MetadataPath))
|
||||||
|
|
||||||
app.use(express.static(this.MetadataPath))
|
app.use(express.static(this.MetadataPath))
|
||||||
app.use(express.static(Path.join(global.appRoot, 'static')))
|
app.use(express.static(Path.join(global.appRoot, 'static')))
|
||||||
app.use(express.urlencoded({ extended: true }));
|
app.use(express.urlencoded({ extended: true }));
|
||||||
@@ -151,44 +199,15 @@ class Server {
|
|||||||
|
|
||||||
// Dynamic routes are not generated on client
|
// Dynamic routes are not generated on client
|
||||||
app.get('/audiobook/:id', (req, res) => res.sendFile(Path.join(distPath, 'index.html')))
|
app.get('/audiobook/:id', (req, res) => res.sendFile(Path.join(distPath, 'index.html')))
|
||||||
|
app.get('/library/:id', (req, res) => res.sendFile(Path.join(distPath, 'index.html')))
|
||||||
|
app.get('/library', (req, res) => res.sendFile(Path.join(distPath, 'index.html')))
|
||||||
|
|
||||||
app.use('/api', this.authMiddleware.bind(this), this.apiController.router)
|
app.use('/api', this.authMiddleware.bind(this), this.apiController.router)
|
||||||
app.use('/hls', this.authMiddleware.bind(this), this.hlsController.router)
|
app.use('/hls', this.authMiddleware.bind(this), this.hlsController.router)
|
||||||
// app.use('/hls', this.hlsController.router)
|
// app.use('/hls', this.hlsController.router)
|
||||||
app.use('/feeds', this.rssFeeds.router)
|
app.use('/feeds', this.rssFeeds.router)
|
||||||
|
|
||||||
app.post('/upload', this.authMiddleware.bind(this), async (req, res) => {
|
app.post('/upload', this.authMiddleware.bind(this), this.handleUpload.bind(this))
|
||||||
var files = Object.values(req.files)
|
|
||||||
var title = req.body.title
|
|
||||||
var author = req.body.author
|
|
||||||
var series = req.body.series
|
|
||||||
|
|
||||||
if (!files.length || !title || !author) {
|
|
||||||
return res.json({
|
|
||||||
error: 'Invalid post data received'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
var outputDirectory = ''
|
|
||||||
if (series && series.length && series !== 'null') {
|
|
||||||
outputDirectory = Path.join(this.AudiobookPath, author, series, title)
|
|
||||||
} else {
|
|
||||||
outputDirectory = Path.join(this.AudiobookPath, author, title)
|
|
||||||
}
|
|
||||||
|
|
||||||
await fs.ensureDir(outputDirectory)
|
|
||||||
Logger.info(`Uploading ${files.length} files to`, outputDirectory)
|
|
||||||
|
|
||||||
for (let i = 0; i < files.length; i++) {
|
|
||||||
var file = files[i]
|
|
||||||
|
|
||||||
var path = Path.join(outputDirectory, file.name)
|
|
||||||
await file.mv(path).catch((error) => {
|
|
||||||
Logger.error('Failed to move file', path, error)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
res.sendStatus(200)
|
|
||||||
})
|
|
||||||
|
|
||||||
app.post('/login', (req, res) => this.auth.login(req, res))
|
app.post('/login', (req, res) => this.auth.login(req, res))
|
||||||
app.post('/logout', this.logout.bind(this))
|
app.post('/logout', this.logout.bind(this))
|
||||||
|
|||||||
+47
-21
@@ -1,15 +1,16 @@
|
|||||||
const Stream = require('./objects/Stream')
|
const Stream = require('./objects/Stream')
|
||||||
const StreamTest = require('./test/StreamTest')
|
// const StreamTest = require('./test/StreamTest')
|
||||||
const Logger = require('./Logger')
|
const Logger = require('./Logger')
|
||||||
const fs = require('fs-extra')
|
const fs = require('fs-extra')
|
||||||
const Path = require('path')
|
const Path = require('path')
|
||||||
|
|
||||||
class StreamManager {
|
class StreamManager {
|
||||||
constructor(db, STREAM_PATH) {
|
constructor(db, MetadataPath) {
|
||||||
this.db = db
|
this.db = db
|
||||||
|
|
||||||
|
this.MetadataPath = MetadataPath
|
||||||
this.streams = []
|
this.streams = []
|
||||||
this.streamPath = STREAM_PATH
|
this.StreamsPath = Path.join(this.MetadataPath, 'streams')
|
||||||
}
|
}
|
||||||
|
|
||||||
get audiobooks() {
|
get audiobooks() {
|
||||||
@@ -25,7 +26,7 @@ class StreamManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async openStream(client, audiobook) {
|
async openStream(client, audiobook) {
|
||||||
var stream = new Stream(this.streamPath, client, audiobook)
|
var stream = new Stream(this.StreamsPath, client, audiobook)
|
||||||
|
|
||||||
stream.on('closed', () => {
|
stream.on('closed', () => {
|
||||||
this.removeStream(stream)
|
this.removeStream(stream)
|
||||||
@@ -44,29 +45,54 @@ class StreamManager {
|
|||||||
return stream
|
return stream
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ensureStreamsDir() {
|
||||||
|
return fs.ensureDir(this.StreamsPath)
|
||||||
|
}
|
||||||
|
|
||||||
removeOrphanStreamFiles(streamId) {
|
removeOrphanStreamFiles(streamId) {
|
||||||
try {
|
try {
|
||||||
var streamPath = Path.join(this.streamPath, streamId)
|
var StreamsPath = Path.join(this.StreamsPath, streamId)
|
||||||
return fs.remove(streamPath)
|
return fs.remove(StreamsPath)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
Logger.debug('No orphan stream', streamId)
|
Logger.debug('No orphan stream', streamId)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async removeOrphanStreams() {
|
async tempCheckStrayStreams() {
|
||||||
try {
|
try {
|
||||||
var dirs = await fs.readdir(this.streamPath)
|
var dirs = await fs.readdir(this.MetadataPath)
|
||||||
if (!dirs || !dirs.length) return true
|
if (!dirs || !dirs.length) return true
|
||||||
|
|
||||||
await Promise.all(dirs.map(async (dirname) => {
|
await Promise.all(dirs.map(async (dirname) => {
|
||||||
var fullPath = Path.join(this.streamPath, dirname)
|
if (dirname !== 'streams' && dirname !== 'books' && dirname !== 'downloads') {
|
||||||
|
var fullPath = Path.join(this.MetadataPath, dirname)
|
||||||
|
Logger.warn(`Removing OLD Orphan Stream ${dirname}`)
|
||||||
|
return fs.remove(fullPath)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
|
||||||
|
return true
|
||||||
|
} catch (error) {
|
||||||
|
Logger.debug('No old orphan streams', error)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async removeOrphanStreams() {
|
||||||
|
await this.tempCheckStrayStreams()
|
||||||
|
try {
|
||||||
|
var dirs = await fs.readdir(this.StreamsPath)
|
||||||
|
if (!dirs || !dirs.length) return true
|
||||||
|
|
||||||
|
await Promise.all(dirs.map(async (dirname) => {
|
||||||
|
var fullPath = Path.join(this.StreamsPath, dirname)
|
||||||
Logger.info(`Removing Orphan Stream ${dirname}`)
|
Logger.info(`Removing Orphan Stream ${dirname}`)
|
||||||
return fs.remove(fullPath)
|
return fs.remove(fullPath)
|
||||||
}))
|
}))
|
||||||
return true
|
return true
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
Logger.debug('No orphan stream', streamId)
|
Logger.debug('No orphan stream', error)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -102,21 +128,21 @@ class StreamManager {
|
|||||||
this.db.updateUserStream(client.user.id, null)
|
this.db.updateUserStream(client.user.id, null)
|
||||||
}
|
}
|
||||||
|
|
||||||
async openTestStream(streamPath, audiobookId) {
|
async openTestStream(StreamsPath, audiobookId) {
|
||||||
Logger.info('Open Stream Test Request', audiobookId)
|
Logger.info('Open Stream Test Request', audiobookId)
|
||||||
var audiobook = this.audiobooks.find(ab => ab.id === audiobookId)
|
// var audiobook = this.audiobooks.find(ab => ab.id === audiobookId)
|
||||||
var stream = new StreamTest(streamPath, audiobook)
|
// var stream = new StreamTest(StreamsPath, audiobook)
|
||||||
|
|
||||||
stream.on('closed', () => {
|
// stream.on('closed', () => {
|
||||||
console.log('Stream closed')
|
// console.log('Stream closed')
|
||||||
})
|
// })
|
||||||
|
|
||||||
var playlistUri = await stream.generatePlaylist()
|
// var playlistUri = await stream.generatePlaylist()
|
||||||
stream.start()
|
// stream.start()
|
||||||
|
|
||||||
Logger.info('Stream Playlist', playlistUri)
|
// Logger.info('Stream Playlist', playlistUri)
|
||||||
Logger.info('Test Stream Opened for audiobook', audiobook.title, 'with streamId', stream.id)
|
// Logger.info('Test Stream Opened for audiobook', audiobook.title, 'with streamId', stream.id)
|
||||||
return playlistUri
|
// return playlistUri
|
||||||
}
|
}
|
||||||
|
|
||||||
streamUpdate(socket, { currentTime, streamId }) {
|
streamUpdate(socket, { currentTime, streamId }) {
|
||||||
|
|||||||
@@ -288,6 +288,12 @@ class Audiobook {
|
|||||||
return hasUpdates
|
return hasUpdates
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Cover Url may be the same, this ensures the lastUpdate is updated
|
||||||
|
updateBookCover(cover) {
|
||||||
|
if (!this.book) return false
|
||||||
|
return this.book.updateCover(cover)
|
||||||
|
}
|
||||||
|
|
||||||
updateAudioTracks(orderedFileData) {
|
updateAudioTracks(orderedFileData) {
|
||||||
var index = 1
|
var index = 1
|
||||||
this.audioFiles = orderedFileData.map((fileData) => {
|
this.audioFiles = orderedFileData.map((fileData) => {
|
||||||
@@ -419,6 +425,10 @@ class Audiobook {
|
|||||||
return this.audioFiles.find(af => af.ino === ino)
|
return this.audioFiles.find(af => af.ino === ino)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getAudioFileByPath(fullPath) {
|
||||||
|
return this.audioFiles.find(af => af.fullPath === fullPath)
|
||||||
|
}
|
||||||
|
|
||||||
setChapters() {
|
setChapters() {
|
||||||
// If 1 audio file without chapters, then no chapters will be set
|
// If 1 audio file without chapters, then no chapters will be set
|
||||||
|
|
||||||
|
|||||||
+21
-2
@@ -18,6 +18,7 @@ class Book {
|
|||||||
this.description = null
|
this.description = null
|
||||||
this.cover = null
|
this.cover = null
|
||||||
this.genres = []
|
this.genres = []
|
||||||
|
this.lastUpdate = null
|
||||||
|
|
||||||
if (book) {
|
if (book) {
|
||||||
this.construct(book)
|
this.construct(book)
|
||||||
@@ -45,6 +46,7 @@ class Book {
|
|||||||
this.description = book.description
|
this.description = book.description
|
||||||
this.cover = book.cover
|
this.cover = book.cover
|
||||||
this.genres = book.genres
|
this.genres = book.genres
|
||||||
|
this.lastUpdate = book.lastUpdate || Date.now()
|
||||||
}
|
}
|
||||||
|
|
||||||
toJSON() {
|
toJSON() {
|
||||||
@@ -62,7 +64,8 @@ class Book {
|
|||||||
publisher: this.publisher,
|
publisher: this.publisher,
|
||||||
description: this.description,
|
description: this.description,
|
||||||
cover: this.cover,
|
cover: this.cover,
|
||||||
genres: this.genres
|
genres: this.genres,
|
||||||
|
lastUpdate: this.lastUpdate
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -97,6 +100,7 @@ class Book {
|
|||||||
this.description = data.description || null
|
this.description = data.description || null
|
||||||
this.cover = data.cover || null
|
this.cover = data.cover || null
|
||||||
this.genres = data.genres || []
|
this.genres = data.genres || []
|
||||||
|
this.lastUpdate = Date.now()
|
||||||
|
|
||||||
if (data.author) {
|
if (data.author) {
|
||||||
this.setParseAuthor(this.author)
|
this.setParseAuthor(this.author)
|
||||||
@@ -145,13 +149,28 @@ class Book {
|
|||||||
hasUpdates = true
|
hasUpdates = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (hasUpdates) {
|
||||||
|
this.lastUpdate = Date.now()
|
||||||
|
}
|
||||||
|
|
||||||
return hasUpdates
|
return hasUpdates
|
||||||
}
|
}
|
||||||
|
|
||||||
|
updateCover(cover) {
|
||||||
|
if (!cover) return false
|
||||||
|
if (!cover.startsWith('http:') && !cover.startsWith('https:')) {
|
||||||
|
cover = Path.normalize(cover)
|
||||||
|
}
|
||||||
|
this.cover = cover
|
||||||
|
this.lastUpdate = Date.now()
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
// If audiobook directory path was changed, check and update properties set from dirnames
|
// If audiobook directory path was changed, check and update properties set from dirnames
|
||||||
// May be worthwhile checking if these were manually updated and not override manual updates
|
// May be worthwhile checking if these were manually updated and not override manual updates
|
||||||
syncPathsUpdated(audiobookData) {
|
syncPathsUpdated(audiobookData) {
|
||||||
var keysToSync = ['author', 'title', 'series', 'publishYear']
|
var keysToSync = ['author', 'title', 'series', 'publishYear', 'volumeNumber']
|
||||||
var syncPayload = {}
|
var syncPayload = {}
|
||||||
keysToSync.forEach((key) => {
|
keysToSync.forEach((key) => {
|
||||||
if (audiobookData[key]) syncPayload[key] = audiobookData[key]
|
if (audiobookData[key]) syncPayload[key] = audiobookData[key]
|
||||||
|
|||||||
@@ -1,9 +1,12 @@
|
|||||||
|
const { CoverDestination } = require('../utils/constants')
|
||||||
|
|
||||||
class ServerSettings {
|
class ServerSettings {
|
||||||
constructor(settings) {
|
constructor(settings) {
|
||||||
this.id = 'server-settings'
|
this.id = 'server-settings'
|
||||||
this.autoTagNew = false
|
this.autoTagNew = false
|
||||||
this.newTagExpireDays = 15
|
this.newTagExpireDays = 15
|
||||||
this.scannerParseSubtitle = false
|
this.scannerParseSubtitle = false
|
||||||
|
this.coverDestination = CoverDestination.METADATA
|
||||||
|
|
||||||
if (settings) {
|
if (settings) {
|
||||||
this.construct(settings)
|
this.construct(settings)
|
||||||
@@ -14,6 +17,7 @@ class ServerSettings {
|
|||||||
this.autoTagNew = settings.autoTagNew
|
this.autoTagNew = settings.autoTagNew
|
||||||
this.newTagExpireDays = settings.newTagExpireDays
|
this.newTagExpireDays = settings.newTagExpireDays
|
||||||
this.scannerParseSubtitle = settings.scannerParseSubtitle
|
this.scannerParseSubtitle = settings.scannerParseSubtitle
|
||||||
|
this.coverDestination = settings.coverDestination || CoverDestination.METADATA
|
||||||
}
|
}
|
||||||
|
|
||||||
toJSON() {
|
toJSON() {
|
||||||
@@ -21,7 +25,8 @@ class ServerSettings {
|
|||||||
id: this.id,
|
id: this.id,
|
||||||
autoTagNew: this.autoTagNew,
|
autoTagNew: this.autoTagNew,
|
||||||
newTagExpireDays: this.newTagExpireDays,
|
newTagExpireDays: this.newTagExpireDays,
|
||||||
scannerParseSubtitle: this.scannerParseSubtitle
|
scannerParseSubtitle: this.scannerParseSubtitle,
|
||||||
|
coverDestination: this.coverDestination
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -171,13 +171,11 @@ class Stream extends EventEmitter {
|
|||||||
this.furthestSegmentCreated = lastSegment
|
this.furthestSegmentCreated = lastSegment
|
||||||
}
|
}
|
||||||
|
|
||||||
// console.log('SORT', [...this.segmentsCreated].slice(0, 200).join(', '), segments.slice(0, 200).join(', '))
|
|
||||||
segments.forEach((seg) => {
|
segments.forEach((seg) => {
|
||||||
if (!current_chunk.length || last_seg_in_chunk + 1 === seg) {
|
if (!current_chunk.length || last_seg_in_chunk + 1 === seg) {
|
||||||
last_seg_in_chunk = seg
|
last_seg_in_chunk = seg
|
||||||
current_chunk.push(seg)
|
current_chunk.push(seg)
|
||||||
} else {
|
} else {
|
||||||
// console.log('Last Seg is not equal to - 1', last_seg_in_chunk, seg)
|
|
||||||
if (current_chunk.length === 1) chunks.push(current_chunk[0])
|
if (current_chunk.length === 1) chunks.push(current_chunk[0])
|
||||||
else chunks.push(`${current_chunk[0]}-${current_chunk[current_chunk.length - 1]}`)
|
else chunks.push(`${current_chunk[0]}-${current_chunk[current_chunk.length - 1]}`)
|
||||||
last_seg_in_chunk = seg
|
last_seg_in_chunk = seg
|
||||||
@@ -191,7 +189,7 @@ class Stream extends EventEmitter {
|
|||||||
|
|
||||||
var perc = (this.segmentsCreated.size * 100 / this.numSegments).toFixed(2) + '%'
|
var perc = (this.segmentsCreated.size * 100 / this.numSegments).toFixed(2) + '%'
|
||||||
Logger.info('[STREAM-CHECK] Check Files', this.segmentsCreated.size, 'of', this.numSegments, perc, `Furthest Segment: ${this.furthestSegmentCreated}`)
|
Logger.info('[STREAM-CHECK] Check Files', this.segmentsCreated.size, 'of', this.numSegments, perc, `Furthest Segment: ${this.furthestSegmentCreated}`)
|
||||||
Logger.info('[STREAM-CHECK] Chunks', chunks.join(', '))
|
Logger.debug('[STREAM-CHECK] Chunks', chunks.join(', '))
|
||||||
|
|
||||||
this.socket.emit('stream_progress', {
|
this.socket.emit('stream_progress', {
|
||||||
stream: this.id,
|
stream: this.id,
|
||||||
@@ -200,7 +198,7 @@ class Stream extends EventEmitter {
|
|||||||
numSegments: this.numSegments
|
numSegments: this.numSegments
|
||||||
})
|
})
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
Logger.error('Failed checkign files', error)
|
Logger.error('Failed checking files', error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -32,6 +32,9 @@ class User {
|
|||||||
get canDownload() {
|
get canDownload() {
|
||||||
return !!this.permissions.download && this.isActive
|
return !!this.permissions.download && this.isActive
|
||||||
}
|
}
|
||||||
|
get canUpload() {
|
||||||
|
return !!this.permissions.upload && this.isActive
|
||||||
|
}
|
||||||
|
|
||||||
getDefaultUserSettings() {
|
getDefaultUserSettings() {
|
||||||
return {
|
return {
|
||||||
@@ -47,7 +50,8 @@ class User {
|
|||||||
return {
|
return {
|
||||||
download: true,
|
download: true,
|
||||||
update: true,
|
update: true,
|
||||||
delete: this.id === 'root'
|
delete: this.type === 'root',
|
||||||
|
upload: this.type === 'root' || this.type === 'admin'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -112,6 +116,8 @@ class User {
|
|||||||
this.createdAt = user.createdAt || Date.now()
|
this.createdAt = user.createdAt || Date.now()
|
||||||
this.settings = user.settings || this.getDefaultUserSettings()
|
this.settings = user.settings || this.getDefaultUserSettings()
|
||||||
this.permissions = user.permissions || this.getDefaultUserPermissions()
|
this.permissions = user.permissions || this.getDefaultUserPermissions()
|
||||||
|
// Upload permission added v1.1.13, make sure root user has upload permissions
|
||||||
|
if (this.type === 'root' && !this.permissions.upload) this.permissions.upload = true
|
||||||
}
|
}
|
||||||
|
|
||||||
update(payload) {
|
update(payload) {
|
||||||
|
|||||||
@@ -5,3 +5,8 @@ module.exports.ScanResult = {
|
|||||||
REMOVED: 3,
|
REMOVED: 3,
|
||||||
UPTODATE: 4
|
UPTODATE: 4
|
||||||
}
|
}
|
||||||
|
|
||||||
|
module.exports.CoverDestination = {
|
||||||
|
METADATA: 0,
|
||||||
|
AUDIOBOOK: 1
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
const Ffmpeg = require('fluent-ffmpeg')
|
const Ffmpeg = require('fluent-ffmpeg')
|
||||||
|
|
||||||
if (process.env.NODE_ENV !== 'production') {
|
if (process.env.FFMPEG_PATH) {
|
||||||
Ffmpeg.setFfmpegPath(process.env.FFMPEG_PATH)
|
Ffmpeg.setFfmpegPath(process.env.FFMPEG_PATH)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -63,3 +63,7 @@ module.exports.getIno = (path) => {
|
|||||||
return null
|
return null
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
module.exports.isAcceptableCoverMimeType = (mimeType) => {
|
||||||
|
return mimeType && mimeType.startsWith('image/')
|
||||||
|
}
|
||||||
+33
-1
@@ -155,10 +155,39 @@ function getAudiobookDataFromDir(abRootPath, dir, parseSubtitle = false) {
|
|||||||
// If there are at least 2 more directories, next furthest will be the series
|
// If there are at least 2 more directories, next furthest will be the series
|
||||||
if (splitDir.length > 1) series = splitDir.pop()
|
if (splitDir.length > 1) series = splitDir.pop()
|
||||||
if (splitDir.length > 0) author = splitDir.pop()
|
if (splitDir.length > 0) author = splitDir.pop()
|
||||||
|
|
||||||
// There could be many more directories, but only the top 3 are used for naming /author/series/title/
|
// There could be many more directories, but only the top 3 are used for naming /author/series/title/
|
||||||
|
|
||||||
|
|
||||||
|
// If in a series directory check for volume number match
|
||||||
|
/* ACCEPTS:
|
||||||
|
Book 2 - Title Here - Subtitle Here
|
||||||
|
Title Here - Subtitle Here - Vol 12
|
||||||
|
Title Here - volume 9 - Subtitle Here
|
||||||
|
Vol. 3 Title Here - Subtitle Here
|
||||||
|
1980 - Book 2-Title Here
|
||||||
|
Title Here-Volume 999-Subtitle Here
|
||||||
|
*/
|
||||||
|
var volumeNumber = null
|
||||||
|
if (series) {
|
||||||
|
var volumeMatch = title.match(/(-? ?)\b((?:Book|Vol.?|Volume) (\d{1,3}))\b( ?-?)/i)
|
||||||
|
if (volumeMatch && volumeMatch.length > 3 && volumeMatch[2] && volumeMatch[3]) {
|
||||||
|
volumeNumber = volumeMatch[3]
|
||||||
|
var replaceChunk = volumeMatch[2]
|
||||||
|
|
||||||
|
// "1980 - Book 2-Title Here"
|
||||||
|
// Group 1 would be "- "
|
||||||
|
// Group 3 would be "-"
|
||||||
|
// Only remove the first group
|
||||||
|
if (volumeMatch[1]) {
|
||||||
|
replaceChunk = volumeMatch[1] + replaceChunk
|
||||||
|
} else if (volumeMatch[4]) {
|
||||||
|
replaceChunk += volumeMatch[4]
|
||||||
|
}
|
||||||
|
title = title.replace(replaceChunk, '').trim()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
var publishYear = null
|
var publishYear = null
|
||||||
// If Title is of format 1999 - Title, then use 1999 as publish year
|
// If Title is of format 1999 - Title, then use 1999 as publish year
|
||||||
var publishYearMatch = title.match(/^([0-9]{4}) - (.+)/)
|
var publishYearMatch = title.match(/^([0-9]{4}) - (.+)/)
|
||||||
@@ -169,7 +198,9 @@ function getAudiobookDataFromDir(abRootPath, dir, parseSubtitle = false) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Subtitle can be parsed from the title if user enabled
|
// Subtitle can be parsed from the title if user enabled
|
||||||
|
// Subtitle is everything after " - "
|
||||||
var subtitle = null
|
var subtitle = null
|
||||||
if (parseSubtitle && title.includes(' - ')) {
|
if (parseSubtitle && title.includes(' - ')) {
|
||||||
var splitOnSubtitle = title.split(' - ')
|
var splitOnSubtitle = title.split(' - ')
|
||||||
@@ -182,6 +213,7 @@ function getAudiobookDataFromDir(abRootPath, dir, parseSubtitle = false) {
|
|||||||
title,
|
title,
|
||||||
subtitle,
|
subtitle,
|
||||||
series,
|
series,
|
||||||
|
volumeNumber,
|
||||||
publishYear,
|
publishYear,
|
||||||
path: dir, // relative audiobook path i.e. /Author Name/Book Name/..
|
path: dir, // relative audiobook path i.e. /Author Name/Book Name/..
|
||||||
fullPath: Path.join(abRootPath, dir) // i.e. /audiobook/Author Name/Book Name/..
|
fullPath: Path.join(abRootPath, dir) // i.e. /audiobook/Author Name/Book Name/..
|
||||||
|
|||||||
Reference in New Issue
Block a user