children.spec.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469
  1. import { patch } from 'web/runtime/patch'
  2. import VNode from 'core/vdom/vnode'
  3. function prop (name) {
  4. return obj => { return obj[name] }
  5. }
  6. function map (fn, list) {
  7. const ret = []
  8. for (let i = 0; i < list.length; i++) {
  9. ret[i] = fn(list[i])
  10. }
  11. return ret
  12. }
  13. function spanNum (n) {
  14. if (typeof n === 'string') {
  15. return new VNode('span', {}, undefined, n)
  16. } else {
  17. return new VNode('span', { key: n }, undefined, n.toString())
  18. }
  19. }
  20. function shuffle (array) {
  21. let currentIndex = array.length
  22. let temporaryValue
  23. let randomIndex
  24. // while there remain elements to shuffle...
  25. while (currentIndex !== 0) {
  26. // pick a remaining element...
  27. randomIndex = Math.floor(Math.random() * currentIndex)
  28. currentIndex -= 1
  29. // and swap it with the current element.
  30. temporaryValue = array[currentIndex]
  31. array[currentIndex] = array[randomIndex]
  32. array[randomIndex] = temporaryValue
  33. }
  34. return array
  35. }
  36. const inner = prop('innerHTML')
  37. const tag = prop('tagName')
  38. describe('children', () => {
  39. let vnode0
  40. beforeEach(() => {
  41. vnode0 = new VNode('p', { attrs: { id: '1' }}, [createTextVNode('hello world')])
  42. patch(null, vnode0)
  43. })
  44. it('should appends elements', () => {
  45. const vnode1 = new VNode('p', {}, [1].map(spanNum))
  46. const vnode2 = new VNode('p', {}, [1, 2, 3].map(spanNum))
  47. let elm = patch(vnode0, vnode1)
  48. expect(elm.children.length).toBe(1)
  49. elm = patch(vnode1, vnode2)
  50. expect(elm.children.length).toBe(3)
  51. expect(elm.children[1].innerHTML).toBe('2')
  52. expect(elm.children[2].innerHTML).toBe('3')
  53. })
  54. it('should prepends elements', () => {
  55. const vnode1 = new VNode('p', {}, [4, 5].map(spanNum))
  56. const vnode2 = new VNode('p', {}, [1, 2, 3, 4, 5].map(spanNum))
  57. let elm = patch(vnode0, vnode1)
  58. expect(elm.children.length).toBe(2)
  59. elm = patch(vnode1, vnode2)
  60. expect(map(inner, elm.children)).toEqual(['1', '2', '3', '4', '5'])
  61. })
  62. it('should add elements in the middle', () => {
  63. const vnode1 = new VNode('p', {}, [1, 2, 4, 5].map(spanNum))
  64. const vnode2 = new VNode('p', {}, [1, 2, 3, 4, 5].map(spanNum))
  65. let elm = patch(vnode0, vnode1)
  66. expect(elm.children.length).toBe(4)
  67. elm = patch(vnode1, vnode2)
  68. expect(map(inner, elm.children)).toEqual(['1', '2', '3', '4', '5'])
  69. })
  70. it('should add elements at begin and end', () => {
  71. const vnode1 = new VNode('p', {}, [2, 3, 4].map(spanNum))
  72. const vnode2 = new VNode('p', {}, [1, 2, 3, 4, 5].map(spanNum))
  73. let elm = patch(vnode0, vnode1)
  74. expect(elm.children.length).toBe(3)
  75. elm = patch(vnode1, vnode2)
  76. expect(map(inner, elm.children)).toEqual(['1', '2', '3', '4', '5'])
  77. })
  78. it('should add children to parent with no children', () => {
  79. const vnode1 = new VNode('p', { key: 'p' })
  80. const vnode2 = new VNode('p', { key: 'p' }, [1, 2, 3].map(spanNum))
  81. let elm = patch(vnode0, vnode1)
  82. expect(elm.children.length).toBe(0)
  83. elm = patch(vnode1, vnode2)
  84. expect(map(inner, elm.children)).toEqual(['1', '2', '3'])
  85. })
  86. it('should remove all children from parent', () => {
  87. const vnode1 = new VNode('p', { key: 'p' }, [1, 2, 3].map(spanNum))
  88. const vnode2 = new VNode('p', { key: 'p' })
  89. let elm = patch(vnode0, vnode1)
  90. expect(map(inner, elm.children)).toEqual(['1', '2', '3'])
  91. elm = patch(vnode1, vnode2)
  92. expect(elm.children.length).toBe(0)
  93. })
  94. it('should remove elements from the beginning', () => {
  95. const vnode1 = new VNode('p', {}, [1, 2, 3, 4, 5].map(spanNum))
  96. const vnode2 = new VNode('p', {}, [3, 4, 5].map(spanNum))
  97. let elm = patch(vnode0, vnode1)
  98. expect(elm.children.length).toBe(5)
  99. elm = patch(vnode1, vnode2)
  100. expect(map(inner, elm.children)).toEqual(['3', '4', '5'])
  101. })
  102. it('should removes elements from end', () => {
  103. const vnode1 = new VNode('p', {}, [1, 2, 3, 4, 5].map(spanNum))
  104. const vnode2 = new VNode('p', {}, [1, 2, 3].map(spanNum))
  105. let elm = patch(vnode0, vnode1)
  106. expect(elm.children.length).toBe(5)
  107. elm = patch(vnode1, vnode2)
  108. expect(elm.children.length).toBe(3)
  109. expect(elm.children[0].innerHTML).toBe('1')
  110. expect(elm.children[1].innerHTML).toBe('2')
  111. expect(elm.children[2].innerHTML).toBe('3')
  112. })
  113. it('should remove elements from the middle', () => {
  114. const vnode1 = new VNode('p', {}, [1, 2, 3, 4, 5].map(spanNum))
  115. const vnode2 = new VNode('p', {}, [1, 2, 4, 5].map(spanNum))
  116. let elm = patch(vnode0, vnode1)
  117. expect(elm.children.length).toBe(5)
  118. elm = patch(vnode1, vnode2)
  119. expect(elm.children.length).toBe(4)
  120. expect(elm.children[0].innerHTML).toBe('1')
  121. expect(elm.children[1].innerHTML).toBe('2')
  122. expect(elm.children[2].innerHTML).toBe('4')
  123. expect(elm.children[3].innerHTML).toBe('5')
  124. })
  125. it('should moves element forward', () => {
  126. const vnode1 = new VNode('p', {}, [1, 2, 3, 4].map(spanNum))
  127. const vnode2 = new VNode('p', {}, [2, 3, 1, 4].map(spanNum))
  128. let elm = patch(vnode0, vnode1)
  129. expect(elm.children.length).toBe(4)
  130. elm = patch(vnode1, vnode2)
  131. expect(elm.children.length).toBe(4)
  132. expect(elm.children[0].innerHTML).toBe('2')
  133. expect(elm.children[1].innerHTML).toBe('3')
  134. expect(elm.children[2].innerHTML).toBe('1')
  135. expect(elm.children[3].innerHTML).toBe('4')
  136. })
  137. it('should move elements to end', () => {
  138. const vnode1 = new VNode('p', {}, [1, 2, 3].map(spanNum))
  139. const vnode2 = new VNode('p', {}, [2, 3, 1].map(spanNum))
  140. let elm = patch(vnode0, vnode1)
  141. expect(elm.children.length).toBe(3)
  142. elm = patch(vnode1, vnode2)
  143. expect(elm.children.length).toBe(3)
  144. expect(elm.children[0].innerHTML).toBe('2')
  145. expect(elm.children[1].innerHTML).toBe('3')
  146. expect(elm.children[2].innerHTML).toBe('1')
  147. })
  148. it('should move element backwards', () => {
  149. const vnode1 = new VNode('p', {}, [1, 2, 3, 4].map(spanNum))
  150. const vnode2 = new VNode('p', {}, [1, 4, 2, 3].map(spanNum))
  151. let elm = patch(vnode0, vnode1)
  152. expect(elm.children.length).toBe(4)
  153. elm = patch(vnode1, vnode2)
  154. expect(elm.children.length).toBe(4)
  155. expect(elm.children[0].innerHTML).toBe('1')
  156. expect(elm.children[1].innerHTML).toBe('4')
  157. expect(elm.children[2].innerHTML).toBe('2')
  158. expect(elm.children[3].innerHTML).toBe('3')
  159. })
  160. it('should swap first and last', () => {
  161. const vnode1 = new VNode('p', {}, [1, 2, 3, 4].map(spanNum))
  162. const vnode2 = new VNode('p', {}, [4, 2, 3, 1].map(spanNum))
  163. let elm = patch(vnode0, vnode1)
  164. expect(elm.children.length).toBe(4)
  165. elm = patch(vnode1, vnode2)
  166. expect(elm.children.length).toBe(4)
  167. expect(elm.children[0].innerHTML).toBe('4')
  168. expect(elm.children[1].innerHTML).toBe('2')
  169. expect(elm.children[2].innerHTML).toBe('3')
  170. expect(elm.children[3].innerHTML).toBe('1')
  171. })
  172. it('should move to left and replace', () => {
  173. const vnode1 = new VNode('p', {}, [1, 2, 3, 4, 5].map(spanNum))
  174. const vnode2 = new VNode('p', {}, [4, 1, 2, 3, 6].map(spanNum))
  175. let elm = patch(vnode0, vnode1)
  176. expect(elm.children.length).toBe(5)
  177. elm = patch(vnode1, vnode2)
  178. expect(elm.children.length).toBe(5)
  179. expect(elm.children[0].innerHTML).toBe('4')
  180. expect(elm.children[1].innerHTML).toBe('1')
  181. expect(elm.children[2].innerHTML).toBe('2')
  182. expect(elm.children[3].innerHTML).toBe('3')
  183. expect(elm.children[4].innerHTML).toBe('6')
  184. })
  185. it('should move to left and leaves hold', () => {
  186. const vnode1 = new VNode('p', {}, [1, 4, 5].map(spanNum))
  187. const vnode2 = new VNode('p', {}, [4, 6].map(spanNum))
  188. let elm = patch(vnode0, vnode1)
  189. expect(elm.children.length).toBe(3)
  190. elm = patch(vnode1, vnode2)
  191. expect(map(inner, elm.children)).toEqual(['4', '6'])
  192. })
  193. it('should handle moved and set to undefined element ending at the end', () => {
  194. const vnode1 = new VNode('p', {}, [2, 4, 5].map(spanNum))
  195. const vnode2 = new VNode('p', {}, [4, 5, 3].map(spanNum))
  196. let elm = patch(vnode0, vnode1)
  197. expect(elm.children.length).toBe(3)
  198. elm = patch(vnode1, vnode2)
  199. expect(elm.children.length).toBe(3)
  200. expect(elm.children[0].innerHTML).toBe('4')
  201. expect(elm.children[1].innerHTML).toBe('5')
  202. expect(elm.children[2].innerHTML).toBe('3')
  203. })
  204. it('should move a key in non-keyed nodes with a size up', () => {
  205. const vnode1 = new VNode('p', {}, [1, 'a', 'b', 'c'].map(spanNum))
  206. const vnode2 = new VNode('p', {}, ['d', 'a', 'b', 'c', 1, 'e'].map(spanNum))
  207. let elm = patch(vnode0, vnode1)
  208. expect(elm.children.length).toBe(4)
  209. expect(elm.textContent, '1abc')
  210. elm = patch(vnode1, vnode2)
  211. expect(elm.children.length).toBe(6)
  212. expect(elm.textContent, 'dabc1e')
  213. })
  214. it('should reverse element', () => {
  215. const vnode1 = new VNode('p', {}, [1, 2, 3, 4, 5, 6, 7, 8].map(spanNum))
  216. const vnode2 = new VNode('p', {}, [8, 7, 6, 5, 4, 3, 2, 1].map(spanNum))
  217. let elm = patch(vnode0, vnode1)
  218. expect(elm.children.length).toBe(8)
  219. elm = patch(vnode1, vnode2)
  220. expect(map(inner, elm.children)).toEqual(['8', '7', '6', '5', '4', '3', '2', '1'])
  221. })
  222. it('something', () => {
  223. const vnode1 = new VNode('p', {}, [0, 1, 2, 3, 4, 5].map(spanNum))
  224. const vnode2 = new VNode('p', {}, [4, 3, 2, 1, 5, 0].map(spanNum))
  225. let elm = patch(vnode0, vnode1)
  226. expect(elm.children.length).toBe(6)
  227. elm = patch(vnode1, vnode2)
  228. expect(map(inner, elm.children)).toEqual(['4', '3', '2', '1', '5', '0'])
  229. })
  230. it('should handle random shuffle', () => {
  231. let n
  232. let i
  233. const arr = []
  234. const opacities = []
  235. const elms = 14
  236. const samples = 5
  237. function spanNumWithOpacity (n, o) {
  238. return new VNode('span', { key: n, style: { opacity: o }}, undefined, n.toString())
  239. }
  240. for (n = 0; n < elms; ++n) { arr[n] = n }
  241. for (n = 0; n < samples; ++n) {
  242. const vnode1 = new VNode('span', {}, arr.map(n => {
  243. return spanNumWithOpacity(n, '1')
  244. }))
  245. const shufArr = shuffle(arr.slice(0))
  246. let elm = patch(vnode0, vnode1)
  247. for (i = 0; i < elms; ++i) {
  248. expect(elm.children[i].innerHTML).toBe(i.toString())
  249. opacities[i] = Math.random().toFixed(5).toString()
  250. }
  251. const vnode2 = new VNode('span', {}, arr.map(n => {
  252. return spanNumWithOpacity(shufArr[n], opacities[n])
  253. }))
  254. elm = patch(vnode1, vnode2)
  255. for (i = 0; i < elms; ++i) {
  256. expect(elm.children[i].innerHTML).toBe(shufArr[i].toString())
  257. expect(opacities[i].indexOf(elm.children[i].style.opacity)).toBe(0)
  258. }
  259. }
  260. })
  261. it('should append elements with updating children without keys', () => {
  262. const vnode1 = new VNode('div', {}, [
  263. new VNode('span', {}, undefined, 'hello')
  264. ])
  265. const vnode2 = new VNode('div', {}, [
  266. new VNode('span', {}, undefined, 'hello'),
  267. new VNode('span', {}, undefined, 'world')
  268. ])
  269. let elm = patch(vnode0, vnode1)
  270. expect(map(inner, elm.children)).toEqual(['hello'])
  271. elm = patch(vnode1, vnode2)
  272. expect(map(inner, elm.children)).toEqual(['hello', 'world'])
  273. })
  274. it('should handle unmoved text nodes with updating children without keys', () => {
  275. const vnode1 = new VNode('div', {}, [
  276. createTextVNode('text'),
  277. new VNode('span', {}, undefined, 'hello')
  278. ])
  279. const vnode2 = new VNode('div', {}, [
  280. createTextVNode('text'),
  281. new VNode('span', {}, undefined, 'hello')
  282. ])
  283. let elm = patch(vnode0, vnode1)
  284. expect(elm.childNodes[0].textContent).toBe('text')
  285. elm = patch(vnode1, vnode2)
  286. expect(elm.childNodes[0].textContent).toBe('text')
  287. })
  288. it('should handle changing text children with updating children without keys', () => {
  289. const vnode1 = new VNode('div', {}, [
  290. createTextVNode('text'),
  291. new VNode('span', {}, undefined, 'hello')
  292. ])
  293. const vnode2 = new VNode('div', {}, [
  294. createTextVNode('text2'),
  295. new VNode('span', {}, undefined, 'hello')
  296. ])
  297. let elm = patch(vnode0, vnode1)
  298. expect(elm.childNodes[0].textContent).toBe('text')
  299. elm = patch(vnode1, vnode2)
  300. expect(elm.childNodes[0].textContent).toBe('text2')
  301. })
  302. it('should prepend element with updating children without keys', () => {
  303. const vnode1 = new VNode('div', {}, [
  304. new VNode('span', {}, undefined, 'world')
  305. ])
  306. const vnode2 = new VNode('div', {}, [
  307. new VNode('span', {}, undefined, 'hello'),
  308. new VNode('span', {}, undefined, 'world')
  309. ])
  310. let elm = patch(vnode0, vnode1)
  311. expect(map(inner, elm.children)).toEqual(['world'])
  312. elm = patch(vnode1, vnode2)
  313. expect(map(inner, elm.children)).toEqual(['hello', 'world'])
  314. })
  315. it('should prepend element of different tag type with updating children without keys', () => {
  316. const vnode1 = new VNode('div', {}, [
  317. new VNode('span', {}, undefined, 'world')
  318. ])
  319. const vnode2 = new VNode('div', {}, [
  320. new VNode('div', {}, undefined, 'hello'),
  321. new VNode('span', {}, undefined, 'world')
  322. ])
  323. let elm = patch(vnode0, vnode1)
  324. expect(map(inner, elm.children)).toEqual(['world'])
  325. elm = patch(vnode1, vnode2)
  326. expect(map(prop('tagName'), elm.children)).toEqual(['DIV', 'SPAN'])
  327. expect(map(inner, elm.children)).toEqual(['hello', 'world'])
  328. })
  329. it('should remove elements with updating children without keys', () => {
  330. const vnode1 = new VNode('div', {}, [
  331. new VNode('span', {}, undefined, 'one'),
  332. new VNode('span', {}, undefined, 'two'),
  333. new VNode('span', {}, undefined, 'three')
  334. ])
  335. const vnode2 = new VNode('div', {}, [
  336. new VNode('span', {}, undefined, 'one'),
  337. new VNode('span', {}, undefined, 'three')
  338. ])
  339. let elm = patch(vnode0, vnode1)
  340. expect(map(inner, elm.children)).toEqual(['one', 'two', 'three'])
  341. elm = patch(vnode1, vnode2)
  342. expect(map(inner, elm.children)).toEqual(['one', 'three'])
  343. })
  344. it('should remove a single text node with updating children without keys', () => {
  345. const vnode1 = new VNode('div', {}, undefined, 'one')
  346. const vnode2 = new VNode('div', {})
  347. let elm = patch(vnode0, vnode1)
  348. expect(elm.textContent).toBe('one')
  349. elm = patch(vnode1, vnode2)
  350. expect(elm.textContent).toBe('')
  351. })
  352. it('should remove a single text node when children are updated', () => {
  353. const vnode1 = new VNode('div', {}, undefined, 'one')
  354. const vnode2 = new VNode('div', {}, [
  355. new VNode('div', {}, undefined, 'two'),
  356. new VNode('span', {}, undefined, 'three')
  357. ])
  358. let elm = patch(vnode0, vnode1)
  359. expect(elm.textContent).toBe('one')
  360. elm = patch(vnode1, vnode2)
  361. expect(map(prop('textContent'), elm.childNodes)).toEqual(['two', 'three'])
  362. })
  363. it('should remove a text node among other elements', () => {
  364. const vnode1 = new VNode('div', {}, [
  365. createTextVNode('one'),
  366. new VNode('span', {}, undefined, 'two')
  367. ])
  368. const vnode2 = new VNode('div', {}, [
  369. new VNode('div', {}, undefined, 'three')
  370. ])
  371. let elm = patch(vnode0, vnode1)
  372. expect(map(prop('textContent'), elm.childNodes)).toEqual(['one', 'two'])
  373. elm = patch(vnode1, vnode2)
  374. expect(elm.childNodes.length).toBe(1)
  375. expect(elm.childNodes[0].tagName).toBe('DIV')
  376. expect(elm.childNodes[0].textContent).toBe('three')
  377. })
  378. it('should reorder elements', () => {
  379. const vnode1 = new VNode('div', {}, [
  380. new VNode('span', {}, undefined, 'one'),
  381. new VNode('div', {}, undefined, 'two'),
  382. new VNode('b', {}, undefined, 'three')
  383. ])
  384. const vnode2 = new VNode('div', {}, [
  385. new VNode('b', {}, undefined, 'three'),
  386. new VNode('span', {}, undefined, 'two'),
  387. new VNode('div', {}, undefined, 'one')
  388. ])
  389. let elm = patch(vnode0, vnode1)
  390. expect(map(inner, elm.children)).toEqual(['one', 'two', 'three'])
  391. elm = patch(vnode1, vnode2)
  392. expect(map(inner, elm.children)).toEqual(['three', 'two', 'one'])
  393. })
  394. it('should handle children with the same key but with different tag', () => {
  395. const vnode1 = new VNode('div', {}, [
  396. new VNode('div', { key: 1 }, undefined, 'one'),
  397. new VNode('div', { key: 2 }, undefined, 'two'),
  398. new VNode('div', { key: 3 }, undefined, 'three'),
  399. new VNode('div', { key: 4 }, undefined, 'four')
  400. ])
  401. const vnode2 = new VNode('div', {}, [
  402. new VNode('div', { key: 4 }, undefined, 'four'),
  403. new VNode('span', { key: 3 }, undefined, 'three'),
  404. new VNode('span', { key: 2 }, undefined, 'two'),
  405. new VNode('div', { key: 1 }, undefined, 'one')
  406. ])
  407. let elm = patch(vnode0, vnode1)
  408. expect(map(tag, elm.children)).toEqual(['DIV', 'DIV', 'DIV', 'DIV'])
  409. expect(map(inner, elm.children)).toEqual(['one', 'two', 'three', 'four'])
  410. elm = patch(vnode1, vnode2)
  411. expect(map(tag, elm.children)).toEqual(['DIV', 'SPAN', 'SPAN', 'DIV'])
  412. expect(map(inner, elm.children)).toEqual(['four', 'three', 'two', 'one'])
  413. })
  414. it('should handle children with the same tag, same key, but one with data and one without data', () => {
  415. const vnode1 = new VNode('div', {}, [
  416. new VNode('div', { class: 'hi' }, undefined, 'one')
  417. ])
  418. const vnode2 = new VNode('div', {}, [
  419. new VNode('div', undefined, undefined, 'four')
  420. ])
  421. let elm = patch(vnode0, vnode1)
  422. const child1 = elm.children[0]
  423. expect(child1.className).toBe('hi')
  424. elm = patch(vnode1, vnode2)
  425. const child2 = elm.children[0]
  426. expect(child1).not.toBe(child2)
  427. expect(child2.className).toBe('')
  428. })
  429. })