22
13

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

モーダル作成で知っておきたいこと[HTML/CSS/jQuery]

Last updated at Posted at 2020-08-13

#はじめに
ひさびさにhtml/css/jqueryでモーダル画面を作成する機会があり、いろいろと学びがあったのでメモ。
基本的なモーダル画面を作成し、そこからいろんなケースに対応するために改良していきます。

#基本編
まず、ボタンをクリックするとチェックボックスが配列されているモーダルウィンドウが表示される、というものをつくっていきます。
modal9.gif

###html

index.html
<main>
  <div class="panel">
        <a class="c-btn js-open-modal" href="#">モーダル表示</a>
        <div class="modal-panel-back disable">
            <div class="modal-panel">
                <div class="modal-panel-ttl">さらに絞る必要がある場合は、下記からせんたくしてください。</div>
                <div class="check-item-area row">
                    <div class="col-sm-6 check-item">
                        <label><input type="checkbox" class="check-init"><span>AAA</span></label>
                    </div>
           (・・・省略・・・)
                    <div class="col-sm-6 check-item">
                        <label><input type="checkbox" class="check-init"><span>KKK</span></label>
                    </div>
                    </div>
                    <div class="btn-area">
                        <a class="c-btn js-close-modal" href="#">設定</a>
                    </div>
                </div>
            </div>
        </div>        
    </main>    
</body>
</html>

htmlはこんな感じ。
ポイントはモーダル表示したときの背景グレー領域(modal-panel-back)とモーダル領域(modal-panel)を作成すること。

###css

style.css
.panel {
  width: 680px;
  max-width: 90%;
  margin: 100px auto;
  padding: 20px;
  -webkit-box-shadow: 0 1px 2px rgba(0,0,0,0.2);
          box-shadow: 0 1px 2px rgba(0,0,0,0.2);
  background-color: white;
  text-align: center;
}
.modal-panel-back {
  position: fixed;
  width: 100%;
  height: 100%;  
  top: 0;
  right: 0;
  bottom: 0;
  left: 0;
  background-color: rgba(200,200,200,.5);
}
.modal-panel-back.disable {
  display: none;
}
.modal-panel {
  width: 600px;
  background-color: white;
  padding: 30px;
  margin: 100px auto;
}

cssはこんな感じ。
.modal-panel-back
グレー背景を画面いっぱいに広げるためにposition:fixedにする。position:absoluteだと親のどこかにposition:relativeが指定されているとそこが基準になり配置されるため、画面いっぱいに広がらない。かといって、ルートのタグ直下にグレー背景用のタグを用意するのはjquery操作がはいるのでめんどくさい。
.modal-panel-back.disable
モーダル領域は隠す
.modal-panel
モーダル領域はシンプルに、横幅widthを指定して、margin: oopx autoで中央に表示する。

###script

index.html
<script>
   $(function(){
       $('.js-open-modal').on('click', function(){
           $(this).next().removeClass('disable');
       });
       $('.js-close-modal').on('click', function(e){
           e.preventDefault();
           $(this).parents('.modal-panel-back').addClass('disable');
       })
   })
</script>

scriptはこんな感じ。
「モーダル表示」をクリックでdisableを削除しモーダル領域を表示。「設定」をクリックするとdisableを追加してモーダルを消す。
e.preventDefault()で明示的にaタグのイベントをキャンセルしています。これがないと「設定」をクリックしたときにhrefで指定したURLに飛ぼうとします。

基本編のモーダル表示はこんな感じです。ここからいろいろなケースを想定して改善していきます。

##ケース1:グレー背景クリックでモーダルを閉じたい
普通のモーダルウィンドウって、背景のグレー領域をクリックすると閉じるのが普通ですよね。なのでそのように実装していきましょう。WEB制作でもここらへんは当たり前に実装しておきたいです。

純粋に考えると、下記コードをscriptに追加しようと考えます。下記コードを追加して、背景部をクリックしてみてください。

index.html
<script>
   $(function(){
       $('.modal-panel-back').on('click', function(){
            $(this).addClass('disable');
        })
   })
</script>

