1
mirror of https://github.com/jakejarvis/jarv.is.git synced 2025-04-26 22:48:29 -04:00

59 lines
1.6 KiB
TypeScript

import NextLink from "next/link";
import clsx from "clsx";
import objStr from "obj-str";
import type { ComponentPropsWithoutRef } from "react";
import styles from "./Link.module.css";
export type LinkProps = ComponentPropsWithoutRef<typeof NextLink> & {
underline?: boolean;
openInNewTab?: boolean;
};
const Link = ({
href,
rel,
target,
prefetch = false,
underline = true,
openInNewTab,
className,
...rest
}: LinkProps) => {
// This component auto-detects whether or not this link should open in the same window (the default for internal
// links) or a new tab (the default for external links). Defaults can be overridden with `openInNewTab={true}`.
const isExternal =
typeof href === "string" &&
!(
["/", "#"].includes(href[0]) ||
(process.env.NEXT_PUBLIC_BASE_URL && href.startsWith(process.env.NEXT_PUBLIC_BASE_URL))
);
if (openInNewTab || isExternal) {
return (
<NextLink
href={href}
target={target || "_blank"}
rel={objStr({
[`${rel}`]: rel, // prepend whatever string is passed via optional `rel` prop
noopener: true,
noreferrer: isExternal, // don't add "noreferrer" if link isn't external, and only opening in a new tab
})}
prefetch={false}
className={clsx(styles.link, underline && styles.underline, className)}
{...rest}
/>
);
}
// If link is to an internal page, simply pass *everything* along as-is to next/link.
return (
<NextLink
className={clsx(styles.link, underline && styles.underline, className)}
{...{ href, rel, target, prefetch, ...rest }}
/>
);
};
export default Link;