mirror of
https://github.com/ENSL/ensl_gathers.git
synced 2025-01-24 10:10:58 +00:00
984 lines
28 KiB
JavaScript
984 lines
28 KiB
JavaScript
"use strict";
|
||
|
||
var VoteButton = React.createClass({displayName: "VoteButton",
|
||
cancelVote: function (e) {
|
||
socket.emit("gather:vote", {
|
||
leader: {
|
||
candidate: null
|
||
}
|
||
});
|
||
},
|
||
vote: function (e) {
|
||
e.preventDefault();
|
||
socket.emit("gather:vote", {
|
||
leader: {
|
||
candidate: parseInt(e.target.value, 10)
|
||
}
|
||
});
|
||
},
|
||
render: function () {
|
||
if (this.props.currentGatherer === null) {
|
||
return false;
|
||
}
|
||
if (this.props.currentGatherer.leaderVote === this.props.candidate.id) {
|
||
return (
|
||
React.createElement("button", {
|
||
onClick: this.cancelVote,
|
||
className: "btn btn-xs btn-success"}, "Voted"
|
||
)
|
||
);
|
||
} else {
|
||
return (
|
||
React.createElement("button", {
|
||
onClick: this.vote,
|
||
className: "btn btn-xs btn-default",
|
||
value: this.props.candidate.id}, "Vote"
|
||
)
|
||
);
|
||
}
|
||
}
|
||
});
|
||
|
||
var SelectPlayerButton = React.createClass({displayName: "SelectPlayerButton",
|
||
selectPlayer: function (e) {
|
||
e.preventDefault();
|
||
socket.emit("gather:select", {
|
||
player: parseInt(e.target.value, 10)
|
||
})
|
||
},
|
||
render: function () {
|
||
if (this.props.gatherer.leader) {
|
||
return (React.createElement("button", {
|
||
className: "btn btn-xs btn-default",
|
||
"data-disabled": "true"}, "Leader"));
|
||
} else if (this.props.gatherer.team !== "lobby") {
|
||
return (React.createElement("button", {
|
||
onClick: this.selectPlayer,
|
||
value: this.props.gatherer.id,
|
||
className: "btn btn-xs btn-default"}, " Reselect"
|
||
)
|
||
);
|
||
} else {
|
||
return (React.createElement("button", {
|
||
onClick: this.selectPlayer,
|
||
value: this.props.gatherer.id,
|
||
className: "btn btn-xs btn-primary"}, " Select"
|
||
)
|
||
);
|
||
}
|
||
}
|
||
});
|
||
|
||
var GathererList = React.createClass({displayName: "GathererList",
|
||
memberList: function () {
|
||
var self = this;
|
||
return this.props.gather.gatherers.filter(function (gatherer) {
|
||
return gatherer.team === self.props.team;
|
||
}).sort(function (gatherer) {
|
||
return (gatherer.leader) ? 1 : -1;
|
||
});
|
||
},
|
||
render: function () {
|
||
var extractGatherer = function (gatherer) {
|
||
var image;
|
||
if (gatherer.leader) {
|
||
image = (React.createElement("img", {src: "/images/commander.png",
|
||
alt: "Commander",
|
||
height: "20",
|
||
width: "20"}));
|
||
}
|
||
return (
|
||
React.createElement("tr", {key: gatherer.id},
|
||
React.createElement("td", {className: "col-md-1"}, image),
|
||
React.createElement("td", {className: "col-md-11"}, gatherer.user.username)
|
||
)
|
||
);
|
||
}
|
||
var members = this.memberList().map(extractGatherer);
|
||
return (
|
||
React.createElement("table", {className: "table"},
|
||
React.createElement("tbody", null,
|
||
members
|
||
)
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
var GatherTeams = React.createClass({displayName: "GatherTeams",
|
||
render: function () {
|
||
return (
|
||
React.createElement("div", {className: "panel-body"},
|
||
React.createElement("div", {className: "row"},
|
||
React.createElement("div", {className: "col-md-6"},
|
||
React.createElement("div", {className: "panel panel-default"},
|
||
React.createElement("div", {className: "panel-heading"},
|
||
"Aliens"
|
||
),
|
||
React.createElement(GathererList, {gather: this.props.gather, team: "alien"})
|
||
)
|
||
),
|
||
React.createElement("div", {className: "col-md-6"},
|
||
React.createElement("div", {className: "panel panel-default"},
|
||
React.createElement("div", {className: "panel-heading"},
|
||
"Marines"
|
||
),
|
||
React.createElement(GathererList, {gather: this.props.gather, team: "marine"})
|
||
)
|
||
)
|
||
)
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
var ElectionProgressBar = React.createClass({displayName: "ElectionProgressBar",
|
||
componentDidMount: function () {
|
||
var self = this;
|
||
this.timer = setInterval(function () {
|
||
self.forceUpdate();
|
||
}, 900);
|
||
},
|
||
progress: function () {
|
||
var interval = this.props.gather.election.interval;
|
||
var startTime = (new Date(this.props.gather.election.startTime)).getTime();
|
||
var msTranspired = Math.floor((new Date()).getTime() - startTime);
|
||
|
||
return {
|
||
num: msTranspired,
|
||
den: interval,
|
||
barMessage: Math.floor((interval - msTranspired) / 1000) + "s remaining"
|
||
}
|
||
},
|
||
componentWillUnmount: function () {
|
||
clearInterval(this.timer);
|
||
},
|
||
render: function () {
|
||
return (React.createElement(ProgressBar, {progress: this.progress()}));
|
||
}
|
||
});
|
||
|
||
var ProgressBar = React.createClass({displayName: "ProgressBar",
|
||
render: function () {
|
||
var style = {
|
||
width: Math.round((this.props.progress.num / this.props.progress.den * 100)) + "%"
|
||
};
|
||
var barMessage = this.props.progress.barMessage || "";
|
||
return (
|
||
React.createElement("div", {className: "progress"},
|
||
React.createElement("div", {className: "progress-bar progress-bar-striped active",
|
||
"data-role": "progressbar",
|
||
"data-aria-valuenow": this.props.progress.num,
|
||
"data-aria-valuemin": "0",
|
||
"data-aria-valuemax": this.props.progress.den,
|
||
style: style}, barMessage
|
||
)
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
var GatherProgress = React.createClass({displayName: "GatherProgress",
|
||
stateDescription: function () {
|
||
switch(this.props.gather.state) {
|
||
case "gathering":
|
||
return "Waiting for more gatherers.";
|
||
case "election":
|
||
return "Currently voting for team leaders.";
|
||
case "selection":
|
||
return "Waiting for leaders to pick teams.";
|
||
case "done":
|
||
return "Gather completed.";
|
||
default:
|
||
return "Initialising gather.";
|
||
}
|
||
},
|
||
gatheringProgress: function () {
|
||
var num = this.props.gather.gatherers.length;
|
||
var den = 12;
|
||
var remaining = den - num;
|
||
var message = (remaining === 1) ? "Waiting for last player" : "Waiting for " + remaining + " more players";
|
||
return {
|
||
num: num,
|
||
den: den,
|
||
message: message
|
||
};
|
||
},
|
||
electionProgress: function () {
|
||
var num = this.props.gather.gatherers.reduce(function (acc, gatherer) {
|
||
if (gatherer.leaderVote) acc++;
|
||
return acc;
|
||
}, 0);
|
||
var den = 12;
|
||
return {
|
||
num: num,
|
||
den: den,
|
||
message: den - num + " more votes required"
|
||
};
|
||
},
|
||
selectionProgress: function () {
|
||
var num = this.props.gather.gatherers.reduce(function (acc, gatherer) {
|
||
if (gatherer.team !== "lobby") acc++;
|
||
return acc;
|
||
}, 0);
|
||
var den = 12;
|
||
|
||
return {
|
||
num: num,
|
||
den: den,
|
||
message: num + " out of " + den + " players assigned"
|
||
};
|
||
},
|
||
render: function () {
|
||
var progress, progressBar;
|
||
var gatherState = this.props.gather.state;
|
||
if (gatherState === 'gathering' && this.props.gather.gatherers.length) {
|
||
progress = this.gatheringProgress();
|
||
progressBar = (React.createElement(ProgressBar, {progress: progress}));
|
||
} else if (gatherState === 'election') {
|
||
progress = this.electionProgress();
|
||
progressBar = (React.createElement(ElectionProgressBar, React.__spread({}, this.props, {progress: progress})));
|
||
} else if (gatherState === 'selection') {
|
||
progress = this.selectionProgress();
|
||
progressBar = (React.createElement(ProgressBar, {progress: progress}));
|
||
}
|
||
|
||
if (!progress) return false;
|
||
|
||
return (
|
||
React.createElement("div", {className: "panel-body no-bottom"},
|
||
React.createElement("p", null, React.createElement("strong", null, this.stateDescription()), " ", progress.message),
|
||
progressBar
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
var GatherActions = React.createClass({displayName: "GatherActions",
|
||
joinGather: function (e) {
|
||
e.preventDefault();
|
||
socket.emit("gather:join");
|
||
},
|
||
leaveGather: function (e) {
|
||
e.preventDefault();
|
||
socket.emit("gather:leave");
|
||
},
|
||
confirmTeam: function (e) {
|
||
e.preventDefault();
|
||
socket.emit("gather:select:confirm");
|
||
},
|
||
inviteToGather: function (e) {
|
||
e.preventDefault();
|
||
alert("Boop!");
|
||
},
|
||
render: function () {
|
||
var joinButton;
|
||
if (this.props.currentGatherer) {
|
||
joinButton = (React.createElement("li", null, React.createElement("button", {
|
||
onClick: this.leaveGather,
|
||
className: "btn btn-danger"}, "Leave Gather")));
|
||
} else if (this.props.gather.state === 'gathering') {
|
||
joinButton = (
|
||
React.createElement("button", {
|
||
onClick: this.joinGather,
|
||
className: "btn btn-success"}, "Join Gather")
|
||
);
|
||
}
|
||
|
||
var confirmTeam;
|
||
if (this.props.currentGatherer &&
|
||
this.props.currentGatherer.leader &&
|
||
this.props.gather.state === 'selection' &&
|
||
this.props.gather.gatherers.every(function (gatherer) {
|
||
return gatherer.team !== 'lobby';
|
||
}) ) {
|
||
if (this.props.currentGatherer.confirm) {
|
||
confirmTeam = (
|
||
React.createElement("li", null,
|
||
React.createElement("button", {
|
||
className: "btn btn-default",
|
||
"data-disabled": "true"
|
||
},
|
||
"Confirmed"
|
||
)
|
||
)
|
||
);
|
||
} else {
|
||
confirmTeam = (
|
||
React.createElement("li", null,
|
||
React.createElement("button", {
|
||
className: "btn btn-success",
|
||
onClick: this.confirmTeam
|
||
},
|
||
"Confirm Team"
|
||
)
|
||
)
|
||
);
|
||
}
|
||
}
|
||
|
||
var inviteButton;
|
||
if (this.props.gather.state === 'gathering') {
|
||
inviteButton = (React.createElement("li", null, React.createElement("button", {
|
||
onClick: this.inviteToGather,
|
||
className: "btn btn-primary"}, "Invite to Gather")));
|
||
}
|
||
|
||
return (
|
||
React.createElement("div", {className: "panel-footer text-right"},
|
||
React.createElement("ul", {className: "list-inline no-bottom"},
|
||
confirmTeam,
|
||
inviteButton,
|
||
joinButton
|
||
)
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
var ServerVoting = React.createClass({displayName: "ServerVoting",
|
||
handleServerVote: function (e) {
|
||
e.preventDefault();
|
||
socket.emit("gather:vote", {
|
||
server: {
|
||
id: parseInt(e.target.value, 10)
|
||
}
|
||
});
|
||
},
|
||
votesForServer: function (server) {
|
||
return this.props.gather.gatherers.reduce(function (acc, gatherer) {
|
||
if (server.id === gatherer.serverVote) acc++;
|
||
return acc;
|
||
}, 0);
|
||
},
|
||
render: function () {
|
||
var self = this;
|
||
var servers = self.props.servers.map(function (server) {
|
||
var voteButton;
|
||
if (self.props.currentGatherer.serverVote === server.id) {
|
||
voteButton = (React.createElement("button", {
|
||
"data-disabled": "true",
|
||
className: "btn btn-xs btn-success"},
|
||
"Voted"))
|
||
} else {
|
||
voteButton = (React.createElement("button", {
|
||
onClick: self.handleServerVote,
|
||
value: server.id,
|
||
className: "btn btn-xs btn-primary"},
|
||
"Vote"));
|
||
}
|
||
return (
|
||
React.createElement("tr", {key: server.id},
|
||
React.createElement("td", {className: "col-md-6"}, server.name),
|
||
React.createElement("td", {className: "col-md-6 text-right"},
|
||
self.votesForServer(server), " Votes ",
|
||
voteButton
|
||
)
|
||
)
|
||
);
|
||
});
|
||
return (
|
||
React.createElement("div", {className: "panel panel-default"},
|
||
React.createElement("div", {className: "panel-heading"},
|
||
"Server Voting"
|
||
),
|
||
React.createElement("table", {id: "serverVoteTable", className: "table table-condensed table-hover voting-table"},
|
||
servers
|
||
)
|
||
)
|
||
);
|
||
}
|
||
})
|
||
|
||
var MapVoting = React.createClass({displayName: "MapVoting",
|
||
handleMapVote: function (e) {
|
||
e.preventDefault();
|
||
socket.emit("gather:vote", {
|
||
map: {
|
||
id: parseInt(e.target.value, 10)
|
||
}
|
||
});
|
||
},
|
||
votesForMap: function (map) {
|
||
return this.props.gather.gatherers.reduce(function (acc, gatherer) {
|
||
if (map.id === gatherer.mapVote) acc++;
|
||
return acc;
|
||
}, 0);
|
||
},
|
||
render: function () {
|
||
var self = this;
|
||
var maps = self.props.maps.map(function (map) {
|
||
var voteButton;
|
||
if (self.props.currentGatherer.mapVote === map.id) {
|
||
voteButton = (React.createElement("button", {
|
||
"data-disabled": "true",
|
||
className: "btn btn-xs btn-success"},
|
||
"Voted"))
|
||
} else {
|
||
voteButton = (React.createElement("button", {
|
||
onClick: self.handleMapVote,
|
||
value: map.id,
|
||
className: "btn btn-xs btn-primary"},
|
||
"Vote"));
|
||
}
|
||
return (
|
||
React.createElement("tr", {key: map.id},
|
||
React.createElement("td", {className: "col-md-6"}, map.name),
|
||
React.createElement("td", {className: "col-md-6 text-right"},
|
||
self.votesForMap(map), " Votes ",
|
||
voteButton
|
||
)
|
||
)
|
||
);
|
||
});
|
||
return (
|
||
React.createElement("div", {className: "panel panel-default"},
|
||
React.createElement("div", {className: "panel-heading"},
|
||
"Map Voting"
|
||
),
|
||
React.createElement("table", {className: "table table-condensed table-hover voting-table"},
|
||
maps
|
||
)
|
||
)
|
||
);
|
||
}
|
||
})
|
||
|
||
var Gather = React.createClass({displayName: "Gather",
|
||
getDefaultProps: function () {
|
||
return {
|
||
gather: {
|
||
gatherers: []
|
||
}
|
||
}
|
||
},
|
||
componentDidMount: function () {
|
||
var self = this;
|
||
socket.on("gather:refresh", function (data) {
|
||
self.setProps(data);
|
||
});
|
||
},
|
||
|
||
render: function () {
|
||
if (this.props.gather.state === 'done') {
|
||
return (React.createElement(CompletedGather, React.__spread({}, this.props)));
|
||
}
|
||
|
||
var voting;
|
||
if (this.props.currentGatherer) {
|
||
voting = (
|
||
React.createElement("div", {className: "panel-body"},
|
||
React.createElement("div", {className: "row"},
|
||
React.createElement("div", {className: "col-md-6"},
|
||
React.createElement(MapVoting, React.__spread({}, this.props))
|
||
),
|
||
React.createElement("div", {className: "col-md-6"},
|
||
React.createElement(ServerVoting, React.__spread({}, this.props))
|
||
)
|
||
)
|
||
)
|
||
);
|
||
}
|
||
|
||
var gatherTeams;
|
||
if (this.props.gather.state === 'selection') {
|
||
gatherTeams = React.createElement(GatherTeams, {gather: this.props.gather})
|
||
}
|
||
return (
|
||
React.createElement("div", {className: "panel panel-default"},
|
||
React.createElement("div", {className: "panel-heading"},
|
||
React.createElement("strong", null, "Current Gather"),
|
||
React.createElement("span", {className: "badge add-left"}, this.props.gather.gatherers.length)
|
||
),
|
||
React.createElement(GatherProgress, React.__spread({}, this.props)),
|
||
React.createElement(Gatherers, React.__spread({}, this.props)),
|
||
gatherTeams,
|
||
voting,
|
||
React.createElement(GatherActions, React.__spread({}, this.props))
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
var Gatherers = React.createClass({displayName: "Gatherers",
|
||
joinGather: function (e) {
|
||
e.preventDefault();
|
||
socket.emit("gather:join");
|
||
},
|
||
render: function () {
|
||
var self = this;
|
||
var gatherers = this.props.gather.gatherers.map(function (gatherer) {
|
||
|
||
// Country
|
||
var country;
|
||
|
||
if (gatherer.user.country) {
|
||
country = (React.createElement("img", {src: "images/blank.gif",
|
||
className: "flag flag-" + gatherer.user.country.toLowerCase(),
|
||
alt: gatherer.user.country}));
|
||
};
|
||
|
||
var division = (React.createElement("span", {className: "label label-primary"}, gatherer.user.ability.division));
|
||
var lifeform = (
|
||
gatherer.user.ability.lifeforms.map(function (lifeform) {
|
||
return (React.createElement("span", {className: "label label-default",
|
||
key: [lifeform, gatherer.id].join("-")}, lifeform));
|
||
})
|
||
);
|
||
var team;
|
||
|
||
if (gatherer.user.team) {
|
||
team = (React.createElement("span", {className: "label label-primary"}, gatherer.user.team.name));
|
||
}
|
||
|
||
var action;
|
||
|
||
if (self.props.gather.state === "election") {
|
||
var votes = self.props.gather.gatherers.reduce(function (acc, voter) {
|
||
if (voter.leaderVote === gatherer.id) acc++;
|
||
return acc;
|
||
}, 0)
|
||
action = (
|
||
React.createElement("span", null,
|
||
React.createElement("small", null, votes + " votes", " "),
|
||
React.createElement(VoteButton, {currentGatherer: self.props.currentGatherer, candidate: gatherer})
|
||
)
|
||
);
|
||
}
|
||
|
||
if (self.props.gather.state === 'selection') {
|
||
if (self.props.currentGatherer && self.props.currentGatherer.leader) {
|
||
action = (
|
||
React.createElement("span", null,
|
||
React.createElement(SelectPlayerButton, {gatherer: gatherer})
|
||
)
|
||
);
|
||
} else {
|
||
if (gatherer.team !== "lobby") {
|
||
action = (React.createElement("span", {className: "label label-success"}, gatherer.team));
|
||
}
|
||
}
|
||
}
|
||
|
||
return (
|
||
React.createElement("tr", {key: gatherer.user.id},
|
||
React.createElement("td", {className: "col-md-5"}, country, " ", gatherer.user.username, " (", gatherer.user.id, ")"),
|
||
React.createElement("td", {className: "col-md-5"},
|
||
lifeform, " ", division, " ", team, " "
|
||
),
|
||
React.createElement("td", {className: "col-md-2 text-right"}, action, " ")
|
||
)
|
||
);
|
||
})
|
||
if (this.props.gather.gatherers.length) {
|
||
return (
|
||
React.createElement("div", {className: "panel-body"},
|
||
React.createElement("div", {className: "panel panel-default"},
|
||
React.createElement("table", {className: "table roster-table"},
|
||
React.createElement("tbody", null,
|
||
gatherers
|
||
)
|
||
)
|
||
)
|
||
)
|
||
);
|
||
} else {
|
||
return (
|
||
React.createElement("div", {className: "panel-body text-center join-hero"},
|
||
React.createElement("button", {
|
||
onClick: this.joinGather,
|
||
className: "btn btn-success btn-lg"}, "Start a Gather")
|
||
)
|
||
);
|
||
}
|
||
}
|
||
});
|
||
|
||
var CompletedGather = React.createClass({displayName: "CompletedGather",
|
||
countVotes: function (voteType) {
|
||
return this.props.gather.gatherers.reduce(function (acc, gatherer) {
|
||
if (gatherer[voteType] !== null) acc.push(gatherer[voteType]);
|
||
return acc;
|
||
}, []);
|
||
},
|
||
selectedMaps: function () {
|
||
return rankVotes(this.countVotes('mapVote'), this.props.maps).slice(0, 2)
|
||
},
|
||
selectedServer: function () {
|
||
return rankVotes(this.countVotes('serverVote'), this.props.servers).slice(0, 1);
|
||
},
|
||
render: function () {
|
||
var maps = this.selectedMaps();
|
||
var server = this.selectedServer().pop();
|
||
return (
|
||
React.createElement("div", {className: "panel panel-default"},
|
||
React.createElement("div", {className: "panel-heading"},
|
||
React.createElement("strong", null, "Gather Details")
|
||
),
|
||
React.createElement(GatherTeams, {gather: this.props.gather}),
|
||
React.createElement("div", {className: "panel-body"},
|
||
React.createElement("dl", {className: "dl-horizontal"},
|
||
React.createElement("dt", null, "Maps"),
|
||
React.createElement("dd", null, maps.map(function(map) { return map.name}).join(" & ")),
|
||
React.createElement("dt", null, "Server"),
|
||
React.createElement("dd", null, server.name),
|
||
React.createElement("dt", null, "Address"),
|
||
React.createElement("dd", null, server.ip, ":", server.port),
|
||
React.createElement("dt", null, "Password"),
|
||
React.createElement("dd", null, server.password),
|
||
React.createElement("br", null),
|
||
React.createElement("dt", null, " "),
|
||
React.createElement("dd", null, React.createElement("a", {href: ["steam://run/4920/connect", server.ip +":"+server.port, server.password].join("/"),
|
||
className: "btn btn-primary"}, "Click to Join"))
|
||
)
|
||
)
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
"use strict";
|
||
|
||
var socket;
|
||
|
||
function initialiseVisibilityMonitoring (socket) {
|
||
var hidden, visibilityChange;
|
||
if (typeof document.hidden !== "undefined") { // Opera 12.10 and Firefox 18 and later support
|
||
hidden = "hidden";
|
||
visibilityChange = "visibilitychange";
|
||
} else if (typeof document.mozHidden !== "undefined") {
|
||
hidden = "mozHidden";
|
||
visibilityChange = "mozvisibilitychange";
|
||
} else if (typeof document.msHidden !== "undefined") {
|
||
hidden = "msHidden";
|
||
visibilityChange = "msvisibilitychange";
|
||
} else if (typeof document.webkitHidden !== "undefined") {
|
||
hidden = "webkitHidden";
|
||
visibilityChange = "webkitvisibilitychange";
|
||
}
|
||
|
||
document.addEventListener(visibilityChange, function () {
|
||
if (document[hidden]) {
|
||
socket.emit("users:away");
|
||
} else {
|
||
socket.emit("users:online");
|
||
}
|
||
}, false);
|
||
}
|
||
|
||
function initialiseComponents () {
|
||
var socketUrl = window.location.protocol + "//" + window.location.host;
|
||
socket = io(socketUrl)
|
||
.on("connect", function () {
|
||
console.log("Connected");
|
||
})
|
||
.on("reconnect", function () {
|
||
console.log("Reconnected");
|
||
})
|
||
.on("disconnect", function () {
|
||
console.log("Disconnected")
|
||
});
|
||
|
||
initialiseVisibilityMonitoring(socket);
|
||
|
||
// Render Page
|
||
React.render(React.createElement(UserMenu, null), document.getElementById('side-menu'));
|
||
React.render(React.createElement(Chatroom, null), document.getElementById('chatroom'));
|
||
React.render(React.createElement(Gather, null), document.getElementById('gathers'));
|
||
React.render(React.createElement(CurrentUser, null), document.getElementById('currentuser'));
|
||
React.render(React.createElement(AdminPanel, null), document.getElementById('admin-menu'));
|
||
};
|
||
|
||
"use strict";
|
||
|
||
var Chatroom = React.createClass({displayName: "Chatroom",
|
||
getDefaultProps: function () {
|
||
return {
|
||
history: []
|
||
};
|
||
},
|
||
componentDidMount: function () {
|
||
var self = this;
|
||
var TIMER_INTERVAL = 5000; // Every minute
|
||
|
||
socket.on("message:new", function (data) {
|
||
var history = self.props.history;
|
||
history.push(data);
|
||
self.setProps({
|
||
history: history
|
||
});
|
||
self.scrollToBottom();
|
||
});
|
||
|
||
// Message History Retrieved
|
||
socket.on("message:refresh", function (data) {
|
||
self.setProps({
|
||
history: data.chatHistory
|
||
});
|
||
self.scrollToBottom();
|
||
});
|
||
|
||
socket.emit("message:refresh", {});
|
||
|
||
self.timer = setInterval(function () {
|
||
self.forceUpdate();
|
||
}, TIMER_INTERVAL);
|
||
},
|
||
|
||
componentDidUnmount: function () {
|
||
clearInterval(this.timer);
|
||
},
|
||
sendMessage: function (message) {
|
||
socket.emit("newMessage", {message: message});
|
||
},
|
||
scrollToBottom: function () {
|
||
var node = React.findDOMNode(this.refs.messageContainer);
|
||
node.scrollTop = node.scrollHeight;
|
||
},
|
||
render: function () {
|
||
var messages = this.props.history.map(function (message) {
|
||
return (
|
||
React.createElement(ChatMessage, {
|
||
message: message,
|
||
key: message.id})
|
||
);
|
||
});
|
||
return (
|
||
React.createElement("div", {className: "panel panel-default"},
|
||
React.createElement("div", {className: "panel-heading"}, "Gather Chat"),
|
||
React.createElement("div", {className: "panel-body"},
|
||
React.createElement("ul", {className: "chat", id: "chatmessages", ref: "messageContainer"},
|
||
messages
|
||
)
|
||
),
|
||
React.createElement("div", {className: "panel-footer"},
|
||
React.createElement(MessageBar, null)
|
||
)
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
var updateMessageCallbacks = [];
|
||
|
||
var timer = setInterval(function () {
|
||
updateMessageCallbacks.forEach(function (callback) {
|
||
callback();
|
||
});
|
||
}, 60000);
|
||
|
||
var ChatMessage = React.createClass({displayName: "ChatMessage",
|
||
componentDidMount: function () {
|
||
var self = this;
|
||
updateMessageCallbacks.push(function () {
|
||
self.forceUpdate();
|
||
});
|
||
},
|
||
render: function () {
|
||
return (
|
||
React.createElement("li", {className: "left clearfix"},
|
||
React.createElement("span", {className: "chat-img pull-left"},
|
||
React.createElement("img", {
|
||
src: this.props.message.author.avatar,
|
||
alt: "User Avatar",
|
||
height: "40",
|
||
width: "40",
|
||
className: "img-circle"})
|
||
),
|
||
React.createElement("div", {className: "chat-body clearfix"},
|
||
React.createElement("div", {className: "header"},
|
||
React.createElement("strong", {className: "primary-font"}, this.props.message.author.username),
|
||
React.createElement("small", {className: "pull-right text-muted"},
|
||
React.createElement("i", {className: "fa fa-clock-o fa-fw"}), " ", $.timeago(this.props.message.createdAt)
|
||
)
|
||
),
|
||
React.createElement("p", null, this.props.message.content)
|
||
)
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
var MessageBar = React.createClass({displayName: "MessageBar",
|
||
sendMessage: function (content) {
|
||
socket.emit("message:new", {
|
||
content: content
|
||
});
|
||
},
|
||
handleSubmit: function (e) {
|
||
e.preventDefault();
|
||
var content = React.findDOMNode(this.refs.content).value.trim();
|
||
if (!content) return;
|
||
React.findDOMNode(this.refs.content).value = '';
|
||
this.sendMessage(content);
|
||
return;
|
||
},
|
||
render: function () {
|
||
return (
|
||
React.createElement("form", {onSubmit: this.handleSubmit},
|
||
React.createElement("div", {className: "input-group"},
|
||
React.createElement("input", {
|
||
id: "btn-input",
|
||
type: "text",
|
||
className: "form-control",
|
||
ref: "content",
|
||
placeholder: "Be polite please..."}),
|
||
React.createElement("span", {className: "input-group-btn"},
|
||
React.createElement("input", {
|
||
type: "submit",
|
||
className: "btn btn-primary",
|
||
id: "btn-chat",
|
||
value: "Send"})
|
||
)
|
||
)
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
"use strict";
|
||
|
||
var UserLogin = React.createClass({displayName: "UserLogin",
|
||
authorizeId: function (id) {
|
||
socket.emit("users:authorize", {
|
||
id: parseInt(id, 10)
|
||
});
|
||
setTimeout(function () {
|
||
socket.emit("gather:refresh");
|
||
}, 1000);
|
||
},
|
||
handleSubmit: function (e) {
|
||
e.preventDefault();
|
||
var id = React.findDOMNode(this.refs.authorize_id).value.trim();
|
||
if (!id) return;
|
||
React.findDOMNode(this.refs.authorize_id).value = '';
|
||
this.authorizeId(id);
|
||
},
|
||
render: function () {
|
||
return (
|
||
React.createElement("form", {onSubmit: this.handleSubmit},
|
||
React.createElement("div", {className: "input-group signin"},
|
||
React.createElement("input", {
|
||
id: "btn-input",
|
||
type: "text",
|
||
className: "form-control",
|
||
ref: "authorize_id",
|
||
placeholder: "Choose an ID..."}),
|
||
React.createElement("span", {className: "input-group-btn"},
|
||
React.createElement("input", {
|
||
type: "submit",
|
||
className: "btn btn-primary",
|
||
id: "btn-chat",
|
||
value: "Login"})
|
||
)
|
||
),
|
||
React.createElement("div", {className: "signin"},
|
||
React.createElement("p", {className: "text-center"}, React.createElement("small", null, "Just a temporary measure until genuine authentication is implemented"))
|
||
)
|
||
)
|
||
);
|
||
}
|
||
})
|
||
|
||
var UserMenu = React.createClass({displayName: "UserMenu",
|
||
getDefaultProps: function () {
|
||
return {
|
||
users: []
|
||
};
|
||
},
|
||
componentDidMount: function () {
|
||
var self = this;
|
||
socket.on('users:update', function (data) {
|
||
self.setProps({
|
||
users: data.users
|
||
});
|
||
});
|
||
},
|
||
render: function () {
|
||
var users = this.props.users.map(function (user) {
|
||
return (
|
||
React.createElement("li", {key: user.id}, React.createElement("a", {href: "#"}, user.username))
|
||
);
|
||
});
|
||
return (
|
||
React.createElement("ul", {className: "nav", id: "side-menu"},
|
||
React.createElement("li", null,
|
||
React.createElement("a", {href: "#"},
|
||
React.createElement("i", {className: "fa fa-users fa-fw"}), " Online",
|
||
React.createElement("span", {className: "badge add-left"}, " ", this.props.users.length, " ")
|
||
)
|
||
),
|
||
users,
|
||
React.createElement("li", null, React.createElement("br", null), React.createElement(UserLogin, null), React.createElement("br", null))
|
||
)
|
||
);
|
||
}
|
||
});
|
||
|
||
var AdminPanel = React.createClass({displayName: "AdminPanel",
|
||
handleGatherReset: function () {
|
||
socket.emit("gather:reset");
|
||
},
|
||
render: function () {
|
||
return (
|
||
React.createElement("ul", {className: "nav", id: "admin-menu"},
|
||
React.createElement("li", null,
|
||
React.createElement("div", {className: "admin-panel"},
|
||
React.createElement("h5", null, "Admin"),
|
||
React.createElement("button", {
|
||
className: "btn btn-danger max-width",
|
||
onClick: this.handleGatherReset},
|
||
"Reset Gather"),
|
||
React.createElement("p", {className: "text-center add-top"}, React.createElement("small", null, "Only responds for admins on staging.ensl.org"))
|
||
)
|
||
)
|
||
)
|
||
)
|
||
}
|
||
});
|
||
|
||
var CurrentUser = React.createClass({displayName: "CurrentUser",
|
||
componentDidMount: function () {
|
||
var self = this;
|
||
socket.on("users:update", function (data) {
|
||
self.setProps({
|
||
user: data.currentUser
|
||
});
|
||
});
|
||
socket.emit("users:refresh");
|
||
},
|
||
render: function () {
|
||
if (this.props.user) {
|
||
return (
|
||
React.createElement("li", {className: "dropdown"},
|
||
React.createElement("a", {className: "dropdown-toggle", "data-toggle": "dropdown", href: "#"},
|
||
this.props.user.username, " ", React.createElement("img", {src: this.props.user.avatar,
|
||
alt: "User Avatar",
|
||
height: "20",
|
||
width: "20"}), " ", React.createElement("i", {className: "fa fa-caret-down"})
|
||
),
|
||
React.createElement("ul", {className: "dropdown-menu dropdown-user"},
|
||
React.createElement("li", null,
|
||
React.createElement("a", {href: "#"}, React.createElement("i", {className: "fa fa-gear fa-fw"}), " Profile")
|
||
),
|
||
React.createElement("li", null,
|
||
React.createElement("a", {href: "#"}, React.createElement("i", {className: "fa fa-flag fa-fw"}), " Notifications")
|
||
),
|
||
React.createElement("li", null,
|
||
React.createElement("a", {href: "#"}, React.createElement("i", {className: "fa fa-music fa-fw"}), " Sounds")
|
||
),
|
||
React.createElement("li", null,
|
||
React.createElement("a", {href: "#", "data-toggle": "modal", "data-target": "#designmodal"}, "Design Goals")
|
||
),
|
||
React.createElement("li", {className: "divider"}),
|
||
React.createElement("li", null, React.createElement("a", {href: "login.html"}, React.createElement("i", {className: "fa fa-sign-out fa-fw"}), " Logout")
|
||
)
|
||
)
|
||
)
|
||
|
||
);
|
||
} else {
|
||
return false;
|
||
}
|
||
}
|
||
});
|