[881] | 1 | "use strict"; |
---|
| 2 | import React from 'react'; |
---|
| 3 | import { WidthProvider, Responsive} from 'react-grid-layout'; |
---|
| 4 | import _ from 'lodash'; |
---|
| 5 | |
---|
| 6 | const ResponsiveReactGridLayout = WidthProvider(Responsive); |
---|
| 7 | |
---|
| 8 | /** |
---|
| 9 | * Component holding all widgets of Framsticks-JS. It bases on 'react-grid-layout' |
---|
| 10 | * component, making it possible to create mini-windows within website. |
---|
| 11 | */ |
---|
| 12 | class WidgetsContainer extends React.Component { |
---|
| 13 | /** |
---|
| 14 | * Creates instance of WidgetsContainer. It starts in the beginning with no |
---|
| 15 | * items. To add items you need to explicitly call object method addItem |
---|
| 16 | * with React.Component. |
---|
| 17 | * @param {any} props Basic props for React Components |
---|
| 18 | */ |
---|
| 19 | constructor(props) { |
---|
| 20 | super(props); |
---|
| 21 | |
---|
| 22 | this.state = { |
---|
| 23 | items: [], |
---|
| 24 | widgetCounter: 0, |
---|
| 25 | }; |
---|
| 26 | this.props = props; |
---|
| 27 | //this.onAddItem = this.onAddItem.bind(this); |
---|
| 28 | this.onBreakpointChange = this.onBreakpointChange.bind(this); |
---|
| 29 | this.onLayoutChange = this.onLayoutChange.bind(this); |
---|
| 30 | } |
---|
| 31 | |
---|
| 32 | /** |
---|
| 33 | * Initializes references. |
---|
| 34 | */ |
---|
| 35 | componentDidMount() { |
---|
| 36 | this.props.onRef(this); |
---|
| 37 | } |
---|
| 38 | |
---|
| 39 | /** |
---|
| 40 | * Unmounts references |
---|
| 41 | */ |
---|
| 42 | componentWillUnmount() { |
---|
| 43 | this.props.onRef(void 0); |
---|
| 44 | } |
---|
| 45 | |
---|
| 46 | /** |
---|
| 47 | * Creates widgets in Framsticks-JS. It is strictly internal method for |
---|
| 48 | * this Component and should not be called outside. It is used in |
---|
| 49 | * render() function. It holds default styles for widgets windows. |
---|
| 50 | * @param {React.Component} el widget to be added to board |
---|
| 51 | * @returns {JSX.Element} new widget |
---|
| 52 | */ |
---|
| 53 | createElement(el) { |
---|
| 54 | const removeStyle = { |
---|
| 55 | right: "2px", |
---|
| 56 | marginRight: "10px", |
---|
| 57 | top: 0, |
---|
| 58 | cursor: "pointer", |
---|
| 59 | }; |
---|
| 60 | const content = el.content ? el.content : "none"; |
---|
| 61 | let gridStyle = { |
---|
| 62 | backgroundColor: '#AAAAAA', |
---|
| 63 | display: 'table', |
---|
| 64 | padding: 0, |
---|
| 65 | boxShadow: "0 4px 8px 0 rgba(0, 0, 0, 0.2), 0 6px 20px 0 rgba(0, 0, 0, 0.19)" |
---|
| 66 | }; |
---|
| 67 | return ( |
---|
| 68 | <div key={el.i} data-grid={el} style={gridStyle}> |
---|
| 69 | {/* <span className="text" style={gridStyle}>{el.content}</span> */} |
---|
| 70 | <div style={{display: 'table-row', textAlign: 'right'}}> |
---|
| 71 | <span |
---|
| 72 | className="remove" |
---|
| 73 | style={removeStyle} |
---|
| 74 | onClick={this.onRemoveItem.bind(this,el.i)} |
---|
| 75 | >✖</span> |
---|
| 76 | </div> |
---|
| 77 | <div style={{display: 'table-row', height: '100%'}}> |
---|
| 78 | {content} |
---|
| 79 | </div> |
---|
| 80 | </div> |
---|
| 81 | ); |
---|
| 82 | } |
---|
| 83 | |
---|
| 84 | /** |
---|
| 85 | * Adds new React.Component to Framsticks-JS board. This method should |
---|
| 86 | * be used externally instead of createElement to add new element. It |
---|
| 87 | * holds and remembers newly added components. |
---|
| 88 | * @param {React.Component} item new widget to be added to board |
---|
| 89 | * @param {string} name name of used component, useful for remembering layout |
---|
| 90 | * @param {number} x OX position of widget |
---|
| 91 | * @param {number} y OY position of widget |
---|
| 92 | * @param {number} w width of widget |
---|
| 93 | * @param {number} h height of widget |
---|
| 94 | * @example <caption>Example of using addItem with GenoChecker widget</caption> |
---|
| 95 | * this.widgetscontainer.addItem(<GenoChecker />); |
---|
| 96 | */ |
---|
| 97 | addItem(item, name, x=0, y=Infinity, w=2, h=2) { |
---|
| 98 | this.setState({ |
---|
| 99 | items: this.state.items.concat({ |
---|
| 100 | i: ""+this.state.widgetCounter+","+name, |
---|
| 101 | content: item, |
---|
| 102 | //x: (this.state.items.length) % (this.state.cols || 12), |
---|
| 103 | //y: Infinity, |
---|
| 104 | x: x, |
---|
| 105 | y: y, |
---|
| 106 | w: w, |
---|
| 107 | h: h |
---|
| 108 | }), |
---|
| 109 | widgetCounter: this.state.widgetCounter+1, |
---|
| 110 | }); |
---|
| 111 | } |
---|
| 112 | |
---|
| 113 | /** |
---|
| 114 | * Uses list of layout items to be distributed on site. |
---|
| 115 | * @param {any} layoutitems list of items. |
---|
| 116 | */ |
---|
| 117 | addMultipleItems(layoutitems) { |
---|
| 118 | this.setState({ |
---|
| 119 | items: layoutitems, |
---|
| 120 | widgetCounter: layoutitems.length |
---|
| 121 | }); |
---|
| 122 | } |
---|
| 123 | |
---|
| 124 | /** |
---|
| 125 | * Method implemented for responsive behaviour of 'react-grid-layout'. |
---|
| 126 | * @param {any} breakpoint new breakpoint for layout |
---|
| 127 | * @param {any} cols new number of columns in grid |
---|
| 128 | */ |
---|
| 129 | onBreakpointChange(breakpoint, cols) { |
---|
| 130 | this.setState({ |
---|
| 131 | breakpoint: breakpoint, |
---|
| 132 | cols: cols |
---|
| 133 | }); |
---|
| 134 | } |
---|
| 135 | |
---|
| 136 | /** |
---|
| 137 | * Event running on change of 'react-grid-layout' Component. |
---|
| 138 | * @param {any} layout layout which should be selected next |
---|
| 139 | */ |
---|
| 140 | onLayoutChange(layout) { |
---|
| 141 | if (this.props.onLayoutChange) this.props.onLayoutChange(layout); |
---|
| 142 | //this.props.onLayoutChange(layout); |
---|
| 143 | this.setState({layout: layout}); |
---|
| 144 | } |
---|
| 145 | |
---|
| 146 | /** |
---|
| 147 | * This events run on removing i-th item in 'react-grid-layout'. |
---|
| 148 | * @param {number} i id of item to be removed |
---|
| 149 | */ |
---|
| 150 | onRemoveItem(i) { |
---|
| 151 | this.setState({items: _.reject(this.state.items, {i:i})}); |
---|
| 152 | } |
---|
| 153 | |
---|
| 154 | /** |
---|
| 155 | * WidgetsContainer render method. |
---|
| 156 | * @returns {JSX.Element} WidgetsContainer |
---|
| 157 | */ |
---|
| 158 | render() { |
---|
| 159 | return ( |
---|
| 160 | <div> |
---|
| 161 | <ResponsiveReactGridLayout |
---|
| 162 | layout={this.state.layout} |
---|
| 163 | onLayoutChange={this.onLayoutChange} |
---|
| 164 | onBreakpointChange={this.onBreakpointChange} |
---|
| 165 | cols={{ lg: 12, md: 10, sm: 6, xs: 4, xxs: 2 }} |
---|
| 166 | className="layout" |
---|
| 167 | {...this.props} |
---|
| 168 | > |
---|
| 169 | {_.map(this.state.items, el=>this.createElement(el))} |
---|
| 170 | </ResponsiveReactGridLayout> |
---|
| 171 | </div> |
---|
| 172 | ); |
---|
| 173 | } |
---|
| 174 | } |
---|
| 175 | |
---|
| 176 | export default WidgetsContainer; |
---|