2018-11-12 12:11:06 +00:00
|
|
|
(function() {
|
|
|
|
// Set up the M object - only pending_js is implemented.
|
|
|
|
window.M = window.M ? window.M : {};
|
2021-11-24 09:47:10 +00:00
|
|
|
const M = window.M;
|
2018-11-12 12:11:06 +00:00
|
|
|
M.util = M.util ? M.util : {};
|
|
|
|
M.util.pending_js = M.util.pending_js ? M.util.pending_js : []; // eslint-disable-line camelcase
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Logs information from this Behat runtime JavaScript, including the time and the 'BEHAT'
|
|
|
|
* keyword so we can easily filter for it if needed.
|
|
|
|
*
|
|
|
|
* @param {string} text Information to log
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const log = function() {
|
|
|
|
const now = new Date();
|
|
|
|
const nowFormatted = String(now.getHours()).padStart(2, '0') + ':' +
|
2018-11-12 12:11:06 +00:00
|
|
|
String(now.getMinutes()).padStart(2, '0') + ':' +
|
|
|
|
String(now.getSeconds()).padStart(2, '0') + '.' +
|
|
|
|
String(now.getMilliseconds()).padStart(2, '0');
|
2021-06-03 09:20:36 +00:00
|
|
|
console.log('BEHAT: ' + nowFormatted, ...arguments); // eslint-disable-line no-console
|
2018-11-12 12:11:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Run after several setTimeouts to ensure queued events are finished.
|
|
|
|
*
|
|
|
|
* @param {function} target function to run
|
|
|
|
* @param {number} count Number of times to do setTimeout (leave blank for 10)
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const runAfterEverything = function(target, count) {
|
2018-11-12 12:11:06 +00:00
|
|
|
if (count === undefined) {
|
|
|
|
count = 10;
|
|
|
|
}
|
|
|
|
setTimeout(function() {
|
|
|
|
count--;
|
|
|
|
if (count == 0) {
|
|
|
|
target();
|
|
|
|
} else {
|
|
|
|
runAfterEverything(target, count);
|
|
|
|
}
|
|
|
|
}, 0);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds a pending key to the array.
|
|
|
|
*
|
|
|
|
* @param {string} key Key to add
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const addPending = function(key) {
|
2018-11-12 12:11:06 +00:00
|
|
|
// Add a special DELAY entry whenever another entry is added.
|
|
|
|
if (window.M.util.pending_js.length == 0) {
|
|
|
|
window.M.util.pending_js.push('DELAY');
|
|
|
|
}
|
|
|
|
window.M.util.pending_js.push(key);
|
|
|
|
|
|
|
|
log('PENDING+: ' + window.M.util.pending_js);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes a pending key from the array. If this would clear the array, the actual clear only
|
|
|
|
* takes effect after the queued events are finished.
|
|
|
|
*
|
|
|
|
* @param {string} key Key to remove
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const removePending = function(key) {
|
2018-11-12 12:11:06 +00:00
|
|
|
// Remove the key immediately.
|
|
|
|
window.M.util.pending_js = window.M.util.pending_js.filter(function(x) { // eslint-disable-line camelcase
|
|
|
|
return x !== key;
|
|
|
|
});
|
|
|
|
log('PENDING-: ' + window.M.util.pending_js);
|
|
|
|
|
|
|
|
// If the only thing left is DELAY, then remove that as well, later...
|
|
|
|
if (window.M.util.pending_js.length === 1) {
|
|
|
|
runAfterEverything(function() {
|
|
|
|
// Check there isn't a spinner...
|
2020-01-22 08:56:01 +00:00
|
|
|
checkUIBlocked();
|
2018-11-12 12:11:06 +00:00
|
|
|
|
|
|
|
// Only remove it if the pending array is STILL empty after all that.
|
|
|
|
if (window.M.util.pending_js.length === 1) {
|
|
|
|
window.M.util.pending_js = []; // eslint-disable-line camelcase
|
|
|
|
log('PENDING-: ' + window.M.util.pending_js);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds a pending key to the array, but removes it after some setTimeouts finish.
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const addPendingDelay = function() {
|
2018-11-12 12:11:06 +00:00
|
|
|
addPending('...');
|
|
|
|
removePending('...');
|
|
|
|
};
|
|
|
|
|
|
|
|
// Override XMLHttpRequest to mark things pending while there is a request waiting.
|
2021-11-24 09:47:10 +00:00
|
|
|
const realOpen = XMLHttpRequest.prototype.open;
|
|
|
|
let requestIndex = 0;
|
2018-11-12 12:11:06 +00:00
|
|
|
XMLHttpRequest.prototype.open = function() {
|
2021-11-24 09:47:10 +00:00
|
|
|
const index = requestIndex++;
|
|
|
|
const key = 'httprequest-' + index;
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-11-10 14:59:32 +00:00
|
|
|
try {
|
|
|
|
// Add to the list of pending requests.
|
|
|
|
addPending(key);
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-11-10 14:59:32 +00:00
|
|
|
// Detect when it finishes and remove it from the list.
|
|
|
|
this.addEventListener('loadend', function() {
|
|
|
|
removePending(key);
|
|
|
|
});
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-11-10 14:59:32 +00:00
|
|
|
return realOpen.apply(this, arguments);
|
2021-11-24 09:47:10 +00:00
|
|
|
} catch (error) {
|
2021-11-10 14:59:32 +00:00
|
|
|
removePending(key);
|
2021-11-24 09:47:10 +00:00
|
|
|
throw error;
|
2021-11-10 14:59:32 +00:00
|
|
|
}
|
2018-11-12 12:11:06 +00:00
|
|
|
};
|
|
|
|
|
2021-11-24 09:47:10 +00:00
|
|
|
let waitingBlocked = false;
|
2018-11-12 12:11:06 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if a loading spinner is present and visible; if so, adds it to the pending array
|
|
|
|
* (and if not, removes it).
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const checkUIBlocked = function() {
|
|
|
|
const blocked = document.querySelector('span.core-loading-spinner, ion-loading, .click-block-active');
|
2020-01-22 08:56:01 +00:00
|
|
|
if (blocked && blocked.offsetParent) {
|
|
|
|
if (!waitingBlocked) {
|
|
|
|
addPending('blocked');
|
|
|
|
waitingBlocked = true;
|
2018-11-12 12:11:06 +00:00
|
|
|
}
|
|
|
|
} else {
|
2020-01-22 08:56:01 +00:00
|
|
|
if (waitingBlocked) {
|
|
|
|
removePending('blocked');
|
|
|
|
waitingBlocked = false;
|
2018-11-12 12:11:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// It would be really beautiful if you could detect CSS transitions and animations, that would
|
|
|
|
// cover almost everything, but sadly there is no way to do this because the transitionstart
|
|
|
|
// and animationcancel events are not implemented in Chrome, so we cannot detect either of
|
|
|
|
// these reliably. Instead, we have to look for any DOM changes and do horrible polling. Most
|
|
|
|
// of the animations are set to 500ms so we allow it to continue from 500ms after any DOM
|
|
|
|
// change.
|
|
|
|
|
2021-11-24 09:47:10 +00:00
|
|
|
let recentMutation = false;
|
|
|
|
let lastMutation;
|
2018-11-12 12:11:06 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Called from the mutation callback to remove the pending tag after 500ms if nothing else
|
|
|
|
* gets mutated.
|
|
|
|
*
|
|
|
|
* This will be called after 500ms, then every 100ms until there have been no mutation events
|
|
|
|
* for 500ms.
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const pollRecentMutation = function() {
|
2018-11-12 12:11:06 +00:00
|
|
|
if (Date.now() - lastMutation > 500) {
|
|
|
|
recentMutation = false;
|
|
|
|
removePending('dom-mutation');
|
|
|
|
} else {
|
|
|
|
setTimeout(pollRecentMutation, 100);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Mutation callback, called whenever the DOM is mutated.
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const mutationCallback = function() {
|
2018-11-12 12:11:06 +00:00
|
|
|
lastMutation = Date.now();
|
|
|
|
if (!recentMutation) {
|
|
|
|
recentMutation = true;
|
|
|
|
addPending('dom-mutation');
|
|
|
|
setTimeout(pollRecentMutation, 500);
|
|
|
|
}
|
|
|
|
// Also update the spinner presence if needed.
|
2020-01-22 08:56:01 +00:00
|
|
|
checkUIBlocked();
|
2018-11-12 12:11:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Set listener using the mutation callback.
|
2021-11-24 09:47:10 +00:00
|
|
|
const observer = new MutationObserver(mutationCallback);
|
2018-11-12 12:11:06 +00:00
|
|
|
observer.observe(document, {attributes: true, childList: true, subtree: true});
|
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
/**
|
|
|
|
* Check if an element is visible.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Element
|
|
|
|
* @param {HTMLElement} container Container
|
|
|
|
* @returns {boolean} Whether the element is visible or not
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const isElementVisible = (element, container) => {
|
2021-04-27 15:43:47 +00:00
|
|
|
if (element.getAttribute('aria-hidden') === 'true' || getComputedStyle(element).display === 'none')
|
|
|
|
return false;
|
|
|
|
|
2021-06-03 09:20:36 +00:00
|
|
|
const parentElement = getParentElement(element);
|
|
|
|
if (parentElement === container)
|
2021-04-27 15:43:47 +00:00
|
|
|
return true;
|
|
|
|
|
2021-06-03 09:20:36 +00:00
|
|
|
if (!parentElement)
|
2021-04-27 15:43:47 +00:00
|
|
|
return false;
|
|
|
|
|
2021-06-03 09:20:36 +00:00
|
|
|
return isElementVisible(parentElement, container);
|
2021-04-27 15:43:47 +00:00
|
|
|
};
|
|
|
|
|
2021-04-29 11:40:42 +00:00
|
|
|
/**
|
|
|
|
* Check if an element is selected.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Element
|
|
|
|
* @param {HTMLElement} container Container
|
|
|
|
* @returns {boolean} Whether the element is selected or not
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const isElementSelected = (element, container) => {
|
2021-04-29 11:40:42 +00:00
|
|
|
const ariaCurrent = element.getAttribute('aria-current');
|
2021-05-11 16:54:18 +00:00
|
|
|
if (
|
|
|
|
(ariaCurrent && ariaCurrent !== 'false') ||
|
2021-05-18 13:53:31 +00:00
|
|
|
(element.getAttribute('aria-selected') === 'true') ||
|
|
|
|
(element.getAttribute('aria-checked') === 'true')
|
2021-05-11 16:54:18 +00:00
|
|
|
)
|
2021-04-29 11:40:42 +00:00
|
|
|
return true;
|
|
|
|
|
2021-06-03 09:20:36 +00:00
|
|
|
const parentElement = getParentElement(element);
|
|
|
|
if (!parentElement || parentElement === container)
|
2021-04-29 11:40:42 +00:00
|
|
|
return false;
|
|
|
|
|
2021-06-03 09:20:36 +00:00
|
|
|
return isElementSelected(parentElement, container);
|
2021-04-29 11:40:42 +00:00
|
|
|
};
|
|
|
|
|
2018-11-12 12:11:06 +00:00
|
|
|
/**
|
2021-05-11 16:54:18 +00:00
|
|
|
* Finds elements within a given container.
|
2018-11-12 12:11:06 +00:00
|
|
|
*
|
2021-04-27 15:43:47 +00:00
|
|
|
* @param {HTMLElement} container Parent element to search the element within
|
|
|
|
* @param {string} text Text to look for
|
2021-05-11 16:54:18 +00:00
|
|
|
* @return {HTMLElement} Elements containing the given text
|
2018-11-12 12:11:06 +00:00
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const findElementsBasedOnTextWithin = (container, text) => {
|
2021-05-11 16:54:18 +00:00
|
|
|
const elements = [];
|
2021-04-27 15:43:47 +00:00
|
|
|
const attributesSelector = `[aria-label*="${text}"], a[title*="${text}"], img[alt*="${text}"]`;
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
for (const foundByAttributes of container.querySelectorAll(attributesSelector)) {
|
2021-05-11 16:54:18 +00:00
|
|
|
if (!isElementVisible(foundByAttributes, container))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
elements.push(foundByAttributes);
|
2021-04-27 15:43:47 +00:00
|
|
|
}
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
const treeWalker = document.createTreeWalker(
|
|
|
|
container,
|
|
|
|
NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_DOCUMENT_FRAGMENT | NodeFilter.SHOW_TEXT,
|
|
|
|
{
|
|
|
|
acceptNode: node => {
|
|
|
|
if (
|
|
|
|
node instanceof HTMLStyleElement ||
|
|
|
|
node instanceof HTMLLinkElement ||
|
|
|
|
node instanceof HTMLScriptElement
|
|
|
|
)
|
|
|
|
return NodeFilter.FILTER_REJECT;
|
|
|
|
|
|
|
|
if (
|
|
|
|
node instanceof HTMLElement && (
|
|
|
|
node.getAttribute('aria-hidden') === 'true' || getComputedStyle(node).display === 'none'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
return NodeFilter.FILTER_REJECT;
|
|
|
|
|
|
|
|
return NodeFilter.FILTER_ACCEPT;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
let currentNode;
|
|
|
|
while (currentNode = treeWalker.nextNode()) {
|
|
|
|
if (currentNode instanceof Text) {
|
|
|
|
if (currentNode.textContent.includes(text)) {
|
2021-05-11 16:54:18 +00:00
|
|
|
elements.push(currentNode.parentElement);
|
2018-11-12 12:11:06 +00:00
|
|
|
}
|
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
continue;
|
2018-11-12 12:11:06 +00:00
|
|
|
}
|
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
const labelledBy = currentNode.getAttribute('aria-labelledby');
|
2021-07-06 16:59:54 +00:00
|
|
|
const labelElement = labelledBy && container.querySelector(`#${labelledBy}`);
|
|
|
|
if (labelElement && labelElement.innerText && labelElement.innerText.includes(text)) {
|
2021-05-11 16:54:18 +00:00
|
|
|
elements.push(currentNode);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
if (currentNode.shadowRoot) {
|
|
|
|
for (const childNode of currentNode.shadowRoot.childNodes) {
|
2021-05-06 11:06:25 +00:00
|
|
|
if (
|
|
|
|
!(childNode instanceof HTMLElement) || (
|
|
|
|
childNode instanceof HTMLStyleElement ||
|
|
|
|
childNode instanceof HTMLLinkElement ||
|
|
|
|
childNode instanceof HTMLScriptElement
|
|
|
|
)
|
|
|
|
) {
|
2021-04-27 15:43:47 +00:00
|
|
|
continue;
|
2018-11-12 12:11:06 +00:00
|
|
|
}
|
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
if (childNode.matches(attributesSelector)) {
|
2021-05-11 16:54:18 +00:00
|
|
|
elements.push(childNode);
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-05-11 16:54:18 +00:00
|
|
|
continue;
|
2021-04-27 15:43:47 +00:00
|
|
|
}
|
2021-05-11 16:54:18 +00:00
|
|
|
|
|
|
|
elements.push(...findElementsBasedOnTextWithin(childNode, text));
|
2021-04-27 15:43:47 +00:00
|
|
|
}
|
2018-11-12 12:11:06 +00:00
|
|
|
}
|
2021-04-27 15:43:47 +00:00
|
|
|
}
|
2021-05-11 16:54:18 +00:00
|
|
|
|
|
|
|
return elements;
|
2021-04-27 15:43:47 +00:00
|
|
|
};
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-05-27 16:11:52 +00:00
|
|
|
/**
|
|
|
|
* Given a list of elements, get the top ancestors among all of them.
|
|
|
|
*
|
|
|
|
* This will remote duplicates and drop any elements nested within each other.
|
|
|
|
*
|
|
|
|
* @param {Array} elements Elements list.
|
|
|
|
* @return {Array} Top ancestors.
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const getTopAncestors = function(elements) {
|
2021-05-27 16:11:52 +00:00
|
|
|
const uniqueElements = new Set(elements);
|
|
|
|
|
|
|
|
for (const element of uniqueElements) {
|
|
|
|
for (otherElement of uniqueElements) {
|
|
|
|
if (otherElement === element) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (element.contains(otherElement)) {
|
|
|
|
uniqueElements.delete(otherElement);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-09 15:06:13 +00:00
|
|
|
return Array.from(uniqueElements);
|
2021-05-27 16:11:52 +00:00
|
|
|
};
|
|
|
|
|
2021-06-03 09:20:36 +00:00
|
|
|
/**
|
|
|
|
* Get parent element, including Shadow DOM parents.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Element.
|
|
|
|
* @return {HTMLElement} Parent element.
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const getParentElement = function(element) {
|
2021-07-06 16:59:54 +00:00
|
|
|
return element.parentElement || (element.getRootNode() && element.getRootNode().host) || null;
|
2021-06-03 09:20:36 +00:00
|
|
|
};
|
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
/**
|
2021-05-11 16:54:18 +00:00
|
|
|
* Function to find elements based on their text or Aria label.
|
2021-04-27 15:43:47 +00:00
|
|
|
*
|
2021-06-03 09:20:36 +00:00
|
|
|
* @param {object} locator Element locator.
|
2021-12-01 13:24:43 +00:00
|
|
|
* @param {boolean} insideSplitView Whether to search only inside the split view contents.
|
2021-05-11 16:54:18 +00:00
|
|
|
* @return {HTMLElement} Found elements
|
2021-04-27 15:43:47 +00:00
|
|
|
*/
|
2021-12-01 13:24:43 +00:00
|
|
|
const findElementsBasedOnText = function(locator, insideSplitView) {
|
|
|
|
let topContainer = document.querySelector('ion-alert, ion-popover, ion-action-sheet, core-ion-tab.show-tab ion-page.show-page, ion-page.show-page, html');
|
|
|
|
|
|
|
|
if (insideSplitView) {
|
|
|
|
topContainer = topContainer.querySelector('core-split-view ion-router-outlet');
|
|
|
|
}
|
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
let container = topContainer;
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-06-03 09:20:36 +00:00
|
|
|
if (topContainer && locator.near) {
|
|
|
|
const nearElements = findElementsBasedOnText(locator.near);
|
2021-04-27 15:43:47 +00:00
|
|
|
|
2021-05-11 16:54:18 +00:00
|
|
|
if (nearElements.length === 0) {
|
|
|
|
throw new Error('There was no match for near text')
|
|
|
|
} else if (nearElements.length > 1) {
|
2021-05-27 16:11:52 +00:00
|
|
|
const nearElementsAncestors = getTopAncestors(nearElements);
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-05-27 16:11:52 +00:00
|
|
|
if (nearElementsAncestors.length > 1) {
|
|
|
|
throw new Error('Too many matches for near text');
|
|
|
|
}
|
|
|
|
|
2021-06-03 09:20:36 +00:00
|
|
|
container = getParentElement(nearElementsAncestors[0]);
|
2021-05-27 16:11:52 +00:00
|
|
|
} else {
|
2021-06-03 09:20:36 +00:00
|
|
|
container = getParentElement(nearElements[0]);
|
2021-05-27 16:11:52 +00:00
|
|
|
}
|
2018-11-12 12:11:06 +00:00
|
|
|
}
|
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
do {
|
2021-06-03 09:20:36 +00:00
|
|
|
const elements = findElementsBasedOnTextWithin(container, locator.text);
|
|
|
|
const filteredElements = locator.selector
|
|
|
|
? elements.filter(element => element.matches(locator.selector))
|
|
|
|
: elements;
|
2021-04-27 15:43:47 +00:00
|
|
|
|
2021-06-03 09:20:36 +00:00
|
|
|
if (filteredElements.length > 0) {
|
|
|
|
return filteredElements;
|
2021-04-27 15:43:47 +00:00
|
|
|
}
|
2021-12-01 13:24:43 +00:00
|
|
|
} while (container !== topContainer && (container = getParentElement(container)) && container !== topContainer);
|
2021-05-11 16:54:18 +00:00
|
|
|
|
|
|
|
return [];
|
2018-11-12 12:11:06 +00:00
|
|
|
};
|
|
|
|
|
2021-06-03 09:21:50 +00:00
|
|
|
/**
|
|
|
|
* Press an element.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Element to press.
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const pressElement = function(element) {
|
|
|
|
// Scroll the item into view.
|
|
|
|
element.scrollIntoView(false);
|
|
|
|
|
|
|
|
const rect = element.getBoundingClientRect();
|
|
|
|
|
|
|
|
// Simulate a mouse click on the button.
|
|
|
|
const eventOptions = {
|
|
|
|
clientX: rect.left + rect.width / 2,
|
|
|
|
clientY: rect.top + rect.height / 2,
|
|
|
|
bubbles: true,
|
|
|
|
view: window,
|
|
|
|
cancelable: true,
|
|
|
|
};
|
|
|
|
|
|
|
|
// Events don't bubble up across Shadow DOM boundaries, and some buttons
|
|
|
|
// may not work without doing this.
|
|
|
|
const parentElement = getParentElement(element);
|
2021-06-03 09:21:50 +00:00
|
|
|
|
2021-11-24 09:47:10 +00:00
|
|
|
if (parentElement && parentElement.matches('ion-button, ion-back-button')) {
|
|
|
|
element = parentElement;
|
|
|
|
}
|
2021-06-03 09:21:50 +00:00
|
|
|
|
2021-11-24 09:47:10 +00:00
|
|
|
// There are some buttons in the app that don't respond to click events, for example
|
|
|
|
// buttons using the core-supress-events directive. That's why we need to send both
|
|
|
|
// click and mouse events.
|
|
|
|
element.dispatchEvent(new MouseEvent('mousedown', eventOptions));
|
2021-06-03 09:21:50 +00:00
|
|
|
|
2021-11-24 09:47:10 +00:00
|
|
|
setTimeout(() => {
|
|
|
|
element.dispatchEvent(new MouseEvent('mouseup', eventOptions));
|
|
|
|
element.click();
|
|
|
|
}, 300);
|
2021-06-03 09:21:50 +00:00
|
|
|
|
|
|
|
// Mark busy until the button click finishes processing.
|
|
|
|
addPendingDelay();
|
|
|
|
};
|
|
|
|
|
2018-11-12 12:11:06 +00:00
|
|
|
/**
|
|
|
|
* Function to find and click an app standard button.
|
|
|
|
*
|
|
|
|
* @param {string} button Type of button to press
|
|
|
|
* @return {string} OK if successful, or ERROR: followed by message
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const behatPressStandard = function(button) {
|
2018-11-12 12:11:06 +00:00
|
|
|
log('Action - Click standard button: ' + button);
|
2021-04-29 11:40:42 +00:00
|
|
|
|
|
|
|
// Find button
|
2021-11-24 09:47:10 +00:00
|
|
|
let foundButton = null;
|
|
|
|
|
|
|
|
switch (button) {
|
|
|
|
case 'back':
|
|
|
|
foundButton = findElementsBasedOnText({ text: 'Back' })[0];
|
|
|
|
break;
|
2021-11-24 09:47:10 +00:00
|
|
|
case 'main menu': // Deprecated name.
|
|
|
|
case 'more menu':
|
2021-11-24 09:47:10 +00:00
|
|
|
foundButton = findElementsBasedOnText({
|
|
|
|
text: 'More',
|
2021-11-29 09:01:55 +00:00
|
|
|
near: { text: 'Messages' },
|
2021-11-24 09:47:10 +00:00
|
|
|
})[0];
|
|
|
|
break;
|
2021-11-24 09:47:10 +00:00
|
|
|
case 'user menu' :
|
2021-11-24 09:47:10 +00:00
|
|
|
foundButton = findElementsBasedOnText({ text: 'Account' })[0];
|
|
|
|
break;
|
|
|
|
case 'page menu':
|
|
|
|
foundButton = findElementsBasedOnText({ text: 'Display options' })[0];
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 'ERROR: Unsupported standard button type';
|
2018-11-12 12:11:06 +00:00
|
|
|
}
|
2021-04-29 11:40:42 +00:00
|
|
|
|
|
|
|
// Click button
|
2021-06-03 09:21:50 +00:00
|
|
|
pressElement(foundButton);
|
2018-11-12 12:11:06 +00:00
|
|
|
|
|
|
|
return 'OK';
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When there is a popup, clicks on the backdrop.
|
|
|
|
*
|
|
|
|
* @return {string} OK if successful, or ERROR: followed by message
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const behatClosePopup = function() {
|
2018-11-12 12:11:06 +00:00
|
|
|
log('Action - Close popup');
|
|
|
|
|
2021-11-24 09:47:10 +00:00
|
|
|
let backdrops = Array.from(document.querySelectorAll('ion-backdrop'));
|
|
|
|
backdrops = backdrops.filter(function(backdrop) {
|
|
|
|
return !!backdrop.offsetParent;
|
2018-11-12 12:11:06 +00:00
|
|
|
});
|
2021-11-24 09:47:10 +00:00
|
|
|
|
|
|
|
if (!backdrops.length) {
|
2018-11-12 12:11:06 +00:00
|
|
|
return 'ERROR: Could not find backdrop';
|
|
|
|
}
|
2021-11-24 09:47:10 +00:00
|
|
|
if (backdrops.length > 1) {
|
2018-11-12 12:11:06 +00:00
|
|
|
return 'ERROR: Found too many backdrops';
|
|
|
|
}
|
2021-11-24 09:47:10 +00:00
|
|
|
const backdrop = backdrops[0];
|
|
|
|
backdrop.click();
|
2018-11-12 12:11:06 +00:00
|
|
|
|
|
|
|
// Mark busy until the click finishes processing.
|
|
|
|
addPendingDelay();
|
|
|
|
|
|
|
|
return 'OK';
|
|
|
|
};
|
|
|
|
|
2021-04-27 15:43:47 +00:00
|
|
|
/**
|
2021-12-01 13:24:43 +00:00
|
|
|
* Function to find an arbitrary element based on its text or aria label.
|
2021-04-27 15:43:47 +00:00
|
|
|
*
|
2021-06-03 09:20:36 +00:00
|
|
|
* @param {object} locator Element locator.
|
2021-12-01 13:24:43 +00:00
|
|
|
* @param {boolean} insideSplitView Whether to search only inside the split view contents.
|
2021-04-27 15:43:47 +00:00
|
|
|
* @return {string} OK if successful, or ERROR: followed by message
|
|
|
|
*/
|
2021-12-01 13:24:43 +00:00
|
|
|
const behatFind = function(locator, insideSplitView) {
|
|
|
|
log('Action - Find', { locator, insideSplitView });
|
2021-04-27 15:43:47 +00:00
|
|
|
|
|
|
|
try {
|
2021-12-01 13:24:43 +00:00
|
|
|
const element = findElementsBasedOnText(locator, insideSplitView)[0];
|
2021-04-27 15:43:47 +00:00
|
|
|
|
|
|
|
if (!element) {
|
|
|
|
return 'ERROR: No matches for text';
|
|
|
|
}
|
|
|
|
|
2021-12-01 13:24:43 +00:00
|
|
|
log('Action - Found', { locator, insideSplitView, element });
|
2021-04-27 15:43:47 +00:00
|
|
|
return 'OK';
|
|
|
|
} catch (error) {
|
|
|
|
return 'ERROR: ' + error.message;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-11-25 12:04:25 +00:00
|
|
|
/**
|
|
|
|
* Scroll an element into view.
|
|
|
|
*
|
|
|
|
* @param {object} locator Element locator.
|
|
|
|
* @return {string} OK if successful, or ERROR: followed by message
|
|
|
|
*/
|
|
|
|
const behatScrollTo = function(locator) {
|
|
|
|
log('Action - scrollTo', { locator });
|
|
|
|
|
|
|
|
try {
|
|
|
|
let element = findElementsBasedOnText(locator)[0];
|
|
|
|
|
|
|
|
if (!element) {
|
|
|
|
return 'ERROR: No matches for text';
|
|
|
|
}
|
|
|
|
|
|
|
|
element = element.closest('ion-item') ?? element.closest('button') ?? element;
|
|
|
|
|
|
|
|
element.scrollIntoView();
|
|
|
|
|
|
|
|
log('Action - Scrolled to', { locator, element });
|
|
|
|
return 'OK';
|
|
|
|
} catch (error) {
|
|
|
|
return 'ERROR: ' + error.message;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-01 13:12:15 +00:00
|
|
|
/**
|
|
|
|
* Load more items form an active list with infinite loader.
|
|
|
|
*
|
|
|
|
* @return {string} OK if successful, or ERROR: followed by message
|
|
|
|
*/
|
|
|
|
const behatLoadMoreItems = async function() {
|
|
|
|
log('Action - loadMoreItems');
|
|
|
|
|
|
|
|
try {
|
|
|
|
const infiniteLoading = Array
|
|
|
|
.from(document.querySelectorAll('core-infinite-loading'))
|
|
|
|
.find(element => !element.closest('.ion-page-hidden'));
|
|
|
|
|
|
|
|
if (!infiniteLoading) {
|
|
|
|
return 'ERROR: There isn\'t an infinite loader in the current page';
|
|
|
|
}
|
|
|
|
|
|
|
|
const initialOffset = infiniteLoading.offsetTop;
|
|
|
|
const isLoading = () => !!infiniteLoading.querySelector('ion-spinner[aria-label]');
|
|
|
|
const isCompleted = () => !isLoading() && !infiniteLoading.querySelector('ion-button');
|
|
|
|
const hasMoved = () => infiniteLoading.offsetTop !== initialOffset;
|
|
|
|
|
|
|
|
if (isCompleted()) {
|
|
|
|
return 'ERROR: All items are already loaded';
|
|
|
|
}
|
|
|
|
|
|
|
|
infiniteLoading.scrollIntoView();
|
|
|
|
|
|
|
|
// Wait 100ms
|
|
|
|
await new Promise(resolve => setTimeout(resolve, 100));
|
|
|
|
|
|
|
|
if (isLoading() || isCompleted() || hasMoved()) {
|
|
|
|
return 'OK';
|
|
|
|
}
|
|
|
|
|
|
|
|
infiniteLoading.querySelector('ion-button').click();
|
|
|
|
|
|
|
|
// Wait 100ms
|
|
|
|
await new Promise(resolve => setTimeout(resolve, 100));
|
|
|
|
|
|
|
|
return (isLoading() || isCompleted() || hasMoved()) ? 'OK' : 'ERROR: Couldn\'t load more items';
|
|
|
|
} catch (error) {
|
|
|
|
return 'ERROR: ' + error.message;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-29 11:40:42 +00:00
|
|
|
/**
|
|
|
|
* Check whether an item is selected or not.
|
|
|
|
*
|
2021-06-03 09:20:36 +00:00
|
|
|
* @param {object} locator Element locator.
|
2021-04-29 11:40:42 +00:00
|
|
|
* @return {string} YES or NO if successful, or ERROR: followed by message
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const behatIsSelected = function(locator) {
|
2021-06-03 09:20:36 +00:00
|
|
|
log('Action - Is Selected', locator);
|
2021-04-29 11:40:42 +00:00
|
|
|
|
|
|
|
try {
|
2021-06-03 09:20:36 +00:00
|
|
|
const element = findElementsBasedOnText(locator)[0];
|
2021-04-29 11:40:42 +00:00
|
|
|
|
|
|
|
return isElementSelected(element, document.body) ? 'YES' : 'NO';
|
|
|
|
} catch (error) {
|
|
|
|
return 'ERROR: ' + error.message;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-12 12:11:06 +00:00
|
|
|
/**
|
|
|
|
* Function to press arbitrary item based on its text or Aria label.
|
|
|
|
*
|
2021-06-03 09:20:36 +00:00
|
|
|
* @param {object} locator Element locator.
|
2018-11-12 12:11:06 +00:00
|
|
|
* @return {string} OK if successful, or ERROR: followed by message
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const behatPress = function(locator) {
|
2021-06-03 09:20:36 +00:00
|
|
|
log('Action - Press', locator);
|
2018-11-12 12:11:06 +00:00
|
|
|
|
2021-11-24 09:47:10 +00:00
|
|
|
let found;
|
2018-11-12 12:11:06 +00:00
|
|
|
try {
|
2021-06-03 09:20:36 +00:00
|
|
|
found = findElementsBasedOnText(locator)[0];
|
2021-04-27 15:43:47 +00:00
|
|
|
|
|
|
|
if (!found) {
|
|
|
|
return 'ERROR: No matches for text';
|
|
|
|
}
|
2018-11-12 12:11:06 +00:00
|
|
|
} catch (error) {
|
|
|
|
return 'ERROR: ' + error.message;
|
|
|
|
}
|
|
|
|
|
2021-06-03 09:21:50 +00:00
|
|
|
pressElement(found);
|
2018-11-12 12:11:06 +00:00
|
|
|
|
|
|
|
return 'OK';
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the currently displayed page header.
|
|
|
|
*
|
|
|
|
* @return {string} OK: followed by header text if successful, or ERROR: followed by message.
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const behatGetHeader = function() {
|
2018-11-12 12:11:06 +00:00
|
|
|
log('Action - Get header');
|
|
|
|
|
2021-11-24 09:47:10 +00:00
|
|
|
let titles = Array.from(document.querySelectorAll('.ion-page:not(.ion-page-hidden) > ion-header h1'));
|
2021-11-24 09:47:10 +00:00
|
|
|
titles = titles.filter(function(title) {
|
|
|
|
return isElementVisible(title, document.body);
|
2018-11-12 12:11:06 +00:00
|
|
|
});
|
|
|
|
|
2021-11-24 09:47:10 +00:00
|
|
|
if (titles.length > 1) {
|
2018-11-12 12:11:06 +00:00
|
|
|
return 'ERROR: Too many possible titles';
|
2021-11-24 09:47:10 +00:00
|
|
|
} else if (!titles.length) {
|
2018-11-12 12:11:06 +00:00
|
|
|
return 'ERROR: No title found';
|
|
|
|
} else {
|
2021-11-24 09:47:10 +00:00
|
|
|
const title = titles[0].innerText.trim();
|
|
|
|
return 'OK:' + title;
|
2018-11-12 12:11:06 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the text of a field to the specified value.
|
|
|
|
*
|
|
|
|
* This currently matches fields only based on the placeholder attribute.
|
|
|
|
*
|
|
|
|
* @param {string} field Field name
|
|
|
|
* @param {string} value New value
|
|
|
|
* @return {string} OK or ERROR: followed by message
|
|
|
|
*/
|
2021-11-24 09:47:10 +00:00
|
|
|
const behatSetField = function(field, value) {
|
2018-11-12 12:11:06 +00:00
|
|
|
log('Action - Set field ' + field + ' to: ' + value);
|
|
|
|
|
2021-11-24 09:47:10 +00:00
|
|
|
const found = findElementsBasedOnText({ text: field, selector: 'input, textarea, [contenteditable="true"]' })[0];
|
|
|
|
if (!found) {
|
|
|
|
return 'ERROR: No matches for text';
|
2018-11-12 12:11:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Functions to get/set value depending on field type.
|
2021-11-24 09:47:10 +00:00
|
|
|
let setValue;
|
|
|
|
let getValue;
|
2018-11-12 12:11:06 +00:00
|
|
|
switch (found.nodeName) {
|
|
|
|
case 'INPUT':
|
|
|
|
case 'TEXTAREA':
|
|
|
|
setValue = function(text) {
|
|
|
|
found.value = text;
|
|
|
|
};
|
|
|
|
getValue = function() {
|
|
|
|
return found.value;
|
|
|
|
};
|
|
|
|
break;
|
|
|
|
case 'DIV':
|
|
|
|
setValue = function(text) {
|
|
|
|
found.innerHTML = text;
|
|
|
|
};
|
|
|
|
getValue = function() {
|
|
|
|
return found.innerHTML;
|
|
|
|
};
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pretend we have cut and pasted the new text.
|
2021-11-24 09:47:10 +00:00
|
|
|
let event;
|
2018-11-12 12:11:06 +00:00
|
|
|
if (getValue() !== '') {
|
|
|
|
event = new InputEvent('input', {bubbles: true, view: window, cancelable: true,
|
|
|
|
inputType: 'devareByCut'});
|
|
|
|
setTimeout(function() {
|
|
|
|
setValue('');
|
|
|
|
found.dispatchEvent(event);
|
|
|
|
}, 0);
|
|
|
|
}
|
|
|
|
if (value !== '') {
|
|
|
|
event = new InputEvent('input', {bubbles: true, view: window, cancelable: true,
|
|
|
|
inputType: 'insertFromPaste', data: value});
|
|
|
|
setTimeout(function() {
|
|
|
|
setValue(value);
|
|
|
|
found.dispatchEvent(event);
|
|
|
|
}, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 'OK';
|
|
|
|
};
|
2019-02-12 10:13:51 +00:00
|
|
|
|
2021-11-09 15:06:13 +00:00
|
|
|
/**
|
|
|
|
* Get an Angular component instance.
|
|
|
|
*
|
|
|
|
* @param {string} selector Element selector
|
|
|
|
* @param {string} className Constructor class name
|
|
|
|
* @return {object} Component instance
|
|
|
|
*/
|
2022-01-10 13:18:10 +00:00
|
|
|
const behatGetAngularInstance = function(selector, className) {
|
|
|
|
log('Action - Get Angular instance ' + selector + ', ' + className);
|
2021-12-01 13:24:43 +00:00
|
|
|
|
2021-11-09 15:06:13 +00:00
|
|
|
const activeElement = Array.from(document.querySelectorAll(`.ion-page:not(.ion-page-hidden) ${selector}`)).pop();
|
|
|
|
|
|
|
|
if (!activeElement || !activeElement.__ngContext__) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return activeElement.__ngContext__.find(node => node?.constructor?.name === className);
|
|
|
|
};
|
|
|
|
|
2019-02-12 10:13:51 +00:00
|
|
|
// Make some functions publicly available for Behat to call.
|
|
|
|
window.behat = {
|
|
|
|
pressStandard : behatPressStandard,
|
|
|
|
closePopup : behatClosePopup,
|
2021-04-27 15:43:47 +00:00
|
|
|
find : behatFind,
|
2021-11-25 12:04:25 +00:00
|
|
|
scrollTo : behatScrollTo,
|
2021-12-01 13:12:15 +00:00
|
|
|
loadMoreItems: behatLoadMoreItems,
|
2021-04-29 11:40:42 +00:00
|
|
|
isSelected : behatIsSelected,
|
2019-02-12 10:13:51 +00:00
|
|
|
press : behatPress,
|
|
|
|
setField : behatSetField,
|
|
|
|
getHeader : behatGetHeader,
|
2022-01-10 13:18:10 +00:00
|
|
|
getAngularInstance: behatGetAngularInstance,
|
2019-02-12 10:13:51 +00:00
|
|
|
};
|
2018-11-12 12:11:06 +00:00
|
|
|
})();
|