使用Phoenix.HTML.Form添加并显示错误消息"使用连接数据"

gui*_*man 5 elixir phoenix-framework

我有一个带有textarea的简单表单,用户可以在其中键入(或复制'n'paste)json字符串.表单将被发送到我的控制器操作,我将验证json字符串,如果它没问题,我将创建一些记录,一切都很好.

要将表单发送到我的控制器操作,我现在使用Phoenix.HTML.Form "With connection data",因此我没有模型/变更集.

<%= form_for @conn, @action, [as: :match], fn f -> %>
  <%= textarea f, :json, rows: 20 %>
  <%#= error_tag @changeset, f, :json %>
<% end %>
Run Code Online (Sandbox Code Playgroud)

如果json由于某种原因无效,我想再次渲染表单并显示错误消息.error_tag是一个视图辅助方法,如果有变更集,它将在字段中显示错误.因此,它现在被注释掉了.

def error_tag(form, field) do
  if error = form.errors[field] do
    content_tag :span, (humanize(field) <> " " <> translate_error(error)), class: "help-block"
  end
end

def error_tag(changeset, form, field) do
  if changeset.action do
    error_tag(form, field)
  end
end
Run Code Online (Sandbox Code Playgroud)

添加错误的正确方法是什么,以便我可以在表单字段中显示它们.我是否必须将错误添加到@connf(表单)或是否有其他方法可以获得?

jos*_*mrb 9

在您的控制器模块中

创建一个包含操作内部错误的关键字列表

errors = [field: "error message"]
Run Code Online (Sandbox Code Playgroud)

然后在调用render方法时传递它

render(conn, "template.html", errors: errors)
Run Code Online (Sandbox Code Playgroud)

在您的视图模块中

添加辅助函数以从关键字列表中提取和格式化消息

def error_tag(errors, field) when is_list(errors) and is_atom(field) do
  case Keyword.fetch(errors, field) do
    {:ok, message} -> content_tag :span, (humanize(field) <> " " <> translate_error(message)), class: "help-block"
    :error -> html_escape("")
  end
end
Run Code Online (Sandbox Code Playgroud)

在您的模板文件中

显示错误消息

<%= form_for @conn, @action, [as: :match], fn f -> %>
  <%= textarea f, :json, rows: 20 %>
  <%= error_tag @errors, :json %>
<% end %>
Run Code Online (Sandbox Code Playgroud)


gui*_*man -1

我决定实现一个验证 json 字符串的模块。模块方法changeset返回一个Ecto.Changeset我添加了自定义错误的值。私有validate_json方法检查 json 架构和内容。毕竟,我将更改和参数设置为表单中给定的参数,并且设置有效?我是否有错误。

def changeset(required_fields, json_field, params) do
  changeset =
    %Ecto.Changeset{model: %MyApp.ModelName{}}
      |> Ecto.Changeset.cast(params, required_fields, ~w())
      |> validate_json(json_field, params)
  %Ecto.Changeset{changeset | changes: params, params: params, valid?: Enum.count(changeset.errors) == 0}
end
Run Code Online (Sandbox Code Playgroud)

通过这种方式,我可以保持视图和部分的原样,并且表单和所有辅助方法都可以像以前一样与我的变更集一起使用。