トップへ戻るボタンにスクロール量を表示する方法

Cocoonカスタマイズ集
この記事は約10分で読めます。
記事内に広告が含まれています。

記事に記載したコードは、スキンに追加されています。

はじめに

トップへ戻るボタンに、ページのスクロール量を表示を追加します。
この記事では、この方法について説明します。

完成イメージ

実装手順

以下の手順で実装します。

  • ラベル
    Cocoon設定

    表1に示す項目を設定します。

    Cocoon設定
    設定大項目項目設定値
    ボタントップへ戻るボタンの表示トップへ戻るボタンを表示するオン
  • ラベル
    JavaScriptを追加

    以下のコードをjavascript.jsに追加します。

    jQuery(function($) {
      $(window).on('load scroll', function() {
        // スクロール量を算出
        var scroll = $(window).scrollTop();
        var height = $(document).height() - $(window).height();
        var progress = scroll / height * 100;
        $('.go-to-top-button').css('--per', progress + '%');
      });
    });
  • ラベル
    CSSを追加

    以下のコードをstyle.cssに追加します。

    .go-to-top-button {
      background-color: transparent;
    }
    
    .go-to-top-button:before {
      --my-color: #eee;       /* プログレス色 */
      --my-progress: #ff0000; /* スクロール量色 */
      background-image: conic-gradient(var(--my-progress) var(--per), var(--my-progress) var(--per), var(--my-color) var(--per), var(--my-color) 100%);
      border-radius: 100%;
      content: '';
      display: block;
      height: 100%;
      mask: radial-gradient(circle, transparent 50%, #000 54%);
      position: absolute;
      width: 100%;
    }

参考:SVGを用い円を描きスクロール量を表示する方法

SVGを使用して円を描き、スクロール量を表示します。
Cocoon標準のトップへ戻るボタンではなく、独自のボタンを使用します。

完成イメージ

実装手順

以下の手順で実装します。

  • ラベル
    Cocoon設定

    表1に示す項目を設定します。

    Cocoon設定
    設定大項目項目設定値
    ボタントップへ戻るボタンの表示トップへ戻るボタンを表示するオン
  • ラベル
    HTMLを追加

    以下のコードをtmp-user/main-before.phpに追加します。

    <div class="go-top-wrap">
      <svg class="go-to-circle" width="50px" height="50px" viewBox="-5 -5 100 100">
        <path d="M 45,0 A 45 45 0 1 1 44,0 z"/>
      </svg>
    </div>
  • ラベル
    JavaScriptを追加

    以下のコードをjavascript.cssに追加します。

    jQuery(function($) {
      var offset = 50;
      var duration = 550;
    
      // style設定
      var progressPath = document.querySelector('.go-top-wrap path');
      var pathLength = progressPath.getTotalLength();
      progressPath.style.strokeDasharray = pathLength + ' ' + pathLength;
      progressPath.style.strokeDashoffset = pathLength;
      progressPath.style.transition = 'stroke-dashoffset 10ms linear';
    
      // スクロール位取得
      function updateProgress() {
        var scroll = $(window).scrollTop();
        var height = $(document).height() - $(window).height();
        var progress = pathLength - (scroll * pathLength / height);
        progressPath.style.strokeDashoffset = progress;
      }
        
      $(window).on('load scroll', function() {
        updateProgress();
    
        if ($(this).scrollTop() > offset) {
          $('.go-top-wrap').addClass('active');
        } else {
          $('.go-top-wrap').removeClass('active');
        }
      });
    
      // ボタンクリックの場合
      $('.go-top-wrap').on('click', function(event) {
        event.preventDefault();
        $('html, body').animate({scrollTop: 0}, duration);
        return false;
      });
    });
  • ラベル
    CSSを追加

    以下のコードをstyle.cssに追加します。

    .go-top-wrap path {
      fill: none;
    }
    
    /* 外枠 */
    .go-top-wrap {
      border-radius: 50%;
      bottom: 25px;
      box-shadow: inset 0 0 0 5px #ccc;
      cursor: pointer;
      display: block;
      height: 50px;
      opacity: 0;
      position: fixed;
      right: 50px;
      transform: translateY(15px);
      transition: all 200ms linear;
      visibility: hidden;
      width: 50px;
      z-index: 9999;
    }
    
    /* 表示 */
    .go-top-wrap.active {
      opacity: 1;
      transform: translateY(0);
      visibility: visible;
    }
    
    /* 矢印 */
    .go-top-wrap:before {
      color: #333;
      content: '\f062';
      display: block;
      font-family: 'Font Awesome 5 Free';
      font-size: 24px;
      font-weight: 900;
      left: 0;
      line-height: 50px;
      position: absolute;
      text-align: center;
      top: 0;
      width: 50px;
      z-index: 2;
    }
    
    /* スクロール量 */
    .go-top-wrap path {
      box-sizing: border-box;
      stroke: #333;
      stroke-width: 10px;
    }
    
    /* マウスオーバー矢印アニメーション */
    .go-top-wrap:hover:before {
      animation-duration: 0.75s;
      animation-name: hvr-icon-up;
      animation-timing-function: ease-out;
    }
    
    @keyframes hvr-icon-up {
      0%, 50%, 100% {
        transform: translateY(0);
      }
      25%, 75% {
        transform: translateY(-8px);
      }
    }

参考:CSSのみでスクロール量を表示する方法

CSSのanimation-timelineを使用すると、JavaScriptなしでスクロール量を表示できます。
ただし、すべてのブラウザに対応しているわけではありません(対応状況は以下を参照)。

以下の例では、ボタンを任意のアイコンに変更し、そのアイコン内にスクロール量を表示します。

完成イメージ

実装方法

  • ラベル
    Cocoon設定

    表1に示す項目を設定します。

    Cocoon設定
    設定大項目項目設定値
    ボタントップへ戻るボタンの表示トップへ戻るボタンを表示するオン
  • ラベル
    CSSを追加

    以下のコードをstyle.cssに追加します。

    /* ブラウザがanimation-timelineサポートしているなら有効 */
    @supports (animation-timeline: scroll()) {
      /* ボタンに影 */
      .go-to-top {
        filter: drop-shadow(rgba(0, 0, 0, 0.5) 1px 1px 2px);
      }
    
      /* スクロール量に重ねるマスク画像 */
      .go-to-top-button-icon-font {
        -webkit-mask-image:url('');
        -webkit-mask-position: center;
        -webkit-mask-repeat: no-repeat;
        -webkit-mask-size: contain;
      }
    
      /* スクロールに応じアニメーション */
      .go-to-top-button-icon-font span {
        animation: progress linear;
        animation-timeline: scroll();
        background-color: #00a9e0;
        display: block;
        height: 100%;
        transform-origin: bottom;
        width: 100%;
      }
    
      .go-to-top-button-icon-font span:before {
        content: none;
      }
    
      @keyframes progress {
        from {
          scale: 1 0;
        }
        to {
          scale: 1 1;
        }
      }
    }

以下のコードをstyle.cssに追加します。

さいごに

今回紹介した方法を使えば、トップへ戻るボタンにスクロール量を表示させることができます。
これにより、ユーザーはページのどの位置にいるかを一目で把握できるようになります。

タイトルとURLをコピーしました