Pen Settings

HTML

CSS

CSS Base

Vendor Prefixing

Add External Stylesheets/Pens

Any URL's added here will be added as <link>s in order, and before the CSS in the editor. If you link to another Pen, it will include the CSS from that Pen. If the preprocessor matches, it will attempt to combine them before processing.

+ add another resource

JavaScript

Babel includes JSX processing.

Add External Scripts/Pens

Any URL's added here will be added as <script>s in order, and run before the JavaScript in the editor. You can use the URL of any other Pen and it will include the JavaScript from that Pen.

+ add another resource

Packages

Add Packages

Search for and use JavaScript packages from npm here. By selecting a package, an import statement will be added to the top of the JavaScript editor for this package.

Behavior

Save Automatically?

If active, Pens will autosave every 30 seconds after being saved once.

Auto-Updating Preview

If enabled, the preview panel updates automatically as you code. If disabled, use the "Run" button to update.

Format on Save

If enabled, your code will be formatted when you actively save your Pen. Note: your code becomes un-folded during formatting.

Editor Settings

Code Indentation

Want to change your Syntax Highlighting theme, Fonts and more?

Visit your global Editor Settings.

HTML

              
                <div class="wrap">
  <form class="c-form" id="js-form">
    <div class="c-form__error-message u-mb40" data-js-form-error-message style="display: none;">入力内容に不備があります<br>お手数ですが再度入力内容をご確認ください</div>
    <div class="c-form-inputs">
      <div class="c-form-inputs__group">
        <label class="c-form-inputs__label"><span class="__required">必須</span>氏名</label>
        <div class="c-form-inputs__input">
          <input type="text" name="name" id="name" required placeholder="例)氏名 太郎" autocomplete="off">
          <div class="c-form-inputs__error-message" data-js-error-message="name"></div>
        </div>
      </div>
      <div class="c-form-inputs__group">
        <label class="c-form-inputs__label"><span class="__required">必須</span>メールアドレス</label>
        <div class="c-form-inputs__input">
          <input type="email" name="email" id="email" required pattern="^[a-zA-Z0-9-_.]+@[a-zA-Z0-9-_.]+$" title="メールアドレスの形式で入力してください" placeholder="例)example@ex.com" autocomplete="off">
          <div class="c-form-inputs__error-message" data-js-error-message="email"></div>
        </div>
      </div>
      <div class="c-form-inputs__group">
        <label class="c-form-inputs__label"><span class="__required">必須</span>電話番号</label>
        <div class="c-form-inputs__input">
          <input type="tel" name="tel" id="tel" required pattern="^0[0-9]{9,10}$" title="電話番号の形式で入力してください" placeholder="例)09012345678" autocomplete="off">
          <div class="c-form-inputs__error-message" data-js-error-message="tel"></div>
          <div class="c-form-inputs__caption"><small>ハイフン無しで入力してください</small></div>
        </div>
      </div>
      <div class="c-form-inputs__group">
        <label class="c-form-inputs__label"><span class="__required">必須</span>年齢</label>
        <div class="c-form-inputs__input">
          <select name="selectbox" id="selectbox" required data-required-error="いずれかを選択してください">
            <option value="">選択してください</option>
            <option value="1">10代</option>
            <option value="2">20代</option>
            <option value="3">30代</option>
            <option value="4">40代</option>
            <option value="5">50代</option>
            <option value="6">60歳以上</option>
          </select>
          <div class="c-form-inputs__error-message" data-js-error-message="selectbox"></div>
        </div>
      </div>
      <div class="c-form-inputs__group--checkbox">
        <label class="c-form-inputs__label"><span class="__required">必須</span>雇用形態</label>
        <div class="c-form-inputs__input">
          <div class="l-flex">
            <div>
              <input type="checkbox" name="job" id="job-1" required data-required-error="1つ以上選択してください">
              <label for="job-1">会社員</label>
            </div>
            <div>
              <input type="checkbox" name="job" id="job-2" required data-required-error="1つ以上選択してください">
              <label for="job-2">公務員</label>
            </div>
            <div>
              <input type="checkbox" name="job" id="job-3" required data-required-error="1つ以上選択してください">
              <label for="job-3">パート・アルバイト</label>
            </div>
            <div>
              <input type="checkbox" name="job" id="job-4" required data-required-error="1つ以上選択してください">
              <label for="job-4">自営業(経営者)</label>
            </div>
            <div>
              <input type="checkbox" name="job" id="job-5" required data-required-error="1つ以上選択してください">
              <label for="job-5">自営業(その他)</label>
            </div>
            <div>
              <input type="checkbox" name="job" id="job-6" required data-required-error="1つ以上選択してください">
              <label for="job-6">その他</label>
            </div>
          </div>
          <div class="c-form-inputs__error-message" data-js-error-message="job"></div>
        </div>
      </div>
      <div class="c-form-inputs__group--radio">
        <label class="c-form-inputs__label"><span class="__required">必須</span>住居</label>
        <div class="c-form-inputs__input">
          <div class="l-flex">
            <div>
              <input type="radio" name="house" id="house-1" required data-required-error="1つ以上選択してください">
              <label for="house-1">持ち家</label>
            </div>
            <div>
              <input type="radio" name="house" id="house-2" required data-required-error="1つ以上選択してください">
              <label for="house-2">賃貸</label>
            </div>
            <div>
              <input type="radio" name="house" id="house-3" required data-required-error="1つ以上選択してください">
              <label for="house-3">その他</label>
            </div>
          </div>
          <div class="c-form-inputs__error-message" data-js-error-message="house"></div>
        </div>
      </div>
      <div class="c-form-inputs__group">
        <label class="c-form-inputs__label"><span class="__any">任意</span>任意項目</label>
        <div class="c-form-inputs__input">
          <textarea name="any"></textarea>
        </div>
      </div>
      <div class="c-form-inputs__group">
        <div class="u-w100p">
          <div class="l-flex __col-center">
            <input type="checkbox" name="privacy-policy" id="privacy-policy" required data-required-error="プライバシーポリシーに同意される場合、チェックを入れてください">
            <label for="privacy-policy"><a class="u-link" href="#" target="_blank" rel="noreferrer noopener">プライバシーポリシー</a>に同意する</label>
          </div>
          <div class="l-flex __col-center">
            <div class="c-form-inputs__error-message" data-js-error-message="privacy-policy"></div>
          </div>
        </div>
      </div>
    </div>
    <div class="c-form-submit">
      <button class="c-btn" type="submit">送信する</button>
    </div>
    <div class="c-form__error-message u-mt40" data-js-form-error-message style="display: none;">入力内容に不備があります<br>お手数ですが再度入力内容をご確認ください</div>
  </form>
