matrix-react-sdk
Version:
SDK for matrix.org using React
157 lines (125 loc) • 19.4 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = exports.Alignment = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _reactDom = _interopRequireDefault(require("react-dom"));
var _classnames = _interopRequireDefault(require("classnames"));
var _replaceableComponent = require("../../../utils/replaceableComponent");
var _dec, _class, _class2, _temp;
const MIN_TOOLTIP_HEIGHT = 25;
let Alignment;
exports.Alignment = Alignment;
(function (Alignment) {
Alignment[Alignment["Natural"] = 0] = "Natural";
Alignment[Alignment["Left"] = 1] = "Left";
Alignment[Alignment["Right"] = 2] = "Right";
Alignment[Alignment["Top"] = 3] = "Top";
Alignment[Alignment["Bottom"] = 4] = "Bottom";
})(Alignment || (exports.Alignment = Alignment = {}));
let Tooltip = (_dec = (0, _replaceableComponent.replaceableComponent)("views.elements.Tooltip"), _dec(_class = (_temp = _class2 = class Tooltip extends _react.default.Component
/*:: <IProps>*/
{
constructor(...args) {
super(...args);
(0, _defineProperty2.default)(this, "tooltipContainer", void 0);
(0, _defineProperty2.default)(this, "tooltip", void 0);
(0, _defineProperty2.default)(this, "parent", void 0);
(0, _defineProperty2.default)(this, "renderTooltip", () => {
// Add the parent's position to the tooltips, so it's correctly
// positioned, also taking into account any window zoom
// NOTE: The additional 6 pixels for the left position, is to take account of the
// tooltips chevron
const style = this.updatePosition({}); // Hide the entire container when not visible. This prevents flashing of the tooltip
// if it is not meant to be visible on first mount.
style.display = this.props.visible ? "block" : "none";
const tooltipClasses = (0, _classnames.default)("mx_Tooltip", this.props.tooltipClassName, {
"mx_Tooltip_visible": this.props.visible,
"mx_Tooltip_invisible": !this.props.visible
});
const tooltip = /*#__PURE__*/_react.default.createElement("div", {
className: tooltipClasses,
style: style
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_Tooltip_chevron"
}), this.props.label); // Render the tooltip manually, as we wish it not to be rendered within the parent
this.tooltip = _reactDom.default.render(tooltip, this.tooltipContainer);
});
}
// Create a wrapper for the tooltip outside the parent and attach it to the body element
componentDidMount() {
this.tooltipContainer = document.createElement("div");
this.tooltipContainer.className = "mx_Tooltip_wrapper";
document.body.appendChild(this.tooltipContainer);
window.addEventListener('scroll', this.renderTooltip, true);
this.parent = _reactDom.default.findDOMNode(this).parentNode;
this.renderTooltip();
}
componentDidUpdate() {
this.renderTooltip();
} // Remove the wrapper element, as the tooltip has finished using it
componentWillUnmount() {
_reactDom.default.unmountComponentAtNode(this.tooltipContainer);
document.body.removeChild(this.tooltipContainer);
window.removeEventListener('scroll', this.renderTooltip, true);
}
updatePosition(style
/*: CSSProperties*/
) {
const parentBox = this.parent.getBoundingClientRect();
let offset = 0;
if (parentBox.height > MIN_TOOLTIP_HEIGHT) {
offset = Math.floor((parentBox.height - MIN_TOOLTIP_HEIGHT) / 2);
} else {
// The tooltip is larger than the parent height: figure out what offset
// we need so that we're still centered.
offset = Math.floor(parentBox.height - MIN_TOOLTIP_HEIGHT);
}
const baseTop = parentBox.top - 2 + this.props.yOffset + window.pageYOffset;
const top = baseTop + offset;
const right = window.innerWidth - parentBox.right - window.pageXOffset - 16;
const left = parentBox.right + window.pageXOffset + 6;
const horizontalCenter = parentBox.right - window.pageXOffset - parentBox.width / 2;
switch (this.props.alignment) {
case Alignment.Natural:
if (parentBox.right > window.innerWidth / 2) {
style.right = right;
style.top = top;
break;
}
// fall through to Right
case Alignment.Right:
style.left = left;
style.top = top;
break;
case Alignment.Left:
style.right = right;
style.top = top;
break;
case Alignment.Top:
style.top = baseTop - 16;
style.left = horizontalCenter;
break;
case Alignment.Bottom:
style.top = baseTop + parentBox.height;
style.left = horizontalCenter;
break;
}
return style;
}
render() {
// Render a placeholder
return /*#__PURE__*/_react.default.createElement("div", {
className: this.props.className
});
}
}, (0, _defineProperty2.default)(_class2, "Alignment", Alignment), (0, _defineProperty2.default)(_class2, "defaultProps", {
visible: true,
yOffset: 0,
alignment: Alignment.Natural
}), _temp)) || _class);
exports.default = Tooltip;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/elements/Tooltip.tsx"],"names":["MIN_TOOLTIP_HEIGHT","Alignment","Tooltip","React","Component","style","updatePosition","display","props","visible","tooltipClasses","tooltipClassName","tooltip","label","ReactDOM","render","tooltipContainer","componentDidMount","document","createElement","className","body","appendChild","window","addEventListener","renderTooltip","parent","findDOMNode","parentNode","componentDidUpdate","componentWillUnmount","unmountComponentAtNode","removeChild","removeEventListener","parentBox","getBoundingClientRect","offset","height","Math","floor","baseTop","top","yOffset","pageYOffset","right","innerWidth","pageXOffset","left","horizontalCenter","width","alignment","Natural","Right","Left","Top","Bottom"],"mappings":";;;;;;;;;;;AAoBA;;AACA;;AACA;;AACA;;;;AAEA,MAAMA,kBAAkB,GAAG,EAA3B;IAEYC,S;;;WAAAA,S;AAAAA,EAAAA,S,CAAAA,S;AAAAA,EAAAA,S,CAAAA,S;AAAAA,EAAAA,S,CAAAA,S;AAAAA,EAAAA,S,CAAAA,S;AAAAA,EAAAA,S,CAAAA,S;GAAAA,S,yBAAAA,S;;IAwBSC,O,WADpB,gDAAqB,wBAArB,C,mCAAD,MACqBA,OADrB,SACqCC,eAAMC;AAD3C;AAC6D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,yDAmFjC,MAAM;AAC1B;AACA;AACA;AACA;AACA,YAAMC,KAAK,GAAG,KAAKC,cAAL,CAAoB,EAApB,CAAd,CAL0B,CAM1B;AACA;;AACAD,MAAAA,KAAK,CAACE,OAAN,GAAgB,KAAKC,KAAL,CAAWC,OAAX,GAAqB,OAArB,GAA+B,MAA/C;AAEA,YAAMC,cAAc,GAAG,yBAAW,YAAX,EAAyB,KAAKF,KAAL,CAAWG,gBAApC,EAAsD;AACzE,8BAAsB,KAAKH,KAAL,CAAWC,OADwC;AAEzE,gCAAwB,CAAC,KAAKD,KAAL,CAAWC;AAFqC,OAAtD,CAAvB;;AAKA,YAAMG,OAAO,gBACT;AAAK,QAAA,SAAS,EAAEF,cAAhB;AAAgC,QAAA,KAAK,EAAEL;AAAvC,sBACI;AAAK,QAAA,SAAS,EAAC;AAAf,QADJ,EAEM,KAAKG,KAAL,CAAWK,KAFjB,CADJ,CAf0B,CAsB1B;;;AACA,WAAKD,OAAL,GAAeE,kBAASC,MAAT,CAAyBH,OAAzB,EAAkC,KAAKI,gBAAvC,CAAf;AACH,KA3GwD;AAAA;;AAezD;AACOC,EAAAA,iBAAP,GAA2B;AACvB,SAAKD,gBAAL,GAAwBE,QAAQ,CAACC,aAAT,CAAuB,KAAvB,CAAxB;AACA,SAAKH,gBAAL,CAAsBI,SAAtB,GAAkC,oBAAlC;AACAF,IAAAA,QAAQ,CAACG,IAAT,CAAcC,WAAd,CAA0B,KAAKN,gBAA/B;AACAO,IAAAA,MAAM,CAACC,gBAAP,CAAwB,QAAxB,EAAkC,KAAKC,aAAvC,EAAsD,IAAtD;AAEA,SAAKC,MAAL,GAAcZ,kBAASa,WAAT,CAAqB,IAArB,EAA2BC,UAAzC;AAEA,SAAKH,aAAL;AACH;;AAEMI,EAAAA,kBAAP,GAA4B;AACxB,SAAKJ,aAAL;AACH,GA7BwD,CA+BzD;;;AACOK,EAAAA,oBAAP,GAA8B;AAC1BhB,sBAASiB,sBAAT,CAAgC,KAAKf,gBAArC;;AACAE,IAAAA,QAAQ,CAACG,IAAT,CAAcW,WAAd,CAA0B,KAAKhB,gBAA/B;AACAO,IAAAA,MAAM,CAACU,mBAAP,CAA2B,QAA3B,EAAqC,KAAKR,aAA1C,EAAyD,IAAzD;AACH;;AAEOnB,EAAAA,cAAR,CAAuBD;AAAvB;AAAA,IAA6C;AACzC,UAAM6B,SAAS,GAAG,KAAKR,MAAL,CAAYS,qBAAZ,EAAlB;AACA,QAAIC,MAAM,GAAG,CAAb;;AACA,QAAIF,SAAS,CAACG,MAAV,GAAmBrC,kBAAvB,EAA2C;AACvCoC,MAAAA,MAAM,GAAGE,IAAI,CAACC,KAAL,CAAW,CAACL,SAAS,CAACG,MAAV,GAAmBrC,kBAApB,IAA0C,CAArD,CAAT;AACH,KAFD,MAEO;AACH;AACA;AACAoC,MAAAA,MAAM,GAAGE,IAAI,CAACC,KAAL,CAAWL,SAAS,CAACG,MAAV,GAAmBrC,kBAA9B,CAAT;AACH;;AAED,UAAMwC,OAAO,GAAIN,SAAS,CAACO,GAAV,GAAgB,CAAhB,GAAoB,KAAKjC,KAAL,CAAWkC,OAAhC,GAA2CnB,MAAM,CAACoB,WAAlE;AACA,UAAMF,GAAG,GAAGD,OAAO,GAAGJ,MAAtB;AACA,UAAMQ,KAAK,GAAGrB,MAAM,CAACsB,UAAP,GAAoBX,SAAS,CAACU,KAA9B,GAAsCrB,MAAM,CAACuB,WAA7C,GAA2D,EAAzE;AACA,UAAMC,IAAI,GAAGb,SAAS,CAACU,KAAV,GAAkBrB,MAAM,CAACuB,WAAzB,GAAuC,CAApD;AACA,UAAME,gBAAgB,GAAGd,SAAS,CAACU,KAAV,GAAkBrB,MAAM,CAACuB,WAAzB,GAAwCZ,SAAS,CAACe,KAAV,GAAkB,CAAnF;;AACA,YAAQ,KAAKzC,KAAL,CAAW0C,SAAnB;AACI,WAAKjD,SAAS,CAACkD,OAAf;AACI,YAAIjB,SAAS,CAACU,KAAV,GAAkBrB,MAAM,CAACsB,UAAP,GAAoB,CAA1C,EAA6C;AACzCxC,UAAAA,KAAK,CAACuC,KAAN,GAAcA,KAAd;AACAvC,UAAAA,KAAK,CAACoC,GAAN,GAAYA,GAAZ;AACA;AACH;;AACD;;AACJ,WAAKxC,SAAS,CAACmD,KAAf;AACI/C,QAAAA,KAAK,CAAC0C,IAAN,GAAaA,IAAb;AACA1C,QAAAA,KAAK,CAACoC,GAAN,GAAYA,GAAZ;AACA;;AACJ,WAAKxC,SAAS,CAACoD,IAAf;AACIhD,QAAAA,KAAK,CAACuC,KAAN,GAAcA,KAAd;AACAvC,QAAAA,KAAK,CAACoC,GAAN,GAAYA,GAAZ;AACA;;AACJ,WAAKxC,SAAS,CAACqD,GAAf;AACIjD,QAAAA,KAAK,CAACoC,GAAN,GAAYD,OAAO,GAAG,EAAtB;AACAnC,QAAAA,KAAK,CAAC0C,IAAN,GAAaC,gBAAb;AACA;;AACJ,WAAK/C,SAAS,CAACsD,MAAf;AACIlD,QAAAA,KAAK,CAACoC,GAAN,GAAYD,OAAO,GAAGN,SAAS,CAACG,MAAhC;AACAhC,QAAAA,KAAK,CAAC0C,IAAN,GAAaC,gBAAb;AACA;AAvBR;;AA0BA,WAAO3C,KAAP;AACH;;AA4BMU,EAAAA,MAAP,GAAgB;AACZ;AACA,wBACI;AAAK,MAAA,SAAS,EAAE,KAAKP,KAAL,CAAWY;AAA3B,MADJ;AAIH;;AAnHwD,C,sDAOtBnB,S,0DAEG;AAClCQ,EAAAA,OAAO,EAAE,IADyB;AAElCiC,EAAAA,OAAO,EAAE,CAFyB;AAGlCQ,EAAAA,SAAS,EAAEjD,SAAS,CAACkD;AAHa,C","sourcesContent":["/*\nCopyright 2015, 2016 OpenMarket Ltd\nCopyright 2019 New Vector Ltd\nCopyright 2019 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2019, 2020 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\n\nimport React, {Component, CSSProperties} from 'react';\nimport ReactDOM from 'react-dom';\nimport classNames from 'classnames';\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\n\nconst MIN_TOOLTIP_HEIGHT = 25;\n\nexport enum Alignment {\n    Natural, // Pick left or right\n    Left,\n    Right,\n    Top, // Centered\n    Bottom, // Centered\n}\n\ninterface IProps {\n        // Class applied to the element used to position the tooltip\n        className?: string;\n        // Class applied to the tooltip itself\n        tooltipClassName?: string;\n        // Whether the tooltip is visible or hidden.\n        // The hidden state allows animating the tooltip away via CSS.\n        // Defaults to visible if unset.\n        visible?: boolean;\n        // the react element to put into the tooltip\n        label: React.ReactNode;\n        alignment?: Alignment; // defaults to Natural\n        yOffset?: number;\n}\n\n@replaceableComponent(\"views.elements.Tooltip\")\nexport default class Tooltip extends React.Component<IProps> {\n    private tooltipContainer: HTMLElement;\n    private tooltip: void | Element | Component<Element, any, any>;\n    private parent: Element;\n\n    // XXX: This is because some components (Field) are unable to `import` the Tooltip class,\n    // so we expose the Alignment options off of us statically.\n    public static readonly Alignment = Alignment;\n\n    public static readonly defaultProps = {\n        visible: true,\n        yOffset: 0,\n        alignment: Alignment.Natural,\n    };\n\n    // Create a wrapper for the tooltip outside the parent and attach it to the body element\n    public componentDidMount() {\n        this.tooltipContainer = document.createElement(\"div\");\n        this.tooltipContainer.className = \"mx_Tooltip_wrapper\";\n        document.body.appendChild(this.tooltipContainer);\n        window.addEventListener('scroll', this.renderTooltip, true);\n\n        this.parent = ReactDOM.findDOMNode(this).parentNode as Element;\n\n        this.renderTooltip();\n    }\n\n    public componentDidUpdate() {\n        this.renderTooltip();\n    }\n\n    // Remove the wrapper element, as the tooltip has finished using it\n    public componentWillUnmount() {\n        ReactDOM.unmountComponentAtNode(this.tooltipContainer);\n        document.body.removeChild(this.tooltipContainer);\n        window.removeEventListener('scroll', this.renderTooltip, true);\n    }\n\n    private updatePosition(style: CSSProperties) {\n        const parentBox = this.parent.getBoundingClientRect();\n        let offset = 0;\n        if (parentBox.height > MIN_TOOLTIP_HEIGHT) {\n            offset = Math.floor((parentBox.height - MIN_TOOLTIP_HEIGHT) / 2);\n        } else {\n            // The tooltip is larger than the parent height: figure out what offset\n            // we need so that we're still centered.\n            offset = Math.floor(parentBox.height - MIN_TOOLTIP_HEIGHT);\n        }\n\n        const baseTop = (parentBox.top - 2 + this.props.yOffset) + window.pageYOffset;\n        const top = baseTop + offset;\n        const right = window.innerWidth - parentBox.right - window.pageXOffset - 16;\n        const left = parentBox.right + window.pageXOffset + 6;\n        const horizontalCenter = parentBox.right - window.pageXOffset - (parentBox.width / 2);\n        switch (this.props.alignment) {\n            case Alignment.Natural:\n                if (parentBox.right > window.innerWidth / 2) {\n                    style.right = right;\n                    style.top = top;\n                    break;\n                }\n                // fall through to Right\n            case Alignment.Right:\n                style.left = left;\n                style.top = top;\n                break;\n            case Alignment.Left:\n                style.right = right;\n                style.top = top;\n                break;\n            case Alignment.Top:\n                style.top = baseTop - 16;\n                style.left = horizontalCenter;\n                break;\n            case Alignment.Bottom:\n                style.top = baseTop + parentBox.height;\n                style.left = horizontalCenter;\n                break;\n        }\n\n        return style;\n    }\n\n    private renderTooltip = () => {\n        // Add the parent's position to the tooltips, so it's correctly\n        // positioned, also taking into account any window zoom\n        // NOTE: The additional 6 pixels for the left position, is to take account of the\n        // tooltips chevron\n        const style = this.updatePosition({});\n        // Hide the entire container when not visible. This prevents flashing of the tooltip\n        // if it is not meant to be visible on first mount.\n        style.display = this.props.visible ? \"block\" : \"none\";\n\n        const tooltipClasses = classNames(\"mx_Tooltip\", this.props.tooltipClassName, {\n            \"mx_Tooltip_visible\": this.props.visible,\n            \"mx_Tooltip_invisible\": !this.props.visible,\n        });\n\n        const tooltip = (\n            <div className={tooltipClasses} style={style}>\n                <div className=\"mx_Tooltip_chevron\" />\n                { this.props.label }\n            </div>\n        );\n\n        // Render the tooltip manually, as we wish it not to be rendered within the parent\n        this.tooltip = ReactDOM.render<Element>(tooltip, this.tooltipContainer);\n    };\n\n    public render() {\n        // Render a placeholder\n        return (\n            <div className={this.props.className}>\n            </div>\n        );\n    }\n}\n"]}