store.js 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. import Vue from 'vue'
  2. import Vuex from 'vuex'
  3. Vue.use(Vuex)
  4. const VUEX_PROPERTIES = ['state', 'getters', 'actions', 'mutations']
  5. let store = {};
  6. (function updateModules () {
  7. store = normalizeRoot(require('../store/index.js'), 'store/index.js')
  8. // If store is an exported method = classic mode (deprecated)
  9. // Enforce store modules
  10. store.modules = store.modules || {}
  11. resolveStoreModules(require('../store/config.js'), 'config.js')
  12. resolveStoreModules(require('../store/product.js'), 'product.js')
  13. // If the environment supports hot reloading...
  14. })()
  15. // createStore
  16. export const createStore = store instanceof Function ? store : () => {
  17. return new Vuex.Store(Object.assign({
  18. strict: (process.env.NODE_ENV !== 'production')
  19. }, store))
  20. }
  21. function normalizeRoot (moduleData, filePath) {
  22. moduleData = moduleData.default || moduleData
  23. if (moduleData.commit) {
  24. throw new Error(`[nuxt] ${filePath} should export a method that returns a Vuex instance.`)
  25. }
  26. if (typeof moduleData !== 'function') {
  27. // Avoid TypeError: setting a property that has only a getter when overwriting top level keys
  28. moduleData = Object.assign({}, moduleData)
  29. }
  30. return normalizeModule(moduleData, filePath)
  31. }
  32. function normalizeModule (moduleData, filePath) {
  33. if (moduleData.state && typeof moduleData.state !== 'function') {
  34. console.warn(`'state' should be a method that returns an object in ${filePath}`)
  35. const state = Object.assign({}, moduleData.state)
  36. // Avoid TypeError: setting a property that has only a getter when overwriting top level keys
  37. moduleData = Object.assign({}, moduleData, { state: () => state })
  38. }
  39. return moduleData
  40. }
  41. function resolveStoreModules (moduleData, filename) {
  42. moduleData = moduleData.default || moduleData
  43. // Remove store src + extension (./foo/index.js -> foo/index)
  44. const namespace = filename.replace(/\.(js|mjs)$/, '')
  45. const namespaces = namespace.split('/')
  46. let moduleName = namespaces[namespaces.length - 1]
  47. const filePath = `store/${filename}`
  48. moduleData = moduleName === 'state'
  49. ? normalizeState(moduleData, filePath)
  50. : normalizeModule(moduleData, filePath)
  51. // If src is a known Vuex property
  52. if (VUEX_PROPERTIES.includes(moduleName)) {
  53. const property = moduleName
  54. const propertyStoreModule = getStoreModule(store, namespaces, { isProperty: true })
  55. // Replace state since it's a function
  56. mergeProperty(propertyStoreModule, moduleData, property)
  57. return
  58. }
  59. // If file is foo/index.js, it should be saved as foo
  60. const isIndexModule = (moduleName === 'index')
  61. if (isIndexModule) {
  62. namespaces.pop()
  63. moduleName = namespaces[namespaces.length - 1]
  64. }
  65. const storeModule = getStoreModule(store, namespaces)
  66. for (const property of VUEX_PROPERTIES) {
  67. mergeProperty(storeModule, moduleData[property], property)
  68. }
  69. if (moduleData.namespaced === false) {
  70. delete storeModule.namespaced
  71. }
  72. }
  73. function normalizeState (moduleData, filePath) {
  74. if (typeof moduleData !== 'function') {
  75. console.warn(`${filePath} should export a method that returns an object`)
  76. const state = Object.assign({}, moduleData)
  77. return () => state
  78. }
  79. return normalizeModule(moduleData, filePath)
  80. }
  81. function getStoreModule (storeModule, namespaces, { isProperty = false } = {}) {
  82. // If ./mutations.js
  83. if (!namespaces.length || (isProperty && namespaces.length === 1)) {
  84. return storeModule
  85. }
  86. const namespace = namespaces.shift()
  87. storeModule.modules[namespace] = storeModule.modules[namespace] || {}
  88. storeModule.modules[namespace].namespaced = true
  89. storeModule.modules[namespace].modules = storeModule.modules[namespace].modules || {}
  90. return getStoreModule(storeModule.modules[namespace], namespaces, { isProperty })
  91. }
  92. function mergeProperty (storeModule, moduleData, property) {
  93. if (!moduleData) {
  94. return
  95. }
  96. if (property === 'state') {
  97. storeModule.state = moduleData || storeModule.state
  98. } else {
  99. storeModule[property] = Object.assign({}, storeModule[property], moduleData)
  100. }
  101. }