</div>
              
            
!

CSS

              
                // Variables
// ##############################
$breackpoint_pc: 1080px
$breackpoint_tb: 750px
$breackpoint_sp: 375px

$c--primary: #014c86
$c--accent: #f5df4d

$c--black: #333
$c--white: #fff

$c--gray-s: #f5f5f5
$c--gray-m: #d5d5d5
$c--gray-l: #939597

$c--text: $c--black
$c--link: $c--primary

$c--safe: #1fda50
$c--warning: #e63444



// Mixin
// ##############################
=max-screen($breakPoint)
  @media screen and (max-width: $breakPoint)
    @content
    
    

// Reset
// ##############################
*
  box-sizing: border-box

/* webkit specific styles */

input[type="color"]::-webkit-color-swatch
  border: none

input[type="color"]::-webkit-color-swatch-wrapper
  padding: 0

// html5doctor.com Reset Stylesheet
// v1.6.1
// Last Updated: 2010-09-17
// Author: Richard Clark - http://richclarkdesign.com
// Twitter: @rich_clark

html,
body,
div,
span,
object,
iframe,
h1,
h2,
h3,
h4,
h5,
h6,
p,
blockquote,
pre,
abbr,
address,
cite,
code,
del,
dfn,
em,
img,
ins,
kbd,
q,
samp,
small,
strong,
sub,
sup,
var,
b,
i,
dl,
dt,
dd,
ol,
ul,
li,
fieldset,
form,
label,
legend,
table,
caption,
tbody,
tfoot,
thead,
tr,
th,
td,
article,
aside,
canvas,
details,
figcaption,
figure,
footer,
header,
hgroup,
menu,
nav,
section,
summary,
time,
mark,
audio,
video
  margin: 0
  padding: 0
  border: 0
  outline: 0
  font-size: 100%
  vertical-align: baseline
  background: transparent
  font-weight: inherit

body
  line-height: 1

article,
aside,
details,
figcaption,
figure,
footer,
header,
hgroup,
menu,
nav,
section
  display: block

