我一直在研究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或通过强制单击第二个按钮来解决此问题,但是到目前为止,我在此问题上尚未取得任何进展。
有没有普遍接受的解决此类问题的方法?
基本上,您首先要触发ajax请求,并在其oncomplete检查中是否成功,然后触发同步请求以下载文件(请注意,您无法使用ajax下载文件)。您可以使用FacesContext#validationFailed()(或OmniFaces Faces.validationFailed())标记验证失败状态,该状态可以通过argsPrimeFaces在oncomplete函数上下文中注入的对象获得(请注意,与ajax相关的属性(例如,oncomplete禁用ajax时不起作用))。
FacesContext#validationFailed()
Faces.validationFailed()
args
oncomplete
像这样:
<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"。
<p:commandButton>
<p:remoteCommand>
ajax="false"