モーダルウインドウ(Modal Window)同一ページ複数実装の最適解?サンプルコード

モーダルウインドウ(Modal Window)を同じページで複数利用したいということは結構あると思うのですがいざやろうとするとこれがなかなか厄介。

モーダルの数分JSやCSSの記述を増やすなんて煩雑な運用は絶対したくない!

ということで、JSやCSSの記述は固定で、HTMLの記述側でモーダルをいくらでも増やせる実装を考えてみました!

動作確認

下記URLから動作確認いただけます。

https://buzz.tadworks.jp/modal.html

実装サンプル

①HTML部

HTML部の記述です。ボタンと表示部分は分けて好きなところへ設置可能です。

<!-- 開くボタン -->
<ul>
<li><a class="modal-open js-modal-open-1" onclick="modalOpen(1);";>モーダルを開く①</a></li>
<li><a class="modal-open js-modal-open-2" onclick="modalOpen(2);";>モーダルを開く②</a></li>
<li><a class="modal-open js-modal-open-3" onclick="modalOpen(3);";>モーダルを開く③</a></li>
</ul>

<!-- モーダル表示部 -->
<div class="modal js-modal-1" onclick="modalOut(1);">
  <div class="modal-container">
    <div class="modal-close js-modal-close-1" onclick="modalClose(1);">×</div>
    <div class="modal-content">
      <p>コンテンツ①</p>
    </div>
  </div>
</div>

<div class="modal js-modal-2" onclick="modalOut(2);">
  <div class="modal-container">
    <div class="modal-close js-modal-close-2" onclick="modalClose(2);">×</div>
    <div class="modal-content">
      <p>コンテンツ②</p>
    </div>
  </div>
</div>

<div class="modal js-modal-3" onclick="modalOut(3);">
  <div class="modal-container">
    <div class="modal-close js-modal-close-3" onclick="modalClose(3);">×</div>
    <div class="modal-content">
      <p>コンテンツ③</p>
    </div>
  </div>
</div>

表示部にはボタンに対応する番号を入れるようにしてください。

<div class="modal js-modal-番号" onclick="modalOut(番号);">
  <div class="modal-container">
    <div class="modal-close js-modal-close-番号" onclick="modalClose(番号);">×</div>
    <div class="modal-content">
      <p>コンテンツ</p>
    </div>
  </div>
</div>

WordPressではビジュアルエディタでonclickが消える場合がありますので、その場合はfunctions.phpに下記記述を追加して自動削除を回避ください。

// onClick を自動削除させない設定
function custom_editor_settings( $initArray ){
$initArray['body_id'] = 'primary';
$initArray['body_class'] = 'post';
$initArray['extended_valid_elements'] = '*[*]';
$initArray['valid_children'] = '+body[style],+div[span],+span[span]+button[onClick],+input[onClick],+button[data-*]';
$initArray['verify_html'] = false;
return $initArray;
}
add_filter( 'tiny_mce_before_init', 'custom_editor_settings' );

②JavaScript部

JavaScript部分です。JQueryを使用しているので読み込み設定をするようにしてください。

<script>
//「開くボタン」をクリックしてモーダルを開く
function modalOpen(num) {
  document.querySelector('.js-modal-' + num).classList.add('is-active');
  document.querySelector('.js-modal-open-' + num).classList.add('is-active');
  document.querySelector('.js-modal-close-' + num).classList.add('is-active');
  $('body').addClass('no_scroll');
}
//「閉じるボタン」をクリックしてモーダルを閉じる
function modalClose(num) {
  document.querySelector('.js-modal-' + num).classList.remove('is-active');
  document.querySelector('.js-modal-open-' + num).classList.remove('is-active');
  document.querySelector('.js-modal-close-' + num).classList.remove('is-active');
  $('body').removeClass('no_scroll');
}
//「モーダルの外側」をクリックしてモーダルを閉じる
function modalOut(num) {
  document.querySelector('.js-modal-' + num).classList.remove('is-active');
  $('body').removeClass('no_scroll');
}
</script>

PCとAndroidではモーダル表示時にページのスクロールをしないように制御しています。

CSS部

CSS部分です。

<style>
/* ページスクロール禁止 */
body.no_scroll{
  overflow: hidden;
}

/* モーダルを開くボタン */
.modal-open{
  font-size: 16px;
  font-weight: bold;
  width: 300px;
  height: 60px;
  color: #fff;
  background: #000;
  border: none;
  cursor: pointer;
  translate: -50% -50%;
}

/* モーダルと背景の指定 */
.modal{
  position: fixed;
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  text-align: center;
  background: rgba(0,0,0,50%);
  padding: 40px 20px;
  overflow: auto;
  opacity: 0;
  visibility: hidden;
  transition: .3s;
  box-sizing: border-box;
}

/* モーダルの擬似要素の指定 */
.modal:before{
  content: "";
  display: inline-block;
  vertical-align: middle;
  height: 100%;
  margin-left: -0.2em;
}

/* クラスが追加された時の指定 */
.modal.is-active{
  opacity: 1;
  visibility: visible;
}

/* モーダル内側の指定 */
.modal-container{
  position: relative;
  display: inline-block;
  vertical-align: middle;
  max-width: 600px;
  width: 90%;
}

/* モーダルを閉じるボタンの指定 */
.modal-close{
  position: absolute;
  display: flex;
  align-items: center;
  justify-content: center;
  top: -20px;
  right: -20px;
  width: 40px;
  height: 40px;
  color: #fff;
  background: #000;
  border-radius: 50%;
  cursor: pointer;
}

/* モーダルのコンテンツ部分の指定 */
.modal-content{
  background: #fff;
  text-align: left;
  line-height: 1.8;
  padding: 20px;
}

/* モーダルのコンテンツ部分のテキストの指定 */
.modal-content p{
  margin: 1em 0;
}
</style>