-
Notifications
You must be signed in to change notification settings - Fork 21
Expand file tree
/
Copy pathgetImagesFromPDF.ts
More file actions
78 lines (74 loc) · 2.23 KB
/
getImagesFromPDF.ts
File metadata and controls
78 lines (74 loc) · 2.23 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
import {
getDocument,
GlobalWorkerOptions,
PDFDocumentProxy,
PDFPageProxy,
version,
} from 'pdfjs-dist'
import { RenderParameters } from 'pdfjs-dist/types/src/display/api'
import { MAX_PDF_SCALE, PDF_RESOLUTION } from '@/common/constants'
GlobalWorkerOptions.workerSrc = `//unpkg.com/pdfjs-dist@${version}/legacy/build/pdf.worker.min.mjs`
const getImageFromPage = async (
_document: PDFDocumentProxy,
pageNumber: number,
resolution = PDF_RESOLUTION,
) => {
const page: PDFPageProxy = await _document.getPage(pageNumber)
const canvas = document.createElement('canvas')
const context = canvas.getContext('2d')
const [, , width, height] = page.view
const newScale = (resolution / (height * width)) ** (1 / 2)
const safeScale = Math.min(newScale, MAX_PDF_SCALE)
const viewport = page.getViewport({ scale: safeScale })
canvas.height = viewport.height
canvas.width = viewport.width
const renderContext = {
canvasContext: context,
viewport: viewport,
}
return page
.render(renderContext as RenderParameters)
.promise.then(() => canvas.toDataURL())
.catch((error) => {
throw new Error(error)
})
}
export default function getImagesFromPDF(
file: string,
maxPages = Infinity,
onSuccess?: () => void,
resolution?: number,
) {
return new Promise<string[]>((resolve, reject) => {
getDocument(file)
.promise.then((document: PDFDocumentProxy) => {
if (document.numPages > maxPages) {
const error = new Error('Too many pages')
error.name = 'TooManyPagesError'
reject(error)
return
}
onSuccess?.()
Promise.allSettled(
Array.from(Array(document.numPages).keys()).map((index) =>
getImageFromPage(document, index + 1, resolution),
),
)
.then((results) => {
const images = results.reduce<string[]>((accumulator, result) => {
if (result.status === 'fulfilled') {
accumulator.push(result.value)
}
return accumulator
}, [])
resolve(images)
})
.catch((error) => {
reject(error)
})
})
.catch((error) => {
reject(error)
})
})
}