1
0
mirror of https://github.com/Unleash/unleash.git synced 2024-10-18 20:09:08 +02:00
unleash.unleash/frontend/src/component/common/index.jsx
Nuno Góis 5872a1c82c refactor: adapt to Vite (#940)
* chore: adapt to vite

* remove react-scripts

* add jsdom dep, update snaps

* fix tests

* fix: formatting

* fix: vercel rewrite

* fix: regeneratorRuntime is not defined

Co-authored-by: Tymoteusz Czech <tymek+gpg@getunleash.ai>
Co-authored-by: Tymoteusz Czech <2625371+Tymek@users.noreply.github.com>
2022-05-05 16:15:22 +01:00

169 lines
4.4 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';
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/${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 = [NaN, Infinity, -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',
},
}),
};