-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathgame.py
313 lines (235 loc) · 10.1 KB
/
game.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
from __future__ import annotations
import typing
from discord.ext import commands
from discord.ext.commands import Context
from discord import Client, User, Embed, Colour
import classes.games.TicTacToe as tic
from classes.Game import Game
from classes.Player import Player
from time import time
import asyncio
class TicTacToeGameCog(commands.Cog):
_channel_to_game: dict[str, tic.TicTacToeGame] = {}
def __init__(self, client: Client):
self.client: Client = client
@commands.command()
async def tchallenge(self, ctx: Context, user: User):
if ctx.guild is None:
await ctx.reply("**You cannot challenge someone outside of a server**")
return
if ctx.author.id == user.id:
await ctx.reply("**You cannot challenge yourself!**")
return
if ctx.channel.id in Game.occupied_channels:
await ctx.reply("**This channel is occupied, by another game!**")
return
if ctx.author.id in Player.occupied_players:
await ctx.reply("**You are already in a game!**")
return
if user.id in Player.occupied_players:
await ctx.reply(f"**{user.mention} is in another game!**")
return
embed = Embed(title="TicTacToe In-Game Commands📝", color=Colour.red())
prefix = "."
embed.add_field(
name="In-Game Commands ⚙️",
value=f"**{prefix}tchallenge @user** (challenge a user to a game of TicTacToe)\n"
f"**{prefix}p pos** (place your symbol at pos, a number from 0-8\n"
f"**{prefix}tsurrender** (forfeit the game)\n"
f"**{prefix}ttimeout** (win, if the opponent hasn't played in 100 seconds)\n"
f"**{prefix}ttie** (propose a tie to your opponent)\n",
)
message = await ctx.send(
content=f"**{ctx.author.mention} has challenged {user.mention} to a game of TicTacToe,"
f" will he accept?**",
embed=embed,
)
reactions = ["✅", "❌"]
accepted = False
for reaction in reactions:
await message.add_reaction(reaction)
def check(react, author):
return author == user and str(react.emoji) in reactions
while True:
try:
reaction, user = await self.client.wait_for(
"reaction_add", timeout=30, check=check
)
if str(reaction.emoji) == "✅":
accepted = True
break
elif str(reaction.emoji) == "❌":
break
except asyncio.TimeoutError:
break
if accepted:
if ctx.channel.id in Game.occupied_channels:
await ctx.reply(
"**This channel was occupied, before the challenge was accepted!**"
)
return
if ctx.author.id in Player.occupied_players:
await ctx.reply(
"**You joined a game, before the challenge was accepted!**"
)
return
if user.id in Player.occupied_players:
await ctx.reply(
f"**{user.mention} joined a game, before the challenge was accepted!**"
)
return
players = [ctx.author.id, user.id]
game = tic.TicTacToeGame(players)
Game.occupied_channels.append(ctx.channel.id)
self._channel_to_game[str(ctx.channel.id)] = game
for player in players:
Player.occupied_players.append(player)
await ctx.send(
content=f"**{ctx.author.mention} ⚔️ {user.mention}\n"
f"-------------------------\n"
f"{game.get_player_by_id(ctx.author.id).emoji}{ctx.author.mention}\n"
f"{game.get_player_by_id(user.id).emoji}{user.mention}**"
)
await asyncio.sleep(1)
game.timer = time()
game.ongoing = True
await self.display(ctx, game) # send the first message
else:
await ctx.reply(
f"**{user.mention} turned down/did not respond the challenge!**"
)
@commands.command()
async def p(self, ctx: Context, pos: int):
try:
game = self._channel_to_game[str(ctx.channel.id)]
if ctx.author.id not in game.player_ids:
await ctx.reply("**You are not part of this TicTacToe game!**")
return
except KeyError:
await ctx.reply("**There is no TicTacToe game running in this channel!**")
return
if pos not in range(0, 9):
await ctx.reply("**Number must be between 0-8!**")
return
if game.current_round_player.discord_id != ctx.author.id:
await ctx.reply("**It's not your turn!**")
return
success = game.board.place(game.current_round_player.symbol, pos)
if success:
if game.board.check_win(game.current_round_player.symbol):
winner = game.current_round_player
loser = game.next_player()
await ctx.send(
f"**{winner.emoji}<@!{winner.discord_id}> defeated "
f"{loser.emoji}<@!{loser.discord_id}> in a game of TicTacToe!**"
)
self.delete_game(ctx, game)
else:
game.next_round()
await self.display(ctx, game)
else:
await ctx.reply(f"**Spot #{pos} is occupied!**")
@commands.command()
async def tsurrender(self, ctx: Context):
try:
game = self._channel_to_game[str(ctx.channel.id)]
if ctx.author.id not in game.player_ids:
await ctx.reply("**You are not part of this TicTacToe game!**")
return
except KeyError:
await ctx.reply("**There is no TicTacToe game running in this channel!**")
return
loser = game.get_player_by_id(ctx.author.id)
winner = [x for x in game.players if x.discord_id != loser.discord_id][0]
await ctx.send(
f"**{loser.emoji}<@!{loser.discord_id}> surrendered to {winner.emoji}<@!{winner.discord_id}>!**"
)
self.delete_game(ctx, game)
@commands.command()
async def ttie(self, ctx: Context):
try:
game = self._channel_to_game[str(ctx.channel.id)]
if ctx.author.id not in game.player_ids:
await ctx.reply("**You are not part of this TicTacToe game!**")
return
except KeyError:
await ctx.reply("**There is no TicTacToe game running in this channel!**")
return
thisplayer = game.get_player_by_id(ctx.author.id)
otherplayer = [
x for x in game.players if x.discord_id != thisplayer.discord_id
][0]
thisplayer.proposed_tie = True
if thisplayer.proposed_tie and otherplayer.proposed_tie:
await ctx.send(
f"**{thisplayer.emoji}<@!{thisplayer.discord_id}> and {otherplayer.emoji}<@!{otherplayer.discord_id}>"
f"agreed to a tie!**"
)
self.delete_game(ctx, game)
else:
await ctx.send(
f"**{thisplayer.emoji}<@!{thisplayer.discord_id}> proposed a tie"
f" to {otherplayer.emoji}<@!{otherplayer.discord_id}>!**"
)
@commands.command()
async def ttimeout(self, ctx: Context):
try:
game = self._channel_to_game[str(ctx.channel.id)]
if ctx.author.id not in game.player_ids:
await ctx.reply("**You are not part of this TicTacToe game!**")
return
except KeyError:
await ctx.reply("**There is no TicTacToe game running in this channel!**")
return
if time() - game.timer > game.TIMEOUT:
game.current_round_player = game.next_player()
winner = game.current_round_player
loser = game.next_player()
await ctx.send(
f"**{winner.emoji}<@!{winner.discord_id}> defeated "
f"{loser.emoji}<@!{loser.discord_id}> due to the inactivity of the latter**"
)
self.delete_game(ctx, game)
else:
await ctx.reply(
f"**<@!{game.next_player().discord_id}> has still ~{game.TIMEOUT - (time() - game.timer)}"
f" seconds to play!**"
)
@classmethod
async def display(cls, ctx: Context, game: tic.TicTacToeGame):
txt = game.board.display()
await ctx.send(
f"**It is <@!{game.current_round_player.discord_id}>'s "
f"turn ({game.current_round_player.emoji})**\n"
)
# these 2 messages are split, because discord will make emojis smaller when they are combined with normal text
await ctx.send(txt)
@classmethod
def delete_game(cls, ctx: Context, game: tic.TicTacToeGame):
Game.occupied_channels.remove(ctx.channel.id)
del cls._channel_to_game[str(ctx.channel.id)]
for player_id in game.player_ids:
Player.occupied_players.remove(player_id)
del game
@tchallenge.error
async def tchallenge_error(self, ctx: Context, error: Exception):
prefix = "."
if isinstance(error, commands.MissingRequiredArgument):
await ctx.reply(
"**You need to specify the user!\n"
f"Example: {prefix}tchallenge @user**"
)
elif isinstance(error, commands.UserNotFound):
await ctx.reply(
"**User does not exist or could not be found!\n"
f"Example: {prefix}tchallenge @user**"
)
@p.error
async def p_error(self, ctx: Context, error: Exception):
prefix = "."
if isinstance(error, commands.MissingRequiredArgument):
await ctx.reply(
"**You need to specify the coordinates!\n" f"Example: {prefix}p 4**"
)
def setup(client):
client.add_cog(TicTacToeGameCog(client))