Bootstrap

【总结】前端JQuery获取Java后端文件流实现常规附件预览功能

项目背景

目前维护的项目是个远古的SpringMVC项目,前端使用的是JQuery和HTML,与Vue相比维护比较复杂,功能实现上没有那么丰富。在做附件预览的需求时,考虑不借助第三方预览服务,通过开源组件实现,需要借助以下服务依赖:
1)Java:引入aspose.words依赖(支持将word文档转换为pdf格式)
注意:需要引入license

1. Java后端处理附件

1.1 将word文档转换为pdf格式

1)在resource下新增license.xml

<License>
	<Data>
		<Products>
			<Product>Aspose.Total for Java</Product>
			<Product>Aspose.Words for Java</Product>
		</Products>
		<EditionType>Enterprise</EditionType>
		<SubscriptionExpiry>20991231</SubscriptionExpiry>
		<LicenseExpiry>20991231</LicenseExpiry>
		<SerialNumber>8bfe198c-7f0c-4ef8-8ff0-acc3237bf0d7</SerialNumber>
	</Data>
	<Signature>sNLLKGMUdF0r8O1kKilWAGdgfs2BvJb/2Xp8p5iuDVfZXmhppo+d0Ran1P9TKdjV4ABwAgKXxJ3jcQTqE/2IRfqwnPf8itN8aFZlV3TJPYeD3yWE7IT55Gz6EijUpC7aKeoohTb4w2fpox58wWoF3SNp6sK6jDfiAUGEHYJ9pjU=</Signature>
</License>

2)word转pdf工具类

/**
 * @program: Mickey
 * @description: pdf工具类
 **/
public class WordToPdfUtil {

    public static byte[] toPdfBytes(HttpServletRequest req, InputStream inputstream) throws IOException {
        ByteArrayOutputStream bos = new ByteArrayOutputStream();
        byte[] b = null;
        try {
            //认证aspose-word,否则转换的pdf会有水印
            if (!getWordLicense(req)) {
                return null;
            }
            LoadOptions options = new LoadOptions();
            options.setLoadFormat(LoadFormat.DOC);
            Document doc = new Document(inputstream, options);
            doc.save(bos, SaveFormat.PDF);
            b = bos.toByteArray();
        } catch (Exception e) {
            e.printStackTrace();
            throw new RuntimeException("word转换pdf出错!");
        } finally {
            bos.close();
        }
        return b;
    }

    /**
     * 获取aspose-word授权
     *
     * @return
     */
    public static boolean getWordLicense(HttpServletRequest req) {
        boolean result = false;
        InputStream is = null;
        try {
            //获取认证文件,并转为输入流
            is = new FileInputStream(req.getSession().getServletContext().getRealPath("") + File.separator + "downFile/pdfFile" + File.separator + "license.xml");
            //创建密钥认证对象
            License aposeLic = new License();
            //进行认证ss
            aposeLic.setLicense(is);
            result = true;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return result;
    }
}

1.2 将文件流返回前端

try (InputStream in = new FileInputStream(new File("文件路径"))) {
	// 预览附件类型白名单
    List<String> whiteFileTypeList = Arrays.asList(DOC_SUFFIX, DOCX_SUFFIX, PDF_SUFFIX, TXT_SUFFIX, OFD_SUFFIX, PNG_SUFFIX, JPG_SUFFIX, JPEG_SUFFIX);
    // 不支持附件预览的类型
    if (StringUtils.isNotBlank(suffix) && !whiteFileTypeList.contains(suffix)) {
        throw new RuntimeException("不支持的附件预览类型,请下载后查看!");
    }
    String fileName = "预览附件名称";
    byte[] toPdfBytes;
    // word文档类型进行格式转换
    if (StringUtils.equals(DOC_SUFFIX, suffix) || StringUtils.equals(DOCX_SUFFIX, suffix)) {
        toPdfBytes = WordToPdfUtil.toPdfBytes(request, in);
    } else {
        toPdfBytes = readInputStream(in);
        if (StringUtils.equals(PNG_SUFFIX, suffix) || StringUtils.equals(JPG_SUFFIX, suffix) || StringUtils.equals(JPEG_SUFFIX, suffix)) {
            String base64Str = Base64Utils.encodeToString(toPdfBytes);
            model.put("base64Str", base64Str);
        }
        if (StringUtils.equals(TXT_SUFFIX, suffix)) {
            model.put("txtView", toPdfBytes);
        }
    }
    if (toPdfBytes != null) {
        try {
            String contextPath = request.getContextPath();
            String basePath = request.getSession().getServletContext().getRealPath(File.separator);
            File outFile = new File(basePath + File.separator + "temp" + File.separator + "fileView");
            if (!outFile.isDirectory()) {
                if (!outFile.exists()) {
                    boolean b = outFile.mkdirs();
                }
            }
            FileOutputStream fo = new FileOutputStream(outFile + File.separator + "附件id" + ".pdf");
            BufferedOutputStream out = new BufferedOutputStream(fo);
            out.write(toPdfBytes, 0, toPdfBytes.length);
            out.flush();
            out.close();
            return contextPath + File.separator + "temp" + File.separator + "fileView" + File.separator + "附件id" + ".pdf";
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
} catch (IOException e) {
    log.error("附件处理异常", e);
} 

2. 前端处理附件预览(JQuery)

2.1 预览pdf文件

引入PDFView组件

<iframe id="pdfFrame" src="${base}/pdfView/web/viewer.html?file=${viewFilePath}" width="900" height="530"></iframe>

2.2 预览ofd文件

引入ofd.js依赖,支持ofd文件查看

<div class="ofdContainer" id="ofdContainer" style="width: 900px; height: 530px; overflow-y: scroll;"></div>
$(function() {
  // ofd文件预览
  if (suffix && suffix === 'ofd') {
    var xhr = new XMLHttpRequest();
    xhr.open('POST', '附件下载地址', true);
    xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
    xhr.responseType = 'blob';
    xhr.send("id=" + 附件id);
    xhr.onload = function(e) {
      if (this.status === 200) {
        var blob = new Blob([this.response], {type: 'application/ofd'});
        // 处理文件流,比如使用 FileReader 进行读取或者直接下载
        const file = new File([blob], fileName + '.ofd', { type: blob.type });
        ofd.parseOfdDocument({
          ofd: file,
          success: function (res) {
            const screenWidth = 800;
            const ofdRenderRes = ofd.renderOfd(screenWidth, res[0]);
            let ofdContainerDiv = document.getElementById('ofdContainer');
            // 清空元素
            ofdContainerDiv.innerHTML = '';
            for (const item of ofdRenderRes) {
              ofdContainerDiv.appendChild(item);
            }
          },
          fail: function (err) {
            console.error(err);
          },
        });
      }
    };
  }
})

2.3 预览图片、txt文档

1)将后端获取的文件流转换为Base64格式,前端展示图片

<img src="data:image/jpeg;base64,${base64Str!}" style="width: 900px; height: 530px; overflow-y: scroll;" />

2)展示txt文档

<pre class="txtContainer" id="txtContainer" style="width: 900px; height: 530px; overflow-y: scroll; text-align: left;"></pre>
fetch('替换为你的后端TXT文件下载地址' + 附件id)
    .then(response => response.text())
    .then(text => {
      document.getElementById('txtContainer').textContent = text;
    })
    .catch(error => console.error('读取文件出错:', error));

共勉。

;