123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334 |
- <template lang="pug">
- v-app(v-scroll='upBtnScroll', :dark='darkMode')
- nav-header
- v-navigation-drawer(
- :class='darkMode ? `grey darken-5` : `primary`'
- dark
- app
- clipped
- mobile-break-point='600'
- :temporary='$vuetify.breakpoint.mdAndDown'
- v-model='navShown'
- :right='$vuetify.rtl'
- )
- vue-scroll(:ops='scrollStyle')
- nav-sidebar(:color='darkMode ? `grey darken-5` : `primary`', :items='sidebar')
- v-fab-transition
- v-btn(
- fab
- color='primary'
- fixed
- bottom
- :right='$vuetify.rtl'
- :left='!$vuetify.rtl'
- small
- @click='navShown = !navShown'
- v-if='$vuetify.breakpoint.mdAndDown'
- v-show='!navShown'
- )
- v-icon mdi-menu
- v-content(ref='content')
- template(v-if='path !== `home`')
- v-toolbar(:color='darkMode ? `grey darken-4-d3` : `grey lighten-3`', flat, dense, v-if='$vuetify.breakpoint.smAndUp')
- //- v-btn.pl-0(v-if='$vuetify.breakpoint.xsOnly', flat, @click='toggleNavigation')
- //- v-icon(color='grey darken-2', left) menu
- //- span Navigation
- v-breadcrumbs.breadcrumbs-nav.pl-0(
- :items='breadcrumbs'
- divider='/'
- )
- template(slot='item', slot-scope='props')
- v-icon(v-if='props.item.path === "/"', small, @click='goHome') mdi-home
- v-btn.ma-0(v-else, :href='props.item.path', small, text) {{props.item.name}}
- template(v-if='!isPublished')
- v-spacer
- .caption.red--text {{$t('common:page.unpublished')}}
- status-indicator.ml-3(negative, pulse)
- v-divider
- v-toolbar.px-2(:color='darkMode ? `grey darken-4-l3` : `grey lighten-4`', flat, :height='90')
- div(style='padding-left: 376px;')
- .headline.grey--text(:class='darkMode ? `text--lighten-2` : `text--darken-3`') {{title}}
- .caption.grey--text.text--darken-1 {{description}}
- v-spacer
- v-divider
- v-container.pl-5.pt-2(fill-height, fluid, grid-list-xl)
- v-layout(row)
- v-flex.page-col-sd(lg3, xl2, fill-height, v-if='$vuetify.breakpoint.lgAndUp', style='margin-top: -90px;')
- v-card(v-if='toc.length')
- .overline.pa-5.pb-0(:class='darkMode ? `indigo--text text--lighten-3` : `primary--text`') {{$t('common:page.toc')}}
- v-list.pb-3(dense, nav, :class='darkMode ? `darken-3-d3` : ``')
- template(v-for='(tocItem, tocIdx) in toc')
- v-list-item(@click='$vuetify.goTo(tocItem.anchor, scrollOpts)')
- v-icon(color='grey', small) mdi-chevron-right
- v-list-item-title.pl-3 {{tocItem.title}}
- //- v-divider(v-if='tocIdx < toc.length - 1 || tocItem.children.length')
- template(v-for='tocSubItem in tocItem.children')
- v-list-item(@click='$vuetify.goTo(tocSubItem.anchor, scrollOpts)')
- v-icon.pl-3(color='grey lighten-1', small) mdi-chevron-right
- v-list-item-title.pl-3.caption.grey--text.text--darken-1 {{tocSubItem.title}}
- //- v-divider(inset, v-if='tocIdx < toc.length - 1')
- v-card.mt-5
- .pa-5.pt-3
- .overline.indigo--text.d-flex.align-center
- span {{$t('common:page.lastEditedBy')}}
- v-spacer
- v-tooltip(top, v-if='isAuthenticated')
- template(v-slot:activator='{ on }')
- v-btn.btn-animate-edit(icon, :href='"/h/" + locale + "/" + path', v-on='on', x-small)
- v-icon(color='grey', dense) mdi-history
- span History
- .body-2.grey--text(:class='darkMode ? `` : `text--darken-3`') {{ authorName }}
- .caption.grey--text.text--darken-1 {{ updatedAt | moment('calendar') }}
- v-card.mt-5(v-if='tags.length > 0 || true')
- .pa-5
- .overline.teal--text.pb-2 Tags
- v-chip.mr-1(
- label
- color='teal lighten-5'
- v-for='(tag, idx) in tags'
- :href='`/t/` + tag.slug'
- :key='tag.slug'
- )
- v-icon(color='teal', left, small) mdi-label
- span.teal--text.text--darken-2 {{tag.text}}
- v-card.mt-5
- .pa-5
- .overline.pb-2.yellow--text.text--darken-4 Rating
- .text-center
- v-rating(
- v-model='rating'
- color='yellow darken-3'
- background-color='grey lighten-1'
- half-increments
- hover
- )
- .caption.grey--text 5 votes
- v-divider
- v-toolbar(:color='darkMode ? `grey darken-3` : `grey lighten-4`', flat, dense)
- v-spacer
- v-tooltip(bottom)
- template(v-slot:activator='{ on }')
- v-btn(icon, tile, small, v-on='on'): v-icon(color='grey') mdi-bookmark
- span {{$t('common:page.bookmark')}}
- v-tooltip(bottom)
- template(v-slot:activator='{ on }')
- v-btn(icon, tile, small, v-on='on'): v-icon(color='grey') mdi-share-variant
- span {{$t('common:page.share')}}
- v-tooltip(bottom)
- template(v-slot:activator='{ on }')
- v-btn(icon, tile, small, v-on='on'): v-icon(color='grey') mdi-printer
- span {{$t('common:page.printFormat')}}
- v-spacer
- v-flex.page-col-content(xs12, lg9, xl10)
- v-tooltip(left, v-if='isAuthenticated')
- template(v-slot:activator='{ on }')
- v-btn.btn-animate-edit(fab, bottom, right, color='primary', fixed, dark, :href='"/e/" + locale + "/" + path', v-on='on')
- v-icon mdi-pencil
- span {{$t('common:page.editPage')}}
- .contents(ref='container')
- slot(name='contents')
- nav-footer
- notify
- search-results
- v-fab-transition
- v-btn(
- v-if='upBtnShown'
- fab
- fixed
- bottom
- :right='$vuetify.rtl'
- :left='!$vuetify.rtl'
- small
- depressed
- @click='$vuetify.goTo(0, scrollOpts)'
- color='primary'
- dark
- :style='$vuetify.rtl ? `right: 235px;` : `left: 235px;`'
- )
- v-icon mdi-arrow-up
- </template>
- <script>
- import { StatusIndicator } from 'vue-status-indicator'
- import Prism from 'prismjs'
- import { get } from 'vuex-pathify'
- import _ from 'lodash'
- Prism.plugins.autoloader.languages_path = '/js/prism/'
- Prism.plugins.NormalizeWhitespace.setDefaults({
- 'remove-trailing': true,
- 'remove-indent': true,
- 'left-trim': true,
- 'right-trim': true,
- 'break-lines': 160,
- 'remove-initial-line-feed': true,
- 'tabs-to-spaces': 2
- })
- export default {
- components: {
- StatusIndicator
- },
- props: {
- pageId: {
- type: Number,
- default: 0
- },
- locale: {
- type: String,
- default: 'en'
- },
- path: {
- type: String,
- default: 'home'
- },
- title: {
- type: String,
- default: 'Untitled Page'
- },
- description: {
- type: String,
- default: ''
- },
- createdAt: {
- type: String,
- default: ''
- },
- updatedAt: {
- type: String,
- default: ''
- },
- tags: {
- type: Array,
- default: () => ([])
- },
- authorName: {
- type: String,
- default: 'Unknown'
- },
- authorId: {
- type: Number,
- default: 0
- },
- isPublished: {
- type: Boolean,
- default: false
- },
- toc: {
- type: Array,
- default: () => []
- },
- sidebar: {
- type: Array,
- default: () => []
- }
- },
- data() {
- return {
- navShown: false,
- navExpanded: false,
- upBtnShown: false,
- scrollOpts: {
- duration: 1500,
- offset: -75,
- easing: 'easeInOutCubic'
- },
- scrollStyle: {
- vuescroll: {},
- scrollPanel: {
- initialScrollX: 0.01, // fix scrollbar not disappearing on load
- scrollingX: false,
- speed: 50
- },
- rail: {
- gutterOfEnds: '2px'
- },
- bar: {
- onlyShowBarOnScroll: false,
- background: '#42A5F5',
- hoverStyle: {
- background: '#64B5F6'
- }
- }
- }
- }
- },
- computed: {
- darkMode: get('site/dark'),
- isAuthenticated: get('user/authenticated'),
- rating: {
- get () {
- return 3.5
- },
- set (val) {
- }
- },
- breadcrumbs() {
- return [{ path: '/', name: 'Home' }].concat(_.reduce(this.path.split('/'), (result, value, key) => {
- result.push({
- path: _.map(result, 'path').join('/') + `/${value}`,
- name: value
- })
- return result
- }, []))
- }
- },
- created() {
- this.$store.commit('page/SET_AUTHOR_ID', this.authorId)
- this.$store.commit('page/SET_AUTHOR_NAME', this.authorName)
- this.$store.commit('page/SET_CREATED_AT', this.createdAt)
- this.$store.commit('page/SET_DESCRIPTION', this.description)
- this.$store.commit('page/SET_IS_PUBLISHED', this.isPublished)
- this.$store.commit('page/SET_ID', this.pageId)
- this.$store.commit('page/SET_LOCALE', this.locale)
- this.$store.commit('page/SET_PATH', this.path)
- this.$store.commit('page/SET_TAGS', this.tags)
- this.$store.commit('page/SET_TITLE', this.title)
- this.$store.commit('page/SET_UPDATED_AT', this.updatedAt)
- this.$store.commit('page/SET_MODE', 'view')
- },
- mounted () {
- Prism.highlightAllUnder(this.$refs.container)
- this.navShown = this.$vuetify.breakpoint.smAndUp
- },
- methods: {
- goHome () {
- window.location.assign('/')
- },
- toggleNavigation () {
- this.navOpen = !this.navOpen
- },
- upBtnScroll () {
- const scrollOffset = window.pageYOffset || document.documentElement.scrollTop
- this.upBtnShown = scrollOffset > window.innerHeight * 0.33
- }
- }
- }
- </script>
- <style lang="scss">
- .breadcrumbs-nav {
- .v-btn {
- min-width: 0;
- &__content {
- text-transform: none;
- }
- }
- .v-breadcrumbs__divider:nth-child(2n) {
- padding: 0 6px;
- }
- .v-breadcrumbs__divider:nth-child(2) {
- padding: 0 6px 0 12px;
- }
- }
- </style>
|