2022-03-30 09:49:48 +00:00
|
|
|
import type { AnchorHTMLAttributes, ButtonHTMLAttributes } from 'react';
|
|
|
|
import { forwardRef } from 'react';
|
2022-03-01 09:26:26 +00:00
|
|
|
import classNames from 'classnames';
|
2022-03-30 09:49:48 +00:00
|
|
|
import type { IconName } from '../icon';
|
|
|
|
import { Icon } from '../icon';
|
2022-03-08 14:03:50 +00:00
|
|
|
import {
|
2022-03-09 13:05:42 +00:00
|
|
|
includesLeftPadding,
|
|
|
|
includesRightPadding,
|
|
|
|
includesBorderWidth,
|
2022-04-20 19:37:44 +00:00
|
|
|
includesHeight,
|
2022-03-08 14:03:50 +00:00
|
|
|
} from '../../utils/class-names';
|
2022-03-07 12:27:56 +00:00
|
|
|
|
|
|
|
interface CommonProps {
|
2022-03-01 09:26:26 +00:00
|
|
|
children?: React.ReactNode;
|
2022-03-22 15:25:45 +00:00
|
|
|
variant?: 'primary' | 'secondary' | 'accent' | 'inline' | 'inline-link';
|
2022-03-01 09:26:26 +00:00
|
|
|
className?: string;
|
2022-03-02 09:56:05 +00:00
|
|
|
prependIconName?: IconName;
|
|
|
|
appendIconName?: IconName;
|
2022-03-01 09:26:26 +00:00
|
|
|
}
|
2022-03-07 12:27:56 +00:00
|
|
|
export interface ButtonProps
|
|
|
|
extends ButtonHTMLAttributes<HTMLButtonElement>,
|
|
|
|
CommonProps {}
|
|
|
|
|
|
|
|
export interface AnchorButtonProps
|
|
|
|
extends AnchorHTMLAttributes<HTMLAnchorElement>,
|
|
|
|
CommonProps {}
|
2022-03-01 09:26:26 +00:00
|
|
|
|
2022-03-23 15:48:50 +00:00
|
|
|
const getClasses = (
|
|
|
|
variant: CommonProps['variant'],
|
|
|
|
paddingLeftProvided: boolean,
|
|
|
|
paddingRightProvided: boolean,
|
2022-04-20 19:37:44 +00:00
|
|
|
borderWidthProvided: boolean,
|
|
|
|
heightProvided: boolean
|
2022-03-07 21:16:06 +00:00
|
|
|
) => {
|
2022-03-23 15:48:50 +00:00
|
|
|
// Add classes into variables if there are multiple classes shared in multiple button styles
|
|
|
|
const sharedClasses =
|
2022-04-20 19:37:44 +00:00
|
|
|
'inline-flex items-center justify-center box-border transition-all disabled:no-underline';
|
2022-03-23 15:48:50 +00:00
|
|
|
const underlineOnHover = 'no-underline hover:underline';
|
|
|
|
const commonHoverAndActiveBorder =
|
|
|
|
'hover:border-black dark:hover:border-white active:border-black dark:active:border-white';
|
|
|
|
const commonDisabled =
|
|
|
|
'disabled:bg-black-10 dark:disabled:bg-white-10 disabled:text-black-60 dark:disabled:text-white-60 disabled:border-black-25 dark:disabled:border-white-25';
|
|
|
|
const inlineTextColour =
|
|
|
|
'text-black-95 dark:text-white-95 hover:text-black hover:dark:text-white active:text-black dark:active:text-vega-yellow';
|
2022-03-09 13:05:42 +00:00
|
|
|
|
2022-03-23 15:48:50 +00:00
|
|
|
const standardButtonPaddingLeft = `${
|
|
|
|
paddingLeftProvided ? paddingLeftProvided : 'pl-28'
|
|
|
|
}`;
|
|
|
|
const standardButtonPaddingRight = `${
|
|
|
|
paddingRightProvided ? paddingRightProvided : 'pr-28'
|
|
|
|
}`;
|
|
|
|
const inlineButtonPaddingLeft = `${
|
|
|
|
paddingLeftProvided ? paddingLeftProvided : 'pl-4'
|
|
|
|
}`;
|
|
|
|
const inlineButtonPaddingRight = `${
|
|
|
|
paddingRightProvided ? paddingRightProvided : 'pr-4'
|
|
|
|
}`;
|
|
|
|
const standardButtonBorderWidth = `${
|
|
|
|
borderWidthProvided ? borderWidthProvided : 'border'
|
|
|
|
}`;
|
2022-04-20 19:37:44 +00:00
|
|
|
const buttonHeight = `${heightProvided ? heightProvided : 'h-28'}`;
|
2022-03-03 11:11:36 +00:00
|
|
|
|
2022-03-23 15:48:50 +00:00
|
|
|
const primaryClasses = [
|
|
|
|
sharedClasses,
|
|
|
|
commonHoverAndActiveBorder,
|
|
|
|
underlineOnHover,
|
|
|
|
commonDisabled,
|
|
|
|
standardButtonPaddingLeft,
|
|
|
|
standardButtonPaddingRight,
|
|
|
|
standardButtonBorderWidth,
|
2022-04-20 19:37:44 +00:00
|
|
|
buttonHeight,
|
2022-03-23 15:48:50 +00:00
|
|
|
'bg-black dark:bg-white hover:bg-black-80 dark:hover:bg-white-80 active:bg-white dark:active:bg-black',
|
|
|
|
'text-ui text-white dark:text-black active:text-black dark:active:text-white',
|
|
|
|
];
|
2022-03-01 09:26:26 +00:00
|
|
|
|
2022-03-23 15:48:50 +00:00
|
|
|
const secondaryClasses = [
|
|
|
|
sharedClasses,
|
|
|
|
commonHoverAndActiveBorder,
|
|
|
|
underlineOnHover,
|
|
|
|
commonDisabled,
|
|
|
|
standardButtonPaddingLeft,
|
|
|
|
standardButtonPaddingRight,
|
|
|
|
standardButtonBorderWidth,
|
2022-04-20 19:37:44 +00:00
|
|
|
buttonHeight,
|
2022-03-23 15:48:50 +00:00
|
|
|
'bg-white dark:bg-black hover:bg-black-25 dark:hover:bg-white-25 active:bg-black dark:active:bg-white',
|
|
|
|
'text-ui text-black dark:text-white active:text-white dark:active:text-black',
|
|
|
|
'border-black-60 dark:border-white-60 hover:border-black',
|
|
|
|
];
|
2022-03-03 14:48:40 +00:00
|
|
|
|
2022-03-23 15:48:50 +00:00
|
|
|
const accentClasses = [
|
|
|
|
sharedClasses,
|
|
|
|
commonHoverAndActiveBorder,
|
|
|
|
underlineOnHover,
|
|
|
|
commonDisabled,
|
|
|
|
standardButtonPaddingLeft,
|
|
|
|
standardButtonPaddingRight,
|
|
|
|
standardButtonBorderWidth,
|
2022-04-20 19:37:44 +00:00
|
|
|
buttonHeight,
|
2022-05-09 15:19:19 +00:00
|
|
|
'bg-vega-yellow dark:bg-vega-yellow hover:bg-yellow/dark dark:hover:bg-vega-yellow/30 active:bg-white dark:active:bg-black',
|
2022-03-23 15:48:50 +00:00
|
|
|
'text-ui uppercase text-black dark:text-black hover:text-white dark:hover:text-white active:text-black dark:active:text-white',
|
|
|
|
'border-transparent dark:border-transparent',
|
|
|
|
];
|
2022-03-01 09:26:26 +00:00
|
|
|
|
2022-03-23 15:48:50 +00:00
|
|
|
const inlineClasses = [
|
|
|
|
sharedClasses,
|
|
|
|
inlineButtonPaddingLeft,
|
|
|
|
inlineButtonPaddingRight,
|
2022-04-20 19:37:44 +00:00
|
|
|
buttonHeight,
|
2022-03-23 15:48:50 +00:00
|
|
|
inlineTextColour,
|
|
|
|
'border-none',
|
|
|
|
'text-ui',
|
|
|
|
];
|
2022-03-01 09:26:26 +00:00
|
|
|
|
2022-03-23 15:48:50 +00:00
|
|
|
const inlineLinkClasses = [
|
|
|
|
sharedClasses,
|
|
|
|
inlineButtonPaddingLeft,
|
|
|
|
inlineButtonPaddingRight,
|
2022-04-20 19:37:44 +00:00
|
|
|
buttonHeight,
|
2022-03-23 15:48:50 +00:00
|
|
|
inlineTextColour,
|
|
|
|
'underline hover:underline',
|
|
|
|
'border-none',
|
|
|
|
];
|
2022-03-03 14:48:40 +00:00
|
|
|
|
2022-03-23 15:48:50 +00:00
|
|
|
switch (variant) {
|
|
|
|
case 'primary':
|
|
|
|
return primaryClasses;
|
|
|
|
case 'secondary':
|
|
|
|
return secondaryClasses;
|
|
|
|
case 'accent':
|
|
|
|
return accentClasses;
|
|
|
|
case 'inline':
|
|
|
|
return inlineClasses;
|
|
|
|
case 'inline-link':
|
|
|
|
return inlineLinkClasses;
|
|
|
|
default:
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const classes = (
|
|
|
|
className: CommonProps['className'],
|
|
|
|
variant: CommonProps['variant']
|
|
|
|
) => {
|
|
|
|
const paddingLeftProvided = includesLeftPadding(className);
|
|
|
|
const paddingRightProvided = includesRightPadding(className);
|
|
|
|
const borderWidthProvided = includesBorderWidth(className);
|
2022-04-20 19:37:44 +00:00
|
|
|
const heightProvided = includesHeight(className);
|
2022-03-23 15:48:50 +00:00
|
|
|
|
|
|
|
return classNames(
|
|
|
|
getClasses(
|
|
|
|
variant,
|
|
|
|
paddingLeftProvided,
|
|
|
|
paddingRightProvided,
|
2022-04-20 19:37:44 +00:00
|
|
|
borderWidthProvided,
|
|
|
|
heightProvided
|
2022-03-23 15:48:50 +00:00
|
|
|
),
|
2022-03-01 09:26:26 +00:00
|
|
|
className
|
|
|
|
);
|
2022-03-07 21:16:06 +00:00
|
|
|
};
|
2022-03-07 12:27:56 +00:00
|
|
|
|
|
|
|
const getContent = (
|
|
|
|
children: React.ReactNode,
|
|
|
|
prependIconName?: IconName,
|
|
|
|
appendIconName?: IconName
|
|
|
|
) => {
|
2022-03-02 09:56:05 +00:00
|
|
|
const iconName = prependIconName || appendIconName;
|
2022-03-07 12:27:56 +00:00
|
|
|
if (iconName === undefined) {
|
|
|
|
return children;
|
2022-03-02 09:56:05 +00:00
|
|
|
}
|
2022-03-07 12:27:56 +00:00
|
|
|
const iconClassName = classNames(['fill-current'], {
|
|
|
|
'mr-8': prependIconName,
|
|
|
|
'ml-8': appendIconName,
|
|
|
|
});
|
|
|
|
const icon = <Icon name={iconName} className={iconClassName} size={16} />;
|
2022-03-01 09:26:26 +00:00
|
|
|
return (
|
2022-03-07 12:27:56 +00:00
|
|
|
<>
|
2022-03-02 09:56:05 +00:00
|
|
|
{prependIconName && icon}
|
2022-03-01 09:26:26 +00:00
|
|
|
{children}
|
2022-03-02 09:56:05 +00:00
|
|
|
{appendIconName && icon}
|
2022-03-07 12:27:56 +00:00
|
|
|
</>
|
2022-03-01 09:26:26 +00:00
|
|
|
);
|
2022-03-07 12:27:56 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
export const Button = forwardRef<HTMLButtonElement, ButtonProps>(
|
|
|
|
(
|
|
|
|
{
|
|
|
|
variant = 'primary',
|
Feat/63 Deal ticket (#82)
* scaffold dealticket package, remove trading views from react-helpers
* add deal ticket component, add intent utils, expand dialog and form group styles
* add splash component, show market not found message if market doesnt exist
* tidy up error handling
* add handleError method for vega tx hook
* add better testname for provider test, flesh out tests a bit more for deal ticket
* Add unit tests for useVegaTransaction and useOrderSubmit hooks
* add wrapper component for order dialog styles
* add vega styled loader to ui toolkit and use in order dialog
* add title prop to order dialog
* split limit and market tickets into own files
* add button radio component
* revert dialog styles
* move splash component to ui-toolkit, add story
* convert intent to enum
* Make button always type=button unless type prop is passed
* inline filter logic for tif selector
* add date-fns, add datetime to helpers
* add order types to wallet package, make price undefined if order type is market
* use enums in deal ticket logic
* tidy up order state by moving submit and transaction hooks out of deal ticket
* add comment for dialog styles
* remove decimal from price input
* add types package, delete old generated types from trading project
* rename types package to graphql
* update generate command to point to correct locations
* fix use order submit test
* use intent shadow helper
* remove date-fns and format manually, update submit button error to use input-error
* remove stray console.log
2022-03-17 19:35:46 +00:00
|
|
|
type = 'button',
|
2022-03-07 12:27:56 +00:00
|
|
|
children,
|
|
|
|
className,
|
|
|
|
prependIconName,
|
|
|
|
appendIconName,
|
2022-03-07 20:45:39 +00:00
|
|
|
...props
|
2022-03-07 12:27:56 +00:00
|
|
|
},
|
|
|
|
ref
|
|
|
|
) => {
|
|
|
|
return (
|
Feat/63 Deal ticket (#82)
* scaffold dealticket package, remove trading views from react-helpers
* add deal ticket component, add intent utils, expand dialog and form group styles
* add splash component, show market not found message if market doesnt exist
* tidy up error handling
* add handleError method for vega tx hook
* add better testname for provider test, flesh out tests a bit more for deal ticket
* Add unit tests for useVegaTransaction and useOrderSubmit hooks
* add wrapper component for order dialog styles
* add vega styled loader to ui toolkit and use in order dialog
* add title prop to order dialog
* split limit and market tickets into own files
* add button radio component
* revert dialog styles
* move splash component to ui-toolkit, add story
* convert intent to enum
* Make button always type=button unless type prop is passed
* inline filter logic for tif selector
* add date-fns, add datetime to helpers
* add order types to wallet package, make price undefined if order type is market
* use enums in deal ticket logic
* tidy up order state by moving submit and transaction hooks out of deal ticket
* add comment for dialog styles
* remove decimal from price input
* add types package, delete old generated types from trading project
* rename types package to graphql
* update generate command to point to correct locations
* fix use order submit test
* use intent shadow helper
* remove date-fns and format manually, update submit button error to use input-error
* remove stray console.log
2022-03-17 19:35:46 +00:00
|
|
|
<button
|
|
|
|
ref={ref}
|
2022-03-23 15:48:50 +00:00
|
|
|
className={classes(className, variant)}
|
Feat/63 Deal ticket (#82)
* scaffold dealticket package, remove trading views from react-helpers
* add deal ticket component, add intent utils, expand dialog and form group styles
* add splash component, show market not found message if market doesnt exist
* tidy up error handling
* add handleError method for vega tx hook
* add better testname for provider test, flesh out tests a bit more for deal ticket
* Add unit tests for useVegaTransaction and useOrderSubmit hooks
* add wrapper component for order dialog styles
* add vega styled loader to ui toolkit and use in order dialog
* add title prop to order dialog
* split limit and market tickets into own files
* add button radio component
* revert dialog styles
* move splash component to ui-toolkit, add story
* convert intent to enum
* Make button always type=button unless type prop is passed
* inline filter logic for tif selector
* add date-fns, add datetime to helpers
* add order types to wallet package, make price undefined if order type is market
* use enums in deal ticket logic
* tidy up order state by moving submit and transaction hooks out of deal ticket
* add comment for dialog styles
* remove decimal from price input
* add types package, delete old generated types from trading project
* rename types package to graphql
* update generate command to point to correct locations
* fix use order submit test
* use intent shadow helper
* remove date-fns and format manually, update submit button error to use input-error
* remove stray console.log
2022-03-17 19:35:46 +00:00
|
|
|
type={type}
|
|
|
|
{...props}
|
|
|
|
>
|
2022-03-07 12:27:56 +00:00
|
|
|
{getContent(children, prependIconName, appendIconName)}
|
|
|
|
</button>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
export const AnchorButton = forwardRef<HTMLAnchorElement, AnchorButtonProps>(
|
|
|
|
(
|
|
|
|
{
|
|
|
|
variant = 'primary',
|
|
|
|
children,
|
|
|
|
className,
|
|
|
|
prependIconName,
|
|
|
|
appendIconName,
|
2022-06-06 16:19:56 +00:00
|
|
|
...props
|
2022-03-07 12:27:56 +00:00
|
|
|
},
|
|
|
|
ref
|
|
|
|
) => {
|
|
|
|
return (
|
2022-06-06 16:19:56 +00:00
|
|
|
<a ref={ref} className={classes(className, variant)} {...props}>
|
2022-03-07 12:27:56 +00:00
|
|
|
{getContent(children, prependIconName, appendIconName)}
|
|
|
|
</a>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|