2
0

model-text.spec.js 11 KB

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