|
|
|
<template lang='pug'>
|
|
|
|
v-app-bar.nav-header(color='black', dark, app, :clipped-left='!$vuetify.rtl', :clipped-right='$vuetify.rtl', fixed, flat, :extended='searchIsShown && $vuetify.breakpoint.smAndDown')
|
|
|
|
v-toolbar(color='deep-purple', flat, slot='extension', v-if='searchIsShown && $vuetify.breakpoint.smAndDown')
|
|
|
|
v-text-field(
|
|
|
|
ref='searchFieldMobile'
|
|
|
|
v-model='search'
|
|
|
|
clearable
|
|
|
|
background-color='deep-purple'
|
|
|
|
color='white'
|
|
|
|
:label='$t(`common:header.search`)'
|
|
|
|
single-line
|
|
|
|
solo
|
|
|
|
flat
|
|
|
|
hide-details
|
|
|
|
prepend-inner-icon='mdi-magnify'
|
|
|
|
:loading='searchIsLoading'
|
|
|
|
@keyup.enter='searchEnter'
|
|
|
|
)
|
|
|
|
v-layout(row)
|
|
|
|
v-flex(xs6, md4)
|
|
|
|
v-toolbar.nav-header-inner.pl-3(color='black', dark, flat)
|
|
|
|
v-menu(open-on-hover, offset-y, bottom, left, min-width='250', transition='slide-y-transition')
|
|
|
|
template(v-slot:activator='{ on }')
|
|
|
|
v-app-bar-nav-icon.btn-animate-app(v-on='on', :class='$vuetify.rtl ? `mx-0` : ``')
|
|
|
|
v-icon mdi-menu
|
|
|
|
v-list(nav, :light='!$vuetify.theme.dark', :dark='$vuetify.theme.dark', :class='$vuetify.theme.dark ? `grey darken-4` : ``')
|
|
|
|
v-list-item.pl-4(href='/')
|
|
|
|
v-list-item-avatar(size='24'): v-icon(color='blue') mdi-home
|
|
|
|
v-list-item-title.body-2 {{$t('common:header.home')}}
|
|
|
|
v-list-item.pl-4(@click='')
|
|
|
|
v-list-item-avatar(size='24'): v-icon(color='grey lighten-2') mdi-file-tree
|
|
|
|
v-list-item-content
|
|
|
|
v-list-item-title.body-2.grey--text.text--ligten-2 {{$t('common:header.siteMap')}}
|
|
|
|
v-list-item-subtitle.overline.grey--text.text--lighten-2 Coming soon
|
|
|
|
v-list-item.pl-4(href='/t')
|
|
|
|
v-list-item-avatar(size='24'): v-icon(color='teal') mdi-tag-multiple
|
|
|
|
v-list-item-title.body-2 {{$t('common:header.browseTags')}}
|
|
|
|
v-list-item.pl-4(@click='assets')
|
|
|
|
v-list-item-avatar(size='24'): v-icon(color='grey lighten-2') mdi-folder-multiple-image
|
|
|
|
v-list-item-content
|
|
|
|
v-list-item-title.body-2.grey--text.text--ligten-2 {{$t('common:header.imagesFiles')}}
|
|
|
|
v-list-item-subtitle.overline.grey--text.text--lighten-2 Coming soon
|
|
|
|
v-toolbar-title(:class='{ "mx-2": $vuetify.breakpoint.mdAndUp, "mx-0": $vuetify.breakpoint.smAndDown }')
|
|
|
|
span.subheading {{title}}
|
|
|
|
v-flex(md4, v-if='$vuetify.breakpoint.mdAndUp')
|
|
|
|
v-toolbar.nav-header-inner(color='black', dark, flat)
|
|
|
|
slot(name='mid')
|
|
|
|
transition(name='navHeaderSearch', v-if='searchIsShown')
|
|
|
|
v-text-field(
|
|
|
|
ref='searchField',
|
|
|
|
v-if='searchIsShown && $vuetify.breakpoint.mdAndUp',
|
|
|
|
v-model='search',
|
|
|
|
color='white',
|
|
|
|
:label='$t(`common:header.search`)',
|
|
|
|
single-line,
|
|
|
|
solo
|
|
|
|
flat
|
|
|
|
rounded
|
|
|
|
hide-details,
|
|
|
|
prepend-inner-icon='mdi-magnify',
|
|
|
|
:loading='searchIsLoading',
|
|
|
|
@keyup.enter='searchEnter'
|
|
|
|
@keyup.esc='searchClose'
|
|
|
|
@focus='searchFocus'
|
|
|
|
@blur='searchBlur'
|
|
|
|
@keyup.down='searchMove(`down`)'
|
|
|
|
@keyup.up='searchMove(`up`)'
|
|
|
|
)
|
|
|
|
v-tooltip(bottom)
|
|
|
|
template(v-slot:activator='{ on }')
|
|
|
|
v-btn.ml-2.mr-0(icon, v-on='on', href='/t')
|
|
|
|
v-icon(color='grey') mdi-tag-multiple
|
|
|
|
span {{$t('common:header.browseTags')}}
|
|
|
|
v-flex(xs6, md4)
|
|
|
|
v-toolbar.nav-header-inner.pr-4(color='black', dark, flat)
|
|
|
|
v-spacer
|
|
|
|
.navHeaderLoading.mr-3
|
|
|
|
v-progress-circular(indeterminate, color='blue', :size='22', :width='2' v-show='isLoading')
|
|
|
|
|
|
|
|
slot(name='actions')
|
|
|
|
|
|
|
|
//- (mobile) SEARCH TOGGLE
|
|
|
|
|
|
|
|
v-btn(
|
|
|
|
v-if='!hideSearch && $vuetify.breakpoint.smAndDown'
|
|
|
|
@click='searchToggle'
|
|
|
|
icon
|
|
|
|
)
|
|
|
|
v-icon(color='grey') mdi-magnify
|
|
|
|
|
|
|
|
//- LANGUAGES
|
|
|
|
|
|
|
|
template(v-if='mode === `view` && locales.length > 0')
|
|
|
|
v-menu(offset-y, bottom, transition='slide-y-transition')
|
|
|
|
template(v-slot:activator='{ on: menu }')
|
|
|
|
v-tooltip(bottom)
|
|
|
|
template(v-slot:activator='{ on: tooltip }')
|
|
|
|
v-btn(icon, v-on='{ ...menu, ...tooltip }', :class='$vuetify.rtl ? `ml-3` : ``', tile, height='64')
|
|
|
|
v-icon(color='grey') mdi-web
|
|
|
|
span {{$t('common:header.language')}}
|
|
|
|
v-list(nav)
|
|
|
|
template(v-for='(lc, idx) of locales')
|
|
|
|
v-list-item(@click='changeLocale(lc)')
|
|
|
|
v-list-item-action(style='min-width:auto;'): v-chip(:color='lc.code === locale ? `blue` : `grey`', small, label, dark) {{lc.code.toUpperCase()}}
|
|
|
|
v-list-item-title {{lc.name}}
|
|
|
|
v-divider(vertical)
|
|
|
|
|
|
|
|
//- PAGE ACTIONS
|
|
|
|
|
|
|
|
template(v-if='isAuthenticated && path && mode !== `edit`')
|
|
|
|
v-menu(offset-y, bottom, transition='slide-y-transition')
|
|
|
|
template(v-slot:activator='{ on: menu }')
|
|
|
|
v-tooltip(bottom)
|
|
|
|
template(v-slot:activator='{ on: tooltip }')
|
|
|
|
v-btn(icon, v-on='{ ...menu, ...tooltip }', :class='$vuetify.rtl ? `ml-3` : ``', tile, height='64')
|
|
|
|
v-icon(color='grey') mdi-file-document-edit-outline
|
|
|
|
span {{$t('common:header.pageActions')}}
|
|
|
|
v-list(nav, :light='!$vuetify.theme.dark', :dark='$vuetify.theme.dark', :class='$vuetify.theme.dark ? `grey darken-4` : ``')
|
|
|
|
.overline.pa-4.grey--text {{$t('common:header.currentPage')}}
|
|
|
|
v-list-item.pl-4(@click='pageView', v-if='mode !== `view`')
|
|
|
|
v-list-item-avatar(size='24'): v-icon(color='indigo') mdi-file-document-box-outline
|
|
|
|
v-list-item-title.body-2 {{$t('common:header.view')}}
|
|
|
|
v-list-item.pl-4(@click='pageEdit', v-if='mode !== `edit` && isAuthenticated')
|
|
|
|
v-list-item-avatar(size='24'): v-icon(color='indigo') mdi-file-document-edit-outline
|
|
|
|
v-list-item-title.body-2 {{$t('common:header.edit')}}
|
|
|
|
v-list-item.pl-4(@click='pageHistory', v-if='mode !== `history`')
|
|
|
|
v-list-item-avatar(size='24'): v-icon(color='grey lighten-2') mdi-history
|
|
|
|
v-list-item-content
|
|
|
|
v-list-item-title.body-2.grey--text.text--ligten-2 {{$t('common:header.history')}}
|
|
|
|
v-list-item-subtitle.overline.grey--text.text--lighten-2 Coming soon
|
|
|
|
v-list-item.pl-4(@click='pageSource', v-if='mode !== `source`')
|
|
|
|
v-list-item-avatar(size='24'): v-icon(color='indigo') mdi-code-tags
|
|
|
|
v-list-item-title.body-2 {{$t('common:header.viewSource')}}
|
|
|
|
v-list-item.pl-4(@click='pageMove', v-if='isAuthenticated')
|
|
|
|
v-list-item-avatar(size='24'): v-icon(color='indigo') mdi-content-save-move-outline
|
|
|
|
v-list-item-content
|
|
|
|
v-list-item-title.body-2 {{$t('common:header.move')}}
|
|
|
|
v-list-item.pl-4(@click='pageDelete', v-if='isAuthenticated')
|
|
|
|
v-list-item-avatar(size='24'): v-icon(color='red darken-2') mdi-trash-can-outline
|
|
|
|
v-list-item-title.body-2 {{$t('common:header.delete')}}
|
|
|
|
v-divider(vertical)
|
|
|
|
|
|
|
|
//- NEW PAGE
|
|
|
|
|
|
|
|
template(v-if='isAuthenticated')
|
|
|
|
v-tooltip(bottom)
|
|
|
|
template(v-slot:activator='{ on }')
|
|
|
|
v-btn(icon, tile, height='64', v-on='on', @click='pageNew')
|
|
|
|
v-icon(color='grey') mdi-file-document-box-plus-outline
|
|
|
|
span {{$t('common:header.newPage')}}
|
|
|
|
v-divider(vertical)
|
|
|
|
|
|
|
|
//- ACCOUNT
|
|
|
|
|
|
|
|
v-menu(v-if='isAuthenticated', offset-y, bottom, min-width='300', transition='slide-y-transition')
|
|
|
|
template(v-slot:activator='{ on: menu }')
|
|
|
|
v-tooltip(bottom)
|
|
|
|
template(v-slot:activator='{ on: tooltip }')
|
|
|
|
v-btn(icon, v-on='{ ...menu, ...tooltip }', :class='$vuetify.rtl ? `ml-0` : ``', tile, height='64')
|
|
|
|
v-icon(v-if='picture.kind === `initials`', color='grey') mdi-account-circle
|
|
|
|
v-avatar(v-else-if='picture.kind === `image`', :size='34')
|
|
|
|
v-img(:src='picture.url')
|
|
|
|
span {{$t('common:header.account')}}
|
|
|
|
v-list(nav)
|
|
|
|
v-list-item.py-3.grey(:class='$vuetify.theme.dark ? `darken-4-l5` : `lighten-5`')
|
|
|
|
v-list-item-avatar
|
|
|
|
v-avatar.blue(v-if='picture.kind === `initials`', :size='40')
|
|
|
|
span.white--text.subheading {{picture.initials}}
|
|
|
|
v-avatar(v-else-if='picture.kind === `image`', :size='40')
|
|
|
|
v-img(:src='picture.url')
|
|
|
|
v-list-item-content
|
|
|
|
v-list-item-title {{name}}
|
|
|
|
v-list-item-subtitle {{email}}
|
|
|
|
v-list-item(href='/w', disabled)
|
|
|
|
v-list-item-action: v-icon(color='blue') mdi-view-compact-outline
|
|
|
|
v-list-item-content
|
|
|
|
v-list-item-title {{$t('common:header.myWiki')}}
|
|
|
|
v-list-item-subtitle.overline Coming soon
|
|
|
|
v-list-item(href='/p', disabled)
|
|
|
|
v-list-item-action: v-icon(color='blue') mdi-face-profile
|
|
|
|
v-list-item-content
|
|
|
|
v-list-item-title {{$t('common:header.profile')}}
|
|
|
|
v-list-item-subtitle.overline Coming soon
|
|
|
|
v-list-item(href='/a', v-if='isAuthenticated && isAdmin')
|
|
|
|
v-list-item-action.btn-animate-rotate: v-icon(:color='$vuetify.theme.dark ? `blue-grey lighten-3` : `blue-grey`') mdi-settings
|
|
|
|
v-list-item-title(:class='$vuetify.theme.dark ? `blue-grey--text text--lighten-3` : `blue-grey--text`') {{$t('common:header.admin')}}
|
|
|
|
v-list-item(@click='logout')
|
|
|
|
v-list-item-action: v-icon(color='red') mdi-logout
|
|
|
|
v-list-item-title.red--text {{$t('common:header.logout')}}
|
|
|
|
|
|
|
|
v-tooltip(v-else, left)
|
|
|
|
template(v-slot:activator='{ on }')
|
|
|
|
v-btn(icon, v-on='on', color='grey darken-3', href='/login')
|
|
|
|
v-icon(color='grey') mdi-account-circle
|
|
|
|
span {{$t('common:header.login')}}
|
|
|
|
|
|
|
|
page-selector(mode='create', v-model='newPageModal', :open-handler='pageNewCreate', :locale='locale')
|
|
|
|
page-selector(mode='move', v-model='movePageModal', :open-handler='pageMoveRename', :path='path', :locale='locale')
|
|
|
|
page-delete(v-model='deletePageModal', v-if='path && path.length')
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
|
|
|
import { get, sync } from 'vuex-pathify'
|
|
|
|
import _ from 'lodash'
|
|
|
|
import Cookies from 'js-cookie'
|
|
|
|
|
|
|
|
import movePageMutation from 'gql/common/common-pages-mutation-move.gql'
|
|
|
|
|
|
|
|
/* global siteLangs */
|
|
|
|
|
|
|
|
export default {
|
|
|
|
components: {
|
|
|
|
PageDelete: () => import('./page-delete.vue')
|
|
|
|
},
|
|
|
|
props: {
|
|
|
|
dense: {
|
|
|
|
type: Boolean,
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
hideSearch: {
|
|
|
|
type: Boolean,
|
|
|
|
default: false
|
|
|
|
}
|
|
|
|
},
|
|
|
|
data() {
|
|
|
|
return {
|
|
|
|
menuIsShown: true,
|
|
|
|
searchIsShown: true,
|
|
|
|
searchAdvMenuShown: false,
|
|
|
|
newPageModal: false,
|
|
|
|
movePageModal: false,
|
|
|
|
deletePageModal: false,
|
|
|
|
locales: siteLangs
|
|
|
|
}
|
|
|
|
},
|
|
|
|
computed: {
|
|
|
|
search: sync('site/search'),
|
|
|
|
searchIsFocused: sync('site/searchIsFocused'),
|
|
|
|
searchIsLoading: sync('site/searchIsLoading'),
|
|
|
|
searchRestrictLocale: sync('site/searchRestrictLocale'),
|
|
|
|
searchRestrictPath: sync('site/searchRestrictPath'),
|
|
|
|
isLoading: get('isLoading'),
|
|
|
|
title: get('site/title'),
|
|
|
|
path: get('page/path'),
|
|
|
|
locale: get('page/locale'),
|
|
|
|
mode: get('page/mode'),
|
|
|
|
name: get('user/name'),
|
|
|
|
email: get('user/email'),
|
|
|
|
pictureUrl: get('user/pictureUrl'),
|
|
|
|
isAuthenticated: get('user/authenticated'),
|
|
|
|
permissions: get('user/permissions'),
|
|
|
|
picture () {
|
|
|
|
if (this.pictureUrl && this.pictureUrl.length > 1) {
|
|
|
|
return {
|
|
|
|
kind: 'image',
|
|
|
|
url: this.pictureUrl
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const nameParts = this.name.toUpperCase().split(' ')
|
|
|
|
let initials = _.head(nameParts).charAt(0)
|
|
|
|
if (nameParts.length > 1) {
|
|
|
|
initials += _.last(nameParts).charAt(0)
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
kind: 'initials',
|
|
|
|
initials
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
isAdmin () {
|
|
|
|
return _.intersection(this.permissions, ['manage:system', 'write:users', 'manage:users', 'write:groups', 'manage:groups', 'manage:navigation', 'manage:theme', 'manage:api']).length > 0
|
|
|
|
}
|
|
|
|
},
|
|
|
|
created () {
|
|
|
|
if (this.hideSearch || this.dense || this.$vuetify.breakpoint.smAndDown) {
|
|
|
|
this.searchIsShown = false
|
|
|
|
}
|
|
|
|
},
|
|
|
|
mounted () {
|
|
|
|
this.$root.$on('pageEdit', () => {
|
|
|
|
this.pageEdit()
|
|
|
|
})
|
|
|
|
this.$root.$on('pageHistory', () => {
|
|
|
|
this.pageHistory()
|
|
|
|
})
|
|
|
|
this.$root.$on('pageSource', () => {
|
|
|
|
this.pageSource()
|
|
|
|
})
|
|
|
|
this.$root.$on('pageMove', () => {
|
|
|
|
this.pageMove()
|
|
|
|
})
|
|
|
|
this.$root.$on('pageDelete', () => {
|
|
|
|
this.pageDelete()
|
|
|
|
})
|
|
|
|
},
|
|
|
|
methods: {
|
|
|
|
searchFocus () {
|
|
|
|
this.searchIsFocused = true
|
|
|
|
},
|
|
|
|
searchBlur () {
|
|
|
|
this.searchIsFocused = false
|
|
|
|
},
|
|
|
|
searchClose () {
|
|
|
|
this.search = ''
|
|
|
|
this.searchBlur()
|
|
|
|
},
|
|
|
|
searchToggle () {
|
|
|
|
this.searchIsShown = !this.searchIsShown
|
|
|
|
if (this.searchIsShown) {
|
|
|
|
_.delay(() => {
|
|
|
|
this.$refs.searchFieldMobile.focus()
|
|
|
|
}, 200)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
searchEnter () {
|
|
|
|
this.$root.$emit('searchEnter', true)
|
|
|
|
},
|
|
|
|
searchMove(dir) {
|
|
|
|
this.$root.$emit('searchMove', dir)
|
|
|
|
},
|
|
|
|
pageNew () {
|
|
|
|
this.newPageModal = true
|
|
|
|
},
|
|
|
|
pageNewCreate ({ path, locale }) {
|
|
|
|
window.location.assign(`/e/${locale}/${path}`)
|
|
|
|
},
|
|
|
|
pageView () {
|
|
|
|
window.location.assign(`/${this.locale}/${this.path}`)
|
|
|
|
},
|
|
|
|
pageEdit () {
|
|
|
|
window.location.assign(`/e/${this.locale}/${this.path}`)
|
|
|
|
},
|
|
|
|
pageHistory () {
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'indigo',
|
|
|
|
message: `Coming soon...`,
|
|
|
|
icon: 'ferry'
|
|
|
|
})
|
|
|
|
// window.location.assign(`/h/${this.locale}/${this.path}`)
|
|
|
|
},
|
|
|
|
pageSource () {
|
|
|
|
window.location.assign(`/s/${this.locale}/${this.path}`)
|
|
|
|
},
|
|
|
|
pageMove () {
|
|
|
|
this.movePageModal = true
|
|
|
|
},
|
|
|
|
async pageMoveRename ({ path, locale }) {
|
|
|
|
this.$store.commit(`loadingStart`, 'page-move')
|
|
|
|
try {
|
|
|
|
const resp = await this.$apollo.mutate({
|
|
|
|
mutation: movePageMutation,
|
|
|
|
variables: {
|
|
|
|
id: this.$store.get('page/id'),
|
|
|
|
destinationLocale: locale,
|
|
|
|
destinationPath: path
|
|
|
|
}
|
|
|
|
})
|
|
|
|
if (_.get(resp, 'data.pages.move.responseResult.succeeded', false)) {
|
|
|
|
window.location.replace(`/${locale}/${path}`)
|
|
|
|
} else {
|
|
|
|
throw new Error(_.get(resp, 'data.pages.move.responseResult.message', this.$t('common:error.unexpected')))
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
this.$store.commit('pushGraphError', err)
|
|
|
|
this.$store.commit(`loadingStop`, 'page-move')
|
|
|
|
}
|
|
|
|
},
|
|
|
|
pageDelete () {
|
|
|
|
this.deletePageModal = true
|
|
|
|
},
|
|
|
|
assets () {
|
|
|
|
// window.location.assign(`/f`)
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'indigo',
|
|
|
|
message: `Coming soon...`,
|
|
|
|
icon: 'ferry'
|
|
|
|
})
|
|
|
|
},
|
|
|
|
async changeLocale (locale) {
|
|
|
|
await this.$i18n.i18next.changeLanguage(locale.code)
|
|
|
|
switch (this.mode) {
|
|
|
|
case 'view':
|
|
|
|
case 'history':
|
|
|
|
window.location.assign(`/${locale.code}/${this.path}`)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
},
|
|
|
|
logout () {
|
|
|
|
Cookies.remove('jwt')
|
|
|
|
window.location.assign('/')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang='scss'>
|
|
|
|
|
|
|
|
.nav-header {
|
|
|
|
//z-index: 1000;
|
|
|
|
|
|
|
|
.v-toolbar__extension {
|
|
|
|
padding: 0;
|
|
|
|
|
|
|
|
.v-toolbar__content {
|
|
|
|
padding: 0;
|
|
|
|
}
|
|
|
|
.v-text-field .v-input__prepend-inner {
|
|
|
|
padding: 0 14px 0 5px;
|
|
|
|
padding-right: 14px;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
&-inner {
|
|
|
|
.v-toolbar__content {
|
|
|
|
padding: 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
&-search-adv {
|
|
|
|
position: absolute;
|
|
|
|
top: 7px;
|
|
|
|
right: 12px;
|
|
|
|
border-radius: 4px !important;
|
|
|
|
|
|
|
|
@at-root .v-application--is-rtl & {
|
|
|
|
right: initial;
|
|
|
|
left: 12px;
|
|
|
|
}
|
|
|
|
|
|
|
|
&::before {
|
|
|
|
border-radius: 4px !important;
|
|
|
|
}
|
|
|
|
|
|
|
|
&:hover, &:focus {
|
|
|
|
position: absolute !important;
|
|
|
|
|
|
|
|
&::before {
|
|
|
|
border-radius: 4px;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.navHeaderSearch {
|
|
|
|
&-enter-active, &-leave-active {
|
|
|
|
transition: opacity .25s ease, transform .25s ease;
|
|
|
|
opacity: 1;
|
|
|
|
}
|
|
|
|
&-enter-active {
|
|
|
|
transition-delay: .25s;
|
|
|
|
}
|
|
|
|
&-enter, &-leave-to {
|
|
|
|
opacity: 0;
|
|
|
|
transform: scale(.7, .7);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
.navHeaderLoading { // To avoid search bar jumping
|
|
|
|
width: 22px;
|
|
|
|
}
|
|
|
|
|
|
|
|
</style>
|