Vue实现PDF导出和打印 放大缩小 功能

这篇博客介绍了如何在Vue.js应用中实现PDF文件的预览、翻页、打印及导出功能。通过引入vue-pdf和vue-to-pdf插件,详细展示了安装过程、组件使用、事件监听以及解决打印中文乱码问题的方法。同时,提供了关键代码片段和配置修改,帮助开发者快速实现类似功能。
摘要由CSDN通过智能技术生成

npm安装

npm install --save vue-pdf
npm install --save vue-to-pdf

页面引入

// pdf预览
import pdf from "vue-pdf";

页面注入

components: {pdf},

页面使用

      <div>
              <div>
                <button type="button" @click="changePdfPage(0)">上一页</button>
                <button type="button" @click="changePdfPage(1)">下一页</button>
              </div>

              <div>
                <button type="button" @click="scaleD()">放大</button>
                <button type="button" @click="scaleX()">缩小</button>
              </div>
              <div>
                <button type="button" @click="printPdf">打印</button>
              </div>
              <div id="exportPdf" ref="exportPdf">
              <pdf
                ref="pdf"
                :src="src"
                :page="currentPage"
                :rotate="pageRotate"
                @num-pages="pageCount = $event"
                @page-loaded="currentPage = $event"
                @loaded="loadPdfHandler"
              ></pdf>
              </div>
              <p>{{ currentPage }} / {{ pageCount }}</p>
              <div>
                <button type="button" @click="clock()">顺时针</button>
                <button type="button" @click="counterClock()">逆时针</button>
              </div>

            </div>

data中定义

    src:"",
      currentPage: 0, // pdf文件页码
      pageCount: 0, // pdf文件总页数
      scale: 100,
      pageRotate: 0,

js逻辑

这里this.$PDFSave会报一个没有被定义的错误,需要在main.js中进行引入

// pdf打印预览
import vueToPdf from 'vue-to-pdf';
Vue.use(vueToPdf);
 // pdf加载时
    loadPdfHandler(e) {
      e;
      this.currentPage = 1; // 加载的时候先加载第一页
    },
    // 改变PDF页码,val传过来区分上一页下一页的值,0上一页,1下一页
    changePdfPage(val) {
      if (val === 0 && this.currentPage > 1) {
        this.currentPage--;
      }
      if (val === 1 && this.currentPage < this.pageCount) {
        this.currentPage++;
      }
    },
    //放大
    scaleD() {
      this.scale += 5;
      this.$refs.pdf.$el.style.width = parseInt(this.scale) + "%";
    },
    //缩小
    scaleX() {
      if (this.scale === 100) {
        return;
      }
      this.scale += -5;
      this.$refs.pdf.$el.style.width = parseInt(this.scale) + "%";
    },
    // 顺时针
    clock() {
      this.pageRotate += 90;
    },
    // 逆时针
    counterClock() {
      this.pageRotate -= 90;
    },
    printPdf(){ // 调用打印
      this.$refs.pdf.print()
    }

但是这个打印存在中文乱码的问题,是框架代码存在问题,GitHub上给出了解决方案,需要修改vue-pdf插件中的pdfjsWrapper.js这个文件。

路径:node_modules\vue-pdf\src\pdfjsWrapper.js

全部替换即可,亲测有效

import { PDFLinkService } from 'pdfjs-dist/es5/web/pdf_viewer';
 
var pendingOperation = Promise.resolve();
 
