async_component_spec.js 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. var Vue = require('src')
  2. var _ = Vue.util
  3. describe('Async components', function () {
  4. var el
  5. beforeEach(function () {
  6. el = document.createElement('div')
  7. document.body.appendChild(el)
  8. })
  9. afterEach(function () {
  10. document.body.removeChild(el)
  11. })
  12. it('normal', function (done) {
  13. var go = jasmine.createSpy()
  14. new Vue({
  15. el: el,
  16. template: '<test foo="bar" @ready="go"></test>',
  17. methods: {
  18. go: go
  19. },
  20. components: {
  21. test: function (resolve) {
  22. setTimeout(function () {
  23. resolve({
  24. props: ['foo'],
  25. template: '{{ foo }}',
  26. ready: function () {
  27. this.$emit('ready')
  28. }
  29. })
  30. next()
  31. }, 0)
  32. }
  33. }
  34. })
  35. function next () {
  36. expect(el.textContent).toBe('bar')
  37. expect(go).toHaveBeenCalled()
  38. done()
  39. }
  40. })
  41. it('dynamic', function (done) {
  42. var vm = new Vue({
  43. el: el,
  44. template: '<component :is="view"></component>',
  45. data: {
  46. view: 'view-a'
  47. },
  48. components: {
  49. 'view-a': function (resolve) {
  50. setTimeout(function () {
  51. resolve({
  52. template: 'A'
  53. })
  54. step1()
  55. }, 0)
  56. },
  57. 'view-b': function (resolve) {
  58. setTimeout(function () {
  59. resolve({
  60. template: 'B'
  61. })
  62. step2()
  63. }, 0)
  64. }
  65. }
  66. })
  67. var aCalled = false
  68. function step1 () {
  69. // ensure A is resolved only once
  70. expect(aCalled).toBe(false)
  71. aCalled = true
  72. expect(el.textContent).toBe('A')
  73. vm.view = 'view-b'
  74. }
  75. function step2 () {
  76. expect(el.textContent).toBe('B')
  77. vm.view = 'view-a'
  78. _.nextTick(function () {
  79. expect(el.textContent).toBe('A')
  80. done()
  81. })
  82. }
  83. })
  84. it('invalidate pending on dynamic switch', function (done) {
  85. var vm = new Vue({
  86. el: el,
  87. template: '<component :is="view"></component>',
  88. data: {
  89. view: 'view-a'
  90. },
  91. components: {
  92. 'view-a': function (resolve) {
  93. setTimeout(function () {
  94. resolve({
  95. template: 'A'
  96. })
  97. step1()
  98. }, 100)
  99. },
  100. 'view-b': function (resolve) {
  101. setTimeout(function () {
  102. resolve({
  103. template: 'B'
  104. })
  105. step2()
  106. }, 200)
  107. }
  108. }
  109. })
  110. expect(el.textContent).toBe('')
  111. vm.view = 'view-b'
  112. function step1 () {
  113. // called after A resolves, but A should have been
  114. // invalidated so no Ctor should be set
  115. expect(vm._directives[0].Component).toBe(null)
  116. }
  117. function step2 () {
  118. // B should resolve successfully
  119. expect(el.textContent).toBe('B')
  120. done()
  121. }
  122. })
  123. it('invalidate pending on teardown', function (done) {
  124. var vm = new Vue({
  125. el: el,
  126. template: '<test></test>',
  127. data: {
  128. view: 'view-a'
  129. },
  130. components: {
  131. test: function (resolve) {
  132. setTimeout(function () {
  133. resolve({
  134. template: 'A'
  135. })
  136. next()
  137. }, 100)
  138. }
  139. }
  140. })
  141. expect(el.textContent).toBe('')
  142. // cache directive isntance before destroy
  143. var dir = vm._directives[0]
  144. vm.$destroy()
  145. function next () {
  146. // called after A resolves, but A should have been
  147. // invalidated so no Ctor should be set
  148. expect(dir.Component).toBe(null)
  149. done()
  150. }
  151. })
  152. it('avoid duplicate requests', function (done) {
  153. var factoryCallCount = 0
  154. var instanceCount = 0
  155. new Vue({
  156. el: el,
  157. template:
  158. '<test></test>' +
  159. '<test></test>',
  160. components: {
  161. test: factory
  162. }
  163. })
  164. function factory (resolve) {
  165. factoryCallCount++
  166. setTimeout(function () {
  167. resolve({
  168. template: 'A',
  169. created: function () {
  170. instanceCount++
  171. }
  172. })
  173. next()
  174. }, 0)
  175. }
  176. function next () {
  177. expect(factoryCallCount).toBe(1)
  178. expect(el.textContent).toBe('AA')
  179. expect(instanceCount).toBe(2)
  180. done()
  181. }
  182. })
  183. it('warn reject', function () {
  184. new Vue({
  185. el: el,
  186. template: '<test></test>',
  187. components: {
  188. test: function (resolve, reject) {
  189. reject('nooooo')
  190. }
  191. }
  192. })
  193. expect('Reason: nooooo').toHaveBeenWarned()
  194. })
  195. it('v-for', function (done) {
  196. new Vue({
  197. el: el,
  198. template: '<test v-for="n in list" :n="n"></test>',
  199. data: {
  200. list: [1, 2, 3]
  201. },
  202. components: {
  203. test: function (resolve) {
  204. setTimeout(function () {
  205. resolve({
  206. props: ['n'],
  207. template: '{{n}}'
  208. })
  209. next()
  210. }, 0)
  211. }
  212. }
  213. })
  214. function next () {
  215. expect(el.textContent).toBe('123')
  216. done()
  217. }
  218. })
  219. })