UNPKG

@nova-ui/bits

Version:

SolarWinds Nova Framework

190 lines 27.9 kB
// © 2022 SolarWinds Worldwide, LLC. All rights reserved. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to // deal in the Software without restriction, including without limitation the // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or // sell copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. import { DOCUMENT } from "@angular/common"; import { Inject, Injectable } from "@angular/core"; import _isNil from "lodash/isNil"; import * as i0 from "@angular/core"; /** * @dynamic * @ignore */ export class PositionService { constructor(document) { this.document = document; } /** * __Description:__ * Takes two HTMLElements (hostElement and targetElement) and calculates a position for the targetElement. * @param hostElement If you want to display tooltip for some textbox, hostElement is textbox. * @param targetElement Tooltip (or popover) element, for which position will be returned. * @param placementAndAlign String in format "placement-align", e.g. "top-center". * 'Placement' could be 'right', 'left', 'top' (default), 'bottom'. It defines how target will be placed * to the host. * 'Align' could be 'right', 'left', 'top', 'bottom' and 'center' (default). It defines whether one of the target's * edges will be aligned to host's one. Be aware that horizontal placements need vertical alignment and vice verca. * Possible values of 'placementAndAlign' could be: 'top' (the same as 'top-center'), 'top-right', 'left-bottom', * 'bottom-left' etc. * @param appendToBody Specifies whether target element is appended to the body. * When not, service seeks for the first positioned parent of host. * @returns Object literal { top:number, left:number } that contains * coordinates for the top left corner of targetElement. */ getPosition(hostElement, targetElement, placementAndAlign, appendToBody) { const placement = placementAndAlign.split("-")[0]; const align = placementAndAlign.split("-")[1]; let shiftByX; let shiftByY; const hostElementPosition = appendToBody ? this.offset(hostElement) : this.position(hostElement); switch (align) { case "right": shiftByX = hostElementPosition.width - targetElement.offsetWidth; shiftByY = (hostElementPosition.height - targetElement.offsetHeight) / 2; break; case "left": shiftByX = 0; shiftByY = (hostElementPosition.height - targetElement.offsetHeight) / 2; break; case "top": shiftByX = (hostElementPosition.width - targetElement.offsetWidth) / 2; shiftByY = 0; break; case "bottom": shiftByX = (hostElementPosition.width - targetElement.offsetWidth) / 2; shiftByY = hostElementPosition.height - targetElement.offsetHeight; break; case "center": default: shiftByX = (hostElementPosition.width - targetElement.offsetWidth) / 2; shiftByY = (hostElementPosition.height - targetElement.offsetHeight) / 2; break; } let targetElementPosition; switch (placement) { case "right": targetElementPosition = { top: hostElementPosition.top + shiftByY, left: hostElementPosition.left + hostElementPosition.width, }; break; case "left": targetElementPosition = { top: hostElementPosition.top + shiftByY, left: hostElementPosition.left - targetElement.offsetWidth, }; break; case "bottom": targetElementPosition = { top: hostElementPosition.top + hostElementPosition.height, left: hostElementPosition.left + shiftByX, }; break; case "top": default: targetElementPosition = { top: hostElementPosition.top - targetElement.offsetHeight, left: hostElementPosition.left + shiftByX, }; break; } return targetElementPosition; } position(nativeElement) { let offsetParentBoundingClientRect = { top: 0, left: 0 }; const elementBoundingClientRect = this.offset(nativeElement); const offsetParentElement = this.parentOffsetEl(nativeElement); if (offsetParentElement !== this.document) { offsetParentBoundingClientRect = this.offset(offsetParentElement); offsetParentBoundingClientRect.top += offsetParentElement.clientTop - offsetParentElement.scrollTop; offsetParentBoundingClientRect.left += offsetParentElement.clientLeft - offsetParentElement.scrollLeft; } const boundingClientRect = nativeElement.getBoundingClientRect(); return { width: boundingClientRect.width || nativeElement.offsetWidth, height: boundingClientRect.height || nativeElement.offsetHeight, top: elementBoundingClientRect.top - offsetParentBoundingClientRect.top, left: elementBoundingClientRect.left - offsetParentBoundingClientRect.left, }; } offset(nativeElement) { const boundingClientRect = nativeElement.getBoundingClientRect(); if (_isNil(this.document.defaultView)) { throw new Error("Document defaultView is not available"); } return { width: boundingClientRect.width || nativeElement.offsetWidth, height: boundingClientRect.height || nativeElement.offsetHeight, top: boundingClientRect.top + (this.document.defaultView.pageYOffset || this.document.documentElement.scrollTop), left: boundingClientRect.left + (this.document.defaultView.pageXOffset || this.document.documentElement.scrollLeft), }; } getStyle(nativeElement, cssProp) { if (_isNil(this.document.defaultView)) { throw new Error("Document defaultView is not available"); } if (this.document.defaultView.getComputedStyle) { return this.document.defaultView.getComputedStyle(nativeElement)[cssProp]; } // finally try and get inline style return nativeElement.style[cssProp]; } isStaticPositioned(nativeElement) { return ((this.getStyle(nativeElement, "position") || "static") === "static"); } parentOffsetEl(nativeElement) { let offsetParent = nativeElement.offsetParent || this.document; while (offsetParent && offsetParent !== this.document && this.isStaticPositioned(offsetParent)) { offsetParent = offsetParent.offsetParent; } return offsetParent || this.document; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PositionService, deps: [{ token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PositionService, providedIn: "root" }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PositionService, decorators: [{ type: Injectable, args: [{ providedIn: "root" }] }], ctorParameters: () => [{ type: Document, decorators: [{ type: Inject, args: [DOCUMENT] }] }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"position.service.js","sourceRoot":"","sources":["../../../src/services/position.service.ts"],"names":[],"mappings":"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","sourcesContent":["// © 2022 SolarWinds Worldwide, LLC. All rights reserved.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n//  of this software and associated documentation files (the \"Software\"), to\n//  deal in the Software without restriction, including without limitation the\n//  rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n//  sell copies of the Software, and to permit persons to whom the Software is\n//  furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n//  all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n//  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n//  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n//  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n//  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n//  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n//  THE SOFTWARE.\n\nimport { DOCUMENT } from \"@angular/common\";\nimport { Inject, Injectable } from \"@angular/core\";\nimport _isNil from \"lodash/isNil\";\n\n/**\n * Service for calculating element's position (e.g. tooltip, popover).\n * For now has one public method 'getCoordinates' (see method description).\n */\n\ntype OurCSS = CSSStyleDeclaration & { [key: string]: any };\n\n/**\n * @dynamic\n * @ignore\n */\n@Injectable({ providedIn: \"root\" })\nexport class PositionService {\n    constructor(@Inject(DOCUMENT) private document: Document) {}\n    /**\n     * __Description:__\n     * Takes two HTMLElements (hostElement and targetElement) and calculates a position for the targetElement.\n     * @param  hostElement If you want to display tooltip for some textbox, hostElement is textbox.\n     * @param   targetElement Tooltip (or popover) element, for which position will be returned.\n     * @param   placementAndAlign String in format \"placement-align\", e.g. \"top-center\".\n     * 'Placement' could be 'right', 'left', 'top' (default), 'bottom'. It defines how target will be placed\n     * to the host.\n     * 'Align' could be 'right', 'left', 'top', 'bottom' and 'center' (default). It defines whether one of the target's\n     * edges will be aligned to host's one. Be aware that horizontal placements need vertical alignment and vice verca.\n     * Possible values of 'placementAndAlign' could be: 'top' (the same as 'top-center'), 'top-right', 'left-bottom',\n     * 'bottom-left' etc.\n     * @param   appendToBody Specifies whether target element is appended to the body.\n     * When not, service seeks for the first positioned parent of host.\n     * @returns Object literal { top:number, left:number } that contains\n     * coordinates for the top left corner of targetElement.\n     */\n    public getPosition(\n        hostElement: HTMLElement,\n        targetElement: HTMLElement,\n        placementAndAlign: string,\n        appendToBody?: boolean\n    ): { top: number; left: number } {\n        const placement = placementAndAlign.split(\"-\")[0];\n        const align = placementAndAlign.split(\"-\")[1];\n        let shiftByX: number;\n        let shiftByY: number;\n        const hostElementPosition = appendToBody\n            ? this.offset(hostElement)\n            : this.position(hostElement);\n        switch (align) {\n            case \"right\":\n                shiftByX =\n                    hostElementPosition.width - targetElement.offsetWidth;\n                shiftByY =\n                    (hostElementPosition.height - targetElement.offsetHeight) /\n                    2;\n                break;\n            case \"left\":\n                shiftByX = 0;\n                shiftByY =\n                    (hostElementPosition.height - targetElement.offsetHeight) /\n                    2;\n                break;\n            case \"top\":\n                shiftByX =\n                    (hostElementPosition.width - targetElement.offsetWidth) / 2;\n                shiftByY = 0;\n                break;\n            case \"bottom\":\n                shiftByX =\n                    (hostElementPosition.width - targetElement.offsetWidth) / 2;\n                shiftByY =\n                    hostElementPosition.height - targetElement.offsetHeight;\n                break;\n            case \"center\":\n            default:\n                shiftByX =\n                    (hostElementPosition.width - targetElement.offsetWidth) / 2;\n                shiftByY =\n                    (hostElementPosition.height - targetElement.offsetHeight) /\n                    2;\n                break;\n        }\n        let targetElementPosition: { top: number; left: number };\n        switch (placement) {\n            case \"right\":\n                targetElementPosition = {\n                    top: hostElementPosition.top + shiftByY,\n                    left: hostElementPosition.left + hostElementPosition.width,\n                };\n                break;\n            case \"left\":\n                targetElementPosition = {\n                    top: hostElementPosition.top + shiftByY,\n                    left: hostElementPosition.left - targetElement.offsetWidth,\n                };\n                break;\n            case \"bottom\":\n                targetElementPosition = {\n                    top: hostElementPosition.top + hostElementPosition.height,\n                    left: hostElementPosition.left + shiftByX,\n                };\n                break;\n            case \"top\":\n            default:\n                targetElementPosition = {\n                    top: hostElementPosition.top - targetElement.offsetHeight,\n                    left: hostElementPosition.left + shiftByX,\n                };\n                break;\n        }\n        return targetElementPosition;\n    }\n\n    private position(nativeElement: HTMLElement): {\n        width: number;\n        height: number;\n        top: number;\n        left: number;\n    } {\n        let offsetParentBoundingClientRect = { top: 0, left: 0 };\n        const elementBoundingClientRect = this.offset(nativeElement);\n        const offsetParentElement = this.parentOffsetEl(nativeElement);\n\n        if (offsetParentElement !== this.document) {\n            offsetParentBoundingClientRect = this.offset(offsetParentElement);\n            offsetParentBoundingClientRect.top +=\n                offsetParentElement.clientTop - offsetParentElement.scrollTop;\n            offsetParentBoundingClientRect.left +=\n                offsetParentElement.clientLeft - offsetParentElement.scrollLeft;\n        }\n\n        const boundingClientRect = nativeElement.getBoundingClientRect();\n\n        return {\n            width: boundingClientRect.width || nativeElement.offsetWidth,\n            height: boundingClientRect.height || nativeElement.offsetHeight,\n            top:\n                elementBoundingClientRect.top -\n                offsetParentBoundingClientRect.top,\n            left:\n                elementBoundingClientRect.left -\n                offsetParentBoundingClientRect.left,\n        };\n    }\n\n    private offset(nativeElement: any): {\n        width: number;\n        height: number;\n        top: number;\n        left: number;\n    } {\n        const boundingClientRect = nativeElement.getBoundingClientRect();\n        if (_isNil(this.document.defaultView)) {\n            throw new Error(\"Document defaultView is not available\");\n        }\n\n        return {\n            width: boundingClientRect.width || nativeElement.offsetWidth,\n            height: boundingClientRect.height || nativeElement.offsetHeight,\n            top:\n                boundingClientRect.top +\n                (this.document.defaultView.pageYOffset ||\n                    this.document.documentElement.scrollTop),\n            left:\n                boundingClientRect.left +\n                (this.document.defaultView.pageXOffset ||\n                    this.document.documentElement.scrollLeft),\n        };\n    }\n\n    private getStyle(nativeElement: HTMLElement, cssProp: string): string {\n        if (_isNil(this.document.defaultView)) {\n            throw new Error(\"Document defaultView is not available\");\n        }\n\n        if (this.document.defaultView.getComputedStyle) {\n            return (\n                this.document.defaultView.getComputedStyle(\n                    nativeElement\n                ) as OurCSS\n            )[cssProp];\n        }\n\n        // finally try and get inline style\n        return (nativeElement.style as OurCSS)[cssProp];\n    }\n\n    private isStaticPositioned(nativeElement: HTMLElement): boolean {\n        return (\n            (this.getStyle(nativeElement, \"position\") || \"static\") === \"static\"\n        );\n    }\n\n    private parentOffsetEl(nativeElement: HTMLElement): any {\n        let offsetParent: any = nativeElement.offsetParent || this.document;\n\n        while (\n            offsetParent &&\n            offsetParent !== this.document &&\n            this.isStaticPositioned(offsetParent)\n        ) {\n            offsetParent = offsetParent.offsetParent;\n        }\n\n        return offsetParent || this.document;\n    }\n}\n"]}