1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-01-25 00:07:47 +01:00
unleash.unleash/frontend/src/component/common/index.jsx
Christopher Kolstad 53354224fc
chore: Bump biome and configure husky (#6589)
Upgrades biome to 1.6.1, and updates husky pre-commit hook.

Most changes here are making type imports explicit.
2024-03-18 13:58:05 +01:00

175 lines
4.5 KiB
JavaScript

import React from 'react';
import PropTypes from 'prop-types';
import { Link } from 'react-router-dom';
import {
List,
MenuItem,
Icon,
ListItem,
ListItemText,
ListItemAvatar,
Button,
Avatar,
Typography,
} from '@mui/material';
import Apps from '@mui/icons-material/Apps';
import styles from './common.module.scss';
import { ConditionallyRender } from './ConditionallyRender/ConditionallyRender';
export { styles };
export const AppsLinkList = ({ apps }) => (
<List>
<ConditionallyRender
condition={apps.length > 0}
show={apps.map(({ appName, description, icon }) => (
<ListItem key={appName} className={styles.listItem}>
<ListItemAvatar>
<Avatar>
<ConditionallyRender
key={`avatar_conditional_${appName}`}
condition={icon}
show={<Icon>{icon}</Icon>}
elseShow={<Apps />}
/>
</Avatar>
</ListItemAvatar>
<ListItemText
primary={
<Link
to={`/applications/${encodeURIComponent(
appName,
)}`}
className={[
styles.listLink,
styles.truncate,
].join(' ')}
>
{appName}
</Link>
}
secondary={description || 'No description'}
/>
</ListItem>
))}
/>
</List>
);
AppsLinkList.propTypes = {
apps: PropTypes.array.isRequired,
};
export const DataTableHeader = ({ title, actions }) => (
<div className={styles.dataTableHeader}>
<div className={styles.title}>
<Typography variant='h2' className={styles.titleText}>
{title}
</Typography>
</div>
{actions && <div className={styles.actions}>{actions}</div>}
</div>
);
DataTableHeader.propTypes = {
title: PropTypes.string,
actions: PropTypes.any,
};
export const FormButtons = ({
submitText = 'Create',
onCancel,
primaryButtonTestId,
}) => (
<div>
<Button
data-testid={primaryButtonTestId}
type='submit'
color='primary'
variant='contained'
>
{submitText}
</Button>
&nbsp;
<Button type='cancel' onClick={onCancel}>
Cancel
</Button>
</div>
);
FormButtons.propTypes = {
submitText: PropTypes.string,
onCancel: PropTypes.func.isRequired,
primaryButtonTestId: PropTypes.string,
};
export const IconLink = ({ url, icon: IconComponent }) => (
<a
href={url}
target='_blank'
rel='noreferrer'
className='mdl-color-text--grey-600'
>
<IconComponent />
</a>
);
IconLink.propTypes = {
url: PropTypes.string,
icon: PropTypes.object,
};
export const MenuItemWithIcon = React.forwardRef(
({ icon: IconComponent, label, disabled, ...menuItemProps }, ref) => (
<MenuItem
disabled={disabled}
style={{ display: 'flex', alignItems: 'center' }}
{...menuItemProps}
>
<IconComponent />
{label}
</MenuItem>
),
);
MenuItemWithIcon.propTypes = {
icon: PropTypes.object,
label: PropTypes.string,
disabled: PropTypes.bool,
};
const badNumbers = [
Number.NaN,
Number.POSITIVE_INFINITY,
Number.NEGATIVE_INFINITY,
];
export function calc(value, total, decimal) {
if (
typeof value !== 'number' ||
typeof total !== 'number' ||
typeof decimal !== 'number'
) {
return null;
}
if (total === 0) {
return 0;
}
badNumbers.forEach((number) => {
if ([value, total, decimal].indexOf(number) > -1) {
return number;
}
});
return ((value / total) * 100).toFixed(decimal);
}
export const selectStyles = {
control: (provided) => ({
...provided,
border: '1px solid #607d8b',
boxShadow: '0',
':hover': {
borderColor: '#607d8b',
boxShadow: '0 0 0 1px #607d8b',
},
}),
};