UNPKG

puppeteer-core

Version:

A high-level API to control headless Chrome over the DevTools Protocol

635 lines 18.5 kB
"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