並べ替え

複数のデータ行を表示する際に、エンド・ユーザによって指定されるカラムに従ってデータを並べ替えなければならないことがよくあります。 Yii は yii\data\Sort オブジェクトを使って並べ替えのスキーマに関する情報を表します。 具体的に言えば、

  • attributes データの並べ替えに使用できる 属性 を指定します。 単純で良ければ、モデルの属性 をこの属性とすることが出来ます。 また、複数のモデル属性や DB のカラムを結合した合成的な属性を指定することも出来ます。詳細については後述します。
  • yii\data\Sort::$attributeOrders 各属性について、 現在リクエストされている並べ替えの方向を指定します。
  • yii\data\Sort::$orders 並べ替えの方向をカラムを使う低レベルな形式で示します。

yii\data\Sort を使用するためには、最初にどの属性が並べ替え可能であるかを宣言します。 次に、現在リクエストされている並べ替え情報を yii\data\Sort::$attributeOrders または yii\data\Sort::$orders から取得して、データのクエリをカスタマイズします。例えば、

use yii\data\Sort;

$sort = new Sort([
    'attributes' => [
        'age',
        'name' => [
            'asc' => ['first_name' => SORT_ASC, 'last_name' => SORT_ASC],
            'desc' => ['first_name' => SORT_DESC, 'last_name' => SORT_DESC],
            'default' => SORT_DESC,
            'label' => '氏名',
        ],
    ],
]);

$articles = Article::find()
    ->where(['status' => 1])
    ->orderBy($sort->orders)
    ->all();

上記の例では、Sort オブジェクトに対して二つの属性が宣言されています。すなわち、agename です。

age 属性は Article アクティブ・レコード・クラスの age 属性に対応する 単純な 属性です。 これは、次の宣言と等価です。

'age' => [
    'asc' => ['age' => SORT_ASC],
    'desc' => ['age' => SORT_DESC],
    'default' => SORT_ASC,
    'label' => Inflector::camel2words('age'),
]

name 属性は Articlefirst_namelast_name によって定義される 合成的な 属性です。 これは次のような配列構造を使って宣言されています。

  • asc および desc の要素は、それぞれ、この属性を昇順および降順に並べ替える方法を指定します。 この値が、データの並べ替えに使用されるべき実際のカラムと方向を表します。 一つまたは複数のカラムを指定して、単純な並べ替えや合成的な並べ替えを示すことが出来ます。
  • default 要素は、最初にリクエストされたときの属性の並べ替えに使用されるべき方向を指定します。 デフォルト値は昇順です。 つまり、以前に並べ替えられたことがない状態でこの属性による並べ替えをリクエストすると、この属性の昇順に従ってデータが並べ替えられることになります。
  • label 要素は、並べ替えのリンクを作成するために yii\data\Sort::link() を呼んだときに、どういうラベルを使用すべきかを指定するものです。 設定されていない場合は、yii\helpers\Inflector::camel2words() が呼ばれて、属性名からラベルが生成されます。 ラベルは HTML エンコードされないことに注意してください。

Info: yii\data\Sort::$orders の値をデータベースのクエリに直接に供給して、 ORDER BY 句を構築することが出来ます。 データベースのクエリが認識できない合成的な属性が入っている場合があるため、 yii\data\Sort::$attributeOrders を使ってはいけません。

yii\data\Sort::link() を呼んでハイパーリンクを生成すれば、それをクリックして、指定した属性によるデータの並べ替えをリクエストすることが出来るようになります。 yii\data\Sort::createUrl() を呼んで並べ替えを実行する URL を生成することも出来ます。 例えば、

// 生成される URL が使用すべきルートを指定する
// これを指定しない場合は、現在リクエストされているルートが使用される
$sort->route = 'article/index';

// 氏名による並べ替えと年齢による並べ替えを実行するリンクを表示
echo $sort->link('name') . ' | ' . $sort->link('age');

// /index.php?r=article%2Findex&sort=age を表示
echo $sort->createUrl('age');

yii\data\Sort は、リクエストの sort クエリ・パラメータをチェックして、どの属性による並べ替えがリクエストされたかを判断します。 このクエリ・パラメータが存在しない場合のデフォルトの並べ替え方法は yii\data\Sort::$defaultOrder によって指定することが出来ます。 また、sortParam プロパティを構成して、このクエリ・パラメータの名前をカスタマイズすることも出来ます。