123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540 |
- import React from 'react';
- import PropTypes from 'prop-types';
- import {
- TextField,
- Grid,
- InputBase,
- Button,
- NativeSelect,
- Box,
- Typography,
- GridList,
- GridListTile,
- TableContainer,
- Table,
- TableHead,
- TableRow,
- TableBody,
- TableCell,
- Dialog,
- DialogTitle,
- DialogContent,
- } from '@material-ui/core';
- import AddIcon from '@material-ui/icons/Add';
- import { withStyles, makeStyles } from '@material-ui/core/styles';
- import Autocomplete, { createFilterOptions } from '@material-ui/lab/Autocomplete';
- const filter = createFilterOptions();
- const BootstrapInput = withStyles((theme) => ({
- root: {
- 'label + &': {
- marginTop: theme.spacing(3),
- },
- },
- input: {
- borderRadius: 4,
- position: 'relative',
- backgroundColor: 'transparent',
- border: '1px solid #ced4da',
- fontSize: 16,
- padding: '10px 26px 10px 12px',
- transition: theme.transitions.create(['border-color', 'box-shadow']),
- // Use the system font instead of the default Roboto font.
- fontFamily: [
- '-apple-system',
- 'BlinkMacSystemFont',
- '"Segoe UI"',
- 'Roboto',
- '"Helvetica Neue"',
- 'Arial',
- 'sans-serif',
- '"Apple Color Emoji"',
- '"Segoe UI Emoji"',
- '"Segoe UI Symbol"',
- ].join(','),
- '&:focus': {
- borderRadius: 4,
- borderColor: '#80bdff',
- boxShadow: '0 0 0 0.2rem rgba(0,123,255,.25)',
- },
- },
- }))(InputBase);
- const useStyles = makeStyles({
- root: {
- width: '100%',
- },
- paper: {
- width: '100%',
- // marginBottom: theme.spacing(2),
- },
- table: {
- minWidth: 750,
- },
- visuallyHidden: {
- border: 0,
- clip: 'rect(0 0 0 0)',
- height: 1,
- margin: -1,
- overflow: 'hidden',
- padding: 0,
- position: 'absolute',
- top: 20,
- width: 1,
- },
- underline: {
- "&&&:before": {
- borderBottom: "none"
- },
- "&&:after": {
- borderBottom: "none"
- }
- },
- actionButton: props => ({
- backgroundColor: props.primaryColor === undefined ? 'grey' : props.primaryColor,
- color: props.actionTextColor === undefined ? 'white' : props.actionTextColor
- })
- });
- function MButton(props) {
- const { action, onCallback, buttonStyle } = props;
- const handleAction = (e) => {
- e.preventDefault();
- onCallback(e);
- }
- return (
- <div className={buttonStyle.root}>
- <Box>
- <Button className={buttonStyle.actionButton}
- variant="contained" style={{ float: 'right', margin: "5px" }}
- onClick={(e) => handleAction(e)}
- >{action.icon}{action.label}</Button>
- </Box>
- </div>
- );
- }
- MButton.propTypes = {
- history: PropTypes.object,
- buttonStyle: PropTypes.any,
- action: PropTypes.object.isRequired,
- onCallback: PropTypes.func.isRequired
- };
- function MkForm(props) {
- const styles = useStyles(props.styles);
- const {
- fields = [],
- data = {},
- onDropdownCreateNew,
- actions = [],
- partHeaders
- } = props;
- const [_data, setDataField] = React.useState(data !== undefined ? data : {});
- const [open, setOpen] = React.useState(false);
- // const [imgCollection, setImageCollection] = React.useState([]);
- const [selectedPhoto, setSelectedPhoto] = React.useState("#");
- const handleTextString = (e, fieldName) => {
- setDataField({ ..._data, [fieldName]: e.target.value });
- }
- const handleTextNumber = (e, fieldName) => {
- setDataField({ ..._data, [fieldName]: e.target.value });
- }
- const handleTextMultiline = (e, fieldName) => {
- setDataField({ ..._data, [fieldName]: e.target.value });
- }
- const handleDate = (e, fieldName) => {
- setDataField({ ..._data, [fieldName]: e.target.value });
- }
- const handleDropDownChange = (e, fieldName) => {
- var selectedIndex = e.target.options.selectedIndex;
- var selectedValue = e.target.options[selectedIndex].getAttribute('name');
- var fn = fieldName.split('_');
- var fieldId = fn[0] + '_' + 'id';
- setDataField({ ..._data, [fieldName]: selectedValue, [fieldId]: e.target.value });
- }
- // const handleImgUpload = (e, fieldName) => {
- // e.preventDefault();
- // let reader = new FileReader();
- // let file = e.target.files[0];
- // reader.onloadend = () => {
- // setImgPreviewPath(reader.result);
- // }
- // reader.readAsDataURL(file);
- // setDataField({ ..._data, [fieldName]: e.target.files[0].name });
- // }
- const handleCanCreateNew = (data) => {
- onDropdownCreateNew(data);
- }
- const onChangeValue = (fieldName, value) => {
- setDataField({ ..._data, [fieldName]: value });
- }
- const onFileChange = (e, f) => {
- }
- const handleSelectItemDialog = () => {
- }
- return (
- <div className={styles.root}>
- <Grid container>
- <Grid item xs={12}>
- {fields.map((f, i) => {
- if (f.type === 'text_string') {
- return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <Grid item xs={12} sm={5}>
- <Box style={{ width: '150px' }}>
- <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
- </Box>
- </Grid>
- <Grid item xs={12} sm={7}>
- <TextField id={f.field_name}
- variant="outlined"
- autoComplete="off"
- size="small"
- style={{ width: '100%' }}
- InputProps={{
- readOnly: f.readOnly ? f.readOnly : false,
- }}
- value={_data !== undefined ? _data[f.field_name] : ''}
- onChange={(e) => handleTextString(e, f.field_name)}
- />
- </Grid>
- </Grid>;
- }
- else if (f.type === 'text_number') {
- return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <Grid item xs={12} sm={5}>
- <Box style={{ width: '150px' }}>
- <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
- </Box>
- </Grid>
- <Grid item xs={12} sm={7}>
- <TextField
- id={f.field_name}
- variant="outlined"
- autoComplete="off"
- size="small"
- style={{ width: '100%' }}
- type="number"
- value={_data !== undefined ? _data[f.field_name] : ''}
- onChange={(e) => handleTextNumber(e, f.field_name)}
- />
- </Grid>
- </Grid>;
- }
- else if (f.type === 'text_multiline') {
- return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <Grid item xs={12} sm={5}>
- <Box style={{ width: '150px' }}>
- <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
- </Box>
- </Grid>
- <Grid item xs={12} sm={7}>
- <TextField
- id={f.field_name}
- multiline
- autoComplete="off"
- rows={3}
- size="small"
- style={{ width: '100%' }}
- value={_data !== undefined ? _data[f.field_name] : ''}
- variant="outlined"
- onChange={(e) => handleTextMultiline(e, f.field_name)}
- />
- </Grid>
- </Grid>;
- }
- else if (f.type === 'date') {
- return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <Grid item xs={12} sm={5}>
- <Box style={{ width: '150px' }}>
- <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
- </Box>
- </Grid>
- <Grid item xs={12} sm={7}>
- <TextField
- id={f.field_name}
- variant="outlined"
- autoComplete="off"
- size="small"
- value={_data !== undefined ? _data[f.field_name] : ''}
- type="date"
- style={{ width: '100%' }}
- onChange={(e) => handleDate(e, f.field_name)}
- />
- </Grid>
- </Grid>;
- }
- else if (f.type === 'dropdown') {
- if (f.options !== undefined && f.option_label_field !== undefined) {
- if (f.field_name === 'priority') {
- return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <Grid item xs={12} sm={5}>
- <Box style={{ width: '150px' }}>
- <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
- </Box>
- </Grid>
- <Grid item xs={12} sm={7}>
- <NativeSelect
- value={_data !== undefined ? _data[f.field_name] : ''}
- onChange={(e) => handleDropDownChange(e, f.field_name)}
- id={f.field_name}
- input={<BootstrapInput />}
- style={{ width: '100%' }}
- >
- <option aria-label="None" value="" >Select</option>
- {f.options.map((d, i) => {
- return (<option name={d.name} value={d.id} key={d.id}>{d.name}</option>);
- })}
- </NativeSelect>
- </Grid>
- </Grid>;
- } else {
- return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <Grid item xs={12} sm={5}>
- <Box style={{ width: '150px' }}>
- <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
- </Box>
- </Grid>
- <Grid item xs={12} sm={7}>
- <Autocomplete
- id="combo-box-demo"
- options={f.options}
- getOptionLabel={(option) => {
- if (typeof option === 'string') {
- return option;
- }
- return option[f.option_label_field];
- }}
- style={{ width: '100%' }}
- size='small'
- value={_data !== undefined ? _data[f.field_name] ? _data[f.field_name] : " " : " "}
- filterOptions={(options, params) => {
- console.log("Autocomplete", f.can_create);
- if (f.can_create) {
- var newFilter = ['+ Add New']
- var filtered = filter(options, params);
- return [...newFilter, ...filtered];
- } else {
- var _filtered = filter(options, params);
- return _filtered;
- }
- }}
- onChange={(event, newValue) => {
- if (typeof newValue === 'string') {
- console.log('f.field_name', f.field_name, " f.can_create", f.can_create);
- var d = {
- "canCreate": f.can_create,
- "fields": f.fields,
- "name": f.name,
- "fieldName": f.field_name
- }
- handleCanCreateNew(d);
- } else {
- if (newValue != null && newValue.inputValue !== '' && newValue.product_desc !== "") {
- onChangeValue(f.field_name, newValue[f.option_label_field]);
- }
- }
- }}
- renderInput={(params) => <TextField {...params} variant="outlined" />}
- />
- </Grid>
- </Grid>;
- }
- }
- }
- else if (f.type === 'photo_list') {
- console.log('photo_list:', _data);
- return <div>
- <Grid
- key={f.field_name} container
- style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <Grid item xs={12} sm={5}>
- <Box style={{ width: '150px' }}>
- <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
- </Box>
- </Grid>
- <Grid item xs={12} sm={7}>
- <div style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <form>
- <div className="form-group">
- <input type="file" name="imgCollection"
- onChange={(e) => onFileChange(e, f.field_name)}
- multiple />
- </div>
- </form>
- </div>
- </Grid>
- </Grid>
- {_data[f.field_name] !== undefined && _data[f.field_name].length !== 0 ?
- <Grid
- key={f.field_name} container
- style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <Grid item xs={12}>
- <div style={{ display: 'block', alignItems: 'center', marginBottom: '10px' }}>
- <GridList className={styles.gridList}>
- {_data[f.field_name] === undefined ? <span /> : _data[f.field_name].map((tile) => (
- <GridListTile key={tile} style={{ width: '100px', height: '100px' }}>
- <img src={tile} alt={tile} onClick={(e) => {
- // setSelectedPhoto(tile);
- setOpen(true);
- }
- } />
- </GridListTile>
- ))}
- </GridList>
- </div>
- </Grid>
- <Dialog maxWidth="lg" aria-labelledby="customized-dialog-title" open={open}>
- <DialogTitle id="customized-dialog-title" onClose={(e) => setOpen(false)} >
- Photos
- </DialogTitle>
- <DialogContent dividers>
- <Grid item xs={12}>
- <Grid>
- <img src={selectedPhoto} className="show-img" alt="logo" />
- </Grid>
- <br />
- <Grid container spacing={3}>
- {_data[f.field_name].length > 0 ? _data[f.field_name].map((value) => (
- <Grid key={value} item>
- <Box className="square" > <img src={value} className="thumnail-img" alt="logo" onClick={(e) => setSelectedPhoto(value)} /></Box>
- </Grid>
- )) : <span />}
- </Grid>
- </Grid>
- </DialogContent>
- </Dialog>
- </Grid>
- : <Grid />}
- </div>;
- }
- else if (f.type === 'list') {
- console.log('list', _data[f.field_name]);
- return <div>
- <Grid
- key={f.field_name} container
- style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <Grid item xs={12} sm={5}>
- <Box style={{ width: '150px' }}>
- <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
- </Box>
- </Grid>
- <Grid item xs={12} sm={7}>
- <div style={{ display: 'block', alignItems: 'center', marginBottom: '10px' }}>
- <Box>
- <Button onClick={handleSelectItemDialog}><AddIcon /></Button>
- </Box>
- </div>
- </Grid>
- </Grid>
- <Grid
- key={f.field_name} container
- style={{ display: 'block', alignItems: 'center', marginBottom: '10px' }}>
- <div style={{ display: 'block', alignItems: 'center', marginBottom: '10px' }}>
- <TableContainer>
- <Table className={styles.table} size="small" aria-label="a dense table">
- <TableHead>
- <TableRow>
- {partHeaders.map((h, i) => {
- return (<TableCell key={h.id} align='left'>{h.label}</TableCell>);
- })}
- </TableRow>
- </TableHead>
- <TableBody>
- {_data[f.field_name].length > 0 ? _data[f.field_name].map((row) => (
- <TableRow key={row.name}>
- {partHeaders.map((h, i) => {
- return (<TableCell key={h.id} align={h.numeric ? 'right' : 'left'}>{row[h.id]}</TableCell>);
- })}
- </TableRow>
- )) : <span />
- }
- </TableBody>
- </Table>
- </TableContainer>
- </div>
- </Grid>
- </div>;
- }
- else if (f.type === 'time') {
- return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
- <Grid item xs={12} sm={5}>
- <Box style={{ width: '150px' }}>
- <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
- </Box>
- </Grid>
- <Grid item xs={12} sm={7}> <TextField
- id="time"
- variant="outlined"
- size="small"
- type="time"
- className={styles.textField}
- InputLabelProps={{
- shrink: true,
- }}
- inputProps={{
- step: 300, // 5 min
- }}
- // onChange={(e) => handleTime(e, f.field_name)}
- />
- </Grid>
- </Grid>;
- }
- })}
- </Grid>
- {/* display actions buttons */}
- {actions.length > 0 ?
- <Grid item xs={12}>
- {actions.map((a) => {
- if (a.status === _data.status) {
- return <MButton action={a} onCallback={(event) => a.callback(event, _data)} buttonStyle={styles} />;
- }
- })}
- </Grid> : <Grid />}
- </Grid>
- </div>
- );
- }
- MkForm.propTypes = {
- history: PropTypes.object,
- fields: PropTypes.array.isRequired,
- data: PropTypes.object,
- isNew: PropTypes.bool,
- actions: PropTypes.array,
- onDropdownCreateNew: PropTypes.func,
- styles: PropTypes.any
- };
- export default (MkForm);
|