作者热门文章
- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在关闭 headless 模式运行 puppeteer,以便自动化和远程控制另一台计算机上的可见 Chromium 浏览器。
有没有一种方法可以触发或模拟浏览器上的缩放,就像在 UI 菜单或 ctrl +
中一样?/crtl -
命令?
注入(inject) CSS 或使用各种记录在案的缩放命令并不能完全复制这一点,例如,使用 vh
定义的元素/vw
单位不会调整。
我目前的解决方案
在 Emulation.setDeviceMetricsOverride
中使用视口(viewport)比例缩小效果很好,但它似乎是在调整页面光栅的大小而不是在目标大小上渲染,导致放大时文本模糊。
调整视口(viewport)大小并使用 Emulation.setPageScaleFactor
在放大时效果很好,但是在我的测试中似乎忽略了小于 1 的 pageScaleFactor。
这两种解决方案的一个问题是,它需要提前知道浏览器窗口的宽度/高度,并且依赖于不改变,而不是具有流畅的视口(viewport)。我也不确定我缺少标准浏览器缩放的其他哪些功能。
我的缩放代码现在是:
async applyFrameZoom(page, zoom) {
// page is a puppeteer.Page instance
// zoom is an integer percentage
const session = await page.target().createCDPSession();
let window = await session.send('Browser.getWindowForTarget', {
targetId: page.target()._targetId
});
let width = window.bounds.width;
let height = window.bounds.height;
if (!zoom || zoom === 100) {
// Unset any zoom
await session.send('Emulation.clearDeviceMetricsOverride');
await session.send('Emulation.resetPageScaleFactor');
} else if (zoom > 100) {
// Unset other zooming method
await session.send('Emulation.clearDeviceMetricsOverride');
// Zoom in by reducing size then adjusting page scale (unable to zoom out using this method)
await page.setViewport({
width: Math.round(width / (zoom / 100)),
height: Math.round(height / (zoom / 100))
});
await session.send('Emulation.setPageScaleFactor', {
pageScaleFactor: (zoom / 100)
});
await session.send('Emulation.setVisibleSize', {
width: width,
height: height
});
} else {
// Unset other zooming method
await session.send('Emulation.resetPageScaleFactor');
// Zoom out by emulating a scaled device (makes text blurry when zooming in with this method)
await session.send('Emulation.setDeviceMetricsOverride', {
width: Math.round(width / (zoom / 100)),
height: Math.round(height / (zoom / 100)),
mobile: false,
deviceScaleFactor: 1,
dontSetVisibleSize: true,
viewport: {
x: 0,
y: 0,
width: width,
height: height,
scale: (zoom / 100)
}
});
}
await this.frame.waitForSelector('html');
this.frame.evaluate(function () {
window.dispatchEvent(new Event('resize'));
});
}
最佳答案
--force-device-scale-factor
命令行选项似乎适用于缩放完整的 chrome UI,无论是放大还是缩小。
使用 puppeteer 将其传递给 chrome:
puppeteer.launch({
args: ["--force-device-scale-factor=0.5"],
headless: false,
})
(用 Chrome 78/puppeteer 1.20.0 测试)
chrome-extension/manifest.json
:
{
"name": "my-extension-name",
"description": "A minimal chrome extension to help puppeteer with zooming",
"version": "1.0",
"manifest_version": 2,
"background": {
"scripts": ["background.js"],
"persistent": false
},
"permissions": []
}
chrome-extension/background.js
:
function setZoom(tabId, zoomFactor) {
chrome.tabs.setZoom(tabId, zoomFactor)
}
main.js
:
const puppeteer = require('puppeteer');
(async () => {
const extensionPath = require('path').join(__dirname, 'chrome-extension');
const extensionName = 'my-extension-name';
const browser = await puppeteer.launch({
args: [
`--disable-extensions-except=${extensionPath}`,
`--load-extension=${extensionPath}`
],
headless: false,
});
const page = await browser.newPage();
await page.goto('https://google.com');
// get the background page of the extension
const targets = await browser.targets();
const extenstionPageTarget = targets.find(
(target) => target._targetInfo.title === extensionName
);
const extensionPage = await extenstionPageTarget.page();
// do the zooming by invoking setZoom of background.js
const zoomFactor = 0.5
await extensionPage.evaluate((zoomFactor) => {
// a tabId of undefined defaults to the currently active tab
const tabId = undefined;
setZoom(tabId, zoomFactor);
}, zoomFactor);
})();
我还没有找到获取
tabId
的方法使用 puppeteer 的页面,尽管这可能通过扩展再次成为可能。
关于puppeteer - 有没有办法触发标准缩放(不是 headless ),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57802309/
我是一名优秀的程序员,十分优秀!