ssrDirectives.spec.ts 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  1. import { renderToString } from '../src/renderToString'
  2. import {
  3. createApp,
  4. h,
  5. withDirectives,
  6. vShow,
  7. vModelText,
  8. vModelRadio,
  9. vModelCheckbox
  10. } from 'vue'
  11. describe('ssr: directives', () => {
  12. describe('template v-show', () => {
  13. test('basic', async () => {
  14. expect(
  15. await renderToString(
  16. createApp({
  17. template: `<div v-show="true"/>`
  18. })
  19. )
  20. ).toBe(`<div style=""></div>`)
  21. expect(
  22. await renderToString(
  23. createApp({
  24. template: `<div v-show="false"/>`
  25. })
  26. )
  27. ).toBe(`<div style="display:none;"></div>`)
  28. })
  29. test('with static style', async () => {
  30. expect(
  31. await renderToString(
  32. createApp({
  33. template: `<div style="color:red" v-show="false"/>`
  34. })
  35. )
  36. ).toBe(`<div style="color:red;display:none;"></div>`)
  37. })
  38. test('with dynamic style', async () => {
  39. expect(
  40. await renderToString(
  41. createApp({
  42. data: () => ({ style: { color: 'red' } }),
  43. template: `<div :style="style" v-show="false"/>`
  44. })
  45. )
  46. ).toBe(`<div style="color:red;display:none;"></div>`)
  47. })
  48. test('with static + dynamic style', async () => {
  49. expect(
  50. await renderToString(
  51. createApp({
  52. data: () => ({ style: { color: 'red' } }),
  53. template: `<div :style="style" style="font-size:12;" v-show="false"/>`
  54. })
  55. )
  56. ).toBe(`<div style="color:red;font-size:12;display:none;"></div>`)
  57. })
  58. })
  59. describe('template v-model', () => {
  60. test('text', async () => {
  61. expect(
  62. await renderToString(
  63. createApp({
  64. data: () => ({ text: 'hello' }),
  65. template: `<input v-model="text">`
  66. })
  67. )
  68. ).toBe(`<input value="hello">`)
  69. })
  70. test('radio', async () => {
  71. expect(
  72. await renderToString(
  73. createApp({
  74. data: () => ({ selected: 'foo' }),
  75. template: `<input type="radio" value="foo" v-model="selected">`
  76. })
  77. )
  78. ).toBe(`<input type="radio" value="foo" checked>`)
  79. expect(
  80. await renderToString(
  81. createApp({
  82. data: () => ({ selected: 'foo' }),
  83. template: `<input type="radio" value="bar" v-model="selected">`
  84. })
  85. )
  86. ).toBe(`<input type="radio" value="bar">`)
  87. // non-string values
  88. expect(
  89. await renderToString(
  90. createApp({
  91. data: () => ({ selected: 'foo' }),
  92. template: `<input type="radio" :value="{}" v-model="selected">`
  93. })
  94. )
  95. ).toBe(`<input type="radio">`)
  96. })
  97. test('checkbox', async () => {
  98. expect(
  99. await renderToString(
  100. createApp({
  101. data: () => ({ checked: true }),
  102. template: `<input type="checkbox" v-model="checked">`
  103. })
  104. )
  105. ).toBe(`<input type="checkbox" checked>`)
  106. expect(
  107. await renderToString(
  108. createApp({
  109. data: () => ({ checked: false }),
  110. template: `<input type="checkbox" v-model="checked">`
  111. })
  112. )
  113. ).toBe(`<input type="checkbox">`)
  114. expect(
  115. await renderToString(
  116. createApp({
  117. data: () => ({ checked: ['foo'] }),
  118. template: `<input type="checkbox" value="foo" v-model="checked">`
  119. })
  120. )
  121. ).toBe(`<input type="checkbox" value="foo" checked>`)
  122. expect(
  123. await renderToString(
  124. createApp({
  125. data: () => ({ checked: [] }),
  126. template: `<input type="checkbox" value="foo" v-model="checked">`
  127. })
  128. )
  129. ).toBe(`<input type="checkbox" value="foo">`)
  130. })
  131. test('textarea', async () => {
  132. expect(
  133. await renderToString(
  134. createApp({
  135. data: () => ({ foo: 'hello' }),
  136. template: `<textarea v-model="foo"/>`
  137. })
  138. )
  139. ).toBe(`<textarea>hello</textarea>`)
  140. })
  141. test('dynamic type', async () => {
  142. expect(
  143. await renderToString(
  144. createApp({
  145. data: () => ({ type: 'text', model: 'hello' }),
  146. template: `<input :type="type" v-model="model">`
  147. })
  148. )
  149. ).toBe(`<input type="text" value="hello">`)
  150. expect(
  151. await renderToString(
  152. createApp({
  153. data: () => ({ type: 'checkbox', model: true }),
  154. template: `<input :type="type" v-model="model">`
  155. })
  156. )
  157. ).toBe(`<input type="checkbox" checked>`)
  158. expect(
  159. await renderToString(
  160. createApp({
  161. data: () => ({ type: 'checkbox', model: false }),
  162. template: `<input :type="type" v-model="model">`
  163. })
  164. )
  165. ).toBe(`<input type="checkbox">`)
  166. expect(
  167. await renderToString(
  168. createApp({
  169. data: () => ({ type: 'checkbox', model: ['hello'] }),
  170. template: `<input :type="type" value="hello" v-model="model">`
  171. })
  172. )
  173. ).toBe(`<input type="checkbox" value="hello" checked>`)
  174. expect(
  175. await renderToString(
  176. createApp({
  177. data: () => ({ type: 'checkbox', model: [] }),
  178. template: `<input :type="type" value="hello" v-model="model">`
  179. })
  180. )
  181. ).toBe(`<input type="checkbox" value="hello">`)
  182. expect(
  183. await renderToString(
  184. createApp({
  185. data: () => ({ type: 'radio', model: 'hello' }),
  186. template: `<input :type="type" value="hello" v-model="model">`
  187. })
  188. )
  189. ).toBe(`<input type="radio" value="hello" checked>`)
  190. expect(
  191. await renderToString(
  192. createApp({
  193. data: () => ({ type: 'radio', model: 'hello' }),
  194. template: `<input :type="type" value="bar" v-model="model">`
  195. })
  196. )
  197. ).toBe(`<input type="radio" value="bar">`)
  198. })
  199. test('with v-bind', async () => {
  200. expect(
  201. await renderToString(
  202. createApp({
  203. data: () => ({
  204. obj: { type: 'radio', value: 'hello' },
  205. model: 'hello'
  206. }),
  207. template: `<input v-bind="obj" v-model="model">`
  208. })
  209. )
  210. ).toBe(`<input type="radio" value="hello" checked>`)
  211. })
  212. })
  213. describe('vnode v-show', () => {
  214. test('basic', async () => {
  215. expect(
  216. await renderToString(
  217. createApp({
  218. render() {
  219. return withDirectives(h('div'), [[vShow, true]])
  220. }
  221. })
  222. )
  223. ).toBe(`<div></div>`)
  224. expect(
  225. await renderToString(
  226. createApp({
  227. render() {
  228. return withDirectives(h('div'), [[vShow, false]])
  229. }
  230. })
  231. )
  232. ).toBe(`<div style="display:none;"></div>`)
  233. })
  234. test('with merge', async () => {
  235. expect(
  236. await renderToString(
  237. createApp({
  238. render() {
  239. return withDirectives(
  240. h('div', {
  241. style: {
  242. color: 'red'
  243. }
  244. }),
  245. [[vShow, false]]
  246. )
  247. }
  248. })
  249. )
  250. ).toBe(`<div style="color:red;display:none;"></div>`)
  251. })
  252. })
  253. describe('vnode v-model', () => {
  254. test('text', async () => {
  255. expect(
  256. await renderToString(
  257. createApp({
  258. render() {
  259. return withDirectives(h('input'), [[vModelText, 'hello']])
  260. }
  261. })
  262. )
  263. ).toBe(`<input value="hello">`)
  264. })
  265. test('radio', async () => {
  266. expect(
  267. await renderToString(
  268. createApp({
  269. render() {
  270. return withDirectives(
  271. h('input', { type: 'radio', value: 'hello' }),
  272. [[vModelRadio, 'hello']]
  273. )
  274. }
  275. })
  276. )
  277. ).toBe(`<input type="radio" value="hello" checked>`)
  278. expect(
  279. await renderToString(
  280. createApp({
  281. render() {
  282. return withDirectives(
  283. h('input', { type: 'radio', value: 'hello' }),
  284. [[vModelRadio, 'foo']]
  285. )
  286. }
  287. })
  288. )
  289. ).toBe(`<input type="radio" value="hello">`)
  290. })
  291. test('checkbox', async () => {
  292. expect(
  293. await renderToString(
  294. createApp({
  295. render() {
  296. return withDirectives(h('input', { type: 'checkbox' }), [
  297. [vModelCheckbox, true]
  298. ])
  299. }
  300. })
  301. )
  302. ).toBe(`<input type="checkbox" checked>`)
  303. expect(
  304. await renderToString(
  305. createApp({
  306. render() {
  307. return withDirectives(h('input', { type: 'checkbox' }), [
  308. [vModelCheckbox, false]
  309. ])
  310. }
  311. })
  312. )
  313. ).toBe(`<input type="checkbox">`)
  314. expect(
  315. await renderToString(
  316. createApp({
  317. render() {
  318. return withDirectives(
  319. h('input', { type: 'checkbox', value: 'foo' }),
  320. [[vModelCheckbox, ['foo']]]
  321. )
  322. }
  323. })
  324. )
  325. ).toBe(`<input type="checkbox" value="foo" checked>`)
  326. expect(
  327. await renderToString(
  328. createApp({
  329. render() {
  330. return withDirectives(
  331. h('input', { type: 'checkbox', value: 'foo' }),
  332. [[vModelCheckbox, []]]
  333. )
  334. }
  335. })
  336. )
  337. ).toBe(`<input type="checkbox" value="foo">`)
  338. })
  339. })
  340. test('custom directive w/ getSSRProps', async () => {
  341. expect(
  342. await renderToString(
  343. createApp({
  344. render() {
  345. return withDirectives(h('div'), [
  346. [
  347. {
  348. getSSRProps({ value }) {
  349. return { id: value }
  350. }
  351. },
  352. 'foo'
  353. ]
  354. ])
  355. }
  356. })
  357. )
  358. ).toBe(`<div id="foo"></div>`)
  359. })
  360. })