【问题标题】:Storing array data in jsonb field with Rails and Postgresql使用 Rails 和 Postgresql 在 jsonb 字段中存储数组数据
【发布时间】:2016-10-01 16:35:12
【问题描述】:

假设我有一个汽车模型,我想在其中显示不同类型的数据。我在表中添加了data 列:

class AddDataToCars < ActiveRecord::Migration[5.0]
  def change
    add_column :cars, :data, :jsonb, null: false, default: '{}'
    add_index  :cars, :data, using: :gin
  end
end

然后我为我想要出现的字段添加一个访问器 (extra_options):

class Car < ApplicationRecord
  store_accessor :data, :wheel_count, :extra_options

  validates :extra_options, presence: true
end

我确保它是cars_controller.rb 中允许的参数:

def car_params
  params.require(:car).permit(:make, :model, :data, :wheel_count, :extra_options)
end

现在我可以在我的_form.html.erb 部分中创建一个文本输入,将数据放入wheel_count,如下所示:

<div class="field">
  <%= f.label :wheel_count %>
  <%= f.text_field :wheel_count %>
</div>

它按预期工作。然后我想要的是一个选择列表(多选)或一组复选框,其中选定的选项存储在extra_options 下。

我试过了:

<%= f.select :extra_options, options_for_select(["1", "2", "3", "4", "5"], car.extra_options), { include_blank: true }, { multiple: true } %>

它产生了以下标记:

<input name="car[extra_options][]" type="hidden" value="" />     
<select multiple="multiple" name="car[extra_options][]" id="car_extra_options">
  <option value=""></option>
  <option value="1">1</option>
  <option value="2">2</option>
  <option value="3">3</option>
  <option value="4">4</option>
  <option value="5">5</option>
</select>

显然,选项标签应该比 1、2、3、4 和 5 更有意义,但更糟糕的是,当我提交表单时,什么都没有存储。

如果我选择 2 和 3 提交时,参数如下所示:

"extra_options"=>["", "2", "3"]

这会导致消息Unpermitted parameter: extra_options,所以看起来我不允许在此字段中放置数组。

为一些额外的选项创建一个 jsonb 字段可能看起来很愚蠢,但它自然也会保存各种其他数据。

那么如何创建一个多选列表或者最好是一组复选框,以便在 jsonb 字段中正确保存数据(当然,在编辑提交时会显示已选择/选中的项目)?

【问题讨论】:

    标签: ruby-on-rails ruby postgresql jsonb


    【解决方案1】:

    在 car_params 中试试这个

    params.require(:car).permit(:make, :model, :data, :wheel_count, extra_options: [])
    

    对于复选框,试试这个

        <%= hidden_field_tag "car[extra_options][]", [] %>
        <% ["1", "2", "3", "4", "5"].each do |o| %>
          <% default_checked = car.extra_options.include?(o.to_s) rescue false %>
          <label class="rightpad">
            <%= check_box_tag "car[extra_options][]", o, default_checked %>
          </label>
          <span>
            <%= o %>
          </span>
        <% end %>
    

    【讨论】:

    • 这完全有效。你知道如何使它成为复选框而不是选择列表吗?我似乎无法找出像上面那样的单线(如果可能的话)。
    • 你知道为什么我不能这样创建一个新对象吗:c = Car.newc.extra_options = [1, 3]我不应该这样做吗?
    猜你喜欢
    • 1970-01-01
    • 2020-11-18
    • 1970-01-01
    • 2017-04-14
    • 2018-03-25
    • 2015-04-13
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多