MENU

HTML5/CSS3とjQueryでマテリアルデザインなタブを作る

だいぶ前にGoogleがマテリアルデザインを発表しましたよね。フラットデザインに階層を追加した真新しいデザインです。ですが、僕はまだ案件でマテリアルデザインを一度も使ったことがありません。なんとなくですが使いづらいですよね。アプリとかなら利用しやすそうですが。そこで、今回はマテリアルデザインを使ったタブを作ったので紹介します。

タブ機能を作る

デモ

  • Tab1
  • Tab2
  • Tab3
  • Tab4
  • Tab1 Contents
  • Tab2 Contents
  • Tab3 Contents
  • Tab4 Contents

必要最低限のタブ機能です。タブを押せば切り替わる感じです。

HTML


  • Tab1
  • Tab2
  • Tab3
  • Tab4
  • Tab1 Contents
  • Tab2 Contents
  • Tab3 Contents
  • Tab4 Contents

.tab-contents の最初の li 要素に active クラスをつけておきます。

CSS


/* RESET */
ul, li {
  display: block;
  margin: 0;
  padding: 0;
  list-style-type: none;
}

.tab-title {
  width: 100%;
}
.tab-title:before, .tab-title:after {
  display: table;
  content: '';
}
.tab-title:after {
  clear: both;
}
.tab-title li {
  float: left;
  width: 25%;
  padding: .8em 0;
  color: #fff;
  text-align: center;
  background-color: #26C6DA;
  cursor: pointer;
}
.tab-contents li {
  display: none;
  padding: 1.4em;
  background-color: #fff;
}
.tab-contents li.active {
  display: block;
}

SCSS


/* RESET */
ul, li {
  display: block;
  margin: 0;
  padding: 0;
  list-style-type: none;
}

.tab-title {
  width: 100%;
  &:before, &:after {
    display: table;
    content: '';
  }
  &:after {
    clear: both;
  }
  li {
    float: left;
    width: 25%;
    padding: .8em 0;
    color: #fff;
    text-align: center;
    background-color: #26C6DA;
    cursor: pointer;
  }
}
.tab-contents {
  li {
    display: none;
    padding: 1.4em;
    background-color: #fff;
    &.active {
      display: block;
    }
  }
}

jQuery


$(function(){

$('.tab-title li').on('click', function(){
  /* タブ */
  var th = $(this).index();  // 何番目のタブがクリックされたか
  var $tab_contents = $('.tab-contents li');
  $tab_contents.removeClass('active');
  $tab_contents.eq(th).addClass('active');
});

});

スライダーを追加する

デモ

  • Tab1
  • Tab2
  • Tab3
  • Tab4
  • Tab1 Contents
  • Tab2 Contents
  • Tab3 Contents
  • Tab4 Contents

HTML


  • Tab1
  • Tab2
  • Tab3
  • Tab4
  • Tab1 Contents
  • Tab2 Contents
  • Tab3 Contents
  • Tab4 Contents

スライダー用に .tab-title-bar を追加しています。擬似要素でもいいのですが、jQueryから直接擬似要素にアクセスできないため、やめました。

CSS


/* RESET */
ul, li {
  display: block;
  margin: 0;
  padding: 0;
  list-style-type: none;
}

.tab-title {
  position: relative;
  width: 100%;
}
.tab-title:before, .tab-title:after {
  display: table;
  content: '';
}
.tab-title:after {
  clear: both;
}
.tab-title li {
  float: left;
  width: 25%;
  padding: .8em 0;
  color: #fff;
  text-align: center;
  background-color: #26C6DA;
  cursor: pointer;
}
.tab-title-bar {
  position: absolute;
  left: 0;
  bottom: 0;
  width: 25%;
  height: 2px;
  background-color: #F06292;
  -webkit-transition: .45s ease-in-out;
  -moz-transition: .45s ease-in-out;
  -o-transition: .45s ease-in-out;
  transition: .45s ease-in-out;
}
.tab-contents li {
  display: none;
  padding: 1.4em;
  background-color: #fff;
}
.tab-contents li.active {
  display: block;
}

SCSS


/* RESET */
ul, li {
  display: block;
  margin: 0;
  padding: 0;
  list-style-type: none;
}

.tab-title {
  position: relative;
  width: 100%;
  &:before, &:after {
    display: table;
    content: '';
  }
  &:after {
    clear: both;
  }
  li {
    float: left;
    width: 25%;
    padding: .8em 0;
    color: #fff;
    text-align: center;
    background-color: #26C6DA;
    cursor: pointer;
  }
}
.tab-title-bar {
  position: absolute;
  left: 0;
  bottom: 0;
  width: 25%;
  height: 2px;
  background-color: #F06292;
  -webkit-transition: .45s ease-in-out;
  -moz-transition: .45s ease-in-out;
  -o-transition: .45s ease-in-out;
  transition: .45s ease-in-out;
}
.tab-contents {
  li {
    display: none;
    padding: 1.4em;
    background-color: #fff;
    &.active {
      display: block;
    }
  }
}

jQuery


$(function(){

$('.tab-title li').on('click', function(){
  /* タブ */
  var th = $(this).index();  // 何番目のタブがクリックされたか
  var $tab_contents = $('.tab-contents li');
  $tab_contents.removeClass('active');
  $tab_contents.eq(th).addClass('active');
  
  /* スライドバー */
  var position = $(this).width() * th;  // スライダーの位置
  $('.tab-title-bar').css('left', position+'px');
});

});

波紋を追加する

