-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgame.py
304 lines (274 loc) · 13.2 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
from player import *
from decks_and_cards import *
from presRound import *
import math
import random
import pandas as pd
class Game(object):
# this object is the only thing that the player interacts with, it manages the playing of rounds and the creation of players,
# it also manages the exchanges of cards before the round starts and different heuristics I was curious about
def __init__(self, numberOfPlayers, name, numRounds, passingRules, numDecks, restOfPlayers, print=False, newPlayer=Player, amountOfNewPlayers=1, anti=False, test=None, generateData=False):
'''
:param numberOfPlayers: the number of players in the game
:param name: the name of the game
:param numRounds: the number of rounds the game should go through
:param passingRules: the passing rules the game should operate under
:param numDecks: number of decks used
:param restOfPlayers: which of the player objects should comprise the game
:param print: True: prints out every single card every bot plays
False: after all games are done, prints out statistics on the games
:param newPlayer: second type of player object (default is normal Player)
:param amountOfNewPlayers: number of players of the type above (default 1)
:param anti: should there be an anti in this game?
:param test: just for passing random things that I want to test to other objects
:param generateData: should the program put the data from the games into a file?
'''
self.name = name
self.deck = False
self.players = []
self.numberOfPlayers = numberOfPlayers
self.numRounds = numRounds
self.playersOutOrder = []
self.startingPlayer = False
self.passingRules = passingRules
self.numDecks = numDecks
self.lastPlayer = newPlayer
self.amountOfNewPlayers = amountOfNewPlayers
self.print = print
self.anti = anti
self.test = test
self.generateData = generateData
self.data = []
# makes players
for i in range(self.numberOfPlayers - self.amountOfNewPlayers):
self.players.append(restOfPlayers(i))
for i in range(self.amountOfNewPlayers):
self.players.append(self.lastPlayer(i + self.numberOfPlayers - self.amountOfNewPlayers))
self.lastPlayer = self.players[self.numberOfPlayers - 1]
self.dealHands()
def dealHands(self):
# makes and shuffles new deck
self.deck = Deck(self.numDecks)
self.deck.shuffle()
# gives players their hands
for player in self.players:
player.resetPlayer()
player.draw(self.deck, 13)
player.sortHand()
def getData(self):
return self.data
def startGame(self):
# checks who has the four of clubs and sets them to be starting player
for player in self.players:
for card in player.showHand():
if card.value == 4 and card.suit == 'Clubs':
self.startingPlayer = player
if self.print:
print('starting player:', self.startingPlayer.name)
break
# heuristic variable setup
numberofTimesAssToPres = 0
numberofTimesPresToPres = 0
numberofTimesVPToPres = 0
numberofTimesVAToPres = 0
numberofTimesNewPlayerPres = 0
isFirstRound = True
# plays the given amount of rounds
for i in range(self.numRounds):
# prints a progress bar
if not self.print: print(f'\r{"Percent Complete: "} {math.ceil((i/self.numRounds)*100)}% ', end='')
# heuristic role assignment
ass = self.players[self.numberOfPlayers - 1]
pres = self.players[0]
vp = self.players[1]
va = self.players[self.numberOfPlayers - 2]
# generates some data about the player's hands for use later
if self.generateData:
cardsForPlayers = []
if self.generateData:
# loops through players
for player in self.players:
playerData = []
# appends the player (this will be deleted later)
playerData.append(player)
# appends if they were the starting player
if player is self.startingPlayer:
playerData.append(1)
else:
playerData.append(0)
# puts cards in the list
for card in player.cardDict:
playerData.append(player.cardDict[card])
cardsForPlayers.append(playerData)
# does the anti
if self.anti:
self.doAnti()
# starts the playing of rounds (this plays one round, the for loop above plays more rounds)
newRound = presRound(self.players, self.startingPlayer, self.print)
self.playersOutOrder = newRound.startRound()
if self.print:
printablePlayersOutOrder = []
for player in self.playersOutOrder:
printablePlayersOutOrder.append(player.name)
print("The below shows the players in order of when they went out.")
print(printablePlayersOutOrder)
print("---------------------------")
# heuristic role checking
if ass == self.playersOutOrder[0]:
numberofTimesAssToPres += 1
if pres == self.playersOutOrder[0]:
numberofTimesPresToPres += 1
if vp == self.playersOutOrder[0]:
numberofTimesVPToPres += 1
if va == self.playersOutOrder[0]:
numberofTimesVAToPres += 1
if pres == self.lastPlayer:
numberofTimesNewPlayerPres += 2
elif vp == self.lastPlayer:
numberofTimesNewPlayerPres += 1
elif va == self.lastPlayer:
numberofTimesNewPlayerPres += -1
elif ass == self.lastPlayer:
numberofTimesNewPlayerPres += -2
# sets players to the new playersOutOrder
self.players = self.playersOutOrder
if self.generateData:
# generates data for use
data = []
for playerDat in cardsForPlayers:
for player in self.players:
if playerDat[0] is player:
playerDat.pop(0)
playerDat = [self.players.index(player)] + playerDat
data.append(playerDat)
# self.data.append(playerDat)
# puts player data into game data
toAppendToSelfData = []
for localData in data:
for thing in localData:
toAppendToSelfData.append(thing)
self.data.append(toAppendToSelfData)
# deals hands
self.dealHands()
# checks passing rules and implements them
if self.passingRules == 'two':
self.doTopTwoCards()
elif self.passingRules == 'one':
self.doTopOneCard()
elif self.passingRules == 'hybrid':
self.doHybridPassing()
elif self.passingRules == 'none':
pass
self.players = self.playersOutOrder
# prints heuristic data
if not self.print:
print("\n")
print(round((numberofTimesPresToPres/self.numRounds)*100,4), "% prez went to p")
print(round((numberofTimesVPToPres/self.numRounds)*100,4), "% vp went to p")
print(round((numberofTimesVAToPres/self.numRounds)*100,4), "% va went to p")
print(round((numberofTimesAssToPres/self.numRounds)*100,4), "% ass went to p")
if (self.amountOfNewPlayers > 0):
print(numberofTimesNewPlayerPres/self.numRounds, " average score (higher is better, avg is 0) of the last player")
def doAnti(self):
cardsInAnti = []
howManyCardsPerPlayer = {}
# loops through all players and adds their cards to the anti
for player in self.players:
cards = player.anti()
cardsInAnti += cards
howManyCardsPerPlayer[player.name] = len(cards)
if self.print:
print("Player: " + str(player.name) + " put in the following to the anti.")
print(cards)
# shuffles anti
random.shuffle(cardsInAnti)
# loops through players and gives them their cards
for player in self.players:
if self.print: print("" + "Player: " + str(player.name) + " got the following in the anti")
for i in range(howManyCardsPerPlayer[player.name]):
card = cardsInAnti.pop()
player.cardDict[card] += 1
if self.print: print(str(card), end="")
if self.print: print("")
def doHybridPassing(self):
# one of the passing rules, this is a hybrid of the two passing rules (functions) below this one
# this was setup to see how powerful pres would be if vp was in its weakest form and p was in its strongest
bestTwo = []
worstTwo = []
# gets cards
for player in self.playersOutOrder:
if self.playersOutOrder.index(player) == 0:
self.startingPlayer = player
worstTwo.append(player.giveLowestCard())
worstTwo.append(player.giveLowestCard())
elif self.playersOutOrder.index(player) == self.numberOfPlayers - 1:
bestTwo.append(player.giveHighestCard())
bestTwo.append(player.giveHighestCard())
# gives cards
for player in self.playersOutOrder:
if self.playersOutOrder.index(player) == 0:
for card in bestTwo:
player.cardDict[card] += 1
elif self.playersOutOrder.index(player) == self.numberOfPlayers - 1:
for card in worstTwo:
player.cardDict[card] += 1
# does va/vp passing
worstOneVA = self.playersOutOrder[self.numberOfPlayers - 2].giveLowestCard()
self.playersOutOrder[1].cardDict[worstOneVA] += 1
worstOneVP = self.playersOutOrder[1].giveLowestCard()
self.playersOutOrder[self.numberOfPlayers - 2].cardDict[worstOneVP] += 1
def doTopOneCard(self):
# a passes best to p, p passes worst to a, va passes worst to vp, vp passes worst to va
self.startingPlayer = self.playersOutOrder[0]
bestOneA = self.playersOutOrder[self.numberOfPlayers - 1].giveHighestCard()
worstOneP = self.playersOutOrder[0].giveLowestCard()
worstOneVA = self.playersOutOrder[self.numberOfPlayers - 2].giveLowestCard()
if self.print:
print("Ass gave pres: ", bestOneA)
print("VA gave VP: ", worstOneVA)
self.playersOutOrder[1].cardDict[worstOneVA] += 1
worstOneVP = self.playersOutOrder[1].giveLowestCard()
if self.print:
print("VP gave VA: ", worstOneVP)
print("Pres gave ass: ", worstOneP)
self.playersOutOrder[0].cardDict[bestOneA] += 1
self.playersOutOrder[self.numberOfPlayers - 2].cardDict[worstOneVP] += 1
self.playersOutOrder[self.numberOfPlayers - 1].cardDict[worstOneP] += 1
def doTopTwoCards(self):
# a passes best two to p, p worst two to a, va best one to vp, vp worst one to va
bestTwo = []
worstTwo = []
worstOne = []
bestOne = []
# gets cards
for player in self.playersOutOrder:
if self.playersOutOrder.index(player) == 0:
self.startingPlayer = player
worstTwo.append(player.giveLowestCard())
worstTwo.append(player.giveLowestCard())
elif self.playersOutOrder.index(player) == 1:
worstOne.append(player.giveLowestCard())
elif self.playersOutOrder.index(player) == self.numberOfPlayers - 2:
bestOne.append(player.giveHighestCard())
elif self.playersOutOrder.index(player) == self.numberOfPlayers - 1:
bestTwo.append(player.giveHighestCard())
bestTwo.append(player.giveHighestCard())
if self.print:
print("Ass gave pres: ", bestTwo)
print("VA gave VP: ", bestOne)
print("VP gave VA: ", worstOne)
print("Pres gave ass: ", worstTwo)
# gives cards
for player in self.playersOutOrder:
if self.playersOutOrder.index(player) == 0:
for card in bestTwo:
player.cardDict[card] += 1
elif self.playersOutOrder.index(player) == 1:
for card in bestOne:
player.cardDict[card] += 1
elif self.playersOutOrder.index(player) == self.numberOfPlayers - 2:
for card in worstOne:
player.cardDict[card] += 1
elif self.playersOutOrder.index(player) == self.numberOfPlayers - 1:
for card in worstTwo:
player.cardDict[card] += 1