mirror of
https://github.com/owntone/owntone-server.git
synced 2025-04-04 03:40:36 -04:00
[web] Add breadcrumb to navigate through the folders
This commit is contained in:
parent
263a197da4
commit
25e005ff32
@ -1,14 +1,18 @@
|
|||||||
<template>
|
<template>
|
||||||
<div
|
<div v-if="$route.query.directory" class="media is-align-items-center">
|
||||||
v-if="$route.query.directory"
|
<figure class="media-left is-clickable" @click="open_parent">
|
||||||
class="media is-align-items-center"
|
<mdicon class="icon" name="chevron-left" size="16" />
|
||||||
@click="open_parent()"
|
|
||||||
>
|
|
||||||
<figure class="media-left is-clickable">
|
|
||||||
<mdicon class="icon" name="subdirectory-arrow-left" size="16" />
|
|
||||||
</figure>
|
</figure>
|
||||||
<div class="media-content is-clickable is-clipped">
|
<div class="media-content is-clipped">
|
||||||
<h1 class="title is-6">..</h1>
|
<nav class="breadcrumb">
|
||||||
|
<ul>
|
||||||
|
<li v-for="directory in directories" :key="directory.index">
|
||||||
|
<a @click="open(directory)">
|
||||||
|
<span v-text="directory.name" />
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</nav>
|
||||||
</div>
|
</div>
|
||||||
<div class="media-right">
|
<div class="media-right">
|
||||||
<slot name="actions" />
|
<slot name="actions" />
|
||||||
@ -20,11 +24,7 @@
|
|||||||
<mdicon class="icon" name="folder" size="16" />
|
<mdicon class="icon" name="folder" size="16" />
|
||||||
</figure>
|
</figure>
|
||||||
<div class="media-content is-clickable is-clipped">
|
<div class="media-content is-clickable is-clipped">
|
||||||
<h1
|
<h1 class="title is-6" v-text="item.name" />
|
||||||
class="title is-6"
|
|
||||||
v-text="item.path.substring(item.path.lastIndexOf('/') + 1)"
|
|
||||||
/>
|
|
||||||
<h2 class="subtitle is-7 has-text-grey" v-text="item.path" />
|
|
||||||
</div>
|
</div>
|
||||||
<div class="media-right">
|
<div class="media-right">
|
||||||
<a @click.prevent.stop="open_dialog(item)">
|
<a @click.prevent.stop="open_dialog(item)">
|
||||||
@ -58,37 +58,34 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
current() {
|
directories() {
|
||||||
return this.$route.query?.directory || '/'
|
const directories = []
|
||||||
|
let path = ''
|
||||||
|
this.$route.query?.directory
|
||||||
|
.split('/')
|
||||||
|
.slice(1, -1)
|
||||||
|
.forEach((name, index) => {
|
||||||
|
path = `${path}/${name}`
|
||||||
|
directories.push({ index, name, path })
|
||||||
|
})
|
||||||
|
return directories
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
open(item) {
|
open(item) {
|
||||||
this.$router.push({
|
const route = { name: 'files' }
|
||||||
name: 'files',
|
if (item.index !== 0) {
|
||||||
query: { directory: item.path }
|
route.query = { directory: item.path }
|
||||||
})
|
}
|
||||||
|
this.$router.push(route)
|
||||||
},
|
},
|
||||||
open_dialog(item) {
|
open_dialog(item) {
|
||||||
this.selected_item = item.path
|
this.selected_item = item.path
|
||||||
this.show_details_modal = true
|
this.show_details_modal = true
|
||||||
},
|
},
|
||||||
open_parent() {
|
open_parent() {
|
||||||
const parent = this.current.slice(0, this.current.lastIndexOf('/'))
|
this.open(this.directories.slice(-1).pop())
|
||||||
if (
|
|
||||||
parent === '' ||
|
|
||||||
this.$store.state.config.directories.includes(this.current)
|
|
||||||
) {
|
|
||||||
this.$router.push({ name: 'files' })
|
|
||||||
} else {
|
|
||||||
this.$router.push({
|
|
||||||
name: 'files',
|
|
||||||
query: {
|
|
||||||
directory: this.current.slice(0, this.current.lastIndexOf('/'))
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -12,6 +12,7 @@ import {
|
|||||||
mdiCellphone,
|
mdiCellphone,
|
||||||
mdiCheck,
|
mdiCheck,
|
||||||
mdiChevronDown,
|
mdiChevronDown,
|
||||||
|
mdiChevronLeft,
|
||||||
mdiChevronUp,
|
mdiChevronUp,
|
||||||
mdiContentSave,
|
mdiContentSave,
|
||||||
mdiDelete,
|
mdiDelete,
|
||||||
@ -57,7 +58,6 @@ import {
|
|||||||
mdiSpeaker,
|
mdiSpeaker,
|
||||||
mdiSpotify,
|
mdiSpotify,
|
||||||
mdiStop,
|
mdiStop,
|
||||||
mdiSubdirectoryArrowLeft,
|
|
||||||
mdiVolumeHigh,
|
mdiVolumeHigh,
|
||||||
mdiVolumeOff,
|
mdiVolumeOff,
|
||||||
mdiWeb
|
mdiWeb
|
||||||
@ -77,6 +77,7 @@ export const icons = {
|
|||||||
mdiCellphone,
|
mdiCellphone,
|
||||||
mdiCheck,
|
mdiCheck,
|
||||||
mdiChevronDown,
|
mdiChevronDown,
|
||||||
|
mdiChevronLeft,
|
||||||
mdiChevronUp,
|
mdiChevronUp,
|
||||||
mdiContentSave,
|
mdiContentSave,
|
||||||
mdiDelete,
|
mdiDelete,
|
||||||
@ -122,7 +123,6 @@ export const icons = {
|
|||||||
mdiSpeaker,
|
mdiSpeaker,
|
||||||
mdiSpotify,
|
mdiSpotify,
|
||||||
mdiStop,
|
mdiStop,
|
||||||
mdiSubdirectoryArrowLeft,
|
|
||||||
mdiVolumeHigh,
|
mdiVolumeHigh,
|
||||||
mdiVolumeOff,
|
mdiVolumeOff,
|
||||||
mdiWeb
|
mdiWeb
|
||||||
|
@ -29,6 +29,7 @@
|
|||||||
.tabs a {
|
.tabs a {
|
||||||
border-bottom-color: $grey-dark;
|
border-bottom-color: $grey-dark;
|
||||||
}
|
}
|
||||||
|
.breadcrumb a:hover,
|
||||||
.tabs a,
|
.tabs a,
|
||||||
.hero.is-light .title,
|
.hero.is-light .title,
|
||||||
.title,
|
.title,
|
||||||
|
@ -2,8 +2,7 @@
|
|||||||
<div>
|
<div>
|
||||||
<content-with-heading>
|
<content-with-heading>
|
||||||
<template #heading-left>
|
<template #heading-left>
|
||||||
<p class="title is-4" v-text="$t('page.files.title')" />
|
<p class="title is-4" v-text="name" />
|
||||||
<p class="title is-7 has-text-grey" v-text="current_directory" />
|
|
||||||
</template>
|
</template>
|
||||||
<template #heading-right>
|
<template #heading-right>
|
||||||
<div class="buttons is-centered">
|
<div class="buttons is-centered">
|
||||||
@ -20,7 +19,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<template #content>
|
<template #content>
|
||||||
<list-directories :items="dirs" />
|
<list-directories :items="directories" />
|
||||||
<list-playlists :items="playlists" />
|
<list-playlists :items="playlists" />
|
||||||
<list-tracks
|
<list-tracks
|
||||||
:expression="play_expression"
|
:expression="play_expression"
|
||||||
@ -28,7 +27,7 @@
|
|||||||
:show_icon="true"
|
:show_icon="true"
|
||||||
/>
|
/>
|
||||||
<modal-dialog-directory
|
<modal-dialog-directory
|
||||||
:item="current_directory"
|
:item="current"
|
||||||
:show="show_details_modal"
|
:show="show_details_modal"
|
||||||
@close="show_details_modal = false"
|
@close="show_details_modal = false"
|
||||||
/>
|
/>
|
||||||
@ -53,25 +52,24 @@ const dataObject = {
|
|||||||
}
|
}
|
||||||
return Promise.resolve()
|
return Promise.resolve()
|
||||||
},
|
},
|
||||||
|
|
||||||
set(vm, response) {
|
set(vm, response) {
|
||||||
if (response) {
|
if (response) {
|
||||||
vm.dirs = response.data.directories
|
vm.directories = response.data.directories.map((directory) =>
|
||||||
vm.playlists = new GroupedList(response.data.playlists)
|
vm.transform(directory.path)
|
||||||
vm.tracks = new GroupedList(response.data.tracks)
|
)
|
||||||
|
} else if (vm.$store.state.config.directories) {
|
||||||
|
vm.directories = vm.$store.state.config.directories.map((path) =>
|
||||||
|
vm.transform(path)
|
||||||
|
)
|
||||||
} else {
|
} else {
|
||||||
if (vm.$store.state.config.directories) {
|
webapi.config().then((config) => {
|
||||||
vm.dirs = vm.$store.state.config.directories.map((dir) => ({
|
vm.directories = config.data.directories.map((path) =>
|
||||||
path: dir
|
vm.transform(path)
|
||||||
}))
|
)
|
||||||
} else {
|
})
|
||||||
webapi.config().then((config) => {
|
|
||||||
vm.dirs = config.data.directories.map((dir) => ({ path: dir }))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
vm.playlists = new GroupedList()
|
|
||||||
vm.tracks = new GroupedList()
|
|
||||||
}
|
}
|
||||||
|
vm.playlists = new GroupedList(response?.data.playlists)
|
||||||
|
vm.tracks = new GroupedList(response?.data.tracks)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -100,7 +98,7 @@ export default {
|
|||||||
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
dirs: [],
|
directories: [],
|
||||||
playlists: new GroupedList(),
|
playlists: new GroupedList(),
|
||||||
show_details_modal: false,
|
show_details_modal: false,
|
||||||
tracks: new GroupedList()
|
tracks: new GroupedList()
|
||||||
@ -108,20 +106,26 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
current_directory() {
|
current() {
|
||||||
if (this.$route.query && this.$route.query.directory) {
|
return this.$route.query?.directory || '/'
|
||||||
return this.$route.query.directory
|
},
|
||||||
|
name() {
|
||||||
|
if (this.current !== '/') {
|
||||||
|
return this.current?.slice(this.current.lastIndexOf('/') + 1)
|
||||||
}
|
}
|
||||||
return '/'
|
return this.$t('page.files.title')
|
||||||
},
|
},
|
||||||
play_expression() {
|
play_expression() {
|
||||||
return `path starts with "${this.current_directory}" order by path asc`
|
return `path starts with "${this.current}" order by path asc`
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
play() {
|
play() {
|
||||||
webapi.player_play_expression(this.play_expression, false)
|
webapi.player_play_expression(this.play_expression, false)
|
||||||
|
},
|
||||||
|
transform(path) {
|
||||||
|
return { path, name: path.slice(path.lastIndexOf('/') + 1) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user