* Upgrade to latest compound-web package * Use a custom render function for jest tests This way we don't need to manually wrap our components with <TooltipProvider> * Pin wrap-ansi to fix broken yarn install * Add playwright helper to find tooltip from element and use it in the failing test * Exclude floating-ui divs/spans from axe testing This is rendered outside .MatrixChat by compound and contains all the tooltips. * Wrap outermost components with TooltipProvider * Remove onChange and use onSelect for toggle * Fix jest tests and update snapshots * Use vector-im/matrix-wysiwig --------- Co-authored-by: Michael Telatynski <7t3chguy@gmail.com>
85 lines
3.1 KiB
TypeScript
85 lines
3.1 KiB
TypeScript
/*
|
|
Copyright 2024 New Vector Ltd.
|
|
Copyright 2022 The Matrix.org Foundation C.I.C.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
|
Please see LICENSE files in the repository root for full details.
|
|
*/
|
|
|
|
import React from "react";
|
|
import ReactDOM from "react-dom";
|
|
import { TooltipProvider } from "@vector-im/compound-web";
|
|
|
|
import PlatformPeg from "../PlatformPeg";
|
|
import LinkWithTooltip from "../components/views/elements/LinkWithTooltip";
|
|
|
|
/**
|
|
* If the platform enabled needsUrlTooltips, recurses depth-first through a DOM tree, adding tooltip previews
|
|
* for link elements. Otherwise, does nothing.
|
|
*
|
|
* @param {Element[]} rootNodes - a list of sibling DOM nodes to traverse to try
|
|
* to add tooltips.
|
|
* @param {Element[]} ignoredNodes: a list of nodes to not recurse into.
|
|
* @param {Element[]} containers: an accumulator of the DOM nodes which contain
|
|
* React components that have been mounted by this function. The initial caller
|
|
* should pass in an empty array to seed the accumulator.
|
|
*/
|
|
export function tooltipifyLinks(rootNodes: ArrayLike<Element>, ignoredNodes: Element[], containers: Element[]): void {
|
|
if (!PlatformPeg.get()?.needsUrlTooltips()) {
|
|
return;
|
|
}
|
|
|
|
let node = rootNodes[0];
|
|
|
|
while (node) {
|
|
if (ignoredNodes.includes(node) || containers.includes(node)) {
|
|
node = node.nextSibling as Element;
|
|
continue;
|
|
}
|
|
|
|
if (
|
|
node.tagName === "A" &&
|
|
node.getAttribute("href") &&
|
|
node.getAttribute("href") !== node.textContent?.trim()
|
|
) {
|
|
let href = node.getAttribute("href")!;
|
|
try {
|
|
href = new URL(href, window.location.href).toString();
|
|
} catch (e) {
|
|
// Not all hrefs will be valid URLs
|
|
}
|
|
|
|
// The node's innerHTML was already sanitized before being rendered in the first place, here we are just
|
|
// wrapping the link with the LinkWithTooltip component, keeping the same children. Ideally we'd do this
|
|
// without the superfluous span but this is not something React trivially supports at this time.
|
|
const tooltip = (
|
|
<TooltipProvider>
|
|
<LinkWithTooltip tooltip={href}>
|
|
<span dangerouslySetInnerHTML={{ __html: node.innerHTML }} />
|
|
</LinkWithTooltip>
|
|
</TooltipProvider>
|
|
);
|
|
|
|
ReactDOM.render(tooltip, node);
|
|
containers.push(node);
|
|
} else if (node.childNodes?.length) {
|
|
tooltipifyLinks(node.childNodes as NodeListOf<Element>, ignoredNodes, containers);
|
|
}
|
|
|
|
node = node.nextSibling as Element;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Unmount tooltip containers created by tooltipifyLinks.
|
|
*
|
|
* It's critical to call this after tooltipifyLinks, otherwise
|
|
* tooltips will leak.
|
|
*
|
|
* @param {Element[]} containers - array of tooltip containers to unmount
|
|
*/
|
|
export function unmountTooltips(containers: Element[]): void {
|
|
for (const container of containers) {
|
|
ReactDOM.unmountComponentAtNode(container);
|
|
}
|
|
}
|