{
    "componentChunkName": "component---src-templates-blog-post-jsx",
    "path": "/post/how-to-validate-form-value-with-constraint-validation/",
    "result": {"data":{"site":{"siteMetadata":{"title":"WEB EGG","author":"Leko - CTO at Yuimedi"}},"markdownRemark":{"id":"984457af-13ec-54f0-ae19-270fbae94629","excerpt":"こんにちは。 Hamee Advent Calendar 2015 8日目の記事です。 Chromeの開発者ツールでHTMLの要素を眺めていたらvalidityという属性を見つけました validityプロパティの中にはpatternMismatch, tooLongなどそれっぽいプロパティとbool…","html":"<p>こんにちは。<br>\n<a href=\"http://qiita.com/advent-calendar/2015/hamee\" target=\"_blank\" rel=\"nofollow noopener noreferrer\">Hamee Advent Calendar 2015</a> 8日目の記事です。</p>\n<p>Chromeの開発者ツールでHTMLの要素を眺めていたら<code>validity</code>という属性を見つけました<br>\nvalidityプロパティの中には<code>patternMismatch</code>, <code>tooLong</code>などそれっぽいプロパティとbool値。</p>\n<p>調べてみたら <strong>フォームのバリデーションはこれだけで十分なのでは</strong><br>\nと思うほどに便利だったので詳しい機能の紹介と、より便利に扱う小ネタの紹介です。</p>\n<!--more-->\n<h2 id=\"参考\" style=\"position:relative;\"><a href=\"#%E5%8F%82%E8%80%83\" aria-label=\"参考 permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>参考</h2>\n<p>情報源として主に以下のページを参考にしました。</p>\n<ul>\n<li><a href=\"http://www.html5rocks.com/en/tutorials/forms/constraintvalidation/\" target=\"_blank\" rel=\"nofollow noopener noreferrer\">Constraint Validation: Native Client Side Validation for Web Forms</a></li>\n<li><a href=\"https://developer.mozilla.org/en-US/docs/Web/API/ValidityState\" target=\"_blank\" rel=\"nofollow noopener noreferrer\">ValidityState – Web APIs | MDN</a></li>\n<li><a href=\"http://dev.w3.org/html5/spec-preview/constraints.html\" target=\"_blank\" rel=\"nofollow noopener noreferrer\">4.10.21 Constraints — HTML5</a></li>\n</ul>\n<h2 id=\"デモ\" style=\"position:relative;\"><a href=\"#%E3%83%87%E3%83%A2\" aria-label=\"デモ permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>デモ</h2>\n<p>今回の記事でご紹介する内容をまとめたデモを<a href=\"https://jsfiddle.net/leko/nwkLcry5/\" target=\"_blank\" rel=\"nofollow noopener noreferrer\">こちら</a>に上げました。<br>\n記事の内容を読みつつデモの動きやコードをご参考ください。</p>\n<h2 id=\"いいこと\" style=\"position:relative;\"><a href=\"#%E3%81%84%E3%81%84%E3%81%93%E3%81%A8\" aria-label=\"いいこと permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>いいこと</h2>\n<h3 id=\"標準仕様\" style=\"position:relative;\"><a href=\"#%E6%A8%99%E6%BA%96%E4%BB%95%E6%A7%98\" aria-label=\"標準仕様 permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>標準仕様</h3>\n<p>単にHTML5のform周りの属性を使用してタグを書くだけでいいので、ライブラリやフレームワークのようにロックインが発生せず、HTML構造にも依存せず、「捨てる/捨てない」の話ではなく、最低限がそれです。非常に良いです。<br>\n独自の仕様が存在しないので、HTMLの属性やjsが汚れずに済みますし、幅広いライブラリを活用できるチャンスが生まれます。</p>\n<h3 id=\"バリデーション処理のメンテナンス実装不要\" style=\"position:relative;\"><a href=\"#%E3%83%90%E3%83%AA%E3%83%87%E3%83%BC%E3%82%B7%E3%83%A7%E3%83%B3%E5%87%A6%E7%90%86%E3%81%AE%E3%83%A1%E3%83%B3%E3%83%86%E3%83%8A%E3%83%B3%E3%82%B9%E5%AE%9F%E8%A3%85%E4%B8%8D%E8%A6%81\" aria-label=\"バリデーション処理のメンテナンス実装不要 permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>バリデーション処理のメンテナンス・実装不要</h3>\n<p>ブラウザ内で入力チェックをしてくれるので、</p>\n<ul>\n<li>inputのtype属性やselect、textareaによる差を吸収して値を取得する</li>\n<li>取得した値と、オレオレ定義を元にバリデーションする</li>\n</ul>\n<p>などの処理が不要になります。<br>\nちょっとしたバリデーションだし、と個々に自前で実装してしまうこともあるかと思います。そういったオレオレ実装をメンテナンスする手間が不要になります。<br>\n少なくとも自分で書くより、ブラウザ側に入力の検証をしてもらうほうが、手間が減るだけでなく圧倒的にバグのリスクやメンテナンスコストが下がります。</p>\n<p>残念ながらエラー内容を表示する処理に関しては現段階では自由度がほぼない（後述）ので、そこは自前で実装する必要があると思います。</p>\n<h3 id=\"便利な擬似クラスが利用できる\" style=\"position:relative;\"><a href=\"#%E4%BE%BF%E5%88%A9%E3%81%AA%E6%93%AC%E4%BC%BC%E3%82%AF%E3%83%A9%E3%82%B9%E3%81%8C%E5%88%A9%E7%94%A8%E3%81%A7%E3%81%8D%E3%82%8B\" aria-label=\"便利な擬似クラスが利用できる permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>便利な擬似クラスが利用できる</h3>\n<p>きちんと属性を定義しておくと、<code>:valid</code>, <code>:invalid</code>, <code>:required</code>, <code>:optional</code>といったフォーム周りの擬似クラスの恩恵が受けられるようになります。<br>\nIE10+なら対応しているので、新しいブラウザ向けのサービスを作る際には検討の余地があるかと思います。</p>\n<h2 id=\"checkvalidityとvaliditystate\" style=\"position:relative;\"><a href=\"#checkvalidity%E3%81%A8validitystate\" aria-label=\"checkvalidityとvaliditystate permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>checkValidityとValidityState</h2>\n<p>では早速本題に入ります。まずは例から。</p>\n<div class=\"gatsby-highlight\" data-language=\"html\"><pre class=\"language-html\"><code class=\"language-html\"><span class=\"token tag\"><span class=\"token tag\"><span class=\"token punctuation\">&lt;</span>input</span> <span class=\"token attr-name\">id</span><span class=\"token attr-value\"><span class=\"token punctuation attr-equals\">=</span><span class=\"token punctuation\">\"</span>form-text<span class=\"token punctuation\">\"</span></span> <span class=\"token attr-name\">type</span><span class=\"token attr-value\"><span class=\"token punctuation attr-equals\">=</span><span class=\"token punctuation\">\"</span>text<span class=\"token punctuation\">\"</span></span> <span class=\"token attr-name\">required</span> <span class=\"token punctuation\">/></span></span></code></pre></div>\n<p>といったHTMLを用意し、以下の様なjsを書いてみます。</p>\n<div class=\"gatsby-highlight\" data-language=\"javascript\"><pre class=\"language-javascript\"><code class=\"language-javascript\">console<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span><span class=\"token function\">checkValidity</span><span class=\"token punctuation\">(</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>    <span class=\"token comment\">// false</span>\n\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>validity<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>   <span class=\"token comment\">// ValidityState {}</span>\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>validity<span class=\"token punctuation\">.</span>valueMissing<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// true</span></code></pre></div>\n<p><code>checkValidity</code>メソッドの戻り値はfalseで、<code>validity.valueMissing</code>プロパティの値は<code>true</code>にっていると思います。<br>\ncheckValidityメソッドはその要素のバリデーション全てに通過した場合trueになります。<br>\nvalidity以下のオブジェクトは、バリデーションに通過した場合はfalse、バリデーションに引っかかった場合trueになります。</p>\n<p>ちょっとややこしですが名前が肯定形になっているので、そこまで勘違いは起きないだろうと思います。<br>\n<code>validity.valid</code>だけやや特殊で、checkValidityメソッドの戻り値と等しくなります。つまりこの値だけture/falseの意味が逆です。ご注意ください</p>\n<h2 id=\"willvalidateとnovalidatedisabled\" style=\"position:relative;\"><a href=\"#willvalidate%E3%81%A8novalidatedisabled\" aria-label=\"willvalidateとnovalidatedisabled permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>willValidateとnovalidate、disabled</h2>\n<p>どんどんいきます。</p>\n<p>willValidateというプロパティもあります。<br>\nこのプロパティは、その要素のバリデーションが必要か否かを表します。<br>\n候補の場合にtrueが、候補でない時にはfalseが、そもそもinput系のタグじゃない場合にはプロパティ自体が存在しません。</p>\n<div class=\"gatsby-highlight\" data-language=\"html\"><pre class=\"language-html\"><code class=\"language-html\"></code></pre></div>\n<div class=\"gatsby-highlight\" data-language=\"javascript\"><pre class=\"language-javascript\"><code class=\"language-javascript\">console<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text1'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// false</span>\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text2'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// true</span>\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text3'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// true</span>\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text4'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// false</span>\n\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text5'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// false</span>\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text6'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// true</span>\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text7'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// true</span>\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text8'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// false</span>\n\ndocument<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text8'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>readOnly <span class=\"token operator\">=</span> <span class=\"token boolean\">false</span><span class=\"token punctuation\">;</span>\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text8'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// true</span>\n\ndocument<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text1'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>value <span class=\"token operator\">=</span> <span class=\"token string\">'hogehoge'</span><span class=\"token punctuation\">;</span>\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span><span class=\"token function\">getElementById</span><span class=\"token punctuation\">(</span><span class=\"token string\">'form-text1'</span><span class=\"token punctuation\">)</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>  <span class=\"token comment\">// false</span>\n\nconsole<span class=\"token punctuation\">.</span><span class=\"token function\">log</span><span class=\"token punctuation\">(</span>document<span class=\"token punctuation\">.</span>forms<span class=\"token punctuation\">[</span><span class=\"token number\">0</span><span class=\"token punctuation\">]</span><span class=\"token punctuation\">.</span>willValidate<span class=\"token punctuation\">)</span><span class=\"token punctuation\">;</span>    <span class=\"token comment\">// undefined</span></code></pre></div>\n<p><code>disabled</code>や<code>readonly</code>になっている要素はfalseになるようです。<br>\njsや開発者ツールなどでHTMLを書き換え、readonlyやdisabledを外すと自動的にwillValidateの値が変わるようです。</p>\n<p>disabledを保ったままvalue属性を書き換えた場合のwillValidateの値は<code>false</code>になってしまいますが、悪意のある入力はどのみちサーバでも弾かなければならないので、<br>\nあくまでHTML+jsでのバリデーションはユーザへのフィードバック速度を向上させること。利便性・使い心地・体感速度を向上させるためのもの、として捉えれば十分だと思います。</p>\n<p>ちなみに<code>novalidate</code>属性を指定してもwillValidateプロパティの値は変わりませんが、ブラウザデフォルトのバリデーションは行われうなくなり、値が空でも送信できるようになります。</p>\n<h2 id=\"バリデーションエラーの例\" style=\"position:relative;\"><a href=\"#%E3%83%90%E3%83%AA%E3%83%87%E3%83%BC%E3%82%B7%E3%83%A7%E3%83%B3%E3%82%A8%E3%83%A9%E3%83%BC%E3%81%AE%E4%BE%8B\" aria-label=\"バリデーションエラーの例 permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>バリデーションエラーの例</h2>\n<h2 id=\"問題点\" style=\"position:relative;\"><a href=\"#%E5%95%8F%E9%A1%8C%E7%82%B9\" aria-label=\"問題点 permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>問題点</h2>\n<p>参考に上げた記事内の<a href=\"http://www.html5rocks.com/en/tutorials/forms/constraintvalidation/#toc-current-implementation-issues\" target=\"_blank\" rel=\"nofollow noopener noreferrer\">Current Implementation Issues and Limitations</a>がとても参考になりました。</p>\n<p>自分でもHTMLとjsを書いてみて不便に思ったことは以下のあたりでした。</p>\n<ul>\n<li>ブラウザデフォルトのバリデーションエラーの見た目をカスタマイズしたい</li>\n<li>エラー文言をカスタマイズできるようにしたい</li>\n<li>ラベル周りは吸収したい</li>\n<li>エラー一覧を取得する薄ーいライブラリ</li>\n<li>正規表現をエラー表示用の値に変換する</li>\n</ul>\n<p>ブラウザごとのバリデーションエラーの要素は<a href=\"http://developer.telerik.com/featured/building-html5-form-validation-bubble-replacements/\" target=\"_blank\" rel=\"nofollow noopener noreferrer\">webkitがベンダープレフィックスつきで提供</a>はしていますが、標準的な仕様はまだ出ていないようです。<br>\nということで自由度が低く現状ではゴリ押しもできないので断念。</p>\n<p>そして表示するエラー文言を生成する簡易的なテンプレート機構が欲しい。<br>\nカスタムエラーや、各ベンダーごとの独自実装はあるのですが、標準的なエラー文言のテンプレート的なものは存在しないようです。<br>\n簡単に設定できて、必要なら上書きできるようになっているとお手軽さがあるなーと思います。</p>\n<p><strong>バリデーションエラーをどう表示するか</strong> は使用しているCSSフレームワークやHTML構造に激しく依存するので、<br>\nロックイン率が高くなるようなライブラリは使いたくない。<br>\nなのでエラーの一覧を配列で返してもらい、それをHTMLとして描画処理は実装する、くらいが抽象化できる落とし所かなと思っています。</p>\n<p>エラーを表示するなら例えば「お名前」などlabelタグの値もメッセージに含めたいこともあると思います。<br>\nChromeならinputタグに対応するlabelタグが<code>labels</code>プロパティで取れるのですが、他のブラウザにはない模様。<br>\nということでinputに対応する項目名（label）を取得する処理をざっくり吸収したい。</p>\n<p>あと、正規表現にマッチしなかった場合に、その正規表現をユーザに見せて理解できるわけがないので、うまいこと日本語にしたい。<br>\n例えば<code>^[A-Za-z0-9-]+$</code>だとしたら<code>半角英数とハイフンのみ</code>といった具合です。<br>\n自動で名前のマッピングは面倒なので、明示的に表示用の別名が指定できればいいなーという感じです。</p>\n<h2 id=\"小ネタ\" style=\"position:relative;\"><a href=\"#%E5%B0%8F%E3%83%8D%E3%82%BF\" aria-label=\"小ネタ permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>小ネタ</h2>\n<p>上記の問題を解消するものをざっくり作ってみました。<br>\n具体的な実装は<a href=\"https://jsfiddle.net/leko/nwkLcry5/\" target=\"_blank\" rel=\"nofollow noopener noreferrer\">前述のデモ</a>にコードが有ります。</p>\n<p><code>var errors = Validator.getErrors(input);</code>のように使用できます。<br>\n引数にはバリデーションするinput要素を渡します。<br>\nエラーの一覧を配列で返します。</p>\n<h2 id=\"まとめ\" style=\"position:relative;\"><a href=\"#%E3%81%BE%E3%81%A8%E3%82%81\" aria-label=\"まとめ permalink\" class=\"autolink-header before\"><svg aria-hidden=\"true\" focusable=\"false\" height=\"16\" version=\"1.1\" viewBox=\"0 0 16 16\" width=\"16\"><path fill-rule=\"evenodd\" d=\"M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z\"></path></svg></a>まとめ</h2>\n<p>HTML5についてはメディア系以外は追っかけているつもりだったのですが、触ったことないAPIがまだまだあるなーと痛感しました。ValidityState、めっちゃ便利です。</p>\n<p>Angularなどのでかいフレームワークを使っているとこの辺一切触ることないですが、<br>\n素のReactなど薄いビュー層を触っているとついバリデーションをオレオレ実装してしまうことがあるので、こういった便利な標準仕様をもっと抑えていこうと思います。</p>","timeToRead":9,"frontmatter":{"title":"HTML5のConstraint validationでライブラリ要らずならくらくバリデーション","tags":["Advent Calendar","HTML5 Constraint validation","JavaScript"],"date":"December 07, 2015","featuredImage":null}}},"pageContext":{"slug":"/how-to-validate-form-value-with-constraint-validation/","previous":{"fields":{"slug":"/how-to-test-api-server-with-api-spec/"},"frontmatter":{"title":"APIドキュメントでAPIサーバをテストする","tags":["API","API Blueprint","Dredd","Go","Nodejs"]}},"next":{"fields":{"slug":"/refactored-ansible-settings-to-setup-golang/"},"frontmatter":{"title":"Goの環境を作るAnsibleの設定をリファクタした","tags":["Ansible","Go"]}}}},
    "staticQueryHashes": ["2585454260","2954598359"]}