- Check for allowed card type was before the check for the mau cards, which caused incorrect cards to be played and mau cards to be passed to wrong players - Ending with a special card is now a faulty move, which grants the player 5 fault cards now - Grabbed cards could be registered as playable as it didnt include next allowed card type - If the first card is a joker, set the next allowed card type to a random other card type to prevent an instant softlock - Other cleanup :)
129 lines
3.4 KiB
C#
129 lines
3.4 KiB
C#
namespace MauMau_Server.Mau;
|
|
|
|
public class Deck
|
|
{
|
|
private readonly List<Card> _unusedDeck = new();
|
|
private readonly List<Card> _usedDeck = new();
|
|
public Card CurrentCard { get; private set; }
|
|
|
|
/**
|
|
* <summary>
|
|
* Creates a new deck instance with a new shuffled set of cards
|
|
* </summary>
|
|
*/
|
|
public Deck()
|
|
{
|
|
CreateSet();
|
|
ShuffleDeck();
|
|
var initialCard = DrawCard();
|
|
CurrentCard = initialCard;
|
|
}
|
|
|
|
/**
|
|
* <summary>
|
|
* Adds the given card to the used cards deck and sets the current card to the given card.
|
|
* </summary>
|
|
* <param name="card">The card to add to the used cards deck.</param>
|
|
*/
|
|
public void AddCardToUsedDeck(Card card)
|
|
{
|
|
_usedDeck.Add(card);
|
|
CurrentCard = card;
|
|
}
|
|
|
|
/**
|
|
* <summary>
|
|
* Adds the given list of cards to the used cards deck.
|
|
* </summary>
|
|
* <param name="cards">The list of cards to add to the used cards deck.</param>
|
|
*/
|
|
public void AddCardsToUsedDeck(IEnumerable<Card> cards)
|
|
{
|
|
_usedDeck.AddRange(cards);
|
|
}
|
|
|
|
/**
|
|
* <summary>
|
|
* Draws a card from the deck.
|
|
* If the deck is empty, the deck is reshuffled with <see cref="ReshuffleDeck"/>.
|
|
* </summary>
|
|
*/
|
|
public Card DrawCard()
|
|
{
|
|
if (_unusedDeck.Count == 0) ReshuffleDeck();
|
|
var card = _unusedDeck[0];
|
|
_unusedDeck.RemoveAt(0);
|
|
return card;
|
|
}
|
|
|
|
/**
|
|
* <summary>
|
|
* Take a given amount of cards from the deck. This method calls <see cref="DrawCard"/> for each card.
|
|
* </summary>
|
|
* <param name="amount">The amount of cards to draw from the deck.</param>
|
|
*/
|
|
public IEnumerable<Card> DrawCards(int amount)
|
|
{
|
|
var cards = new List<Card>();
|
|
for (var i = 0; i < amount; i++)
|
|
{
|
|
cards.Add(DrawCard());
|
|
}
|
|
return cards;
|
|
}
|
|
|
|
/**
|
|
* <summary>
|
|
* Creates a new deck of cards and adds them to the unused deck.
|
|
* </summary>
|
|
*/
|
|
private void CreateSet()
|
|
{
|
|
foreach (CardType cardType in Enum.GetValues(typeof(CardType)))
|
|
{
|
|
if (cardType == CardType.JOKER)
|
|
{
|
|
_unusedDeck.Add(new Card(cardType, CardValue.RED));
|
|
_unusedDeck.Add(new Card(cardType, CardValue.BLACK));
|
|
continue;
|
|
}
|
|
foreach (CardValue cardValue in Enum.GetValues(typeof(CardValue)))
|
|
{
|
|
if (cardValue is CardValue.RED or CardValue.BLACK) continue;
|
|
_unusedDeck.Add(new Card(cardType, cardValue));
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* <summary>
|
|
* Moves all the used cards back to the unused deck and shuffles it.
|
|
* If there are no cards to reshuffle, a new set of cards is created and shuffled.
|
|
* </summary>
|
|
*/
|
|
private void ReshuffleDeck()
|
|
{
|
|
_unusedDeck.AddRange(_usedDeck);
|
|
_usedDeck.Clear();
|
|
|
|
if (_unusedDeck.Count == 0)
|
|
{
|
|
CreateSet();
|
|
}
|
|
|
|
ShuffleDeck();
|
|
}
|
|
|
|
/**
|
|
* <summary>
|
|
* Shuffles all the cards in the deck using the Fisher-Yates algorithm.
|
|
* </summary>
|
|
*/
|
|
private void ShuffleDeck()
|
|
{
|
|
var unusedDeckCopy = new List<Card>(_unusedDeck);
|
|
_unusedDeck.Clear();
|
|
unusedDeckCopy = unusedDeckCopy.OrderBy(x => Guid.NewGuid()).ToList();
|
|
_unusedDeck.AddRange(unusedDeckCopy);
|
|
}
|
|
} |