for.spec.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  1. import Vue from 'vue'
  2. describe('Directive v-for', () => {
  3. it('should render array of primitive values', done => {
  4. const vm = new Vue({
  5. template: `
  6. <div>
  7. <span v-for="item in list">{{$index}}-{{item}}</span>
  8. </div>
  9. `,
  10. data: {
  11. list: ['a', 'b', 'c']
  12. }
  13. }).$mount()
  14. expect(vm.$el.innerHTML).toBe('<span>0-a</span><span>1-b</span><span>2-c</span>')
  15. Vue.set(vm.list, 0, 'd')
  16. waitForUpdate(() => {
  17. expect(vm.$el.innerHTML).toBe('<span>0-d</span><span>1-b</span><span>2-c</span>')
  18. vm.list.push('d')
  19. }).then(() => {
  20. expect(vm.$el.innerHTML).toBe('<span>0-d</span><span>1-b</span><span>2-c</span><span>3-d</span>')
  21. vm.list.splice(1, 2)
  22. }).then(() => {
  23. expect(vm.$el.innerHTML).toBe('<span>0-d</span><span>1-d</span>')
  24. vm.list = ['x', 'y']
  25. }).then(() => {
  26. expect(vm.$el.innerHTML).toBe('<span>0-x</span><span>1-y</span>')
  27. done()
  28. }).catch(done)
  29. })
  30. it('should render array of object values', done => {
  31. const vm = new Vue({
  32. template: `
  33. <div>
  34. <span v-for="item in list">{{$index}}-{{item.value}}</span>
  35. </div>
  36. `,
  37. data: {
  38. list: [
  39. { value: 'a' },
  40. { value: 'b' },
  41. { value: 'c' }
  42. ]
  43. }
  44. }).$mount()
  45. expect(vm.$el.innerHTML).toBe('<span>0-a</span><span>1-b</span><span>2-c</span>')
  46. Vue.set(vm.list, 0, { value: 'd' })
  47. waitForUpdate(() => {
  48. expect(vm.$el.innerHTML).toBe('<span>0-d</span><span>1-b</span><span>2-c</span>')
  49. vm.list[0].value = 'e'
  50. }).then(() => {
  51. expect(vm.$el.innerHTML).toBe('<span>0-e</span><span>1-b</span><span>2-c</span>')
  52. vm.list.push({})
  53. }).then(() => {
  54. expect(vm.$el.innerHTML).toBe('<span>0-e</span><span>1-b</span><span>2-c</span><span>3-</span>')
  55. vm.list.splice(1, 2)
  56. }).then(() => {
  57. expect(vm.$el.innerHTML).toBe('<span>0-e</span><span>1-</span>')
  58. vm.list = [{ value: 'x' }, { value: 'y' }]
  59. }).then(() => {
  60. expect(vm.$el.innerHTML).toBe('<span>0-x</span><span>1-y</span>')
  61. done()
  62. }).catch(done)
  63. })
  64. it('should render an Object', done => {
  65. const vm = new Vue({
  66. template: `
  67. <div>
  68. <span v-for="val in obj">{{val}}-{{$key}}</span>
  69. </div>
  70. `,
  71. data: {
  72. obj: { a: 0, b: 1, c: 2 }
  73. }
  74. }).$mount()
  75. expect(vm.$el.innerHTML).toBe('<span>0-a</span><span>1-b</span><span>2-c</span>')
  76. vm.obj.a = 3
  77. waitForUpdate(() => {
  78. expect(vm.$el.innerHTML).toBe('<span>3-a</span><span>1-b</span><span>2-c</span>')
  79. Vue.set(vm.obj, 'd', 4)
  80. }).then(() => {
  81. expect(vm.$el.innerHTML).toBe('<span>3-a</span><span>1-b</span><span>2-c</span><span>4-d</span>')
  82. Vue.delete(vm.obj, 'a')
  83. }).then(() => {
  84. expect(vm.$el.innerHTML).toBe('<span>1-b</span><span>2-c</span><span>4-d</span>')
  85. done()
  86. }).catch(done)
  87. })
  88. it('should render each key of data', done => {
  89. const vm = new Vue({
  90. template: `
  91. <div>
  92. <span v-for="val in $data">{{val}}-{{$key}}</span>
  93. </div>
  94. `,
  95. data: { a: 0, b: 1, c: 2 }
  96. }).$mount()
  97. expect(vm.$el.innerHTML).toBe('<span>0-a</span><span>1-b</span><span>2-c</span>')
  98. vm.a = 3
  99. waitForUpdate(() => {
  100. expect(vm.$el.innerHTML).toBe('<span>3-a</span><span>1-b</span><span>2-c</span>')
  101. done()
  102. }).catch(done)
  103. })
  104. describe('alternative syntax', () => {
  105. it('should render array of primitive values', done => {
  106. const vm = new Vue({
  107. template: `
  108. <div>
  109. <span v-for="(i, item) in list">{{i}}-{{item}}</span>
  110. </div>
  111. `,
  112. data: {
  113. list: ['a', 'b', 'c']
  114. }
  115. }).$mount()
  116. expect(vm.$el.innerHTML).toBe('<span>0-a</span><span>1-b</span><span>2-c</span>')
  117. Vue.set(vm.list, 0, 'd')
  118. waitForUpdate(() => {
  119. expect(vm.$el.innerHTML).toBe('<span>0-d</span><span>1-b</span><span>2-c</span>')
  120. vm.list.push('d')
  121. }).then(() => {
  122. expect(vm.$el.innerHTML).toBe('<span>0-d</span><span>1-b</span><span>2-c</span><span>3-d</span>')
  123. vm.list.splice(1, 2)
  124. }).then(() => {
  125. expect(vm.$el.innerHTML).toBe('<span>0-d</span><span>1-d</span>')
  126. vm.list = ['x', 'y']
  127. }).then(() => {
  128. expect(vm.$el.innerHTML).toBe('<span>0-x</span><span>1-y</span>')
  129. done()
  130. }).catch(done)
  131. })
  132. it('should render array of object values', done => {
  133. const vm = new Vue({
  134. template: `
  135. <div>
  136. <span v-for="(i, item) in list">{{i}}-{{item.value}}</span>
  137. </div>
  138. `,
  139. data: {
  140. list: [
  141. { value: 'a' },
  142. { value: 'b' },
  143. { value: 'c' }
  144. ]
  145. }
  146. }).$mount()
  147. expect(vm.$el.innerHTML).toBe('<span>0-a</span><span>1-b</span><span>2-c</span>')
  148. Vue.set(vm.list, 0, { value: 'd' })
  149. waitForUpdate(() => {
  150. expect(vm.$el.innerHTML).toBe('<span>0-d</span><span>1-b</span><span>2-c</span>')
  151. vm.list[0].value = 'e'
  152. }).then(() => {
  153. expect(vm.$el.innerHTML).toBe('<span>0-e</span><span>1-b</span><span>2-c</span>')
  154. vm.list.push({})
  155. }).then(() => {
  156. expect(vm.$el.innerHTML).toBe('<span>0-e</span><span>1-b</span><span>2-c</span><span>3-</span>')
  157. vm.list.splice(1, 2)
  158. }).then(() => {
  159. expect(vm.$el.innerHTML).toBe('<span>0-e</span><span>1-</span>')
  160. vm.list = [{ value: 'x' }, { value: 'y' }]
  161. }).then(() => {
  162. expect(vm.$el.innerHTML).toBe('<span>0-x</span><span>1-y</span>')
  163. done()
  164. }).catch(done)
  165. })
  166. it('should render an Object', done => {
  167. const vm = new Vue({
  168. template: `
  169. <div>
  170. <span v-for="(k, v) in obj">{{v}}-{{k}}</span>
  171. </div>
  172. `,
  173. data: {
  174. obj: { a: 0, b: 1, c: 2 }
  175. }
  176. }).$mount()
  177. expect(vm.$el.innerHTML).toBe('<span>0-a</span><span>1-b</span><span>2-c</span>')
  178. vm.obj.a = 3
  179. waitForUpdate(() => {
  180. expect(vm.$el.innerHTML).toBe('<span>3-a</span><span>1-b</span><span>2-c</span>')
  181. Vue.set(vm.obj, 'd', 4)
  182. }).then(() => {
  183. expect(vm.$el.innerHTML).toBe('<span>3-a</span><span>1-b</span><span>2-c</span><span>4-d</span>')
  184. Vue.delete(vm.obj, 'a')
  185. }).then(() => {
  186. expect(vm.$el.innerHTML).toBe('<span>1-b</span><span>2-c</span><span>4-d</span>')
  187. done()
  188. }).catch(done)
  189. })
  190. it('should render each key of data', done => {
  191. const vm = new Vue({
  192. template: `
  193. <div>
  194. <span v-for="(k, v) in $data">{{v}}-{{k}}</span>
  195. </div>
  196. `,
  197. data: { a: 0, b: 1, c: 2 }
  198. }).$mount()
  199. expect(vm.$el.innerHTML).toBe('<span>0-a</span><span>1-b</span><span>2-c</span>')
  200. vm.a = 3
  201. waitForUpdate(() => {
  202. expect(vm.$el.innerHTML).toBe('<span>3-a</span><span>1-b</span><span>2-c</span>')
  203. done()
  204. }).catch(done)
  205. })
  206. })
  207. it('check priorities: v-if before v-for', function () {
  208. const vm = new Vue({
  209. data: {
  210. items: [1, 2, 3]
  211. },
  212. template: '<div><div v-if="item < 3" v-for="item in items">{{item}}</div></div>'
  213. }).$mount()
  214. expect(vm.$el.textContent).toBe('12')
  215. })
  216. it('check priorities: v-if after v-for', function () {
  217. const vm = new Vue({
  218. data: {
  219. items: [1, 2, 3]
  220. },
  221. template: '<div><div v-for="item in items" v-if="item < 3">{{item}}</div></div>'
  222. }).$mount()
  223. expect(vm.$el.textContent).toBe('12')
  224. })
  225. it('range v-for', () => {
  226. const vm = new Vue({
  227. template: '<div><div v-for="n in 3">{{n}}</div></div>'
  228. }).$mount()
  229. expect(vm.$el.textContent).toBe('123')
  230. })
  231. it('without track-by', done => {
  232. const vm = new Vue({
  233. data: {
  234. items: [
  235. { id: 1, msg: 'a' },
  236. { id: 2, msg: 'b' },
  237. { id: 3, msg: 'c' }
  238. ]
  239. },
  240. template: '<div><div v-for="item in items">{{ item.msg }}</div></div>'
  241. }).$mount()
  242. expect(vm.$el.textContent).toBe('abc')
  243. const first = vm.$el.children[0]
  244. vm.items.reverse()
  245. waitForUpdate(() => {
  246. expect(vm.$el.textContent).toBe('cba')
  247. // assert reusing DOM element in place
  248. expect(vm.$el.children[0]).toBe(first)
  249. done()
  250. })
  251. })
  252. it('with track-by', done => {
  253. const vm = new Vue({
  254. data: {
  255. items: [
  256. { id: 1, msg: 'a' },
  257. { id: 2, msg: 'b' },
  258. { id: 3, msg: 'c' }
  259. ]
  260. },
  261. template: '<div><div v-for="item in items" track-by="item.id">{{ item.msg }}</div></div>'
  262. }).$mount()
  263. expect(vm.$el.textContent).toBe('abc')
  264. const first = vm.$el.children[0]
  265. vm.items.reverse()
  266. waitForUpdate(() => {
  267. expect(vm.$el.textContent).toBe('cba')
  268. // assert moving DOM element
  269. expect(vm.$el.children[0]).not.toBe(first)
  270. expect(vm.$el.children[2]).toBe(first)
  271. done()
  272. })
  273. })
  274. it('nested loops', () => {
  275. const vm = new Vue({
  276. data: {
  277. items: [
  278. { items: [{ a: 1 }, { a: 2 }], a: 1 },
  279. { items: [{ a: 3 }, { a: 4 }], a: 2 }
  280. ]
  281. },
  282. template:
  283. '<div>' +
  284. '<div v-for="(i, item) in items">' +
  285. '<p v-for="subItem in item.items">{{$index}} {{subItem.a}} {{i}} {{item.a}}</p>' +
  286. '</div>' +
  287. '</div>'
  288. }).$mount()
  289. expect(vm.$el.innerHTML).toBe(
  290. '<div><p>0 1 0 1</p><p>1 2 0 1</p></div>' +
  291. '<div><p>0 3 1 2</p><p>1 4 1 2</p></div>'
  292. )
  293. })
  294. it('template v-for', done => {
  295. const vm = new Vue({
  296. data: {
  297. list: [
  298. { a: 1 },
  299. { a: 2 },
  300. { a: 3 }
  301. ]
  302. },
  303. template:
  304. '<div>' +
  305. '<template v-for="item in list">' +
  306. '<p>{{item.a}}</p>' +
  307. '<p>{{item.a + 1}}</p>' +
  308. '</template>' +
  309. '</div>'
  310. }).$mount()
  311. assertMarkup()
  312. vm.list.reverse()
  313. waitForUpdate(() => {
  314. assertMarkup()
  315. vm.list.splice(1, 1)
  316. }).then(() => {
  317. assertMarkup()
  318. vm.list.splice(1, 0, { a: 2 })
  319. done()
  320. }).catch(done)
  321. function assertMarkup () {
  322. var markup = vm.list.map(function (item) {
  323. return '<p>' + item.a + '</p><p>' + (item.a + 1) + '</p>'
  324. }).join('')
  325. expect(vm.$el.innerHTML).toBe(markup)
  326. }
  327. })
  328. it('component v-for', done => {
  329. const vm = new Vue({
  330. data: {
  331. list: [
  332. { a: 1 },
  333. { a: 2 },
  334. { a: 3 }
  335. ]
  336. },
  337. template:
  338. '<div>' +
  339. '<test v-for="item in list" :msg="item.a">' +
  340. '<span>{{item.a}}</span>' +
  341. '</test>' +
  342. '</div>',
  343. components: {
  344. test: {
  345. props: ['msg'],
  346. template: '<p>{{msg}}<slot></slot></p>'
  347. }
  348. }
  349. }).$mount()
  350. assertMarkup()
  351. vm.list.reverse()
  352. waitForUpdate(() => {
  353. assertMarkup()
  354. vm.list.splice(1, 1)
  355. }).then(() => {
  356. assertMarkup()
  357. vm.list.splice(1, 0, { a: 2 })
  358. done()
  359. }).catch(done)
  360. function assertMarkup () {
  361. var markup = vm.list.map(function (item) {
  362. return `<p>${item.a}<span>${item.a}</span></p>`
  363. }).join('')
  364. expect(vm.$el.innerHTML).toBe(markup)
  365. }
  366. })
  367. it('dynamic component v-for', done => {
  368. const vm = new Vue({
  369. data: {
  370. list: [
  371. { type: 'one' },
  372. { type: 'two' }
  373. ]
  374. },
  375. template:
  376. '<div>' +
  377. '<component v-for="item in list" :is="item.type"></component>' +
  378. '</div>',
  379. components: {
  380. one: {
  381. template: '<p>One!</p>'
  382. },
  383. two: {
  384. template: '<div>Two!</div>'
  385. }
  386. }
  387. }).$mount()
  388. expect(vm.$el.innerHTML).toContain('<p>One!</p><div>Two!</div>')
  389. vm.list.reverse()
  390. waitForUpdate(() => {
  391. expect(vm.$el.innerHTML).toContain('<div>Two!</div><p>One!</p>')
  392. done()
  393. }).catch(done)
  394. })
  395. })