library.js 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. import Vue from 'vue'
  2. import router from '@/router'
  3. import api from '@/api'
  4. import {
  5. AllTags
  6. } from '@/api/queries'
  7. import {
  8. parseNodesQueryParams,
  9. getUniqueNodesIds,
  10. getRelatedNodesIds,
  11. buildTree
  12. } from '@/store/utils.js'
  13. export default {
  14. state: {
  15. trees: {},
  16. nodebook: undefined,
  17. // LibraryOptions options
  18. strangeness: [0, 1, 2, 3, 4],
  19. tagsOptions: [],
  20. // LibraryOptions state
  21. mode: undefined,
  22. nodeDepartId: undefined,
  23. search: ''
  24. },
  25. mutations: {
  26. // ╷ ╶┬╴┌─╮┌─╮╭─┐┌─╮╷ ╷
  27. // │ │ │╶┤├┬╯├─┤├┬╯╰─┤
  28. // ╰─╴╶┴╴└─╯╵ ╰╵ ╵╵ ╰╶─╯
  29. 'SET_NODEBOOK' (state, nodes) {
  30. state.nodebook = nodes
  31. },
  32. 'ADD_NODEBOOK_NODE' (state, [stack, parentId, childId]) {
  33. if (stack === undefined) {
  34. stack = childId === undefined ? [parentId] : [parentId, childId]
  35. state.nodebook.push(stack)
  36. } else {
  37. stack.push(childId)
  38. }
  39. },
  40. 'REMOVE_NODEBOOK_NODE' (state, [stack, nodeId]) {
  41. if (stack[0] === nodeId) {
  42. state.nodebook.splice(state.nodebook.indexOf(stack), 1)
  43. } else {
  44. stack.splice(stack.indexOf(nodeId), 1)
  45. }
  46. },
  47. 'ADD_NODE_TREE' (state, [id, tree]) {
  48. Vue.set(state.trees, id, tree)
  49. },
  50. // ╭─╮┌─╮╶┬╴╶┬╴╭─╮╭╮╷╭─╴
  51. // │ │├─╯ │ │ │ ││││╰─╮
  52. // ╰─╯╵ ╵ ╶┴╴╰─╯╵╰╯╶─╯
  53. 'SET_TAGS_OPTIONS' (state, tags) {
  54. state.tagsOptions = tags.map(tag => ({ value: tag.id, text: tag.name }))
  55. },
  56. 'SET_MODE' (state, mode) {
  57. state.mode = mode
  58. },
  59. 'SET_NODE_DEPART_ID' (state, id) {
  60. state.nodeDepartId = id
  61. },
  62. 'SET_SEARCH' (state, str) {
  63. state.search = str
  64. }
  65. },
  66. actions: {
  67. // ╷ ╶┬╴┌─╮┌─╮╭─┐┌─╮╷ ╷
  68. // │ │ │╶┤├┬╯├─┤├┬╯╰─┤
  69. // ╰─╴╶┴╴└─╯╵ ╰╵ ╵╵ ╰╶─╯
  70. async 'INIT_LIBRARY' ({ state, commit, dispatch, rootState }) {
  71. const departIds = await dispatch('GET_ALL_NODES_IDS', 'depart')
  72. await dispatch('GET_NODES', { ids: departIds, dataLevel: 'initial' })
  73. dispatch('GET_ALL_TAGS')
  74. return departIds
  75. },
  76. async 'INIT_LIBRARY_TREE' ({ state, commit, dispatch, rootState }) {
  77. const ids = await dispatch('INIT_LIBRARY')
  78. return dispatch('SET_NODE_DEPART_ID', ids[ids.length - 1])
  79. },
  80. async 'INIT_LIBRARY_MAP' ({ state, commit, dispatch }) {
  81. const departIds = await dispatch('INIT_LIBRARY')
  82. const departNodes = await dispatch('GET_NODES', { ids: departIds.slice(0, 3), dataLevel: 'partial' })
  83. const relatedIds = getRelatedNodesIds(departNodes)
  84. const relatedNodes = await dispatch('GET_NODES', { ids: relatedIds, dataLevel: 'partial' })
  85. return [...departNodes, ...relatedNodes]
  86. },
  87. async 'INIT_LIBRARY_LIST' ({ state, commit, dispatch }) {
  88. const departIds = await dispatch('INIT_LIBRARY')
  89. return dispatch('GET_NODES', { ids: departIds, dataLevel: 'partial' })
  90. },
  91. async 'UPDATE_NODEBOOK' ({ state, commit, dispatch }, query) {
  92. const { mode = 'tree', nodebook } = parseNodesQueryParams(query)
  93. commit('SET_MODE', mode)
  94. await dispatch('GET_NODES', { ids: [].concat(...nodebook), dataLevel: 'full' })
  95. commit('SET_NODEBOOK', nodebook)
  96. },
  97. // ╭╮╷╭─╮┌─╮┌─╴╭─╴
  98. // ││││ ││ │├─╴╰─╮
  99. // ╵╰╯╰─╯└─╯╰─╴╶─╯
  100. 'UPDATE_QUERY_NODES' ({ state }, from) {
  101. const query = {
  102. mode: state.mode,
  103. nodes: [...state.nodebook.map(ids => ids.join(','))]
  104. }
  105. router.push({ query })
  106. },
  107. async 'OPEN_NODE' ({ state, commit, dispatch }, [parentId, childId]) {
  108. const stack = state.nodebook.find(stack => stack[0] === parentId)
  109. if (stack && (childId === undefined || stack.includes(childId))) return
  110. commit('ADD_NODEBOOK_NODE', [stack, parentId, childId])
  111. dispatch('UPDATE_QUERY_NODES')
  112. },
  113. 'CLOSE_NODE' ({ state, commit, dispatch }, [parentId, childId]) {
  114. const stack = state.nodebook.find(stack => stack.includes(parentId) && (childId ? stack.includes(childId) : true))
  115. commit('REMOVE_NODEBOOK_NODE', [stack, childId || parentId])
  116. dispatch('UPDATE_QUERY_NODES')
  117. },
  118. // ╭─╮┌─╮╶┬╴╶┬╴╭─╮╭╮╷╭─╴
  119. // │ │├─╯ │ │ │ ││││╰─╮
  120. // ╰─╯╵ ╵ ╶┴╴╰─╯╵╰╯╶─╯
  121. 'GET_ALL_TAGS' ({ state, commit }) {
  122. if (state.tags !== undefined) return state.tags
  123. return api.query(AllTags).then(data => {
  124. commit('SET_TAGS_OPTIONS', data.tags)
  125. return state.tags
  126. })
  127. },
  128. async 'SET_NODE_DEPART_ID' ({ state, rootState, commit, dispatch, getters }, id) {
  129. if (state.nodeDepartId === id) return
  130. commit('SET_NODE_DEPART_ID', id)
  131. if (!(id in state.trees)) {
  132. const treeData = await api.queryRecursiveNodes([id])
  133. const nodes = await dispatch('GET_NODES', { ids: getUniqueNodesIds(treeData), dataLevel: 'initial' })
  134. commit('ADD_NODE_TREE', [id, buildTree(treeData, nodes)])
  135. }
  136. return getters.nodeTree
  137. },
  138. 'UPDATE_QUERY_MODE' (store, mode) {
  139. if (router.currentRoute.query.mode === mode) return
  140. router.push({ query: { mode } })
  141. }
  142. },
  143. getters: {
  144. // LibraryOptions state
  145. mode: state => state.mode,
  146. nodeDepartId: state => state.nodeDepartId,
  147. search: state => state.search,
  148. // LibraryOptions options
  149. tagsOptions: state => state.tagsOptions,
  150. nodesDepartsOptions: (state, getters, rootState) => {
  151. const departIds = rootState.ids.depart
  152. if (departIds === undefined || rootState.nodes[departIds[0]] === undefined) return
  153. return departIds.map(id => {
  154. const text = rootState.nodes[id]
  155. const firstAuthor = text.authors !== null ? text.authors[0].name : 'Pad de noms'
  156. return {
  157. text: `${firstAuthor}, ${text.title} (${id})`,
  158. value: id
  159. }
  160. })
  161. },
  162. // Library
  163. nodebook: (state, getters, rootState) => {
  164. if (!state.nodebook) return []
  165. return state.nodebook.map(([parentId, ...childrenIds]) => {
  166. return {
  167. parent: rootState.nodes[parentId],
  168. children: childrenIds.map(id => rootState.nodes[id])
  169. }
  170. })
  171. },
  172. // LibraryList
  173. orderedTextsDepart: (state, getters, rootState) => {
  174. const departIds = rootState.ids.depart
  175. if (departIds === undefined || rootState.nodes[departIds[0]] === undefined) return
  176. return rootState.ids.depart.map(id => rootState.nodes[id]).sort((a, b) => {
  177. if (!b.authors) return -1
  178. if (!a.authors) return +1
  179. if (a.authors[0].last_name < b.authors[0].last_name) return -1
  180. if (a.authors[0].last_name > b.authors[0].last_name) return 1
  181. return 0
  182. }).reduce((dict, text) => {
  183. const firstChar = text.authors ? text.authors[0].last_name[0] : 'Pas de noms'
  184. if (!(firstChar in dict)) dict[firstChar] = []
  185. dict[firstChar].push(text)
  186. return dict
  187. }, {})
  188. },
  189. // LibraryTree
  190. nodeTree: (state, rootState) => {
  191. return state.trees[state.nodeDepartId] || { nodes: [], links: [] }
  192. },
  193. // Commons
  194. activeNodes: state => {
  195. if (!state.nodebook) return []
  196. return state.nodebook.reduce((acc, ids) => [...acc, ...ids], [])
  197. }
  198. }
  199. }