我一直在研究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,或者强行点击第二个按钮,但到目前为止,我还没有在这个问题上取得任何进展。
有没有一种普遍接受的解决方法可以解决这个问题?
答案 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 && !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"
。