utils.js 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. import { hierarchy } from 'd3-hierarchy'
  2. export const RELATIONS = [
  3. 'parents',
  4. 'siblings',
  5. 'children'
  6. ]
  7. export const DATA_LEVELS = {
  8. initial: 0,
  9. partial: 1,
  10. full: 2
  11. }
  12. export const ID_VARIANTS = {
  13. 9: 'depart',
  14. 22: 'critique',
  15. 63: 'echo',
  16. 6: 'reflexion',
  17. 7: 'lecture',
  18. 8: 'sensible',
  19. 23: 'kit',
  20. 0: 'creation'
  21. }
  22. export const VARIANT_IDS = Object.fromEntries(
  23. Object.entries(ID_VARIANTS).map(([key, value]) => [value, parseInt(key)])
  24. )
  25. export const SEARCH_KEYS = ['content', 'title', 'preTitle', 'authors']
  26. export function parseNodesQueryParams ({ mode, nodes = [] }) {
  27. let nodebook = typeof nodes === 'string' ? [nodes] : nodes
  28. nodebook = nodebook.map(ids => {
  29. return typeof ids === 'string'
  30. ? ids.split(',').map(id => parseInt(id))
  31. : [...ids]
  32. })
  33. return { mode, nodebook }
  34. }
  35. export function reduceQueryLevels (lvl, lowestLvl = -1) {
  36. return Array(lvl - lowestLvl).fill(lowestLvl + 1).map((v, i) => v + i)
  37. }
  38. export function reduceQueryIds (ids, dataLevel, data) {
  39. const lvl = DATA_LEVELS[dataLevel]
  40. const idsToQuery = []
  41. let lowestLvl = lvl
  42. for (const id of ids) {
  43. const nodeLvl = id in data ? data[id].dataLevel : -1
  44. if (nodeLvl < lvl) {
  45. idsToQuery.push(id)
  46. }
  47. if (nodeLvl < lowestLvl) {
  48. lowestLvl = nodeLvl
  49. }
  50. }
  51. return idsToQuery.length === 0 ? {} : {
  52. idsToQuery,
  53. levelsToQuery: reduceQueryLevels(lvl, lowestLvl)
  54. }
  55. }
  56. export function getUniqueNodesIds (tree) {
  57. function extractId (ids, node) {
  58. ids.add(node.id)
  59. for (const relation of RELATIONS) {
  60. if (relation in node && node[relation]) {
  61. for (const subNode of node[relation]) {
  62. extractId(ids, subNode)
  63. }
  64. }
  65. }
  66. return ids
  67. }
  68. return Array.from(extractId(new Set(), tree))
  69. }
  70. export function getRelatedNodesIds (nodes, ignored) {
  71. const ids = new Set()
  72. for (const node of nodes) {
  73. for (const relation of RELATIONS) {
  74. if (relation in node && node[relation]) {
  75. node[relation].forEach(({ id }) => {
  76. if (ignored.includes(id)) return
  77. ids.add(id)
  78. })
  79. }
  80. }
  81. }
  82. return Array.from(ids)
  83. }
  84. export function getRandomIds (ids, count = 3) {
  85. const randomIds = []
  86. while (randomIds.length < count) {
  87. const id = ids[Math.floor(Math.random() * ids.length)]
  88. if (!randomIds.includes(id)) {
  89. randomIds.push(id)
  90. }
  91. }
  92. return randomIds
  93. }
  94. export function parseTree (treeData) {
  95. if ('creations' in treeData) {
  96. if (treeData.creations) {
  97. if (!Array.isArray(treeData.children)) {
  98. treeData.children = []
  99. }
  100. treeData.creations.forEach(item => {
  101. item.variant = [{ id: 0 }]
  102. treeData.children.push(item)
  103. })
  104. }
  105. delete treeData.creations
  106. }
  107. for (const relation of RELATIONS) {
  108. if (relation in treeData && treeData[relation]) {
  109. treeData[relation] = treeData[relation].map(node => parseTree(node))
  110. }
  111. }
  112. return treeData
  113. }
  114. export function buildTree (treeData, nodesData) {
  115. const uniqueIds = []
  116. const nodes = []
  117. const links = []
  118. const h = hierarchy(treeData, (node) => {
  119. return RELATIONS.reduce((acc, relation) => {
  120. if (node[relation]) {
  121. node[relation].forEach((item, i) => {
  122. item.linkType = relation
  123. })
  124. return [...acc, ...node[relation]]
  125. }
  126. return acc
  127. }, [])
  128. })
  129. h.each(node => {
  130. node.id = node.data.id
  131. if (!uniqueIds.includes(node.id)) {
  132. uniqueIds.push(node.id)
  133. node.data = nodesData.find(n => n.id === node.id)
  134. // Add `x` and `y` keys so Vue can update on these values updates
  135. Object.assign(node, { x: undefined, y: undefined })
  136. nodes.push(node)
  137. if (node.children) {
  138. node.children.forEach(child => {
  139. const asSource = links.find(link => link.source === node.id && link.target === child.data.id)
  140. const asTarget = links.find(link => link.source === child.data.id && link.target === node.id)
  141. if (!asSource && !asTarget) {
  142. links.push({ source: node.id, target: child.data.id, linkType: child.data.linkType })
  143. }
  144. })
  145. }
  146. }
  147. })
  148. return {
  149. nodes,
  150. links
  151. }
  152. }
  153. export function searchInNode (search, node) {
  154. if (!search) return true
  155. for (const key of SEARCH_KEYS) {
  156. if (!node[key]) continue
  157. let match = false
  158. if (key === 'authors') {
  159. match = node[key].some(author => author.name.toLowerCase().includes(search))
  160. } else {
  161. match = node[key].toLowerCase().includes(search)
  162. }
  163. if (match) return true
  164. }
  165. return false
  166. }
  167. export function tagsInNode (tags, nodeTags) {
  168. if (tags.length === 0) return true
  169. if (!nodeTags || nodeTags.length === 0) return false
  170. return tags.every(tag => nodeTags.some(nodeTag => nodeTag.name === tag))
  171. }
  172. export function getRelation (node) {
  173. if (node.type === 'prod' && node.parents && node.parents.length) {
  174. return { parentId: node.parents[0].id, childId: node.id }
  175. }
  176. return { parentId: node.id }
  177. }
  178. export function getStrangenessOpacity (currentStrangeness, node) {
  179. if (currentStrangeness === undefined) return 1
  180. const strangeness = (
  181. node.type === 'prod' && node.parents && node.parents.length ? node.parents[0] : node
  182. ).strangeness
  183. return 1 / (Math.abs(strangeness - currentStrangeness) + 1)
  184. }