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

    Just a temporary measure until genuine authentication is implemented

    -
    -
    - ); - } -}) - -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 ( -
    -
    Gather Chat
    -
    -
      - {messages} -
    -
    -
    - -
    -
    - ); - } -}); - -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 ( -
  • - - User Avatar - -
    -
    - {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}  User Avatar - ); - } 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 = (Commander); - } - - 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 ( -
    -
    -
    -
    Roster
    -
    - - - {gatherers} - -
    -
    -
    - ); - } 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 = (Commander); + } + return ( + + {image} + {gatherer.user.username} + + ); + } + var marines = this.marineGatherers().map(extractGatherer); + var aliens = this.alienGatherers().map(extractGatherer); + return ( +
    +
    +
    +
    +
    + Aliens +
    + + + {aliens} + +
    +
    +
    +
    +
    +
    + Marines +
    + + + {marines} + +
    +
    +
    +
    +
    + ); + } +}) + +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 ( +
    +
    +
    +
    Roster
    +
    + + + {gatherers} + +
    +
    +
    + ); + } 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 ( +
    +
    Gather Chat
    +
    +
      + {messages} +
    +
    +
    + +
    +
    + ); + } +}); + +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 ( +
  • + + User Avatar + +
    +
    + {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 ( +
    +
    + + + + +
    +
    +

    Just a temporary measure until genuine authentication is implemented

    +
    +
    + ); + } +}) + +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 ( +
      + + {users} +


    • +
    + ); + } +}); + +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}  User Avatar + ); + } 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