デモ

  • Tab1
  • Tab2
  • Tab3
  • Tab4
  • Tab1 Contents
  • Tab2 Contents
  • Tab3 Contents
  • Tab4 Contents

マテリアルデザインといえば波紋エフェクトですよね!

HTML


  • Tab1
  • Tab2
  • Tab3
  • Tab4
  • Tab1 Contents
  • Tab2 Contents
  • Tab3 Contents
  • Tab4 Contents

HTMLは先ほどと変わりません。

CSS


/* RESET */
ul, li {
  display: block;
  margin: 0;
  padding: 0;
  list-style-type: none;
}

.tab-title {
  position: relative;
  width: 100%;
}
.tab-title:before, .tab-title:after {
  display: table;
  content: '';
}
.tab-title:after {
  clear: both;
}
.tab-title li {
  position: relative;
  float: left;
  width: 25%;
  padding: .8em 0;
  color: #fff;
  text-align: center;
  background-color: #26C6DA;
  cursor: pointer;
  overflow: hidden;
}
.tab-title-bar {
  position: absolute;
  left: 0;
  bottom: 0;
  width: 25%;
  height: 2px;
  background-color: #F06292;
  -webkit-transition: .45s ease-in-out;
  -moz-transition: .45s ease-in-out;
  -o-transition: .45s ease-in-out;
  transition: .45s ease-in-out;
}
.tab-contents li {
  display: none;
  padding: 1.4em;
  background-color: #fff;
}
.tab-contents li.active {
  display: block;
}

/* 波紋 */
.ripple {
  position: absolute;
  width: 0;
  height: 0;
  border-radius: 100%;
  background-color: rgba(255, 255, 255, 0.4);
  opacity: 1;
  -webkit-transform: scale(0);
  -moz-transform: scale(0);
  -ms-transform: scale(0);
  transform: scale(0);
}
.ripple-animation {
  -webkit-animation: ripple .6s linear;
  -moz-animation: ripple .6s linear;
  animation: ripple .6s linear;
}
@-webkit-keyframes ripple {
  100% {
    opacity: 0;
    -webkit-transform: scale(2);
    -moz-transform: scale(2);
    -ms-transform: scale(2);
    transform: scale(2);
  }
}
@keyframes ripple {
  100% {
    opacity: 0;
    -webkit-transform: scale(2);
    -moz-transform: scale(2);
    -ms-transform: scale(2);
    transform: scale(2);
  }
}

波紋用のスタイルを追加しています。

SCSS


/* RESET */
ul, li {
  display: block;
  margin: 0;
  padding: 0;
  list-style-type: none;
}

.tab-title {
  position: relative;
  width: 100%;
  &:before, &:after {
    display: table;
    content: '';
  }
  &:after {
    clear: both;
  }
  li {
    position: relative;
    float: left;
    width: 25%;
    padding: .8em 0;
    color: #fff;
    text-align: center;
    background-color: #26C6DA;
    cursor: pointer;
    overflow: hidden;
  }
}
.tab-title-bar {
  position: absolute;
  left: 0;
  bottom: 0;
  width: 25%;
  height: 2px;
  background-color: #F06292;
  -webkit-transition: .45s ease-in-out;
  -moz-transition: .45s ease-in-out;
  -o-transition: .45s ease-in-out;
  transition: .45s ease-in-out;
}
.tab-contents {
  li {
    display: none;
    padding: 1.4em;
    background-color: #fff;
    &.active {
      display: block;
    }
  }
}

/* 波紋 */
.ripple {
  position: absolute;
  width: 0;
  height: 0;
  border-radius: 100%;
  background-color: rgba(255, 255, 255, .4);
  opacity: 1;
  -webkit-transform: scale(0);
  -moz-transform: scale(0);
  -ms-transform: scale(0);
  transform: scale(0);
}
.ripple-animation {
  -webkit-animation: ripple .6s linear;
  -moz-animation: ripple .6s linear;
  animation: ripple .6s linear;
}
@-webkit-keyframes ripple {
  100% {
    opacity: 0;
    -webkit-transform: scale(2);
    -moz-transform: scale(2);
    -ms-transform: scale(2);
    transform: scale(2);
  }
}
@keyframes ripple {
  100% {
    opacity: 0;
    -webkit-transform: scale(2);
    -moz-transform: scale(2);
    -ms-transform: scale(2);
    transform: scale(2);
  }
}

jQuery


$(function(){

$('.tab-title li').on('click', function(e){
  /* タブ */
  var th = $(this).index();  // 何番目のタブがクリックされたか
  var $tab_contents = $('.tab-contents li');
  $tab_contents.removeClass('active');
  $tab_contents.eq(th).addClass('active');
  
  /* スライドバー */
  var position = $(this).width() * th;  // 位置
  $('.tab-title-bar').css('left', position+'px');
  
  /* 波紋エフェクト */
  $('.tab .ripple').remove();  // 前の波紋要素を消す
  
  var tab_width = $(this).width(),
      tab_height = $(this).height(),
      tab_x = $(this).offset().left,
      tab_y = $(this).offset().top;
  
  // 大きい方の幅に合わせる
  var size = tab_width >= tab_height ? tab_width : tab_height;
  
  // 波紋要素を追加
  $(this).prepend('
'); // 中心座標を算出 var x = e.pageX - tab_x - size / 2, y = e.pageY - tab_y - size / 2; $('.ripple').css({ width: size, height: size, top: y + 'px', left: x + 'px' }).addClass('ripple-animation'); }); });