MkForm.js 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540
  1. import React from 'react';
  2. import PropTypes from 'prop-types';
  3. import {
  4. TextField,
  5. Grid,
  6. InputBase,
  7. Button,
  8. NativeSelect,
  9. Box,
  10. Typography,
  11. GridList,
  12. GridListTile,
  13. TableContainer,
  14. Table,
  15. TableHead,
  16. TableRow,
  17. TableBody,
  18. TableCell,
  19. Dialog,
  20. DialogTitle,
  21. DialogContent,
  22. } from '@material-ui/core';
  23. import AddIcon from '@material-ui/icons/Add';
  24. import { withStyles, makeStyles } from '@material-ui/core/styles';
  25. import Autocomplete, { createFilterOptions } from '@material-ui/lab/Autocomplete';
  26. const filter = createFilterOptions();
  27. const BootstrapInput = withStyles((theme) => ({
  28. root: {
  29. 'label + &': {
  30. marginTop: theme.spacing(3),
  31. },
  32. },
  33. input: {
  34. borderRadius: 4,
  35. position: 'relative',
  36. backgroundColor: 'transparent',
  37. border: '1px solid #ced4da',
  38. fontSize: 16,
  39. padding: '10px 26px 10px 12px',
  40. transition: theme.transitions.create(['border-color', 'box-shadow']),
  41. // Use the system font instead of the default Roboto font.
  42. fontFamily: [
  43. '-apple-system',
  44. 'BlinkMacSystemFont',
  45. '"Segoe UI"',
  46. 'Roboto',
  47. '"Helvetica Neue"',
  48. 'Arial',
  49. 'sans-serif',
  50. '"Apple Color Emoji"',
  51. '"Segoe UI Emoji"',
  52. '"Segoe UI Symbol"',
  53. ].join(','),
  54. '&:focus': {
  55. borderRadius: 4,
  56. borderColor: '#80bdff',
  57. boxShadow: '0 0 0 0.2rem rgba(0,123,255,.25)',
  58. },
  59. },
  60. }))(InputBase);
  61. const useStyles = makeStyles({
  62. root: {
  63. width: '100%',
  64. },
  65. paper: {
  66. width: '100%',
  67. // marginBottom: theme.spacing(2),
  68. },
  69. table: {
  70. minWidth: 750,
  71. },
  72. visuallyHidden: {
  73. border: 0,
  74. clip: 'rect(0 0 0 0)',
  75. height: 1,
  76. margin: -1,
  77. overflow: 'hidden',
  78. padding: 0,
  79. position: 'absolute',
  80. top: 20,
  81. width: 1,
  82. },
  83. underline: {
  84. "&&&:before": {
  85. borderBottom: "none"
  86. },
  87. "&&:after": {
  88. borderBottom: "none"
  89. }
  90. },
  91. actionButton: props => ({
  92. backgroundColor: props.primaryColor === undefined ? 'grey' : props.primaryColor,
  93. color: props.actionTextColor === undefined ? 'white' : props.actionTextColor
  94. })
  95. });
  96. function MButton(props) {
  97. const { action, onCallback, buttonStyle } = props;
  98. const handleAction = (e) => {
  99. e.preventDefault();
  100. onCallback(e);
  101. }
  102. return (
  103. <div className={buttonStyle.root}>
  104. <Box>
  105. <Button className={buttonStyle.actionButton}
  106. variant="contained" style={{ float: 'right', margin: "5px" }}
  107. onClick={(e) => handleAction(e)}
  108. >{action.icon}{action.label}</Button>
  109. </Box>
  110. </div>
  111. );
  112. }
  113. MButton.propTypes = {
  114. history: PropTypes.object,
  115. buttonStyle: PropTypes.any,
  116. action: PropTypes.object.isRequired,
  117. onCallback: PropTypes.func.isRequired
  118. };
  119. function MkForm(props) {
  120. const styles = useStyles(props.styles);
  121. const {
  122. fields = [],
  123. data = {},
  124. onDropdownCreateNew,
  125. actions = [],
  126. partHeaders
  127. } = props;
  128. const [_data, setDataField] = React.useState(data !== undefined ? data : {});
  129. const [open, setOpen] = React.useState(false);
  130. // const [imgCollection, setImageCollection] = React.useState([]);
  131. const [selectedPhoto, setSelectedPhoto] = React.useState("#");
  132. const handleTextString = (e, fieldName) => {
  133. setDataField({ ..._data, [fieldName]: e.target.value });
  134. }
  135. const handleTextNumber = (e, fieldName) => {
  136. setDataField({ ..._data, [fieldName]: e.target.value });
  137. }
  138. const handleTextMultiline = (e, fieldName) => {
  139. setDataField({ ..._data, [fieldName]: e.target.value });
  140. }
  141. const handleDate = (e, fieldName) => {
  142. setDataField({ ..._data, [fieldName]: e.target.value });
  143. }
  144. const handleDropDownChange = (e, fieldName) => {
  145. var selectedIndex = e.target.options.selectedIndex;
  146. var selectedValue = e.target.options[selectedIndex].getAttribute('name');
  147. var fn = fieldName.split('_');
  148. var fieldId = fn[0] + '_' + 'id';
  149. setDataField({ ..._data, [fieldName]: selectedValue, [fieldId]: e.target.value });
  150. }
  151. // const handleImgUpload = (e, fieldName) => {
  152. // e.preventDefault();
  153. // let reader = new FileReader();
  154. // let file = e.target.files[0];
  155. // reader.onloadend = () => {
  156. // setImgPreviewPath(reader.result);
  157. // }
  158. // reader.readAsDataURL(file);
  159. // setDataField({ ..._data, [fieldName]: e.target.files[0].name });
  160. // }
  161. const handleCanCreateNew = (data) => {
  162. onDropdownCreateNew(data);
  163. }
  164. const onChangeValue = (fieldName, value) => {
  165. setDataField({ ..._data, [fieldName]: value });
  166. }
  167. const onFileChange = (e, f) => {
  168. }
  169. const handleSelectItemDialog = () => {
  170. }
  171. return (
  172. <div className={styles.root}>
  173. <Grid container>
  174. <Grid item xs={12}>
  175. {fields.map((f, i) => {
  176. if (f.type === 'text_string') {
  177. return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  178. <Grid item xs={12} sm={5}>
  179. <Box style={{ width: '150px' }}>
  180. <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
  181. </Box>
  182. </Grid>
  183. <Grid item xs={12} sm={7}>
  184. <TextField id={f.field_name}
  185. variant="outlined"
  186. autoComplete="off"
  187. size="small"
  188. style={{ width: '100%' }}
  189. InputProps={{
  190. readOnly: f.readOnly ? f.readOnly : false,
  191. }}
  192. value={_data !== undefined ? _data[f.field_name] : ''}
  193. onChange={(e) => handleTextString(e, f.field_name)}
  194. />
  195. </Grid>
  196. </Grid>;
  197. }
  198. else if (f.type === 'text_number') {
  199. return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  200. <Grid item xs={12} sm={5}>
  201. <Box style={{ width: '150px' }}>
  202. <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
  203. </Box>
  204. </Grid>
  205. <Grid item xs={12} sm={7}>
  206. <TextField
  207. id={f.field_name}
  208. variant="outlined"
  209. autoComplete="off"
  210. size="small"
  211. style={{ width: '100%' }}
  212. type="number"
  213. value={_data !== undefined ? _data[f.field_name] : ''}
  214. onChange={(e) => handleTextNumber(e, f.field_name)}
  215. />
  216. </Grid>
  217. </Grid>;
  218. }
  219. else if (f.type === 'text_multiline') {
  220. return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  221. <Grid item xs={12} sm={5}>
  222. <Box style={{ width: '150px' }}>
  223. <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
  224. </Box>
  225. </Grid>
  226. <Grid item xs={12} sm={7}>
  227. <TextField
  228. id={f.field_name}
  229. multiline
  230. autoComplete="off"
  231. rows={3}
  232. size="small"
  233. style={{ width: '100%' }}
  234. value={_data !== undefined ? _data[f.field_name] : ''}
  235. variant="outlined"
  236. onChange={(e) => handleTextMultiline(e, f.field_name)}
  237. />
  238. </Grid>
  239. </Grid>;
  240. }
  241. else if (f.type === 'date') {
  242. return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  243. <Grid item xs={12} sm={5}>
  244. <Box style={{ width: '150px' }}>
  245. <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
  246. </Box>
  247. </Grid>
  248. <Grid item xs={12} sm={7}>
  249. <TextField
  250. id={f.field_name}
  251. variant="outlined"
  252. autoComplete="off"
  253. size="small"
  254. value={_data !== undefined ? _data[f.field_name] : ''}
  255. type="date"
  256. style={{ width: '100%' }}
  257. onChange={(e) => handleDate(e, f.field_name)}
  258. />
  259. </Grid>
  260. </Grid>;
  261. }
  262. else if (f.type === 'dropdown') {
  263. if (f.options !== undefined && f.option_label_field !== undefined) {
  264. if (f.field_name === 'priority') {
  265. return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  266. <Grid item xs={12} sm={5}>
  267. <Box style={{ width: '150px' }}>
  268. <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
  269. </Box>
  270. </Grid>
  271. <Grid item xs={12} sm={7}>
  272. <NativeSelect
  273. value={_data !== undefined ? _data[f.field_name] : ''}
  274. onChange={(e) => handleDropDownChange(e, f.field_name)}
  275. id={f.field_name}
  276. input={<BootstrapInput />}
  277. style={{ width: '100%' }}
  278. >
  279. <option aria-label="None" value="" >Select</option>
  280. {f.options.map((d, i) => {
  281. return (<option name={d.name} value={d.id} key={d.id}>{d.name}</option>);
  282. })}
  283. </NativeSelect>
  284. </Grid>
  285. </Grid>;
  286. } else {
  287. return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  288. <Grid item xs={12} sm={5}>
  289. <Box style={{ width: '150px' }}>
  290. <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
  291. </Box>
  292. </Grid>
  293. <Grid item xs={12} sm={7}>
  294. <Autocomplete
  295. id="combo-box-demo"
  296. options={f.options}
  297. getOptionLabel={(option) => {
  298. if (typeof option === 'string') {
  299. return option;
  300. }
  301. return option[f.option_label_field];
  302. }}
  303. style={{ width: '100%' }}
  304. size='small'
  305. value={_data !== undefined ? _data[f.field_name] ? _data[f.field_name] : " " : " "}
  306. filterOptions={(options, params) => {
  307. console.log("Autocomplete", f.can_create);
  308. if (f.can_create) {
  309. var newFilter = ['+ Add New']
  310. var filtered = filter(options, params);
  311. return [...newFilter, ...filtered];
  312. } else {
  313. var _filtered = filter(options, params);
  314. return _filtered;
  315. }
  316. }}
  317. onChange={(event, newValue) => {
  318. if (typeof newValue === 'string') {
  319. console.log('f.field_name', f.field_name, " f.can_create", f.can_create);
  320. var d = {
  321. "canCreate": f.can_create,
  322. "fields": f.fields,
  323. "name": f.name,
  324. "fieldName": f.field_name
  325. }
  326. handleCanCreateNew(d);
  327. } else {
  328. if (newValue != null && newValue.inputValue !== '' && newValue.product_desc !== "") {
  329. onChangeValue(f.field_name, newValue[f.option_label_field]);
  330. }
  331. }
  332. }}
  333. renderInput={(params) => <TextField {...params} variant="outlined" />}
  334. />
  335. </Grid>
  336. </Grid>;
  337. }
  338. }
  339. }
  340. else if (f.type === 'photo_list') {
  341. console.log('photo_list:', _data);
  342. return <div>
  343. <Grid
  344. key={f.field_name} container
  345. style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  346. <Grid item xs={12} sm={5}>
  347. <Box style={{ width: '150px' }}>
  348. <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
  349. </Box>
  350. </Grid>
  351. <Grid item xs={12} sm={7}>
  352. <div style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  353. <form>
  354. <div className="form-group">
  355. <input type="file" name="imgCollection"
  356. onChange={(e) => onFileChange(e, f.field_name)}
  357. multiple />
  358. </div>
  359. </form>
  360. </div>
  361. </Grid>
  362. </Grid>
  363. {_data[f.field_name] !== undefined && _data[f.field_name].length !== 0 ?
  364. <Grid
  365. key={f.field_name} container
  366. style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  367. <Grid item xs={12}>
  368. <div style={{ display: 'block', alignItems: 'center', marginBottom: '10px' }}>
  369. <GridList className={styles.gridList}>
  370. {_data[f.field_name] === undefined ? <span /> : _data[f.field_name].map((tile) => (
  371. <GridListTile key={tile} style={{ width: '100px', height: '100px' }}>
  372. <img src={tile} alt={tile} onClick={(e) => {
  373. // setSelectedPhoto(tile);
  374. setOpen(true);
  375. }
  376. } />
  377. </GridListTile>
  378. ))}
  379. </GridList>
  380. </div>
  381. </Grid>
  382. <Dialog maxWidth="lg" aria-labelledby="customized-dialog-title" open={open}>
  383. <DialogTitle id="customized-dialog-title" onClose={(e) => setOpen(false)} >
  384. Photos
  385. </DialogTitle>
  386. <DialogContent dividers>
  387. <Grid item xs={12}>
  388. <Grid>
  389. <img src={selectedPhoto} className="show-img" alt="logo" />
  390. </Grid>
  391. <br />
  392. <Grid container spacing={3}>
  393. {_data[f.field_name].length > 0 ? _data[f.field_name].map((value) => (
  394. <Grid key={value} item>
  395. <Box className="square" > <img src={value} className="thumnail-img" alt="logo" onClick={(e) => setSelectedPhoto(value)} /></Box>
  396. </Grid>
  397. )) : <span />}
  398. </Grid>
  399. </Grid>
  400. </DialogContent>
  401. </Dialog>
  402. </Grid>
  403. : <Grid />}
  404. </div>;
  405. }
  406. else if (f.type === 'list') {
  407. console.log('list', _data[f.field_name]);
  408. return <div>
  409. <Grid
  410. key={f.field_name} container
  411. style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  412. <Grid item xs={12} sm={5}>
  413. <Box style={{ width: '150px' }}>
  414. <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
  415. </Box>
  416. </Grid>
  417. <Grid item xs={12} sm={7}>
  418. <div style={{ display: 'block', alignItems: 'center', marginBottom: '10px' }}>
  419. <Box>
  420. <Button onClick={handleSelectItemDialog}><AddIcon /></Button>
  421. </Box>
  422. </div>
  423. </Grid>
  424. </Grid>
  425. <Grid
  426. key={f.field_name} container
  427. style={{ display: 'block', alignItems: 'center', marginBottom: '10px' }}>
  428. <div style={{ display: 'block', alignItems: 'center', marginBottom: '10px' }}>
  429. <TableContainer>
  430. <Table className={styles.table} size="small" aria-label="a dense table">
  431. <TableHead>
  432. <TableRow>
  433. {partHeaders.map((h, i) => {
  434. return (<TableCell key={h.id} align='left'>{h.label}</TableCell>);
  435. })}
  436. </TableRow>
  437. </TableHead>
  438. <TableBody>
  439. {_data[f.field_name].length > 0 ? _data[f.field_name].map((row) => (
  440. <TableRow key={row.name}>
  441. {partHeaders.map((h, i) => {
  442. return (<TableCell key={h.id} align={h.numeric ? 'right' : 'left'}>{row[h.id]}</TableCell>);
  443. })}
  444. </TableRow>
  445. )) : <span />
  446. }
  447. </TableBody>
  448. </Table>
  449. </TableContainer>
  450. </div>
  451. </Grid>
  452. </div>;
  453. }
  454. else if (f.type === 'time') {
  455. return <Grid key={f.field_name} container style={{ display: 'flex', alignItems: 'center', marginBottom: '10px' }}>
  456. <Grid item xs={12} sm={5}>
  457. <Box style={{ width: '150px' }}>
  458. <Typography style={{ paddingRight: '30px', color: 'grey' }}>{f.label}</Typography>
  459. </Box>
  460. </Grid>
  461. <Grid item xs={12} sm={7}> <TextField
  462. id="time"
  463. variant="outlined"
  464. size="small"
  465. type="time"
  466. className={styles.textField}
  467. InputLabelProps={{
  468. shrink: true,
  469. }}
  470. inputProps={{
  471. step: 300, // 5 min
  472. }}
  473. // onChange={(e) => handleTime(e, f.field_name)}
  474. />
  475. </Grid>
  476. </Grid>;
  477. }
  478. })}
  479. </Grid>
  480. {/* display actions buttons */}
  481. {actions.length > 0 ?
  482. <Grid item xs={12}>
  483. {actions.map((a) => {
  484. if (a.status === _data.status) {
  485. return <MButton action={a} onCallback={(event) => a.callback(event, _data)} buttonStyle={styles} />;
  486. }
  487. })}
  488. </Grid> : <Grid />}
  489. </Grid>
  490. </div>
  491. );
  492. }
  493. MkForm.propTypes = {
  494. history: PropTypes.object,
  495. fields: PropTypes.array.isRequired,
  496. data: PropTypes.object,
  497. isNew: PropTypes.bool,
  498. actions: PropTypes.array,
  499. onDropdownCreateNew: PropTypes.func,
  500. styles: PropTypes.any
  501. };
  502. export default (MkForm);