-
Notifications
You must be signed in to change notification settings - Fork 306
/
Copy pathelement-to-pdf.js
102 lines (91 loc) · 3.01 KB
/
element-to-pdf.js
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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
/**
* Copyright 2018 Google Inc. All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* @author ebidel@ (Eric Bidelman)
*/
/**
* Takes a screenshot of the latest tweet in a user's timeline and creates a
* PDF of it. Shows how to use Puppeteer to:
*
* 1. screenshot a DOM element
* 2. craft an HTML page on-the-fly
* 3. produce an image of the element and PDF of the page with the image embedded
*
* Usage:
* node element-to-pdf.js
* USERNAME=ChromiumDev node element-to-pdf.js
*
* --searchable makes "find in page" work:
* node element-to-pdf.js --searchable
*
* Output:
* tweet.png and tweet.pdf
*/
const puppeteer = require('puppeteer');
const username = process.env.USERNAME || 'ebidel';
const searchable = process.argv.includes('--searchable');
(async() => {
const browser = await puppeteer.launch();
const page = await browser.newPage();
await page.setViewport({width: 1200, height: 800, deviceScaleFactor: 2});
await page.goto(`https://twitter.com/${username}`);
// Can't use elementHandle.click() because it clicks the center of the element
// with the mouse. On tweets like https://twitter.com/ebidel/status/915996563234631680
// there is an embedded card link to another tweet that it clicks.
await page.$eval(`.tweet[data-screen-name="${username}"]`, tweet => tweet.click());
await page.waitForSelector('.tweet.permalink-tweet', {visible: true});
const overlay = await page.$('.tweet.permalink-tweet');
const screenshot = await overlay.screenshot({path: 'tweet.png'});
if (searchable) {
await page.evaluate(tweet => {
const width = getComputedStyle(tweet).width;
tweet = tweet.cloneNode(true);
tweet.style.width = width;
document.body.innerHTML = `
<div style="display:flex;justify-content:center;align-items:center;height:100vh;">;
${tweet.outerHTML}
</div>
`;
}, overlay);
} else {
await page.setContent(`
<!DOCTYPE html>
<html>
<head>
<style>
html, body {
height: 100vh;
margin: 0;
display: flex;
justify-content: center;
align-items: center;
background: #fafafa;
}
img {
max-width: 60%;
box-shadow: 3px 3px 6px #eee;
border-radius: 6px;
}
</style>
</head>
<body>
<img src="data:img/png;base64,${screenshot.toString('base64')}">
</body>
</html>
`);
}
await page.pdf({path: 'tweet.pdf', printBackground: true});
await browser.close();
})();