modal2.gif

きちんと消えました。
次にモーダルを表示したあと、チェックボックスをクリックしてみてください。
modal3.gif

チェックボックスをクリックすると、モーダル領域が消えました。
ちょっと焦りますが、これはイベントバブリングという現象です。

#####イベントバブリング
子要素でイベントが発生したときに、親やさらに先の親にも同じイベントが発生するというのがイベントバブリングです。詳しい説明は割愛しますが、要は.modal-panelのモーダル領域でクリックすると、親要素である.modal-panel-hideのグレー背景もクリックされるということです。
これを回避するために、下記のコードを追加しましょう。これで.modal-panelのモーダル領域で発生したイベントは親要素.modal-panel-hideに伝搬しなくなります。

index.html
<script>
   $(function(){
       $('.modal-panel').on('click', function(e){
            e.stopPropagation();
        })
   })
</script>

これで背景部をクリックしてモーダルを閉じる動作が実装できました。

##ケース2:画面がスクロールする場合
ここまでの例では、「モーダル表示」ボタンが配置されている画面はスクロールしませんでした。ただ、大概のウェブサイトはスクロールするので、その状況にも対応していきましょう。
まず、cssファイルに下記を追加しましょう。これでスクロールするようになります。

style.css
html, body {
  height: 1500px; //←追加
}

modal10.gif

この状態で「モーダル表示」ボタンを表示してみます。
modal5.gif

モーダル表示しているのに元画面のスクロールが働いてしまっています。これではブサイクなのでスクロールできないようにしましょう。モーダル表示したときはbodyのスクロールが働かないようにコードを追加します。

index.html
<script>
   $(function(){
       $('.js-open-modal').on('click', function(){
            $(this).next().removeClass('disable');
            $('body').addClass('modal-open');  //追加
        });
        $('.js-close-modal').on('click', function(e){
            e.preventDefault();
            $(this).parents('.modal-panel-back').addClass('disable');
            $('body').removeClass('modal-open'); //追加
        })
        $('.modal-panel-back').on('click', function(){
            $(this).addClass('disable');
            $('body').removeClass('modal-open'); //追加
        })
   })
</script>
style.css
body.modal-open {
    overflow: hidden;
}

モーダル表示したときはbodyのスクロールが効かないようにしました。
これでモーダルを表示してみましょう。
modal6.gif

スクロールしないようにできました。
・・が、よくよく見るとモーダル表示したとき「モーダル表示」ボタンを配置している領域が微妙に右側に動いてしまっています。今回の例はかなりシンプルでわかりにくいですが、普通のサイトでこれが再現するとかなり気持ち悪いです。なので修正します。
下記コードを追加してください。

style.css
body.modal-open {
    overflow: hidden;
    padding-right: 15px; //追加
}

padding-rightを追加しました。微妙に右に動いてしまうのは、スクロールの領域がなくなった分、全体が右側に移動してしまうからです。なので動かないようにスクロールの幅分paddingを与えてやります。これで気持ち悪さはなくなります。
modal7.gif

##ケース3:モーダル画面がアクティブウィンドウからはみ出る場合
ここまでのモーダル画面は、アクティブウィンドウからはみ出していませんでしたが、はみ出してしまう場合を考えます。
チェックボタンの要素を増やし、モーダル画面がアクティブウィンドウからはみ出すようにしましょう。

これの状態でモーダル画面を表示するとスクロールしないのでしたのほうを確認することができません。
modal11.gif

モーダルがスクロールするように修正します。

.modal-panel-backのcssにoverflow-y:scrollを指定します。

style.css
.modal-panel-back {
  position: fixed;
  width: 100%;
  height: 100%;  
  top: 0;
  right: 0;
  bottom: 0;
  left: 0;
  background-color: rgba(200,200,200,.5);
  overflow-y: scroll; //追加
}

この状態でモーダルを表示するとスクロールできるようになります。
modal12.gif

#おわりに
モーダル表示といえど意外と考えないといけないことが多かったので、まとめてみました。
だれかのお役に立てればうれしいです。

22
13
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
22
13

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?