3

我为我们的支持者制作了一个支持票系统 - 使用 ruby​​ on rails 编程(Ruby 1.9.3 Rails 3.2)

有一个belongs_to与用户(支持者)相关联的票模型。

我使用 Ernie 的宝石 Ransack 进行搜索。当支持者搜索他自己的门票(由tickets_controller.index 处理)时,他还应该在搜索结果(views/tickets/index)中看到未关联的门票(结合其他搜索条件,例如“日期”或其他内容)

(这样他就可以拿开票)

预期的 SQL 语句:

SELECT * FROM tickets WHERE ... AND (user_id=5 OR user_id IS NULL) AND ... 

我在索引视图的搜索表单中尝试过user_id_null_or_user_id_eq-但这不起作用(没有方法错误)

如何制作自定义谓词(或 ransacker)来解决此问题?(感谢@Magnuss 的评论)

不幸的是,我没有使用 Arel 的经验

就像是:

ransacker :user_null_or_eq do
    ...
end

在门票模型中。

我希望这能更详细地解释问题。

谢谢你的帮助。

4

3 回答 3

5

我也必须弄清楚这一点,@TomG 的回答很有帮助!你真的不需要为这个问题创建一个 ransacker,它可以通过使用 ransack 的 build_grouping 方法来修复(在任何地方都没有记录......)并将你需要的条件放入 'or' 条件。

我用它来搜索要添加到笼子的项目,但想只向用户显示我的笼子中还没有的项目(也没有分配给任何其他笼子),所以在我的控制器中:

   @search = Cage.ransack(params[:q])
   @search.build_grouping({:m => 'or', :storage_id_not_eq => @cage_item.id, :storage_id_null => true})  
   @results_count = @search.result(:distinct => true).length
   @items = @search.result(:distinct => true).paginate(:per_page => 20, :page => params[:page] || 1)

   @search.build_condition if @search.conditions.empty?
   @search.build_sort if @search.sorts.empty?

因此,当我打印查询 ransack 正在运行时,它将显示:

 @search.result.to_sql
"SELECT `specimens`.* FROM `specimens`  WHERE (`specimens`.`deleted_at` IS NULL) AND ((`specimens`.`storage_id` != 183 OR `specimens`.`storage_id` IS NULL))"

如果我添加一个名称查询:

@search.result.to_sql
"SELECT `specimens`.* FROM `specimens`  WHERE (`specimens`.`deleted_at` IS NULL) AND ((`specimens`.`name` LIKE '%root%' AND (`specimens`.`storage_id` != 183 OR `specimens`.`storage_id` IS NULL)))"

希望这可以帮助您或其他人在搜索条件下使用 OR 而不是基本 AND 时遇到问题。

于 2014-01-01T12:36:59.567 回答
1

好的 - 这是我自己的解决方案 - 灵感来自 graywh(非常感谢):ransack/issues/290

您可以将带有“或”组合器的新分组节点添加到 ransack 对象。

在索引视图中,我更改f.select :user_id_eqselect_tag ...

门票控制器.index:

@q = Ticket.search(params[:q])  
if params[:user_id_eq] && params[:user_id_eq]!=''  
  @q.build_grouping({:m => 'or', :user_id_eq => params[:user_id_eq], :user_id_null => true})  
end
于 2013-10-21T06:41:46.343 回答
1

您可以像这样构建查询参数

q: {g: [ {m: 'or', user_id_eq: 5, user_id_null: 1} ] }

这在哪里:

user_id_null: 1

只是用于搜索空值的标志

于 2016-02-09T10:31:42.047 回答