nav ul
  list-style: none

blockquote,
q
  quotes: none

blockquote:before,
blockquote:after,
q:before,
q:after
  content: ''
  content: none

a
  margin: 0
  padding: 0
  font-size: 100%
  vertical-align: baseline
  background: transparent

/* change colours to suit your needs */

mark
  background-color: #ff9
  color: #000
  font-style: italic
  font-weight: bold

del
  text-decoration: line-through

abbr[title],
dfn[title]
  border-bottom: 1px dotted
  cursor: help

table
  border-collapse: collapse
  border-spacing: 0

/* change border colour to suit your needs */

hr
  display: block
  height: 1px
  border: 0
  border-top: 1px solid #cccccc
  margin: 1em 0
  padding: 0

input,
select
  vertical-align: middle

input:focus
  outline: none

ul,
ol
  list-style-type: none

// Add

body
  -ms-text-size-adjust: 100%
  -webkit-text-size-adjust: 100%

main
  display: block

li
  list-style: none

h1,
h2,
h3,
h4,
h5,
h6
  font-size: 16px

a
  color: inherit
  width: 100%
  text-decoration: none
  &:hover
    color: inherit

img
  vertical-align: middle
  max-width: 100%
  height: auto

address
  font-style: normal

sup
  vertical-align: super
  font-size: smaller

input,
button,
textarea,
select
  display: block
  color: $c--text
  font-size: 16px
  background: none
  border: none
  border-radius: 0
  outline: none
  width: 100%
  margin: 0
  padding: 0
  -webkit-appearance: none
  -moz-appearance: none
  appearance: none
  -moz-appearance: textfield

  // オートフィルで入力した際の文字色と入力欄の背景色
  &:-webkit-autofill
    color: $c--text
    -webkit-box-shadow: 0 0 0px 1000px $c--white inset

  // プレースホルダーの文字色
  &::placeholder
    color: #b5b5b6

  // input[type="number"]の増減ボタン非表示
  &::-webkit-outer-spin-button,
  &::-webkit-inner-spin-button
    -webkit-appearance: none
    margin: 0

input,
textarea,
select
  line-height: 1.5
  background-color: $c--white
  border: solid 1px $c--gray-m
  border-radius: 3px
  padding: 10px 12px
  &[readonly]
    color: rgba($c--black, .7)
    border: none
    border-radius: 0

textarea
  min-height: 100px

select
  // IEの矢印を非表示
  &::-ms-expand
    display: none

button
  cursor: pointer

%__input_radio_check_hide
  // `display: none`で消すのはアクセシビリティ的によろしくない
  // @see https://github.com/mike-engel/a11y-css-reset
  position: absolute
  white-space: nowrap
  width: 1px
  height: 1px
  border: 0
  margin: -1px
  padding: 0
  overflow: hidden
  clip: rect(0 0 0 0)
  clip-path: inset(50%)

input[type="radio"]
  $parent_radio: &
  @extend %__input_radio_check_hide
  + label
    display: block
    font-size: 14px
    margin-right: 20px
    margin-bottom: 4px
    padding-left: 28px
    position: relative
    cursor: pointer
    &::before
      content: ""
      display: block
      width: 20px
      height: 20px
      background-color: $c--gray-s
      border: solid 1px $c--gray-m
      border-radius: 50%
      position: absolute
      top: 0
      left: 0
      @at-root #{$parent_radio}:checked + label::before
        border: solid 1px $c--primary
    &::after
      content: ""
      display: none
      // codepenだと何故か中心からズレるから小数点に...
      width: 13.5px
      height: 13.5px
      background-color: $c--primary
      border-radius: 50%
      position: absolute
      top: 4px
      left: 4px
      @at-root #{$parent_radio}:checked + label::after
        display: block

input[type="checkbox"]
  $parent_checkbox: &
  @extend %__input_radio_check_hide
  + label
    display: block
    font-size: 14px
    margin-right: 20px
    margin-bottom: 4px
    padding-left: 28px
    position: relative
    cursor: pointer
    &::before
      content: ""
      display: block
      width: 20px
      height: 20px
      background-color: $c--gray-s
      border: solid 1px $c--gray-m
      border-radius: 2px
      position: absolute
      top: 0
      left: 0
      @at-root #{$parent_checkbox}:checked + label::before
        background-color: $c--primary
        border-color: $c--primary
    &::after
      content: ""
      display: none
      width: 6px
      height: 10px
      border: solid $c--white
      border-width: 0 2px 2px 0
      position: absolute
      top: 3px
      left: 7px
      transform: rotate(45deg)
      @at-root #{$parent_checkbox}:checked + label::after
        display: block

