diff --git a/client/lib/avalon.css b/client/lib/avalon.css index 4d2280d..08f759c 100644 --- a/client/lib/avalon.css +++ b/client/lib/avalon.css @@ -101,6 +101,11 @@ ul.bar_tabs > li { width: auto; } +.btn, +.btn:hover { + border-color: transparent; +} + /* * Modified */ diff --git a/imports/api/groups/groups.jsx b/imports/api/groups/groups.jsx index 7c823d1..20ec244 100644 --- a/imports/api/groups/groups.jsx +++ b/imports/api/groups/groups.jsx @@ -50,11 +50,11 @@ TeamsSchema = new SimpleSchema({ // Index of leader's id which is included in `players.id`, corresponds to the index of this team in `missions.teams` memberIndices: { type: [Number], defaultValue: [] }, // Indices of players who were selected to be sent out on the mission, correspond to `players.id` approvals: { type: [Boolean], defaultValue: [] }, // Indicate players whether to approve the mission team make-up or not, with indices correspond to `players.id` - successVotes: { type: [Boolean], defaultValue: [] }, // Indicate members whether to vote for the mission to success or not, with indices correspond to `missions.teams.memberIndices` }); MissionsSchema = new SimpleSchema({ teams: { type: [TeamsSchema], defaultValue: [] }, + votes: { type: [Boolean], defaultValue: [] }, // Indicate members whether to vote for the mission to success or not, with indices correspond to `missions.teams.memberIndices` }); MessagesSchema = new SimpleSchema({ @@ -67,6 +67,7 @@ Groups.schema = new SimpleSchema({ ownerId: { type: String, regEx: SimpleSchema.RegEx.Id }, // Owner's id name: { type: String }, players: { type: [PlayersSchema], defaultValue: [] }, // Group players, include the owner + firstLeaderIndex: { type: Number, defaultValue: 0 }, missions: { type: [MissionsSchema], defaultValue: [] }, // Mission proposals guessMerlin: { type: Boolean, optional: true }, // Indicate whether Assassin correctly guesses Merlin's identity or not messages: { type: [MessagesSchema], defaultValue: [] }, @@ -75,24 +76,22 @@ Groups.schema = new SimpleSchema({ Groups.attachSchema(Groups.schema); -// This represents the keys from Groups objects that should be published -// to the client. If we add secret properties to Group objects, don't list -// them here to keep them private to the server. -Groups.publicFieldsWhenFindAll = { - ownerId: 1, - name: 1, - 'players.id': 1, - 'missions.length': 1, -}; - -// TODO: Remove secret properties to keep them private -Groups.publicFieldsWhenFindOne = { - ownerId: 1, - name: 1, - players: 1, - missions: 1, - guessMerlin: 1, - messages: 1, +Groups.publicFields = { + findAll: { + ownerId: 1, + name: 1, + 'players.id': 1, + 'missions.length': 1, + }, + findOne: { // TODO: Remove secret properties to keep them private + ownerId: 1, + name: 1, + players: 1, + firstLeaderIndex: 1, + missions: 1, + guessMerlin: 1, + messages: 1, + } }; Groups.helpers({ @@ -118,35 +117,41 @@ Groups.helpers({ isPlaying() { return this.missions.length != 0; }, + getLastMission() { // Private use only + return this.missions[this.missions.length - 1]; + }, getLastTeam() { // Force return `null` if missions list or teams list is empty (instead of `undefined`) - const lastMission = this.missions[this.missions.length - 1]; + const lastMission = this.getLastMission(); return (lastMission || null) && lastMission.teams && lastMission.teams[lastMission.teams.length - 1] || null; }, getTeamsCount() { return this.missions.reduce((c, m) => c + m.teams.length, 0); }, getLastTeamsCount() { - return this.missions[this.missions.length - 1].teams.length; + return this.getLastMission().teams.length; }, findRequiredFailVotesCount(missionIndex) { return missionIndex == 3 && this.players.length >= 7 ? 2 : 1; }, getSummaries() { - return this.missions.map((m, i) => - m.teams.map((t, j) => { - const summary = { memberIndices: t.memberIndices, denierIndices: [], failVotesCount: null, result: undefined }; + let lastMissionsTeamsCount = 0; + return this.missions.map((m, i) => { + const mission = m.teams.map((t, j) => { + const team = { leaderIndex: (this.firstLeaderIndex + lastMissionsTeamsCount + j) % this.players.length, memberIndices: t.memberIndices, denierIndices: [], failVotesCount: null, result: undefined }; if (t.memberIndices.length != 0 && t.approvals.indexOf(null) == -1) { - summary.denierIndices = t.approvals.map((a, i) => a ? -1 : i).filter(i => i >= 0); + team.denierIndices = t.approvals.map((a, i) => a ? -1 : i).filter(i => i >= 0); if (j < Groups.MISSION_TEAMS_COUNT - 1 && t.approvals.filter(a => !a).length >= t.approvals.filter(a => a).length) { // Denied, Hammer - summary.result = null; - } else if (t.successVotes.indexOf(null) == -1) { - summary.failVotesCount = t.successVotes.filter(a => !a).length; - summary.result = summary.failVotesCount < this.findRequiredFailVotesCount(i) ? true : false; + team.result = null; + } else if (m.votes.indexOf(null) == -1) { + team.failVotesCount = m.votes.filter(a => !a).length; + team.result = team.failVotesCount < this.findRequiredFailVotesCount(i) ? true : false; } } - return summary; - }) - ); + return team; + }); + lastMissionsTeamsCount += m.teams.length; + return mission; + }); }, startNewMission() { const summaryResults = this.getSummaries().map(m => m[m.length - 1].result); @@ -157,7 +162,7 @@ Groups.helpers({ this.startGuessingMerlin(); return; } - const newMission = { teams: [] }; + const newMission = { teams: [], votes: [] }; Groups.update(this._id, { $push: { missions: newMission } }); @@ -172,10 +177,10 @@ Groups.helpers({ // const memberIndices = _.shuffle(Array.from(new Array(this.players.length), (_, i) => i)).slice(0, Groups.MISSIONS_MEMBERS_COUNT[this.players.length][this.missions.length - 1]); // TEST const memberIndices = []; Groups.update(this._id, { - $push: { [`missions.${this.missions.length - 1}.teams`]: { memberIndices: memberIndices, approvals: [], successVotes: [] } } + $push: { [`missions.${this.missions.length - 1}.teams`]: { memberIndices: memberIndices, approvals: [] } } }); // FIXME: Remove redundancy (@ref 'methods' `selectedMembers`) - this.missions[this.missions.length - 1].teams.push({ memberIndices: memberIndices, approvals: [], successVotes: [] }); // Update local variable + this.getLastMission().teams.push({ memberIndices: memberIndices, approvals: [] }); // Update local variable if (!this.isSelectingMembers()) { this.startWaitingForApproval(); } @@ -195,23 +200,29 @@ Groups.helpers({ } }, startWaitingForVote() { - const lastMission = this.missions[this.missions.length - 1]; - // const successVotes = Array.from(new Array(Groups.MISSIONS_MEMBERS_COUNT[this.players.length][this.missions.length - 1]), (_, i) => Groups.ROLES[this.players[lastMission.teams[lastMission.teams.length - 1].memberIndices[i]].role].side ? true : Math.random() > 0.3 ? true : false); // TEST - const successVotes = Array.from(new Array(Groups.MISSIONS_MEMBERS_COUNT[this.players.length][this.missions.length - 1]), (_, i) => Groups.ROLES[this.players[lastMission.teams[lastMission.teams.length - 1].memberIndices[i]].role].side ? null : null); + const lastMission = this.getLastMission(); + // const votes = Array.from(new Array(Groups.MISSIONS_MEMBERS_COUNT[this.players.length][this.missions.length - 1]), (_, i) => Groups.ROLES[this.players[lastMission.teams[lastMission.teams.length - 1].memberIndices[i]].role].side ? true : Math.random() > 0.3 ? true : false); // TEST + const votes = Array.from(new Array(Groups.MISSIONS_MEMBERS_COUNT[this.players.length][this.missions.length - 1]), (_, i) => Groups.ROLES[this.players[lastMission.teams[lastMission.teams.length - 1].memberIndices[i]].role].side ? null : null); Groups.update(this._id, { - $set: { [`missions.${this.missions.length - 1}.teams.${lastMission.teams.length - 1}.successVotes`]: successVotes } + $set: { [`missions.${this.missions.length - 1}.votes`]: votes } }); // FIXME: Remove redundancy (@ref 'methods' `vote`) - this.getLastTeam().successVotes = successVotes; // Update local variable + this.getLastMission().votes = votes; // Update local variable if (!this.isWaitingForVote()) { this.startNewMission(); } }, startGuessingMerlin() { + // const guessMerlin = Math.random() > 0.5 ? true : false; // TEST + const guessMerlin = null; Groups.update(this._id, { - $set: { guessMerlin: null } - // $set: { guessMerlin: Math.random() > 0.5 ? true : false } // TEST + $set: { guessMerlin: guessMerlin } }); + // FIXME: Remove redundancy (@ref 'methods' `guess`) + this.guessMerlin = guessMerlin; //Update local variable + if (this.guessMerlin != null) { + this.finish(); + } }, finish() { const playerActivities = []; @@ -239,13 +250,13 @@ Groups.helpers({ } }, getLeader() { - return this.missions.length > 0 ? Meteor.users.findOne(this.players[(this.getTeamsCount() - 1) % this.players.length].id) : null; + return this.missions.length > 0 ? Meteor.users.findOne(this.players[(this.firstLeaderIndex + this.getTeamsCount() - 1) % this.players.length].id) : null; }, hasLeader(userId) { - return this.missions.length > 0 ? this.players[(this.getTeamsCount() - 1) % this.players.length].id == userId : false; + return this.missions.length > 0 ? this.players[(this.firstLeaderIndex + this.getTeamsCount() - 1) % this.players.length].id == userId : false; }, hasHammer(userId) { - return this.missions.length > 0 ? this.players[(this.getTeamsCount() - this.getLastTeamsCount() + Groups.MISSION_TEAMS_COUNT - 1) % this.players.length].id == userId : false; + return this.missions.length > 0 ? this.players[(this.firstLeaderIndex + this.getTeamsCount() - this.getLastTeamsCount() + Groups.MISSION_TEAMS_COUNT - 1) % this.players.length].id == userId : false; }, hasMember(userId) { return this.getLastTeam() != null && this.getLastTeam().memberIndices.find(i => this.players[i].id == userId) != undefined; @@ -266,11 +277,17 @@ Groups.helpers({ return this.getLastTeam() != null && this.getLastTeam().approvals.indexOf(null) == -1 && this.getLastTeam().approvals.filter(a => !a).length >= this.getLastTeam().approvals.filter(a => a).length; }, isWaitingForVote() { - return this.getLastTeam() != null && this.getLastTeam().successVotes.indexOf(null) != -1; + return this.getLastMission() != null && this.getLastMission().votes.indexOf(null) != -1; }, isGuessingMerlin() { return this.guessMerlin === null; }, + checkPlayerHasApproved(userId) { + return this.getLastTeam() != null && this.getLastTeam().approvals[this.players.map(p => p.id).indexOf(userId)] != null; + }, + checkMemberHasVoted(userId) { + return this.getLastMission() != null && this.getLastMission().votes[this.getLastTeam().memberIndices.indexOf(this.players.map(p => p.id).indexOf(userId))] != null; + }, getSituation() { const situation = { status: '', slot: undefined, result: undefined }; if (!this.isPlaying()) { @@ -312,7 +329,7 @@ Groups.helpers({ }, findInformation(userId, playerId) { const indices = this.players.map(p => p.id); - const userRole = this.players[indices.indexOf(userId)].role; + const userRole = this.players[indices.indexOf(userId)] && this.players[indices.indexOf(userId)].role || 'Undecided'; const playerIndex = indices.indexOf(playerId); const playerRole = this.players[playerIndex].role; const otherPlayer = userId != playerId; @@ -321,10 +338,10 @@ Groups.helpers({ let status = ''; if (this.isWaitingForApproval()) { const approval = this.getLastTeam().approvals[playerIndex]; - status = approval == null ? 'Undecided' : approval ? 'Approved' : 'Denied'; + status = approval === undefined ? '' : otherPlayer || approval == null ? 'Waiting' : approval ? 'Approved' : 'Denied'; } if (this.isWaitingForVote()) { - const vote = this.getLastTeam().successVotes[this.getLastTeam().memberIndices.indexOf(playerIndex)]; + const vote = this.getLastMission().votes[this.getLastTeam().memberIndices.indexOf(playerIndex)]; status = vote === undefined ? '' : otherPlayer || vote == null ? 'Waiting' : vote ? 'Voted Success' : 'Voted Fail'; } return { role: role, side: side, status: status }; @@ -354,7 +371,7 @@ Groups.helpers({ {approval == null ? '' : `(You ${approval ? 'approved' : 'denied'})`}

; } else if (this.isWaitingForVote() && this.hasMember(userId)) { - const vote = this.getLastTeam().successVotes[this.getLastTeam().memberIndices.indexOf(this.players.map(p => p.id).indexOf(userId))]; + const vote = this.getLastMission().votes[this.getLastTeam().memberIndices.indexOf(this.players.map(p => p.id).indexOf(userId))]; suggestion =

Press buttons to vote for the mission success or fail diff --git a/imports/api/groups/methods.js b/imports/api/groups/methods.js index df29dca..3c7ca60 100644 --- a/imports/api/groups/methods.js +++ b/imports/api/groups/methods.js @@ -88,13 +88,13 @@ export const start = new ValidatedMethod({ const minions = Array.from(new Array(evilPlayersCount - roles.filter(r => !Groups.ROLES[r].side).length)).map(() => 'Minion'); _.shuffle(roles.concat(servants, minions)).forEach((r, i) => players[i].role = r); Groups.update(groupId, { - $set: { players: players } + $set: { players: players, firstLeaderIndex: Math.floor(Math.random() * playersCount) } }); group = Groups.findOne(groupId); // Update local variable group.startNewMission(); } else { Groups.update(groupId, { - $set: { players: group.players.map(player => ({ id: player.id, role: 'Undecided' })), messages: [] }, + $set: { players: group.players.map(player => ({ id: player.id, role: 'Undecided' })), firstLeaderIndex: 0, messages: [] }, $unset: { guessMerlin: 1 } }); } @@ -149,7 +149,7 @@ export const vote = new ValidatedMethod({ run({ groupId, success }) { let group = Groups.findOne(groupId); Groups.update(groupId, { - $set: { [`missions.${group.missions.length - 1}.teams.${group.missions[group.missions.length - 1].teams.length - 1}.successVotes.${group.getLastTeam().memberIndices.indexOf(group.players.map(p => p.id).indexOf(this.userId))}`]: success } + $set: { [`missions.${group.missions.length - 1}.votes.${group.getLastTeam().memberIndices.indexOf(group.players.map(p => p.id).indexOf(this.userId))}`]: success } }); group = Groups.findOne(groupId); // Update local variable if (!group.isWaitingForVote()) { diff --git a/imports/api/groups/server/publications.js b/imports/api/groups/server/publications.js index bb4a468..eb084c1 100644 --- a/imports/api/groups/server/publications.js +++ b/imports/api/groups/server/publications.js @@ -12,10 +12,10 @@ Meteor.publish('groups.findAll', (name, page) => { const lastGroup = Groups.find(selector, { sort: { createdAt: -1 }, limit: skippedGroupsCount }).fetch().pop(); selector.createdAt = { $lt: lastGroup.createdAt }; } - return Groups.find(selector, { fields: Groups.publicFieldsWhenFindAll, sort: { createdAt: -1 }, limit: GROUPS_PER_PAGE }); + return Groups.find(selector, { fields: Groups.publicFields.findAll, sort: { createdAt: -1 }, limit: GROUPS_PER_PAGE }); }); -Meteor.publish('groups.findOne', function (id) { // Do not use arrow function here +Meteor.publish('groups.findOne', id => { check(id, String); - return Groups.find({ _id: id }, { fields: Groups.publicFieldsWhenFindOne }); + return Groups.find({ _id: id }, { fields: Groups.publicFields.findOne }); }); diff --git a/imports/startup/server/fixtures.js b/imports/startup/server/fixtures.js index 4f4ef78..8f8a4a4 100644 --- a/imports/startup/server/fixtures.js +++ b/imports/startup/server/fixtures.js @@ -5,17 +5,22 @@ import { Groups } from '../../api/groups/groups.jsx'; Meteor.startup(() => { // Code to run on server at startup if (Meteor.users.find().count() == 0) { - for (const i of Array(15).keys()) { + for (const i of Array(30).keys()) { Accounts.createUser({ username: `player${i + 1}`, password: 'password' }); } } if (Groups.find().count() == 0) { - let j = 1; - const players = []; - for (const i of Array(10).keys()) { + let c = 1; + let players = []; + for (const i of Array(30).keys()) { const user = Meteor.users.findOne({ username: `player${i + 1}` }); - Groups.update(Groups.insert({ ownerId: user._id, name: `group${j++}` }), { $push: { players: { $each: players } } }); - players.push({ id: user._id, role: 'Undecided' }); + const groupsCount = Groups.find().count(); + if (i + 1 < (groupsCount + 1) * (groupsCount + 2)) { + players.push({ id: user._id, role: 'Undecided' }); + } else { + Groups.update(Groups.insert({ ownerId: user._id, name: `group${c++}` }), { $push: { players: { $each: players.reverse() } } }); + players = []; + } } } }); diff --git a/imports/ui/components/group_row.jsx b/imports/ui/components/group_row.jsx index 83ebfa5..dc919bb 100644 --- a/imports/ui/components/group_row.jsx +++ b/imports/ui/components/group_row.jsx @@ -1,5 +1,4 @@ import React from 'react'; -import { Groups } from '../../api/groups/groups.jsx'; // Constants only import { join, leave } from '../../api/groups/methods.js'; import { MediumAndSmallDevice, TinyDevice } from '../layouts/devices.jsx'; @@ -13,9 +12,9 @@ export default class GroupRow extends React.Component { // REGION: Component Specifications render() { - const { group } = this.props; + const { group, joinedOtherGroup } = this.props; const situation = group.getSituation(); - const joined = group.hasPlayer(Meteor.userId()); + const joinedThisGroup = group.hasPlayer(Meteor.userId()); const isPlaying = group.isPlaying(); return ( @@ -61,9 +60,9 @@ export default class GroupRow extends React.Component { Go to { - !!Meteor.userId() && (joined || situation.slot != false) && !isPlaying ? - !joined && !isPlaying ? - Join : + !!Meteor.userId() && (joinedThisGroup || situation.slot != false) && !isPlaying ? + !joinedThisGroup && !isPlaying ? + !joinedOtherGroup ? Join : null : Leave : null } @@ -106,6 +105,7 @@ export default class GroupRow extends React.Component { GroupRow.propTypes = { group: React.PropTypes.object, + joinedOtherGroup: React.PropTypes.bool, }; GroupRow.contextTypes = { diff --git a/imports/ui/components/group_table.jsx b/imports/ui/components/groups_table.jsx similarity index 70% rename from imports/ui/components/group_table.jsx rename to imports/ui/components/groups_table.jsx index 6be1acd..8a5e9d0 100644 --- a/imports/ui/components/group_table.jsx +++ b/imports/ui/components/groups_table.jsx @@ -1,8 +1,9 @@ import React from 'react'; +import { Groups } from '../../api/groups/groups.jsx'; import GroupRow from './group_row.jsx'; import { MediumAndSmallDevice } from '../layouts/devices.jsx'; -export default class GroupTable extends React.Component { +export default class GroupsTable extends React.Component { // REGION: Component Specifications @@ -20,7 +21,7 @@ export default class GroupTable extends React.Component { - {groups.map(g => )} + {groups.map(g => 0}/>)} @@ -28,6 +29,6 @@ export default class GroupTable extends React.Component { } } -GroupTable.propTypes = { +GroupsTable.propTypes = { groups: React.PropTypes.array, }; diff --git a/imports/ui/components/players_content.jsx b/imports/ui/components/players_content.jsx index 332726b..4e45444 100644 --- a/imports/ui/components/players_content.jsx +++ b/imports/ui/components/players_content.jsx @@ -72,14 +72,14 @@ export default class PlayersContent extends React.Component {

{isSelectingMembers ? : null} { - group.isWaitingForApproval() && group.hasPlayer(Meteor.userId()) ? + group.isWaitingForApproval() && group.hasPlayer(Meteor.userId()) && !group.checkPlayerHasApproved(Meteor.userId()) ? : null } { - group.isWaitingForVote() && group.hasMember(Meteor.userId()) ? + group.isWaitingForVote() && group.hasMember(Meteor.userId()) && !group.checkMemberHasVoted(Meteor.userId()) ? { diff --git a/imports/ui/components/summaries_content.jsx b/imports/ui/components/summaries_content.jsx index 07055fe..0caff6d 100644 --- a/imports/ui/components/summaries_content.jsx +++ b/imports/ui/components/summaries_content.jsx @@ -8,7 +8,7 @@ export default class SummariesContent extends React.Component { render() { const { group } = this.props; - const players = group.getPlayers(); + const players = group.getPlayers().map(p => p.user); return (
@@ -27,6 +27,7 @@ export default class SummariesContent extends React.Component { # + Leader Team members Deniers Result @@ -37,8 +38,9 @@ export default class SummariesContent extends React.Component { m.map((t, j) => {j + 1} - {t.memberIndices.map(i =>
{players[i].user.username}
)} - {t.denierIndices.map(i =>
{players[i].user.username}
)} + {players[t.leaderIndex].username} + {t.memberIndices.map(i =>
{players[i].username}
)} + {t.denierIndices.map(i =>
{players[i].username}
)} {t.result === undefined ? '' : t.result == null ?
Denied by
{t.denierIndices.length} player(s)
: t.result ?
Success with
{t.failVotesCount} fail vote(s)
:
Fail with
{t.failVotesCount} fail vote(s)
} ) diff --git a/imports/ui/pages/lobby.jsx b/imports/ui/pages/lobby.jsx index 66fe76d..4152e69 100644 --- a/imports/ui/pages/lobby.jsx +++ b/imports/ui/pages/lobby.jsx @@ -1,5 +1,5 @@ import React from 'react'; -import GroupTable from '../components/group_table.jsx'; +import GroupsTable from '../components/groups_table.jsx'; import { insert } from '../../api/groups/methods.js'; export default class Lobby extends React.Component { @@ -39,7 +39,7 @@ export default class Lobby extends React.Component {
{formCreateGroup} - +
) : null; diff --git a/imports/ui/pages/login.jsx b/imports/ui/pages/login.jsx index fff923f..71826f0 100644 --- a/imports/ui/pages/login.jsx +++ b/imports/ui/pages/login.jsx @@ -30,7 +30,7 @@ export default class Login extends React.Component {
- +
diff --git a/imports/ui/pages/signup.jsx b/imports/ui/pages/signup.jsx index ab12ac6..9c1ea35 100644 --- a/imports/ui/pages/signup.jsx +++ b/imports/ui/pages/signup.jsx @@ -30,7 +30,7 @@ export default class Signup extends React.Component {
- +