【问题标题】:How to handle global errors in forms in Play Framework 2 (Java)如何在 Play Framework 2 (Java) 中处理表单中的全局错误
【发布时间】:2015-05-20 16:31:04
【问题描述】:

在 Play Framework 2 (Java) 中处理表单中的错误时,我想呈现一个整体错误消息,警告用户出现问题,例如“Form X 未更新”。这样做的一种方法是,除了每个表单字段的错误之外,创建一个全局错误,然后在每个模板上呈现它:

控制器:

if(form.hasErrors()) {
    form.reject("Form X not updated."); // This creates a global error
}

模板:

@if(form.hasGlobalErrors) {
    @someFunctionToPresentErrors(from.globalError.message))
}

但是,将这段相同的代码添加到包含表单的每个模板违反了DRY 原则。

到目前为止,我最好的解决方案是将错误消息添加到Http.Context,然后在每个模板调用的主模板中拦截错误消息:

控制器:

Http.Context.current().args.put("error", "Form X not updated.");

主模板:

@if(Http.Context.current().args.containsKey("error")){
    @someFunctionToPresentErrors(
        Http.Context.current().args.get("error").asInstanceOf[String])
}

这更好,因为它允许我将错误演示文稿保存在一个地方,但感觉就像滥用Http.Context。请注意,flash(...) 也可用于隐式传递变量,并提供与Http.Context 相同的效果,但在调用badRequest(...) 时不允许这样做,因为flash 只能用于重定向。无论如何,我认为应该有更好的方法来使用 Play Framework 提供的表单机制。

对此有最佳实践吗?

【问题讨论】:

    标签: java playframework playframework-2.3


    【解决方案1】:

    我认为您在这种情况下高估了 DRY 违规 - 实际上我意识到即使 global 错误也不是那么全局 - 实际上每个表单都有它自己的上下文。无论如何,为了让工作更快,您可以使用 template tagsFlash scope

    首先添加两个通用公共方法,即在您的 Application 控制器中

    public static void flashGlobalError() {
        flashGlobalError("Oppps... error occurred and I don't know the reason ;(");
    }
    
    public static void flashGlobalError(String msg) {
        flash("globalError", msg);
    }
    

    然后创建一个无参数的tag template,即views/tags/globalErrorByFlash.scala.html

    @if(flash.get("globalError")) {
        <div style="background: red">
            <h4>Error!</h4>
            @flash.get("globalError")
        </div>
    }
    

    并将其包含在您的 main 模板中(或仅在选定的视图中):

    <body>
        @tags.globalErrorByFlash()
        @content
    </body>
    

    所以每次当你想显示这个错误时,你可以在你的操作中做到这一点:

    if(form.hasErrors()) {
        Application.flashGlobalError();
        return badRequest(...);
    }
    

    if(form.hasErrors()) {
        Application.flashGlobalError("This case is quite bit different...");
        return badRequest(...);
    }
    

    这种方法的最大好处是您无需依赖Form.form(T),因为您可以在任何地方使用它:

    if (1 != 0){
        Application.flashGlobalError("Stupido! One isn't equal to Zero!");
    }
    

    最后:请记住,保存在 Flash 范围内的任何值存在仅用于一个请求,因此它可能在重定向后消失。视图/模板渲染也被计算在内,因此如果您将错误重定向到新操作,请将其转发到控制器中,以便重定向视图可以看到它。

    【讨论】:

    • 其实我已经考虑过这个解决方案了。但是,当您调用badRequest() 时(当您在表单中出现错误时应该这样做),您不应该同时使用flash。事实上,如果你这样做,这会产生一个警告:你正在使用带闪烁的状态代码“400”,它应该只与重定向状态一起使用!。所以这就是为什么我决定使用Http.Context而不是flash。然而,这个想法是相同的,即避免传递参数。
    • 我现在已经用有关 flash 的信息更新了问题。
    • TBH 我不明白生成此警告的原因(顺便说一句。仅在 DEV 模式下)。就个人而言,直到它工作(至少直到 2.3.x)我将使用它来填充简单的消息,即使没有重定向。如果未来的版本会禁用它 - 我会寻找其他解决方案:)
    • 我尝试过使用 flash,它确实会出现一些问题,如您链接到的 Google Group 中所述。例如,我有时会连续两次看到“密码错误”消息,即使我第一次尝试输入了错误的密码,而第二次却成功了。无论如何,我没有理由不使用Http.Context 而不是flash,因为效果是一样的。 (但也许问题也一样?我没有遇到任何问题,但这并不意味着它们不存在......)
    • 这种方法在scala中不起作用,你有什么想法如何让它在scala中起作用吗?
    猜你喜欢
    • 2015-08-30
    • 1970-01-01
    • 1970-01-01
    • 2019-10-26
    • 2016-02-08
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2017-08-30
    相关资源
    最近更新 更多