PDF.js 和 viewer.js。将流或 blob 传递给查看器

IT技术 javascript pdf pdf.js pdf-viewer
2021-02-27 08:06:04

我在为此寻找解决方案时遇到了麻烦:我以这种方式使用 Javascript 从 SQL 文件流字段中检索 PDF blob(这是一个灯开关项目)

var blob = new Blob([screen.WebReportsPdfFilesStream.selectedItem.Pdf], { type: "application/pdf;base64" });

我有 blob,我什至可以将它转换为文件流或 base64(“JVBERi0 .....”或“%PDF 1.6 ......”等)

到目前为止没有问题。

现在我需要在查看器中显示它。我更喜欢查看器在新窗口中打开,但我愿意以某种方式将其嵌入到我的页面中。

我想知道是否可以直接将 blob 或流传递给查看器并显示文档。我试过类似的东西

PDFView.open(pdfAsArray, 0)

在这种情况下,嵌入式查看器中没有任何react。

pdfAsArray很好,因为我可以显示它,将流附加到同一页面内的画布上。我只想显示查看器,而不是将 PDF 嵌入画布中,可能是在新窗口中。

谁能提供几行代码来说明如何在 Javascript 中实现这一目标?

5个回答

我正在使用 PDFJS.version = '1.0.1040'; PDFJS.build = '997096f';

对我来说加载 base64 pdf 数据的代码是这样的:

function (base64Data) {
  var pdfData = base64ToUint8Array(base64Data);
  PDFJS.getDocument(pdfData).then(function (pdf) {
    pdf.getPage(1).then(function (page) {
      var scale = 1;
      var viewport = page.getViewport(scale);
      var canvas = document.getElementById('myCanvas');
      var context = canvas.getContext('2d');
      canvas.height = viewport.height;
      canvas.width = viewport.width;
      page.render({ canvasContext: context, viewport: viewport });
    });
  });

  function base64ToUint8Array(base64) {
    var raw = atob(base64);
    var uint8Array = new Uint8Array(raw.length);
    for (var i = 0; i < raw.length; i++) {
      uint8Array[i] = raw.charCodeAt(i);
    }
    return uint8Array;
  }
}

此函数可能是 api 调用Promise的成功函数。我在这里做的是将 pdf 渲染到画布元素上myCanvas

<canvas id="myCanvas"></canvas>

这显示了 pdf 的第一页,但没有功能。我可以理解为什么观众是可取的。如果我将其连接到查看器 (viewer.html/viewer.js),我将编辑我的答案。

编辑:如何连接查看器

1bower.json,添加"pdfjs-viewer": "1.0.1040"

2文本:

<iframe id="pdfViewer" src="lib/pdfjs-viewer/web/viewer.html" style="width: 100%; height: 700px;" allowfullscreen="" webkitallowfullscreen=""></iframe>

3更改文件中的愚蠢默认文档viewer.js

var DEFAULT_URL = '';

4控制器:

var pdfjsframe = document.getElementById('pdfViewer');
pdfjsframe.onload = function() {
  LoadPdfDocument();
};

$scope.myApiCallThatReturnsBase64PdfData.then(
  function(base64Data) {
    $scope.base64Data = base64Data;
    LoadPdfDocument();
  },
  function(failure) {

    //NotificationService.error(failure.Message);

  });

function LoadPdfDocument() {
  if ($scope.PdfDocumentLoaded)
    return;
  if (!$scope.base64Data)
    return;

  var pdfData = base64ToUint8Array($scope.base64Data);
  pdfjsframe.contentWindow.PDFViewerApplication.open(pdfData);

  $scope.PdfDocumentLoaded = true;
}

function base64ToUint8Array(base64) {
  var raw = atob(base64);
  var uint8Array = new Uint8Array(raw.length);
  for (var i = 0; i < raw.length; i++) {
    uint8Array[i] = raw.charCodeAt(i);
  }
  return uint8Array;
}
救命稻草。这个答案应该包含在 pdf.js 文档中,从未见过如此糟糕的文档。还是谢谢你!!!!
2021-05-06 08:06:04
差不多一年后,你的回答救了我的命,谢谢!=)
2021-05-09 08:06:04
@JeffreyA.Gochin,在代码示例中,var pdfData我相信data:在您的 uri之后包含数据(没有标题的实际数据)。无论您使用 angular 还是仅使用 js,您都应该能够修改此代码并像这样加载查看器。是的,IE 很烂。
2021-05-11 08:06:04
@Matt,在该行的 Chrome 调试器中中断,看看有什么没有定义。pdfjsframe?内容窗口?PDF查看器应用程序?数据通常在文档加载后进入。让我知道,我们将从那里开始。
2021-05-14 08:06:04
@QuentinJanon,抱歉, $scope来自 AngularJS。你需要改变的是你看到的任何地方$scope,让它成为一个变量。不是函数内部的局部变量,而是在任何函数之外创建一个。
2021-05-15 08:06:04

如果您有一个类型化数组(例如Uint8Array),则可以使用PDFView.open(typedarray, 0);.

如果您有一个BloborFile对象,则必须将数据转换为类型化数组,然后才能查看它:

var fr = new FileReader();
fr.onload = function() {
    var arraybuffer = this.result;
    var uint8array = new Uint8Array(arraybuffer);
    PDFView.open(uint8array, 0);
};   
fr.readAsArrayBuffer(blob);

