Ruby on Rails:更改表单提交中的 JSON
Ruby on Rails: Change the JSON from a form submission
我对 jsonb 的经验为 0,如果这是一个基本问题,我深表歉意。
我设法构建了我的动态表单并提交到数据库,但我想
更改数据进入数据库的格式。我正在使用 Postgres,我的“状态”字段是 jsonb。
在我的表单设计中,它存储的数据如下:
{
"9" => "23",
"11" => "1",
"13" => "1",
"14" => "1",
"18" => "32"
}
我希望按以下方式存储数据:
[
{
id: "9"
value: "23"
},
{
id: "13"
value: "23"
}
]
我的模特:
class Client < ApplicationRecord
extend FriendlyId
friendly_id :name, use: :slugged
validates :name, presence: true, uniqueness: true
belongs_to :country
belongs_to :client_unit, optional: true
belongs_to :industry, optional: true
end
我的表格:
<div class="row">
<% schema = JSON.parse @client.schema %>
<%= form_for @client, url: {action: "update"} do |f| %>
<%= f.fields_for :status, OpenStruct.new(@client.status) do |df| %>
<% schema.each do |initiative| %>
<div class="card card-block col-6">
<div class='card-header bg-primary text-light'>
<h5><%= initiative['name'] %></h5>
<small> Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. <i class="fas fa-sm fa-info-circle"></i> </small>
</div>
<div class='card-body'>
<div class='form-group'>
<% initiative['milestones'].each do |milestone| %>
<% if milestone['category'] == 'Boolean' %>
<div class='custom-control custom-checkbox pt-2' style='padding-left: -30px; padding-bottom: 0px'>
<%= df.check_box milestone["id"], class: 'custom-control-input', include_hidden: false%>
<%= df.label milestone['id'], milestone['name'], class: 'custom-control-label' %>
<%= check_box_tag 'status[value]', true %> ###### <-- This is me trying stuff out
</div>
<% else %>
<div class='input-group input-group-sm pt-2'>
<%= df.text_field milestone["id"], class: 'form-control', style: 'max-width: 45px' %>
<label class="control-label" style="padding-left: 10px; padding-top: 3px"><%= milestone['name'] %></label>
</div>
<% end %>
<small style="padding-bottom: 10px;"> Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. <i class="fas fa-sm fa-info-circle text-primary"></i> </small>
<% end %>
</div>
</div>
</div>
<% end %>
<% end %>
<div class='card-footer'>
<%= button_to :submit %>
</div>
<% end %>
</div>
要在 Rails(以及一般的 Rack)中将数组作为表单数据传递,您需要在键中使用空括号:
irb(main):019:0> qs = "client[milestones][][id]=9&client[milestones][][value]=23&client[milestones][][id]=13&client[milestones][][value]=23"
irb(main):020:0> p = Rack::Utils.parse_nested_query(qs)
=> {"client"=>{"milestones"=>[{"id"=>"9", "value"=>"23"}, {"id"=>"13", "value"=>"23"}]}}
您可以使用 fields_for
通过使用结构作为模型来设置嵌套输入。我不太确定该表格实际上是做什么的,因为有很多疯狂的事情正在发生。在任何情况下,您都不应该在视图中进行 JSON 解析之类的操作。
我强烈感觉您遗漏了一些关键领域对象。
要将哈希数组列入白名单,请传递具有允许属性数组的哈希键:
irb(main):005:0> params = ActionController::Parameters.new({"client"=>{"mileston
es"=>[{"id"=>"9", "value"=>"23", "id\n "=>"13"}, {"value"=>"23"}]}})
irb(main):006:0> params.require(:client).permit(:foo, :bar, milestones: [:id, :v
alue])
=> <ActionController::Parameters {"milestones"=>[<ActionController::Parameters {"id"=>"9", "value"=>"23"} permitted: true>, <ActionController::Parameters {"value"=>"23"} permitted: true>]} permitted: true>
我对 jsonb 的经验为 0,如果这是一个基本问题,我深表歉意。 我设法构建了我的动态表单并提交到数据库,但我想 更改数据进入数据库的格式。我正在使用 Postgres,我的“状态”字段是 jsonb。
在我的表单设计中,它存储的数据如下:
{
"9" => "23",
"11" => "1",
"13" => "1",
"14" => "1",
"18" => "32"
}
我希望按以下方式存储数据:
[
{
id: "9"
value: "23"
},
{
id: "13"
value: "23"
}
]
我的模特:
class Client < ApplicationRecord
extend FriendlyId
friendly_id :name, use: :slugged
validates :name, presence: true, uniqueness: true
belongs_to :country
belongs_to :client_unit, optional: true
belongs_to :industry, optional: true
end
我的表格:
<div class="row">
<% schema = JSON.parse @client.schema %>
<%= form_for @client, url: {action: "update"} do |f| %>
<%= f.fields_for :status, OpenStruct.new(@client.status) do |df| %>
<% schema.each do |initiative| %>
<div class="card card-block col-6">
<div class='card-header bg-primary text-light'>
<h5><%= initiative['name'] %></h5>
<small> Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. <i class="fas fa-sm fa-info-circle"></i> </small>
</div>
<div class='card-body'>
<div class='form-group'>
<% initiative['milestones'].each do |milestone| %>
<% if milestone['category'] == 'Boolean' %>
<div class='custom-control custom-checkbox pt-2' style='padding-left: -30px; padding-bottom: 0px'>
<%= df.check_box milestone["id"], class: 'custom-control-input', include_hidden: false%>
<%= df.label milestone['id'], milestone['name'], class: 'custom-control-label' %>
<%= check_box_tag 'status[value]', true %> ###### <-- This is me trying stuff out
</div>
<% else %>
<div class='input-group input-group-sm pt-2'>
<%= df.text_field milestone["id"], class: 'form-control', style: 'max-width: 45px' %>
<label class="control-label" style="padding-left: 10px; padding-top: 3px"><%= milestone['name'] %></label>
</div>
<% end %>
<small style="padding-bottom: 10px;"> Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. <i class="fas fa-sm fa-info-circle text-primary"></i> </small>
<% end %>
</div>
</div>
</div>
<% end %>
<% end %>
<div class='card-footer'>
<%= button_to :submit %>
</div>
<% end %>
</div>
要在 Rails(以及一般的 Rack)中将数组作为表单数据传递,您需要在键中使用空括号:
irb(main):019:0> qs = "client[milestones][][id]=9&client[milestones][][value]=23&client[milestones][][id]=13&client[milestones][][value]=23"
irb(main):020:0> p = Rack::Utils.parse_nested_query(qs)
=> {"client"=>{"milestones"=>[{"id"=>"9", "value"=>"23"}, {"id"=>"13", "value"=>"23"}]}}
您可以使用 fields_for
通过使用结构作为模型来设置嵌套输入。我不太确定该表格实际上是做什么的,因为有很多疯狂的事情正在发生。在任何情况下,您都不应该在视图中进行 JSON 解析之类的操作。
我强烈感觉您遗漏了一些关键领域对象。
要将哈希数组列入白名单,请传递具有允许属性数组的哈希键:
irb(main):005:0> params = ActionController::Parameters.new({"client"=>{"mileston
es"=>[{"id"=>"9", "value"=>"23", "id\n "=>"13"}, {"value"=>"23"}]}})
irb(main):006:0> params.require(:client).permit(:foo, :bar, milestones: [:id, :v
alue])
=> <ActionController::Parameters {"milestones"=>[<ActionController::Parameters {"id"=>"9", "value"=>"23"} permitted: true>, <ActionController::Parameters {"value"=>"23"} permitted: true>]} permitted: true>