bind.spec.js 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  1. import Vue from 'vue'
  2. describe('Directive v-bind', () => {
  3. it('normal attr', done => {
  4. const vm = new Vue({
  5. template: '<div><span :test="foo">hello</span></div>',
  6. data: { foo: 'ok' }
  7. }).$mount()
  8. expect(vm.$el.firstChild.getAttribute('test')).toBe('ok')
  9. vm.foo = 'again'
  10. waitForUpdate(() => {
  11. expect(vm.$el.firstChild.getAttribute('test')).toBe('again')
  12. vm.foo = null
  13. }).then(() => {
  14. expect(vm.$el.firstChild.hasAttribute('test')).toBe(false)
  15. vm.foo = false
  16. }).then(() => {
  17. expect(vm.$el.firstChild.hasAttribute('test')).toBe(false)
  18. vm.foo = true
  19. }).then(() => {
  20. expect(vm.$el.firstChild.getAttribute('test')).toBe('true')
  21. vm.foo = 0
  22. }).then(() => {
  23. expect(vm.$el.firstChild.getAttribute('test')).toBe('0')
  24. }).then(done)
  25. })
  26. it('should set property for input value', done => {
  27. const vm = new Vue({
  28. template: `
  29. <div>
  30. <input type="text" :value="foo">
  31. <input type="checkbox" :checked="bar">
  32. </div>
  33. `,
  34. data: {
  35. foo: 'ok',
  36. bar: false
  37. }
  38. }).$mount()
  39. expect(vm.$el.firstChild.value).toBe('ok')
  40. expect(vm.$el.lastChild.checked).toBe(false)
  41. vm.bar = true
  42. waitForUpdate(() => {
  43. expect(vm.$el.lastChild.checked).toBe(true)
  44. }).then(done)
  45. })
  46. it('xlink', done => {
  47. const vm = new Vue({
  48. template: '<svg><a :xlink:special="foo"></a></svg>',
  49. data: {
  50. foo: 'ok'
  51. }
  52. }).$mount()
  53. const xlinkNS = 'http://www.w3.org/1999/xlink'
  54. expect(vm.$el.firstChild.getAttributeNS(xlinkNS, 'special')).toBe('ok')
  55. vm.foo = 'again'
  56. waitForUpdate(() => {
  57. expect(vm.$el.firstChild.getAttributeNS(xlinkNS, 'special')).toBe('again')
  58. vm.foo = null
  59. }).then(() => {
  60. expect(vm.$el.firstChild.hasAttributeNS(xlinkNS, 'special')).toBe(false)
  61. vm.foo = true
  62. }).then(() => {
  63. expect(vm.$el.firstChild.getAttributeNS(xlinkNS, 'special')).toBe('true')
  64. }).then(done)
  65. })
  66. it('enumerated attr', done => {
  67. const vm = new Vue({
  68. template: '<div><span :draggable="foo">hello</span></div>',
  69. data: { foo: true }
  70. }).$mount()
  71. expect(vm.$el.firstChild.getAttribute('draggable')).toBe('true')
  72. vm.foo = 'again'
  73. waitForUpdate(() => {
  74. expect(vm.$el.firstChild.getAttribute('draggable')).toBe('true')
  75. vm.foo = null
  76. }).then(() => {
  77. expect(vm.$el.firstChild.getAttribute('draggable')).toBe('false')
  78. vm.foo = ''
  79. }).then(() => {
  80. expect(vm.$el.firstChild.getAttribute('draggable')).toBe('true')
  81. vm.foo = false
  82. }).then(() => {
  83. expect(vm.$el.firstChild.getAttribute('draggable')).toBe('false')
  84. vm.foo = 'false'
  85. }).then(() => {
  86. expect(vm.$el.firstChild.getAttribute('draggable')).toBe('false')
  87. }).then(done)
  88. })
  89. it('boolean attr', done => {
  90. const vm = new Vue({
  91. template: '<div><span :disabled="foo">hello</span></div>',
  92. data: { foo: true }
  93. }).$mount()
  94. expect(vm.$el.firstChild.getAttribute('disabled')).toBe('disabled')
  95. vm.foo = 'again'
  96. waitForUpdate(() => {
  97. expect(vm.$el.firstChild.getAttribute('disabled')).toBe('disabled')
  98. vm.foo = null
  99. }).then(() => {
  100. expect(vm.$el.firstChild.hasAttribute('disabled')).toBe(false)
  101. vm.foo = ''
  102. }).then(() => {
  103. expect(vm.$el.firstChild.hasAttribute('disabled')).toBe(true)
  104. }).then(done)
  105. })
  106. it('.prop modifier', () => {
  107. const vm = new Vue({
  108. template: '<div><span v-bind:text-content.prop="foo"></span><span :inner-html.prop="bar"></span></div>',
  109. data: {
  110. foo: 'hello',
  111. bar: '<span>qux</span>'
  112. }
  113. }).$mount()
  114. expect(vm.$el.children[0].textContent).toBe('hello')
  115. expect(vm.$el.children[1].innerHTML).toBe('<span>qux</span>')
  116. })
  117. it('.prop modifier with normal attribute binding', () => {
  118. const vm = new Vue({
  119. template: '<input :some.prop="some" :id="id">',
  120. data: {
  121. some: 'hello',
  122. id: false
  123. }
  124. }).$mount()
  125. expect(vm.$el.some).toBe('hello')
  126. expect(vm.$el.getAttribute('id')).toBe(null)
  127. })
  128. it('.camel modifier', () => {
  129. const vm = new Vue({
  130. template: '<svg :view-box.camel="viewBox"></svg>',
  131. data: {
  132. viewBox: '0 0 1 1'
  133. }
  134. }).$mount()
  135. expect(vm.$el.getAttribute('viewBox')).toBe('0 0 1 1')
  136. })
  137. it('.sync modifier', done => {
  138. const vm = new Vue({
  139. template: `<test :foo-bar.sync="bar"/>`,
  140. data: {
  141. bar: 1
  142. },
  143. components: {
  144. test: {
  145. props: ['fooBar'],
  146. template: `<div @click="$emit('update:fooBar', 2)">{{ fooBar }}</div>`
  147. }
  148. }
  149. }).$mount()
  150. expect(vm.$el.textContent).toBe('1')
  151. triggerEvent(vm.$el, 'click')
  152. waitForUpdate(() => {
  153. expect(vm.$el.textContent).toBe('2')
  154. }).then(done)
  155. })
  156. it('bind object', done => {
  157. const vm = new Vue({
  158. template: '<input v-bind="test">',
  159. data: {
  160. test: {
  161. id: 'test',
  162. class: 'ok',
  163. value: 'hello'
  164. }
  165. }
  166. }).$mount()
  167. expect(vm.$el.getAttribute('id')).toBe('test')
  168. expect(vm.$el.getAttribute('class')).toBe('ok')
  169. expect(vm.$el.value).toBe('hello')
  170. vm.test.id = 'hi'
  171. vm.test.value = 'bye'
  172. waitForUpdate(() => {
  173. expect(vm.$el.getAttribute('id')).toBe('hi')
  174. expect(vm.$el.getAttribute('class')).toBe('ok')
  175. expect(vm.$el.value).toBe('bye')
  176. }).then(done)
  177. })
  178. it('bind object with overwrite', done => {
  179. const vm = new Vue({
  180. template: '<input v-bind="test" id="foo" :class="test.value">',
  181. data: {
  182. test: {
  183. id: 'test',
  184. class: 'ok',
  185. value: 'hello'
  186. }
  187. }
  188. }).$mount()
  189. expect(vm.$el.getAttribute('id')).toBe('foo')
  190. expect(vm.$el.getAttribute('class')).toBe('hello')
  191. expect(vm.$el.value).toBe('hello')
  192. vm.test.id = 'hi'
  193. vm.test.value = 'bye'
  194. waitForUpdate(() => {
  195. expect(vm.$el.getAttribute('id')).toBe('foo')
  196. expect(vm.$el.getAttribute('class')).toBe('bye')
  197. expect(vm.$el.value).toBe('bye')
  198. }).then(done)
  199. })
  200. it('bind object with class/style', done => {
  201. const vm = new Vue({
  202. template: '<input class="a" style="color:red" v-bind="test">',
  203. data: {
  204. test: {
  205. id: 'test',
  206. class: ['b', 'c'],
  207. style: { fontSize: '12px' }
  208. }
  209. }
  210. }).$mount()
  211. expect(vm.$el.id).toBe('test')
  212. expect(vm.$el.className).toBe('a b c')
  213. expect(vm.$el.style.color).toBe('red')
  214. expect(vm.$el.style.fontSize).toBe('12px')
  215. vm.test.id = 'hi'
  216. vm.test.class = ['d']
  217. vm.test.style = { fontSize: '14px' }
  218. waitForUpdate(() => {
  219. expect(vm.$el.id).toBe('hi')
  220. expect(vm.$el.className).toBe('a d')
  221. expect(vm.$el.style.color).toBe('red')
  222. expect(vm.$el.style.fontSize).toBe('14px')
  223. }).then(done)
  224. })
  225. it('bind object as prop', done => {
  226. const vm = new Vue({
  227. template: '<input v-bind.prop="test">',
  228. data: {
  229. test: {
  230. id: 'test',
  231. className: 'ok',
  232. value: 'hello'
  233. }
  234. }
  235. }).$mount()
  236. expect(vm.$el.id).toBe('test')
  237. expect(vm.$el.className).toBe('ok')
  238. expect(vm.$el.value).toBe('hello')
  239. vm.test.id = 'hi'
  240. vm.test.className = 'okay'
  241. vm.test.value = 'bye'
  242. waitForUpdate(() => {
  243. expect(vm.$el.id).toBe('hi')
  244. expect(vm.$el.className).toBe('okay')
  245. expect(vm.$el.value).toBe('bye')
  246. }).then(done)
  247. })
  248. it('bind array', done => {
  249. const vm = new Vue({
  250. template: '<input v-bind="test">',
  251. data: {
  252. test: [
  253. { id: 'test', class: 'ok' },
  254. { value: 'hello' }
  255. ]
  256. }
  257. }).$mount()
  258. expect(vm.$el.getAttribute('id')).toBe('test')
  259. expect(vm.$el.getAttribute('class')).toBe('ok')
  260. expect(vm.$el.value).toBe('hello')
  261. vm.test[0].id = 'hi'
  262. vm.test[1].value = 'bye'
  263. waitForUpdate(() => {
  264. expect(vm.$el.getAttribute('id')).toBe('hi')
  265. expect(vm.$el.getAttribute('class')).toBe('ok')
  266. expect(vm.$el.value).toBe('bye')
  267. }).then(done)
  268. })
  269. it('warn expect object', () => {
  270. new Vue({
  271. template: '<input v-bind="test">',
  272. data: {
  273. test: 1
  274. }
  275. }).$mount()
  276. expect('v-bind without argument expects an Object or Array value').toHaveBeenWarned()
  277. })
  278. it('set value for option element', () => {
  279. const vm = new Vue({
  280. template: '<select><option :value="val">val</option></select>',
  281. data: {
  282. val: 'val'
  283. }
  284. }).$mount()
  285. // check value attribute
  286. expect(vm.$el.options[0].getAttribute('value')).toBe('val')
  287. })
  288. // a vdom patch edge case where the user has several un-keyed elements of the
  289. // same tag next to each other, and toggling them.
  290. it('properly update for toggling un-keyed children', done => {
  291. const vm = new Vue({
  292. template: `
  293. <div>
  294. <div v-if="ok" id="a" data-test="1"></div>
  295. <div v-if="!ok" id="b"></div>
  296. </div>
  297. `,
  298. data: {
  299. ok: true
  300. }
  301. }).$mount()
  302. expect(vm.$el.children[0].id).toBe('a')
  303. expect(vm.$el.children[0].getAttribute('data-test')).toBe('1')
  304. vm.ok = false
  305. waitForUpdate(() => {
  306. expect(vm.$el.children[0].id).toBe('b')
  307. expect(vm.$el.children[0].getAttribute('data-test')).toBe(null)
  308. }).then(done)
  309. })
  310. })