TeaWeb/shared/js/text/bbcode/url.tsx

131 lines
5.3 KiB
TypeScript

import * as contextmenu from "tc-shared/ui/elements/ContextMenu";
import {copyToClipboard} from "tc-shared/utils/helpers";
import * as loader from "tc-loader";
import {ElementRenderer} from "vendor/xbbcode/renderer/base";
import {BBCodeTagElement} from "vendor/xbbcode/elements";
import * as React from "react";
import ReactRenderer from "vendor/xbbcode/renderer/react";
import {rendererReact, rendererText, BBCodeHandlerContext} from "tc-shared/text/bbcode/renderer";
import {ClientTag} from "tc-shared/ui/tree/EntryTags";
import {isYoutubeLink, YoutubeRenderer} from "tc-shared/text/bbcode/YoutubeRenderer";
import {InviteLinkRenderer, isInviteLink} from "tc-shared/text/bbcode/InviteRenderer";
function spawnUrlContextMenu(pageX: number, pageY: number, target: string) {
contextmenu.spawn_context_menu(pageX, pageY, {
callback: () => {
const win = window.open(target, '_blank');
win.focus();
},
name: tr("Open URL"),
type: contextmenu.MenuEntryType.ENTRY,
icon_class: "client-browse-addon-online"
}, {
callback: () => {
//TODO
},
name: tr("Open URL in Browser"),
type: contextmenu.MenuEntryType.ENTRY,
visible: __build.target === "client" && false // Currently not possible
}, contextmenu.Entry.HR(), {
callback: () => copyToClipboard(target),
name: tr("Copy URL to clipboard"),
type: contextmenu.MenuEntryType.ENTRY,
icon_class: "client-copy"
});
}
const ClientUrlRegex = /client:\/\/([0-9]+)\/([-A-Za-z0-9+/=]+)~/g;
export const SimpleUrlRenderer = (props: { target: string, children }) => {
return (
<a className={"xbbcode xbbcode-tag-url"} href={props.target} target={"_blank"} onContextMenu={event => {
event.preventDefault();
spawnUrlContextMenu(event.pageX, event.pageY, props.target);
}}>
{props.children}
</a>
);
}
loader.register_task(loader.Stage.JAVASCRIPT_INITIALIZING, {
name: "XBBCode code tag init",
function: async () => {
let reactId = 0;
const regexUrl = /^(?:[a-zA-Z]{1,16}):(?:\/{1,3}|\\)[-a-zA-Z0-9:;,@#%&()~_?+=\/\\.]*$/g;
rendererReact.registerCustomRenderer(new class extends ElementRenderer<BBCodeTagElement, React.ReactNode> {
render(element: BBCodeTagElement, renderer: ReactRenderer): React.ReactNode {
let target: string;
if (!element.options) {
target = rendererText.render(element);
} else {
target = element.options;
}
regexUrl.lastIndex = 0;
if (!regexUrl.test(target)) {
target = '#';
}
return (
<BBCodeHandlerContext.Consumer key={"er-" + ++reactId}>
{handlerId => {
if(handlerId) {
/* TS3-Protocol for a client */
if(target.match(ClientUrlRegex)) {
const clientData = target.match(ClientUrlRegex);
const clientDatabaseId = parseInt(clientData[1]);
const clientUniqueId = clientDatabaseId[2];
return (
<ClientTag
key={"er-" + ++reactId}
clientName={rendererText.renderContent(element).join("")}
clientUniqueId={clientUniqueId}
clientDatabaseId={clientDatabaseId > 0 ? clientDatabaseId : undefined}
handlerId={handlerId}
/>
);
}
if(isInviteLink(target)) {
return <InviteLinkRenderer key={"er-" + ++reactId} handlerId={handlerId} url={target} />;
}
}
const body = (
<SimpleUrlRenderer key={"er-" + ++reactId} target={target}>
{renderer.renderContent(element)}
</SimpleUrlRenderer>
);
if(isYoutubeLink(target)) {
return (
<YoutubeRenderer url={target}>{body}</YoutubeRenderer>
);
} else {
return body;
}
}}
</BBCodeHandlerContext.Consumer>
);
}
tags(): string | string[] {
return "url";
}
});
},
priority: 10
});
export function fixupJQueryUrlTags(container: JQuery) {
container.find("a").on('contextmenu', event => {
if(event.isDefaultPrevented())
return;
event.preventDefault();
spawnUrlContextMenu(event.pageX, event.pageY, $(event.target).attr("href"));
});
}