export default function (PDFJS) {
 
	function isPDFDocumentLoadingTask(obj) {
 
		return typeof(obj) === 'object' && obj !== null && obj.__PDFDocumentLoadingTask === true;
		// or: return obj.constructor.name === 'PDFDocumentLoadingTask';
	}
 
	function createLoadingTask(src, options) {
		var source;
		if ( typeof(src) === 'string' )
			source = { url: src };
		else if ( src instanceof Uint8Array )
			source = { data: src };
		else if ( typeof(src) === 'object' && src !== null )
			source = Object.assign({}, src);
		else
			throw new TypeError('invalid src type');
		if ( options && options.withCredentials )
			source.withCredentials = options.withCredentials;
 
		var loadingTask = PDFJS.getDocument(source);
		loadingTask.__PDFDocumentLoadingTask = true; // since PDFDocumentLoadingTask is not public
 
		if ( options && options.onPassword )
			loadingTask.onPassword = options.onPassword;
 
		if ( options && options.onProgress )
			loadingTask.onProgress = options.onProgress;
 
		return loadingTask;
	}
 
 
	function PDFJSWrapper(canvasElt, annotationLayerElt, emitEvent) {
		var pdfDoc = null;
		var pdfPage = null;
		var pdfRender = null;
		var canceling = false;
		canvasElt.getContext('2d').save();
		function clearCanvas() {
			canvasElt.getContext('2d').clearRect(0, 0, canvasElt.width, canvasElt.height);
		}
 
		function clearAnnotations() {
			while ( annotationLayerElt.firstChild )
				annotationLayerElt.removeChild(annotationLayerElt.firstChild);
		}
 
		this.destroy = function() {
			if ( pdfDoc === null )
				return;
 
			// Aborts all network requests and destroys worker.
			pendingOperation = pdfDoc.destroy();
			pdfDoc = null;
		}
 
		this.getResolutionScale = function() {
			return canvasElt.offsetWidth / canvasElt.width;
		}
		this.printPage = function(dpi, pageNumberOnly) {
			if ( pdfPage === null )
				return;
 
			// 1in == 72pt
			// 1in == 96px
			var PRINT_RESOLUTION = dpi === undefined ? 150 : dpi;
			var PRINT_UNITS = PRINT_RESOLUTION / 72.0;
			var CSS_UNITS = 96.0 / 72.0;
 
      //var iframeElt = document.createElement('iframe');
      var printContainerElement = document.createElement('div');
			printContainerElement.setAttribute('id', 'print-container')
 
			// function removeIframe() {
 
      // 	iframeElt.parentNode.removeChild(iframeElt);
      function removePrintContainer() {
				printContainerElement.parentNode.removeChild(printContainerElement);
			}
 
			new Promise(function(resolve, reject) {
 
        printContainerElement.frameBorder = '0';
				printContainerElement.scrolling = 'no';
				printContainerElement.width = '0px;'
				printContainerElement.height = '0px;'
				printContainerElement.style.cssText = 'position: absolute; top: 0; left: 0';
 
				window.document.body.appendChild(printContainerElement);
				resolve(window)
			})
			.then(function(win) {
 
				win.document.title = '';
 
				return pdfDoc.getPage(1)
				.then(function(page) {
					var viewport = page.getViewport({ scale: 1 });
          //win.document.head.appendChild(win.document.createElement('style')).textContent =
          printContainerElement.appendChild(win.document.createElement('style')).textContent =
						'@supports ((size:A4) and (size:1pt 1pt)) {' +
							'@page { margin: 1pt; size: ' + ((viewport.width * PRINT_UNITS) / CSS_UNITS) + 'pt ' + ((viewport.height * PRINT_UNITS) / CSS_UNITS) + 'pt; }' +
						'}' +
 
            '#print-canvas { display: none }' +
 
						'@media print {' +
							'body { margin: 0 }' +
              '#print-canvas { page-break-before: avoid; page-break-after: always; page-break-inside: avoid; display: block }' +
							'body > *:not(#print-container) { display: none; }' +
						'}'+
 
						'@media screen {' +
							'body { margin: 0 }' +
						'}'
					return win;
				})
			})
			.then(function(win) {
 
				var allPages = [];
 
				for ( var pageNumber = 1; pageNumber <= pdfDoc.numPages; ++pageNumber ) {
 
					if ( pageNumberOnly !== undefined && pageNumberOnly.indexOf(pageNumber) === -1 )
						continue;
 
					allPages.push(
						pdfDoc.getPage(pageNumber)
						.then(function(page) {
 
							var viewport = page.getViewport({ scale: 1 });
 
              //var printCanvasElt = win.document.body.appendChild(win.document.createElement('canvas'));
              var printCanvasElt = printContainerElement.appendChild(win.document.createElement('canvas'));
							printCanvasElt.setAttribute('id', 'print-canvas')
							printCanvasElt.width = (viewport.width * PRINT_UNITS);
							printCanvasElt.height = (viewport.height * PRINT_UNITS);
 
							return page.render({
								canvasContext: printCanvasElt.getContext('2d'),
								transform: [ // Additional transform, applied just before viewport transform.
									PRINT_UNITS, 0, 0,
									PRINT_UNITS, 0, 0
								],
								viewport: viewport,
								intent: 'print'
							}).promise;
						})
					);
				}
 
				Promise.all(allPages)
				.then(function() {
					win.focus(); // Required for IE
					if (win.document.queryCommandSupported('print')) {
						win.document.execCommand('print', false, null);
					} else {
            win.print();
          }
					removePrintContainer();
				})
				.catch(function(err) {
 
					removePrintContainer();
					emitEvent('error', err);
				})
			})
		}
 
		this.renderPage = function(rotate) {
			if ( pdfRender !== null ) {
 
				if ( canceling )
					return;
				canceling = true;
				pdfRender.cancel();
				return;
			}
 
			if ( pdfPage === null )
				return;
 
			var pageRotate = (pdfPage.rotate === undefined ? 0 : pdfPage.rotate) + (rotate === undefined ? 0 : rotate);
 
			var scale = canvasElt.offsetWidth / pdfPage.getViewport({ scale: 1 }).width * (window.devicePixelRatio || 1);
			var viewport = pdfPage.getViewport({ scale: scale, rotation:pageRotate });
 
			emitEvent('page-size', viewport.width, viewport.height, scale);
 
			canvasElt.width = viewport.width;
			canvasElt.height = viewport.height;
 
			pdfRender = pdfPage.render({
				canvasContext: canvasElt.getContext('2d'),
				viewport: viewport
			});
 
			annotationLayerElt.style.visibility = 'hidden';
			clearAnnotations();
 
			var viewer = {
				scrollPageIntoView: function(params) {
					emitEvent('link-clicked', params.pageNumber)
				},
			};
 
			var linkService = new PDFLinkService();
			linkService.setDocument(pdfDoc);
			linkService.setViewer(viewer);
 
			pendingOperation = pendingOperation.then(function() {
 
				var getAnnotationsOperation =
				pdfPage.getAnnotations({ intent: 'display' })
				.then(function(annotations) {
 
					PDFJS.AnnotationLayer.render({
						viewport: viewport.clone({ dontFlip: true }),
						div: annotationLayerElt,
						annotations: annotations,
						page: pdfPage,
						linkService: linkService,
						renderInteractiveForms: false
					});
				});
 
				var pdfRenderOperation =
				pdfRender.promise
				.then(function() {
 
					annotationLayerElt.style.visibility = '';
					canceling = false;
					pdfRender = null;
				})
				.catch(function(err) {
 
					pdfRender = null;
					if ( err instanceof PDFJS.RenderingCancelledException ) {
 
						canceling = false;
						this.renderPage(rotate);
						return;
					}
					emitEvent('error', err);
				}.bind(this))
 
				return Promise.all([getAnnotationsOperation, pdfRenderOperation]);
			}.bind(this));
		}
 
 
		this.forEachPage = function(pageCallback) {
 
			var numPages = pdfDoc.numPages;
 
			(function next(pageNum) {
 
				pdfDoc.getPage(pageNum)
				.then(pageCallback)
				.then(function() {
 
					if ( ++pageNum <= numPages )
						next(pageNum);
				})
			})(1);
		}
 
 
		this.loadPage = function(pageNumber, rotate) {
 
			pdfPage = null;
 
			if ( pdfDoc === null )
				return;
 
			pendingOperation = pendingOperation.then(function() {
 
				return pdfDoc.getPage(pageNumber);
			})
			.then(function(page) {
 
				pdfPage = page;
				this.renderPage(rotate);
				emitEvent('page-loaded', page.pageNumber);
			}.bind(this))
			.catch(function(err) {
 
				clearCanvas();
				clearAnnotations();
				emitEvent('error', err);
			});
		}
 
		this.loadDocument = function(src) {
 
			pdfDoc = null;
			pdfPage = null;
 
			emitEvent('num-pages', undefined);
 
			if ( !src ) {
 
				canvasElt.removeAttribute('width');
				canvasElt.removeAttribute('height');
				clearAnnotations();
				return;
			}
 
			// wait for pending operation ends
			pendingOperation = pendingOperation.then(function() {
 
				var loadingTask;
				if ( isPDFDocumentLoadingTask(src) ) {
 
					if ( src.destroyed ) {
 
						emitEvent('error', new Error('loadingTask has been destroyed'));
						return
					}
 
					loadingTask = src;
				} else {
 
					loadingTask = createLoadingTask(src, {
						onPassword: function(updatePassword, reason) {
 
							var reasonStr;
							switch (reason) {
								case PDFJS.PasswordResponses.NEED_PASSWORD:
									reasonStr = 'NEED_PASSWORD';
									break;
								case PDFJS.PasswordResponses.INCORRECT_PASSWORD:
									reasonStr = 'INCORRECT_PASSWORD';
									break;
							}
							emitEvent('password', updatePassword, reasonStr);
						},
						onProgress: function(status) {
 
							var ratio = status.loaded / status.total;
							emitEvent('progress', Math.min(ratio, 1));
						}
					});
				}
 
				return loadingTask.promise;
			})
			.then(function(pdf) {
 
				pdfDoc = pdf;
				emitEvent('num-pages', pdf.numPages);
				emitEvent('loaded');
			})
			.catch(function(err) {
 
				clearCanvas();
				clearAnnotations();
				emitEvent('error', err);
			})
		}
 
		annotationLayerElt.style.transformOrigin = '0 0';
	}
 
	return {
		createLoadingTask: createLoadingTask,
		PDFJSWrapper: PDFJSWrapper,
	}
}

