フォームの入力時にフロートする入力欄のラベル

CSSのみで実装したサンプル

DEMO (CSSのみ)

DEMO (JSのクリアボタン有り)

仕組み

  • 入力欄の枠要素上部に margin を持たせ、フォーカスした際と入力されている場合でプレースホルダーが表示されていなときに、ラベルを浮かせて小さく上部に transformtransition で表示してる
  • 入力欄に値がない場合は input:placeholder-shown
  • 入力欄に値がある場合は input:not(:placeholder-shown)
  • クリアボタンが必要な場合は、JavaScriptで入力内容をクリアする処理を別途実装

コード

HTML


<div class="input_wrapper">
  <input class="input" type="text" id="name" name="name" placeholder="Name">
  <label class="label" for="name">名前</label>
  <!-- ↓クリアボタンを設置する場合のみ -->
  <button class="clear_button" aria-label="Clear input">✕</button>
  <!-- ↑クリアボタンを設置する場合のみ -->
</div>
          

フロートする入力欄のラベルのCSS


/* 入力欄枠の指定 ======================= */
.input_wrapper {
  position: relative;
  margin-top: 2.5rem; /* ラベルをフローティング表示するための余白指定 */
  margin-bottom: 2rem;
  max-width: 400px;
}

/* テキスト入力欄の指定 ======================= */
.input_wrapper input {
  font-size: 1rem;
  width: 100%;
  padding: 0.5rem 0;
  color: #333;
  border: none;
  border-bottom: 1px solid #ddd;
  box-sizing: border-box;
  transition: border-color 250ms;
  background-color: transparent;
}
.input_wrapper input:focus {
  outline: none;
  border-bottom-color: #777;
}
.input_wrapper input::placeholder {
  color: transparent;
}
/* Safariのオートフィルボタンを隠す */
.input_wrapper input::-webkit-contacts-auto-fill-button {
  visibility: hidden;
  pointer-events: none;
  position: absolute;
}

/* 未入力時のラベル要素指定 ======================= */
.input_wrapper label {
  position: absolute;
  top: 0.2rem;
  left: 0;
  color: #43454e;
  pointer-events: none;
  transform-origin: left center;
  transition: transform 250ms;
}

/* 入力時にラベルをフロートして再配置 (※フォーカス時かプレースホルダーが表示されない場合) */
.input_wrapper input:focus + .label,
.input_wrapper input:not(:placeholder-shown) + .label {
  transform: translateY(-100%) scale(0.75);
  color: #000;
}
          

クリアボタンのCSS


/* クリアボタンの非表示指定 (※プレースホルダーが表示されている状態の場合) */
.input_wrapper input:placeholder-shown + .label + .clear_button {
  display: none;
}

/* クリアボタンの表示指定 ======================= */
.input_wrapper .clear_button {
  height: 30px;
  width: 30px;
  position: absolute;
  right: -4px; /* 入力欄の右端に設置 */
  top: 50%;
  transform: translateY(-50%);
  display: flex;
  align-items: center;
  justify-content: center;
  background: none;
  border: none;
  border-radius: 50%;
  color: #777;
  transition: color 250ms;
  cursor: pointer;
  appearance: none;
  -webkit-appearance: none;
}
.input_wrapper .clear_button:hover,
.input_wrapper .clear_button:focus {
  color: #333;
}

/* クリアボタンと入力テキストが重ならないようにするための余白指定 */
.input_wrapper:has(.clear_button) input {
  padding-right: 24px;
}
          

クリアボタンのJavaScript


[...document.querySelectorAll('.input_wrapper .clear_button')].forEach(clearButtonElem => {
  const $searchInput = clearButtonElem.parentElement.querySelector('.input');
  clearButtonElem.addEventListener('click', () => {
    $searchInput.value = '';
    $searchInput.focus();
  });
});