2014-08-10 03:17:10 +00:00
|
|
|
/*
|
|
|
|
Copyright 2014 The Camlistore Authors
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
goog.provide('cam.Header');
|
|
|
|
|
2014-08-16 17:55:55 +00:00
|
|
|
goog.require('goog.Uri');
|
|
|
|
|
2014-08-10 03:17:10 +00:00
|
|
|
goog.require('cam.reactUtil');
|
|
|
|
goog.require('cam.SpritedImage');
|
|
|
|
|
|
|
|
cam.Header = React.createClass({
|
|
|
|
displayName: 'Header',
|
|
|
|
|
|
|
|
PIGGY_NATIVE_WIDTH: 88,
|
|
|
|
PIGGY_NATIVE_HEIGHT: 62,
|
|
|
|
PIGGY_MARGIN: {
|
2014-08-29 23:08:19 +00:00
|
|
|
LEFT: 1,
|
|
|
|
RIGHT: 4,
|
|
|
|
TOP: -1,
|
|
|
|
BOTTOM: 1,
|
2014-08-10 03:17:10 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
SEARCH_MARGIN: {
|
|
|
|
LEFT: 180,
|
|
|
|
RIGHT: 145,
|
|
|
|
},
|
|
|
|
|
|
|
|
propTypes: {
|
|
|
|
currentSearch: React.PropTypes.string,
|
2014-08-30 14:17:24 +00:00
|
|
|
errors: React.PropTypes.arrayOf(
|
|
|
|
React.PropTypes.shape({
|
|
|
|
error: React.PropTypes.string.isRequired,
|
|
|
|
onClick: React.PropTypes.func,
|
|
|
|
url: React.PropTypes.string,
|
|
|
|
}).isRequired
|
|
|
|
).isRequired,
|
|
|
|
homeURL: React.PropTypes.instanceOf(goog.Uri).isRequired,
|
2014-08-10 03:17:10 +00:00
|
|
|
height: React.PropTypes.number.isRequired,
|
2014-08-30 13:41:19 +00:00
|
|
|
importersURL: React.PropTypes.instanceOf(goog.Uri).isRequired,
|
2014-08-10 03:17:10 +00:00
|
|
|
mainControls: React.PropTypes.arrayOf(React.PropTypes.renderable),
|
|
|
|
onNewPermanode: React.PropTypes.func,
|
|
|
|
onSearch: React.PropTypes.func,
|
2014-08-16 17:55:55 +00:00
|
|
|
searchRootsURL: React.PropTypes.instanceOf(goog.Uri).isRequired,
|
2014-08-21 17:36:39 +00:00
|
|
|
statusURL: React.PropTypes.instanceOf(goog.Uri).isRequired,
|
2014-08-10 03:17:10 +00:00
|
|
|
subControls: React.PropTypes.arrayOf(React.PropTypes.renderable),
|
|
|
|
timer: React.PropTypes.shape({setTimeout:React.PropTypes.func.isRequired, clearTimeout:React.PropTypes.func.isRequired}).isRequired,
|
|
|
|
width: React.PropTypes.number.isRequired,
|
|
|
|
},
|
|
|
|
|
|
|
|
focusSearch: function() {
|
|
|
|
this.getSearchNode_().focus();
|
|
|
|
this.getSearchNode_().select();
|
|
|
|
},
|
|
|
|
|
|
|
|
getInitialState: function() {
|
|
|
|
return {
|
|
|
|
currentSearch: this.props.currentSearch,
|
|
|
|
menuVisible: false,
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
componentWillReceiveProps: function(nextProps) {
|
|
|
|
if (nextProps.currentSearch != this.props.currentSearch) {
|
|
|
|
this.setState({currentSearch: nextProps.currentSearch});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
render: function() {
|
|
|
|
return React.DOM.div(
|
|
|
|
{
|
|
|
|
className: React.addons.classSet({
|
|
|
|
'cam-header': true,
|
|
|
|
'cam-header-sub-active': this.props.subControls.length,
|
|
|
|
}),
|
|
|
|
style: {
|
|
|
|
width: this.props.width,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
React.DOM.table(
|
|
|
|
{
|
|
|
|
className: 'cam-header-main',
|
|
|
|
},
|
|
|
|
React.DOM.tr(null,
|
|
|
|
this.getPiggy_(),
|
|
|
|
this.getTitle_(),
|
|
|
|
this.getSearchbox_(),
|
|
|
|
this.getMainControls_()
|
|
|
|
)
|
|
|
|
),
|
|
|
|
this.getSubheader_(),
|
|
|
|
this.getMenuDropdown_()
|
|
|
|
)
|
|
|
|
},
|
|
|
|
|
|
|
|
getPiggy_: function() {
|
2014-08-29 23:08:19 +00:00
|
|
|
var props = {
|
|
|
|
sheetWidth: 11,
|
|
|
|
spriteWidth: this.PIGGY_NATIVE_WIDTH,
|
|
|
|
spriteHeight: this.PIGGY_NATIVE_HEIGHT,
|
|
|
|
style: cam.reactUtil.getVendorProps({
|
|
|
|
position: 'absolute',
|
|
|
|
left: this.PIGGY_MARGIN.LEFT,
|
|
|
|
top: this.PIGGY_MARGIN.TOP,
|
|
|
|
transform: 'scale(' + this.getPiggyScale_() + ')',
|
|
|
|
transformOrigin: '0 0',
|
|
|
|
}),
|
|
|
|
};
|
|
|
|
|
2014-08-30 14:17:24 +00:00
|
|
|
var image = function() {
|
|
|
|
if (this.props.errors.length) {
|
2014-08-29 23:08:19 +00:00
|
|
|
return cam.SpritedAnimation(cam.object.extend(props, {
|
|
|
|
key: 'error',
|
|
|
|
loopDelay: 10 * 1000,
|
|
|
|
numFrames: 65,
|
|
|
|
src: 'glitch/npc_piggy__x1_too_much_nibble_png_1354829441.png',
|
|
|
|
}));
|
|
|
|
} else {
|
|
|
|
return cam.SpritedImage(cam.object.extend(props, {
|
|
|
|
key: 'ok',
|
|
|
|
index: 5,
|
|
|
|
src: 'glitch/npc_piggy__x1_chew_png_1354829433.png',
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-08-10 03:17:10 +00:00
|
|
|
return React.DOM.td(
|
|
|
|
{
|
|
|
|
className: 'cam-header-item',
|
|
|
|
style: {
|
|
|
|
minWidth: this.getPiggyWidth_() + this.PIGGY_MARGIN.LEFT + this.PIGGY_MARGIN.RIGHT,
|
|
|
|
},
|
2014-08-10 15:43:16 +00:00
|
|
|
onClick: this.handleClick_,
|
2014-08-10 03:17:10 +00:00
|
|
|
onMouseEnter: this.handleMouseEnter_,
|
|
|
|
onMouseLeave: this.handleMouseLeave_,
|
|
|
|
},
|
2014-08-30 14:17:24 +00:00
|
|
|
image.call(this)
|
2014-08-10 03:17:10 +00:00
|
|
|
)
|
|
|
|
},
|
|
|
|
|
|
|
|
getTitle_: function() {
|
|
|
|
return React.DOM.td(
|
|
|
|
{
|
|
|
|
className: 'cam-header-item cam-header-title',
|
2014-08-10 15:43:16 +00:00
|
|
|
onClick: this.handleClick_,
|
2014-08-10 03:17:10 +00:00
|
|
|
onMouseEnter: this.handleMouseEnter_,
|
|
|
|
onMouseLeave: this.handleMouseLeave_,
|
|
|
|
},
|
|
|
|
React.DOM.span(null, 'Pudgy'),
|
|
|
|
React.DOM.span(null, '\u25BE')
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
getSearchbox_: function() {
|
|
|
|
return React.DOM.td(
|
|
|
|
{
|
|
|
|
className: 'cam-header-item',
|
|
|
|
style: {
|
|
|
|
width: '100%',
|
|
|
|
}
|
|
|
|
},
|
|
|
|
React.DOM.form(
|
|
|
|
{
|
|
|
|
onSubmit: this.handleSearchSubmit_,
|
|
|
|
},
|
|
|
|
React.DOM.input(
|
|
|
|
{
|
|
|
|
onChange: this.handleSearchChange_,
|
|
|
|
placeholder: 'Search...',
|
|
|
|
ref: 'searchbox',
|
|
|
|
value: this.state.currentSearch,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
},
|
|
|
|
|
|
|
|
getMainControls_: function() {
|
|
|
|
return React.DOM.td(
|
|
|
|
{
|
|
|
|
className: React.addons.classSet({
|
|
|
|
'cam-header-item': true,
|
|
|
|
'cam-header-main-controls': true,
|
|
|
|
'cam-header-main-controls-empty': !this.props.mainControls.length,
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
this.props.mainControls
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
getMenuDropdown_: function() {
|
2014-08-30 14:17:24 +00:00
|
|
|
var errorItems = this.props.errors.map(function(err) {
|
2014-08-29 23:08:19 +00:00
|
|
|
var children = [
|
|
|
|
React.DOM.i({className: 'fa fa-exclamation-circle cam-header-menu-item-icon'}),
|
|
|
|
err.error
|
|
|
|
];
|
2014-08-30 14:17:24 +00:00
|
|
|
return this.getMenuItem_(children, err.url, err.onClick, 'cam-header-menu-item-error');
|
2014-08-29 23:08:19 +00:00
|
|
|
}, this);
|
|
|
|
|
2014-08-10 03:17:10 +00:00
|
|
|
return React.DOM.div(
|
|
|
|
{
|
|
|
|
className: 'cam-header-menu-dropdown',
|
|
|
|
onClick: this.handleDropdownClick_,
|
|
|
|
onMouseEnter: this.handleMouseEnter_,
|
|
|
|
onMouseLeave: this.handleMouseLeave_,
|
|
|
|
style: cam.reactUtil.getVendorProps({
|
|
|
|
transform: 'translate3d(0, ' + this.getMenuTranslate_() + '%, 0)',
|
|
|
|
}),
|
|
|
|
},
|
2014-08-30 14:17:24 +00:00
|
|
|
this.getMenuItem_('Home', this.props.homeURL),
|
2014-10-26 08:41:19 +00:00
|
|
|
this.getMenuItem_('Upload...', null, this.props.onUpload),
|
2014-08-16 17:55:55 +00:00
|
|
|
|
|
|
|
// TODO(aa): Create a new permanode UI that delays creating the permanode until the user confirms, then change this to a link to that UI.
|
2014-08-16 18:00:05 +00:00
|
|
|
// TODO(aa): Also I keep going back and forth about whether we should call this 'permanode' or 'set' in the UI. Hrm.
|
2014-08-30 14:17:24 +00:00
|
|
|
this.getMenuItem_('New set', null, this.props.onNewPermanode),
|
2014-08-16 17:55:55 +00:00
|
|
|
|
2014-08-30 14:17:24 +00:00
|
|
|
this.getMenuItem_('Importers', this.props.importersURL),
|
|
|
|
this.getMenuItem_('Server status', this.props.statusURL),
|
|
|
|
this.getMenuItem_('Search roots', this.props.searchRootsURL),
|
2014-08-29 23:08:19 +00:00
|
|
|
errorItems
|
2014-08-16 17:55:55 +00:00
|
|
|
);
|
|
|
|
},
|
|
|
|
|
2014-08-30 14:17:24 +00:00
|
|
|
getMenuItem_: function(text, opt_link, opt_onClick, opt_class) {
|
|
|
|
if (!text || (!opt_onClick && !opt_link)) {
|
2014-08-21 17:36:39 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2014-08-29 23:08:19 +00:00
|
|
|
var className = 'cam-header-menu-item';
|
|
|
|
if (opt_class) {
|
|
|
|
className += ' ' + opt_class;
|
|
|
|
}
|
2014-08-10 03:17:10 +00:00
|
|
|
|
2014-08-30 14:17:24 +00:00
|
|
|
var ctor = opt_link ? React.DOM.a : React.DOM.div;
|
|
|
|
return ctor(
|
2014-08-10 03:17:10 +00:00
|
|
|
{
|
2014-08-30 14:17:24 +00:00
|
|
|
className: className,
|
|
|
|
href: opt_link,
|
|
|
|
onClick: opt_onClick,
|
2014-08-10 03:17:10 +00:00
|
|
|
},
|
|
|
|
text
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
getSubheader_: function() {
|
|
|
|
return React.DOM.div(
|
|
|
|
{
|
|
|
|
className: 'cam-header-sub',
|
|
|
|
},
|
|
|
|
this.props.subControls
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
getMenuTranslate_: function() {
|
|
|
|
if (this.state.menuVisible) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
// 110% because it has a shadow that we don't want to double-up with the shadow from the header.
|
|
|
|
return -110;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
getPiggyHeight_: function() {
|
|
|
|
return this.props.height - this.PIGGY_MARGIN.TOP - this.PIGGY_MARGIN.BOTTOM;
|
|
|
|
},
|
|
|
|
|
|
|
|
getPiggyWidth_: function() {
|
|
|
|
return this.getPiggyScale_() * this.PIGGY_NATIVE_WIDTH;
|
|
|
|
},
|
|
|
|
|
|
|
|
getPiggyScale_: function() {
|
|
|
|
return this.getPiggyHeight_() / this.PIGGY_NATIVE_HEIGHT;
|
|
|
|
},
|
|
|
|
|
2014-08-10 15:43:16 +00:00
|
|
|
handleClick_: function() {
|
2014-08-10 16:15:52 +00:00
|
|
|
this.setState({menuVisible: !this.state.menuVisible});
|
2014-08-10 15:43:16 +00:00
|
|
|
},
|
|
|
|
|
2014-08-10 03:17:10 +00:00
|
|
|
handleMouseEnter_: function() {
|
|
|
|
this.clearTimer_();
|
|
|
|
this.setTimer_(true);
|
|
|
|
},
|
|
|
|
|
|
|
|
handleMouseLeave_: function() {
|
|
|
|
this.clearTimer_();
|
|
|
|
this.setTimer_(false);
|
|
|
|
},
|
|
|
|
|
2014-08-10 15:43:16 +00:00
|
|
|
handleDropdownClick_: function(e) {
|
2014-08-10 03:17:10 +00:00
|
|
|
this.clearTimer_();
|
|
|
|
this.setState({menuVisible:false});
|
|
|
|
},
|
|
|
|
|
|
|
|
setTimer_: function(show) {
|
|
|
|
this.timerId_ = this.props.timer.setTimeout(this.handleTimer_.bind(null, show), 250);
|
|
|
|
},
|
|
|
|
|
|
|
|
clearTimer_: function() {
|
|
|
|
if (this.timerId_) {
|
|
|
|
this.props.timer.clearTimeout(this.timerId_);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
handleTimer_: function(show) {
|
|
|
|
this.setState({menuVisible:show});
|
|
|
|
},
|
|
|
|
|
|
|
|
handleSearchChange_: function(e) {
|
|
|
|
this.setState({currentSearch: e.target.value});
|
|
|
|
},
|
|
|
|
|
|
|
|
handleSearchSubmit_: function(e) {
|
|
|
|
this.props.onSearch(this.getSearchNode_().value);
|
|
|
|
e.preventDefault();
|
|
|
|
},
|
|
|
|
|
|
|
|
getSearchNode_: function() {
|
|
|
|
return this.refs['searchbox'].getDOMNode();
|
|
|
|
},
|
|
|
|
});
|