async_component_spec.js 4.7 KB

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