UNPKG

@testplane/webdriverio

Version:

Next-gen browser and mobile automation test framework for Node.js

104 lines (103 loc) 16.2 kB
// src/scripts/isElementClickable.ts function isElementClickable(elem) { if (!elem.getBoundingClientRect || !elem.scrollIntoView || !elem.contains || !elem.getClientRects || !document.elementFromPoint) { return false; } const isOldEdge = !!window.StyleMedia; const scrollIntoViewFullSupport = !(window.safari || isOldEdge); function getOverlappingElement(elem2, context) { context = context || document; const elemDimension = elem2.getBoundingClientRect(); const x = elemDimension.left + elem2.clientWidth / 2; const y = elemDimension.top + elem2.clientHeight / 2; return context.elementFromPoint(x, y); } function getOverlappingRects(elem2, context) { context = context || document; const rects = elem2.getClientRects(); const rect = rects[0]; const x = rect.left + rect.width / 2; const y = rect.top + rect.height / 2; return [context.elementFromPoint(x, y)]; } function getOverlappingElements(elem2, context) { return [getOverlappingElement(elem2, context)].concat(getOverlappingRects(elem2, context)); } function nodeContains(elem2, otherNode) { if (isOldEdge) { let tmpElement = otherNode; while (tmpElement) { if (tmpElement === elem2) { return true; } tmpElement = tmpElement.parentNode; if (tmpElement && tmpElement.nodeType === 11 && tmpElement.host) { tmpElement = tmpElement.host; } } return false; } return elem2.contains(otherNode); } function isOverlappingElementMatch(elementsFromPoint, elem2) { if (elementsFromPoint.some(function(elementFromPoint) { return elementFromPoint === elem2 || nodeContains(elem2, elementFromPoint); })) { return true; } let elemsWithShadowRoot = [].concat(elementsFromPoint); elemsWithShadowRoot = elemsWithShadowRoot.filter(function(x) { return x && x.shadowRoot && x.shadowRoot.elementFromPoint; }); let shadowElementsFromPoint = []; for (let i = 0; i < elemsWithShadowRoot.length; ++i) { const shadowElement = elemsWithShadowRoot[i]; shadowElementsFromPoint = shadowElementsFromPoint.concat( getOverlappingElements(elem2, shadowElement.shadowRoot) ); } shadowElementsFromPoint = [].concat(shadowElementsFromPoint); shadowElementsFromPoint = shadowElementsFromPoint.filter(function(x) { return !elementsFromPoint.includes(x); }); if (shadowElementsFromPoint.length === 0) { return false; } return isOverlappingElementMatch(shadowElementsFromPoint, elem2); } function isElementInViewport(elem2) { if (!elem2.getBoundingClientRect) { return false; } const rect = elem2.getBoundingClientRect(); const windowHeight = window.innerHeight || document.documentElement.clientHeight; const windowWidth = window.innerWidth || document.documentElement.clientWidth; const vertInView = rect.top <= windowHeight && rect.top + rect.height > 0; const horInView = rect.left <= windowWidth && rect.left + rect.width > 0; return vertInView && horInView; } function isEnabled(elem2) { return elem2.disabled !== true; } function hasOverlaps(elem2) { return !isOverlappingElementMatch(getOverlappingElements(elem2), elem2); } function isFullyDisplayedInViewport(elem2) { return isElementInViewport(elem2) && !hasOverlaps(elem2); } let _isFullyDisplayedInViewport = isFullyDisplayedInViewport(elem); if (!_isFullyDisplayedInViewport) { const { x: originalX, y: originalY } = elem.getBoundingClientRect(); elem.scrollIntoView(scrollIntoViewFullSupport ? { block: "center", inline: "center" } : false); _isFullyDisplayedInViewport = isFullyDisplayedInViewport(elem); const { x, y } = elem.getBoundingClientRect(); if (x !== originalX || y !== originalY) { elem.scroll(scrollX, scrollY); } } return _isFullyDisplayedInViewport && isEnabled(elem); } export { isElementClickable as default }; //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/scripts/isElementClickable.ts"],
  "sourceRoot": "/Users/kroman512/gemini-testing/webdriverio/packages/webdriverio",
  "sourcesContent": ["/**\n * check if element is within the viewport or is overlapped by another element or disabled\n * @param  {HTMLElement} elem  element to check\n * @return {Boolean}           false if element is not overlapped\n */\nexport default function isElementClickable (elem: HTMLElement) {\n    if (!elem.getBoundingClientRect || !elem.scrollIntoView || !elem.contains || !elem.getClientRects || !document.elementFromPoint) {\n        return false\n    }\n\n    // Edge before switching to Chromium\n    const isOldEdge = !!(window as unknown as { StyleMedia: unknown }).StyleMedia\n    // returns true for Chrome and Firefox and false for Safari, Edge and IE\n    const scrollIntoViewFullSupport = !((window as { safari?: boolean }).safari || isOldEdge)\n\n    // get overlapping element\n    function getOverlappingElement (elem: HTMLElement, context?: Document) {\n        context = context || document\n        const elemDimension = elem.getBoundingClientRect()\n        const x = elemDimension.left + (elem.clientWidth / 2)\n        const y = elemDimension.top + (elem.clientHeight / 2)\n        return context.elementFromPoint(x, y)\n    }\n\n    // get overlapping element rects (currently only the first)\n    // applicable if element's text is multiline.\n    function getOverlappingRects (elem: HTMLElement, context?: Document) {\n        context = context || document\n\n        const rects = elem.getClientRects()\n        // webdriver clicks on center of the first element's rect (line of text), it might change in future\n        const rect = rects[0]\n        const x = rect.left + (rect.width / 2)\n        const y = rect.top + (rect.height / 2)\n        return [context.elementFromPoint(x, y)]\n    }\n\n    // get overlapping elements\n    function getOverlappingElements (elem: HTMLElement, context?: Document) {\n        return [getOverlappingElement(elem, context)].concat(getOverlappingRects(elem, context))\n    }\n\n    // is a node a descendant of a given node\n    function nodeContains (elem: HTMLElement, otherNode: HTMLElement) {\n        // Edge doesn't support neither Shadow Dom nor contains if ShadowRoot polyfill is used\n        if (isOldEdge) {\n            let tmpElement = otherNode as HTMLElement | ShadowRoot | Element\n            while (tmpElement) {\n                if (tmpElement === elem) {\n                    return true\n                }\n\n                tmpElement = tmpElement.parentNode as ShadowRoot\n                // DocumentFragment / ShadowRoot polyfill like ShadyRoot\n                if (tmpElement && tmpElement.nodeType === 11 && tmpElement.host) {\n                    tmpElement = tmpElement.host\n                }\n            }\n            return false\n        }\n\n        return elem.contains(otherNode)\n    }\n\n    // is one of overlapping elements the `elem` or one of its child\n    function isOverlappingElementMatch (elementsFromPoint: HTMLElement[], elem: HTMLElement): boolean {\n        if (elementsFromPoint.some(function (elementFromPoint) {\n            return elementFromPoint === elem || nodeContains(elem, elementFromPoint)\n        })) {\n            return true\n        }\n\n        // shadow root\n        // filter unique elements with shadowRoot\n        // @ts-ignore\n        let elemsWithShadowRoot = [].concat(elementsFromPoint)\n        elemsWithShadowRoot = elemsWithShadowRoot.filter(function (x: HTMLElement) {\n            return x && x.shadowRoot && (x.shadowRoot as ShadowRoot).elementFromPoint\n        })\n\n        // getOverlappingElements of every element with shadowRoot\n        let shadowElementsFromPoint: HTMLElement[] = []\n        for (let i = 0; i < elemsWithShadowRoot.length; ++i) {\n            const shadowElement = elemsWithShadowRoot[i]\n            shadowElementsFromPoint = shadowElementsFromPoint.concat(\n                getOverlappingElements(elem, (shadowElement as HTMLElement).shadowRoot as unknown as Document) as HTMLElement[]\n            )\n        }\n        // remove duplicates and parents\n        // @ts-ignore\n        shadowElementsFromPoint = [].concat(shadowElementsFromPoint)\n        shadowElementsFromPoint = shadowElementsFromPoint.filter(function (x) {\n            return !elementsFromPoint.includes(x)\n        })\n\n        if (shadowElementsFromPoint.length === 0) {\n            return false\n        }\n\n        return isOverlappingElementMatch(shadowElementsFromPoint, elem)\n    }\n\n    // copied from `isElementInViewport.js`\n    function isElementInViewport (elem: HTMLElement) {\n        if (!elem.getBoundingClientRect) {\n            return false\n        }\n\n        const rect = elem.getBoundingClientRect()\n\n        const windowHeight = (window.innerHeight || document.documentElement.clientHeight)\n        const windowWidth = (window.innerWidth || document.documentElement.clientWidth)\n\n        const vertInView = (rect.top <= windowHeight) && ((rect.top + rect.height) > 0)\n        const horInView = (rect.left <= windowWidth) && ((rect.left + rect.width) > 0)\n\n        return (vertInView && horInView)\n    }\n\n    function isEnabled(elem: HTMLFormElement) {\n        return elem.disabled !== true\n    }\n\n    function hasOverlaps(elem: HTMLElement) {\n        return !isOverlappingElementMatch(getOverlappingElements(elem) as unknown as HTMLElement[], elem)\n    }\n\n    function isFullyDisplayedInViewport(elem: HTMLElement) {\n        return isElementInViewport(elem) && !hasOverlaps(elem)\n    }\n\n    // scroll the element to the center of the viewport when\n    // it is not fully displayed in the viewport or is overlapped by another element\n    // to check if it still overlapped/not in the viewport\n    // afterwards we scroll back to the original position\n    let _isFullyDisplayedInViewport = isFullyDisplayedInViewport(elem)\n    if (!_isFullyDisplayedInViewport) {\n        const { x: originalX, y: originalY } = elem.getBoundingClientRect()\n\n        elem.scrollIntoView(scrollIntoViewFullSupport ? { block: 'center', inline: 'center' } : false)\n\n        _isFullyDisplayedInViewport = isFullyDisplayedInViewport(elem)\n\n        const { x, y } = elem.getBoundingClientRect()\n        if (x !== originalX || y !== originalY) {\n            elem.scroll(scrollX, scrollY)\n        }\n    }\n\n    return _isFullyDisplayedInViewport && isEnabled(elem as HTMLFormElement)\n}\n"],
  "mappings": "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",
  "names": ["elem"]
}
