mirror of
https://github.com/puppeteer/puppeteer
synced 2024-06-14 14:02:48 +00:00
Mess with lusha's waitFor for fun. (#133)
This commit is contained in:
parent
4adf11427a
commit
49e363c121
@ -422,7 +422,7 @@ class WaitTask {
|
|||||||
constructor(frame, pageScript, timeout) {
|
constructor(frame, pageScript, timeout) {
|
||||||
this._frame = frame;
|
this._frame = frame;
|
||||||
this._pageScript = pageScript;
|
this._pageScript = pageScript;
|
||||||
this._runningTask = null;
|
this._runCount = 0;
|
||||||
frame._waitTasks.add(this);
|
frame._waitTasks.add(this);
|
||||||
this.promise = new Promise((resolve, reject) => {
|
this.promise = new Promise((resolve, reject) => {
|
||||||
this._resolve = resolve;
|
this._resolve = resolve;
|
||||||
@ -438,30 +438,34 @@ class WaitTask {
|
|||||||
* @param {!Error} error
|
* @param {!Error} error
|
||||||
*/
|
*/
|
||||||
terminate(error) {
|
terminate(error) {
|
||||||
|
this._terminated = true;
|
||||||
this._reject(error);
|
this._reject(error);
|
||||||
this._cleanup();
|
this._cleanup();
|
||||||
}
|
}
|
||||||
|
|
||||||
rerun() {
|
async rerun() {
|
||||||
let runningTask = this._frame._evaluateExpression(this._pageScript, true).then(finish.bind(this), finish.bind(this, false));
|
const runCount = ++this._runCount;
|
||||||
this._runningTask = runningTask;
|
let success = false;
|
||||||
|
let error = null;
|
||||||
/**
|
try {
|
||||||
* @param {boolean} success
|
success = await this._frame._evaluateExpression(this._pageScript, true);
|
||||||
* @param {?Error=} error
|
} catch (e) {
|
||||||
*/
|
error = e;
|
||||||
function finish(success, error) {
|
|
||||||
if (runningTask !== this._runningTask)
|
|
||||||
return;
|
|
||||||
// Ignore timeouts in pageScript - we track timeouts ourselves.
|
|
||||||
if (!success && !error)
|
|
||||||
return;
|
|
||||||
if (error)
|
|
||||||
this._reject(error);
|
|
||||||
else
|
|
||||||
this._resolve();
|
|
||||||
this._cleanup();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this._terminated || runCount !== this._runCount)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Ignore timeouts in pageScript - we track timeouts ourselves.
|
||||||
|
if (!success && !error)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (error)
|
||||||
|
this._reject(error);
|
||||||
|
else
|
||||||
|
this._resolve();
|
||||||
|
|
||||||
|
this._cleanup();
|
||||||
}
|
}
|
||||||
|
|
||||||
_cleanup() {
|
_cleanup() {
|
||||||
@ -477,30 +481,28 @@ class WaitTask {
|
|||||||
* @param {number} timeout
|
* @param {number} timeout
|
||||||
* @return {!Promise<boolean>}
|
* @return {!Promise<boolean>}
|
||||||
*/
|
*/
|
||||||
function waitForSelectorPageFunction(selector, visible, timeout) {
|
async function waitForSelectorPageFunction(selector, visible, timeout) {
|
||||||
const resultPromise = visible ? waitForVisible(selector) : waitInDOM(selector);
|
let timedOut = false;
|
||||||
const timeoutPromise = new Promise(fulfill => setTimeout(fulfill, timeout));
|
setTimeout(() => timedOut = true, timeout);
|
||||||
return Promise.race([
|
await waitForDOM();
|
||||||
resultPromise.then(() => true),
|
await waitForVisible();
|
||||||
timeoutPromise.then(() => false)
|
return !timedOut;
|
||||||
]);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {string} selector
|
|
||||||
* @return {!Promise<!Element>}
|
* @return {!Promise<!Element>}
|
||||||
*/
|
*/
|
||||||
function waitInDOM(selector) {
|
function waitForDOM() {
|
||||||
let node = document.querySelector(selector);
|
let node = document.querySelector(selector);
|
||||||
if (node)
|
if (node)
|
||||||
return Promise.resolve(node);
|
return Promise.resolve();
|
||||||
|
|
||||||
let fulfill;
|
let fulfill;
|
||||||
const result = new Promise(x => fulfill = x);
|
const result = new Promise(x => fulfill = x);
|
||||||
const observer = new MutationObserver(mutations => {
|
const observer = new MutationObserver(mutations => {
|
||||||
const node = document.querySelector(selector);
|
const node = document.querySelector(selector);
|
||||||
if (node) {
|
if (node || timedOut) {
|
||||||
observer.disconnect();
|
observer.disconnect();
|
||||||
fulfill(node);
|
fulfill();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
observer.observe(document, {
|
observer.observe(document, {
|
||||||
@ -511,23 +513,26 @@ function waitForSelectorPageFunction(selector, visible, timeout) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {string} selector
|
|
||||||
* @return {!Promise<!Element>}
|
* @return {!Promise<!Element>}
|
||||||
*/
|
*/
|
||||||
async function waitForVisible(selector) {
|
function waitForVisible() {
|
||||||
let fulfill;
|
let fulfill;
|
||||||
const result = new Promise(x => fulfill = x);
|
const result = new Promise(x => fulfill = x);
|
||||||
onRaf();
|
onRaf();
|
||||||
return result;
|
return result;
|
||||||
|
|
||||||
async function onRaf() {
|
function onRaf() {
|
||||||
const node = await waitInDOM(selector);
|
if (timedOut) {
|
||||||
const style = window.getComputedStyle(node);
|
fulfill();
|
||||||
if (style.display === 'none' || style.visibility === 'hidden') {
|
return;
|
||||||
|
}
|
||||||
|
const node = document.querySelector(selector);
|
||||||
|
const style = node ? window.getComputedStyle(node) : null;
|
||||||
|
if (!style || style.display === 'none' || style.visibility === 'hidden') {
|
||||||
requestAnimationFrame(onRaf);
|
requestAnimationFrame(onRaf);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
fulfill(node);
|
fulfill();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user