-
Notifications
You must be signed in to change notification settings - Fork 58
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
React-ify scheduling check output tables #1721
Changes from all commits
c5694e3
13248a4
ecb6011
75f3c01
9f5f43c
82e87d7
0b18d5f
b7f2ba2
af60cfa
0bc91b6
eeb1a74
4647488
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -16,6 +16,15 @@ var SchedulingCheckList = React.createClass({ | |
* | ||
* This might or might not be loaded yet; clicking the heading will load the | ||
* data from the server and expand it. | ||
* | ||
* There is supposedly a functionality in which clicking on a table row will cause it | ||
* to be greyed out, however whether this actually happens depends on your | ||
* scheduling_checks.css (clicking could possibly do nothing, or possibly | ||
* do other things). | ||
* | ||
* Likewise, clicking on a table header will sort by that column. You can also | ||
* add formatting in scheduling_checks.css so that the selected header will | ||
* look different, e.g. be colored differently. | ||
*/ | ||
var SchedulingCheck = React.createClass({ | ||
propTypes: { | ||
|
@@ -28,8 +37,45 @@ var SchedulingCheck = React.createClass({ | |
open: false, | ||
failed: false, | ||
timestamp: "never", | ||
tableState: { | ||
greyed: {}, | ||
sort: -1, | ||
reverse: false | ||
}, | ||
}; | ||
}, | ||
|
||
sortColumn: function (column) { | ||
if (this.state.tableState.sort === column) { | ||
this.setState( { | ||
tableState: React.addons.update(this.state.tableState, | ||
{ reverse: {$set: !this.state.tableState.reverse} } ), | ||
} ); | ||
} else { | ||
this.setState( { | ||
tableState: React.addons.update(this.state.tableState, { sort: {$set: column}, reverse: {$set: false} }), | ||
} ); | ||
} | ||
}, | ||
|
||
greyRow: function(item ){ | ||
newkey = {}; | ||
newkey[item] = !this.state.tableState.greyed[item]; | ||
this.setState( { | ||
tableState: React.addons.update(this.state.tableState, { greyed: {$merge: newkey} } ), | ||
} ); | ||
}, | ||
|
||
resetTable: function () { | ||
this.setState({ | ||
tableState: { | ||
greyed: {}, | ||
sort: -1, | ||
reverse: false | ||
}, | ||
}); | ||
|
||
}, | ||
|
||
handleClick: function () { | ||
if (this.state.open) { | ||
|
@@ -80,18 +126,40 @@ var SchedulingCheck = React.createClass({ | |
} else if (!this.state.data) { | ||
body = <div className="placeholder">loading...</div>; | ||
} else { | ||
var data = JSON.parse(this.state.data); // Might not work on old browsers | ||
var table; | ||
if (data.headings.length == 0) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This case and the other two I can find in this diff will be okay, but it's safer to more or less always use |
||
table = <SelectTable rows = {data.body} header = {false} | ||
saveState = {this.state.tableState} | ||
clickHeader = {this.sortColumn} clickRow = {this.greyRow} | ||
/>; | ||
} else { | ||
var columns = []; | ||
for (i = 0; i < data.headings.length; i++) { | ||
if (data.headings[i]) { | ||
columns[i] = {key: String(i), label: data.headings[i]}; | ||
} else { | ||
columns[i] = {key: String(i), label: "--"}; | ||
} | ||
} | ||
table = <SelectTable rows = {data.body} columns = {columns} | ||
header = {true} saveState = {this.state.tableState} | ||
clickHeader = {this.sortColumn} clickRow = {this.greyRow} | ||
/>; | ||
} | ||
body = <div> | ||
<div className="placeholder"> | ||
(loaded {this.state.timestamp}, click title to close) | ||
</div> | ||
<div className="data" dangerouslySetInnerHTML={{__html: this.state.data}} /> | ||
{table} | ||
</div>; | ||
} | ||
|
||
return <div className="scheduling-check"> | ||
<div className="scheduling-check-title"> | ||
<span onClick={this.handleClick}>{this.props.title}</span> | ||
<RefreshButton onClick={this.loadData} /> | ||
<ResetButton onClick={this.resetTable} /> | ||
</div> | ||
<div className="scheduling-check-body"> | ||
{body} | ||
|
@@ -114,3 +182,89 @@ var RefreshButton = React.createClass({ | |
</button>; | ||
}, | ||
}); | ||
|
||
/** | ||
* Calls its onClick prop to reset table greying/sorting | ||
*/ | ||
var ResetButton = React.createClass({ | ||
propTypes: { | ||
onClick: React.PropTypes.func.isRequired, | ||
}, | ||
|
||
render: function () { | ||
return <button onClick={this.props.onClick} className="reset-button"> | ||
Reset | ||
</button>; | ||
}, | ||
}); | ||
|
||
// Modified from react-json-table example code. | ||
var SelectTable = React.createClass({ | ||
|
||
propTypes: { | ||
rows: React.PropTypes.array.isRequired, | ||
saveState: React.PropTypes.shape({ | ||
greyed: React.PropTypes.object.isRequired, | ||
sort: React.PropTypes.any.isRequired, | ||
reverse: React.PropTypes.bool.isRequired | ||
}).isRequired, | ||
header: React.PropTypes.bool.isRequired, | ||
columns: React.PropTypes.array, | ||
clickHeader: React.PropTypes.func.isRequired, | ||
clickRow: React.PropTypes.func.isRequired | ||
}, | ||
|
||
getInitialState: function(){ | ||
return {}; | ||
}, | ||
render: function(){ | ||
// clone the rows | ||
items = this.props.rows.slice(); | ||
|
||
items = _.sortBy(items, this.props.saveState.sort); | ||
|
||
if (this.props.saveState.reverse) items.reverse(); | ||
|
||
return <JsonTable | ||
rows={items} | ||
columns={this.props.columns} | ||
settings={ this.getSettings() } | ||
onClickHeader={ this.onClickHeader } | ||
onClickRow={ this.onClickRow } | ||
/>; | ||
}, | ||
|
||
getSettings: function(){ | ||
var me = this; | ||
// We will add some classes to the selected rows and cells | ||
return { | ||
headerClass: function( current, key ){ | ||
if( me.props.saveState.sort == key ) { | ||
if ( me.props.saveState.reverse) { | ||
return current + ' headerSelected sortReversed'; | ||
} else { | ||
return current + ' headerSelected'; | ||
} | ||
} else { | ||
return current; | ||
} | ||
}, | ||
rowClass: function( current, item ){ | ||
if( me.props.saveState.greyed[item] ) { | ||
return current + ' rowGreyed'; | ||
} else { | ||
return current; | ||
} | ||
}, | ||
header: this.props.header | ||
}; | ||
}, | ||
|
||
onClickHeader: function( e, column ){ | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. These two handlers are where you want to pass the state updates upwards, not in There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I did that initially; it didn't work. The parent gets passed an outdated version of the state (in particular, the most recent state change isn't processed). There's a thing on https://facebook.github.io/react/docs/component-api.html which seems to be relevant, about how it doesn't immediately mutate the state. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. So I think what you want to do is to have all of the state exist in the SchedulingCheck, and none in the SelectTable -- SchedulingCheck can just pass what rows are grayed and such down to SelectTable as a prop. Then the event handler here just calls the callback from SchedulingCheck to update its state, which will then rerender a new SelectTable with a different set of rows grayed. (In fact, React is smart and will do the minimal set of DOM updates necessary to accomplish this.) Either that or I don't understand the issue -- you are setting the new state to whatever you want it to be; React may batch updates, but if you make a change that change will get there eventually. |
||
this.props.clickHeader(column); | ||
}, | ||
|
||
onClickRow: function( e, item ){ | ||
this.props.clickRow(item); | ||
} | ||
}); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why is this in state? It doesn't seem like it ever gets modified.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
(this is no longer the case)