puppeteer-core
Version:
A high-level API to control headless Chrome over the DevTools Protocol
635 lines • 18.5 kB
JavaScript
"use strict";
/**
* @license
* Copyright 2017 Google Inc.
* SPDX-License-Identifier: Apache-2.0
*/
Object.defineProperty(exports, "__esModule", { value: true });
exports.BidiTouchscreen = exports.BidiMouse = exports.BidiKeyboard = void 0;
const Input_js_1 = require("../api/Input.js");
const Errors_js_1 = require("../common/Errors.js");
const Errors_js_2 = require("../common/Errors.js");
var SourceActionsType;
(function (SourceActionsType) {
SourceActionsType["None"] = "none";
SourceActionsType["Key"] = "key";
SourceActionsType["Pointer"] = "pointer";
SourceActionsType["Wheel"] = "wheel";
})(SourceActionsType || (SourceActionsType = {}));
var ActionType;
(function (ActionType) {
ActionType["Pause"] = "pause";
ActionType["KeyDown"] = "keyDown";
ActionType["KeyUp"] = "keyUp";
ActionType["PointerUp"] = "pointerUp";
ActionType["PointerDown"] = "pointerDown";
ActionType["PointerMove"] = "pointerMove";
ActionType["Scroll"] = "scroll";
})(ActionType || (ActionType = {}));
const getBidiKeyValue = (key) => {
switch (key) {
case '\r':
case '\n':
key = 'Enter';
break;
}
// Measures the number of code points rather than UTF-16 code units.
if ([...key].length === 1) {
return key;
}
switch (key) {
case 'Cancel':
return '\uE001';
case 'Help':
return '\uE002';
case 'Backspace':
return '\uE003';
case 'Tab':
return '\uE004';
case 'Clear':
return '\uE005';
case 'Enter':
return '\uE007';
case 'Shift':
case 'ShiftLeft':
return '\uE008';
case 'Control':
case 'ControlLeft':
return '\uE009';
case 'Alt':
case 'AltLeft':
return '\uE00A';
case 'Pause':
return '\uE00B';
case 'Escape':
return '\uE00C';
case 'PageUp':
return '\uE00E';
case 'PageDown':
return '\uE00F';
case 'End':
return '\uE010';
case 'Home':
return '\uE011';
case 'ArrowLeft':
return '\uE012';
case 'ArrowUp':
return '\uE013';
case 'ArrowRight':
return '\uE014';
case 'ArrowDown':
return '\uE015';
case 'Insert':
return '\uE016';
case 'Delete':
return '\uE017';
case 'NumpadEqual':
return '\uE019';
case 'Numpad0':
return '\uE01A';
case 'Numpad1':
return '\uE01B';
case 'Numpad2':
return '\uE01C';
case 'Numpad3':
return '\uE01D';
case 'Numpad4':
return '\uE01E';
case 'Numpad5':
return '\uE01F';
case 'Numpad6':
return '\uE020';
case 'Numpad7':
return '\uE021';
case 'Numpad8':
return '\uE022';
case 'Numpad9':
return '\uE023';
case 'NumpadMultiply':
return '\uE024';
case 'NumpadAdd':
return '\uE025';
case 'NumpadSubtract':
return '\uE027';
case 'NumpadDecimal':
return '\uE028';
case 'NumpadDivide':
return '\uE029';
case 'F1':
return '\uE031';
case 'F2':
return '\uE032';
case 'F3':
return '\uE033';
case 'F4':
return '\uE034';
case 'F5':
return '\uE035';
case 'F6':
return '\uE036';
case 'F7':
return '\uE037';
case 'F8':
return '\uE038';
case 'F9':
return '\uE039';
case 'F10':
return '\uE03A';
case 'F11':
return '\uE03B';
case 'F12':
return '\uE03C';
case 'Meta':
case 'MetaLeft':
return '\uE03D';
case 'ShiftRight':
return '\uE050';
case 'ControlRight':
return '\uE051';
case 'AltRight':
return '\uE052';
case 'MetaRight':
return '\uE053';
case 'Digit0':
return '0';
case 'Digit1':
return '1';
case 'Digit2':
return '2';
case 'Digit3':
return '3';
case 'Digit4':
return '4';
case 'Digit5':
return '5';
case 'Digit6':
return '6';
case 'Digit7':
return '7';
case 'Digit8':
return '8';
case 'Digit9':
return '9';
case 'KeyA':
return 'a';
case 'KeyB':
return 'b';
case 'KeyC':
return 'c';
case 'KeyD':
return 'd';
case 'KeyE':
return 'e';
case 'KeyF':
return 'f';
case 'KeyG':
return 'g';
case 'KeyH':
return 'h';
case 'KeyI':
return 'i';
case 'KeyJ':
return 'j';
case 'KeyK':
return 'k';
case 'KeyL':
return 'l';
case 'KeyM':
return 'm';
case 'KeyN':
return 'n';
case 'KeyO':
return 'o';
case 'KeyP':
return 'p';
case 'KeyQ':
return 'q';
case 'KeyR':
return 'r';
case 'KeyS':
return 's';
case 'KeyT':
return 't';
case 'KeyU':
return 'u';
case 'KeyV':
return 'v';
case 'KeyW':
return 'w';
case 'KeyX':
return 'x';
case 'KeyY':
return 'y';
case 'KeyZ':
return 'z';
case 'Semicolon':
return ';';
case 'Equal':
return '=';
case 'Comma':
return ',';
case 'Minus':
return '-';
case 'Period':
return '.';
case 'Slash':
return '/';
case 'Backquote':
return '`';
case 'BracketLeft':
return '[';
case 'Backslash':
return '\\';
case 'BracketRight':
return ']';
case 'Quote':
return '"';
default:
throw new Error(`Unknown key: "${key}"`);
}
};
/**
* @internal
*/
class BidiKeyboard extends Input_js_1.Keyboard {
#page;
constructor(page) {
super();
this.#page = page;
}
async down(key, _options) {
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Key,
id: "__puppeteer_keyboard" /* InputId.Keyboard */,
actions: [
{
type: ActionType.KeyDown,
value: getBidiKeyValue(key),
},
],
},
]);
}
async up(key) {
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Key,
id: "__puppeteer_keyboard" /* InputId.Keyboard */,
actions: [
{
type: ActionType.KeyUp,
value: getBidiKeyValue(key),
},
],
},
]);
}
async press(key, options = {}) {
const { delay = 0 } = options;
const actions = [
{
type: ActionType.KeyDown,
value: getBidiKeyValue(key),
},
];
if (delay > 0) {
actions.push({
type: ActionType.Pause,
duration: delay,
});
}
actions.push({
type: ActionType.KeyUp,
value: getBidiKeyValue(key),
});
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Key,
id: "__puppeteer_keyboard" /* InputId.Keyboard */,
actions,
},
]);
}
async type(text, options = {}) {
const { delay = 0 } = options;
// This spread separates the characters into code points rather than UTF-16
// code units.
const values = [...text].map(getBidiKeyValue);
const actions = [];
if (delay <= 0) {
for (const value of values) {
actions.push({
type: ActionType.KeyDown,
value,
}, {
type: ActionType.KeyUp,
value,
});
}
}
else {
for (const value of values) {
actions.push({
type: ActionType.KeyDown,
value,
}, {
type: ActionType.Pause,
duration: delay,
}, {
type: ActionType.KeyUp,
value,
});
}
}
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Key,
id: "__puppeteer_keyboard" /* InputId.Keyboard */,
actions,
},
]);
}
async sendCharacter(char) {
// Measures the number of code points rather than UTF-16 code units.
if ([...char].length > 1) {
throw new Error('Cannot send more than 1 character.');
}
const frame = await this.#page.focusedFrame();
await frame.isolatedRealm().evaluate(async (char) => {
document.execCommand('insertText', false, char);
}, char);
}
}
exports.BidiKeyboard = BidiKeyboard;
const getBidiButton = (button) => {
switch (button) {
case Input_js_1.MouseButton.Left:
return 0;
case Input_js_1.MouseButton.Middle:
return 1;
case Input_js_1.MouseButton.Right:
return 2;
case Input_js_1.MouseButton.Back:
return 3;
case Input_js_1.MouseButton.Forward:
return 4;
}
};
/**
* @internal
*/
class BidiMouse extends Input_js_1.Mouse {
#page;
#lastMovePoint = { x: 0, y: 0 };
constructor(page) {
super();
this.#page = page;
}
async reset() {
this.#lastMovePoint = { x: 0, y: 0 };
await this.#page.mainFrame().browsingContext.releaseActions();
}
async move(x, y, options = {}) {
const from = this.#lastMovePoint;
const to = {
x: Math.round(x),
y: Math.round(y),
};
const actions = [];
const steps = options.steps ?? 0;
for (let i = 0; i < steps; ++i) {
actions.push({
type: ActionType.PointerMove,
x: from.x + (to.x - from.x) * (i / steps),
y: from.y + (to.y - from.y) * (i / steps),
origin: options.origin,
});
}
actions.push({
type: ActionType.PointerMove,
...to,
origin: options.origin,
});
// https://w3c.github.io/webdriver-bidi/#command-input-performActions:~:text=input.PointerMoveAction%20%3D%20%7B%0A%20%20type%3A%20%22pointerMove%22%2C%0A%20%20x%3A%20js%2Dint%2C
this.#lastMovePoint = to;
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Pointer,
id: "__puppeteer_mouse" /* InputId.Mouse */,
actions,
},
]);
}
async down(options = {}) {
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Pointer,
id: "__puppeteer_mouse" /* InputId.Mouse */,
actions: [
{
type: ActionType.PointerDown,
button: getBidiButton(options.button ?? Input_js_1.MouseButton.Left),
},
],
},
]);
}
async up(options = {}) {
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Pointer,
id: "__puppeteer_mouse" /* InputId.Mouse */,
actions: [
{
type: ActionType.PointerUp,
button: getBidiButton(options.button ?? Input_js_1.MouseButton.Left),
},
],
},
]);
}
async click(x, y, options = {}) {
const actions = [
{
type: ActionType.PointerMove,
x: Math.round(x),
y: Math.round(y),
origin: options.origin,
},
];
const pointerDownAction = {
type: ActionType.PointerDown,
button: getBidiButton(options.button ?? Input_js_1.MouseButton.Left),
};
const pointerUpAction = {
type: ActionType.PointerUp,
button: pointerDownAction.button,
};
for (let i = 1; i < (options.count ?? 1); ++i) {
actions.push(pointerDownAction, pointerUpAction);
}
actions.push(pointerDownAction);
if (options.delay) {
actions.push({
type: ActionType.Pause,
duration: options.delay,
});
}
actions.push(pointerUpAction);
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Pointer,
id: "__puppeteer_mouse" /* InputId.Mouse */,
actions,
},
]);
}
async wheel(options = {}) {
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Wheel,
id: "__puppeteer_wheel" /* InputId.Wheel */,
actions: [
{
type: ActionType.Scroll,
...(this.#lastMovePoint ?? {
x: 0,
y: 0,
}),
deltaX: options.deltaX ?? 0,
deltaY: options.deltaY ?? 0,
},
],
},
]);
}
drag() {
throw new Errors_js_1.UnsupportedOperation();
}
dragOver() {
throw new Errors_js_1.UnsupportedOperation();
}
dragEnter() {
throw new Errors_js_1.UnsupportedOperation();
}
drop() {
throw new Errors_js_1.UnsupportedOperation();
}
dragAndDrop() {
throw new Errors_js_1.UnsupportedOperation();
}
}
exports.BidiMouse = BidiMouse;
/**
* @internal
*/
class BidiTouchHandle {
#started = false;
#x;
#y;
#bidiId;
#page;
#touchScreen;
#properties;
constructor(page, touchScreen, id, x, y, properties) {
this.#page = page;
this.#touchScreen = touchScreen;
this.#x = Math.round(x);
this.#y = Math.round(y);
this.#properties = properties;
this.#bidiId = `${"__puppeteer_finger" /* InputId.Finger */}_${id}`;
}
async start(options = {}) {
if (this.#started) {
throw new Errors_js_2.TouchError('Touch has already started');
}
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Pointer,
id: this.#bidiId,
parameters: {
pointerType: "touch" /* Bidi.Input.PointerType.Touch */,
},
actions: [
{
type: ActionType.PointerMove,
x: this.#x,
y: this.#y,
origin: options.origin,
},
{
...this.#properties,
type: ActionType.PointerDown,
button: 0,
},
],
},
]);
this.#started = true;
}
move(x, y) {
const newX = Math.round(x);
const newY = Math.round(y);
return this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Pointer,
id: this.#bidiId,
parameters: {
pointerType: "touch" /* Bidi.Input.PointerType.Touch */,
},
actions: [
{
...this.#properties,
type: ActionType.PointerMove,
x: newX,
y: newY,
},
],
},
]);
}
async end() {
await this.#page.mainFrame().browsingContext.performActions([
{
type: SourceActionsType.Pointer,
id: this.#bidiId,
parameters: {
pointerType: "touch" /* Bidi.Input.PointerType.Touch */,
},
actions: [
{
type: ActionType.PointerUp,
button: 0,
},
],
},
]);
this.#touchScreen.removeHandle(this);
}
}
/**
* @internal
*/
class BidiTouchscreen extends Input_js_1.Touchscreen {
#page;
constructor(page) {
super();
this.#page = page;
}
async touchStart(x, y, options = {}) {
const id = this.idGenerator();
const properties = {
width: 0.5 * 2, // 2 times default touch radius.
height: 0.5 * 2, // 2 times default touch radius.
pressure: 0.5,
altitudeAngle: Math.PI / 2,
};
const touch = new BidiTouchHandle(this.#page, this, id, x, y, properties);
await touch.start(options);
this.touches.push(touch);
return touch;
}
}
exports.BidiTouchscreen = BidiTouchscreen;
//# sourceMappingURL=Input.js.map