You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

278 lines
8.3 KiB

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