tree.ts 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. import { expect } from 'chai';
  2. import cloneDeep from 'lodash/cloneDeep';
  3. import { Tree, TreeNode, TreeRaw } from '../src/tree';
  4. function getIDList(nodes: TreeNode[]): string[] {
  5. return nodes.map(node => node.ID);
  6. }
  7. const rawData: TreeRaw[] = [
  8. { ID: '1', parentID: '-1', seq: 1 },
  9. { ID: '2', parentID: '-1', seq: 3 },
  10. { ID: '3', parentID: '-1', seq: 2 },
  11. { ID: '4', parentID: '-1', seq: 4 },
  12. { ID: '5', parentID: '-1', seq: 5 },
  13. { ID: '6', parentID: '1', seq: 2 },
  14. { ID: '7', parentID: '1', seq: 1 },
  15. { ID: '8', parentID: '7', seq: 1 },
  16. { ID: '9', parentID: '2', seq: 1 },
  17. ];
  18. const complicatedRawData: TreeRaw[] = [
  19. { ID: '1', parentID: '-1', seq: 1 },
  20. { ID: '2', parentID: '-1', seq: 4 },
  21. { ID: '3', parentID: '-1', seq: 3 },
  22. { ID: '4', parentID: '-1', seq: 5 },
  23. { ID: '5', parentID: '-1', seq: 6 },
  24. { ID: '6', parentID: '1', seq: 2 },
  25. { ID: '7', parentID: '1', seq: 1 },
  26. { ID: '8', parentID: '7', seq: 1 },
  27. { ID: '9', parentID: '2', seq: 1 },
  28. { ID: '10', parentID: '1', seq: 3 },
  29. { ID: '11', parentID: '1', seq: 4 },
  30. { ID: '12', parentID: '1', seq: 5 },
  31. { ID: '13', parentID: '1', seq: 6 },
  32. { ID: '14', parentID: '-1', seq: 2 },
  33. ];
  34. describe('Tree', () => {
  35. const tree = new Tree(cloneDeep(rawData));
  36. it('genData', () => {
  37. const IDList = tree.data.map(item => item.ID);
  38. expect(IDList).to.have.ordered.members([
  39. '1',
  40. '7',
  41. '8',
  42. '6',
  43. '3',
  44. '2',
  45. '9',
  46. '4',
  47. '5',
  48. ]);
  49. });
  50. it('find', () => {
  51. const node = tree.find('3');
  52. expect(node).to.have.property('ID', '3');
  53. });
  54. it('findParent', () => {
  55. const node = tree.findParent('6');
  56. expect(node).to.have.property('ID', '1');
  57. });
  58. it('findNext', () => {
  59. const node = tree.findNext('7');
  60. expect(node).to.have.property('ID', '6');
  61. });
  62. it('findPrev', () => {
  63. const node = tree.findPrev('6');
  64. expect(node).to.have.property('ID', '7');
  65. });
  66. it('getNodesPosterity', () => {
  67. const node1 = tree.find('1');
  68. const node2 = tree.find('2');
  69. if (node1 && node2) {
  70. const nodes = tree.getNodesPosterity([node1, node2]);
  71. const IDList = getIDList(nodes);
  72. expect(IDList).to.have.ordered.members(['1', '7', '8', '6', '2', '9']);
  73. const exNodes = tree.getNodesPosterity([node1, node2], false);
  74. const exIDList = getIDList(exNodes);
  75. expect(exIDList).to.have.ordered.members(['7', '8', '6', '9']);
  76. }
  77. });
  78. it('isOperable', () => {
  79. const node1 = tree.find('1');
  80. const node2 = tree.find('2');
  81. const node3 = tree.find('3');
  82. const node6 = tree.find('6');
  83. const node7 = tree.find('7');
  84. const node8 = tree.find('8');
  85. if (node1 && node2 && node3 && node6 && node7 && node8) {
  86. expect(tree.isOperable([node1, node2, node3])).to.be.equal(false);
  87. expect(tree.isOperable([node1, node3, node2])).to.be.equal(false);
  88. expect(tree.isOperable([node7, node6, node8])).to.be.equal(false);
  89. expect(tree.isOperable([node7, node8, node6])).to.be.equal(true);
  90. }
  91. });
  92. it('sameDepthNodes-empty-depth', () => {
  93. const nodes = tree.sameDepthNodes(tree.data);
  94. const IDList = getIDList(nodes);
  95. expect(IDList).to.have.ordered.members(['1', '3', '2', '4', '5']);
  96. });
  97. it('sameDepthNodes-specific-depth', () => {
  98. const nodes = tree.sameDepthNodes(tree.data, 1);
  99. const IDList = getIDList(nodes);
  100. expect(IDList).to.have.ordered.members(['7', '6', '9']);
  101. });
  102. });
  103. describe('Tree change', () => {
  104. it('insert', () => {
  105. const tree = new Tree(cloneDeep(rawData));
  106. const treeRaw: TreeRaw[] = [
  107. {
  108. ID: '10',
  109. parentID: '2',
  110. seq: 2,
  111. },
  112. {
  113. ID: '11',
  114. parentID: '7',
  115. seq: 2,
  116. },
  117. ];
  118. const updateData = tree.prepareInsert(treeRaw);
  119. tree.insert(treeRaw, updateData);
  120. const IDList = tree.data.map(item => item.ID);
  121. expect(IDList).to.have.ordered.members([
  122. '1',
  123. '7',
  124. '8',
  125. '11',
  126. '6',
  127. '3',
  128. '2',
  129. '9',
  130. '10',
  131. '4',
  132. '5',
  133. ]);
  134. });
  135. it('complicated-insert', () => {
  136. const tree = new Tree(cloneDeep(complicatedRawData));
  137. const treeRaw: TreeRaw[] = [
  138. {
  139. ID: '15',
  140. parentID: '7',
  141. seq: 1,
  142. },
  143. {
  144. ID: '16',
  145. parentID: '7',
  146. seq: 1,
  147. },
  148. {
  149. ID: '17',
  150. parentID: '7',
  151. seq: 2,
  152. },
  153. {
  154. ID: '18',
  155. parentID: '12',
  156. seq: 1,
  157. },
  158. {
  159. ID: '19',
  160. parentID: '18',
  161. seq: 1,
  162. },
  163. ];
  164. const updateData = tree.prepareInsert(treeRaw);
  165. tree.insert(treeRaw, updateData);
  166. const dataIDList = getIDList(tree.data);
  167. expect(dataIDList).to.have.ordered.members([
  168. '1',
  169. '7',
  170. '15',
  171. '16',
  172. '8',
  173. '17',
  174. '6',
  175. '10',
  176. '11',
  177. '12',
  178. '18',
  179. '19',
  180. '13',
  181. '14',
  182. '3',
  183. '2',
  184. '9',
  185. '4',
  186. '5',
  187. ]);
  188. expect(tree.parentMap).to.have.all.keys('-1', '1', '7', '12', '18', '2');
  189. const parentD1 = getIDList(tree.parentMap['-1']);
  190. expect(parentD1).to.have.ordered.members(['1', '14', '3', '2', '4', '5']);
  191. const parent1 = getIDList(tree.parentMap['1']);
  192. expect(parent1).to.have.ordered.members(['7', '6', '10', '11', '12', '13']);
  193. const parent7 = getIDList(tree.parentMap['7']);
  194. expect(parent7).to.have.ordered.members(['15', '16', '8', '17']);
  195. const parent12 = getIDList(tree.parentMap['12']);
  196. expect(parent12).to.have.ordered.members(['18']);
  197. const parent18 = getIDList(tree.parentMap['18']);
  198. expect(parent18).to.have.ordered.members(['19']);
  199. const parent2 = getIDList(tree.parentMap['2']);
  200. expect(parent2).to.have.ordered.members(['9']);
  201. });
  202. it('delete', () => {
  203. const tree = new Tree(cloneDeep(rawData));
  204. const deleteNodes: TreeNode[] = [tree.find('1') as TreeNode];
  205. // const updateData = tree.prepareDelete(deleteNodes);
  206. tree.delete(deleteNodes);
  207. const IDList = tree.data.map(item => item.ID);
  208. expect(IDList).to.have.ordered.members(['3', '2', '9', '4', '5']);
  209. });
  210. it('upMove', () => {
  211. const tree = new Tree(cloneDeep(rawData));
  212. const nodes = [tree.find('2'), tree.find('4')];
  213. const updateData = tree.prepareUpMove(nodes as TreeNode[]);
  214. tree.move(nodes as TreeNode[], updateData);
  215. const IDList = getIDList(tree.data);
  216. expect(IDList).to.have.ordered.members([
  217. '1',
  218. '7',
  219. '8',
  220. '6',
  221. '2',
  222. '9',
  223. '4',
  224. '3',
  225. '5',
  226. ]);
  227. expect(nodes[0]).to.have.property('seq', 2);
  228. expect(nodes[1]).to.have.property('seq', 3);
  229. const node3 = tree.find('3');
  230. expect(node3).to.have.property('seq', 4);
  231. });
  232. it('specialUpMove', () => {
  233. const tree = new Tree([
  234. { ID: '1', parentID: '-1', seq: 1 },
  235. { ID: '2', parentID: '-1', seq: 2 },
  236. { ID: '3', parentID: '-1', seq: 2 },
  237. { ID: '4', parentID: '-1', seq: 2 },
  238. { ID: '5', parentID: '-1', seq: 5 },
  239. ]);
  240. const nodes = [tree.find('3'), tree.find('4')];
  241. const updateData = tree.prepareUpMove(nodes as TreeNode[]);
  242. tree.move(nodes as TreeNode[], updateData);
  243. const IDList = getIDList(tree.data);
  244. expect(IDList).to.have.ordered.members(['1', '3', '4', '2', '5']);
  245. expect(nodes[0]).to.have.property('seq', 1);
  246. expect(nodes[1]).to.have.property('seq', 2);
  247. const node3 = tree.find('1');
  248. expect(node3).to.have.property('seq', 0);
  249. });
  250. it('downMove', () => {
  251. const tree = new Tree(cloneDeep(rawData));
  252. const nodes = [tree.find('3'), tree.find('2')];
  253. const updateData = tree.prepareDownMove(nodes as TreeNode[]);
  254. tree.move(nodes as TreeNode[], updateData);
  255. const IDList = getIDList(tree.data);
  256. expect(IDList).to.have.ordered.members([
  257. '1',
  258. '7',
  259. '8',
  260. '6',
  261. '4',
  262. '3',
  263. '2',
  264. '9',
  265. '5',
  266. ]);
  267. expect(nodes[0]).to.have.property('seq', 3);
  268. expect(nodes[1]).to.have.property('seq', 4);
  269. const node4 = tree.find('4');
  270. expect(node4).to.have.property('seq', 2);
  271. });
  272. it('specialDownMove', () => {
  273. const tree = new Tree([
  274. { ID: '1', parentID: '-1', seq: 1 },
  275. { ID: '2', parentID: '-1', seq: 2 },
  276. { ID: '3', parentID: '-1', seq: 2 },
  277. { ID: '4', parentID: '-1', seq: 4 },
  278. { ID: '5', parentID: '-1', seq: 5 },
  279. ]);
  280. const nodes = [tree.find('2')];
  281. const updateData = tree.prepareDownMove(nodes as TreeNode[]);
  282. tree.move(nodes as TreeNode[], updateData);
  283. const IDList = getIDList(tree.data);
  284. expect(IDList).to.have.ordered.members(['1', '3', '2', '4', '5']);
  285. expect(nodes[0]).to.have.property('seq', 2);
  286. const node3 = tree.find('3');
  287. expect(node3).to.have.property('seq', 1);
  288. });
  289. it('single-upLevel', () => {
  290. const tree = new Tree(cloneDeep(complicatedRawData));
  291. const node10 = tree.find('10');
  292. if (node10) {
  293. const updateData = tree.prepareUpLevel([node10]);
  294. tree.upLevel([node10], updateData);
  295. const dataIDList = getIDList(tree.data);
  296. expect(dataIDList).to.have.ordered.members([
  297. '1',
  298. '7',
  299. '8',
  300. '6',
  301. '10',
  302. '11',
  303. '12',
  304. '13',
  305. '14',
  306. '3',
  307. '2',
  308. '9',
  309. '4',
  310. '5',
  311. ]);
  312. expect(tree.parentMap).to.have.all.keys('-1', '1', '7', '10', '2');
  313. const parentD1 = getIDList(tree.parentMap['-1']);
  314. expect(parentD1).to.have.ordered.members([
  315. '1',
  316. '10',
  317. '14',
  318. '3',
  319. '2',
  320. '4',
  321. '5',
  322. ]);
  323. const parent1 = getIDList(tree.parentMap['1']);
  324. expect(parent1).to.have.ordered.members(['7', '6']);
  325. const parent7 = getIDList(tree.parentMap['7']);
  326. expect(parent7).to.have.ordered.members(['8']);
  327. const parent10 = getIDList(tree.parentMap['10']);
  328. expect(parent10).to.have.ordered.members(['11', '12', '13']);
  329. const parent2 = getIDList(tree.parentMap['2']);
  330. expect(parent2).to.have.ordered.members(['9']);
  331. }
  332. });
  333. it('multi-upLevel', () => {
  334. const tree = new Tree(cloneDeep(complicatedRawData));
  335. const node10 = tree.find('10');
  336. const node11 = tree.find('11');
  337. if (node10 && node11) {
  338. const updateData = tree.prepareUpLevel([node10, node11]);
  339. tree.upLevel([node10, node11], updateData);
  340. const dataIDList = getIDList(tree.data);
  341. expect(dataIDList).to.have.ordered.members([
  342. '1',
  343. '7',
  344. '8',
  345. '6',
  346. '10',
  347. '11',
  348. '12',
  349. '13',
  350. '14',
  351. '3',
  352. '2',
  353. '9',
  354. '4',
  355. '5',
  356. ]);
  357. expect(tree.parentMap).to.have.all.keys('-1', '1', '7', '11', '2');
  358. const parentD1 = getIDList(tree.parentMap['-1']);
  359. expect(parentD1).to.have.ordered.members([
  360. '1',
  361. '10',
  362. '11',
  363. '14',
  364. '3',
  365. '2',
  366. '4',
  367. '5',
  368. ]);
  369. const parent1 = getIDList(tree.parentMap['1']);
  370. expect(parent1).to.have.ordered.members(['7', '6']);
  371. const parent7 = getIDList(tree.parentMap['7']);
  372. expect(parent7).to.have.ordered.members(['8']);
  373. const parent11 = getIDList(tree.parentMap['11']);
  374. expect(parent11).to.have.ordered.members(['12', '13']);
  375. const parent2 = getIDList(tree.parentMap['2']);
  376. expect(parent2).to.have.ordered.members(['9']);
  377. }
  378. });
  379. it('single-downLevel', () => {
  380. const tree = new Tree(cloneDeep(complicatedRawData));
  381. const node10 = tree.find('10');
  382. if (node10) {
  383. const updateData = tree.prepareDownLevel([node10]);
  384. tree.downLevel([node10], updateData);
  385. const dataIDList = getIDList(tree.data);
  386. expect(dataIDList).to.have.ordered.members([
  387. '1',
  388. '7',
  389. '8',
  390. '6',
  391. '10',
  392. '11',
  393. '12',
  394. '13',
  395. '14',
  396. '3',
  397. '2',
  398. '9',
  399. '4',
  400. '5',
  401. ]);
  402. expect(tree.parentMap).to.have.all.keys('-1', '1', '7', '6', '2');
  403. const parentD1 = getIDList(tree.parentMap['-1']);
  404. expect(parentD1).to.have.ordered.members(['1', '14', '3', '2', '4', '5']);
  405. const parent1 = getIDList(tree.parentMap['1']);
  406. expect(parent1).to.have.ordered.members(['7', '6', '11', '12', '13']);
  407. const parent7 = getIDList(tree.parentMap['7']);
  408. expect(parent7).to.have.ordered.members(['8']);
  409. const parent11 = getIDList(tree.parentMap['6']);
  410. expect(parent11).to.have.ordered.members(['10']);
  411. const parent2 = getIDList(tree.parentMap['2']);
  412. expect(parent2).to.have.ordered.members(['9']);
  413. }
  414. });
  415. it('multi-downLevel', () => {
  416. const tree = new Tree(cloneDeep(complicatedRawData));
  417. const node6 = tree.find('6');
  418. const node10 = tree.find('10');
  419. if (node6 && node10) {
  420. const updateData = tree.prepareDownLevel([node6, node10]);
  421. tree.downLevel([node6, node10], updateData);
  422. const dataIDList = getIDList(tree.data);
  423. expect(dataIDList).to.have.ordered.members([
  424. '1',
  425. '7',
  426. '8',
  427. '6',
  428. '10',
  429. '11',
  430. '12',
  431. '13',
  432. '14',
  433. '3',
  434. '2',
  435. '9',
  436. '4',
  437. '5',
  438. ]);
  439. expect(tree.parentMap).to.have.all.keys('-1', '1', '7', '2');
  440. const parentD1 = getIDList(tree.parentMap['-1']);
  441. expect(parentD1).to.have.ordered.members(['1', '14', '3', '2', '4', '5']);
  442. const parent1 = getIDList(tree.parentMap['1']);
  443. expect(parent1).to.have.ordered.members(['7', '11', '12', '13']);
  444. const parent7 = getIDList(tree.parentMap['7']);
  445. expect(parent7).to.have.ordered.members(['8', '6', '10']);
  446. const parent2 = getIDList(tree.parentMap['2']);
  447. expect(parent2).to.have.ordered.members(['9']);
  448. }
  449. });
  450. });