select
  cursor: pointer
  padding-right: 32px
  background-repeat: no-repeat
  background-position: right 8px center
  background-size: 10px 10px
  background-image: url()

		

// Base
// ##############################
body,
input,
button,
textarea,
select
  color: $c--text
  font-size: 16px
  font-family: "游ゴシック Medium", "游ゴシック体", "Yu Gothic Medium", YuGothic, "ヒラギノ角ゴ ProN", "Hiragino Kaku Gothic ProN", "メイリオ", Meiryo, "MS Pゴシック", "MS PGothic", sans-serif

.wrap
  line-height: 1.7
  padding: 40px 20px 80px
  
  

// Layout
// ##############################
.l-flex
  display: flex
  flex-wrap: wrap
  &.__col-center
    justify-content: center
  &.__col-between
    justify-content: space-between
  &.__col-start
    justify-content: flex-start
  &.__col-end
    justify-content: flex-end

	

// Component
// ##############################
.c-form
  max-width: 600px
  margin: auto
  &__error-message
    color: $c--warning
    font-size: 13px
    text-align: center
    background-color: rgba($c--warning, .1)
    border: solid 1px $c--warning
    border-radius: 2px
    padding: 12px 12px
  &-inputs
    &__group
      $parent_inputs_group: &
      display: flex
      align-items: flex-start
      &:not(:first-of-type)
        margin-top: 40px
      +max-screen($breackpoint_tb)
        display: block
      &--checkbox
        @extend #{$parent_inputs_group}
      &--radio
        @extend #{$parent_inputs_group}
    &__label
      flex: 0 0 auto
      display: flex
      align-items: center
      font-size: 15px
      width: 100%
      max-width: 200px
      +max-screen($breackpoint_tb)
        max-width: 100%
      .__required,
      .__any
        display: inline-block
        font-size: 12px
        border-radius: 2px
        margin-right: 8px
        padding: 2px 6px
      .__required
        color: $c--white
        background-color: $c--warning
      .__any
        background-color: $c--gray-s
    &__input
      width: 100%
      padding-left: 40px
      +max-screen($breackpoint_tb)
        margin-top: 10px
        padding-left: 0
      input,
      textarea,
      select
        &.__error
          background-color: rgba($c--warning, .1)
          border: solid 1px $c--warning
        &.__secure
          background-color: rgba($c--safe, .05)
          border: solid 1px $c--safe
    &__error-message
      color: $c--warning
      font-size: 12px
      margin-top: 4px
    &__caption
      color: $c--gray-l
      font-size: 12px
      margin-top: 4px
  &-submit
    max-width: 200px
    margin: 64px auto 0
    
.c-btn
  color: $c--white
  background-color: $c--primary
  border-radius: 2px
  padding: 16px 20px
  transition: opacity .2s
  &:hover
    opacity: 0.7
    transition: opacity .2s



// Utility
// ##############################
.u
  &-link
    color: $c--link
    text-decoration: underline
    &:hover
      text-decoration: none
  &-w100p
    width: 100% !important

$marginType: (top: ("mt", "margin-top"), left: ("ml", "margin-left"), right: ("mr", "margin-right"), bottom: ("mb", "margin-bottom"))
@each $key, $value in $marginType
  @for $i from 0 through 20
    .u-#{nth($value, 1)}#{$i * 4}
      #{nth($value, 2)}: #{$i * 4}px !important
              
            
!

JS

              
                /**
 * フォームのバリデーションを実行するクラス
 * `elmForm`, `elmTargetInputs`, `elmSubmitBtn`は必須
 *
 * 場合によっては、以下ライブラリでバリデーションするでも良いかも
 * 公式: https://imbrn.github.io/v8n/#what-s-v8n
 * 日本語解説記事: https://co.bsnws.net/article/182
 */