转载地址:

vue实现 pdf 预览、翻页、打印、导出功能_文i的博客-CSDN博客_vue打印预览页功能

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要通过vue-pdf实现PDF放大缩小功能,你可以使用vue-pdf的内置方法来控制PDF的缩放级别。以下是一个示例: 1. 首先,确保你已经安装了vue-pdf库。可以通过运行以下命令来安装它: ```shell npm install vue-pdf ``` 2. 在你的Vue组件中,导入vue-pdf库并注册它: ```vue <template> <div> <pdf :src="pdfSrc" :page="currentPage" @num-pages="setTotalPages" @page-loaded="setLoadedPages"></pdf> <button @click="zoomIn">放大</button> <button @click="zoomOut">缩小</button> </div> </template> <script> import { pdf } from 'vue-pdf'; export default { components: { pdf }, data() { return { pdfSrc: 'path_to_your_pdf_file.pdf', currentPage: 1, totalPages: 0, loadedPages: [], zoomLevel: 1 }; }, methods: { setTotalPages(totalPages) { this.totalPages = totalPages; }, setLoadedPages(loadedPages) { this.loadedPages = loadedPages; }, zoomIn() { this.zoomLevel += 0.1; this.$refs.pdf.setScale(this.zoomLevel); }, zoomOut() { this.zoomLevel -= 0.1; this.$refs.pdf.setScale(this.zoomLevel); } } }; </script> ``` 3. 在上面的示例中,我们在模板中添加了两个按钮,分别用于放大缩小。当按钮被点击时,我们调用相应的方法`zoomIn`和`zoomOut`。 4. 在`zoomIn`和`zoomOut`方法中,我们增加或减少`zoomLevel`变量的值,并通过`this.$refs.pdf.setScale`方法来设置PDF的缩放级别。`this.$refs.pdf`引用了vue-pdf组件的实例,我们可以通过它来访问vue-pdf提供的方法和属性。 这样,你就可以使用vue-pdf实现PDF放大缩小功能了。注意,这只是一个简单的示例,你可以根据需要进行调整和扩展。确保在vue-pdf组件上使用`ref`属性来获取组件的引用,以便在方法中使用它。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值