HandsontableModal.jsx 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. import React from 'react';
  2. import PropTypes from 'prop-types';
  3. import Modal from 'react-bootstrap/es/Modal';
  4. import Button from 'react-bootstrap/es/Button';
  5. import Navbar from 'react-bootstrap/es/Navbar';
  6. import ButtonGroup from 'react-bootstrap/es/ButtonGroup';
  7. import Collapse from 'react-bootstrap/es/Collapse';
  8. import FormGroup from 'react-bootstrap/es/FormGroup';
  9. import ControlLabel from 'react-bootstrap/es/ControlLabel';
  10. import FormControl from 'react-bootstrap/es/FormControl';
  11. import Handsontable from 'handsontable';
  12. import { HotTable } from '@handsontable/react';
  13. import MarkdownTable from '../../models/MarkdownTable';
  14. import HandsontableUtil from './HandsontableUtil';
  15. export default class HandsontableModal extends React.Component {
  16. constructor(props) {
  17. super(props);
  18. this.state = {
  19. show: false,
  20. markdownTableOnInit: HandsontableModal.getDefaultMarkdownTable(),
  21. markdownTable: HandsontableModal.getDefaultMarkdownTable(),
  22. handsontableSetting: HandsontableModal.getDefaultHandsontableSetting()
  23. };
  24. this.init = this.init.bind(this);
  25. this.reset = this.reset.bind(this);
  26. this.cancel = this.cancel.bind(this);
  27. this.save = this.save.bind(this);
  28. }
  29. init(markdownTable) {
  30. const initMarkdownTable = markdownTable || HandsontableModal.getDefaultMarkdownTable();
  31. this.setState(
  32. {
  33. markdownTableOnInit: initMarkdownTable,
  34. markdownTable: initMarkdownTable.clone(),
  35. handsontableSetting: Object.assign({}, this.state.handsontableSetting, {
  36. /*
  37. * The afterUpdateSettings hook is called when this component state changes.
  38. *
  39. * In detail, when this component state changes, React will re-render HotTable because it is passed some state values of this component.
  40. * HotTable#shouldComponentUpdate is called in this process and it call the updateSettings method for the Handsontable instance.
  41. * After updateSetting is executed, Handsontable calls a AfterUpdateSetting hook.
  42. */
  43. afterUpdateSettings: HandsontableUtil.createHandlerToSynchronizeHandontableAlignWith(initMarkdownTable.options.align)
  44. })
  45. }
  46. );
  47. }
  48. show(markdownTable) {
  49. this.init(markdownTable);
  50. this.setState({ show: true });
  51. }
  52. reset() {
  53. this.setState({ markdownTable: this.state.markdownTableOnInit.clone() });
  54. }
  55. cancel() {
  56. this.setState({ show: false });
  57. }
  58. save() {
  59. let newMarkdownTable = this.state.markdownTable.clone();
  60. newMarkdownTable.options.align = HandsontableUtil.getMarkdownTableAlignmentFrom(this.refs.hotTable.hotInstance);
  61. if (this.props.onSave != null) {
  62. this.props.onSave(newMarkdownTable);
  63. }
  64. this.setState({ show: false });
  65. }
  66. setClassNameToColumns(className) {
  67. const selectedRange = this.refs.hotTable.hotInstance.getSelectedRange();
  68. if (selectedRange == null) return;
  69. let startCol;
  70. let endCol;
  71. if (selectedRange[0].from.col < selectedRange[0].to.col) {
  72. startCol = selectedRange[0].from.col;
  73. endCol = selectedRange[0].to.col;
  74. }
  75. else {
  76. startCol = selectedRange[0].to.col;
  77. endCol = selectedRange[0].from.col;
  78. }
  79. HandsontableUtil.setClassNameToColumns(this.refs.hotTable.hotInstance, startCol, endCol, className);
  80. }
  81. render() {
  82. return (
  83. <Modal show={this.state.show} onHide={this.cancel} bsSize="large" dialogClassName="handsontable-modal">
  84. <Modal.Header closeButton>
  85. <Modal.Title>Edit Table</Modal.Title>
  86. </Modal.Header>
  87. <Modal.Body className="p-0">
  88. <Navbar>
  89. <Navbar.Form>
  90. <Button className="m-r-20" onClick={() => this.setState({ open: !this.state.open })}>Data Import</Button>
  91. <ButtonGroup>
  92. <Button onClick={() => { this.setClassNameToColumns('htLeft') }}><i className="ti-align-left"></i></Button>
  93. <Button onClick={() => { this.setClassNameToColumns('htCenter') }}><i className="ti-align-center"></i></Button>
  94. <Button onClick={() => { this.setClassNameToColumns('htRight') }}><i className="ti-align-right"></i></Button>
  95. </ButtonGroup>
  96. </Navbar.Form>
  97. </Navbar>
  98. <Collapse in={this.state.open}>
  99. <div className="p-30">
  100. <form action="">
  101. <FormGroup controlId="formControlsSelect">
  102. <ControlLabel>Select Data Format</ControlLabel>
  103. <FormControl componentClass="select" placeholder="select">
  104. <option value="select">CSV</option>
  105. <option value="other">TSV</option>
  106. <option value="other">HTML</option>
  107. </FormControl>
  108. </FormGroup>
  109. <FormGroup controlId="formControlsTextarea">
  110. <ControlLabel>Import Data</ControlLabel>
  111. <FormControl componentClass="textarea" placeholder="textarea" />
  112. </FormGroup>
  113. </form>
  114. </div>
  115. </Collapse>
  116. <div className="p-4">
  117. <HotTable ref='hotTable' data={this.state.markdownTable.table} settings={this.state.handsontableSetting} />
  118. </div>
  119. </Modal.Body>
  120. <Modal.Footer>
  121. <div className="d-flex justify-content-between">
  122. <Button bsStyle="danger" onClick={this.reset}>Reset</Button>
  123. <div className="d-flex">
  124. <Button bsStyle="default" onClick={this.cancel}>Cancel</Button>
  125. <Button bsStyle="primary" onClick={this.save}>Done</Button>
  126. </div>
  127. </div>
  128. </Modal.Footer>
  129. </Modal>
  130. );
  131. }
  132. static getDefaultMarkdownTable() {
  133. return new MarkdownTable(
  134. [
  135. ['col1', 'col2', 'col3'],
  136. ['', '', ''],
  137. ['', '', ''],
  138. ],
  139. {
  140. align: ['', '', '']
  141. }
  142. );
  143. }
  144. static getDefaultHandsontableSetting() {
  145. return {
  146. height: 300,
  147. rowHeaders: true,
  148. colHeaders: true,
  149. contextMenu: {
  150. items: {
  151. 'row_above': {}, 'row_below': {}, 'col_left': {}, 'col_right': {},
  152. 'separator1': Handsontable.plugins.ContextMenu.SEPARATOR,
  153. 'remove_row': {}, 'remove_col': {},
  154. 'separator2': Handsontable.plugins.ContextMenu.SEPARATOR,
  155. 'custom_alignment': {
  156. name: 'Align columns',
  157. key: 'align_columns',
  158. submenu: {
  159. items: [{
  160. name: 'Left',
  161. key: 'align_columns:1',
  162. callback: function(key, selection) {
  163. HandsontableUtil.setClassNameToColumns(this, selection[0].start.col, selection[0].end.col, 'htLeft');
  164. }}, {
  165. name: 'Center',
  166. key: 'align_columns:2',
  167. callback: function(key, selection) {
  168. HandsontableUtil.setClassNameToColumns(this, selection[0].start.col, selection[0].end.col, 'htCenter');
  169. }}, {
  170. name: 'Right',
  171. key: 'align_columns:3',
  172. callback: function(key, selection) {
  173. HandsontableUtil.setClassNameToColumns(this, selection[0].start.col, selection[0].end.col, 'htRight');
  174. }}
  175. ]
  176. }
  177. }
  178. }
  179. },
  180. selectionMode: 'multiple',
  181. outsideClickDeselects: false,
  182. modifyColWidth: function(width) {
  183. if (width < 100) {
  184. return 100;
  185. }
  186. if (width > 300) {
  187. return 300;
  188. }
  189. }
  190. };
  191. }
  192. }
  193. HandsontableModal.propTypes = {
  194. onSave: PropTypes.func
  195. };