<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
  <channel>
    <title>表单校验 on Saiga</title>
    <link>http://localhost:1313/tags/%E8%A1%A8%E5%8D%95%E6%A0%A1%E9%AA%8C/</link>
    <description>Recent content in 表单校验 on Saiga</description>
    <generator>Hugo</generator>
    <language>zh-cn</language>
    <managingEditor>wuwenzen@outlook.com (wuwj)</managingEditor>
    <webMaster>wuwenzen@outlook.com (wuwj)</webMaster>
    <lastBuildDate>Tue, 18 Jun 2019 00:00:00 +0000</lastBuildDate>
    <atom:link href="http://localhost:1313/tags/%E8%A1%A8%E5%8D%95%E6%A0%A1%E9%AA%8C/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>复杂表单实战：我在项目里踩过的 Vue 坑</title>
      <link>http://localhost:1313/posts/2019-06-18-vue-form-best-practices/</link>
      <pubDate>Tue, 18 Jun 2019 00:00:00 +0000</pubDate><author>wuwenzen@outlook.com (wuwj)</author>
      <guid>http://localhost:1313/posts/2019-06-18-vue-form-best-practices/</guid>
      <description>&lt;p&gt;表单是前端最常见、也最容易写成一坨的东西。简单表单随便放几个 &lt;code&gt;v-model&lt;/code&gt; 都能跑，一旦字段多起来、存在动态联动、跨步校验，代码就开始失控。&lt;/p&gt;&#xA;&lt;p&gt;这篇文章基于我做过的一个&amp;quot;几十个字段 + 多步 + 条件显示&amp;quot;的表单项目，总结一些在 Vue 里踩过的坑和解决方案。&lt;/p&gt;&#xA;&lt;hr&gt;&#xA;&lt;h2 id=&#34;需求背景一个略显超标的表单&#34;&gt;需求背景：一个略显「超标」的表单&lt;/h2&gt;&#xA;&lt;p&gt;当时的表单大概长这样：&lt;/p&gt;&#xA;&lt;ul&gt;&#xA;&lt;li&gt;3 个步骤：基础信息 → 业务信息 → 确认提交&lt;/li&gt;&#xA;&lt;li&gt;每步有 8～15 个字段&lt;/li&gt;&#xA;&lt;li&gt;部分字段需要「根据上一步选择动态显示或隐藏」&lt;/li&gt;&#xA;&lt;li&gt;有跨字段校验（比如：结束日期 ≥ 开始日期）&lt;/li&gt;&#xA;&lt;li&gt;需要支持「草稿保存」+「再次打开自动回填」&lt;/li&gt;&#xA;&lt;/ul&gt;&#xA;&lt;p&gt;如果&lt;strong&gt;一股脑全写在一个组件里&lt;/strong&gt;，大概会出现：&lt;/p&gt;&#xA;&lt;ul&gt;&#xA;&lt;li&gt;data 里一个巨大的 &lt;code&gt;form = { ... }&lt;/code&gt;。&lt;/li&gt;&#xA;&lt;li&gt;各种 &lt;code&gt;watch&lt;/code&gt; 做联动逻辑。&lt;/li&gt;&#xA;&lt;li&gt;methods 里各种 &lt;code&gt;validateXXX&lt;/code&gt;，互相依赖。&lt;/li&gt;&#xA;&lt;li&gt;维护成本爆炸。&lt;/li&gt;&#xA;&lt;/ul&gt;&#xA;&lt;hr&gt;&#xA;&lt;h2 id=&#34;核心思路拆数据结构--拆-ui--拆校验&#34;&gt;核心思路：&lt;strong&gt;拆数据结构 + 拆 UI + 拆校验&lt;/strong&gt;&lt;/h2&gt;&#xA;&lt;h3 id=&#34;1-数据结构要先定清楚&#34;&gt;1. 数据结构要先定清楚&lt;/h3&gt;&#xA;&lt;p&gt;我一般会在一开始，就写出一个完整的 &lt;code&gt;formModel&lt;/code&gt;：&lt;/p&gt;&#xA;&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;&#34;&gt;&lt;code class=&#34;language-js&#34; data-lang=&#34;js&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;const&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;defaultForm&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; () =&amp;gt; ({&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  &lt;span style=&#34;color:#a6e22e&#34;&gt;basic&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; {&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#a6e22e&#34;&gt;name&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; &lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#39;&amp;#39;&lt;/span&gt;,&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#a6e22e&#34;&gt;idNo&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; &lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#39;&amp;#39;&lt;/span&gt;,&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#a6e22e&#34;&gt;phone&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; &lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#39;&amp;#39;&lt;/span&gt;,&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  },&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  &lt;span style=&#34;color:#a6e22e&#34;&gt;business&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; {&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#a6e22e&#34;&gt;type&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; &lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#39;&amp;#39;&lt;/span&gt;,&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#a6e22e&#34;&gt;amount&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; &lt;span style=&#34;color:#66d9ef&#34;&gt;null&lt;/span&gt;,&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#a6e22e&#34;&gt;startDate&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; &lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#39;&amp;#39;&lt;/span&gt;,&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#a6e22e&#34;&gt;endDate&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; &lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#39;&amp;#39;&lt;/span&gt;,&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  },&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  &lt;span style=&#34;color:#a6e22e&#34;&gt;extra&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; {&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#a6e22e&#34;&gt;agree&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; &lt;span style=&#34;color:#66d9ef&#34;&gt;false&lt;/span&gt;,&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#a6e22e&#34;&gt;comment&lt;/span&gt;&lt;span style=&#34;color:#f92672&#34;&gt;:&lt;/span&gt; &lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#39;&amp;#39;&lt;/span&gt;,&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  },&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;});&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;在组件里：&lt;/p&gt;</description>
    </item>
  </channel>
</rss>
