jQueryでUIコンポーネントの作り方を学ぶ前に、jQueryプラグインの作り方と仕組みをしっかりとおさえよう。
この連載で取り扱っているjQueryはJavaScriptのライブラリエンジンです。
世の中には実にさまざまなプログラミング言語があります。昨今主流となっているものの大半は、オブジェクト指向プログラミング言語と呼ばれているものです。
代表的な例を挙げると、Java、C#、Ruby、Python、Objective-Cといったところでしょう。これらの言語はクラスベースというカテゴリに属しています。クラスベースはアプリケーションにおけるさまざまな機能をクラスと呼ばれる単位でキッチリと分割することで、プログラム全体をキレイに整理整頓できるという特徴を持っています。大規模な開発になるほど、この特徴が威力を発揮するため、先に挙げたプログラミング言語が積極的に採用される要因の1つとなっているわけですね。
一方のJavaScriptですが、オブジェクト指向プログラミング言語ではあるものの、こちらはクラスベースではなくプロトタイプベースというカテゴリに属しています。詳しい解説は省きますが、プロトタイプベースはクラスのような仕組みを持っていないため、機能ごとにキッチリ分割して整理整頓することができません。もちろん先人たちの知恵と努力によってそれらしく見せかけるノウハウはありますが、あくまで擬似的な域を超えられていません。
もともと、JavaScriptはWebブラウザ上で動作する軽量なスクリプト言語として世に出てきたため、昨今に見られるような大規模開発に用いられることは想定されていませんでした。
しかし時代は変わって、Webブラウザ上でリッチなアプリケーションが提供されることがトレンドとなった今、おのずとJavaScriptでの大規模開発が迫られています。プログラマたちはどうにかして JavaScriptでも効率良く開発が進められるようにクラスベースのようなアーキテクチャを構築できないかと試行錯誤してきました。その動きは今もなお活発で、今回紹介するjQueryプラグインもそんなアーキテクチャの1つです。
随分と前置きが長くなってしまいました。今後、この連載ではjQueryでさまざまなUIコンポーネントの作り方をご紹介していく予定ですが、そのためにはまずjQueryプラグインの作り方についてしっかりと学んでおく必要があります。仕組みさえ分かってしまえばそれほど難しいものでもありませんし、手間の掛かるものでもありません。
以下の環境を用意しましょう。
今回のサンプルファイルはwakamsha/learning-jqueryplugin - GitHubからダウンロードできます。必要に応じてご参照ください。
HTMLファイルにてjQueryファイルを読み込む際は、以下のコードのようにオフィシャルサイトのCDNを利用します。
<script src="http://code.jquery.com/jquery-latest.min.js"></script>
ファイルそのものを直接ローカル環境にダウンロードしても問題ありませんが、管理するファイルが増えるのも手間なので、ネットワーク上から直接読み込む方を推奨します。
細かい差はあれど、基本的にjQueryプラグインは以下のような構造で成り立っています。
// 匿名関数で全体をラップ - (5) (function($) { // このプラグインの名前 - (1) $.fn.name_space = function() { //要素を退避 - (2) var elements = this; // 要素を1つずつ処理 - (3) elements.each(function() { // 具体的な処理をここに記述 }); // method chain用に要素を返す - (4) return this; }; }) (jQuery);
では実際にサンプルを作りながら詳しく見ていきましょう。
まずは最小規模なものから。やはり最初に実行するプログラムということで、K&R以来の伝統にのっとって "Hello, World!" から始めたいと思います。「またかよ……」と思われる方もいらっしゃるかもしれませんが、新しいものを学ぶには、"Hello, World!"はとても重要なのです。
テキストエディタで新規ファイルを作成し、"jquery.myplugin.js"と名前を付けて保存します。一般的にjQueryプラグインのファイル名にはjquery.という接頭辞を付けることで、中身を見ずともそれがjQueryプラグインであることが分かるようにするのがお決まりとなっています(※すべてのプラグインがそうなっているわけではなく、あくまで慣習的なものです)。ファイルが作成できたら以下のコードを書いてみましょう。
// このプラグインの名前 $.fn.myplugin = function() { //要素を変数に退避 var elements = this; // 要素を1つずつ処理 elements.each(function() { $('body').append('<p>Message: ' + this.innerHTML + '</p>'); }); // method chain用に要素を返す return this; };
では上記のコードを1つずつ解説していきましょう。
jQueryプラグインはカスタムjQueryメソッドとしてjQueryを拡張したものです。定義するには$.fnオブジェクトをプラグインの名前(カスタムメソッド名)で拡張します。
thisには指定された要素が格納されています。これを変数に退避しておきます(※必須というわけではありません。していないプラグインもたくさんあります)。
jQueryオブジェクトは要素が複数含まれていることが前提となっています。そのためelementsにも複数の要素が含まれていると考え、それらに対してすべて同様に処理しなくてはなりません。ということで each() を使ってelementsをループに掛けて、要素を1つずつ処理してきます。
jQueryにはMethod chainという処理を連鎖させるアーキテクチャがあります。
$('div') // div要素を取得 .text('new context') // 隠したdiv要素にテキストを'new context'にする .addClass('update') // クラスを追加する .show(); // div要素を表示する
このような書き方が成立するのは、hide()やtext()といったメソッドが処理の最後に対象となった要素を丸ごと返しているからです。jQueryプラグインでカスタムメソッドを定義する際も、最後に要素を返すことで処理を連鎖(chain)させられます。返す方法はカスタムメソッド内の最後にreturn this;と書くだけです。決まり文句として覚えておきましょう。
これで初めてのjQueryプラグインができました。使い方はjQueryデフォルトの関数とまったく同じです。HTMLファイルで作成したjQueryプラグインファイルを読み込んだ後、以下のように新しいメソッド名を指定すれば実行されます。
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>My plugin | Learning jQuery plugin</title> <style> p { color: #4c5069; font-size: 3em; } </style> </head> <body> <div>Hello, world!</div> <script src="http://code.jquery.com/jquery-latest.min.js"></script> <script src="js/jquery.myplugin.js"></script> <script> $('div').myplugin(); </script> </body> </html>
カスタムメソッドにオプションを渡すことで、プラグインの利便性を何倍にも高めることができます。オプションを渡す方法は、optionsオブジェクトを通じてカスタムメソッドに渡すのが一般的です。optionsオブジェクトにすることで、複数のパラメータをまとめて渡すことができるようになるからです。
プラグインにオプション機能を追加する際は、デフォルト値を定義するのがセオリーです。そのデフォルト値に対してプラグインのユーザーは任意のオプションを渡して(※必須というわけではありません。していないプラグインもたくさんあります)上書きできるようにします。
jquery.shake.jsというファイルを新規に作成し、以下のコードを記述します。
$.fn.shake = function(options) { // 要素を退避 var elements = this; // 渡されたオプションとデフォルトをマージする var opts = $.extend({}, $.fn.shake.defaults, options); // 要素を1つずつ処理 elements.each(function() { for (var i=0; i<opts.shakes; i++) { $(this).animate({marginLeft: opts.x}, opts.speed) .animate({marginLeft: opts.x * -1}, opts.speed); } // 要素を元に戻す $(this).animate({marginLeft: 0}, opts.speed); }); // method chain用に要素を返す return this; }; // shakeプラグインのデフォルトオプション $.fn.shake.defaults = { speed: 'slow', shakes: 2, x: 10 };
デフォルトオプションは上記のようにして定義します。これを extend() メソッドを使って、ユーザーが指定したオプションで上書きします。extend() の第一引数に {} (空オブジェクト)を指定すると、デフォルトオプションそのものは上書きされずにデフォルトオプションとユーザー指定オプションがマージされたものが作られます。ちょっとした小技ですが、覚えておいて損はありません。
デフォルトオプションを定義することで、プラグインのユーザーは必要なオプションのみを指定することができます。
// オプションを1つだけ指定 $('div').shake({speed: 'fast'}); // オプションをすべて指定 $('div').shake({speed: 'fast', shakes: 10, x: 20}); // オプションを指定しない $('div').shake();
Copyright © ITmedia, Inc. All Rights Reserved.