model-text.spec.js 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  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. // should let strings pass through
  46. vm.$el.value = 'f'
  47. triggerEvent(vm.$el, 'input')
  48. expect(vm.test).toBe('f')
  49. })
  50. it('.trim modifier', () => {
  51. const vm = new Vue({
  52. data: {
  53. test: 'hi'
  54. },
  55. template: '<input v-model.trim="test">'
  56. }).$mount()
  57. expect(vm.test).toBe('hi')
  58. vm.$el.value = ' what '
  59. triggerEvent(vm.$el, 'input')
  60. expect(vm.test).toBe('what')
  61. })
  62. it('.number focus and typing', (done) => {
  63. const vm = new Vue({
  64. data: {
  65. test: 0,
  66. update: 0
  67. },
  68. template:
  69. '<div>' +
  70. '<input ref="input" v-model.number="test">{{ update }}' +
  71. '<input ref="blur">' +
  72. '</div>'
  73. }).$mount()
  74. document.body.appendChild(vm.$el)
  75. vm.$refs.input.focus()
  76. expect(vm.test).toBe(0)
  77. vm.$refs.input.value = '1.0'
  78. triggerEvent(vm.$refs.input, 'input')
  79. expect(vm.test).toBe(1)
  80. vm.update++
  81. waitForUpdate(() => {
  82. expect(vm.$refs.input.value).toBe('1.0')
  83. vm.$refs.blur.focus()
  84. vm.update++
  85. }).then(() => {
  86. expect(vm.$refs.input.value).toBe('1')
  87. }).then(done)
  88. })
  89. it('.trim focus and typing', (done) => {
  90. const vm = new Vue({
  91. data: {
  92. test: 'abc',
  93. update: 0
  94. },
  95. template:
  96. '<div>' +
  97. '<input ref="input" v-model.trim="test" type="text">{{ update }}' +
  98. '<input ref="blur"/>' +
  99. '</div>'
  100. }).$mount()
  101. document.body.appendChild(vm.$el)
  102. vm.$refs.input.focus()
  103. vm.$refs.input.value = ' abc '
  104. triggerEvent(vm.$refs.input, 'input')
  105. expect(vm.test).toBe('abc')
  106. vm.update++
  107. waitForUpdate(() => {
  108. expect(vm.$refs.input.value).toBe(' abc ')
  109. vm.$refs.blur.focus()
  110. vm.update++
  111. }).then(() => {
  112. expect(vm.$refs.input.value).toBe('abc')
  113. }).then(done)
  114. })
  115. it('multiple inputs', (done) => {
  116. const spy = jasmine.createSpy()
  117. const vm = new Vue({
  118. data: {
  119. selections: [[1, 2, 3], [4, 5]],
  120. inputList: [
  121. {
  122. name: 'questionA',
  123. data: ['a', 'b', 'c']
  124. },
  125. {
  126. name: 'questionB',
  127. data: ['1', '2']
  128. }
  129. ]
  130. },
  131. watch: {
  132. selections: spy
  133. },
  134. template:
  135. '<div>' +
  136. '<div v-for="(inputGroup, idx) in inputList">' +
  137. '<div>' +
  138. '<span v-for="(item, index) in inputGroup.data">' +
  139. '<input v-bind:name="item" type="text" v-model.number="selections[idx][index]" v-bind:id="idx+\'-\'+index"/>' +
  140. '<label>{{item}}</label>' +
  141. '</span>' +
  142. '</div>' +
  143. '</div>' +
  144. '<span ref="rs">{{selections}}</span>' +
  145. '</div>'
  146. }).$mount()
  147. var inputs = vm.$el.getElementsByTagName('input')
  148. inputs[1].value = 'test'
  149. triggerEvent(inputs[1], 'input')
  150. waitForUpdate(() => {
  151. expect(spy).toHaveBeenCalled()
  152. expect(vm.selections).toEqual([[1, 'test', 3], [4, 5]])
  153. }).then(done)
  154. })
  155. if (isIE9) {
  156. it('IE9 selectionchange', done => {
  157. const vm = new Vue({
  158. data: {
  159. test: 'foo'
  160. },
  161. template: '<input v-model="test">'
  162. }).$mount()
  163. const input = vm.$el
  164. input.value = 'bar'
  165. document.body.appendChild(input)
  166. input.focus()
  167. triggerEvent(input, 'selectionchange')
  168. waitForUpdate(() => {
  169. expect(vm.test).toBe('bar')
  170. input.value = 'a'
  171. triggerEvent(input, 'selectionchange')
  172. expect(vm.test).toBe('a')
  173. }).then(done)
  174. })
  175. }
  176. if (!isAndroid) {
  177. it('compositionevents', function (done) {
  178. const vm = new Vue({
  179. data: {
  180. test: 'foo'
  181. },
  182. template: '<input v-model="test">'
  183. }).$mount()
  184. const input = vm.$el
  185. triggerEvent(input, 'compositionstart')
  186. input.value = 'baz'
  187. // input before composition unlock should not call set
  188. triggerEvent(input, 'input')
  189. expect(vm.test).toBe('foo')
  190. // after composition unlock it should work
  191. triggerEvent(input, 'compositionend')
  192. triggerEvent(input, 'input')
  193. expect(vm.test).toBe('baz')
  194. done()
  195. })
  196. }
  197. it('warn inline value attribute', () => {
  198. const vm = new Vue({
  199. data: {
  200. test: 'foo'
  201. },
  202. template: '<input v-model="test" value="bar">'
  203. }).$mount()
  204. expect(vm.test).toBe('foo')
  205. expect(vm.$el.value).toBe('foo')
  206. expect('inline value attributes will be ignored').toHaveBeenWarned()
  207. })
  208. it('warn textarea inline content', function () {
  209. const vm = new Vue({
  210. data: {
  211. test: 'foo'
  212. },
  213. template: '<textarea v-model="test">bar</textarea>'
  214. }).$mount()
  215. expect(vm.test).toBe('foo')
  216. expect(vm.$el.value).toBe('foo')
  217. expect('inline content inside <textarea> will be ignored').toHaveBeenWarned()
  218. })
  219. it('warn invalid tag', () => {
  220. new Vue({
  221. data: {
  222. test: 'foo'
  223. },
  224. template: '<div v-model="test"></div>'
  225. }).$mount()
  226. expect('v-model is not supported on element type: <div>').toHaveBeenWarned()
  227. })
  228. // #3468
  229. it('should have higher priority than user v-on events', () => {
  230. const spy = jasmine.createSpy()
  231. const vm = new Vue({
  232. data: {
  233. a: 'a'
  234. },
  235. template: '<input v-model="a" @input="onInput">',
  236. methods: {
  237. onInput (e) {
  238. spy(e.target.value)
  239. }
  240. }
  241. }).$mount()
  242. vm.$el.value = 'b'
  243. triggerEvent(vm.$el, 'input')
  244. expect(spy).toHaveBeenCalledWith('b')
  245. })
  246. it('warn binding to v-for alias', () => {
  247. new Vue({
  248. data: {
  249. strings: ['hi']
  250. },
  251. template: `
  252. <div>
  253. <div v-for="str in strings">
  254. <input v-model="str">
  255. </div>
  256. </div>
  257. `
  258. }).$mount()
  259. expect('You are binding v-model directly to a v-for iteration alias').toHaveBeenWarned()
  260. })
  261. })