model-text.spec.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. import Vue from 'vue'
  2. import { isIE9, isIE, isAndroid } from 'core/util/env'
  3. describe('Directive v-model text', () => {
  4. it('should update value both ways', done => {
  5. const vm = new Vue({
  6. data: {
  7. test: 'b'
  8. },
  9. template: '<input v-model="test">'
  10. }).$mount()
  11. expect(vm.$el.value).toBe('b')
  12. vm.test = 'a'
  13. waitForUpdate(() => {
  14. expect(vm.$el.value).toBe('a')
  15. vm.$el.value = 'c'
  16. triggerEvent(vm.$el, 'input')
  17. expect(vm.test).toBe('c')
  18. }).then(done)
  19. })
  20. it('should work with space ended expression in v-model', () => {
  21. const vm = new Vue({
  22. data: {
  23. obj: {
  24. test: 'b'
  25. }
  26. },
  27. template: '<input v-model="obj.test ">'
  28. }).$mount()
  29. triggerEvent(vm.$el, 'input')
  30. expect(vm.obj['test ']).toBe(undefined)
  31. expect(vm.obj.test).toBe('b')
  32. })
  33. it('.lazy modifier', () => {
  34. const vm = new Vue({
  35. data: {
  36. test: 'b'
  37. },
  38. template: '<input v-model.lazy="test">'
  39. }).$mount()
  40. expect(vm.$el.value).toBe('b')
  41. expect(vm.test).toBe('b')
  42. vm.$el.value = 'c'
  43. triggerEvent(vm.$el, 'input')
  44. expect(vm.test).toBe('b')
  45. triggerEvent(vm.$el, 'change')
  46. expect(vm.test).toBe('c')
  47. })
  48. it('.number modifier', () => {
  49. const vm = new Vue({
  50. data: {
  51. test: 1
  52. },
  53. template: '<input v-model.number="test">'
  54. }).$mount()
  55. expect(vm.test).toBe(1)
  56. vm.$el.value = '2'
  57. triggerEvent(vm.$el, 'input')
  58. expect(vm.test).toBe(2)
  59. // should let strings pass through
  60. vm.$el.value = 'f'
  61. triggerEvent(vm.$el, 'input')
  62. expect(vm.test).toBe('f')
  63. })
  64. it('.trim modifier', () => {
  65. const vm = new Vue({
  66. data: {
  67. test: 'hi'
  68. },
  69. template: '<input v-model.trim="test">'
  70. }).$mount()
  71. expect(vm.test).toBe('hi')
  72. vm.$el.value = ' what '
  73. triggerEvent(vm.$el, 'input')
  74. expect(vm.test).toBe('what')
  75. })
  76. it('.number focus and typing', (done) => {
  77. const vm = new Vue({
  78. data: {
  79. test: 0,
  80. update: 0
  81. },
  82. template:
  83. '<div>' +
  84. '<input ref="input" v-model.number="test">{{ update }}' +
  85. '<input ref="blur">' +
  86. '</div>'
  87. }).$mount()
  88. document.body.appendChild(vm.$el)
  89. vm.$refs.input.focus()
  90. expect(vm.test).toBe(0)
  91. vm.$refs.input.value = '1.0'
  92. triggerEvent(vm.$refs.input, 'input')
  93. expect(vm.test).toBe(1)
  94. vm.update++
  95. waitForUpdate(() => {
  96. expect(vm.$refs.input.value).toBe('1.0')
  97. vm.$refs.blur.focus()
  98. vm.update++
  99. }).then(() => {
  100. expect(vm.$refs.input.value).toBe('1')
  101. }).then(done)
  102. })
  103. it('.trim focus and typing', (done) => {
  104. const vm = new Vue({
  105. data: {
  106. test: 'abc',
  107. update: 0
  108. },
  109. template:
  110. '<div>' +
  111. '<input ref="input" v-model.trim="test" type="text">{{ update }}' +
  112. '<input ref="blur"/>' +
  113. '</div>'
  114. }).$mount()
  115. document.body.appendChild(vm.$el)
  116. vm.$refs.input.focus()
  117. vm.$refs.input.value = ' abc '
  118. triggerEvent(vm.$refs.input, 'input')
  119. expect(vm.test).toBe('abc')
  120. vm.update++
  121. waitForUpdate(() => {
  122. expect(vm.$refs.input.value).toBe(' abc ')
  123. vm.$refs.blur.focus()
  124. vm.update++
  125. }).then(() => {
  126. expect(vm.$refs.input.value).toBe('abc')
  127. }).then(done)
  128. })
  129. it('multiple inputs', (done) => {
  130. const spy = jasmine.createSpy()
  131. const vm = new Vue({
  132. data: {
  133. selections: [[1, 2, 3], [4, 5]],
  134. inputList: [
  135. {
  136. name: 'questionA',
  137. data: ['a', 'b', 'c']
  138. },
  139. {
  140. name: 'questionB',
  141. data: ['1', '2']
  142. }
  143. ]
  144. },
  145. watch: {
  146. selections: spy
  147. },
  148. template:
  149. '<div>' +
  150. '<div v-for="(inputGroup, idx) in inputList">' +
  151. '<div>' +
  152. '<span v-for="(item, index) in inputGroup.data">' +
  153. '<input v-bind:name="item" type="text" v-model.number="selections[idx][index]" v-bind:id="idx+\'-\'+index"/>' +
  154. '<label>{{item}}</label>' +
  155. '</span>' +
  156. '</div>' +
  157. '</div>' +
  158. '<span ref="rs">{{selections}}</span>' +
  159. '</div>'
  160. }).$mount()
  161. const inputs = vm.$el.getElementsByTagName('input')
  162. inputs[1].value = 'test'
  163. triggerEvent(inputs[1], 'input')
  164. waitForUpdate(() => {
  165. expect(spy).toHaveBeenCalled()
  166. expect(vm.selections).toEqual([[1, 'test', 3], [4, 5]])
  167. }).then(done)
  168. })
  169. if (isIE9) {
  170. it('IE9 selectionchange', done => {
  171. const vm = new Vue({
  172. data: {
  173. test: 'foo'
  174. },
  175. template: '<input v-model="test">'
  176. }).$mount()
  177. const input = vm.$el
  178. input.value = 'bar'
  179. document.body.appendChild(input)
  180. input.focus()
  181. triggerEvent(input, 'selectionchange')
  182. waitForUpdate(() => {
  183. expect(vm.test).toBe('bar')
  184. input.value = 'a'
  185. triggerEvent(input, 'selectionchange')
  186. expect(vm.test).toBe('a')
  187. }).then(done)
  188. })
  189. }
  190. it('compositionevents', function (done) {
  191. const vm = new Vue({
  192. data: {
  193. test: 'foo'
  194. },
  195. template: '<input v-model="test">'
  196. }).$mount()
  197. const input = vm.$el
  198. triggerEvent(input, 'compositionstart')
  199. input.value = 'baz'
  200. // input before composition unlock should not call set
  201. triggerEvent(input, 'input')
  202. expect(vm.test).toBe('foo')
  203. // after composition unlock it should work
  204. triggerEvent(input, 'compositionend')
  205. triggerEvent(input, 'input')
  206. expect(vm.test).toBe('baz')
  207. done()
  208. })
  209. it('warn invalid tag', () => {
  210. new Vue({
  211. data: {
  212. test: 'foo'
  213. },
  214. template: '<div v-model="test"></div>'
  215. }).$mount()
  216. expect('<div v-model="test">: v-model is not supported on this element type').toHaveBeenWarned()
  217. })
  218. // #3468
  219. it('should have higher priority than user v-on events', () => {
  220. const spy = jasmine.createSpy()
  221. const vm = new Vue({
  222. data: {
  223. a: 'a'
  224. },
  225. template: '<input v-model="a" @input="onInput">',
  226. methods: {
  227. onInput (e) {
  228. spy(this.a)
  229. }
  230. }
  231. }).$mount()
  232. vm.$el.value = 'b'
  233. triggerEvent(vm.$el, 'input')
  234. expect(spy).toHaveBeenCalledWith('b')
  235. })
  236. it('warn binding to v-for alias', () => {
  237. new Vue({
  238. data: {
  239. strings: ['hi']
  240. },
  241. template: `
  242. <div>
  243. <div v-for="str in strings">
  244. <input v-model="str">
  245. </div>
  246. </div>
  247. `
  248. }).$mount()
  249. expect('You are binding v-model directly to a v-for iteration alias').toHaveBeenWarned()
  250. })
  251. it('warn if v-model and v-bind:value conflict', () => {
  252. new Vue({
  253. data: {
  254. test: 'foo'
  255. },
  256. template: '<input type="text" v-model="test" v-bind:value="test">'
  257. }).$mount()
  258. expect('v-bind:value="test" conflicts with v-model').toHaveBeenWarned()
  259. })
  260. it('warn if v-model and :value conflict', () => {
  261. new Vue({
  262. data: {
  263. test: 'foo'
  264. },
  265. template: '<input type="text" v-model="test" :value="test">'
  266. }).$mount()
  267. expect(':value="test" conflicts with v-model').toHaveBeenWarned()
  268. })
  269. it('should not warn on radio, checkbox, or custom component', () => {
  270. new Vue({
  271. data: { test: '' },
  272. components: {
  273. foo: {
  274. props: ['model', 'value'],
  275. model: { prop: 'model', event: 'change' },
  276. template: `<div/>`
  277. }
  278. },
  279. template: `
  280. <div>
  281. <input type="checkbox" v-model="test" :value="test">
  282. <input type="radio" v-model="test" :value="test">
  283. <foo v-model="test" :value="test"/>
  284. </div>
  285. `
  286. }).$mount()
  287. expect('conflicts with v-model').not.toHaveBeenWarned()
  288. })
  289. it('should not warn on input with dynamic type binding', () => {
  290. new Vue({
  291. data: {
  292. type: 'checkbox',
  293. test: 'foo'
  294. },
  295. template: '<input :type="type" v-model="test" :value="test">'
  296. }).$mount()
  297. expect('conflicts with v-model').not.toHaveBeenWarned()
  298. })
  299. if (!isAndroid) {
  300. it('does not trigger extra input events with single compositionend', () => {
  301. const spy = jasmine.createSpy()
  302. const vm = new Vue({
  303. data: {
  304. a: 'a'
  305. },
  306. template: '<input v-model="a" @input="onInput">',
  307. methods: {
  308. onInput (e) {
  309. spy(e.target.value)
  310. }
  311. }
  312. }).$mount()
  313. expect(spy.calls.count()).toBe(0)
  314. vm.$el.value = 'b'
  315. triggerEvent(vm.$el, 'input')
  316. expect(spy.calls.count()).toBe(1)
  317. triggerEvent(vm.$el, 'compositionend')
  318. expect(spy.calls.count()).toBe(1)
  319. })
  320. it('triggers extra input on compositionstart + end', () => {
  321. const spy = jasmine.createSpy()
  322. const vm = new Vue({
  323. data: {
  324. a: 'a'
  325. },
  326. template: '<input v-model="a" @input="onInput">',
  327. methods: {
  328. onInput (e) {
  329. spy(e.target.value)
  330. }
  331. }
  332. }).$mount()
  333. expect(spy.calls.count()).toBe(0)
  334. vm.$el.value = 'b'
  335. triggerEvent(vm.$el, 'input')
  336. expect(spy.calls.count()).toBe(1)
  337. triggerEvent(vm.$el, 'compositionstart')
  338. triggerEvent(vm.$el, 'compositionend')
  339. expect(spy.calls.count()).toBe(2)
  340. })
  341. // #4392
  342. it('should not update value with modifiers when in focus if post-conversion values are the same', done => {
  343. const vm = new Vue({
  344. data: {
  345. a: 1,
  346. foo: false
  347. },
  348. template: '<div>{{ foo }}<input ref="input" v-model.number="a"></div>'
  349. }).$mount()
  350. document.body.appendChild(vm.$el)
  351. vm.$refs.input.focus()
  352. vm.$refs.input.value = '1.000'
  353. vm.foo = true
  354. waitForUpdate(() => {
  355. expect(vm.$refs.input.value).toBe('1.000')
  356. }).then(done)
  357. })
  358. // #6552
  359. // This was original introduced due to the microtask between DOM events issue
  360. // but fixed after switching to MessageChannel.
  361. it('should not block input when another input listener with modifier is used', done => {
  362. const vm = new Vue({
  363. data: {
  364. a: 'a',
  365. foo: false
  366. },
  367. template: `
  368. <div>
  369. <input ref="input" v-model="a" @input.capture="onInput">{{ a }}
  370. <div v-if="foo">foo</div>
  371. </div>
  372. `,
  373. methods: {
  374. onInput (e) {
  375. this.foo = true
  376. }
  377. }
  378. }).$mount()
  379. document.body.appendChild(vm.$el)
  380. vm.$refs.input.focus()
  381. vm.$refs.input.value = 'b'
  382. triggerEvent(vm.$refs.input, 'input')
  383. // not using wait for update here because there will be two update cycles
  384. // one caused by onInput in the first listener
  385. setTimeout(() => {
  386. expect(vm.a).toBe('b')
  387. expect(vm.$refs.input.value).toBe('b')
  388. done()
  389. }, 16)
  390. })
  391. it('should create and make reactive non-existent properties', done => {
  392. const vm = new Vue({
  393. data: {
  394. foo: {}
  395. },
  396. template: '<input v-model="foo.bar">'
  397. }).$mount()
  398. expect(vm.$el.value).toBe('')
  399. vm.$el.value = 'a'
  400. triggerEvent(vm.$el, 'input')
  401. expect(vm.foo.bar).toBe('a')
  402. vm.foo.bar = 'b'
  403. waitForUpdate(() => {
  404. expect(vm.$el.value).toBe('b')
  405. vm.foo = {}
  406. }).then(() => {
  407. expect(vm.$el.value).toBe('')
  408. }).then(done)
  409. })
  410. }
  411. // #7138
  412. if (isIE && !isIE9) {
  413. it('should not fire input on initial render of textarea with placeholder in IE10/11', done => {
  414. const el = document.createElement('div')
  415. document.body.appendChild(el)
  416. const vm = new Vue({
  417. el,
  418. data: { foo: null },
  419. template: `<textarea v-model="foo" placeholder="bar"></textarea>`
  420. })
  421. setTimeout(() => {
  422. expect(vm.foo).toBe(null)
  423. done()
  424. }, 17)
  425. })
  426. }
  427. })