pdf-view.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473
  1. <template>
  2. <div class="home_wrap">
  3. <!-- <button class="show-catalogue" @click="showDrawer">目录</button> -->
  4. <button type="button" class="catalogue-btn" @click="showDrawer">
  5. <img src="@/assets/icon/目录.png" alt="" />
  6. 目录
  7. </button>
  8. <pdf-catalogue v-model="visible" :list="gData" @click-menu="handleSelect" />
  9. <div
  10. ref="getheight"
  11. id="scrollBox"
  12. :class="{ 'loading-hide': loading }"
  13. :style="{ width: pdf_div_width, margin: '0 auto' }"
  14. >
  15. <div v-for="(page, index) in pdf_pages" :key="index" class="canvas-box">
  16. <canvas :id="'the_canvas' + page" :key="page"></canvas>
  17. <div class="ws" :style="{ fontSize: maxWidth / 40 + 'px' }">
  18. {{ watermarkText || userInfo.phone || "搞一下" }}
  19. </div>
  20. </div>
  21. </div>
  22. <div class="loading" v-if="loading">
  23. <van-circle
  24. v-model="rate"
  25. :rate="currentRate"
  26. :speed="100"
  27. :text="currentRate + '%'"
  28. />
  29. <van-loading size="24px" color="#0094ff" style="margin-top: 10px"
  30. >数据加载中...</van-loading
  31. >
  32. </div>
  33. </div>
  34. </template>
  35. <script>
  36. import PdfCatalogue from "@/components/pdf/pdf-catalogue.vue";
  37. import { OSS_URL } from "@/common/config";
  38. let PDFJS = require("pdfjs-dist");
  39. PDFJS.GlobalWorkerOptions.workerSrc = require("pdfjs-dist/build/pdf.worker.entry.js");
  40. require("@/wasm/wasm_exec.js");
  41. export default {
  42. components: {
  43. PdfCatalogue,
  44. },
  45. props: {
  46. path: {
  47. default: "",
  48. },
  49. is_encrypt: {
  50. default: 1,
  51. },
  52. platform: {
  53. default: "h5",
  54. },
  55. watermarkText: {
  56. default: "",
  57. },
  58. },
  59. data() {
  60. return {
  61. expandedKeys: [],
  62. pageArr: [],
  63. gData: [],
  64. visible: false,
  65. placement: "left",
  66. pdf_scale: 1.0, //pdf放大系数
  67. pdf_pages: [],
  68. pdf_div_width: "",
  69. pdf_src: null,
  70. pdf_data: null,
  71. userInfo: {},
  72. loading: true,
  73. loadingEndSize: 0,
  74. rate: 10,
  75. currentRate: 0,
  76. maxWidth: 0,
  77. };
  78. },
  79. mounted() {
  80. try {
  81. this.userInfo = JSON.parse(localStorage.getItem("user_info") || `{}`);
  82. } catch (e) {
  83. this.userInfo = {};
  84. }
  85. if (this.path) {
  86. this.currentRate = 5;
  87. this.waitWasm();
  88. }
  89. },
  90. methods: {
  91. handleSelect(b) {
  92. this.visible = false;
  93. let height = this.$refs.getheight.offsetHeight;
  94. let top = 0;
  95. // console.log(this.pageArr);
  96. for (let index = 1; index < this.pageArr.length; index++) {
  97. // console.log(this.pageArr[index].num , b.selectedNodes[0].data.props.dest[0].num);
  98. if (this.pageArr[index].num > b[0].num) {
  99. top = index;
  100. break;
  101. }
  102. }
  103. let offsetHeight =
  104. top * this.$refs.getheight.children[0].getBoundingClientRect().height;
  105. offsetHeight = (offsetHeight * b[0].num) / this.pageArr[top].num;
  106. window.scrollTo({
  107. top: offsetHeight + 2 * (top - 1),
  108. // behavior: 'smooth' // 滚动行为:smooth平滑滚动,instant瞬间滚动,默认值auto,等同于instant
  109. });
  110. console.log(height);
  111. console.log(this.pageArr[0]);
  112. },
  113. onClose() {
  114. this.visible = false;
  115. },
  116. showDrawer() {
  117. this.visible = true;
  118. },
  119. scaleD() {
  120. //放大
  121. let max = 0;
  122. if (window.screen.width > 1440) {
  123. max = 1.4;
  124. } else {
  125. max = 1.2;
  126. }
  127. if (this.pdf_scale >= max) {
  128. return;
  129. }
  130. this.pdf_scale = this.pdf_scale + 0.1;
  131. this._loadFile(this.pdf_data);
  132. },
  133. scaleX() {
  134. //缩小
  135. let min = 1.0;
  136. if (this.pdf_scale <= min) {
  137. return;
  138. }
  139. this.pdf_scale = this.pdf_scale - 0.1;
  140. this._loadFile(this.pdf_data);
  141. },
  142. // 加载wasm
  143. waitWasm() {
  144. if (parseInt(this.is_encrypt || 0) === 1) {
  145. const go = new Go();
  146. const that = this;
  147. WebAssembly.instantiateStreaming(
  148. // fetch("wasm/libgetpdf.wasm"),
  149. // fetch('https://gaoyixia.oss-cn-hangzhou.aliyuncs.com/wenjian/libgetpdf.wasm'),
  150. fetch("https://gaoyixia.oss-cn-hangzhou.aliyuncs.com/libgetpdf.wasm"),
  151. go.importObject
  152. ).then((result) => {
  153. go.run(result.instance);
  154. console.log(GetVersion());
  155. this.currentRate = 30;
  156. ReadPdf(this.path).then((pd) => {
  157. that.readPdfData(pd);
  158. });
  159. });
  160. } else {
  161. const path =
  162. this.path.indexOf("http") > -1 ? this.path : OSS_URL + this.path;
  163. this.readPdfData(path);
  164. }
  165. },
  166. readPdfData(pdf_data) {
  167. this.pdf_data = pdf_data;
  168. this._loadFile(this.pdf_data);
  169. },
  170. _loadFile(pdf_data) {
  171. //初始化pdf
  172. let loadingTask;
  173. if (parseInt(this.is_encrypt || 0) === 1) {
  174. loadingTask = PDFJS.getDocument({
  175. data: pdf_data,
  176. });
  177. } else {
  178. // 未加密
  179. loadingTask = PDFJS.getDocument(pdf_data);
  180. }
  181. loadingTask.promise.then((pdf) => {
  182. this.pdfDoc = pdf;
  183. this.pdf_pages = this.pdfDoc.numPages;
  184. pdf.getOutline().then((r) => {
  185. console.log("dddd");
  186. console.log(r);
  187. this.gData = r;
  188. if (r != null) {
  189. console.log("888888");
  190. r.forEach((it) => {
  191. console.log(it);
  192. pdf
  193. .getPageIndex({
  194. ref: it.dest[0],
  195. })
  196. .then((vvv) => {
  197. console.log(vvv);
  198. });
  199. // $("#div").html(it.title)
  200. });
  201. }
  202. });
  203. //debugger
  204. this.$nextTick(() => {
  205. this.currentRate = 50;
  206. this._renderPage(1);
  207. });
  208. });
  209. },
  210. _renderPage(num) {
  211. //渲染pdf页
  212. const that = this;
  213. this.pdfDoc.getPage(num).then((page) => {
  214. that.pageArr.push(page.ref);
  215. let canvas = document.getElementById("the_canvas" + num);
  216. let ctx = canvas.getContext("2d");
  217. let dpr = window.devicePixelRatio || 1;
  218. let bsr =
  219. ctx.webkitBackingStorePixelRatio ||
  220. ctx.mozBackingStorePixelRatio ||
  221. ctx.msBackingStorePixelRatio ||
  222. ctx.oBackingStorePixelRatio ||
  223. ctx.backingStorePixelRatio ||
  224. 1;
  225. let ratio = dpr / bsr;
  226. let viewport = page.getViewport({
  227. scale: this.pdf_scale,
  228. });
  229. canvas.width = viewport.width * ratio;
  230. canvas.height = viewport.height * ratio;
  231. // canvas.style.width = viewport.width + "px";
  232. // that.pdf_div_width = viewport.width + "px";
  233. // canvas.style.height = viewport.height + "px";
  234. canvas.style.width = "100%";
  235. canvas.style.maxWidth = viewport.width + "px";
  236. this.maxWidth = viewport.width;
  237. that.pdf_div_width = "100%";
  238. ctx.setTransform(ratio, 0, 0, ratio, 0, 0);
  239. let renderContext = {
  240. canvasContext: ctx,
  241. viewport: viewport,
  242. };
  243. page.render(renderContext).promise.then(() => {
  244. // 添加水印
  245. // ctx.rotate(-20 * Math.PI / 180); // 逆时针旋转π/9
  246. // ctx.font = `${20 * ratio}px Vedana`; // 设置字体
  247. // ctx.fillStyle = 'rgba(0, 0, 0, 0.20)'; // 设置字体的颜色
  248. // ctx.textAlign = 'center'; // 文本对齐方式
  249. // ctx.textBaseline = 'Middle'; // 文本基线
  250. // ctx.fillText(
  251. // `${this.watermarkText || this.userInfo.phone || '搞一下'}`,
  252. // 60 * ratio,
  253. // viewport.height * ratio / 3
  254. // ); // 绘制文字
  255. // 绘制加载进度
  256. this.loadingEndSize += 1;
  257. this.currentRate =
  258. 50 +
  259. parseFloat(
  260. ((this.loadingEndSize / this.pdf_pages) * 30).toFixed(0)
  261. );
  262. // console.log(this.loadingEndSize, this.pdf_pages, this.currentRate, new Date().getTime())
  263. if (this.loadingEndSize === this.pdf_pages) {
  264. setTimeout(() => {
  265. this.loading = false;
  266. try {
  267. if (this.platform === "ios") {
  268. loadFinished();
  269. }
  270. } catch (e) {
  271. console.error(e);
  272. }
  273. try {
  274. if (this.platform === "ios") {
  275. window.webkit.messageHandlers.loadFinished.postMessage(
  276. "h5_pdf_view"
  277. );
  278. }
  279. } catch (e) {
  280. console.error(e);
  281. }
  282. }, 300);
  283. }
  284. if (this.pdf_pages > num) {
  285. setTimeout(() => {
  286. this._renderPage(num + 1);
  287. }, 10);
  288. }
  289. });
  290. });
  291. },
  292. },
  293. };
  294. </script>
  295. <style scoped>
  296. .home_wrap {
  297. width: 100%;
  298. min-height: 50vh;
  299. position: relative;
  300. }
  301. .home_wrap .show-catalogue {
  302. position: fixed !important;
  303. right: 0 !important;
  304. top: 20% !important;
  305. z-index: 99999999 !important;
  306. height: 20px;
  307. font-size: 14px;
  308. color: #ffffff;
  309. background: rgba(0, 0, 0, 0.5);
  310. display: flex;
  311. align-items: center;
  312. padding: 5px 10px;
  313. }
  314. .home_wrap .pdf_down {
  315. position: absolute;
  316. display: flex;
  317. z-index: 20;
  318. right: 26px;
  319. bottom: 7%;
  320. }
  321. .home_wrap .pdf_down .pdf_set_left {
  322. width: 30px;
  323. height: 40px;
  324. color: #408fff;
  325. font-size: 11px;
  326. padding-top: 25px;
  327. text-align: center;
  328. margin-right: 5px;
  329. cursor: pointer;
  330. }
  331. .home_wrap .pdf_down .pdf_set_middle {
  332. width: 30px;
  333. height: 40px;
  334. color: #408fff;
  335. font-size: 11px;
  336. padding-top: 25px;
  337. text-align: center;
  338. margin-right: 5px;
  339. cursor: pointer;
  340. }
  341. .position-phone {
  342. display: flex;
  343. align-items: center;
  344. position: fixed;
  345. top: 50%;
  346. left: 66.666666%;
  347. color: #ffffff;
  348. background: rgba(0, 0, 0, 0.5);
  349. border-radius: 2.5vw;
  350. height: 5vw;
  351. padding: 0 2vw;
  352. opacity: 0.8;
  353. font-size: 2.5vw;
  354. }
  355. #scrollBox {
  356. display: flex;
  357. flex-direction: column;
  358. align-items: center;
  359. }
  360. .loading {
  361. width: 100%;
  362. height: 100%;
  363. background: #ffffff;
  364. position: absolute;
  365. top: 0;
  366. left: 0;
  367. z-index: 99;
  368. display: flex;
  369. flex-direction: column;
  370. justify-content: center;
  371. align-items: center;
  372. }
  373. .loading-hide {
  374. position: fixed;
  375. left: 200vw;
  376. top: 200vh;
  377. opacity: 0;
  378. }
  379. .catalogue-btn {
  380. color: #ffffff;
  381. font-size: 13px;
  382. font-weight: bold;
  383. width: 72px;
  384. height: 28px;
  385. background-color: #2a63f3;
  386. border-radius: 40px;
  387. display: flex;
  388. justify-content: center;
  389. align-items: center;
  390. position: fixed;
  391. top: 70%;
  392. left: 12px;
  393. z-index: 999;
  394. }
  395. .catalogue-btn > img {
  396. width: 14px;
  397. margin-right: 3px;
  398. }
  399. .canvas-box {
  400. position: relative;
  401. }
  402. .ws {
  403. width: max-content;
  404. transform: translateX(-50%) rotateZ(315deg);
  405. color: rgba(0, 0, 0, 0.2);
  406. font-weight: bold;
  407. pointer-events: none;
  408. position: absolute;
  409. z-index: 9;
  410. top: 50%;
  411. left: 50%;
  412. }
  413. /*const fontSize = parseInt(canvasWrapper.style.width) / 12;*/
  414. /*var cover = document.createElement('div');*/
  415. /*cover.className = "cover";*/
  416. /*cover.style.height = fontSize + 'px';*/
  417. /*cover.style.lineHeight = fontSize + 'px';*/
  418. /*cover.style.position = 'absolute';*/
  419. /*cover.style.left = '50%';*/
  420. /*cover.style.top = `calc(50% - ${fontSize}px)`;*/
  421. /*cover.style.display = 'flex';*/
  422. /*cover.style.alignItems = 'center';*/
  423. /*cover.style.justifyContent = 'center';*/
  424. /*cover.style.transform = 'translateX(-50%)';*/
  425. /*cover.style.pointerEvents = 'none' // 取消所有事件*/
  426. /* let c = document.createElement('div')*/
  427. /*c.className = "cover"*/
  428. /*c.style.width = 'max-content';*/
  429. /*c.style.fontSize = fontSize + 'px';*/
  430. /*c.style.transform = "rotateZ(315deg)";*/
  431. /*c.style.color = "rgba(0, 0, 0, 0.20)";*/
  432. /*c.style.fontWeight = 'bold';*/
  433. /*c.innerText = watermarkText;*/
  434. /*c.style.pointerEvents = 'none' // 取消所有事件*/
  435. /* cover.appendChild(c);*/
  436. /*if (this.annotationLayer?.div) {*/
  437. /* div.insertBefore(textLayerDiv, this.annotationLayer.div);*/
  438. /* div.appendChild(cover);*/
  439. /*} else {*/
  440. /* div.appendChild(textLayerDiv);*/
  441. /* div.appendChild(cover);*/
  442. /* }*/
  443. </style>