| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638 |
- <?xml version="1.0" encoding="UTF-8"?>
- <!-- Reviewed: no -->
- <!-- EN-Revision: 24249 -->
- <sect1 id="zend.paginator.usage">
- <title>使用法</title>
- <sect2 id="zend.paginator.usage.paginating">
- <title>データコレクションのページ処理</title>
- <para>
- ページ処理を行うには、<classname>Zend_Paginator</classname>
- がデータにアクセスするための汎用的な方法が必要です。
- そのため、データへのアクセスはすべてデータソースアダプタを用いて行います。
- Zend Framework には、いくつかのアダプタが標準で同梱されています。
- </para>
- <table id="zend.paginator.usage.paginating.adapters">
- <title>Zend_Paginator 用のアダプタ</title>
- <tgroup cols="2">
- <thead>
- <row>
- <entry>アダプタ</entry>
- <entry>説明</entry>
- </row>
- </thead>
- <tbody>
- <row>
- <entry>Array</entry>
- <entry><acronym>PHP</acronym> の配列を使用します。</entry>
- </row>
- <row>
- <entry>DbSelect</entry>
- <entry>
- <link linkend="zend.db.select"><classname>Zend_Db_Select</classname></link>
- のインスタンスを使用し、配列を返します。
- </entry>
- </row>
- <row>
- <entry>DbTableSelect</entry>
- <entry>
- <link linkend="zend.db.table.fetch-all"><classname>Zend_Db_Table_Select</classname></link>
- のインスタンスを使用し、
- <classname>Zend_Db_Table_Rowset_Abstract</classname>
- のインスタンスを返します。
- これは、結果セットについての追加情報
- (カラム名など) を提供します。
- </entry>
- </row>
- <row>
- <entry>Iterator</entry>
- <entry>
- <ulink url="http://www.php.net/~helly/php/ext/spl/interfaceIterator.html"><classname>Iterator</classname></ulink>
- のインスタンスを使用します。
- </entry>
- </row>
- <row>
- <entry>Null</entry>
- <entry>
- データのページ処理を管理する際に <classname>Zend_Paginator</classname>
- を使用しません。その場合でもページ処理コントロールの機能を使うことはできます。
- </entry>
- </row>
- </tbody>
- </tgroup>
- </table>
- <note>
- <para>
- 指定したクエリにマッチするすべての行を取得するのではなく、
- DbSelect アダプタや DbTableSelect アダプタは
- 現在のページの表示のための必要最小限のデータのみを取得します。
- </para>
- <para>
- そのため、マッチする行の総数を得るための別のクエリが動的に生成されます。
- しかし、総数を直接指定したり、総数を求めるクエリを直接指定したりすることもできます。
- 詳細な情報は、DbSelect アダプタの
- <methodname>setRowCount()</methodname> メソッドを参照ください。
- </para>
- </note>
- <para>
- <classname>Zend_Paginator</classname> のインスタンスを作成するには、
- コンストラクタでアダプタを指定しなければなりません。
- </para>
- <programlisting language="php"><![CDATA[
- $paginator = new Zend_Paginator(new Zend_Paginator_Adapter_Array($array));
- ]]></programlisting>
- <para>
- 利便性を確保するために、Zend Framework に同梱されているアダプタ用の静的メソッド
- <methodname>factory()</methodname> も用意されています。
- </para>
- <programlisting language="php"><![CDATA[
- $paginator = Zend_Paginator::factory($array);
- ]]></programlisting>
- <note>
- <para>
- <classname>Null</classname> アダプタの場合は、
- データコレクションのかわりに要素数をコンストラクタで指定します。
- </para>
- </note>
- <para>
- この状態でも技術的には既に使用可能ですが、
- ユーザが要求したページ番号をコントローラのアクション内で教えてやる必要があります。
- これにより、データを読み進めていくことが可能となります。
- </para>
- <programlisting language="php"><![CDATA[
- $paginator->setCurrentPageNumber($page);
- ]]></programlisting>
- <para>
- ページ番号は <acronym>URL</acronym> で指定するのがもっともシンプルな方法でしょう。
- <classname>Zend_Controller_Router_Interface</classname>
- 互換のルータを使うことを推奨しますが、
- それが必須というわけではありません。
- </para>
- <para>
- <acronym>INI</acronym> ファイルで定義するルートの例を次に示します。
- </para>
- <programlisting language="php"><![CDATA[
- routes.example.route = articles/:articleName/:page
- routes.example.defaults.controller = articles
- routes.example.defaults.action = view
- routes.example.defaults.page = 1
- routes.example.reqs.articleName = \w+
- routes.example.reqs.page = \d+
- ]]></programlisting>
- <para>
- この設定を使った (そして Zend Framework の <acronym>MVC</acronym> コンポーネントを使った)
- 場合、現在のページ番号を設定するコードはこのようになります。
- </para>
- <programlisting language="php"><![CDATA[
- $paginator->setCurrentPageNumber($this->_getParam('page'));
- ]]></programlisting>
- <para>
- それ以外にもオプションがあります。詳細は
- <link linkend="zend.paginator.configuration">設定</link>
- を参照ください。
- </para>
- <para>
- 最後に、paginator のインスタンスをビューに割り当てます。
- <classname>Zend_View</classname> と ViewRenderer アクションヘルパーを使っている場合は、
- 次のようになります。
- </para>
- <programlisting language="php"><![CDATA[
- $this->view->paginator = $paginator;
- ]]></programlisting>
- </sect2>
- <sect2 id="zend.paginator.usage.dbselect">
- <title>DbSelect および DbTableSelect アダプタ</title>
- <para>
- 大半のアダプタの使用法は非常にわかりやすいものです。
- しかし、データベースアダプタについては、
- データベースからのデータの取得方法や件数の数え方についてのより詳細な説明が必要です。
- </para>
- <para>
- DbSelect アダプタや DbTableSelect アダプタを使う際には、
- 事前にデータベースからデータを取得する必要はありません。
- どちらのアダプタも、自動的にデータを取得して総ページ数を計算します。
- もしデータベースからのデータに対して何らかの処理が必要となるのなら、
- アダプタの <methodname>getItems()</methodname> メソッドをアプリケーション内で継承します。
- </para>
- <para>
- さらに、これらのアダプタは、
- 数を数える際にデータベースの全レコードを取得するわけでは
- <emphasis>ありません</emphasis>。そのかわりに、アダプタが元のクエリを修正し、
- それに対応する COUNT クエリを作成します。
- Paginator は、その COUNT クエリを実行して行数を取得するのです。
- そのぶんデータベースとの余分なやりとりが必要となりますが、結果セット全体を取得して
- <methodname>count()</methodname> を使うよりは何倍も高速になります。
- 大量のデータを扱う場合などは特にそうです。
- </para>
- <para>
- データベースアダプタは、すべてのモダンなデータベース上で実行できる
- もっとも効率的なクエリを作成しようとします。
- しかし、使用するデータベースやスキーマ設定によっては
- 行数を取得するのにもっと効率的な方法があるかもしれません。
- そのような場合のために、データベースアダプタでは独自の COUNT
- クエリを設定できるようにもなっています。たとえば、
- 別々のテーブルにある blog の投稿の数を調べるには、
- 次の用に設定すればより高速な count クエリが使用できるでしょう。
- </para>
- <programlisting language="php"><![CDATA[
- $adapter = new Zend_Paginator_Adapter_DbSelect($db->select()->from('posts'));
- $adapter->setRowCount(
- $db->select()
- ->from(
- 'item_counts',
- array(
- Zend_Paginator_Adapter_DbSelect::ROW_COUNT_COLUMN => 'post_count'
- )
- )
- );
- $paginator = new Zend_Paginator($adapter);
- ]]></programlisting>
- <para>
- この方法は、小規模なデータや単純な select クエリの場合にはあまり劇的な効果を得られません。
- しかし、複雑なクエリや大規模なデータを扱う場合は
- かなりパフォーマンスが向上することでしょう。
- </para>
- </sect2>
- <sect2 id="zend.paginator.rendering">
- <title>ビュースクリプトによるページのレンダリング</title>
- <para>
- ビュースクリプトを使用してページ項目のレンダリング
- (<classname>Zend_Paginator</classname> を使うよう設定している場合)
- とページ処理コントロールの表示を行います。
- </para>
- <para>
- <classname>Zend_Paginator</classname> は <acronym>SPL</acronym> の
- <ulink url="http://www.php.net/~helly/php/ext/spl/interfaceIteratorAggregate.html"><classname>IteratorAggregate</classname></ulink>
- インターフェイスを実装しているので、
- 項目を順次処理したり表示したりするのは簡単です。
- </para>
- <programlisting language="php"><![CDATA[
- <html>
- <body>
- <h1>Example</h1>
- <?php if (count($this->paginator)): ?>
- <ul>
- <?php foreach ($this->paginator as $item): ?>
- <li><?php echo $item; ?></li>
- <?php endforeach; ?>
- </ul>
- <?php endif; ?>
- <?php echo $this->paginationControl($this->paginator,
- 'Sliding',
- 'my_pagination_control.phtml'); ?>
- </body>
- </html>
- ]]></programlisting>
- <para>
- 最後のほうでビューヘルパーをコールしているところに注目しましょう。
- PaginationControl 4 つまでのパラメータを受け取ります。
- paginator のインスタンス、スクロール形式、
- そして追加パラメータの配列です。
- </para>
- <para>
- 2 番目と 3 番目のパラメータは重要です。
- ビュー partial はページ処理コントロールの
- <emphasis>見た目</emphasis>を決めるために用いられ、
- 一方スクロール形式はその <emphasis>振る舞い</emphasis>
- を決めるために用いられます。ビュー partial
- が、次のようなページ処理コントロール形式だっととしましょう。
- </para>
- <para>
- <inlinegraphic align="center" valign="middle"
- fileref="figures/zend.paginator.usage.rendering.control.png"
- format="PNG"/>
- </para>
- <para>
- ここで "next" リンクを数回クリックしたときに、いったい何が起こるでしょう?
- そう、いろんなことが起こりえます。
- クリックし続けても現在のページがずっと中央に表示される (Yahoo!
- 形式) かもしれませんし、
- 表示される範囲はそのままで現在のページの位置がどんどん右にずれていき、
- 表示範囲の最後をページでさらに "next" をクリックしたときに一番左に戻るかもしれません。
- ページを進めるたびにページ数そのものが増加 ("scroll")
- していく (Google 形式) も考えられます。
- </para>
- <para>
- 4 種類のスクロール形式が Zend Framework に組み込まれています。
- </para>
- <table id="zend.paginator.usage.rendering.scrolling-styles">
- <title>Zend_Paginator のスクロール形式</title>
- <tgroup cols="2">
- <thead>
- <row>
- <entry>スクロール形式</entry>
- <entry>説明</entry>
- </row>
- </thead>
- <tbody>
- <row>
- <entry>All</entry>
- <entry>
- すべてのページを返します。
- 総ページ数が比較的少なめのときなど、
- ドロップダウンメニュー形式でページ選択をさせる際に便利です。
- そのような場合は、利用できるすべてのページを
- 一度にユーザに見せることになるでしょう。
- </entry>
- </row>
- <row>
- <entry>Elastic</entry>
- <entry>
- Google 風のスクロール形式で、
- ユーザがページを移動するのにあわせて拡大・縮小します。
- </entry>
- </row>
- <row>
- <entry>Jumping</entry>
- <entry>
- ユーザがページを進めるにつれて、
- ページ番号が表示範囲の最後に向けて進んでいきます。
- 表示範囲を超えると、新しい範囲の最初の位置に移動します。
- </entry>
- </row>
- <row>
- <entry>Sliding</entry>
- <entry>
- Yahoo! 風のスクロール形式で、
- 現在表示されているページが常にページ範囲の中央
- (あるいは可能な限りそれに近い場所)
- にあるようにします。これがデフォルトの形式です。
- </entry>
- </row>
- </tbody>
- </tgroup>
- </table>
- <para>
- 4 番目、そして最後のパラメータはオプションの連想配列です。
- ここで、ビューパーシャルから (<varname>$this</varname> を用いて)
- 使用したい追加変数を指定します。
- たとえば、ページ移動用のリンクに使用する追加の
- <acronym>URL</acronym> パラメータなどを含めることができます。
- </para>
- <para>
- デフォルトのビュー partial とスクロール形式、
- そしてビューのインスタンスを設定してしまえば、
- PaginationControl のコールを完全に除去できます。
- </para>
- <programlisting language="php"><![CDATA[
- Zend_Paginator::setDefaultScrollingStyle('Sliding');
- Zend_View_Helper_PaginationControl::setDefaultViewPartial(
- 'my_pagination_control.phtml'
- );
- $paginator->setView($view);
- ]]></programlisting>
- <para>
- これらの値をすべて設定すると、
- ビュースクリプト内で単純な echo
- 文を使用するだけでページ処理コントロールをレンダリングできるようになります。
- </para>
- <programlisting language="php"><![CDATA[
- <?php echo $this->paginator; ?>
- ]]></programlisting>
- <note>
- <para>
- もちろん、<classname>Zend_Paginator</classname>
- を別のテンプレートエンジンで使用することもできます。
- たとえば、Smarty を使用する場合は次のようになります。
- </para>
- <programlisting language="php"><![CDATA[
- $smarty->assign('pages', $paginator->getPages());
- ]]></programlisting>
- <para>
- そして、テンプレートからは次のようにして paginator の値にアクセスします。
- </para>
- <programlisting language="php"><![CDATA[
- {$pages->pageCount}
- ]]></programlisting>
- </note>
- <sect3 id="zend.paginator.usage.rendering.example-controls">
- <title>ページ処理コントロールの例</title>
- <para>
- 次のページ処理コントロールの例が、
- とりあえず使い始めるにあたっての参考となることでしょう。
- </para>
- <para>
- 検索のページ処理
- </para>
- <programlisting language="php"><![CDATA[
- <!--
- See http://developer.yahoo.com/ypatterns/pattern.php?pattern=searchpagination
- -->
- <?php if ($this->pageCount): ?>
- <div class="paginationControl">
- <!-- 前のページへのリンク -->
- <?php if (isset($this->previous)): ?>
- <a href="<?php echo $this->url(array('page' => $this->previous)); ?>">
- < Previous
- </a> |
- <?php else: ?>
- <span class="disabled">< Previous</span> |
- <?php endif; ?>
- <!-- ページ番号へのリンク -->
- <?php foreach ($this->pagesInRange as $page): ?>
- <?php if ($page != $this->current): ?>
- <a href="<?php echo $this->url(array('page' => $page)); ?>">
- <?php echo $page; ?>
- </a> |
- <?php else: ?>
- <?php echo $page; ?> |
- <?php endif; ?>
- <?php endforeach; ?>
- <!-- 次のページへのリンク -->
- <?php if (isset($this->next)): ?>
- <a href="<?php echo $this->url(array('page' => $this->next)); ?>">
- Next >
- </a>
- <?php else: ?>
- <span class="disabled">Next ></span>
- <?php endif; ?>
- </div>
- <?php endif; ?>
- ]]></programlisting>
- <para>
- 項目のページ処理
- </para>
- <programlisting language="php"><![CDATA[
- <!--
- See http://developer.yahoo.com/ypatterns/pattern.php?pattern=itempagination
- -->
- <?php if ($this->pageCount): ?>
- <div class="paginationControl">
- <?php echo $this->firstItemNumber; ?> - <?php echo $this->lastItemNumber; ?>
- of <?php echo $this->totalItemCount; ?>
- <!-- 最初のページへのリンク -->
- <?php if (isset($this->previous)): ?>
- <a href="<?php echo $this->url(array('page' => $this->first)); ?>">
- First
- </a> |
- <?php else: ?>
- <span class="disabled">First</span> |
- <?php endif; ?>
- <!-- 前のページへのリンク -->
- <?php if (isset($this->previous)): ?>
- <a href="<?php echo $this->url(array('page' => $this->previous)); ?>">
- < Previous
- </a> |
- <?php else: ?>
- <span class="disabled">< Previous</span> |
- <?php endif; ?>
- <!-- 次のページへのリンク -->
- <?php if (isset($this->next)): ?>
- <a href="<?php echo $this->url(array('page' => $this->next)); ?>">
- Next >
- </a> |
- <?php else: ?>
- <span class="disabled">Next ></span> |
- <?php endif; ?>
- <!-- 最後のページへのリンク -->
- <?php if (isset($this->next)): ?>
- <a href="<?php echo $this->url(array('page' => $this->last)); ?>">
- Last
- </a>
- <?php else: ?>
- <span class="disabled">Last</span>
- <?php endif; ?>
- </div>
- <?php endif; ?>
- ]]></programlisting>
- <para>
- ドロップダウンのページ処理
- </para>
- <programlisting language="php"><![CDATA[
- <?php if ($this->pageCount): ?>
- <select id="paginationControl" size="1">
- <?php foreach ($this->pagesInRange as $page): ?>
- <?php $selected = ($page == $this->current) ? ' selected="selected"' : ''; ?>
- <option value="<?php
- echo $this->url(array('page' => $page));?>"<?php echo $selected ?>>
- <?php echo $page; ?>
- </option>
- <?php endforeach; ?>
- </select>
- <?php endif; ?>
- <script type="text/javascript"
- src="http://ajax.googleapis.com/ajax/libs/prototype/1.6.0.2/prototype.js">
- </script>
- <script type="text/javascript">
- $('paginationControl').observe('change', function() {
- window.location = this.options[this.selectedIndex].value;
- })
- </script>
- ]]></programlisting>
- </sect3>
- <sect3 id="zend.paginator.usage.rendering.properties">
- <title>プロパティの一覧</title>
- <para>
- 次のオプションが、ページ処理コントロールのビュー
- partial で使用可能です。
- </para>
- <table id="zend.paginator.usage.rendering.properties.table">
- <title>ビュー partial のプロパティ</title>
- <tgroup cols="3">
- <thead>
- <row>
- <entry>プロパティ</entry>
- <entry>型</entry>
- <entry>説明</entry>
- </row>
- </thead>
- <tbody>
- <row>
- <entry>first</entry>
- <entry>integer</entry>
- <entry>最初のページ番号 (つまり 1)</entry>
- </row>
- <row>
- <entry>firstItemNumber</entry>
- <entry>integer</entry>
- <entry>
- このページの最初の項目の番号
- </entry>
- </row>
- <row>
- <entry>firstPageInRange</entry>
- <entry>integer</entry>
- <entry>
- スクロール形式で返された範囲内の最初のページ
- </entry>
- </row>
- <row>
- <entry>current</entry>
- <entry>integer</entry>
- <entry>現在のページ番号</entry>
- </row>
- <row>
- <entry>currentItemCount</entry>
- <entry>integer</entry>
- <entry>このページの項目の数</entry>
- </row>
- <row>
- <entry>itemCountPerPage</entry>
- <entry>integer</entry>
- <entry>各ページに表示できる項目の最大数</entry>
- </row>
- <row>
- <entry>last</entry>
- <entry>integer</entry>
- <entry>最後のページ番号</entry>
- </row>
- <row>
- <entry>lastItemNumber</entry>
- <entry>integer</entry>
- <entry>
- このページの最後の項目の番号
- </entry>
- </row>
- <row>
- <entry>lastPageInRange</entry>
- <entry>integer</entry>
- <entry>
- スクロール形式で返された範囲内の最後のページ
- </entry>
- </row>
- <row>
- <entry>next</entry>
- <entry>integer</entry>
- <entry>次のページ番号</entry>
- </row>
- <row>
- <entry>pageCount</entry>
- <entry>integer</entry>
- <entry>ページ数</entry>
- </row>
- <row>
- <entry>pagesInRange</entry>
- <entry>array</entry>
- <entry>
- スクロール形式で返されたページの配列
- </entry>
- </row>
- <row>
- <entry>previous</entry>
- <entry>integer</entry>
- <entry>前のページ番号</entry>
- </row>
- <row>
- <entry>totalItemCount</entry>
- <entry>integer</entry>
- <entry>項目の総数</entry>
- </row>
- </tbody>
- </tgroup>
- </table>
- </sect3>
- </sect2>
- </sect1>
- <!--
- vim:se ts=4 sw=4 et:
- -->
|