1

希望我在这里遗漏了一些简单的东西。我正在为我的蛋糕 (2.3.4) 应用程序使用 CakeDC 搜索和标签插件。

除了按字段进行通用搜索外,我还希望用户能够按标签进行搜索。我几乎已经完成了这项工作,但是如果您搜索单个标签而不是多个标签,则搜索只会显示结果。例如,如果我添加带有以下标签的文章 - 黑色、白色、红色。如果我搜索单个标签(例如黑色)而不是全部 3 个,甚至 2 个标签,该文章只会显示在搜索结果中......我错过了什么?

这是我的代码:

文章.php模型

class Article extends AppModel {
public $actsAs = array(
    'Upload.Upload' => array(
        'screenshot1' => array (
            'fields' => array (
                'dir' => 'dir'
            ),
            'thumbnailMethod' => 'php',
            'thumbnailSizes' => array(
                'xvga' => '1024x768',
                'vga' => '640x480',
                'thumb' => '80x80'
            ),
        ),
    ),
'Search.Searchable',
'Tags.Taggable'

);

// Search plugin filters
public $filterArgs = array(      
    'title' => array('type' => 'like'),
    'environment' => array('type' => 'like'),
    'description' => array('type' => 'like'),
    'error' => array('type' => 'like'),
    'cause' => array('type' => 'like'),
    'resolution' => array('type' => 'like'),
    'live' => array('type' => 'value'),
    'synced' => array('type' => 'value'),
    'tags' => array('type' => 'subquery', 'method' => 'findByTags', 'field' => 'Article.id'),
    array('name' => 'search', 'type' => 'query', 'method' => 'filterQuery'),
);

// This is the OR query that runs when the user uses the client side signle field search
public function filterQuery($data = array()) {
    if(empty($data['search'])) { // search is the name of my search field
        return array();
    }

    $query = '%'.$data['search'].'%';
    return array(
        'OR' => array(
            'Article.title LIKE' => $query,
            'Article.description LIKE' => $query,
            'Article.error LIKE' => $query,
            )
        );
}

// Find by tags method
public function findByTags($data = array()) {
    $this->Tagged->Behaviors->attach('Containable', array('autoFields' => false));
    $this->Tagged->Behaviors->attach('Search.Searchable');
    $query = $this->Tagged->getQuery('all', array(
        'conditions' => array('Tag.name'  => $data['tags']),
        'fields' => array('foreign_key'),
        'contain' => array('Tag')
    ));
    return $query;
}

public $hasAndBelongsToMany = array(
    'Category' => array(
        'className' => 'Category',
        'joinTable' => 'articles_categories',
        'foreignKey' => 'article_id',
        'associationForeignKey' => 'category_id',
        'unique' => 'keepExisting',
        'conditions' => '',
        'fields' => '',
        'order' => '',
        'limit' => '',
        'offset' => '',
        'finderQuery' => '',
        'deleteQuery' => '',
        'insertQuery' => ''
    ),
    'Tag' => array('with' => 'Tagged')
);

控制器方法

public function admin_advancedSearch() {

        // Disable validation for this action
        $this->Article->validate = array();

        // For search plugin
        $this->Prg->commonProcess();

        // Set searched for details
        $this->set('searchedFor', $this->passedArgs);

        // If passed args are all empty
        if ($this->passedArgs) {

            if (empty($this->passedArgs['title']) AND
                empty($this->passedArgs['environment']) AND
                empty($this->passedArgs['description']) AND
                empty($this->passedArgs['error']) AND
                empty($this->passedArgs['cause']) AND
                empty($this->passedArgs['resolution']) AND
                empty($this->passedArgs['live']) AND
                empty($this->passedArgs['synced']) AND
                empty($this->passedArgs['tags'])) {

                    $this->Session->setFlash('Please enter at least one search criteria', 'flash_failure');
                    // Set this var for checks in view
                    $this->set('emptySeach', true);
            } else {

                $paginateConditions = $this->Article->parseCriteria($this->passedArgs);
                $this->paginate = array('conditions' => array(
                                            $paginateConditions),
                                        'limit' => 10,
                                        'order' => array('Article.modified' => 'DESC'));            
                $this->set('articles', $this->paginate());

                // Count number of results
                $count = 0;
                foreach ($this->paginate() as $result) {
                    $count++;
                }

                $this->set('resultsCount', $count);

                // Search was not empty - set flag for view
                $this->set('emptySeach', false);
            }
        }

        // Set layout
        $this->layout = 'admin';

        //debug($this->passedArgs);
    }

所有插件都从引导程序成功加载,搜索工作正常,没有标签。仅当输入一个标签时,带有标签的搜索才有效....

*编辑 * 如果我从 findByTags 方法调试 $query 我得到这个:

'SELECT `Tagged`.`foreign_key` FROM `knowledgebase`.`tagged` AS `Tagged` LEFT JOIN `knowledgebase`.`tags` AS `Tag` ON (`Tagged`.`tag_id` = `Tag`.`id`)  WHERE `Tag`.`name` = 'kia, rio, red''

所以看起来它试图找到一个标签,其名称中包含所有搜索过的标签。如何使 WHERE 部分成为 IN?

例如:

WHERE `Tag`.`name` IN ('kia', 'rio', 'red')

谢谢

4

1 回答 1

1

使用“方法” https://github.com/cakedc/search#behavior-and-model-configuration键将搜索参数传递给自定义方法。

这里的示例向您展示了它是如何工作的https://github.com/cakedc/search#full-example-for-modelcontroller-configuration-with-overriding

public $filterArgs = array(
    'some_related_table_id' => array('type' => 'value'),
    'search'=> array('type' => 'like', 'encode' => true, 'before' => false, 'after' => false, 'field' => array('ThisModel.name', 'OtherModel.name')),
    'name'=> array('type' => 'query', 'method' => 'searchNameCondition')
);

public function searchNameCondition($data = array()) {
    $filter = $data['name'];
    $cond = array(
        'OR' => array(
            $this->alias . '.name LIKE' => '' . $this->formatLike($filter) . '',
            $this->alias . '.invoice_number LIKE' => '' . $this->formatLike($filter) . '',
    ));
    return $cond;
}

在您的自定义方法中explode() 标记并使其成为一个数组,以便CakePHP 将它们用作IN() 或更好(in 可能会变慢)使其成为AND 或OR 链。

于 2013-05-30T08:15:21.547 回答