App.js 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. import Vue from 'vue'
  2. import { decode, parsePath, withoutBase, withoutTrailingSlash, normalizeURL } from 'ufo'
  3. import { getMatchedComponentsInstances, getChildrenComponentInstancesUsingFetch, promisify, globalHandleError, urlJoin, sanitizeComponent } from './utils'
  4. import NuxtError from '../layouts/error.vue'
  5. import '../node_modules/element-ui/lib/theme-chalk/index.css'
  6. import '../assets/iconfont/iconfont.css'
  7. import '../assets/css/reset.scss'
  8. import '../assets/css/common.scss'
  9. import '../assets/css/element.scss'
  10. import '../assets/css/flex-custom.scss'
  11. import _c991692a from '../layouts/blank_layout.vue'
  12. import _6f6c098b from '../layouts/default.vue'
  13. import _32b9add8 from '../layouts/product_builder_layout.vue'
  14. const layouts = { "_blank_layout": sanitizeComponent(_c991692a),"_default": sanitizeComponent(_6f6c098b),"_product_builder_layout": sanitizeComponent(_32b9add8) }
  15. export default {
  16. render (h, props) {
  17. const layoutEl = h(this.layout || 'nuxt')
  18. const templateEl = h('div', {
  19. domProps: {
  20. id: '__layout'
  21. },
  22. key: this.layoutName
  23. }, [layoutEl])
  24. const transitionEl = h('transition', {
  25. props: {
  26. name: 'layout',
  27. mode: 'out-in'
  28. },
  29. on: {
  30. beforeEnter (el) {
  31. // Ensure to trigger scroll event after calling scrollBehavior
  32. window.$nuxt.$nextTick(() => {
  33. window.$nuxt.$emit('triggerScroll')
  34. })
  35. }
  36. }
  37. }, [templateEl])
  38. return h('div', {
  39. domProps: {
  40. id: '__nuxt'
  41. }
  42. }, [
  43. transitionEl
  44. ])
  45. },
  46. data: () => ({
  47. isOnline: true,
  48. layout: null,
  49. layoutName: '',
  50. nbFetching: 0
  51. }),
  52. beforeCreate () {
  53. Vue.util.defineReactive(this, 'nuxt', this.$options.nuxt)
  54. },
  55. created () {
  56. // Add this.$nuxt in child instances
  57. this.$root.$options.$nuxt = this
  58. if (process.client) {
  59. // add to window so we can listen when ready
  60. window.$nuxt = this
  61. this.refreshOnlineStatus()
  62. // Setup the listeners
  63. window.addEventListener('online', this.refreshOnlineStatus)
  64. window.addEventListener('offline', this.refreshOnlineStatus)
  65. }
  66. // Add $nuxt.error()
  67. this.error = this.nuxt.error
  68. // Add $nuxt.context
  69. this.context = this.$options.context
  70. },
  71. watch: {
  72. 'nuxt.err': 'errorChanged'
  73. },
  74. computed: {
  75. isOffline () {
  76. return !this.isOnline
  77. },
  78. isFetching () {
  79. return this.nbFetching > 0
  80. },
  81. },
  82. methods: {
  83. refreshOnlineStatus () {
  84. if (process.client) {
  85. if (typeof window.navigator.onLine === 'undefined') {
  86. // If the browser doesn't support connection status reports
  87. // assume that we are online because most apps' only react
  88. // when they now that the connection has been interrupted
  89. this.isOnline = true
  90. } else {
  91. this.isOnline = window.navigator.onLine
  92. }
  93. }
  94. },
  95. async refresh () {
  96. const pages = getMatchedComponentsInstances(this.$route)
  97. if (!pages.length) {
  98. return
  99. }
  100. const promises = pages.map(async (page) => {
  101. let p = []
  102. // Old fetch
  103. if (page.$options.fetch && page.$options.fetch.length) {
  104. p.push(promisify(page.$options.fetch, this.context))
  105. }
  106. if (page.$options.asyncData) {
  107. p.push(
  108. promisify(page.$options.asyncData, this.context)
  109. .then((newData) => {
  110. for (const key in newData) {
  111. Vue.set(page.$data, key, newData[key])
  112. }
  113. })
  114. )
  115. }
  116. // Wait for asyncData & old fetch to finish
  117. await Promise.all(p)
  118. // Cleanup refs
  119. p = []
  120. if (page.$fetch) {
  121. p.push(page.$fetch())
  122. }
  123. // Get all component instance to call $fetch
  124. for (const component of getChildrenComponentInstancesUsingFetch(page.$vnode.componentInstance)) {
  125. p.push(component.$fetch())
  126. }
  127. return Promise.all(p)
  128. })
  129. try {
  130. await Promise.all(promises)
  131. } catch (error) {
  132. globalHandleError(error)
  133. this.error(error)
  134. }
  135. },
  136. errorChanged () {
  137. if (this.nuxt.err) {
  138. let errorLayout = (NuxtError.options || NuxtError).layout;
  139. if (typeof errorLayout === 'function') {
  140. errorLayout = errorLayout(this.context)
  141. }
  142. this.setLayout(errorLayout)
  143. }
  144. },
  145. setLayout (layout) {
  146. if (!layout || !layouts['_' + layout]) {
  147. layout = 'default'
  148. }
  149. this.layoutName = layout
  150. this.layout = layouts['_' + layout]
  151. return this.layout
  152. },
  153. loadLayout (layout) {
  154. if (!layout || !layouts['_' + layout]) {
  155. layout = 'default'
  156. }
  157. return Promise.resolve(layouts['_' + layout])
  158. },
  159. },
  160. }