LoginSignup
13
18

More than 5 years have passed since last update.

[JavaScript]jQueryプラグイン、select2を始める!そして、select2の改善事項

Last updated at Posted at 2016-09-09

1%足りないエレメント、select

HTML5ではselectエレメントを利用してComboBoxコントロールを実装することができます。しかし、基本エレメントだけでは期待値が高い使用者の厳しい要求を受け入れることに機能があまりにも足りないです。
このようなselectエレメントの足りない機能を強化するアドオン性格のライブラリが結構ありますが、今回の文章では多く使用されるjQueryプラグイン、Select2の説明と改善事項に関して説明させていただきたいお思っております。

Select2の特徴

本格的に使用法を整理するのは🔗ドキュメントに本当に優しく説明しておりますので、ご覧ください。Select2ライブラリーの基本的な特徴は以下の通りであります。
自動完成(AutoComplete)とAJAXを利用した遠隔データ読み込みが可能であります。
自動完成の場合は日本語は文字単位から可能です。また、jQuery基盤のプラグインでjQueryが必須です。Bootstrapテーマを支援して違和感なく使用が可能です。
GPL v2ライセンスです。商業的な目的の修正がない以上、無料使用が可能です。

概要

皆様、🔗select2というjQueryプラグインをよく使っていますね。
利用ケースは🔗こちらにある例たちをご覧ください。
image

前回のプロジェクトでこのプラグインを利用しましたが、
改善事項がいくつか必要になりましたので、同じ状況の方に共有したいと思っています。

なぜ、改善が必要か

前回のプロジェクトで求めている<select>要素は以下のとおりでした。
・Pulldownリストで選択
・マルチ選択
・選択した項目はタグとして配置
・チェックボックスがあるPulldownリスト

🔗ドキュメントにある例を見て、「以外と簡単に出来るじゃん〜:-)」と思いました。
しかし、 問題がDone!!!YA・BA・I !
一応以下の内容で動作しておりますが、いくつかの改善事項が必要でした。

✪求めているの通り動作するシンプルな設定方法

select2_example.js
$(".user_list").select2({
    closeOnSelect : false
});

結果のご参考は🔗こちら

✪改善事項
① マルチ選択する場合、Pulldown ListエリアのScrollTopがリセットされる。
② マルチ選択する場合、Pulldown Listエリアが固定され、入力タグが積み重ねるエリアを隠してしまう。
image
③ マルチ選択する場合、PulldownList以外のエリアをクリックしてもPulldownListエリアが閉められない。
④ マルチ選択する場合、PulldownListにチェックボックスがあるフォマットを提供していないので改善が必要
⑤ 選択のエリアとタグが重ねるエリアを分けるようなフォマットを提供していないので改善が必要

改善内容

① マルチ選択する場合、Pulldown ListエリアのScrollTopが維持されるように対応
以下のコードをご覧するとすぐご存知だと思いますが、選択したり選択したものを外する場合、クリックした要素(Object)のoffsetTopを利用しております。

select2u.js
const SELECT2_PULLDOWNLIST_OPTIONS_AREA_SELECT_NAME = '.select2-dropdown > .select2-results > .select2-results__options';
const SELECT2_PULLDOWNLIST_OPTIONS_SELECT_NAME = '.select2-results__option';
const SELECT2_PULLDOWNLIST_OPTION_MOUSE_OVER_CLASS_NAME = 'select2-results__option--highlighted';

var select2u = {

    //このObjectの全判に扱う要素を定義する
    $select2: null,

    //メインのjQueryプラグインSelect2の要素に設定する
    setInit: function(targetClassName) {
        select2u.$select2 = $("."+targetClassName).select2({ /*options*/ });
    }

    setSelect2SelectEventHandler: function() {
        select2u.$select2.on("select2:select", function (e) {
            select2u.getEventCtl(e);
        });
    },

    setSelect2UnSelectEventHandler: function() {
        select2u.$select2.on("select2:unselect", function (e) {  
            select2u.getEventCtl(e);
        });
    }, 

    //selectとunselectの場合の処理が一緒で、違うパラメーターだけ渡す
    getEventCtl: function(e) {
        var targetElem = e.params.data.element.offsetTop;

        //PulldownListAreaのスクロールを維持する
        $(SELECT2_PULLDOWNLIST_OPTIONS_AREA_SELECT_NAME).scrollTop(targetElem*1.4);
        //PulldownListAreaのOptionたちのmouseOverの表示(背景)をリセットする
        $(SELECT2_PULLDOWNLIST_OPTIONS_SELECT_NAME).removeClass(SELECT2_PULLDOWNLIST_OPTION_MOUSE_OVER_CLASS_NAME);
        //PulldownListAreaのOptionの正しいターゲットにmouseOverの表示(背景)ををセット
        $(targetElemId).addClass(SELECT2_PULLDOWNLIST_OPTION_MOUSE_OVER_CLASS_NAME);
    }
};

