質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

91.80%

  • PHP

    13276questions

    PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

  • CakePHP

    1767questions

    CakePHPは、PHPで書かれたWebアプリケーション開発用のフレームワークです。 Ruby on Railsの考え方を多く取り入れており、Railsの高速性とPHPの機動性を兼ね備えています。 MVCやORMなどを「規約優先の考え方」で利用するため、コードを書く手間を省くことができます。 外部のライブラリに依存しないので、単体での利用が可能です。

  • Framework

    70questions

    Frameworkは、アプリケーションソフトを開発する際に、一般的な機能をより簡単に、より早く完了させる事を目的とした、ソフトウェアやライブラリのセットを指します。開発にフレームワークを使用する事で、追加で必要となる機能だけを開発するだけで済む為、開発効率の向上が見込めます。

cakephp3、FriendsOfCake/Searchプラグインで、concatで文字結合したフィールドを検索したい

解決済

回答 2

投稿 2017/01/27 13:53 ・編集 2017/01/27 17:52

  • 評価
  • クリップ 0
  • VIEW 761

maifukuhara

score 65

cakephp3で顧客テーブルをfriendsofcake/Searchで検索する部分を開発しています。
このテーブルで、住所1、2を文字列結合して検索したいです。
イメージ説明

最終的に発行したいSQLは下記の様になるかと思います。

select * from customers where concat(address_1, address_2) LIKE '%words(検索文字列)%'

cakephp2.xでは、仮想フィールド(virtualfield) を使い
「full_address」等の名前でconcatしたフィールドを設定しておけば
検索クエリでも、そのまま使えたらしいのですが、

cakephp3.xからは、下記理由のためできなくなったようです。
http://palmgate.co.jp/masato_kato/article.html&id=164

他画面を「firiendsofcake/searchプラグイン」を使って
開発を進めてきたため、できればプラグインを使ったまま対応したいです。

下記の記事を見つけましたが、cakephp3そのものの
経験が浅く、思うようなクエリを発行してくれません
http://stackoverflow.com/questions/37333873/concat-fields-in-search-plugin-cakephp-3

アドバイスをお願いできないでしょうか。
よろしくお願いします。

↓ C:\xampp\htdocs\cake3\src\Template\Customers\index.ctp

echo $this->Form->create();
echo $this->Form->input('id');   //idを入力
echo $this->Form->input('words'); //住所を入力
echo $this->Form->button("送信", ['type' => 'submit']);
echo $this->Form->end();
debug($customers);

↓ C:\xampp\htdocs\cake3\src\Model\Table\CustomersTable.php

use Search\Manager;   //Searchプラグイン用 