另一种方法是为 Blob/File 对象创建一个 URL:

var url = URL.createObjectURL(blob);
PDFView.open(url, 0);

如果 PDF 查看器与您嵌入框架的网站位于同一来源,那么您还可以通过将 blob URL 传递给查看器来查看 PDF:

var url = URL.createObjectURL(blob);
var viewerUrl = 'web/viewer.html?file=' + encodeURIComponent(url);
// TODO: Load the PDF.js viewer in a frame or new tab/window.
谢谢罗布。我知道前两部分,甚至有可能使用 PDFView.load 函数打开 pdf.js 返回的 pdfDocument。我不知道最后一段代码 var url = URL.createObjectURL(blob); var viewerUrl = 'web/viewer.html?file=' + encodeURIComponent(url); // TODO:在框架或新选项卡/窗口中加载 PDF.js 查看器。我认为这将使我找到正确的位置,因为我的偏好是在新窗口中打开查看器而不是在嵌入式查看器中加载文件
2021-05-01 08:06:04
所以,这应该有效: var url = URL.createObjectURL(blob); var viewerUrl = 'Scripts/pdfViewer/web/viewer.html?file=' + encodeURIComponent(url); window.open(viewerUrl, "_blank");
2021-05-06 08:06:04

我终于使该PDFView.open方法起作用了。现在,如果我将查看器嵌入到我的页面中,并按照 Rob 在前两个示例中建议的那样调用 open 函数,它将起作用。

对于那些正在寻找这种解决方案的人,我在这里提供了一些代码行:

这是我的 Lightswitch mainPage.lsml.js 中的代码。js 脚本(pdf.js、查看器和其他)在 Lightswitch 项目的主 html 页面(Default.html)中被引用;我认为它应该适用于任何其他不基于 Lightswitch 的 html 页面。

myapp.MainPage.ShowPdf_execute = function (screen) {
// Write code here.
// Getting the stream from sql
var blob = new Blob([screen.WebReportsPdfFilesStream.selectedItem.Pdf], { type: "application/pdf;base64" });
// Pass the stream to an aspx page that makes some manipulations and returns a response
var formData = new FormData();
formData.tagName = pdfName;
formData.append(pdfName, blob);
var xhr = new XMLHttpRequest();
var url = "../OpenPdf.aspx";
xhr.open('POST', url, false);
xhr.onload = function (e) {
    var response = e.target.response;
        var pdfAsArray = convertDataURIToBinary("data:application/pdf;base64, " + response);
        var pdfDocument;
// Use PDFJS to render a pdfDocument from pdf array
    PDFJS.getDocument(pdfAsArray).then(function (pdf) {
        pdfDocument = pdf;
        var url = URL.createObjectURL(blob);
        PDFView.load(pdfDocument, 1.5)
    })
};
xhr.send(formData);  // multipart/form-data 
};

这是convertDataURIToBinary功能

function convertDataURIToBinary(dataURI) {
var base64Index = dataURI.indexOf(BASE64_MARKER) + BASE64_MARKER.length;
var base64 = dataURI.substring(base64Index);
var raw = window.atob(base64);
var rawLength = raw.length;
var array = new Uint8Array(new ArrayBuffer(rawLength));

for (i = 0; i < rawLength; i++) {
    array[i] = raw.charCodeAt(i);
}
return array;
}

仍然错过的是将流直接传递到viewer.html页面以便在新窗口中打开它并有一个单独的 ui 进行渲染的可能性

这段代码仍然不起作用,因为我有一个没有文档的空查看器:

var blob = new Blob([screen.WebReportsPdfFilesStream.selectedItem.Pdf], { type: "application/pdf;base64" });
var url = URL.createObjectURL(blob);
var viewerUrl = 'Scripts/pdfViewer/web/viewer.html?file=' + encodeURIComponent(url);
window.open(viewerUrl);

看起来encodeURIComponent(url)没有传递给查看器的好对象加载到查看器中。有什么想法或建议吗?

嗨@Efrael。您的解决方案真的很酷,但我不确定BASE64_MARKER 是什么?从哪里得到它?或者我需要在那个变量中放入什么,你能描述一下吗?:)
2021-04-24 08:06:04
只是一个字符串: var BASE64_MARKER = ';base64, ';
2021-04-30 08:06:04
活的救星。这解决了我的问题,应该被接受为正确答案。
2021-05-05 08:06:04
var url = URL.createObjectURL(blob);

var viewerUrl = 'web/viewer.html?file=' + encodeURIComponent(url);

// TODO: Load the PDF.js viewer in a frame or new tab/window.

//-- Abobe code opens a new window but with errors : 'Missing PDF 
// blob://http://server-addr:port/converted-blob' 

我在 iframe 中使用查看器;

 <iframe
    id="pdfIframe"
    src="pdfjs/web/viewer.html"
    style="width: 100%; height: 100%;"
  >
  </iframe>

和 fetch API 使用如下;

 fetch(pdfSourceUrl).then((response: Response) => {
            response.blob().then((blob) => {
              var url = URL.createObjectURL(blob);
              pdfIframe.src = `pdfjs/web/viewer.html?file=${url}`;
            });
          });

最终 iframe src 创建如下;

http://localhost:9000/pdfjs/web/viewer.html?file=blob:http://localhost:9000/14f6a2ec-ad25-40ab-9db8-560c15e90f6e