class FormValidator {
  /**
   * @property {Object} elmForm 【必須】form要素
   * @property {Array} elmTargetInputs 【必須】バリデーション対象となるinput要素の配列
   * @property {Object} elmSubmitBtn 【必須】送信ボタンの要素
   * @property {Array} elmFormErrorMessages フォーム全体のエラーメッセージ要素の配列
   * @property {String} classErrorInput エラーの場合、input要素に付与されるclass
   * @property {String} classSecureInput エラーが無い場合、input要素に付与されるclass
   * @property {String} attrElmErrorMessage エラーメッセージ要素をinput要素と紐付けるための属性名
   * @property {String} attrRequiredErrorMessage `required`のエラーメッセージの文言を変更するための属性名
   * @property {String} defaultErrorMessage デフォルトのエラーメッセージ
   * @property {Object} inputStatuses バリデーション対象となる全てのinput要素のオブジェクト. エラーの状態などのプロパティを持つ
   * @property {Boolean} isFirstSubmit 1回でも送信ボタンをクリックしたら`true`
   */
  constructor(_parm) {
    this.elmForm = document.querySelector(_parm.form) || false;
    this.elmTargetInputs = [
      ...this.elmForm.querySelectorAll(_parm.targetInputs)
    ];
    this.elmSubmitBtn = this.elmForm.querySelector(_parm.submitBtn);
    this.elmFormErrorMessages = [
      ...this.elmForm.querySelectorAll("[data-js-form-error-message]")
    ];
    this.classErrorInput = _parm.classErrorInput || "__error";
    this.classSecureInput = _parm.classSecureInput || "__secure";
    this.attrElmErrorMessage =
      _parm.attrElmErrorMessage || "data-js-error-message";
    this.attrRequiredErrorMessage =
      _parm.attrRequiredErrorMessage || "data-required-error";
    this.defaultErrorMessage =
      _parm.defaultErrorMessage || "必須項目を入力してください";
    const createInputStatuses = this.elmTargetInputs.map((_item) => {
      let result = [];
      result["name"] = _item.getAttribute("name");
      result["isError"] = true;
      return result;
    });
    this.inputStatuses = createInputStatuses.filter(
      (_item, _index, _self) =>
        _self.findIndex((_ev) => _ev.name === _item.name) === _index
    );
    this.isFirstSubmit = false;
  }

  /**
   * input要素が一つでもバリデーションエラーなら`true`を返す
   * @return {Boolean}
   */
  getIsFormError() {
    return this.inputStatuses.every((_item) => _item["isError"] === false)
      ? false
      : true;
  }

  /**
   * バリデーション対象となるinput要素の種類を返す
   * @param {Object} _elmInput バリデーション対象のinput要素
   * @return {String} 'checkOrRadio', 'select', 'input'
   */
  getInputType(_elmInput) {
    if (_elmInput.tagName === "SELECT") return "select";
    if (_elmInput.getAttribute("type").match(/checkbox|radio/))
      return "checkOrRadio";
    return "input";
  }

  /**
   * input要素に付与されている`required`と`pattern`でバリデーションチェックを行う
   * エラーなら`true`を返す
   * @param {Object} _elmInput バリデーション対象のinput要素
   * @returns {Boolean}
   */
  errorCheck(_elmInput) {
    const patternValidate = _elmInput.getAttribute("pattern") || false;
    const inputType = this.getInputType(_elmInput);
    // セレクトボックスの場合
    if (inputType === "select")
      return !_elmInput.validity.patternMismatch && _elmInput.value.length
        ? false
        : true;
    // チェックボックスかラジオボタンの場合
    if (inputType === "checkOrRadio") return _elmInput.checked ? false : true;
    // input(パターン有り)の場合
    if (patternValidate)
      return !_elmInput.validity.patternMismatch && _elmInput.value.length
        ? false
        : true;
    // input(パターン無し)の場合
    return _elmInput.validity.valueMissing;
  }

  /**
   * input要素に紐づくエラーメッセージ要素のテキストを描画
   * @param {Object} _elmInput バリデーション対象のinput要素
   */
  createInputErrorMessage(_elmInput) {
    const value = _elmInput.value;
    const name = _elmInput.getAttribute("name");
    const elmErrorMessage = this.elmForm.querySelector(
      `[${this.attrElmErrorMessage}="${name}"]`
    );
    const patternErrorMessage = _elmInput.getAttribute("title") || false;
    const requiredErrorMessage =
      _elmInput.getAttribute(this.attrRequiredErrorMessage) ||
      this.defaultErrorMessage;
    let errorMessage = "";
    if (patternErrorMessage) {
      errorMessage = value.length ? patternErrorMessage : requiredErrorMessage;
    } else {
      errorMessage = requiredErrorMessage;
    }
    elmErrorMessage.textContent = errorMessage;
    return;
  }

