有条件地提供文件下载或显示导出验证错误消息

时间:2015-09-15 17:06:55

标签: ajax validation jsf primefaces download

我一直在研究CRUD应用程序,我需要将数据从数据库导出到csv文件。 为了导出,我必须以下面的代码中显示的方式禁用ajax:

<p:commandButton value="Export" ajax="false" action="myController.export"/>

在调用的方法中,我创建文件并通过OmniFaces实用程序方法下载:

Faces.sendFile(file, true);

使用相同的方法,检查是否确实存在任何数据,如果没有任何数据,则会显示警告对话框:

RequestContext.getCurrentInstance().showMessageInDialog(new FacesMessage(FacesMessage.SEVERITY_WARN, "Warning!", "No available data for export."));

现在,虽然所有这些都按预期工作,但问题是因为禁用了ajax,所以无法动态显示对话框,并重新加载页面。如果启用了ajax,则会动态显示对话框,但文件下载不会启动。

我一直在努力解决这个问题,使用Monitor download,或者强行点击第二个按钮,但到目前为止,我还没有在这个问题上取得任何进展。

有没有一种普遍接受的解决方法可以解决这个问题?

1 个答案:

答案 0 :(得分:9)

  

是否有任何一般可接受的方法来解决像这样的问题?

你基本上想要首先触发ajax请求并在其完成检查中是否成功,然后触发同步请求以下载文件(请注意你can't用ajax下载文件)。您可以使用FacesContext#validationFailed()(或OmniFaces Faces.validationFailed())来标记验证失败状态,该状态可通过args对象获得,PrimeFaces在oncomplete函数上下文中注入该对象(请注意ajax相关属性,例如oncomplete在禁用ajax时不起作用。)

这样的事情:

<p:commandButton 
    value="Export" 
    action="#{bean.export}" 
    oncomplete="if (args &amp;&amp; !args.validationFailed) PF('download').jq.click()" />
<p:commandButton 
    widgetVar="download" 
    styleClass="ui-helper-hidden" 
    action="#{bean.download}" ajax="false" />
public void export() {
    // Prepare file locally.

    if (fail) {
        // Show message your way and then set validation failed as below.
        Faces.validationFailed();
    }
}

public void download() throws IOException {
    Faces.sendFile(file, true);

    // Delete local file afterwards?
}

请注意,正在使用隐藏的<p:commandButton>代替<p:remoteCommand>,因为后者不支持ajax="false"