bind.spec.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  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('.prop modifier shorthand', () => {
  129. const vm = new Vue({
  130. template: '<div><span .text-content="foo"></span><span .inner-html="bar"></span></div>',
  131. data: {
  132. foo: 'hello',
  133. bar: '<span>qux</span>'
  134. }
  135. }).$mount()
  136. expect(vm.$el.children[0].textContent).toBe('hello')
  137. expect(vm.$el.children[1].innerHTML).toBe('<span>qux</span>')
  138. })
  139. it('.camel modifier', () => {
  140. const vm = new Vue({
  141. template: '<svg :view-box.camel="viewBox"></svg>',
  142. data: {
  143. viewBox: '0 0 1 1'
  144. }
  145. }).$mount()
  146. expect(vm.$el.getAttribute('viewBox')).toBe('0 0 1 1')
  147. })
  148. it('.sync modifier', done => {
  149. const vm = new Vue({
  150. template: `<test :foo-bar.sync="bar"/>`,
  151. data: {
  152. bar: 1
  153. },
  154. components: {
  155. test: {
  156. props: ['fooBar'],
  157. template: `<div @click="$emit('update:fooBar', 2)">{{ fooBar }}</div>`
  158. }
  159. }
  160. }).$mount()
  161. document.body.appendChild(vm.$el)
  162. expect(vm.$el.textContent).toBe('1')
  163. triggerEvent(vm.$el, 'click')
  164. waitForUpdate(() => {
  165. expect(vm.$el.textContent).toBe('2')
  166. document.body.removeChild(vm.$el)
  167. }).then(done)
  168. })
  169. it('.sync modifier with kebab case event', done => {
  170. const vm = new Vue({
  171. template: `<test ref="test" :foo-bar.sync="bar"/>`,
  172. data: {
  173. bar: 1
  174. },
  175. components: {
  176. test: {
  177. props: ['fooBar'],
  178. template: `<div>{{ fooBar }}</div>`,
  179. methods: {
  180. update () {
  181. this.$emit('update:foo-bar', 2)
  182. }
  183. }
  184. }
  185. }
  186. }).$mount()
  187. expect(vm.$el.textContent).toBe('1')
  188. vm.$refs.test.update()
  189. waitForUpdate(() => {
  190. expect(vm.$el.textContent).toBe('2')
  191. }).then(done)
  192. })
  193. it('bind object', done => {
  194. const vm = new Vue({
  195. template: '<input v-bind="test">',
  196. data: {
  197. test: {
  198. id: 'test',
  199. class: 'ok',
  200. value: 'hello'
  201. }
  202. }
  203. }).$mount()
  204. expect(vm.$el.getAttribute('id')).toBe('test')
  205. expect(vm.$el.getAttribute('class')).toBe('ok')
  206. expect(vm.$el.value).toBe('hello')
  207. vm.test.id = 'hi'
  208. vm.test.value = 'bye'
  209. waitForUpdate(() => {
  210. expect(vm.$el.getAttribute('id')).toBe('hi')
  211. expect(vm.$el.getAttribute('class')).toBe('ok')
  212. expect(vm.$el.value).toBe('bye')
  213. }).then(done)
  214. })
  215. it('bind object with explicit overrides', () => {
  216. const vm = new Vue({
  217. template: `<test v-bind="test" data-foo="foo" dataBar="bar"/>`,
  218. components: {
  219. test: {
  220. template: '<div :data-foo="dataFoo" :data-bar="dataBar"></div>',
  221. props: ['dataFoo', 'dataBar']
  222. }
  223. },
  224. data: {
  225. test: {
  226. dataFoo: 'hi',
  227. dataBar: 'bye'
  228. }
  229. }
  230. }).$mount()
  231. expect(vm.$el.getAttribute('data-foo')).toBe('foo')
  232. expect(vm.$el.getAttribute('data-bar')).toBe('bar')
  233. })
  234. it('.sync modifier with bind object', done => {
  235. const vm = new Vue({
  236. template: `<test v-bind.sync="test"/>`,
  237. data: {
  238. test: {
  239. fooBar: 1
  240. }
  241. },
  242. components: {
  243. test: {
  244. props: ['fooBar'],
  245. template: `<div @click="handleUpdate">{{ fooBar }}</div>`,
  246. methods: {
  247. handleUpdate () {
  248. this.$emit('update:fooBar', 2)
  249. }
  250. }
  251. }
  252. }
  253. }).$mount()
  254. document.body.appendChild(vm.$el)
  255. expect(vm.$el.textContent).toBe('1')
  256. triggerEvent(vm.$el, 'click')
  257. waitForUpdate(() => {
  258. expect(vm.$el.textContent).toBe('2')
  259. vm.test.fooBar = 3
  260. }).then(() => {
  261. expect(vm.$el.textContent).toBe('3')
  262. document.body.removeChild(vm.$el)
  263. }).then(done)
  264. })
  265. it('bind object with overwrite', done => {
  266. const vm = new Vue({
  267. template: '<input v-bind="test" id="foo" :class="test.value">',
  268. data: {
  269. test: {
  270. id: 'test',
  271. class: 'ok',
  272. value: 'hello'
  273. }
  274. }
  275. }).$mount()
  276. expect(vm.$el.getAttribute('id')).toBe('foo')
  277. expect(vm.$el.getAttribute('class')).toBe('hello')
  278. expect(vm.$el.value).toBe('hello')
  279. vm.test.id = 'hi'
  280. vm.test.value = 'bye'
  281. waitForUpdate(() => {
  282. expect(vm.$el.getAttribute('id')).toBe('foo')
  283. expect(vm.$el.getAttribute('class')).toBe('bye')
  284. expect(vm.$el.value).toBe('bye')
  285. }).then(done)
  286. })
  287. it('bind object with class/style', done => {
  288. const vm = new Vue({
  289. template: '<input class="a" style="color:red" v-bind="test">',
  290. data: {
  291. test: {
  292. id: 'test',
  293. class: ['b', 'c'],
  294. style: { fontSize: '12px' }
  295. }
  296. }
  297. }).$mount()
  298. expect(vm.$el.id).toBe('test')
  299. expect(vm.$el.className).toBe('a b c')
  300. expect(vm.$el.style.color).toBe('red')
  301. expect(vm.$el.style.fontSize).toBe('12px')
  302. vm.test.id = 'hi'
  303. vm.test.class = ['d']
  304. vm.test.style = { fontSize: '14px' }
  305. waitForUpdate(() => {
  306. expect(vm.$el.id).toBe('hi')
  307. expect(vm.$el.className).toBe('a d')
  308. expect(vm.$el.style.color).toBe('red')
  309. expect(vm.$el.style.fontSize).toBe('14px')
  310. }).then(done)
  311. })
  312. it('bind object as prop', done => {
  313. const vm = new Vue({
  314. template: '<input v-bind.prop="test">',
  315. data: {
  316. test: {
  317. id: 'test',
  318. className: 'ok',
  319. value: 'hello'
  320. }
  321. }
  322. }).$mount()
  323. expect(vm.$el.id).toBe('test')
  324. expect(vm.$el.className).toBe('ok')
  325. expect(vm.$el.value).toBe('hello')
  326. vm.test.id = 'hi'
  327. vm.test.className = 'okay'
  328. vm.test.value = 'bye'
  329. waitForUpdate(() => {
  330. expect(vm.$el.id).toBe('hi')
  331. expect(vm.$el.className).toBe('okay')
  332. expect(vm.$el.value).toBe('bye')
  333. }).then(done)
  334. })
  335. it('bind array', done => {
  336. const vm = new Vue({
  337. template: '<input v-bind="test">',
  338. data: {
  339. test: [
  340. { id: 'test', class: 'ok' },
  341. { value: 'hello' }
  342. ]
  343. }
  344. }).$mount()
  345. expect(vm.$el.getAttribute('id')).toBe('test')
  346. expect(vm.$el.getAttribute('class')).toBe('ok')
  347. expect(vm.$el.value).toBe('hello')
  348. vm.test[0].id = 'hi'
  349. vm.test[1].value = 'bye'
  350. waitForUpdate(() => {
  351. expect(vm.$el.getAttribute('id')).toBe('hi')
  352. expect(vm.$el.getAttribute('class')).toBe('ok')
  353. expect(vm.$el.value).toBe('bye')
  354. }).then(done)
  355. })
  356. it('warn expect object', () => {
  357. new Vue({
  358. template: '<input v-bind="test">',
  359. data: {
  360. test: 1
  361. }
  362. }).$mount()
  363. expect('v-bind without argument expects an Object or Array value').toHaveBeenWarned()
  364. })
  365. it('set value for option element', () => {
  366. const vm = new Vue({
  367. template: '<select><option :value="val">val</option></select>',
  368. data: {
  369. val: 'val'
  370. }
  371. }).$mount()
  372. // check value attribute
  373. expect(vm.$el.options[0].getAttribute('value')).toBe('val')
  374. })
  375. // a vdom patch edge case where the user has several un-keyed elements of the
  376. // same tag next to each other, and toggling them.
  377. it('properly update for toggling un-keyed children', done => {
  378. const vm = new Vue({
  379. template: `
  380. <div>
  381. <div v-if="ok" id="a" data-test="1"></div>
  382. <div v-if="!ok" id="b"></div>
  383. </div>
  384. `,
  385. data: {
  386. ok: true
  387. }
  388. }).$mount()
  389. expect(vm.$el.children[0].id).toBe('a')
  390. expect(vm.$el.children[0].getAttribute('data-test')).toBe('1')
  391. vm.ok = false
  392. waitForUpdate(() => {
  393. expect(vm.$el.children[0].id).toBe('b')
  394. expect(vm.$el.children[0].getAttribute('data-test')).toBe(null)
  395. }).then(done)
  396. })
  397. describe('bind object with special attribute', () => {
  398. function makeInstance (options) {
  399. return new Vue({
  400. template: `<div>${options.parentTemp}</div>`,
  401. data: {
  402. attrs: {
  403. [options.attr]: options.value
  404. }
  405. },
  406. components: {
  407. comp: {
  408. template: options.childTemp
  409. }
  410. }
  411. }).$mount()
  412. }
  413. it('key', () => {
  414. const vm = makeInstance({
  415. attr: 'key',
  416. value: 'test',
  417. parentTemp: '<div v-bind="attrs"></div>'
  418. })
  419. expect(vm._vnode.children[0].key).toBe('test')
  420. })
  421. it('ref', () => {
  422. const vm = makeInstance({
  423. attr: 'ref',
  424. value: 'test',
  425. parentTemp: '<div v-bind="attrs"></div>'
  426. })
  427. expect(vm.$refs.test).toBe(vm.$el.firstChild)
  428. })
  429. it('slot', () => {
  430. const vm = makeInstance({
  431. attr: 'slot',
  432. value: 'test',
  433. parentTemp: '<comp><span v-bind="attrs">123</span></comp>',
  434. childTemp: '<div>slot:<slot name="test"></slot></div>'
  435. })
  436. expect(vm.$el.innerHTML).toBe('<div>slot:<span>123</span></div>')
  437. })
  438. it('is', () => {
  439. const vm = makeInstance({
  440. attr: 'is',
  441. value: 'comp',
  442. parentTemp: '<component v-bind="attrs"></component>',
  443. childTemp: '<div>comp</div>'
  444. })
  445. expect(vm.$el.innerHTML).toBe('<div>comp</div>')
  446. })
  447. })
  448. })