2020-10-20 13:10:02 -04:00
|
|
|
// @flow
|
|
|
|
import { KNOWN_APP_DOMAINS } from 'config';
|
|
|
|
import * as ICONS from 'constants/icons';
|
|
|
|
import * as React from 'react';
|
|
|
|
import { isURIValid } from 'lbry-redux';
|
|
|
|
import Button from 'component/button';
|
|
|
|
import ClaimLink from 'component/claimLink';
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
href: string,
|
|
|
|
title?: string,
|
|
|
|
embed?: boolean,
|
|
|
|
children: React.Node,
|
|
|
|
parentCommentId?: string,
|
|
|
|
isMarkdownPost?: boolean,
|
2020-10-22 14:16:42 -04:00
|
|
|
simpleLinks?: boolean,
|
2020-10-20 13:10:02 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
function MarkdownLink(props: Props) {
|
2020-11-04 14:18:45 -05:00
|
|
|
const { children, href, title, embed = false, parentCommentId, isMarkdownPost, simpleLinks = false } = props;
|
|
|
|
|
2020-11-19 15:24:58 -05:00
|
|
|
let decodedUri;
|
|
|
|
try {
|
|
|
|
decodedUri = decodeURI(href);
|
|
|
|
} catch (e) {}
|
|
|
|
|
|
|
|
if (!href || !decodedUri) {
|
2020-10-20 13:10:02 -04:00
|
|
|
return children || null;
|
|
|
|
}
|
|
|
|
|
2020-10-22 14:06:28 -04:00
|
|
|
let element = <span>{children}</span>;
|
|
|
|
|
2020-10-20 13:10:02 -04:00
|
|
|
// Regex for url protocol
|
|
|
|
const protocolRegex = new RegExp('^(https?|lbry|mailto)+:', 'i');
|
|
|
|
const protocol = href ? protocolRegex.exec(href) : null;
|
2020-10-22 14:06:28 -04:00
|
|
|
|
2020-10-22 18:09:17 -04:00
|
|
|
let linkUrlObject;
|
|
|
|
try {
|
|
|
|
linkUrlObject = new URL(decodedUri);
|
|
|
|
} catch (e) {}
|
|
|
|
|
|
|
|
let lbryUrlFromLink;
|
2020-12-10 16:25:29 +08:00
|
|
|
if (linkUrlObject && !href.startsWith('mailto:')) {
|
2020-10-22 14:06:28 -04:00
|
|
|
const linkDomain = linkUrlObject.host;
|
|
|
|
const isKnownAppDomainLink = KNOWN_APP_DOMAINS.includes(linkDomain);
|
|
|
|
if (isKnownAppDomainLink) {
|
2020-11-04 14:18:45 -05:00
|
|
|
let linkPathname;
|
|
|
|
try {
|
|
|
|
// This could be anything
|
|
|
|
linkPathname = decodeURIComponent(
|
|
|
|
linkUrlObject.pathname.startsWith('//') ? linkUrlObject.pathname.slice(2) : linkUrlObject.pathname.slice(1)
|
|
|
|
);
|
|
|
|
} catch (e) {}
|
2020-10-23 13:14:12 -04:00
|
|
|
|
2020-11-04 14:18:45 -05:00
|
|
|
const linkPathPlusHash = linkPathname ? `${linkPathname}${linkUrlObject.hash}` : undefined;
|
|
|
|
const possibleLbryUrl = linkPathPlusHash ? `lbry://${linkPathPlusHash.replace(/:/g, '#')}` : undefined;
|
2020-10-23 13:14:12 -04:00
|
|
|
|
2020-11-04 14:18:45 -05:00
|
|
|
const lbryLinkIsValid = possibleLbryUrl && isURIValid(possibleLbryUrl);
|
2020-10-22 14:06:28 -04:00
|
|
|
if (lbryLinkIsValid) {
|
|
|
|
lbryUrlFromLink = possibleLbryUrl;
|
|
|
|
}
|
2020-10-20 13:10:02 -04:00
|
|
|
}
|
2020-10-22 18:09:17 -04:00
|
|
|
}
|
2020-10-20 13:10:02 -04:00
|
|
|
|
2020-12-15 15:15:02 +08:00
|
|
|
if (href.startsWith('?t=')) {
|
|
|
|
// Video timestamp markers
|
|
|
|
element = (
|
|
|
|
<Button
|
|
|
|
button="link"
|
|
|
|
iconRight={undefined}
|
|
|
|
title={title || decodedUri}
|
|
|
|
label={children}
|
|
|
|
className="button--external-link"
|
|
|
|
onClick={() => {
|
|
|
|
if (window.player) {
|
|
|
|
window.player.currentTime(parseInt(href.substr(3)));
|
|
|
|
window.scrollTo(0, 0);
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
} else if (!simpleLinks && ((protocol && protocol[0] === 'lbry:' && isURIValid(decodedUri)) || lbryUrlFromLink)) {
|
|
|
|
// Return plain text if no valid url
|
|
|
|
// Return external link if protocol is http or https
|
|
|
|
// Return local link if protocol is lbry uri
|
2020-10-20 13:10:02 -04:00
|
|
|
element = (
|
|
|
|
<ClaimLink
|
|
|
|
uri={lbryUrlFromLink || decodedUri}
|
|
|
|
autoEmbed={embed}
|
|
|
|
parentCommentId={parentCommentId}
|
|
|
|
isMarkdownPost={isMarkdownPost}
|
|
|
|
>
|
|
|
|
{children}
|
|
|
|
</ClaimLink>
|
|
|
|
);
|
2020-10-22 14:16:42 -04:00
|
|
|
} else if (
|
|
|
|
simpleLinks ||
|
|
|
|
(protocol && (protocol[0] === 'http:' || protocol[0] === 'https:' || protocol[0] === 'mailto:'))
|
|
|
|
) {
|
2020-10-22 14:30:40 -04:00
|
|
|
const isLbryLink = href.startsWith('lbry://');
|
|
|
|
|
2020-10-20 13:10:02 -04:00
|
|
|
element = (
|
|
|
|
<Button
|
|
|
|
button="link"
|
2020-10-22 14:30:40 -04:00
|
|
|
iconRight={isLbryLink ? undefined : ICONS.EXTERNAL}
|
2020-10-20 13:10:02 -04:00
|
|
|
title={title || decodedUri}
|
|
|
|
label={children}
|
|
|
|
className="button--external-link"
|
2020-10-22 14:30:40 -04:00
|
|
|
navigate={isLbryLink ? href : undefined}
|
2020-11-04 14:18:45 -05:00
|
|
|
href={isLbryLink ? undefined : href}
|
2020-10-20 13:10:02 -04:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return <>{element}</>;
|
|
|
|
}
|
|
|
|
|
|
|
|
export default MarkdownLink;
|