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; |
---|