  /**
   * フォーム全体のエラーメッセージの表示を切り替える
   * @param {Boolean} _show `true`: 表示, `false`: 非表示
   */
  toggleFormErrorMessage(_show) {
    this.elmFormErrorMessages.forEach((_elmFormErrorMessage) => {
      _show
        ? (_elmFormErrorMessage.style.display = "block")
        : (_elmFormErrorMessage.style.display = "none");
    });
  }

  /**
   * input要素のエラーをリセット
   * @param {Object} _elmInput バリデーション対象のinput要素
   */
  errorReset(_elmInput) {
    const name = _elmInput.getAttribute("name");
    const elmErrorMessage = this.elmForm.querySelector(
      `[${this.attrElmErrorMessage}="${name}"]`
    );
    _elmInput.classList.remove(this.classErrorInput);
    _elmInput.classList.remove(this.classSecureInput);
    elmErrorMessage.textContent = "";
    return;
  }

  /**
   * input要素に対してバリデーションチェックやエラーメッセージの描画など、バリデーションに関する関数を全て実行する
   * @param {Object} _elmInput バリデーション対象のinput要素
   */
  validate(_elmInput) {
    // エラーリセット
    this.errorReset(_elmInput);

    // 必要な変数定義
    const isError = this.errorCheck(_elmInput);
    const targetInputStatus = this.inputStatuses.find((_item) => {
      return _item.name === _elmInput.getAttribute("name");
    });
    const changeInputStatusArray = (_isErrorValue) => {
      targetInputStatus["isError"] = _isErrorValue;
    };

    // チェックボックスかラジオボタンの場合、いずれかがチェックされていればエラーにしない
    if (this.getInputType(_elmInput) === "checkOrRadio") {
      const ElmsCheckOrRadio = [
        ...document.querySelectorAll(
          `input[name="${targetInputStatus["name"]}"]`
        )
      ];
      const getIsAnyChecked = () => {
        return ElmsCheckOrRadio.some((_elm) => {
          return _elm.checked;
        });
      };
      const toggleAllCheckOrRadioRequired = (_value) => {
        ElmsCheckOrRadio.forEach((_elm) => {
          _elm.required = _value;
        });
      };
      if (getIsAnyChecked()) {
        toggleAllCheckOrRadioRequired(false);
        changeInputStatusArray(false);
      } else {
        toggleAllCheckOrRadioRequired(true);
        changeInputStatusArray(true);
        this.createInputErrorMessage(_elmInput);
      }
      return;
    }

    // バリデーションチェックやエラーメッセージの描画などを実行
    if (isError) {
      _elmInput.classList.add(this.classErrorInput);
      changeInputStatusArray(true);
      this.createInputErrorMessage(_elmInput);
    } else {
      changeInputStatusArray(false);
      _elmInput.classList.add(this.classSecureInput);
    }
    return;
  }

  addEvent() {
    /**
     * Input
     */
    this.elmTargetInputs.forEach((_elmTargetInput) => {
      // 入力時
      _elmTargetInput.addEventListener("change", (_ev) => {
        this.validate(_elmTargetInput);
        this.isFirstSubmit
          ? this.getIsFormError()
            ? this.toggleFormErrorMessage(true)
            : this.toggleFormErrorMessage(false)
          : false;
      });
      // エラー時
      _elmTargetInput.addEventListener("invalid", (_ev) => {
        this.validate(_elmTargetInput);
      });
    });

    /**
     * SubmitBtn
     */
    this.elmSubmitBtn.addEventListener("click", (_ev) => {
      this.isFirstSubmit = true;
      this.getIsFormError()
        ? this.toggleFormErrorMessage(true)
        : this.toggleFormErrorMessage(false);
    });

    /**
     * Form
     */
    this.elmForm.addEventListener("submit", (_ev) => {
      _ev.preventDefault();
      if (!this.getIsFormError()) {
        alert("Validate OK!");
        // this.elmForm.submit();
      }
    });
  }

  init() {
    if (!this.elmForm) return;
    this.addEvent();
  }
}

window.addEventListener("DOMContentLoaded", () => {
  const formValidator = new FormValidator({
    form: "#js-form",
    targetInputs: "input[required], select[required]",
    submitBtn: 'button[type="submit"]'
  });
  formValidator.init();
});

              
            
!
999px

Console