スキップしてメイン コンテンツに移動

Bootstrap3のAffixバグについて

【Affixとは】




サイドメニューなどを、画面と一緒にスクロールさせずに固定位置で表示するためのBootstrapのビルトイン便利JSです。
http://getbootstrap.com/javascript/#affix

また、サイドメニューの上部高さとフッターの高さを指定すると、上部高さまで、および、フッターが表示されるとフッターに押されて画面と一緒にスクロールされます。

上部高さはoffsetオプションのtopで指定し、フッターの高さはoffsetオプションのbottomで指定します。
具体例は以下

$('サイドメニュー').affix({

  offset: {

    top: function () {

      return (this.top = $('ヘッダー部分').outerHeight(true))

    },

    bottom: function () {

      return (this.bottom = $('フッター部分').outerHeight(true))

    }

  }

});


【バグの事象】(v3.1.1で確認)

スクロール中は動的にJSにより.affixクラスが付与され、position:fixed(自分でCSSに設定)となりサイドメニューの位置は固定され、bottom(フッター位置)までスクロールすると、動的にJSにより.affix-bottomクラスが付与され、position:absolude(自分でCSSに設定。top位置はJSが計算)となり画面と一緒に上に移動してゆきますが、スクロールで上部に切り返しても.affix-bottomが解除されず位置がフッターの上部に固定されたままになってしまいます。
(一旦、画面上部までスクロールすると.affix-topが付与され、サイドメニューは初期位置までジャンプします。その後画面下部にスクロールすると.affixが付与され正常動作(位置固定)しますが、bottomまでスクロールされると上記不具合(.affix-bottomクラスが付与され解除されない)が発生します。)


【バグの報告】

以下でバグ報告が挙げられています。
https://github.com/twbs/bootstrap/issues/9342
https://github.com/twbs/bootstrap/issues/12736


【修正されたaffix.js】

以下からダウンロード可能です。(2014.3.26現在)
https://github.com/twbs/bootstrap/blob/master/js/affix.js


【主な修正点】

<NG>
    if (affix == 'bottom') {

      this.$element.offset({ top: scrollHeight - offsetBottom - this.$element.height() })

    }

<OK>
    if (affix == 'bottom') {

      this.$element.offset({ top: position.top })

    }

NGは、スクロール高さ(=サイト全体の高さ)からオプションで指定したbottomの高さとサイドメニュー自身の高さを引いた位置をtopの位置としていましたが、サイドメニューの高さや、フッターの高さに影響され不具合が発生する場合があります。
OKは、単純に自身の位置をサイドメニューのtopの位置に変更されています。

以上です。


Bootstrapはとても便利で今やRWDのサイト制作には欠かせないフレームワークですが、Bootstrap自体にバグがあるとにっちもさっちもゆかなくなります。(自分のコードに問題がなくても事象が解決されないので何時間も悩んでしまいます。)

それでもいちからRWDを自前で制作しようとすると
・幅pxを%に計算すると...
・このJS使いたいけどレスポンシブ対応してるかしら...
・こんな共通クラスあったら便利だからまず作っておいて...
・わー、このサイトもう自分しか解読できないな...
みたいなことで悩む時間をBootstrapが解決してくれますから断然Bootstrapの使用をオススメします。

コメント

このブログの人気の投稿

WordPressのwp_get_archivesで月別リストを生成する際に<li>をカスタムする方法

【About】 WordPressの wp_get_archives('type=monthly') で月別のリスト<li>を生成する際に、特定の年月のアーカイブページを表示した際に<li>に特定のクラス(class="active"など)を設定する方法を記載します。
また、おまけとして、functions.phpによらず、テンプレートで<li>をカスタマイズする方法も記載します。

【通常の書き方】 <WordPress>ーテンプレート
<p>月別アーカイブ</p> <ul>     <?php wp_get_archives('type=monthly')); ?> </ul>
とすると

<HTML><p>月別アーカイブ</p> <ul>     <li><a href="http://hoge.com/2013/12/">2013年12月</a></li>     <li><a href="http://hoge.com/2013/11/">2013年11月</a></li> </ul>
などと生成されます。

これを、ある月を表示しているときに、該当のリスト<li>に class="active" を追加したいとします。

【カスタマイズ方法】 <WordPress>ーテンプレート
<p>月別アーカイブ</p> <ul>     <?php wp_get_archives('type=monthly')); ?> </ul> (変更なし)

ーfunctions.php
functions.phpに以下を追加
<?php ・・・ function my_archives_link($link_html){   $my_month = '/' . get_query_var('year') . &#…

PhotoSwipeでテキストリンクから特定の画像のPhotoSwipeギャラリーを起動する方法

【PhotoSwipeとは】 スマホサイトで写真ギャラリーを設置したいときに使えるライブラリ
スワイプで画像がめくれて便利!
レスポンシブサイトの画像ギャラリーにオススメ。
(optionに、imageScaleMethod: "fitNoUpscale" とすれば画像が最大100%で表示されるのでPCサイトでも使えます。)
https://github.com/codecomputerlove/PhotoSwipe


【実現したいこと】 通常は画像サムネイルのクリック(or タップ)でPhotoswipeギャラリーを起動します。
それとは別に、新着情報などのテキストリンクから特定の画像のPhotoSwipeギャラリーを起動したい。


【方針】var instance = $("#Gallery a").photoSwipe(options); instance.show(0); で、最初のPhotoSwipeギャラリーを起動できることから、テキストリンクに任意の要素(data-swipeとする)と、クラス(.swipe-targetとする)を設置し、.swipe-targetの上から順番にdata-swipeに番号を振ってゆき、.swipe-targetがクリックされたら、data-swipeから番号xを取得し、instance.show(x)で目的のPhotoSwipeギャラリーを起動する。


【HTML】 ヘッダー(もしくはフッター)部分(jQueryバージョンを使用)<script src="//ajax.googleapis.com/ajax/libs/jquery/1.8.2/jquery.min.js"></script> <script>window.jQuery || document.write('<script src="/js/vendor/jquery-1.8.2.min.js"><\/script>')</script> <script src="/js/vendor/klass.min.js"></script> <scri…

<input type="file" ...> のvalueをクリアする方法

選び直す場合は再度「ファイルを選択」ボタン(IEの場合「参照」ボタン?)で選び直せばよいのですが、選択必須ではない任意選択項目の場合、一度選んでしまったファイルのvalueを解除する必要があります。

しかし

input type="file"のvalueにはブラウザのセキュリティー上アクセスできないようなので
(確かにスクリプトでvalueを変更できたら大変ですよね。)

$('type="file"の要素').val('');

とか

$('type="file"の要素').replaceWith('<input type="file" ...>');

とかやっても解除されません。。。orz


なので、typeを"text"にして、textのvalueを変更することで元のtype="file"のvalueがリセットできました。


[sample code]
$('type="file"の要素').attr("type","text");
$('type="file"の要素').val('dummy');


無理やり〜(^^;