2018-03-26 14:32:43 -07:00
|
|
|
// @flow
|
2019-06-28 03:27:55 -04:00
|
|
|
import type { Node } from 'react';
|
2019-11-15 16:53:31 -05:00
|
|
|
import React, { forwardRef, useRef } from 'react';
|
2018-03-26 14:32:43 -07:00
|
|
|
import Icon from 'component/common/icon';
|
|
|
|
import classnames from 'classnames';
|
2019-04-04 17:05:23 -04:00
|
|
|
import { NavLink } from 'react-router-dom';
|
2019-12-02 12:30:08 -05:00
|
|
|
import { formatLbryUrlForWeb } from 'util/url';
|
2019-10-15 14:53:55 -04:00
|
|
|
import * as PAGES from 'constants/pages';
|
2019-11-15 16:53:31 -05:00
|
|
|
import useCombinedRefs from 'effects/use-combined-refs';
|
2018-03-26 14:32:43 -07:00
|
|
|
|
|
|
|
type Props = {
|
2019-06-11 20:22:21 -06:00
|
|
|
id: ?string,
|
2018-03-26 14:32:43 -07:00
|
|
|
href: ?string,
|
|
|
|
title: ?string,
|
|
|
|
label: ?string,
|
2020-04-02 19:09:59 -04:00
|
|
|
largestLabel: ?string,
|
2018-03-26 14:32:43 -07:00
|
|
|
icon: ?string,
|
|
|
|
iconRight: ?string,
|
|
|
|
disabled: ?boolean,
|
2019-06-28 03:27:55 -04:00
|
|
|
children: ?Node,
|
2018-03-26 14:32:43 -07:00
|
|
|
navigate: ?string,
|
|
|
|
className: ?string,
|
|
|
|
description: ?string,
|
|
|
|
type: string,
|
|
|
|
button: ?string, // primary, secondary, alt, link
|
2018-12-19 00:44:53 -05:00
|
|
|
iconSize?: number,
|
2019-06-28 03:27:55 -04:00
|
|
|
iconColor?: string,
|
2019-03-28 12:53:13 -04:00
|
|
|
activeClass?: string,
|
2019-06-05 00:07:53 -06:00
|
|
|
innerRef: ?any,
|
2020-11-17 14:10:14 -05:00
|
|
|
authSrc?: string,
|
2019-06-05 00:07:53 -06:00
|
|
|
// Events
|
|
|
|
onClick: ?(any) => any,
|
|
|
|
onMouseEnter: ?(any) => any,
|
|
|
|
onMouseLeave: ?(any) => any,
|
2019-10-15 14:53:55 -04:00
|
|
|
pathname: string,
|
|
|
|
emailVerified: boolean,
|
|
|
|
requiresAuth: ?boolean,
|
|
|
|
myref: any,
|
|
|
|
dispatch: any,
|
2020-05-12 17:22:36 +08:00
|
|
|
'aria-label'?: string,
|
2021-06-17 18:52:21 -04:00
|
|
|
user: ?User,
|
2018-03-26 14:32:43 -07:00
|
|
|
};
|
|
|
|
|
2019-06-28 03:27:55 -04:00
|
|
|
// use forwardRef to allow consumers to pass refs to the button content if they want to
|
|
|
|
// flow requires forwardRef have default type arguments passed to it
|
|
|
|
const Button = forwardRef<any, {}>((props: Props, ref: any) => {
|
|
|
|
const {
|
|
|
|
type = 'button',
|
|
|
|
onClick,
|
|
|
|
href,
|
|
|
|
title,
|
|
|
|
label,
|
2020-04-02 19:09:59 -04:00
|
|
|
largestLabel,
|
2019-06-28 03:27:55 -04:00
|
|
|
icon,
|
2020-11-17 14:10:14 -05:00
|
|
|
|
2019-06-28 03:27:55 -04:00
|
|
|
// This should rarely be used. Regular buttons should just use `icon`
|
|
|
|
// `iconRight` is used for the header (home) button with the LBRY icon and external links that are displayed inline
|
|
|
|
iconRight,
|
|
|
|
disabled,
|
|
|
|
children,
|
|
|
|
navigate,
|
|
|
|
className,
|
|
|
|
description,
|
|
|
|
button,
|
|
|
|
iconSize,
|
|
|
|
iconColor,
|
|
|
|
activeClass,
|
2019-10-15 14:53:55 -04:00
|
|
|
emailVerified,
|
|
|
|
requiresAuth,
|
|
|
|
myref,
|
|
|
|
dispatch, // <button> doesn't know what to do with dispatch
|
|
|
|
pathname,
|
2021-06-17 18:52:21 -04:00
|
|
|
user,
|
2020-11-17 14:10:14 -05:00
|
|
|
authSrc,
|
2019-06-28 03:27:55 -04:00
|
|
|
...otherProps
|
|
|
|
} = props;
|
2018-03-26 14:32:43 -07:00
|
|
|
|
2021-06-17 18:52:21 -04:00
|
|
|
const disable = disabled || (user === null && requiresAuth);
|
|
|
|
|
2019-06-28 03:27:55 -04:00
|
|
|
const combinedClassName = classnames(
|
|
|
|
'button',
|
|
|
|
button
|
|
|
|
? {
|
|
|
|
'button--primary': button === 'primary',
|
|
|
|
'button--secondary': button === 'secondary',
|
|
|
|
'button--alt': button === 'alt',
|
|
|
|
'button--inverse': button === 'inverse',
|
|
|
|
'button--close': button === 'close',
|
2021-06-17 18:52:21 -04:00
|
|
|
'button--disabled': disable,
|
2019-06-28 03:27:55 -04:00
|
|
|
'button--link': button === 'link',
|
|
|
|
}
|
|
|
|
: 'button--no-style',
|
|
|
|
className
|
|
|
|
);
|
2018-03-26 14:32:43 -07:00
|
|
|
|
2019-11-15 16:53:31 -05:00
|
|
|
const innerRef = useRef(null);
|
|
|
|
const combinedRef = useCombinedRefs(ref, innerRef, myref);
|
2020-04-01 14:43:50 -04:00
|
|
|
const size = iconSize || (!label && !children) ? 18 : undefined; // Fall back to default
|
2019-11-15 16:53:31 -05:00
|
|
|
|
2021-07-13 15:19:52 -05:00
|
|
|
// Label can be a string or object ( use title instead )
|
|
|
|
const ariaLabel = description || (typeof label === 'string' ? label : title);
|
|
|
|
|
2019-06-28 03:27:55 -04:00
|
|
|
const content = (
|
2020-04-10 20:46:03 -04:00
|
|
|
<span className="button__content">
|
2021-07-14 13:44:22 -05:00
|
|
|
{icon && <Icon icon={icon} iconColor={iconColor} size={iconSize} />}
|
2020-04-10 20:24:23 -04:00
|
|
|
|
2020-09-23 11:00:10 -04:00
|
|
|
{!largestLabel && label && (
|
|
|
|
<span dir="auto" className="button__label">
|
|
|
|
{label}
|
|
|
|
</span>
|
|
|
|
)}
|
2020-04-17 16:33:02 -04:00
|
|
|
|
|
|
|
{/* largestLabel is used when a single button has two different labels based on hover state */}
|
|
|
|
{largestLabel && (
|
2020-08-20 13:27:28 +02:00
|
|
|
<div dir="auto" className="button__label" style={{ position: 'relative' }}>
|
2020-04-10 20:24:23 -04:00
|
|
|
<div
|
|
|
|
style={{
|
|
|
|
position: 'relative',
|
|
|
|
left: '50%',
|
|
|
|
top: '50%',
|
|
|
|
transform: `translate(-50%, 0%)`,
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<span style={{ visibility: 'hidden' }}>
|
|
|
|
{largestLabel || label}
|
2020-04-03 19:29:39 -04:00
|
|
|
<div
|
|
|
|
style={{
|
2020-04-10 20:24:23 -04:00
|
|
|
position: 'absolute',
|
2020-04-03 19:29:39 -04:00
|
|
|
left: '50%',
|
2020-04-10 20:24:23 -04:00
|
|
|
top: '50%',
|
2020-04-03 19:29:39 -04:00
|
|
|
transform: `translate(-50%, -50%)`,
|
|
|
|
}}
|
|
|
|
>
|
2020-04-12 15:10:47 -04:00
|
|
|
<span style={{ visibility: 'visible' }}>{label}</span>
|
2020-04-03 19:29:39 -04:00
|
|
|
</div>
|
2020-04-10 20:24:23 -04:00
|
|
|
</span>
|
|
|
|
</div>
|
2020-04-03 19:29:39 -04:00
|
|
|
</div>
|
2020-04-10 20:24:23 -04:00
|
|
|
)}
|
|
|
|
|
2019-06-28 03:27:55 -04:00
|
|
|
{children && children}
|
2021-07-14 13:44:22 -05:00
|
|
|
{iconRight && <Icon icon={iconRight} iconColor={iconColor} size={size} />}
|
2019-06-28 03:27:55 -04:00
|
|
|
</span>
|
|
|
|
);
|
2020-06-04 10:56:57 -04:00
|
|
|
|
2020-09-23 11:00:10 -04:00
|
|
|
if (href || (navigate && navigate.startsWith('http'))) {
|
|
|
|
// TODO: replace the below with an outbound link tracker for matomo
|
2019-06-28 03:27:55 -04:00
|
|
|
return (
|
2020-09-23 11:00:10 -04:00
|
|
|
<a
|
|
|
|
target="_blank"
|
|
|
|
rel="noopener noreferrer"
|
|
|
|
href={href || navigate}
|
|
|
|
className={combinedClassName}
|
2020-10-02 12:12:06 +08:00
|
|
|
title={title}
|
2020-09-23 11:00:10 -04:00
|
|
|
onClick={onClick}
|
2021-07-13 16:38:50 -05:00
|
|
|
aria-label={ariaLabel}
|
2021-09-18 10:23:30 -04:00
|
|
|
disabled={disabled} // is there a reason this wasn't here before?
|
2020-09-23 11:00:10 -04:00
|
|
|
>
|
2019-06-28 03:27:55 -04:00
|
|
|
{content}
|
2020-06-02 16:52:34 -04:00
|
|
|
</a>
|
2018-03-26 14:32:43 -07:00
|
|
|
);
|
2019-06-28 03:27:55 -04:00
|
|
|
}
|
2018-03-26 14:32:43 -07:00
|
|
|
|
2019-06-28 03:27:55 -04:00
|
|
|
// Handle lbry:// uris passed in, or already formatted web urls
|
|
|
|
let path = navigate;
|
|
|
|
if (path) {
|
|
|
|
if (path.startsWith('lbry://')) {
|
2019-12-02 12:30:08 -05:00
|
|
|
path = formatLbryUrlForWeb(path);
|
2019-06-28 03:27:55 -04:00
|
|
|
} else if (!path.startsWith('/')) {
|
|
|
|
// Force a leading slash so new paths aren't appended on to the current path
|
|
|
|
path = `/${path}`;
|
2019-03-28 12:53:13 -04:00
|
|
|
}
|
2018-03-26 14:32:43 -07:00
|
|
|
}
|
2019-06-28 03:27:55 -04:00
|
|
|
|
2020-05-12 17:22:36 +08:00
|
|
|
// Try to generate a tooltip using available text and display it through
|
|
|
|
// the 'title' mechanism, but only if it isn't being used.
|
|
|
|
let defaultTooltip;
|
|
|
|
if (!title) {
|
|
|
|
if (props['aria-label']) {
|
|
|
|
defaultTooltip = props['aria-label'];
|
|
|
|
} else if (description) {
|
|
|
|
defaultTooltip = description;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-15 14:53:55 -04:00
|
|
|
if (requiresAuth && !emailVerified) {
|
2020-11-17 14:10:14 -05:00
|
|
|
let redirectUrl = `/$/${PAGES.AUTH}?redirect=${pathname}`;
|
|
|
|
|
|
|
|
if (authSrc) {
|
|
|
|
redirectUrl += `&src=${authSrc}`;
|
|
|
|
}
|
|
|
|
|
2019-10-15 14:53:55 -04:00
|
|
|
return (
|
|
|
|
<NavLink
|
|
|
|
exact
|
2021-06-17 18:52:21 -04:00
|
|
|
onClick={(e) => {
|
2019-10-23 14:59:33 -04:00
|
|
|
e.stopPropagation();
|
|
|
|
}}
|
2020-11-17 14:10:14 -05:00
|
|
|
to={redirectUrl}
|
2020-05-12 17:22:36 +08:00
|
|
|
title={title || defaultTooltip}
|
2021-06-17 18:52:21 -04:00
|
|
|
disabled={disable}
|
2019-10-15 14:53:55 -04:00
|
|
|
className={combinedClassName}
|
|
|
|
activeClassName={activeClass}
|
2021-07-13 16:38:50 -05:00
|
|
|
aria-label={ariaLabel}
|
2019-10-15 14:53:55 -04:00
|
|
|
>
|
|
|
|
{content}
|
|
|
|
</NavLink>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-06-28 03:27:55 -04:00
|
|
|
return path ? (
|
|
|
|
<NavLink
|
|
|
|
exact
|
|
|
|
to={path}
|
2020-05-12 17:22:36 +08:00
|
|
|
title={title || defaultTooltip}
|
2021-06-17 18:52:21 -04:00
|
|
|
disabled={disable}
|
|
|
|
onClick={(e) => {
|
2019-06-28 03:27:55 -04:00
|
|
|
e.stopPropagation();
|
|
|
|
if (onClick) {
|
|
|
|
onClick();
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
className={combinedClassName}
|
|
|
|
activeClassName={activeClass}
|
2021-07-13 16:38:50 -05:00
|
|
|
aria-label={ariaLabel}
|
2020-01-06 13:32:35 -05:00
|
|
|
{...otherProps}
|
2019-06-28 03:27:55 -04:00
|
|
|
>
|
|
|
|
{content}
|
|
|
|
</NavLink>
|
|
|
|
) : (
|
|
|
|
<button
|
2019-11-15 16:53:31 -05:00
|
|
|
ref={combinedRef}
|
2020-05-12 17:22:36 +08:00
|
|
|
title={title || defaultTooltip}
|
2021-07-13 15:19:52 -05:00
|
|
|
aria-label={ariaLabel}
|
2019-06-28 03:27:55 -04:00
|
|
|
className={combinedClassName}
|
2021-06-17 18:52:21 -04:00
|
|
|
onClick={(e) => {
|
2019-08-13 01:35:13 -04:00
|
|
|
if (onClick) {
|
|
|
|
e.stopPropagation();
|
|
|
|
onClick(e);
|
|
|
|
}
|
|
|
|
}}
|
2021-06-17 18:52:21 -04:00
|
|
|
disabled={disable}
|
2019-06-28 03:27:55 -04:00
|
|
|
type={type}
|
|
|
|
{...otherProps}
|
|
|
|
>
|
|
|
|
{content}
|
|
|
|
</button>
|
|
|
|
);
|
|
|
|
});
|
2018-03-26 14:32:43 -07:00
|
|
|
|
|
|
|
export default Button;
|