class CustomersTable extends Table
{
    public function initialize(array $config)
    {
      parent::initialize($config);

       /* Searchプラグイン用 */ 
       $this->addBehavior('Search.Search');
       $this->searchManager()
          ->value('id')
          ->add('words', 'Search.Like', [
                'before' => true,
                'after' => true,
                'fieldMode' => 'OR',
                'comparison' => 'LIKE',
                'wildcardAny' => '*',
                'wildcardOne' => '?',

        //試したこと1 仮想フィールドを指定・・・ダメ
        //         'field' => 'full_address'

        //試したこと2 concatを直接書く・・・ダメ
        // こんなSQLになる→   WHERE   Customers.concat(address_1, address_2) like '%words%' 
        //         'field' => 'concat(address_1, address_2)'
        //    ]);

//試したこと3 下記を見て、コールバックを設定してみる
//http://stackoverflow.com/questions/37333873/concat-fields-in-search-plugin-cakephp-3
    ->add('word', 'Search.Callback', [
    'callback' => function ($query, $args, $manager) {
    return $query->select([
                   'full_address' => $query->func()->concat([
                    'address_1' => 'identifier',
                    'address_2' => 'identifier'
                     ])
                  ])
                  ->having(['full_address LIKE' => '%' . $args['words'] . '%']);
      }
        ]);
:
以下略

仮想フィールドは上手くいきませんでしたが、こんなかんじで書きました。
↓C:\xampp\htdocs\cake3\src\Model\Entity\Customer.php

class Customer extends Entity
{
    protected $_virtual = [
        'full_address',
    ];

    protected function _getFullAddress()
    {
        return $this->_properties['address_1'].
               $this->_properties['address_2'];
    }
  :
以下略

       

↓C:\xampp\htdocs\cake3\src\src\Controller\CustomersController.php

class CustomersController extends AppController
{
     public function initialize()
     {
       parent::initialize();
       //サーチプラグインの読み込み
       $this->loadComponent('Search.Prg', [
         'actions' => ['index']
       ]);
     }

    public function index()
    {
    $selected = $this->Customers
      ->find('search', ['search' => $this->request->query])
    $this->set('customers', $this->paginate($selected));
  :
以下略

■参考にしたもの
http://stackoverflow.com/questions/37333873/concat-fields-in-search-plugin-cakephp-3
http://www.naidim.org/cakephp-3-tutorial-11-virtual-fields

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • icchii

    2017/01/27 17:40

    SerchではなくSearchですね...

    キャンセル

  • maifukuhara

    2017/01/27 17:54

    あ、ほんとだ。誤字でしたね。

    キャンセル

回答 2

checkベストアンサー

+1

以下のような感じでしょうか。callbackを使って実現しています。

<?php
namespace App\Model\Table;

use Cake\ORM\Table;
use Search\Manager; //Searchプラグイン用

class CustomersTable extends Table
{
    public function initialize(array $config)
    {
        parent::initialize($config);

        /* Searchプラグイン用 */
        $this->addBehavior('Search.Search');
        $this->searchManager()
        ->value('id')
        ->add('words', 'Search.Callback', [
            'callback' => function ($query, $args, $manager) {
                return $query->where(function ($exp, $query) use ($args) {
                    $address = $query->func()->concat([
                        'address_1' => 'identifier',
                        'address_2' => 'identifier'
                    ]);
                    return $exp->like($address, '%' . $args['words'] . '%');
                });
            },
        ]);
    }
}

投稿 2017/01/27 17:33

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    以下のような回答は評価を下げられます

    • 間違っている回答
    • 質問の回答になっていない投稿
    • 不快な投稿

    評価を下げる際はその理由をコメントに書き込んでください。

  • 2017/01/27 19:06

    icchiiさん、ありがとうございます!
    おかげで思った通りのSQLが発行されました。
    ここでずっとハマってたのですごく助かりました。

    キャンセル

0

icchiiさんに教えて頂いた方法でできました。
カナ住所も一緒に検索したくなり、
最終的には下記のようになりました。

カナ住所も含めて

<?php
namespace App\Model\Table;

use Cake\ORM\Table;
use Search\Manager; //Searchプラグイン用

class CustomersTable extends Table
{
    public function initialize(array $config)
    {
        parent::initialize($config);

        /* Searchプラグイン用 */
        $this->addBehavior('Search.Search');
        $this->searchManager()
        ->value('id')
          ->add('words', 'Search.Callback', [
            'callback' => function ($query, $args, $manager) {
               return $query->where(function ($exp, $query) use ($args) {
                      $fulladdress = $query->func()->concat([
                          'customers.address_1' => 'identifier',
                          'customers.address_2' => 'identifier',
                          'customers.address_1_kana' => 'identifier',
                          'customers.address_2_kana' => 'identifier'
                      ]);
                      return $exp->like($fulladdress, '%' . $args['words'] . '%');
                  });
              }
          ]);
    }
}

投稿 2017/01/27 19:11

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    以下のような回答は評価を下げられます

    • 間違っている回答
    • 質問の回答になっていない投稿
    • 不快な投稿

    評価を下げる際はその理由をコメントに書き込んでください。

15分調べてもわからないことは、teratailで質問しよう!

ただいまの回答率

91.80%

関連した質問

同じタグがついた質問を見る

  • PHP

    13276questions

    PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

  • CakePHP

    1767questions

    CakePHPは、PHPで書かれたWebアプリケーション開発用のフレームワークです。 Ruby on Railsの考え方を多く取り入れており、Railsの高速性とPHPの機動性を兼ね備えています。 MVCやORMなどを「規約優先の考え方」で利用するため、コードを書く手間を省くことができます。 外部のライブラリに依存しないので、単体での利用が可能です。

  • Framework

    70questions

    Frameworkは、アプリケーションソフトを開発する際に、一般的な機能をより簡単に、より早く完了させる事を目的とした、ソフトウェアやライブラリのセットを指します。開発にフレームワークを使用する事で、追加で必要となる機能だけを開発するだけで済む為、開発効率の向上が見込めます。

閲覧数の多いPHPの質問