diff --git a/gulpfile.js b/gulpfile.js
new file mode 100644
index 0000000..ad0e1db
--- /dev/null
+++ b/gulpfile.js
@@ -0,0 +1,21 @@
+"use strict";
+
+var gulp = require("gulp");
+var react = require("gulp-react");
+var concat = require("gulp-concat");
+var watch = require("gulp-watch");
+var plumber = require("gulp-plumber");
+
+gulp.task('default', ['compile']);
+
+gulp.task('compile', function () {
+ return gulp.src('lib/react/**')
+ .pipe(plumber())
+ .pipe(concat('app.js'))
+ .pipe(react())
+ .pipe(gulp.dest('public/js/'));
+});
+
+gulp.task('watch', function () {
+ gulp.watch('lib/react/**', ['compile']);
+});
\ No newline at end of file
diff --git a/lib/gather/controller.js b/lib/gather/controller.js
index 3496d97..f126cbb 100644
--- a/lib/gather/controller.js
+++ b/lib/gather/controller.js
@@ -53,7 +53,11 @@ async.parallel(instructions, function (error) {
console.log("Error while adding gatherers", error);
} else {
console.log("Loaded gatherers");
-
+ gather.gatherers.forEach(function (gatherer, index, array) {
+ var candidate = Math.floor(Math.random() * array.length);
+ array[index].leaderVote = array[candidate].id;
+ });
+ console.log("Assigned vote for each gatherer");
}
});
@@ -65,7 +69,7 @@ module.exports = function (namespace) {
namespace.sockets.forEach(function (socket) {
socket.emit("gather:refresh", {
gather: gather.toJson(),
- currentUser: socket._user
+ currentGatherer: gather.getGatherer(socket._user)
});
});
};
diff --git a/lib/gather/gather.js b/lib/gather/gather.js
index 637f26e..e08161b 100644
--- a/lib/gather/gather.js
+++ b/lib/gather/gather.js
@@ -164,6 +164,14 @@ Gather.prototype.voteForLeader = function (voter, candidate) {
});
};
+Gather.prototype.getGatherer = function (user) {
+ var matchingGatherer = null;
+ this.gatherers.forEach(function (gatherer) {
+ if (gatherer.id === user.id) matchingGatherer = gatherer;
+ });
+ return matchingGatherer;
+};
+
// Gather States
// - Gathering
// - Election
diff --git a/lib/react/app.jsx b/lib/react/app.jsx
deleted file mode 100644
index fa35ebb..0000000
--- a/lib/react/app.jsx
+++ /dev/null
@@ -1,573 +0,0 @@
-$(function () {
-
-"use strict";
-
-var UserCounter = React.createClass({
- render: function () {
- return (
-
-
- Online
- {this.props.count}
-
-
- );
- }
-});
-
-var UserLogin = React.createClass({
- authorizeId: function (id) {
- id = parseInt(id, 10);
- socket.emit("users:authorize", {
- id: id
- });
- },
- 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);
- return;
- },
- render: function () {
- return (
-
- );
- }
-})
-
-var UserMenu = React.createClass({
- getDefaultProps: function () {
- return {
- count: 0,
- users: []
- };
- },
- componentDidMount: function () {
- socket.on('users:update', this.updateUsers);
- },
- updateUsers: function (data) {
- this.setProps({
- count: data.count,
- users: data.users
- });
- },
- render: function () {
- var users = this.props.users.map(function (user) {
- return (
- {user.username}
- );
- });
- return (
-
- );
- }
-});
-
-var Chatroom = React.createClass({
- getDefaultProps: function () {
- return {
- history: []
- };
- },
- componentDidMount: function () {
- var self = this;
- var TIMER_INTERVAL = 60000; // 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 () {
- if (self.refs.messages) self.refs.messages.refreshTime();
- }, 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 (
-
- );
- });
- return (
-
- );
- }
-});
-
-var ChatMessage = React.createClass({
- getInitialState: function () {
- return {
- timeAgo: $.timeago(this.props.createdAt)
- }
- },
- refreshTime: function () {
- var self = this;
- self.setState({
- timeAgo: $.timeago(self.props.createdAt)
- });
- },
- render: function () {
- return (
-
-
-
-
-
-
- {this.props.username}
-
- {this.state.timeAgo}
-
-
-
{this.props.content}
-
-
- );
- }
-});
-
-var CurrentUser = React.createClass({
- 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 (
- {this.props.user.username}
- );
- } else {
- return false;
- }
- }
-});
-
-var VoteButton = React.createClass({
- 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 (
-
- );
- } else {
- return (
-
- );
- }
- }
-});
-
-var MessageBar = React.createClass({
- 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 (
-
- );
- }
-});
-
-var JoinGatherButton = React.createClass({
- joinGather: function (e) {
- e.preventDefault();
- socket.emit("gather:join", {});
- },
- render: function () {
- var message = this.props.buttonName || "Join Gather";
- var buttonClass = "btn btn-primary";
- if (this.props.buttonClass) {
- buttonClass += " " + this.props.buttonClass;
- }
- return ()
- }
-});
-
-var GatherProgress = React.createClass({
- gatheringProgress: function () {
- var num = this.props.gather.gatherers.length;
- var den = 12;
- return {
- num: num,
- den: den,
- message: num + " / " + den
- };
- },
- 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;
- var gatherState = this.props.gather.state;
- if (gatherState === 'gathering' && this.props.gather.gatherers.length) {
- progress = this.gatheringProgress();
- } else if (gatherState === 'election') {
- progress = this.electionProgress();
- } else if (gatherState === 'selection') {
- progress = this.selectionProgress();
- }
- if (progress) {
- var style = {
- width: Math.round((progress.num / progress.den * 100)) + "%"
- };
- return (
-
-
Gather Progress
-
-
- {progress.message}
-
-
-
- );
- } else {
- return false;
- }
- }
-});
-
-var Gather = React.createClass({
- getDefaultProps: function () {
- return {
- gather: {
- gatherers: []
- }
- }
- },
- joinedGather: function () {
- var self = this;
- return this.props.gather.gatherers.some(function (gatherer) {
- return gatherer.user.id === self.props.currentUser.id;
- });
- },
- componentDidMount: function () {
- var self = this;
- socket.on("gather:refresh", function (data) {
- self.setProps({
- gather: data.gather,
- currentUser: data.currentUser
- });
- });
- },
- 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 picking teams";
- case "done":
- return "Gather completed";
- default:
- return "Initialising gather";
- }
- },
- leaveGather: function (e) {
- e.preventDefault();
- socket.emit("gather:leave", {});
- },
- inviteToGather: function (e) {
- e.preventDefault();
- },
- currentGatherer: function () {
- var current = null;
- var self = this;
- this.props.gather.gatherers.forEach(function (gatherer) {
- if (gatherer.id === self.props.currentUser.id) current = gatherer;
- });
- return current;
- },
- render: function () {
- var joinButton;
- if (this.joinedGather()) {
- joinButton = ();
- } else {
- joinButton = ();
- }
- var inviteButton;
- if (this.props.gather.state === 'gathering') {
- inviteButton = ();
- }
- return (
-
-
- NS2 Gather
- {this.props.gather.gatherers.length}
-
- {this.stateDescription()}
-
-
-
-
-
- {inviteButton}
- {joinButton}
-
-
-
- );
- }
-});
-
-var LeaderPoll = React.createClass({
- render: function () {
- return (
-
-
- );
- }
-});
-
-var Gatherers = React.createClass({
- render: function () {
- var self = this;
- var gatherers = this.props.gather.gatherers.map(function (gatherer) {
- var lifeforms = (
- gatherer.user.ability.lifeforms.map(function (lifeform) {
- return ({lifeform});
- })
- );
-
- var commBadge;
- if (gatherer.user.ability.commander) {
- commBadge = ();
- }
-
- var division = ({gatherer.user.ability.division});
- var action = lifeforms;
- 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 = (
-
- {votes + " votes"}
-
-
-
- );
- }
-
- return (
-
- {commBadge} |
- {gatherer.user.username} |
- {division} |
- {action} |
-
- );
- })
- if (this.props.gather.gatherers.length) {
- return (
-
- );
- } else {
- return (
);
- }
- }
-});
-
-var socket;
-
-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")
- });
-
- React.render(, document.getElementById('side-menu'));
- React.render(, document.getElementById('chatroom'));
- React.render(, document.getElementById('gathers'));
- React.render(, document.getElementById('currentuser'));
-};
-
-initialiseComponents();
-
-
-
-});
-
diff --git a/lib/react/gather.jsx b/lib/react/gather.jsx
new file mode 100644
index 0000000..4e3ab91
--- /dev/null
+++ b/lib/react/gather.jsx
@@ -0,0 +1,354 @@
+"use strict";
+
+var VoteButton = React.createClass({
+ 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 (
+
+ );
+ } else {
+ return (
+
+ );
+ }
+ }
+});
+
+var JoinGatherButton = React.createClass({
+ joinGather: function (e) {
+ e.preventDefault();
+ socket.emit("gather:join", {});
+ },
+ render: function () {
+ var message = this.props.buttonName || "Join Gather";
+ var buttonClass = "btn btn-primary";
+ if (this.props.buttonClass) {
+ buttonClass += " " + this.props.buttonClass;
+ }
+ return ()
+ }
+});
+
+var SelectPlayerButton = React.createClass({
+ selectPlayer: function (e) {
+ e.preventDefault();
+ },
+ render: function () {
+ if (!this.props.currentGatherer.leader) {
+ return false;
+ } else {
+ return (
+ );
+ }
+ }
+})
+
+var GatherTeams = React.createClass({
+ alienGatherers: function () {
+ return this.props.gather.gatherers.filter(function (gatherer) {
+ return gatherer.team === "alien";
+ }).sort(function (gatherer) {
+ return (gatherer.leader) ? 1 : 0;
+ });
+ },
+ marineGatherers: function () {
+ return this.props.gather.gatherers.filter(function (gatherer) {
+ return gatherer.team === "marine";
+ }).sort(function (gatherer) {
+ return (gatherer.leader) ? 1 : 0;
+ });
+ },
+ render: function () {
+ var extractGatherer = function (gatherer) {
+ var image;
+ if (gatherer.leader) {
+ image = ();
+ }
+ return (
+
+ {image} |
+ {gatherer.user.username} |
+
+ );
+ }
+ var marines = this.marineGatherers().map(extractGatherer);
+ var aliens = this.alienGatherers().map(extractGatherer);
+ return (
+
+ );
+ }
+})
+
+var GatherProgress = React.createClass({
+ 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 picking 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;
+ var gatherState = this.props.gather.state;
+ if (gatherState === 'gathering' && this.props.gather.gatherers.length) {
+ progress = this.gatheringProgress();
+ } else if (gatherState === 'election') {
+ progress = this.electionProgress();
+ } else if (gatherState === 'selection') {
+ progress = this.selectionProgress();
+ }
+ if (progress) {
+ var style = {
+ width: Math.round((progress.num / progress.den * 100)) + "%"
+ };
+ return (
+
+
{this.stateDescription()} {progress.message}
+
+
+ );
+ } else {
+ return false;
+ }
+ }
+});
+
+var Gather = React.createClass({
+ getDefaultProps: function () {
+ return {
+ gather: {
+ gatherers: []
+ }
+ }
+ },
+ joinedGather: function () {
+ return this.props.currentGatherer !== null;
+ },
+ componentDidMount: function () {
+ var self = this;
+ socket.on("gather:refresh", function (data) {
+ self.setProps({
+ gather: data.gather,
+ currentGatherer: data.currentGatherer
+ });
+ });
+ },
+ leaveGather: function (e) {
+ e.preventDefault();
+ socket.emit("gather:leave", {});
+ },
+ inviteToGather: function (e) {
+ e.preventDefault();
+ },
+ render: function () {
+ var joinButton;
+ if (this.joinedGather()) {
+ joinButton = ();
+ } else {
+ joinButton = ();
+ }
+ var inviteButton;
+ if (this.props.gather.state === 'gathering') {
+ inviteButton = ();
+ }
+ var gatherTeams;
+ if (this.props.gather.state === 'selection') {
+ gatherTeams =
+ }
+ return (
+
+
+ NS2 Gather
+ {this.props.gather.gatherers.length}
+
+
+ {gatherTeams}
+
+
+
+ {inviteButton}
+ {joinButton}
+
+
+
+ );
+ }
+});
+
+var Gatherers = React.createClass({
+ render: function () {
+ var self = this;
+ var gatherers = this.props.gather.gatherers.map(function (gatherer) {
+ var lifeforms = (
+ gatherer.user.ability.lifeforms.map(function (lifeform) {
+ return ({lifeform});
+ })
+ );
+
+ // Switch this to online status
+ var online= ( );
+
+ var division = ({gatherer.user.ability.division});
+ var action = lifeforms;
+ 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 = (
+
+ {votes + " votes"}
+
+
+ );
+ }
+
+ return (
+
+ {online} |
+ {gatherer.user.username} |
+ {division} |
+ {action} |
+
+ );
+ })
+ if (this.props.gather.gatherers.length) {
+ return (
+
+ );
+ } else {
+ return (
);
+ }
+ }
+});
+
+
+
diff --git a/lib/react/init.jsx b/lib/react/init.jsx
new file mode 100644
index 0000000..dfbbd5c
--- /dev/null
+++ b/lib/react/init.jsx
@@ -0,0 +1,22 @@
+"use strict";
+
+var socket;
+
+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")
+ });
+
+ React.render(, document.getElementById('side-menu'));
+ React.render(, document.getElementById('chatroom'));
+ React.render(, document.getElementById('gathers'));
+ React.render(, document.getElementById('currentuser'));
+};
\ No newline at end of file
diff --git a/lib/react/message.jsx b/lib/react/message.jsx
new file mode 100644
index 0000000..503a97d
--- /dev/null
+++ b/lib/react/message.jsx
@@ -0,0 +1,146 @@
+"use strict";
+
+var Chatroom = React.createClass({
+ getDefaultProps: function () {
+ return {
+ history: []
+ };
+ },
+ componentDidMount: function () {
+ var self = this;
+ var TIMER_INTERVAL = 60000; // 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 () {
+ if (self.refs.messages) self.refs.messages.refreshTime();
+ }, 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 (
+
+ );
+ });
+ return (
+
+ );
+ }
+});
+
+var ChatMessage = React.createClass({
+ getInitialState: function () {
+ return {
+ timeAgo: $.timeago(this.props.createdAt)
+ }
+ },
+ refreshTime: function () {
+ var self = this;
+ self.setState({
+ timeAgo: $.timeago(self.props.createdAt)
+ });
+ },
+ render: function () {
+ return (
+
+
+
+
+
+
+ {this.props.username}
+
+ {this.state.timeAgo}
+
+
+
{this.props.content}
+
+
+ );
+ }
+});
+
+var MessageBar = React.createClass({
+ 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 (
+
+ );
+ }
+});
diff --git a/lib/react/user.jsx b/lib/react/user.jsx
new file mode 100644
index 0000000..9c6fe65
--- /dev/null
+++ b/lib/react/user.jsx
@@ -0,0 +1,111 @@
+"use strict";
+
+var UserCounter = React.createClass({
+ render: function () {
+ return (
+
+
+ Online
+ {this.props.count}
+
+
+ );
+ }
+});
+
+var UserLogin = React.createClass({
+ authorizeId: function (id) {
+ id = parseInt(id, 10);
+ socket.emit("users:authorize", {
+ id: id
+ });
+ },
+ 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);
+ return;
+ },
+ render: function () {
+ return (
+
+ );
+ }
+})
+
+var UserMenu = React.createClass({
+ getDefaultProps: function () {
+ return {
+ count: 0,
+ users: []
+ };
+ },
+ componentDidMount: function () {
+ socket.on('users:update', this.updateUsers);
+ },
+ updateUsers: function (data) {
+ this.setProps({
+ count: data.count,
+ users: data.users
+ });
+ },
+ render: function () {
+ var users = this.props.users.map(function (user) {
+ return (
+ {user.username}
+ );
+ });
+ return (
+
+ );
+ }
+});
+
+var CurrentUser = React.createClass({
+ 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 (
+ {this.props.user.username}
+ );
+ } else {
+ return false;
+ }
+ }
+});
diff --git a/npm-shrinkwrap.json b/npm-shrinkwrap.json
index 73e051f..c08b804 100644
--- a/npm-shrinkwrap.json
+++ b/npm-shrinkwrap.json
@@ -2,6 +2,11 @@
"name": "sws_gathers",
"version": "1.0.0",
"dependencies": {
+ "async": {
+ "version": "1.4.0",
+ "from": "async@",
+ "resolved": "https://registry.npmjs.org/async/-/async-1.4.0.tgz"
+ },
"express": {
"version": "4.13.1",
"from": "express@",
@@ -405,6 +410,781 @@
"from": "extend@",
"resolved": "https://registry.npmjs.org/extend/-/extend-3.0.0.tgz"
},
+ "gulp": {
+ "version": "3.9.0",
+ "from": "gulp@",
+ "resolved": "https://registry.npmjs.org/gulp/-/gulp-3.9.0.tgz",
+ "dependencies": {
+ "archy": {
+ "version": "1.0.0",
+ "from": "archy@^1.0.0",
+ "resolved": "https://registry.npmjs.org/archy/-/archy-1.0.0.tgz"
+ },
+ "chalk": {
+ "version": "1.1.0",
+ "from": "chalk@^1.0.0",
+ "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.0.tgz",
+ "dependencies": {
+ "ansi-styles": {
+ "version": "2.1.0",
+ "from": "ansi-styles@^2.1.0",
+ "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.1.0.tgz"
+ },
+ "escape-string-regexp": {
+ "version": "1.0.3",
+ "from": "escape-string-regexp@^1.0.2",
+ "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.3.tgz"
+ },
+ "has-ansi": {
+ "version": "2.0.0",
+ "from": "has-ansi@^2.0.0",
+ "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz",
+ "dependencies": {
+ "ansi-regex": {
+ "version": "2.0.0",
+ "from": "ansi-regex@^2.0.0",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.0.0.tgz"
+ }
+ }
+ },
+ "strip-ansi": {
+ "version": "3.0.0",
+ "from": "strip-ansi@^3.0.0",
+ "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.0.tgz",
+ "dependencies": {
+ "ansi-regex": {
+ "version": "2.0.0",
+ "from": "ansi-regex@^2.0.0",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.0.0.tgz"
+ }
+ }
+ },
+ "supports-color": {
+ "version": "2.0.0",
+ "from": "supports-color@^2.0.0",
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz"
+ }
+ }
+ },
+ "deprecated": {
+ "version": "0.0.1",
+ "from": "deprecated@^0.0.1",
+ "resolved": "https://registry.npmjs.org/deprecated/-/deprecated-0.0.1.tgz"
+ },
+ "gulp-util": {
+ "version": "3.0.6",
+ "from": "gulp-util@^3.0.0",
+ "resolved": "https://registry.npmjs.org/gulp-util/-/gulp-util-3.0.6.tgz",
+ "dependencies": {
+ "array-differ": {
+ "version": "1.0.0",
+ "from": "array-differ@^1.0.0",
+ "resolved": "https://registry.npmjs.org/array-differ/-/array-differ-1.0.0.tgz"
+ },
+ "array-uniq": {
+ "version": "1.0.2",
+ "from": "array-uniq@^1.0.2",
+ "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.2.tgz"
+ },
+ "beeper": {
+ "version": "1.1.0",
+ "from": "beeper@^1.0.0",
+ "resolved": "https://registry.npmjs.org/beeper/-/beeper-1.1.0.tgz"
+ },
+ "dateformat": {
+ "version": "1.0.11",
+ "from": "dateformat@^1.0.11",
+ "resolved": "https://registry.npmjs.org/dateformat/-/dateformat-1.0.11.tgz",
+ "dependencies": {
+ "get-stdin": {
+ "version": "4.0.1",
+ "from": "get-stdin@*",
+ "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-4.0.1.tgz"
+ },
+ "meow": {
+ "version": "3.3.0",
+ "from": "meow@*",
+ "resolved": "https://registry.npmjs.org/meow/-/meow-3.3.0.tgz",
+ "dependencies": {
+ "camelcase-keys": {
+ "version": "1.0.0",
+ "from": "camelcase-keys@^1.0.0",
+ "resolved": "https://registry.npmjs.org/camelcase-keys/-/camelcase-keys-1.0.0.tgz",
+ "dependencies": {
+ "camelcase": {
+ "version": "1.1.0",
+ "from": "camelcase@^1.0.1",
+ "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-1.1.0.tgz"
+ },
+ "map-obj": {
+ "version": "1.0.1",
+ "from": "map-obj@^1.0.0",
+ "resolved": "https://registry.npmjs.org/map-obj/-/map-obj-1.0.1.tgz"
+ }
+ }
+ },
+ "indent-string": {
+ "version": "1.2.2",
+ "from": "indent-string@^1.1.0",
+ "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-1.2.2.tgz",
+ "dependencies": {
+ "repeating": {
+ "version": "1.1.3",
+ "from": "repeating@^1.1.0",
+ "resolved": "https://registry.npmjs.org/repeating/-/repeating-1.1.3.tgz",
+ "dependencies": {
+ "is-finite": {
+ "version": "1.0.1",
+ "from": "is-finite@^1.0.0",
+ "resolved": "https://registry.npmjs.org/is-finite/-/is-finite-1.0.1.tgz",
+ "dependencies": {
+ "number-is-nan": {
+ "version": "1.0.0",
+ "from": "number-is-nan@^1.0.0",
+ "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.0.tgz"
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ },
+ "lodash._reescape": {
+ "version": "3.0.0",
+ "from": "lodash._reescape@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash._reescape/-/lodash._reescape-3.0.0.tgz"
+ },
+ "lodash._reevaluate": {
+ "version": "3.0.0",
+ "from": "lodash._reevaluate@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash._reevaluate/-/lodash._reevaluate-3.0.0.tgz"
+ },
+ "lodash._reinterpolate": {
+ "version": "3.0.0",
+ "from": "lodash._reinterpolate@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash._reinterpolate/-/lodash._reinterpolate-3.0.0.tgz"
+ },
+ "lodash.template": {
+ "version": "3.6.2",
+ "from": "lodash.template@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash.template/-/lodash.template-3.6.2.tgz",
+ "dependencies": {
+ "lodash._basecopy": {
+ "version": "3.0.1",
+ "from": "lodash._basecopy@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash._basecopy/-/lodash._basecopy-3.0.1.tgz"
+ },
+ "lodash._basetostring": {
+ "version": "3.0.1",
+ "from": "lodash._basetostring@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash._basetostring/-/lodash._basetostring-3.0.1.tgz"
+ },
+ "lodash._basevalues": {
+ "version": "3.0.0",
+ "from": "lodash._basevalues@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash._basevalues/-/lodash._basevalues-3.0.0.tgz"
+ },
+ "lodash._isiterateecall": {
+ "version": "3.0.9",
+ "from": "lodash._isiterateecall@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash._isiterateecall/-/lodash._isiterateecall-3.0.9.tgz"
+ },
+ "lodash.escape": {
+ "version": "3.0.0",
+ "from": "lodash.escape@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash.escape/-/lodash.escape-3.0.0.tgz"
+ },
+ "lodash.keys": {
+ "version": "3.1.2",
+ "from": "lodash.keys@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash.keys/-/lodash.keys-3.1.2.tgz",
+ "dependencies": {
+ "lodash._getnative": {
+ "version": "3.9.1",
+ "from": "lodash._getnative@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash._getnative/-/lodash._getnative-3.9.1.tgz"
+ },
+ "lodash.isarguments": {
+ "version": "3.0.4",
+ "from": "lodash.isarguments@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash.isarguments/-/lodash.isarguments-3.0.4.tgz"
+ },
+ "lodash.isarray": {
+ "version": "3.0.4",
+ "from": "lodash.isarray@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash.isarray/-/lodash.isarray-3.0.4.tgz"
+ }
+ }
+ },
+ "lodash.restparam": {
+ "version": "3.6.1",
+ "from": "lodash.restparam@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash.restparam/-/lodash.restparam-3.6.1.tgz"
+ },
+ "lodash.templatesettings": {
+ "version": "3.1.0",
+ "from": "lodash.templatesettings@^3.0.0",
+ "resolved": "https://registry.npmjs.org/lodash.templatesettings/-/lodash.templatesettings-3.1.0.tgz"
+ }
+ }
+ },
+ "multipipe": {
+ "version": "0.1.2",
+ "from": "multipipe@^0.1.2",
+ "resolved": "https://registry.npmjs.org/multipipe/-/multipipe-0.1.2.tgz",
+ "dependencies": {
+ "duplexer2": {
+ "version": "0.0.2",
+ "from": "duplexer2@0.0.2",
+ "resolved": "https://registry.npmjs.org/duplexer2/-/duplexer2-0.0.2.tgz",
+ "dependencies": {
+ "readable-stream": {
+ "version": "1.1.13",
+ "from": "readable-stream@~1.1.9",
+ "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.13.tgz",
+ "dependencies": {
+ "core-util-is": {
+ "version": "1.0.1",
+ "from": "core-util-is@~1.0.0",
+ "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.1.tgz"
+ },
+ "isarray": {
+ "version": "0.0.1",
+ "from": "isarray@0.0.1",
+ "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz"
+ },
+ "string_decoder": {
+ "version": "0.10.31",
+ "from": "string_decoder@~0.10.x",
+ "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz"
+ },
+ "inherits": {
+ "version": "2.0.1",
+ "from": "inherits@~2.0.1",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz"
+ }
+ }
+ }
+ }
+ }
+ }
+ },
+ "object-assign": {
+ "version": "3.0.0",
+ "from": "object-assign@^3.0.0",
+ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-3.0.0.tgz"
+ },
+ "replace-ext": {
+ "version": "0.0.1",
+ "from": "replace-ext@0.0.1",
+ "resolved": "https://registry.npmjs.org/replace-ext/-/replace-ext-0.0.1.tgz"
+ },
+ "through2": {
+ "version": "2.0.0",
+ "from": "through2@^2.0.0",
+ "resolved": "https://registry.npmjs.org/through2/-/through2-2.0.0.tgz",
+ "dependencies": {
+ "readable-stream": {
+ "version": "2.0.2",
+ "from": "readable-stream@~2.0.0",
+ "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.0.2.tgz",
+ "dependencies": {
+ "core-util-is": {
+ "version": "1.0.1",
+ "from": "core-util-is@~1.0.0",
+ "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.1.tgz"
+ },
+ "inherits": {
+ "version": "2.0.1",
+ "from": "inherits@~2.0.1",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz"
+ },
+ "isarray": {
+ "version": "0.0.1",
+ "from": "isarray@0.0.1",
+ "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz"
+ },
+ "process-nextick-args": {
+ "version": "1.0.2",
+ "from": "process-nextick-args@~1.0.0",
+ "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-1.0.2.tgz"
+ },
+ "string_decoder": {
+ "version": "0.10.31",
+ "from": "string_decoder@~0.10.x",
+ "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz"
+ },
+ "util-deprecate": {
+ "version": "1.0.1",
+ "from": "util-deprecate@~1.0.1",
+ "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.1.tgz"
+ }
+ }
+ },
+ "xtend": {
+ "version": "4.0.0",
+ "from": "xtend@>=4.0.0 <4.1.0-0",
+ "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.0.tgz"
+ }
+ }
+ },
+ "vinyl": {
+ "version": "0.5.0",
+ "from": "vinyl@^0.5.0",
+ "resolved": "https://registry.npmjs.org/vinyl/-/vinyl-0.5.0.tgz",
+ "dependencies": {
+ "clone": {
+ "version": "1.0.2",
+ "from": "clone@^1.0.0",
+ "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.2.tgz"
+ },
+ "clone-stats": {
+ "version": "0.0.1",
+ "from": "clone-stats@^0.0.1",
+ "resolved": "https://registry.npmjs.org/clone-stats/-/clone-stats-0.0.1.tgz"
+ }
+ }
+ }
+ }
+ },
+ "interpret": {
+ "version": "0.6.5",
+ "from": "interpret@^0.6.2",
+ "resolved": "https://registry.npmjs.org/interpret/-/interpret-0.6.5.tgz"
+ },
+ "liftoff": {
+ "version": "2.1.0",
+ "from": "liftoff@^2.1.0",
+ "resolved": "https://registry.npmjs.org/liftoff/-/liftoff-2.1.0.tgz",
+ "dependencies": {
+ "extend": {
+ "version": "2.0.1",
+ "from": "extend@^2.0.1",
+ "resolved": "https://registry.npmjs.org/extend/-/extend-2.0.1.tgz"
+ },
+ "findup-sync": {
+ "version": "0.2.1",
+ "from": "findup-sync@^0.2.1",
+ "resolved": "https://registry.npmjs.org/findup-sync/-/findup-sync-0.2.1.tgz",
+ "dependencies": {
+ "glob": {
+ "version": "4.3.5",
+ "from": "glob@~4.3.0",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-4.3.5.tgz",
+ "dependencies": {
+ "inflight": {
+ "version": "1.0.4",
+ "from": "inflight@^1.0.4",
+ "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.4.tgz",
+ "dependencies": {
+ "wrappy": {
+ "version": "1.0.1",
+ "from": "wrappy@1",
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.1.tgz"
+ }
+ }
+ },
+ "inherits": {
+ "version": "2.0.1",
+ "from": "inherits@2",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz"
+ },
+ "minimatch": {
+ "version": "2.0.10",
+ "from": "minimatch@^2.0.1",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-2.0.10.tgz",
+ "dependencies": {
+ "brace-expansion": {
+ "version": "1.1.0",
+ "from": "brace-expansion@^1.0.0",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.0.tgz",
+ "dependencies": {
+ "balanced-match": {
+ "version": "0.2.0",
+ "from": "balanced-match@^0.2.0",
+ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-0.2.0.tgz"
+ },
+ "concat-map": {
+ "version": "0.0.1",
+ "from": "concat-map@0.0.1",
+ "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz"
+ }
+ }
+ }
+ }
+ },
+ "once": {
+ "version": "1.3.2",
+ "from": "once@^1.3.0",
+ "resolved": "https://registry.npmjs.org/once/-/once-1.3.2.tgz",
+ "dependencies": {
+ "wrappy": {
+ "version": "1.0.1",
+ "from": "wrappy@1",
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.1.tgz"
+ }
+ }
+ }
+ }
+ }
+ }
+ },
+ "flagged-respawn": {
+ "version": "0.3.1",
+ "from": "flagged-respawn@^0.3.1",
+ "resolved": "https://registry.npmjs.org/flagged-respawn/-/flagged-respawn-0.3.1.tgz"
+ },
+ "rechoir": {
+ "version": "0.6.2",
+ "from": "rechoir@^0.6.0",
+ "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.6.2.tgz"
+ },
+ "resolve": {
+ "version": "1.1.6",
+ "from": "resolve@^1.1.6",
+ "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.1.6.tgz"
+ }
+ }
+ },
+ "minimist": {
+ "version": "1.1.2",
+ "from": "minimist@^1.1.0",
+ "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.1.2.tgz"
+ },
+ "orchestrator": {
+ "version": "0.3.7",
+ "from": "orchestrator@^0.3.0",
+ "resolved": "https://registry.npmjs.org/orchestrator/-/orchestrator-0.3.7.tgz",
+ "dependencies": {
+ "end-of-stream": {
+ "version": "0.1.5",
+ "from": "end-of-stream@~0.1.5",
+ "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-0.1.5.tgz",
+ "dependencies": {
+ "once": {
+ "version": "1.3.2",
+ "from": "once@~1.3.0",
+ "resolved": "https://registry.npmjs.org/once/-/once-1.3.2.tgz",
+ "dependencies": {
+ "wrappy": {
+ "version": "1.0.1",
+ "from": "wrappy@1",
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.1.tgz"
+ }
+ }
+ }
+ }
+ },
+ "sequencify": {
+ "version": "0.0.7",
+ "from": "sequencify@~0.0.7",
+ "resolved": "https://registry.npmjs.org/sequencify/-/sequencify-0.0.7.tgz"
+ },
+ "stream-consume": {
+ "version": "0.1.0",
+ "from": "stream-consume@~0.1.0",
+ "resolved": "https://registry.npmjs.org/stream-consume/-/stream-consume-0.1.0.tgz"
+ }
+ }
+ },
+ "pretty-hrtime": {
+ "version": "1.0.0",
+ "from": "pretty-hrtime@^1.0.0",
+ "resolved": "https://registry.npmjs.org/pretty-hrtime/-/pretty-hrtime-1.0.0.tgz"
+ },
+ "semver": {
+ "version": "4.3.6",
+ "from": "semver@^4.1.0",
+ "resolved": "https://registry.npmjs.org/semver/-/semver-4.3.6.tgz"
+ },
+ "tildify": {
+ "version": "1.1.0",
+ "from": "tildify@^1.0.0",
+ "resolved": "https://registry.npmjs.org/tildify/-/tildify-1.1.0.tgz",
+ "dependencies": {
+ "os-homedir": {
+ "version": "1.0.1",
+ "from": "os-homedir@^1.0.0",
+ "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.1.tgz"
+ }
+ }
+ },
+ "v8flags": {
+ "version": "2.0.9",
+ "from": "v8flags@^2.0.2",
+ "resolved": "https://registry.npmjs.org/v8flags/-/v8flags-2.0.9.tgz",
+ "dependencies": {
+ "user-home": {
+ "version": "1.1.1",
+ "from": "user-home@^1.1.1",
+ "resolved": "https://registry.npmjs.org/user-home/-/user-home-1.1.1.tgz"
+ }
+ }
+ },
+ "vinyl-fs": {
+ "version": "0.3.13",
+ "from": "vinyl-fs@^0.3.0",
+ "resolved": "https://registry.npmjs.org/vinyl-fs/-/vinyl-fs-0.3.13.tgz",
+ "dependencies": {
+ "defaults": {
+ "version": "1.0.2",
+ "from": "defaults@^1.0.0",
+ "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.2.tgz",
+ "dependencies": {
+ "clone": {
+ "version": "0.1.19",
+ "from": "clone@~0.1.5",
+ "resolved": "https://registry.npmjs.org/clone/-/clone-0.1.19.tgz"
+ }
+ }
+ },
+ "glob-stream": {
+ "version": "3.1.18",
+ "from": "glob-stream@^3.1.5",
+ "resolved": "https://registry.npmjs.org/glob-stream/-/glob-stream-3.1.18.tgz",
+ "dependencies": {
+ "glob": {
+ "version": "4.5.3",
+ "from": "glob@^4.3.1",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-4.5.3.tgz",
+ "dependencies": {
+ "inflight": {
+ "version": "1.0.4",
+ "from": "inflight@^1.0.4",
+ "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.4.tgz",
+ "dependencies": {
+ "wrappy": {
+ "version": "1.0.1",
+ "from": "wrappy@1",
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.1.tgz"
+ }
+ }
+ },
+ "inherits": {
+ "version": "2.0.1",
+ "from": "inherits@2",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz"
+ },
+ "once": {
+ "version": "1.3.2",
+ "from": "once@^1.3.0",
+ "resolved": "https://registry.npmjs.org/once/-/once-1.3.2.tgz",
+ "dependencies": {
+ "wrappy": {
+ "version": "1.0.1",
+ "from": "wrappy@1",
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.1.tgz"
+ }
+ }
+ }
+ }
+ },
+ "minimatch": {
+ "version": "2.0.10",
+ "from": "minimatch@^2.0.1",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-2.0.10.tgz",
+ "dependencies": {
+ "brace-expansion": {
+ "version": "1.1.0",
+ "from": "brace-expansion@^1.0.0",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.0.tgz",
+ "dependencies": {
+ "balanced-match": {
+ "version": "0.2.0",
+ "from": "balanced-match@^0.2.0",
+ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-0.2.0.tgz"
+ },
+ "concat-map": {
+ "version": "0.0.1",
+ "from": "concat-map@0.0.1",
+ "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz"
+ }
+ }
+ }
+ }
+ },
+ "ordered-read-streams": {
+ "version": "0.1.0",
+ "from": "ordered-read-streams@^0.1.0",
+ "resolved": "https://registry.npmjs.org/ordered-read-streams/-/ordered-read-streams-0.1.0.tgz"
+ },
+ "glob2base": {
+ "version": "0.0.12",
+ "from": "glob2base@^0.0.12",
+ "resolved": "https://registry.npmjs.org/glob2base/-/glob2base-0.0.12.tgz",
+ "dependencies": {
+ "find-index": {
+ "version": "0.1.1",
+ "from": "find-index@^0.1.1",
+ "resolved": "https://registry.npmjs.org/find-index/-/find-index-0.1.1.tgz"
+ }
+ }
+ },
+ "unique-stream": {
+ "version": "1.0.0",
+ "from": "unique-stream@^1.0.0",
+ "resolved": "https://registry.npmjs.org/unique-stream/-/unique-stream-1.0.0.tgz"
+ }
+ }
+ },
+ "glob-watcher": {
+ "version": "0.0.6",
+ "from": "glob-watcher@^0.0.6",
+ "resolved": "https://registry.npmjs.org/glob-watcher/-/glob-watcher-0.0.6.tgz",
+ "dependencies": {
+ "gaze": {
+ "version": "0.5.1",
+ "from": "gaze@^0.5.1",
+ "resolved": "https://registry.npmjs.org/gaze/-/gaze-0.5.1.tgz",
+ "dependencies": {
+ "globule": {
+ "version": "0.1.0",
+ "from": "globule@~0.1.0",
+ "resolved": "https://registry.npmjs.org/globule/-/globule-0.1.0.tgz",
+ "dependencies": {
+ "lodash": {
+ "version": "1.0.2",
+ "from": "lodash@~1.0.1",
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-1.0.2.tgz"
+ },
+ "glob": {
+ "version": "3.1.21",
+ "from": "glob@~3.1.21",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-3.1.21.tgz",
+ "dependencies": {
+ "graceful-fs": {
+ "version": "1.2.3",
+ "from": "graceful-fs@~1.2.0",
+ "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-1.2.3.tgz"
+ },
+ "inherits": {
+ "version": "1.0.0",
+ "from": "inherits@1",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-1.0.0.tgz"
+ }
+ }
+ },
+ "minimatch": {
+ "version": "0.2.14",
+ "from": "minimatch@~0.2.11",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-0.2.14.tgz",
+ "dependencies": {
+ "lru-cache": {
+ "version": "2.6.5",
+ "from": "lru-cache@2",
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-2.6.5.tgz"
+ },
+ "sigmund": {
+ "version": "1.0.1",
+ "from": "sigmund@~1.0.0",
+ "resolved": "https://registry.npmjs.org/sigmund/-/sigmund-1.0.1.tgz"
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ },
+ "graceful-fs": {
+ "version": "3.0.8",
+ "from": "graceful-fs@^3.0.0",
+ "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-3.0.8.tgz"
+ },
+ "mkdirp": {
+ "version": "0.5.1",
+ "from": "mkdirp@^0.5.0",
+ "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz",
+ "dependencies": {
+ "minimist": {
+ "version": "0.0.8",
+ "from": "minimist@0.0.8",
+ "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz"
+ }
+ }
+ },
+ "strip-bom": {
+ "version": "1.0.0",
+ "from": "strip-bom@^1.0.0",
+ "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-1.0.0.tgz",
+ "dependencies": {
+ "first-chunk-stream": {
+ "version": "1.0.0",
+ "from": "first-chunk-stream@^1.0.0",
+ "resolved": "https://registry.npmjs.org/first-chunk-stream/-/first-chunk-stream-1.0.0.tgz"
+ },
+ "is-utf8": {
+ "version": "0.2.0",
+ "from": "is-utf8@^0.2.0",
+ "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.0.tgz"
+ }
+ }
+ },
+ "through2": {
+ "version": "0.6.5",
+ "from": "through2@^0.6.1",
+ "resolved": "https://registry.npmjs.org/through2/-/through2-0.6.5.tgz",
+ "dependencies": {
+ "readable-stream": {
+ "version": "1.0.33",
+ "from": "readable-stream@>=1.0.33-1 <1.1.0-0",
+ "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.0.33.tgz",
+ "dependencies": {
+ "core-util-is": {
+ "version": "1.0.1",
+ "from": "core-util-is@~1.0.0",
+ "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.1.tgz"
+ },
+ "isarray": {
+ "version": "0.0.1",
+ "from": "isarray@0.0.1",
+ "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz"
+ },
+ "string_decoder": {
+ "version": "0.10.31",
+ "from": "string_decoder@~0.10.x",
+ "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz"
+ },
+ "inherits": {
+ "version": "2.0.1",
+ "from": "inherits@~2.0.1",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz"
+ }
+ }
+ },
+ "xtend": {
+ "version": "4.0.0",
+ "from": "xtend@>=4.0.0 <4.1.0-0",
+ "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.0.tgz"
+ }
+ }
+ },
+ "vinyl": {
+ "version": "0.4.6",
+ "from": "vinyl@^0.4.0",
+ "resolved": "https://registry.npmjs.org/vinyl/-/vinyl-0.4.6.tgz",
+ "dependencies": {
+ "clone": {
+ "version": "0.2.0",
+ "from": "clone@^0.2.0",
+ "resolved": "https://registry.npmjs.org/clone/-/clone-0.2.0.tgz"
+ },
+ "clone-stats": {
+ "version": "0.0.1",
+ "from": "clone-stats@^0.0.1",
+ "resolved": "https://registry.npmjs.org/clone-stats/-/clone-stats-0.0.1.tgz"
+ }
+ }
+ }
+ }
+ }
+ }
+ },
"javascript-state-machine": {
"version": "2.3.5",
"from": "javascript-state-machine@",
diff --git a/package.json b/package.json
index af576e5..91b1369 100644
--- a/package.json
+++ b/package.json
@@ -10,8 +10,8 @@
"scripts": {
"test": "NODE_ENV=test mocha spec/",
"start": "npm run compile && node index.js",
- "compile:dev": "node_modules/react-tools/bin/jsx --watch --no-cache-dir --source-map-inline -x jsx lib/react/ public/js/",
- "compile": "node_modules/react-tools/bin/jsx -x jsx lib/react/ public/js/",
+ "watch": "node_modules/gulp/bin/gulp.js watch",
+ "compile": "node_modules/gulp/bin/gulp.js",
"dev": "nodemon index.js"
},
"repository": {
@@ -29,16 +29,22 @@
"express": "~4.13.1",
"express-handlebars": "~2.0.1",
"extend": "^3.0.0",
+ "gulp": "^3.9.0",
+ "gulp-concat": "^2.6.0",
+ "gulp-react": "^3.0.1",
"javascript-state-machine": "^2.3.5",
"morgan": "~1.6.1",
"node-mysql": "~0.4.2",
"react-tools": "~0.13.3",
"request": "~2.60.0",
"socket.io": "~1.3.5",
+ "gulp-watch": "^4.3.4",
"winston": "~1.0.1"
},
"devDependencies": {
"chai": "~3.1.0",
+ "gulp": "^3.9.0",
+ "gulp-plumber": "^1.0.1",
"mocha": "~2.2.5",
"nodemon": "~1.3.7",
"supertest": "~1.0.1"
diff --git a/public/css/app.css b/public/css/app.css
index 750a8af..5eb4315 100644
--- a/public/css/app.css
+++ b/public/css/app.css
@@ -19,4 +19,14 @@
.signin {
margin: 0px 10px;
+}
+
+.user-online {
+ border-radius: 50%;
+ width: 10px;
+ height: 10px;
+
+ background-image: -moz-radial-gradient(45px 45px 45deg, circle cover, yellow 0%, orange 100%, red 95%);
+ background-image: -webkit-radial-gradient(45px 45px, circle cover, yellow, orange);
+ background-image: radial-gradient(45px 45px 45deg, circle cover, yellow 0%, orange 100%, red 95%);
}
\ No newline at end of file
diff --git a/public/js/app.js b/public/js/app.js
index 988e1a8..0314448 100644
--- a/public/js/app.js
+++ b/public/js/app.js
@@ -1,93 +1,382 @@
-$(function () {
-
"use strict";
-var UserCounter = React.createClass({displayName: "UserCounter",
+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 () {
- return (
- 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.count, " ")
+ 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 UserLogin = React.createClass({displayName: "UserLogin",
- authorizeId: function (id) {
- id = parseInt(id, 10);
- socket.emit("users:authorize", {
- id: id
- });
- },
- handleSubmit: function (e) {
+var JoinGatherButton = React.createClass({displayName: "JoinGatherButton",
+ joinGather: 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);
- return;
+ socket.emit("gather:join", {});
},
render: function () {
+ var message = this.props.buttonName || "Join Gather";
+ var buttonClass = "btn btn-primary";
+ if (this.props.buttonClass) {
+ buttonClass += " " + this.props.buttonClass;
+ }
+ return (React.createElement("button", {
+ onClick: this.joinGather,
+ className: buttonClass}, message))
+ }
+});
+
+var SelectPlayerButton = React.createClass({displayName: "SelectPlayerButton",
+ selectPlayer: function (e) {
+ e.preventDefault();
+ },
+ render: function () {
+ if (!this.props.currentGatherer.leader) {
+ return false;
+ } else {
+ return (React.createElement("button", {
+ onClick: this.selectPlayer,
+ className: "btn btn-xs btn-primary"}, " Select"
+ )
+ );
+ }
+ }
+})
+
+var GatherTeams = React.createClass({displayName: "GatherTeams",
+ alienGatherers: function () {
+ return this.props.gather.gatherers.filter(function (gatherer) {
+ return gatherer.team === "alien";
+ }).sort(function (gatherer) {
+ return (gatherer.leader) ? 1 : 0;
+ });
+ },
+ marineGatherers: function () {
+ return this.props.gather.gatherers.filter(function (gatherer) {
+ return gatherer.team === "marine";
+ }).sort(function (gatherer) {
+ return (gatherer.leader) ? 1 : 0;
+ });
+ },
+ 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 marines = this.marineGatherers().map(extractGatherer);
+ var aliens = this.alienGatherers().map(extractGatherer);
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: "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("table", {className: "table"},
+ React.createElement("tbody", null,
+ aliens
+ )
+ )
+ )
+ ),
+ React.createElement("div", {className: "col-md-6"},
+ React.createElement("div", {className: "panel panel-default"},
+ React.createElement("div", {className: "panel-heading"},
+ "Marines"
+ ),
+ React.createElement("table", {className: "table"},
+ React.createElement("tbody", null,
+ marines
+ )
+ )
+ )
)
- ),
- 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 () {
+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 picking 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 {
- count: 0,
- users: []
+ num: num,
+ den: den,
+ message: message
};
},
- componentDidMount: function () {
- socket.on('users:update', this.updateUsers);
+ 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"
+ };
},
- updateUsers: function (data) {
- this.setProps({
- count: data.count,
- users: data.users
- });
+ 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 users = this.props.users.map(function (user) {
+ var progress;
+ var gatherState = this.props.gather.state;
+ if (gatherState === 'gathering' && this.props.gather.gatherers.length) {
+ progress = this.gatheringProgress();
+ } else if (gatherState === 'election') {
+ progress = this.electionProgress();
+ } else if (gatherState === 'selection') {
+ progress = this.selectionProgress();
+ }
+ if (progress) {
+ var style = {
+ width: Math.round((progress.num / progress.den * 100)) + "%"
+ };
return (
- React.createElement("li", {key: user.id}, React.createElement("a", {href: "#"}, user.username))
+ React.createElement("div", {className: "panel-body"},
+ React.createElement("p", null, React.createElement("strong", null, this.stateDescription()), " ", progress.message),
+ React.createElement("div", {className: "progress"},
+ React.createElement("div", {className: "progress-bar progress-bar-striped active",
+ "data-role": "progressbar",
+ "data-aria-valuenow": progress.num,
+ "data-aria-valuemin": "0",
+ "data-aria-valuemax": progress.den,
+ style: style}
+ )
+ )
+ )
);
+ } else {
+ return false;
+ }
+ }
+});
+
+var Gather = React.createClass({displayName: "Gather",
+ getDefaultProps: function () {
+ return {
+ gather: {
+ gatherers: []
+ }
+ }
+ },
+ joinedGather: function () {
+ return this.props.currentGatherer !== null;
+ },
+ componentDidMount: function () {
+ var self = this;
+ socket.on("gather:refresh", function (data) {
+ self.setProps({
+ gather: data.gather,
+ currentGatherer: data.currentGatherer
+ });
});
+ },
+ leaveGather: function (e) {
+ e.preventDefault();
+ socket.emit("gather:leave", {});
+ },
+ inviteToGather: function (e) {
+ e.preventDefault();
+ },
+ render: function () {
+ var joinButton;
+ if (this.joinedGather()) {
+ joinButton = (React.createElement("li", null, React.createElement("button", {
+ onClick: this.leaveGather,
+ className: "btn btn-danger"}, "Leave Gather")));
+ } else {
+ joinButton = (React.createElement("li", null, React.createElement(JoinGatherButton, null)));
+ }
+ 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")));
+ }
+ var gatherTeams;
+ if (this.props.gather.state === 'selection') {
+ gatherTeams = React.createElement(GatherTeams, {gather: this.props.gather})
+ }
return (
- React.createElement("ul", {className: "nav", id: "side-menu"},
- React.createElement(UserCounter, React.__spread({}, this.props)),
- users,
- React.createElement("li", null, React.createElement("br", null), React.createElement(UserLogin, null), React.createElement("br", null))
+ React.createElement("div", {className: "panel panel-default"},
+ React.createElement("div", {className: "panel-heading"},
+ React.createElement("strong", null, "NS2 Gather "),
+ React.createElement("span", {className: "badge add-left"}, this.props.gather.gatherers.length)
+ ),
+ React.createElement(Gatherers, {gather: this.props.gather, currentGatherer: this.props.currentGatherer}),
+ gatherTeams,
+ React.createElement(GatherProgress, {gather: this.props.gather}),
+ React.createElement("div", {className: "panel-footer text-right"},
+ React.createElement("ul", {className: "list-inline"},
+ inviteButton,
+ joinButton
+ )
+ )
)
);
}
});
+var Gatherers = React.createClass({displayName: "Gatherers",
+ render: function () {
+ var self = this;
+ var gatherers = this.props.gather.gatherers.map(function (gatherer) {
+ var lifeforms = (
+ gatherer.user.ability.lifeforms.map(function (lifeform) {
+ return (React.createElement("span", {className: "label label-default"}, lifeform));
+ })
+ );
+
+ // Switch this to online status
+ var online= (React.createElement("span", {src: "/images/commander.png",
+ alt: "online",
+ className: "user-online"}, " "));
+
+ var division = (React.createElement("span", {className: "label label-primary"}, gatherer.user.ability.division));
+ var action = lifeforms;
+ 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})
+ )
+ );
+ }
+
+ return (
+ React.createElement("tr", {key: gatherer.user.id},
+ React.createElement("td", {className: "col-md-1"}, online),
+ React.createElement("td", {className: "col-md-5"}, gatherer.user.username),
+ React.createElement("td", {className: "col-md-3"}, division, " "),
+ 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("div", {className: "panel-heading"},
+ React.createElement("h5", {className: "panel-title"}, "Roster")
+ ),
+ React.createElement("table", {className: "table roster-table"},
+ React.createElement("tbody", null,
+ gatherers
+ )
+ )
+ )
+ )
+ );
+ } else {
+ return (React.createElement("div", {className: "panel-body text-center"}, React.createElement(JoinGatherButton, {buttonClass: "btn-lg", buttonName: "Start a Gather"})));
+ }
+ }
+});
+
+
+
+
+"use strict";
+
+var socket;
+
+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")
+ });
+
+ 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'));
+};
+"use strict";
+
var Chatroom = React.createClass({displayName: "Chatroom",
getDefaultProps: function () {
return {
@@ -196,69 +485,6 @@ var ChatMessage = React.createClass({displayName: "ChatMessage",
}
});
-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("a", {href: "#"}, this.props.user.username, " ", React.createElement("img", {src: this.props.user.avatar,
- alt: "User Avatar",
- height: "20",
- width: "20"}))
- );
- } else {
- return false;
- }
- }
-});
-
-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 MessageBar = React.createClass({displayName: "MessageBar",
sendMessage: function (content) {
socket.emit("message:new", {
@@ -296,280 +522,114 @@ var MessageBar = React.createClass({displayName: "MessageBar",
}
});
-var JoinGatherButton = React.createClass({displayName: "JoinGatherButton",
- joinGather: function (e) {
- e.preventDefault();
- socket.emit("gather:join", {});
- },
+"use strict";
+
+var UserCounter = React.createClass({displayName: "UserCounter",
render: function () {
- var message = this.props.buttonName || "Join Gather";
- var buttonClass = "btn btn-primary";
- if (this.props.buttonClass) {
- buttonClass += " " + this.props.buttonClass;
- }
- return (React.createElement("button", {
- onClick: this.joinGather,
- className: buttonClass}, message))
+ return (
+ 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.count, " ")
+ )
+ )
+ );
}
});
-var GatherProgress = React.createClass({displayName: "GatherProgress",
- gatheringProgress: function () {
- var num = this.props.gather.gatherers.length;
- var den = 12;
- return {
- num: num,
- den: den,
- message: num + " / " + den
- };
+var UserLogin = React.createClass({displayName: "UserLogin",
+ authorizeId: function (id) {
+ id = parseInt(id, 10);
+ socket.emit("users:authorize", {
+ id: id
+ });
},
- 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"
- };
+ 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);
+ return;
},
render: function () {
- var progress;
- var gatherState = this.props.gather.state;
- if (gatherState === 'gathering' && this.props.gather.gatherers.length) {
- progress = this.gatheringProgress();
- } else if (gatherState === 'election') {
- progress = this.electionProgress();
- } else if (gatherState === 'selection') {
- progress = this.selectionProgress();
- }
- if (progress) {
- var style = {
- width: Math.round((progress.num / progress.den * 100)) + "%"
- };
- return (
- React.createElement("div", {className: "panel-body"},
- React.createElement("p", null, "Gather Progress"),
- React.createElement("div", {className: "progress"},
- React.createElement("div", {className: "progress-bar progress-bar-striped active",
- "data-role": "progressbar",
- "data-aria-valuenow": progress.num,
- "data-aria-valuemin": "0",
- "data-aria-valuemax": progress.den,
- style: style},
- progress.message
- )
- )
+ 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 {
+ count: 0,
+ users: []
+ };
+ },
+ componentDidMount: function () {
+ socket.on('users:update', this.updateUsers);
+ },
+ updateUsers: function (data) {
+ this.setProps({
+ count: data.count,
+ 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(UserCounter, React.__spread({}, this.props)),
+ users,
+ React.createElement("li", null, React.createElement("br", null), React.createElement(UserLogin, null), React.createElement("br", null))
+ )
+ );
+ }
+});
+
+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("a", {href: "#"}, this.props.user.username, " ", React.createElement("img", {src: this.props.user.avatar,
+ alt: "User Avatar",
+ height: "20",
+ width: "20"}))
);
} else {
return false;
}
}
});
-
-var Gather = React.createClass({displayName: "Gather",
- getDefaultProps: function () {
- return {
- gather: {
- gatherers: []
- }
- }
- },
- joinedGather: function () {
- var self = this;
- return this.props.gather.gatherers.some(function (gatherer) {
- return gatherer.user.id === self.props.currentUser.id;
- });
- },
- componentDidMount: function () {
- var self = this;
- socket.on("gather:refresh", function (data) {
- self.setProps({
- gather: data.gather,
- currentUser: data.currentUser
- });
- });
- },
- 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 picking teams";
- case "done":
- return "Gather completed";
- default:
- return "Initialising gather";
- }
- },
- leaveGather: function (e) {
- e.preventDefault();
- socket.emit("gather:leave", {});
- },
- inviteToGather: function (e) {
- e.preventDefault();
- },
- currentGatherer: function () {
- var current = null;
- var self = this;
- this.props.gather.gatherers.forEach(function (gatherer) {
- if (gatherer.id === self.props.currentUser.id) current = gatherer;
- });
- return current;
- },
- render: function () {
- var joinButton;
- if (this.joinedGather()) {
- joinButton = (React.createElement("li", null, React.createElement("button", {
- onClick: this.leaveGather,
- className: "btn btn-danger"}, "Leave Gather")));
- } else {
- joinButton = (React.createElement("li", null, React.createElement(JoinGatherButton, null)));
- }
- 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 panel-default"},
- React.createElement("div", {className: "panel-heading"},
- React.createElement("strong", null, "NS2 Gather "),
- React.createElement("span", {className: "badge add-left"}, this.props.gather.gatherers.length),
- React.createElement("br", null),
- this.stateDescription()
- ),
- React.createElement(Gatherers, {gather: this.props.gather, currentGatherer: this.currentGatherer()}),
- React.createElement(GatherProgress, {gather: this.props.gather}),
- React.createElement("div", {className: "panel-footer text-right"},
- React.createElement("ul", {className: "list-inline"},
- inviteButton,
- joinButton
- )
- )
- )
- );
- }
-});
-
-var LeaderPoll = React.createClass({displayName: "LeaderPoll",
- render: function () {
- return (
- React.createElement("div", {className: "panel-body"}
- )
- );
- }
-});
-
-var Gatherers = React.createClass({displayName: "Gatherers",
- render: function () {
- var self = this;
- var gatherers = this.props.gather.gatherers.map(function (gatherer) {
- var lifeforms = (
- gatherer.user.ability.lifeforms.map(function (lifeform) {
- return (React.createElement("span", {className: "label label-default"}, lifeform));
- })
- );
-
- var commBadge;
- if (gatherer.user.ability.commander) {
- commBadge = (React.createElement("img", {src: "/images/commander.png",
- alt: "Commander",
- height: "20",
- width: "20"}));
- }
-
- var division = (React.createElement("span", {className: "label label-primary"}, gatherer.user.ability.division));
- var action = lifeforms;
- 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("div", {className: "text-right"},
- React.createElement("small", null, votes + " votes", " "),
-
- React.createElement(VoteButton, {currentGatherer: self.props.currentGatherer, candidate: gatherer})
- )
- );
- }
-
- return (
- React.createElement("tr", {key: gatherer.user.id},
- React.createElement("td", {className: "col-md-1"}, commBadge),
- React.createElement("td", {className: "col-md-5"}, gatherer.user.username),
- React.createElement("td", {className: "col-md-3"}, division, " "),
- React.createElement("td", {className: "col-md-2"}, action, " ")
- )
- );
- })
- if (this.props.gather.gatherers.length) {
- return (
- React.createElement("div", {className: "panel-body"},
- React.createElement("div", {className: "panel panel-default"},
- React.createElement("div", {className: "panel-heading"},
- React.createElement("h5", {className: "panel-title"}, "Roster")
- ),
- React.createElement("table", {className: "table roster-table"},
- React.createElement("tbody", null,
- gatherers
- )
- )
- )
- )
- );
- } else {
- return (React.createElement("div", {className: "panel-body text-center"}, React.createElement(JoinGatherButton, {buttonClass: "btn-lg", buttonName: "Start a Gather"})));
- }
- }
-});
-
-var socket;
-
-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")
- });
-
- 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'));
-};
-
-initialiseComponents();
-
-
-
-});
-
-
-//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transformed.js","sources":[null],"names":[],"mappings":"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","sourcesContent":["$(function () {\n\n\"use strict\";\n\nvar UserCounter = React.createClass({\n\trender: function () {\n\t\treturn (\n\t\t\t<li>\n\t\t\t\t<a href=\"#\">\n\t\t\t\t\t<i className=\"fa fa-users fa-fw\"></i> Online \n\t\t\t\t\t<span className=\"badge add-left\"> {this.props.count} </span>\n\t\t\t\t</a>\n\t\t\t</li>\n\t\t);\n\t}\n});\n\nvar UserLogin = React.createClass({\n\tauthorizeId: function (id) {\n\t\tid = parseInt(id, 10);\n\t\tsocket.emit(\"users:authorize\", {\n\t\t\tid: id\n\t\t});\n\t},\n\thandleSubmit: function (e) {\n\t\te.preventDefault();\n\t\tvar id = React.findDOMNode(this.refs.authorize_id).value.trim();\n\t\tif (!id) return;\n\t\tReact.findDOMNode(this.refs.authorize_id).value = '';\n\t\tthis.authorizeId(id);\n\t\treturn;\n\t},\n\trender: function () {\n\t\treturn (\n\t\t\t<form onSubmit={this.handleSubmit} >\n\t\t\t\t<div className=\"input-group signin\">\n\t\t\t\t\t<input \n\t\t\t\t\t\tid=\"btn-input\" \n\t\t\t\t\t\ttype=\"text\" \n\t\t\t\t\t\tclassName=\"form-control\" \n\t\t\t\t\t\tref=\"authorize_id\"\n\t\t\t\t\t\tplaceholder=\"Choose an ID...\" />\n\t\t\t\t\t<span className=\"input-group-btn\">\n\t\t\t\t\t\t<input \n\t\t\t\t\t\t\ttype=\"submit\" \n\t\t\t\t\t\t\tclassName=\"btn btn-primary\" \n\t\t\t\t\t\t\tid=\"btn-chat\" \n\t\t\t\t\t\t\tvalue=\"Login\" />\n\t\t\t\t\t</span>\n\t\t\t\t</div>\n\t\t\t\t<div className=\"signin\">\n\t\t\t\t<p className=\"text-center\"><small>Just a temporary measure until genuine authentication is implemented</small></p>\n\t\t\t\t</div>\n\t\t\t</form>\n\t\t);\n\t}\n})\n\nvar UserMenu = React.createClass({\n\tgetDefaultProps: function () {\n\t\treturn {\n\t\t\tcount: 0,\n\t\t\tusers: []\n\t\t};\n\t},\n\tcomponentDidMount: function () {\n\t\tsocket.on('users:update', this.updateUsers);\n\t},\n\tupdateUsers: function (data) {\n\t\tthis.setProps({\n\t\t\tcount: data.count,\n\t\t\tusers: data.users\n\t\t});\n\t},\n\trender: function () {\n\t\tvar users = this.props.users.map(function (user) {\n\t\t\treturn (\n\t\t\t\t<li key={user.id}><a href=\"#\">{user.username}</a></li>\n\t\t\t);\n\t\t});\n\t\treturn (\n\t\t\t<ul className=\"nav\" id=\"side-menu\">\n\t\t\t\t<UserCounter {...this.props} />\n\t\t\t\t{users}\n\t\t\t\t<li><br /><UserLogin /><br /></li>\n\t\t\t</ul>\n\t\t);\n\t}\n});\n\nvar Chatroom = React.createClass({\n\tgetDefaultProps: function () {\n\t\treturn {\n\t\t\thistory: []\n\t\t};\n\t},\n\tcomponentDidMount: function () {\n\t\tvar self = this;\n\t\tvar TIMER_INTERVAL = 60000; // Every minute\n\n\t\tsocket.on(\"message:new\", function (data) {\n\t\t\tvar history = self.props.history;\n\t\t\thistory.push(data);\n\t\t\tself.setProps({\n\t\t\t\thistory: history\n\t\t\t});\n\t\t\tself.scrollToBottom();\n\t\t});\n\n\t\t// Message History Retrieved\n\t\tsocket.on(\"message:refresh\", function (data) {\n\t\t\tself.setProps({\n\t\t\t\thistory: data.chatHistory\n\t\t\t});\n\t\t\tself.scrollToBottom();\n\t\t});\n\n\t\tsocket.emit(\"message:refresh\", {});\n\n\t\tself.timer = setInterval(function () {\n\t\t\tif (self.refs.messages) self.refs.messages.refreshTime();\n\t\t}, TIMER_INTERVAL);\n\t},\n\n\tcomponentDidUnmount: function () {\n\t\tclearInterval(this.timer);\n\t},\n\tsendMessage: function (message) {\n\t\tsocket.emit(\"newMessage\", {message: message});\n\t},\n\tscrollToBottom: function () {\n\t\tvar node = React.findDOMNode(this.refs.messageContainer);\n\t  node.scrollTop = node.scrollHeight;\n\t},\n\trender: function () {\n\t\tvar messages = this.props.history.map(function (message) {\n\t\t\treturn (\n\t\t\t\t<ChatMessage \n\t\t\t\t\tavatar={message.author.avatar} \n\t\t\t\t\tusername={message.author.username}\n\t\t\t\t\tcontent={message.content}\n\t\t\t\t\tref=\"messages\"\n\t\t\t\t\tcreatedAt={message.createdAt} />\n\t\t\t);\n\t\t});\n\t\treturn (\n\t\t\t<div className=\"panel panel-default\">\n\t\t\t\t<div className=\"panel-heading\">Gather Chat</div>\n\t\t\t\t<div className=\"panel-body\">\n\t\t\t\t\t<ul className=\"chat\" id=\"chatmessages\" ref=\"messageContainer\">\n\t\t\t\t\t\t{messages}\n\t\t\t\t\t</ul>\n\t\t\t\t</div>\n\t\t\t\t<div className=\"panel-footer\">\n\t\t\t\t\t<MessageBar />\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t);\n\t}\n});\n\nvar ChatMessage = React.createClass({\n\tgetInitialState: function () {\n\t\treturn {\n\t\t\ttimeAgo: $.timeago(this.props.createdAt)\n\t\t}\n\t},\n\trefreshTime: function () {\n\t\tvar self = this;\n\t\tself.setState({\n\t\t\ttimeAgo: $.timeago(self.props.createdAt)\n\t\t});\n\t},\n\trender: function () {\n\t\treturn (\n\t\t\t<li className=\"left clearfix\">\n\t\t\t\t<span className=\"chat-img pull-left\">\n\t\t\t\t\t\t<img \n\t\t\t\t\t\t\tsrc={this.props.avatar} \n\t\t\t\t\t\t\talt=\"User Avatar\" \n\t\t\t\t\t\t\theight=\"40\"\n\t\t\t\t\t\t\twidth=\"40\"\n\t\t\t\t\t\t\tclassName=\"img-circle\" />\n\t\t\t\t</span>\n\t\t\t\t<div className=\"chat-body clearfix\">\n\t\t\t\t\t<div className=\"header\">\n\t\t\t\t\t\t<strong className=\"primary-font\">{this.props.username}</strong>\n\t\t\t\t\t\t<small className=\"pull-right text-muted\">\n\t\t\t\t\t\t\t<i className=\"fa fa-clock-o fa-fw\"></i> {this.state.timeAgo}\n\t\t\t\t\t\t</small>\n\t\t\t\t\t</div>\n\t\t\t\t\t<p>{this.props.content}</p>\n\t\t\t\t</div>\n\t\t\t</li>\n\t\t);\n\t}\n});\n\nvar CurrentUser = React.createClass({\n\tcomponentDidMount: function () {\n\t\tvar self = this;\n\t\tsocket.on(\"users:update\", function (data) {\n\t\t\tself.setProps({\n\t\t\t\tuser: data.currentUser\n\t\t\t});\n\t\t});\n\t\tsocket.emit(\"users:refresh\", {});\n\t},\n\trender: function () {\n\t\tif (this.props.user) {\n\t\t\treturn (\n\t\t\t\t<a href=\"#\">{this.props.user.username} &nbsp;<img src={this.props.user.avatar}\n\t\t\t\t\talt=\"User Avatar\" \n\t\t\t\t\theight=\"20\"\n\t\t\t\t\twidth=\"20\" /></a>\n\t\t\t);\n\t\t} else {\n\t\t\treturn false;\n\t\t}\n\t}\n});\n\nvar VoteButton = React.createClass({\n\tcancelVote: function (e) {\n\t\tsocket.emit(\"gather:vote\", {\n\t\t\tleader: {\n\t\t\t\tcandidate: null\n\t\t\t}\n\t\t});\n\t},\n\tvote: function (e) {\n\t\te.preventDefault();\n\t\tsocket.emit(\"gather:vote\", {\n\t\t\tleader: {\n\t\t\t\tcandidate: parseInt(e.target.value, 10)\n\t\t\t}\n\t\t});\n\t},\n\trender: function () {\n\t\tif (this.props.currentGatherer === null) {\n\t\t\treturn false;\n\t\t}\n\t\tif (this.props.currentGatherer.leaderVote === this.props.candidate.id) {\n\t\t\treturn (\n\t\t\t\t<button \n\t\t\t\t\tonClick={this.cancelVote} \n\t\t\t\t\tclassName=\"btn btn-xs btn-success\">Voted\n\t\t\t\t</button>\n\t\t\t);\n\t\t} else {\n\t\t\treturn (\n\t\t\t\t<button \n\t\t\t\t\tonClick={this.vote} \n\t\t\t\t\tclassName=\"btn btn-xs btn-default\"\n\t\t\t\t\tvalue={this.props.candidate.id}>Vote\n\t\t\t\t</button>\n\t\t\t);\n\t\t}\n\t}\n});\n\nvar MessageBar = React.createClass({\n\tsendMessage: function (content) {\n\t\tsocket.emit(\"message:new\", {\n\t\t\tcontent: content\n\t\t});\n\t},\n\thandleSubmit: function (e) {\n\t\te.preventDefault();\n\t\tvar content = React.findDOMNode(this.refs.content).value.trim();\n\t\tif (!content) return;\n\t\tReact.findDOMNode(this.refs.content).value = '';\n\t\tthis.sendMessage(content);\n\t\treturn;\n\t},\n\trender: function () {\n\t\treturn (\n\t\t\t<form onSubmit={this.handleSubmit} >\n\t\t\t\t<div className=\"input-group\">\n\t\t\t\t\t<input \n\t\t\t\t\t\tid=\"btn-input\" \n\t\t\t\t\t\ttype=\"text\" \n\t\t\t\t\t\tclassName=\"form-control\" \n\t\t\t\t\t\tref=\"content\"\n\t\t\t\t\t\tplaceholder=\"Be polite please...\" />\n\t\t\t\t\t<span className=\"input-group-btn\">\n\t\t\t\t\t\t<input \n\t\t\t\t\t\t\ttype=\"submit\" \n\t\t\t\t\t\t\tclassName=\"btn btn-primary\" \n\t\t\t\t\t\t\tid=\"btn-chat\" \n\t\t\t\t\t\t\tvalue=\"Send\" />\n\t\t\t\t\t</span>\n\t\t\t\t</div>\n\t\t\t</form>\n\t\t);\n\t}\n});\n\nvar JoinGatherButton = React.createClass({\n\tjoinGather: function (e) {\n\t\te.preventDefault();\n\t\tsocket.emit(\"gather:join\", {});\n\t},\n\trender: function () {\n\t\tvar message = this.props.buttonName || \"Join Gather\";\n\t\tvar buttonClass = \"btn btn-primary\";\n\t\tif (this.props.buttonClass) {\n\t\t\tbuttonClass += \" \" + this.props.buttonClass;\n\t\t}\n\t\treturn (<button \n\t\t\t\t\t\t\tonClick={this.joinGather} \n\t\t\t\t\t\t\tclassName={buttonClass}>{message}</button>)\n\t}\n});\n\nvar GatherProgress = React.createClass({\n\tgatheringProgress: function () {\n\t\tvar num = this.props.gather.gatherers.length;\n\t\tvar den = 12;\n\t\treturn {\n\t\t\tnum: num,\n\t\t\tden: den,\n\t\t\tmessage: num + \" / \" + den\n\t\t};\n\t},\n\telectionProgress: function () {\n\t\tvar num = this.props.gather.gatherers.reduce(function (acc, gatherer) {\n\t\t\tif (gatherer.leaderVote) acc++;\n\t\t\treturn acc;\n\t\t}, 0);\n\t\tvar den = 12;\n\t\treturn {\n\t\t\tnum: num,\n\t\t\tden: den,\n\t\t\tmessage: den - num + \" more votes required\"\n\t\t};\n\t},\n\tselectionProgress: function () {\n\t\tvar num = this.props.gather.gatherers.reduce(function (acc, gatherer) {\n\t\t\tif (gatherer.team !== \"lobby\") acc++;\n\t\t\treturn acc;\n\t\t}, 0);\n\t\tvar den = 12;\n\n\t\treturn {\n\t\t\tnum: num,\n\t\t\tden: den,\n\t\t\tmessage: num + \" out of \" + den + \" players assigned\"\n\t\t};\n\t},\n\trender: function () {\n\t\tvar progress;\n\t\tvar gatherState = this.props.gather.state;\n\t\tif (gatherState === 'gathering' && this.props.gather.gatherers.length) {\n\t\t\tprogress = this.gatheringProgress();\n\t\t} else if (gatherState === 'election') {\n\t\t\tprogress = this.electionProgress();\n\t\t} else if (gatherState === 'selection') {\n\t\t\tprogress = this.selectionProgress();\n\t\t}\n\t\tif (progress) {\n\t\t\tvar style = {\n\t\t\t\twidth: Math.round((progress.num / progress.den * 100)) + \"%\"\n\t\t\t};\n\t\t\treturn (\n\t\t\t\t<div className=\"panel-body\">\n\t\t\t\t\t<p>Gather Progress</p>\n\t\t\t\t\t<div className=\"progress\">\n\t\t\t\t\t  <div className=\"progress-bar progress-bar-striped active\" \n\t\t\t\t\t  \tdata-role=\"progressbar\" \n\t\t\t\t\t  \tdata-aria-valuenow={progress.num} \n\t\t\t\t\t  \tdata-aria-valuemin=\"0\" \n\t\t\t\t\t  \tdata-aria-valuemax={progress.den} \n\t\t\t\t\t  \tstyle={style}>\n\t\t\t\t\t    {progress.message}\n\t\t\t\t\t  </div>\n\t\t\t\t  </div>\n\t\t\t\t</div>\n\t\t\t);\n\t\t} else {\n\t\t\treturn false;\n\t\t}\n\t}\n});\n\nvar Gather = React.createClass({\n\tgetDefaultProps: function () {\n\t\treturn {\n\t\t\tgather: {\n\t\t\t\tgatherers: []\n\t\t\t}\n\t\t}\n\t},\n\tjoinedGather: function () {\n\t\tvar self = this;\n\t\treturn this.props.gather.gatherers.some(function (gatherer) {\n\t\t\treturn gatherer.user.id === self.props.currentUser.id;\n\t\t});\n\t},\n\tcomponentDidMount: function () {\n\t\tvar self = this;\n\t\tsocket.on(\"gather:refresh\", function (data) {\n\t\t\tself.setProps({\n\t\t\t\tgather: data.gather,\n\t\t\t\tcurrentUser: data.currentUser\n\t\t\t});\n\t\t});\n\t},\n\tstateDescription: function () {\n\t\tswitch(this.props.gather.state) {\n\t\t\tcase \"gathering\":\n\t\t\t\treturn \"Waiting for more gatherers\";\n\t\t\tcase \"election\":\n\t\t\t\treturn \"Currently voting for team leaders\";\n\t\t\tcase \"selection\":\n\t\t\t\treturn \"Waiting for leaders to picking teams\";\n\t\t\tcase \"done\":\n\t\t\t\treturn \"Gather completed\";\n\t\t\tdefault:\n\t\t\t\treturn \"Initialising gather\";\n\t\t}\n\t},\n\tleaveGather: function (e) {\n\t\te.preventDefault();\n\t\tsocket.emit(\"gather:leave\", {});\n\t},\n\tinviteToGather: function (e) {\n\t\te.preventDefault();\n\t},\n\tcurrentGatherer: function () {\n\t\tvar current = null;\n\t\tvar self = this;\n\t\tthis.props.gather.gatherers.forEach(function (gatherer) {\n\t\t\tif (gatherer.id === self.props.currentUser.id) current = gatherer;\n\t\t});\n\t\treturn current;\n\t},\n\trender: function () {\n\t\tvar joinButton;\n\t\tif (this.joinedGather()) {\n\t\t\tjoinButton = (<li><button \n\t\t\t\t\t\t\tonClick={this.leaveGather} \n\t\t\t\t\t\t\tclassName=\"btn btn-danger\">Leave Gather</button></li>);\n\t\t} else {\n\t\t\tjoinButton = (<li><JoinGatherButton /></li>);\n\t\t}\n\t\tvar inviteButton;\n\t\tif (this.props.gather.state === 'gathering') {\n\t\t\tinviteButton = (<li><button\n\t\t\t\t\t\t\tonClick={this.inviteToGather}\n\t\t\t\t\t\t\tclassName=\"btn btn-primary\">Invite to Gather</button></li>);\n\t\t}\n\t\treturn (\n\t\t\t<div className=\"panel panel-default\">\n\t\t\t\t<div className=\"panel-heading\">\n\t\t\t\t\t<strong>NS2 Gather </strong>\n\t\t\t\t\t<span className=\"badge add-left\">{this.props.gather.gatherers.length}</span>\n\t\t\t\t\t<br />\n\t\t\t\t\t{this.stateDescription()}\n\t\t\t\t</div>\n\t\t\t\t<Gatherers gather={this.props.gather} currentGatherer={this.currentGatherer()} />\n\t\t\t\t<GatherProgress gather={this.props.gather} />\n\t\t\t\t<div className=\"panel-footer text-right\">\n\t\t\t\t\t<ul className=\"list-inline\">\n\t\t\t\t\t\t{inviteButton}\n\t\t\t\t\t\t{joinButton}\n\t\t\t\t\t</ul>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t);\n\t}\n});\n\nvar LeaderPoll = React.createClass({\n\trender: function () {\n\t\treturn (\n\t\t\t<div className=\"panel-body\">\n\t\t\t</div>\n\t\t);\n\t}\n});\n\nvar Gatherers = React.createClass({\n\trender: function () {\n\t\tvar self = this;\n\t\tvar gatherers = this.props.gather.gatherers.map(function (gatherer) {\n\t\t\tvar lifeforms = (\n\t\t\t\tgatherer.user.ability.lifeforms.map(function (lifeform) {\n\t\t\t\t\treturn (<span className=\"label label-default\">{lifeform}</span>);\n\t\t\t\t})\n\t\t\t);\n\n\t\t\tvar commBadge;\n\t\t\tif (gatherer.user.ability.commander) {\n\t\t\t\tcommBadge = (<img src=\"/images/commander.png\" \n\t\t\t\t\t\t\talt=\"Commander\" \n\t\t\t\t\t\t\theight=\"20\"\n\t\t\t\t\t\t\twidth=\"20\" />);\n\t\t\t}\n\n\t\t\tvar division = (<span className=\"label label-primary\">{gatherer.user.ability.division}</span>);\n\t\t\tvar action = lifeforms;\n\t\t\tif (self.props.gather.state === \"election\") {\n\t\t\t\tvar votes = self.props.gather.gatherers.reduce(function (acc, voter) {\n\t\t\t\t\tif (voter.leaderVote === gatherer.id) acc++;\n\t\t\t\t\treturn acc;\n\t\t\t\t}, 0)\n\t\t\t\taction = (\n\t\t\t\t\t<div className=\"text-right\">\n\t\t\t\t\t<small>{votes + \" votes\"} &nbsp;</small>\n\n\t\t\t\t\t<VoteButton currentGatherer={self.props.currentGatherer} candidate={gatherer} />\n\t\t\t\t\t</div>\n\t\t\t\t);\n\t\t\t}\n\n\t\t\treturn (\n\t\t\t\t<tr key={gatherer.user.id}>\n\t\t\t\t\t<td className=\"col-md-1\">{commBadge}</td>\n\t\t\t\t\t<td className=\"col-md-5\">{gatherer.user.username}</td>\n\t\t\t\t\t<td className=\"col-md-3\">{division}&nbsp;</td>\n\t\t\t\t\t<td className=\"col-md-2\">{action}&nbsp;</td>\n\t\t\t\t</tr>\n\t\t\t);\n\t\t})\n\t\tif (this.props.gather.gatherers.length) {\n\t\t\treturn (\n\t\t\t\t<div className=\"panel-body\">\n\t\t\t\t\t<div className=\"panel panel-default\">\n\t\t\t\t\t\t<div className=\"panel-heading\">\n\t\t\t\t\t\t\t<h5 className=\"panel-title\">Roster</h5>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<table className=\"table roster-table\">\n\t\t\t\t\t\t\t<tbody>\n\t\t\t\t\t\t\t\t{gatherers}\n\t\t\t\t\t\t\t</tbody>\n\t\t\t\t\t\t</table>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t);\n\t\t} else {\n\t\t\treturn (<div className=\"panel-body text-center\"><JoinGatherButton buttonClass=\"btn-lg\" buttonName=\"Start a Gather\" /></div>);\n\t\t}\n\t}\n});\n\nvar socket;\n\nfunction initialiseComponents () {\n\tvar socketUrl = window.location.protocol + \"//\" + window.location.host;\n\tsocket = io(socketUrl)\n\t\t.on(\"connect\", function () {\n\t\t\tconsole.log(\"Connected\");\n\t\t})\n\t\t.on(\"reconnect\", function () {\n\t\t\tconsole.log(\"Reconnected\");\n\t\t})\n\t\t.on(\"disconnect\", function () {\n\t\t\tconsole.log(\"Disconnected\")\n\t\t});\n\n\tReact.render(<UserMenu />, document.getElementById('side-menu'));\n\tReact.render(<Chatroom />, document.getElementById('chatroom'));\n\tReact.render(<Gather />, document.getElementById('gathers'));\n\tReact.render(<CurrentUser />, document.getElementById('currentuser'));\n};\n\ninitialiseComponents();\n\n\n\n});\n\n"]}
\ No newline at end of file
diff --git a/public/js/client.js b/public/js/client.js
new file mode 100644
index 0000000..9732ed5
--- /dev/null
+++ b/public/js/client.js
@@ -0,0 +1,4 @@
+$(function () {
+ initialiseComponents();
+
+});
diff --git a/public/js/gather.js b/public/js/gather.js
new file mode 100644
index 0000000..88e6e3c
--- /dev/null
+++ b/public/js/gather.js
@@ -0,0 +1,282 @@
+"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 JoinGatherButton = React.createClass({displayName: "JoinGatherButton",
+ joinGather: function (e) {
+ e.preventDefault();
+ socket.emit("gather:join", {});
+ },
+ render: function () {
+ var message = this.props.buttonName || "Join Gather";
+ var buttonClass = "btn btn-primary";
+ if (this.props.buttonClass) {
+ buttonClass += " " + this.props.buttonClass;
+ }
+ return (React.createElement("button", {
+ onClick: this.joinGather,
+ className: buttonClass}, message))
+ }
+});
+
+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 picking 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;
+ var gatherState = this.props.gather.state;
+ if (gatherState === 'gathering' && this.props.gather.gatherers.length) {
+ progress = this.gatheringProgress();
+ } else if (gatherState === 'election') {
+ progress = this.electionProgress();
+ } else if (gatherState === 'selection') {
+ progress = this.selectionProgress();
+ }
+ if (progress) {
+ var style = {
+ width: Math.round((progress.num / progress.den * 100)) + "%"
+ };
+ return (
+ React.createElement("div", {className: "panel-body"},
+ React.createElement("p", null, React.createElement("strong", null, this.stateDescription()), " ", progress.message),
+ React.createElement("div", {className: "progress"},
+ React.createElement("div", {className: "progress-bar progress-bar-striped active",
+ "data-role": "progressbar",
+ "data-aria-valuenow": progress.num,
+ "data-aria-valuemin": "0",
+ "data-aria-valuemax": progress.den,
+ style: style}
+ )
+ )
+ )
+ );
+ } else {
+ return false;
+ }
+ }
+});
+
+var Gather = React.createClass({displayName: "Gather",
+ getDefaultProps: function () {
+ return {
+ gather: {
+ gatherers: []
+ }
+ }
+ },
+ joinedGather: function () {
+ var self = this;
+ return this.props.gather.gatherers.some(function (gatherer) {
+ return gatherer.user.id === self.props.currentUser.id;
+ });
+ },
+ componentDidMount: function () {
+ var self = this;
+ socket.on("gather:refresh", function (data) {
+ self.setProps({
+ gather: data.gather,
+ currentUser: data.currentUser
+ });
+ });
+ },
+ leaveGather: function (e) {
+ e.preventDefault();
+ socket.emit("gather:leave", {});
+ },
+ inviteToGather: function (e) {
+ e.preventDefault();
+ },
+ currentGatherer: function () {
+ var current = null;
+ var self = this;
+ this.props.gather.gatherers.forEach(function (gatherer) {
+ if (gatherer.id === self.props.currentUser.id) current = gatherer;
+ });
+ return current;
+ },
+ render: function () {
+ var joinButton;
+ if (this.joinedGather()) {
+ joinButton = (React.createElement("li", null, React.createElement("button", {
+ onClick: this.leaveGather,
+ className: "btn btn-danger"}, "Leave Gather")));
+ } else {
+ joinButton = (React.createElement("li", null, React.createElement(JoinGatherButton, null)));
+ }
+ 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 panel-default"},
+ React.createElement("div", {className: "panel-heading"},
+ React.createElement("strong", null, "NS2 Gather "),
+ React.createElement("span", {className: "badge add-left"}, this.props.gather.gatherers.length)
+ ),
+ React.createElement(Gatherers, {gather: this.props.gather, currentGatherer: this.currentGatherer()}),
+ React.createElement(GatherProgress, {gather: this.props.gather}),
+ React.createElement("div", {className: "panel-footer text-right"},
+ React.createElement("ul", {className: "list-inline"},
+ inviteButton,
+ joinButton
+ )
+ )
+ )
+ );
+ }
+});
+
+var Gatherers = React.createClass({displayName: "Gatherers",
+ render: function () {
+ var self = this;
+ var gatherers = this.props.gather.gatherers.map(function (gatherer) {
+ var lifeforms = (
+ gatherer.user.ability.lifeforms.map(function (lifeform) {
+ return (React.createElement("span", {className: "label label-default"}, lifeform));
+ })
+ );
+
+ var commBadge;
+ if (gatherer.user.ability.commander) {
+ commBadge = (React.createElement("img", {src: "/images/commander.png",
+ alt: "Commander",
+ height: "20",
+ width: "20"}));
+ }
+
+ var division = (React.createElement("span", {className: "label label-primary"}, gatherer.user.ability.division));
+ var action = lifeforms;
+ 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})
+ )
+ );
+ }
+
+ return (
+ React.createElement("tr", {key: gatherer.user.id},
+ React.createElement("td", {className: "col-md-1"}, commBadge),
+ React.createElement("td", {className: "col-md-5"}, gatherer.user.username),
+ React.createElement("td", {className: "col-md-3"}, division, " "),
+ 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("div", {className: "panel-heading"},
+ React.createElement("h5", {className: "panel-title"}, "Roster")
+ ),
+ React.createElement("table", {className: "table roster-table"},
+ React.createElement("tbody", null,
+ gatherers
+ )
+ )
+ )
+ )
+ );
+ } else {
+ return (React.createElement("div", {className: "panel-body text-center"}, React.createElement(JoinGatherButton, {buttonClass: "btn-lg", buttonName: "Start a Gather"})));
+ }
+ }
+});
+
+
+
+
+//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transformed.js","sources":[null],"names":[],"mappings":"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","sourcesContent":["\"use strict\";\n\nvar VoteButton = React.createClass({\n\tcancelVote: function (e) {\n\t\tsocket.emit(\"gather:vote\", {\n\t\t\tleader: {\n\t\t\t\tcandidate: null\n\t\t\t}\n\t\t});\n\t},\n\tvote: function (e) {\n\t\te.preventDefault();\n\t\tsocket.emit(\"gather:vote\", {\n\t\t\tleader: {\n\t\t\t\tcandidate: parseInt(e.target.value, 10)\n\t\t\t}\n\t\t});\n\t},\n\trender: function () {\n\t\tif (this.props.currentGatherer === null) {\n\t\t\treturn false;\n\t\t}\n\t\tif (this.props.currentGatherer.leaderVote === this.props.candidate.id) {\n\t\t\treturn (\n\t\t\t\t<button \n\t\t\t\t\tonClick={this.cancelVote} \n\t\t\t\t\tclassName=\"btn btn-xs btn-success\">Voted\n\t\t\t\t</button>\n\t\t\t);\n\t\t} else {\n\t\t\treturn (\n\t\t\t\t<button \n\t\t\t\t\tonClick={this.vote} \n\t\t\t\t\tclassName=\"btn btn-xs btn-default\"\n\t\t\t\t\tvalue={this.props.candidate.id}>Vote\n\t\t\t\t</button>\n\t\t\t);\n\t\t}\n\t}\n});\n\nvar JoinGatherButton = React.createClass({\n\tjoinGather: function (e) {\n\t\te.preventDefault();\n\t\tsocket.emit(\"gather:join\", {});\n\t},\n\trender: function () {\n\t\tvar message = this.props.buttonName || \"Join Gather\";\n\t\tvar buttonClass = \"btn btn-primary\";\n\t\tif (this.props.buttonClass) {\n\t\t\tbuttonClass += \" \" + this.props.buttonClass;\n\t\t}\n\t\treturn (<button \n\t\t\t\t\t\t\tonClick={this.joinGather} \n\t\t\t\t\t\t\tclassName={buttonClass}>{message}</button>)\n\t}\n});\n\nvar GatherProgress = React.createClass({\n\tstateDescription: function () {\n\t\tswitch(this.props.gather.state) {\n\t\t\tcase \"gathering\":\n\t\t\t\treturn \"Waiting for more gatherers.\";\n\t\t\tcase \"election\":\n\t\t\t\treturn \"Currently voting for team leaders.\";\n\t\t\tcase \"selection\":\n\t\t\t\treturn \"Waiting for leaders to picking teams.\";\n\t\t\tcase \"done\":\n\t\t\t\treturn \"Gather completed.\";\n\t\t\tdefault:\n\t\t\t\treturn \"Initialising gather.\";\n\t\t}\n\t},\n\tgatheringProgress: function () {\n\t\tvar num = this.props.gather.gatherers.length;\n\t\tvar den = 12;\n\t\tvar remaining = den - num;\n\t\tvar message = (remaining === 1) ? \"Waiting for last player\" : \"Waiting for \" + remaining + \" more players\";\n\t\treturn {\n\t\t\tnum: num,\n\t\t\tden: den,\n\t\t\tmessage: message\n\t\t};\n\t},\n\telectionProgress: function () {\n\t\tvar num = this.props.gather.gatherers.reduce(function (acc, gatherer) {\n\t\t\tif (gatherer.leaderVote) acc++;\n\t\t\treturn acc;\n\t\t}, 0);\n\t\tvar den = 12;\n\t\treturn {\n\t\t\tnum: num,\n\t\t\tden: den,\n\t\t\tmessage: den - num + \" more votes required\"\n\t\t};\n\t},\n\tselectionProgress: function () {\n\t\tvar num = this.props.gather.gatherers.reduce(function (acc, gatherer) {\n\t\t\tif (gatherer.team !== \"lobby\") acc++;\n\t\t\treturn acc;\n\t\t}, 0);\n\t\tvar den = 12;\n\n\t\treturn {\n\t\t\tnum: num,\n\t\t\tden: den,\n\t\t\tmessage: num + \" out of \" + den + \" players assigned\"\n\t\t};\n\t},\n\trender: function () {\n\t\tvar progress;\n\t\tvar gatherState = this.props.gather.state;\n\t\tif (gatherState === 'gathering' && this.props.gather.gatherers.length) {\n\t\t\tprogress = this.gatheringProgress();\n\t\t} else if (gatherState === 'election') {\n\t\t\tprogress = this.electionProgress();\n\t\t} else if (gatherState === 'selection') {\n\t\t\tprogress = this.selectionProgress();\n\t\t}\n\t\tif (progress) {\n\t\t\tvar style = {\n\t\t\t\twidth: Math.round((progress.num / progress.den * 100)) + \"%\"\n\t\t\t};\n\t\t\treturn (\n\t\t\t\t<div className=\"panel-body\">\n\t\t\t\t\t<p><strong>{this.stateDescription()}</strong> {progress.message}</p>\n\t\t\t\t\t<div className=\"progress\">\n\t\t\t\t\t  <div className=\"progress-bar progress-bar-striped active\" \n\t\t\t\t\t  \tdata-role=\"progressbar\" \n\t\t\t\t\t  \tdata-aria-valuenow={progress.num} \n\t\t\t\t\t  \tdata-aria-valuemin=\"0\" \n\t\t\t\t\t  \tdata-aria-valuemax={progress.den} \n\t\t\t\t\t  \tstyle={style}>\n\t\t\t\t\t  </div>\n\t\t\t\t  </div>\n\t\t\t\t</div>\n\t\t\t);\n\t\t} else {\n\t\t\treturn false;\n\t\t}\n\t}\n});\n\nvar Gather = React.createClass({\n\tgetDefaultProps: function () {\n\t\treturn {\n\t\t\tgather: {\n\t\t\t\tgatherers: []\n\t\t\t}\n\t\t}\n\t},\n\tjoinedGather: function () {\n\t\tvar self = this;\n\t\treturn this.props.gather.gatherers.some(function (gatherer) {\n\t\t\treturn gatherer.user.id === self.props.currentUser.id;\n\t\t});\n\t},\n\tcomponentDidMount: function () {\n\t\tvar self = this;\n\t\tsocket.on(\"gather:refresh\", function (data) {\n\t\t\tself.setProps({\n\t\t\t\tgather: data.gather,\n\t\t\t\tcurrentUser: data.currentUser\n\t\t\t});\n\t\t});\n\t},\n\tleaveGather: function (e) {\n\t\te.preventDefault();\n\t\tsocket.emit(\"gather:leave\", {});\n\t},\n\tinviteToGather: function (e) {\n\t\te.preventDefault();\n\t},\n\tcurrentGatherer: function () {\n\t\tvar current = null;\n\t\tvar self = this;\n\t\tthis.props.gather.gatherers.forEach(function (gatherer) {\n\t\t\tif (gatherer.id === self.props.currentUser.id) current = gatherer;\n\t\t});\n\t\treturn current;\n\t},\n\trender: function () {\n\t\tvar joinButton;\n\t\tif (this.joinedGather()) {\n\t\t\tjoinButton = (<li><button \n\t\t\t\t\t\t\tonClick={this.leaveGather} \n\t\t\t\t\t\t\tclassName=\"btn btn-danger\">Leave Gather</button></li>);\n\t\t} else {\n\t\t\tjoinButton = (<li><JoinGatherButton /></li>);\n\t\t}\n\t\tvar inviteButton;\n\t\tif (this.props.gather.state === 'gathering') {\n\t\t\tinviteButton = (<li><button\n\t\t\t\t\t\t\tonClick={this.inviteToGather}\n\t\t\t\t\t\t\tclassName=\"btn btn-primary\">Invite to Gather</button></li>);\n\t\t}\n\t\treturn (\n\t\t\t<div className=\"panel panel-default\">\n\t\t\t\t<div className=\"panel-heading\">\n\t\t\t\t\t<strong>NS2 Gather </strong>\n\t\t\t\t\t<span className=\"badge add-left\">{this.props.gather.gatherers.length}</span>\n\t\t\t\t</div>\n\t\t\t\t<Gatherers gather={this.props.gather} currentGatherer={this.currentGatherer()} />\n\t\t\t\t<GatherProgress gather={this.props.gather} />\n\t\t\t\t<div className=\"panel-footer text-right\">\n\t\t\t\t\t<ul className=\"list-inline\">\n\t\t\t\t\t\t{inviteButton}\n\t\t\t\t\t\t{joinButton}\n\t\t\t\t\t</ul>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t);\n\t}\n});\n\nvar Gatherers = React.createClass({\n\trender: function () {\n\t\tvar self = this;\n\t\tvar gatherers = this.props.gather.gatherers.map(function (gatherer) {\n\t\t\tvar lifeforms = (\n\t\t\t\tgatherer.user.ability.lifeforms.map(function (lifeform) {\n\t\t\t\t\treturn (<span className=\"label label-default\">{lifeform}</span>);\n\t\t\t\t})\n\t\t\t);\n\n\t\t\tvar commBadge;\n\t\t\tif (gatherer.user.ability.commander) {\n\t\t\t\tcommBadge = (<img src=\"/images/commander.png\" \n\t\t\t\t\t\t\talt=\"Commander\" \n\t\t\t\t\t\t\theight=\"20\"\n\t\t\t\t\t\t\twidth=\"20\" />);\n\t\t\t}\n\n\t\t\tvar division = (<span className=\"label label-primary\">{gatherer.user.ability.division}</span>);\n\t\t\tvar action = lifeforms;\n\t\t\tif (self.props.gather.state === \"election\") {\n\t\t\t\tvar votes = self.props.gather.gatherers.reduce(function (acc, voter) {\n\t\t\t\t\tif (voter.leaderVote === gatherer.id) acc++;\n\t\t\t\t\treturn acc;\n\t\t\t\t}, 0)\n\t\t\t\taction = (\n\t\t\t\t\t<span>\n\t\t\t\t\t\t<small>{votes + \" votes\"} &nbsp;</small>\n\t\t\t\t\t\t<VoteButton currentGatherer={self.props.currentGatherer} candidate={gatherer} />\n\t\t\t\t\t</span>\n\t\t\t\t);\n\t\t\t}\n\n\t\t\treturn (\n\t\t\t\t<tr key={gatherer.user.id}>\n\t\t\t\t\t<td className=\"col-md-1\">{commBadge}</td>\n\t\t\t\t\t<td className=\"col-md-5\">{gatherer.user.username}</td>\n\t\t\t\t\t<td className=\"col-md-3\">{division}&nbsp;</td>\n\t\t\t\t\t<td className=\"col-md-2 text-right\">{action}&nbsp;</td>\n\t\t\t\t</tr>\n\t\t\t);\n\t\t})\n\t\tif (this.props.gather.gatherers.length) {\n\t\t\treturn (\n\t\t\t\t<div className=\"panel-body\">\n\t\t\t\t\t<div className=\"panel panel-default\">\n\t\t\t\t\t\t<div className=\"panel-heading\">\n\t\t\t\t\t\t\t<h5 className=\"panel-title\">Roster</h5>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<table className=\"table roster-table\">\n\t\t\t\t\t\t\t<tbody>\n\t\t\t\t\t\t\t\t{gatherers}\n\t\t\t\t\t\t\t</tbody>\n\t\t\t\t\t\t</table>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t);\n\t\t} else {\n\t\t\treturn (<div className=\"panel-body text-center\"><JoinGatherButton buttonClass=\"btn-lg\" buttonName=\"Start a Gather\" /></div>);\n\t\t}\n\t}\n});\n\n\n\n"]}
\ No newline at end of file
diff --git a/public/js/init.js b/public/js/init.js
new file mode 100644
index 0000000..b4f148b
--- /dev/null
+++ b/public/js/init.js
@@ -0,0 +1,23 @@
+"use strict";
+
+var socket;
+
+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")
+ });
+
+ 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'));
+};
+//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhbnNmb3JtZWQuanMiLCJzb3VyY2VzIjpbbnVsbF0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFlBQVksQ0FBQzs7QUFFYixJQUFJLE1BQU0sQ0FBQzs7QUFFWCxTQUFTLG9CQUFvQixJQUFJO0NBQ2hDLElBQUksU0FBUyxHQUFHLE1BQU0sQ0FBQyxRQUFRLENBQUMsUUFBUSxHQUFHLElBQUksR0FBRyxNQUFNLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQztDQUN2RSxNQUFNLEdBQUcsRUFBRSxDQUFDLFNBQVMsQ0FBQztHQUNwQixFQUFFLENBQUMsU0FBUyxFQUFFLFlBQVk7R0FDMUIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxXQUFXLENBQUMsQ0FBQztHQUN6QixDQUFDO0dBQ0QsRUFBRSxDQUFDLFdBQVcsRUFBRSxZQUFZO0dBQzVCLE9BQU8sQ0FBQyxHQUFHLENBQUMsYUFBYSxDQUFDLENBQUM7R0FDM0IsQ0FBQztHQUNELEVBQUUsQ0FBQyxZQUFZLEVBQUUsWUFBWTtHQUM3QixPQUFPLENBQUMsR0FBRyxDQUFDLGNBQWMsQ0FBQztBQUM5QixHQUFHLENBQUMsQ0FBQzs7Q0FFSixLQUFLLENBQUMsTUFBTSxDQUFDLG9CQUFDLFFBQVEsRUFBQSxJQUFBLENBQUcsQ0FBQSxFQUFFLFFBQVEsQ0FBQyxjQUFjLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQztDQUNqRSxLQUFLLENBQUMsTUFBTSxDQUFDLG9CQUFDLFFBQVEsRUFBQSxJQUFBLENBQUcsQ0FBQSxFQUFFLFFBQVEsQ0FBQyxjQUFjLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQztDQUNoRSxLQUFLLENBQUMsTUFBTSxDQUFDLG9CQUFDLE1BQU0sRUFBQSxJQUFBLENBQUcsQ0FBQSxFQUFFLFFBQVEsQ0FBQyxjQUFjLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQztDQUM3RCxLQUFLLENBQUMsTUFBTSxDQUFDLG9CQUFDLFdBQVcsRUFBQSxJQUFBLENBQUcsQ0FBQSxFQUFFLFFBQVEsQ0FBQyxjQUFjLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQztDQUN0RSIsInNvdXJjZXNDb250ZW50IjpbIlwidXNlIHN0cmljdFwiO1xuXG52YXIgc29ja2V0O1xuXG5mdW5jdGlvbiBpbml0aWFsaXNlQ29tcG9uZW50cyAoKSB7XG5cdHZhciBzb2NrZXRVcmwgPSB3aW5kb3cubG9jYXRpb24ucHJvdG9jb2wgKyBcIi8vXCIgKyB3aW5kb3cubG9jYXRpb24uaG9zdDtcblx0c29ja2V0ID0gaW8oc29ja2V0VXJsKVxuXHRcdC5vbihcImNvbm5lY3RcIiwgZnVuY3Rpb24gKCkge1xuXHRcdFx0Y29uc29sZS5sb2coXCJDb25uZWN0ZWRcIik7XG5cdFx0fSlcblx0XHQub24oXCJyZWNvbm5lY3RcIiwgZnVuY3Rpb24gKCkge1xuXHRcdFx0Y29uc29sZS5sb2coXCJSZWNvbm5lY3RlZFwiKTtcblx0XHR9KVxuXHRcdC5vbihcImRpc2Nvbm5lY3RcIiwgZnVuY3Rpb24gKCkge1xuXHRcdFx0Y29uc29sZS5sb2coXCJEaXNjb25uZWN0ZWRcIilcblx0XHR9KTtcblxuXHRSZWFjdC5yZW5kZXIoPFVzZXJNZW51IC8+LCBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnc2lkZS1tZW51JykpO1xuXHRSZWFjdC5yZW5kZXIoPENoYXRyb29tIC8+LCBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnY2hhdHJvb20nKSk7XG5cdFJlYWN0LnJlbmRlcig8R2F0aGVyIC8+LCBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnZ2F0aGVycycpKTtcblx0UmVhY3QucmVuZGVyKDxDdXJyZW50VXNlciAvPiwgZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ2N1cnJlbnR1c2VyJykpO1xufTsiXX0=
\ No newline at end of file
diff --git a/public/js/message.js b/public/js/message.js
new file mode 100644
index 0000000..6fce96a
--- /dev/null
+++ b/public/js/message.js
@@ -0,0 +1,148 @@
+"use strict";
+
+var Chatroom = React.createClass({displayName: "Chatroom",
+ getDefaultProps: function () {
+ return {
+ history: []
+ };
+ },
+ componentDidMount: function () {
+ var self = this;
+ var TIMER_INTERVAL = 60000; // 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 () {
+ if (self.refs.messages) self.refs.messages.refreshTime();
+ }, 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, {
+ avatar: message.author.avatar,
+ username: message.author.username,
+ content: message.content,
+ ref: "messages",
+ createdAt: message.createdAt})
+ );
+ });
+ 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 ChatMessage = React.createClass({displayName: "ChatMessage",
+ getInitialState: function () {
+ return {
+ timeAgo: $.timeago(this.props.createdAt)
+ }
+ },
+ refreshTime: function () {
+ var self = this;
+ self.setState({
+ timeAgo: $.timeago(self.props.createdAt)
+ });
+ },
+ render: function () {
+ return (
+ React.createElement("li", {className: "left clearfix"},
+ React.createElement("span", {className: "chat-img pull-left"},
+ React.createElement("img", {
+ src: this.props.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.username),
+ React.createElement("small", {className: "pull-right text-muted"},
+ React.createElement("i", {className: "fa fa-clock-o fa-fw"}), " ", this.state.timeAgo
+ )
+ ),
+ React.createElement("p", null, this.props.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"})
+ )
+ )
+ )
+ );
+ }
+});
+
+//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transformed.js","sources":[null],"names":[],"mappings":"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","sourcesContent":["\"use strict\";\n\nvar Chatroom = React.createClass({\n\tgetDefaultProps: function () {\n\t\treturn {\n\t\t\thistory: []\n\t\t};\n\t},\n\tcomponentDidMount: function () {\n\t\tvar self = this;\n\t\tvar TIMER_INTERVAL = 60000; // Every minute\n\n\t\tsocket.on(\"message:new\", function (data) {\n\t\t\tvar history = self.props.history;\n\t\t\thistory.push(data);\n\t\t\tself.setProps({\n\t\t\t\thistory: history\n\t\t\t});\n\t\t\tself.scrollToBottom();\n\t\t});\n\n\t\t// Message History Retrieved\n\t\tsocket.on(\"message:refresh\", function (data) {\n\t\t\tself.setProps({\n\t\t\t\thistory: data.chatHistory\n\t\t\t});\n\t\t\tself.scrollToBottom();\n\t\t});\n\n\t\tsocket.emit(\"message:refresh\", {});\n\n\t\tself.timer = setInterval(function () {\n\t\t\tif (self.refs.messages) self.refs.messages.refreshTime();\n\t\t}, TIMER_INTERVAL);\n\t},\n\n\tcomponentDidUnmount: function () {\n\t\tclearInterval(this.timer);\n\t},\n\tsendMessage: function (message) {\n\t\tsocket.emit(\"newMessage\", {message: message});\n\t},\n\tscrollToBottom: function () {\n\t\tvar node = React.findDOMNode(this.refs.messageContainer);\n\t  node.scrollTop = node.scrollHeight;\n\t},\n\trender: function () {\n\t\tvar messages = this.props.history.map(function (message) {\n\t\t\treturn (\n\t\t\t\t<ChatMessage \n\t\t\t\t\tavatar={message.author.avatar} \n\t\t\t\t\tusername={message.author.username}\n\t\t\t\t\tcontent={message.content}\n\t\t\t\t\tref=\"messages\"\n\t\t\t\t\tcreatedAt={message.createdAt} />\n\t\t\t);\n\t\t});\n\t\treturn (\n\t\t\t<div className=\"panel panel-default\">\n\t\t\t\t<div className=\"panel-heading\">Gather Chat</div>\n\t\t\t\t<div className=\"panel-body\">\n\t\t\t\t\t<ul className=\"chat\" id=\"chatmessages\" ref=\"messageContainer\">\n\t\t\t\t\t\t{messages}\n\t\t\t\t\t</ul>\n\t\t\t\t</div>\n\t\t\t\t<div className=\"panel-footer\">\n\t\t\t\t\t<MessageBar />\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t);\n\t}\n});\n\nvar ChatMessage = React.createClass({\n\tgetInitialState: function () {\n\t\treturn {\n\t\t\ttimeAgo: $.timeago(this.props.createdAt)\n\t\t}\n\t},\n\trefreshTime: function () {\n\t\tvar self = this;\n\t\tself.setState({\n\t\t\ttimeAgo: $.timeago(self.props.createdAt)\n\t\t});\n\t},\n\trender: function () {\n\t\treturn (\n\t\t\t<li className=\"left clearfix\">\n\t\t\t\t<span className=\"chat-img pull-left\">\n\t\t\t\t\t\t<img \n\t\t\t\t\t\t\tsrc={this.props.avatar} \n\t\t\t\t\t\t\talt=\"User Avatar\" \n\t\t\t\t\t\t\theight=\"40\"\n\t\t\t\t\t\t\twidth=\"40\"\n\t\t\t\t\t\t\tclassName=\"img-circle\" />\n\t\t\t\t</span>\n\t\t\t\t<div className=\"chat-body clearfix\">\n\t\t\t\t\t<div className=\"header\">\n\t\t\t\t\t\t<strong className=\"primary-font\">{this.props.username}</strong>\n\t\t\t\t\t\t<small className=\"pull-right text-muted\">\n\t\t\t\t\t\t\t<i className=\"fa fa-clock-o fa-fw\"></i> {this.state.timeAgo}\n\t\t\t\t\t\t</small>\n\t\t\t\t\t</div>\n\t\t\t\t\t<p>{this.props.content}</p>\n\t\t\t\t</div>\n\t\t\t</li>\n\t\t);\n\t}\n});\n\nvar MessageBar = React.createClass({\n\tsendMessage: function (content) {\n\t\tsocket.emit(\"message:new\", {\n\t\t\tcontent: content\n\t\t});\n\t},\n\thandleSubmit: function (e) {\n\t\te.preventDefault();\n\t\tvar content = React.findDOMNode(this.refs.content).value.trim();\n\t\tif (!content) return;\n\t\tReact.findDOMNode(this.refs.content).value = '';\n\t\tthis.sendMessage(content);\n\t\treturn;\n\t},\n\trender: function () {\n\t\treturn (\n\t\t\t<form onSubmit={this.handleSubmit} >\n\t\t\t\t<div className=\"input-group\">\n\t\t\t\t\t<input \n\t\t\t\t\t\tid=\"btn-input\" \n\t\t\t\t\t\ttype=\"text\" \n\t\t\t\t\t\tclassName=\"form-control\" \n\t\t\t\t\t\tref=\"content\"\n\t\t\t\t\t\tplaceholder=\"Be polite please...\" />\n\t\t\t\t\t<span className=\"input-group-btn\">\n\t\t\t\t\t\t<input \n\t\t\t\t\t\t\ttype=\"submit\" \n\t\t\t\t\t\t\tclassName=\"btn btn-primary\" \n\t\t\t\t\t\t\tid=\"btn-chat\" \n\t\t\t\t\t\t\tvalue=\"Send\" />\n\t\t\t\t\t</span>\n\t\t\t\t</div>\n\t\t\t</form>\n\t\t);\n\t}\n});\n"]}
\ No newline at end of file
diff --git a/public/js/user.js b/public/js/user.js
new file mode 100644
index 0000000..4bea783
--- /dev/null
+++ b/public/js/user.js
@@ -0,0 +1,114 @@
+"use strict";
+
+var UserCounter = React.createClass({displayName: "UserCounter",
+ render: function () {
+ return (
+ 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.count, " ")
+ )
+ )
+ );
+ }
+});
+
+var UserLogin = React.createClass({displayName: "UserLogin",
+ authorizeId: function (id) {
+ id = parseInt(id, 10);
+ socket.emit("users:authorize", {
+ id: id
+ });
+ },
+ 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);
+ return;
+ },
+ 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 {
+ count: 0,
+ users: []
+ };
+ },
+ componentDidMount: function () {
+ socket.on('users:update', this.updateUsers);
+ },
+ updateUsers: function (data) {
+ this.setProps({
+ count: data.count,
+ 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(UserCounter, React.__spread({}, this.props)),
+ users,
+ React.createElement("li", null, React.createElement("br", null), React.createElement(UserLogin, null), React.createElement("br", null))
+ )
+ );
+ }
+});
+
+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("a", {href: "#"}, this.props.user.username, " ", React.createElement("img", {src: this.props.user.avatar,
+ alt: "User Avatar",
+ height: "20",
+ width: "20"}))
+ );
+ } else {
+ return false;
+ }
+ }
+});
+
+alert("foo")
+//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transformed.js","sources":[null],"names":[],"mappings":"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","sourcesContent":["\"use strict\";\n\nvar UserCounter = React.createClass({\n\trender: function () {\n\t\treturn (\n\t\t\t<li>\n\t\t\t\t<a href=\"#\">\n\t\t\t\t\t<i className=\"fa fa-users fa-fw\"></i> Online \n\t\t\t\t\t<span className=\"badge add-left\"> {this.props.count} </span>\n\t\t\t\t</a>\n\t\t\t</li>\n\t\t);\n\t}\n});\n\nvar UserLogin = React.createClass({\n\tauthorizeId: function (id) {\n\t\tid = parseInt(id, 10);\n\t\tsocket.emit(\"users:authorize\", {\n\t\t\tid: id\n\t\t});\n\t},\n\thandleSubmit: function (e) {\n\t\te.preventDefault();\n\t\tvar id = React.findDOMNode(this.refs.authorize_id).value.trim();\n\t\tif (!id) return;\n\t\tReact.findDOMNode(this.refs.authorize_id).value = '';\n\t\tthis.authorizeId(id);\n\t\treturn;\n\t},\n\trender: function () {\n\t\treturn (\n\t\t\t<form onSubmit={this.handleSubmit} >\n\t\t\t\t<div className=\"input-group signin\">\n\t\t\t\t\t<input \n\t\t\t\t\t\tid=\"btn-input\" \n\t\t\t\t\t\ttype=\"text\" \n\t\t\t\t\t\tclassName=\"form-control\" \n\t\t\t\t\t\tref=\"authorize_id\"\n\t\t\t\t\t\tplaceholder=\"Choose an ID...\" />\n\t\t\t\t\t<span className=\"input-group-btn\">\n\t\t\t\t\t\t<input \n\t\t\t\t\t\t\ttype=\"submit\" \n\t\t\t\t\t\t\tclassName=\"btn btn-primary\" \n\t\t\t\t\t\t\tid=\"btn-chat\" \n\t\t\t\t\t\t\tvalue=\"Login\" />\n\t\t\t\t\t</span>\n\t\t\t\t</div>\n\t\t\t\t<div className=\"signin\">\n\t\t\t\t<p className=\"text-center\"><small>Just a temporary measure until genuine authentication is implemented</small></p>\n\t\t\t\t</div>\n\t\t\t</form>\n\t\t);\n\t}\n})\n\nvar UserMenu = React.createClass({\n\tgetDefaultProps: function () {\n\t\treturn {\n\t\t\tcount: 0,\n\t\t\tusers: []\n\t\t};\n\t},\n\tcomponentDidMount: function () {\n\t\tsocket.on('users:update', this.updateUsers);\n\t},\n\tupdateUsers: function (data) {\n\t\tthis.setProps({\n\t\t\tcount: data.count,\n\t\t\tusers: data.users\n\t\t});\n\t},\n\trender: function () {\n\t\tvar users = this.props.users.map(function (user) {\n\t\t\treturn (\n\t\t\t\t<li key={user.id}><a href=\"#\">{user.username}</a></li>\n\t\t\t);\n\t\t});\n\t\treturn (\n\t\t\t<ul className=\"nav\" id=\"side-menu\">\n\t\t\t\t<UserCounter {...this.props} />\n\t\t\t\t{users}\n\t\t\t\t<li><br /><UserLogin /><br /></li>\n\t\t\t</ul>\n\t\t);\n\t}\n});\n\nvar CurrentUser = React.createClass({\n\tcomponentDidMount: function () {\n\t\tvar self = this;\n\t\tsocket.on(\"users:update\", function (data) {\n\t\t\tself.setProps({\n\t\t\t\tuser: data.currentUser\n\t\t\t});\n\t\t});\n\t\tsocket.emit(\"users:refresh\", {});\n\t},\n\trender: function () {\n\t\tif (this.props.user) {\n\t\t\treturn (\n\t\t\t\t<a href=\"#\">{this.props.user.username} &nbsp;<img src={this.props.user.avatar}\n\t\t\t\t\talt=\"User Avatar\" \n\t\t\t\t\theight=\"20\"\n\t\t\t\t\twidth=\"20\" /></a>\n\t\t\t);\n\t\t} else {\n\t\t\treturn false;\n\t\t}\n\t}\n});\n\nalert(\"foo\")"]}
\ No newline at end of file
diff --git a/spec/gather.js b/spec/gather.js
index 530b590..f27c7bd 100644
--- a/spec/gather.js
+++ b/spec/gather.js
@@ -147,6 +147,7 @@ describe("Gather Model:", function () {
});
});
});
+
describe("addUser", function () {
it ("adds gatherer to lobby", function () {
gather.addUser(user);
@@ -159,6 +160,7 @@ describe("Gather Model:", function () {
assert.equal(gather.gatherers.length, 1);
});
});
+
describe("removeUser", function () {
it ("removes gatherer altogether", function () {
gather.addUser(user);
@@ -168,6 +170,7 @@ describe("Gather Model:", function () {
assert.equal(gather.gatherers.length, 0);
});
});
+
describe("moveToMarine", function () {
it ("moves a player to marine", function () {
gather.addUser(user);
@@ -183,6 +186,7 @@ describe("Gather Model:", function () {
});
});
});
+
describe("moveToAlien", function () {
it ("moves a player to alien", function () {
gather.addUser(user);
@@ -198,6 +202,7 @@ describe("Gather Model:", function () {
});
});
});
+
describe("moveToLobby", function () {
it ("moves a player to lobby", function () {
gather.addUser(user);
@@ -208,6 +213,7 @@ describe("Gather Model:", function () {
assert.equal(gather.lobby()[0].id, user.id);
});
});
+
describe("aliens", function () {
it ("returns all gatherers in aliens", function () {
gather.addUser(user);
@@ -215,6 +221,7 @@ describe("Gather Model:", function () {
assert.equal(gather.aliens().length, 1);
});
});
+
describe("marines", function () {
it ("returns all gatherers in marines", function () {
gather.addUser(user);
@@ -222,12 +229,14 @@ describe("Gather Model:", function () {
assert.equal(gather.marines().length, 1);
});
});
+
describe("lobby", function () {
it ("returns all gatherers in lobby", function () {
gather.addUser(user);
assert.equal(gather.lobby().length, 1);
});
});
+
describe("toJson", function () {
it ("returns a json representation of the gather instance", function () {
var output = gather.toJson();
@@ -235,6 +244,7 @@ describe("Gather Model:", function () {
assert.isString(output.state);
});
});
+
describe("leaderVotes", function () {
beforeEach(function () {
gatherers.forEach(function (user) {
@@ -261,6 +271,7 @@ describe("Gather Model:", function () {
assert.equal(gather.leaderVotes().length, 0);
});
});
+
describe("voteForLeader", function () {
beforeEach(function () {
gatherers.forEach(function (user) {
@@ -289,6 +300,7 @@ describe("Gather Model:", function () {
assert.equal(votes[0], secondCandidate.id);
});
});
+
describe("alienLeader", function () {
beforeEach(function () {
gatherers.forEach(function (gatherer) {
@@ -304,6 +316,7 @@ describe("Gather Model:", function () {
assert.isUndefined(gather.alienLeader());
});
});
+
describe("marineLeader", function () {
beforeEach(function () {
gatherers.forEach(function (gatherer) {
@@ -319,6 +332,7 @@ describe("Gather Model:", function () {
assert.isUndefined(gather.marineLeader());
});
});
+
describe("assignMarineLeader", function () {
it ("assigns a marine leader", function () {
gather.addUser(user);
@@ -327,6 +341,7 @@ describe("Gather Model:", function () {
assert.equal(leader.id, user.id);
});
});
+
describe("assignAlienLeader", function () {
it ("assigns an alien leader", function () {
gather.addUser(user);
@@ -335,6 +350,21 @@ describe("Gather Model:", function () {
assert.equal(leader.id, user.id);
});
});
+
+ describe("getGatherer", function () {
+ beforeEach(function () {
+ gather.addGatherer(user);
+ });
+ it ("returns a gatherer given a user", function () {
+ var gatherer = gather.getGatherer(user);
+ assert.equal(gatherer.id, user.id);
+ });
+ it ("returns null if user is not a gatherer", function () {
+ var gatherer = gather.getGatherer(gatherers[0]);
+ assert.isNull(gatherer);
+ });
+ });
+
describe("confirmTeam", function () {
var leader;
beforeEach(function () {
diff --git a/views/partials/foot.hbs b/views/partials/foot.hbs
index 13d57a5..806d4e9 100644
--- a/views/partials/foot.hbs
+++ b/views/partials/foot.hbs
@@ -1 +1,2 @@
-
\ No newline at end of file
+
+
\ No newline at end of file