HandsontableModal.jsx 6.5 KB

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