page.vue 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. <template lang="pug">
  2. v-app(v-scroll='upBtnScroll', :dark='darkMode')
  3. nav-header
  4. v-navigation-drawer(
  5. :class='darkMode ? `grey darken-5` : `primary`'
  6. dark
  7. app
  8. clipped
  9. mobile-break-point='600'
  10. :temporary='$vuetify.breakpoint.mdAndDown'
  11. v-model='navShown'
  12. :right='$vuetify.rtl'
  13. )
  14. vue-scroll(:ops='scrollStyle')
  15. nav-sidebar(:color='darkMode ? `grey darken-5` : `primary`')
  16. slot(name='sidebar')
  17. v-fab-transition
  18. v-btn(
  19. fab
  20. color='primary'
  21. fixed
  22. bottom
  23. :right='$vuetify.rtl'
  24. :left='!$vuetify.rtl'
  25. small
  26. @click='navShown = !navShown'
  27. v-if='$vuetify.breakpoint.mdAndDown'
  28. v-show='!navShown'
  29. )
  30. v-icon menu
  31. v-content(ref='content')
  32. template(v-if='path !== `home`')
  33. v-toolbar(:color='darkMode ? `grey darken-4-d3` : `grey lighten-3`', flat, dense, v-if='$vuetify.breakpoint.smAndUp')
  34. //- v-btn.pl-0(v-if='$vuetify.breakpoint.xsOnly', flat, @click='toggleNavigation')
  35. //- v-icon(color='grey darken-2', left) menu
  36. //- span Navigation
  37. v-breadcrumbs.breadcrumbs-nav.pl-0(
  38. :items='breadcrumbs'
  39. divider='/'
  40. )
  41. template(slot='item', slot-scope='props')
  42. v-icon(v-if='props.item.path === "/"', small, @click='goHome') home
  43. v-btn.ma-0(v-else, :href='props.item.path', small, flat) {{props.item.name}}
  44. template(v-if='!isPublished')
  45. v-spacer
  46. .caption.red--text {{$t('common:page.unpublished')}}
  47. status-indicator.ml-3(negative, pulse)
  48. v-divider
  49. v-layout(row)
  50. v-flex(xs12, lg9, xl10)
  51. v-toolbar(:color='darkMode ? `grey darken-4-l3` : `grey lighten-4`', flat, :height='90')
  52. div
  53. .headline.grey--text(:class='darkMode ? `text--lighten-2` : `text--darken-3`') {{title}}
  54. .caption.grey--text.text--darken-1 {{description}}
  55. v-divider
  56. .contents(ref='container')
  57. slot(name='contents')
  58. v-flex(lg3, xl2, fill-height, v-if='$vuetify.breakpoint.lgAndUp')
  59. v-toolbar(:color='darkMode ? `grey darken-4-l3` : `grey lighten-4`', flat, :height='90')
  60. div
  61. .caption.grey--text.text--lighten-1 {{$t('common:page.lastEditedBy')}}
  62. .body-2.grey--text(:class='darkMode ? `` : `text--darken-3`') {{ authorName }}
  63. .caption.grey--text.text--darken-1 {{ updatedAt | moment('calendar') }}
  64. template(v-if='isAuthenticated')
  65. v-spacer
  66. v-tooltip(left)
  67. v-btn.btn-animate-edit(icon, slot='activator', :href='"/e/" + locale + "/" + path')
  68. v-icon(color='grey') edit
  69. span {{$t('common:page.editPage')}}
  70. v-divider
  71. template(v-if='toc.length')
  72. v-list.grey.pb-3(dense, :class='darkMode ? `darken-3-d3` : `lighten-3`')
  73. v-subheader.pl-4(:class='darkMode ? `indigo--text text--lighten-3` : `primary--text`') {{$t('common:page.toc')}}
  74. template(v-for='(tocItem, tocIdx) in toc')
  75. v-list-tile(@click='$vuetify.goTo(tocItem.anchor, scrollOpts)')
  76. v-icon(color='grey') arrow_right
  77. v-list-tile-title.pl-3 {{tocItem.title}}
  78. v-divider.ml-4(v-if='tocIdx < toc.length - 1 || tocItem.children.length')
  79. template(v-for='tocSubItem in tocItem.children')
  80. v-list-tile(@click='$vuetify.goTo(tocSubItem.anchor, scrollOpts)')
  81. v-icon.pl-3(color='grey lighten-1') arrow_right
  82. v-list-tile-title.pl-3.caption {{tocSubItem.title}}
  83. v-divider(inset, v-if='tocIdx < toc.length - 1')
  84. v-divider
  85. //- v-list.grey(dense, :class='darkMode ? `darken-3` : `lighten-4`')
  86. //- v-subheader.pl-4.yellow--text.text--darken-4 Rating
  87. //- .text-xs-center
  88. //- v-rating(
  89. //- v-model='rating'
  90. //- color='yellow darken-3'
  91. //- background-color='grey lighten-1'
  92. //- half-increments
  93. //- hover
  94. //- )
  95. //- .pb-2.caption.grey--text 5 votes
  96. //- v-divider
  97. template(v-if='tags.length')
  98. v-list.grey(dense, :class='darkMode ? `darken-3-d3` : `lighten-3`')
  99. v-subheader.pl-4.teal--text Tags
  100. template(v-for='(tag, idx) in tags')
  101. v-list-tile(:href='`/t/` + tag.slug')
  102. v-list-tile-avatar: v-icon(color='teal') label
  103. v-list-tile-title {{tag.title}}
  104. v-divider(inset, v-if='idx < tags.length - 1')
  105. v-divider
  106. v-toolbar(:color='darkMode ? `grey darken-3` : `grey lighten-4`', flat, dense)
  107. v-spacer
  108. v-tooltip(bottom)
  109. v-btn(icon, slot='activator'): v-icon(color='grey') bookmark
  110. span {{$t('common:page.bookmark')}}
  111. v-tooltip(bottom)
  112. v-btn(icon, slot='activator'): v-icon(color='grey') share
  113. span {{$t('common:page.share')}}
  114. v-tooltip(bottom)
  115. v-btn(icon, slot='activator'): v-icon(color='grey') print
  116. span {{$t('common:page.printFormat')}}
  117. v-spacer
  118. nav-footer
  119. notify
  120. search-results
  121. v-fab-transition
  122. v-btn(
  123. v-if='upBtnShown'
  124. fab
  125. fixed
  126. bottom
  127. :right='!$vuetify.rtl'
  128. :left='$vuetify.rtl'
  129. small
  130. @click='$vuetify.goTo(0, scrollOpts)'
  131. color='primary'
  132. )
  133. v-icon arrow_upward
  134. </template>
  135. <script>
  136. import { StatusIndicator } from 'vue-status-indicator'
  137. import Prism from 'prismjs'
  138. import { get } from 'vuex-pathify'
  139. import _ from 'lodash'
  140. export default {
  141. components: {
  142. StatusIndicator
  143. },
  144. props: {
  145. pageId: {
  146. type: Number,
  147. default: 0
  148. },
  149. locale: {
  150. type: String,
  151. default: 'en'
  152. },
  153. path: {
  154. type: String,
  155. default: 'home'
  156. },
  157. title: {
  158. type: String,
  159. default: 'Untitled Page'
  160. },
  161. description: {
  162. type: String,
  163. default: ''
  164. },
  165. createdAt: {
  166. type: String,
  167. default: ''
  168. },
  169. updatedAt: {
  170. type: String,
  171. default: ''
  172. },
  173. tags: {
  174. type: Array,
  175. default: () => ([])
  176. },
  177. authorName: {
  178. type: String,
  179. default: 'Unknown'
  180. },
  181. authorId: {
  182. type: Number,
  183. default: 0
  184. },
  185. isPublished: {
  186. type: Boolean,
  187. default: false
  188. },
  189. toc: {
  190. type: Array,
  191. default: () => []
  192. }
  193. },
  194. data() {
  195. return {
  196. navShown: false,
  197. navExpanded: false,
  198. upBtnShown: false,
  199. scrollOpts: {
  200. duration: 1500,
  201. offset: -75,
  202. easing: 'easeInOutCubic'
  203. },
  204. scrollStyle: {
  205. vuescroll: {},
  206. scrollPanel: {
  207. initialScrollX: 0.01, // fix scrollbar not disappearing on load
  208. scrollingX: false,
  209. speed: 50
  210. },
  211. rail: {
  212. gutterOfEnds: '2px'
  213. },
  214. bar: {
  215. onlyShowBarOnScroll: false,
  216. background: '#42A5F5',
  217. hoverStyle: {
  218. background: '#64B5F6'
  219. }
  220. }
  221. }
  222. }
  223. },
  224. computed: {
  225. darkMode: get('site/dark'),
  226. isAuthenticated: get('user/authenticated'),
  227. rating: {
  228. get () {
  229. return 3.5
  230. },
  231. set (val) {
  232. }
  233. },
  234. breadcrumbs() {
  235. return [{ path: '/', name: 'Home' }].concat(_.reduce(this.path.split('/'), (result, value, key) => {
  236. result.push({
  237. path: _.map(result, 'path').join('/') + `/${value}`,
  238. name: value
  239. })
  240. return result
  241. }, []))
  242. }
  243. },
  244. created() {
  245. this.$store.commit('page/SET_AUTHOR_ID', this.authorId)
  246. this.$store.commit('page/SET_AUTHOR_NAME', this.authorName)
  247. this.$store.commit('page/SET_CREATED_AT', this.createdAt)
  248. this.$store.commit('page/SET_DESCRIPTION', this.description)
  249. this.$store.commit('page/SET_IS_PUBLISHED', this.isPublished)
  250. this.$store.commit('page/SET_ID', this.pageId)
  251. this.$store.commit('page/SET_LOCALE', this.locale)
  252. this.$store.commit('page/SET_PATH', this.path)
  253. this.$store.commit('page/SET_TAGS', this.tags)
  254. this.$store.commit('page/SET_TITLE', this.title)
  255. this.$store.commit('page/SET_UPDATED_AT', this.updatedAt)
  256. this.$store.commit('page/SET_MODE', 'view')
  257. },
  258. mounted () {
  259. Prism.highlightAllUnder(this.$refs.container)
  260. this.navShown = this.$vuetify.breakpoint.smAndUp
  261. },
  262. methods: {
  263. goHome () {
  264. window.location.assign('/')
  265. },
  266. toggleNavigation () {
  267. this.navOpen = !this.navOpen
  268. },
  269. upBtnScroll () {
  270. const scrollOffset = window.pageYOffset || document.documentElement.scrollTop
  271. this.upBtnShown = scrollOffset > window.innerHeight * 0.33
  272. }
  273. }
  274. }
  275. </script>
  276. <style lang="scss">
  277. .breadcrumbs-nav {
  278. .v-btn {
  279. min-width: 0;
  280. &__content {
  281. text-transform: none;
  282. }
  283. }
  284. .v-breadcrumbs__divider:nth-child(2n) {
  285. padding: 0 6px;
  286. }
  287. .v-breadcrumbs__divider:nth-child(2) {
  288. padding: 0 6px 0 12px;
  289. }
  290. }
  291. </style>