② マルチ選択する場合、Pulldown Listエリアが固定され、入力タグが積み重ねるエリアを隠してしまう現状解決
こちらは、上端で紹介した、closeOnSelect : falseの設定をやめて、実装しました。
対応はものすごく簡単です。

select2u.js
var select2u = {

    //このObjectの全判に扱う要素を定義する
    $select2: null,

    //メインのjQueryプラグインSelect2の要素に設定する
    setInit: function(targetClassName) {
        select2u.$select2 = $("."+targetClassName).select2({ /*options*/ });
    }

    setSelect2SelectEventHandler: function() {
        select2u.$select2.on("select2:select", function (e) {
            select2u.getEventCtl(e);
        });
    },

    setSelect2UnSelectEventHandler: function() {
        select2u.$select2.on("select2:unselect", function (e) {  
            select2u.getEventCtl(e);
        });
    }, 

    //selectとunselectの場合の処理が一緒で、違うパラメーターだけ渡す
    getEventCtl: function(e) {
        select2u.$select2.select2("open");
    }
};

image

④ マルチ選択する場合、PulldownListにチェックボックスがあるフォマットを提供していないので改善
プラグインのオブジェクトから提供しているtemplateオブジェクト設定で実装を入れます。
templateResultのオプションで内部項目に関して調整とその意外の処理をしています。

select2u.js
const SELECT2_PULLDOWNLIST_CHECKED_OPTIONS_SELECT_NAME = '.select2-results__option[aria-selected = true] > input';

var select2u = {

    //このObjectの全判に扱う要素を定義する
    $select2: null,

    //メインのjQueryプラグインSelect2の要素に設定する
    setInit: function(targetClassName) {
        select2u.$select2 = $("."+targetClassName).select2({
            //要素(options)を操作するテンプレートメソッド
            templateResult: function (data) {
                if (data.id === '') {
                    return '選択する項目がありません';
                }
                //要素(options)を確認して内容にcheckboxを追加してReturnする
                return "<input type='checkbox' class='select2-pulldown-checkbox'><label for='checkboxId'>"+data.text+"</label>";
            },
            //マークアップに設定する(checkboxのため)
            escapeMarkup: function (markup) { return markup; },
            //その以外の設定(項目以外入れる & 全体Cancelボタンの設定を一応コメントアウトする)
            // allowClear: true,
            // tags: true
        });

    }

    setSelect2SelectEventHandler: function() {
        select2u.$select2.on("select2:select", function (e) {
            select2u.getEventCtl(e);
        });
    },

    setSelect2UnSelectEventHandler: function() {
        select2u.$select2.on("select2:unselect", function (e) {  
            select2u.getEventCtl(e);
        });
    }, 

    //selectとunselectの場合の処理が一緒で、違うパラメーターだけ渡す
    getEventCtl: function(e) {
        $(SELECT2_PULLDOWNLIST_CHECKED_OPTIONS_SELECT_NAME).prop('checked', true);
    }
};

image

⑤ 選択のエリアとタグが重ねるエリアを分けるようなフォマットを提供していないので改善
templateResult: function (data) {}select2u.$select2OnlyDisplay.val(select2u.$select2.val()).trigger("change");の部分が重要ポイントです。
選択のみの<select>要素に選択してもタグが生成されない設定をしておいて、
タグのみの要素には、Pulldown Listが出ないように設定しております。
そして、選択のみの予想から選択した場合、タグ要素の値をトリガーします。

select2u.js
var select2u = {

    //このObjectの全判に扱う要素を定義する
    $select2: null,
    $select2OnlyDisplay: null,

    //メインのjQueryプラグインSelect2の要素に設定する
    setInit: function(targetClassName) {
        select2u.$select2 = $("."+targetClassName).select2({ /* options */ });
    },

    //タグ表示のみの要素の設定する
    setOnlyDisplayInit: function(targetClassNameToOnlyDisplay) {
        return new Promise((resolve, reject) => {
            select2u.$select2OnlyDisplay = $("."+targetClassNameToOnlyDisplay).select2({
                // allowClear: true,
                // tags: true,
                //PulldownListが表示されないように何もしない
                templateResult: function (data) {}
            });
        });
    },

    setSelect2SelectEventHandler: function() {
        select2u.$select2.on("select2:select", function (e) {
            select2u.getEventCtl(e);
        });
    },

    setSelect2UnSelectEventHandler: function() {
        select2u.$select2.on("select2:unselect", function (e) {  
            select2u.getEventCtl(e);
        });
    }, 

    //selectとunselectの場合の処理が一緒で、違うパラメーターだけ渡す
    getEventCtl: function(e) {
        select2u.$select2OnlyDisplay.val(select2u.$select2.val()).trigger("change"); 
    }
};

まとめ

👏👏👏👏👏👏👏

13
18
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
13
18