package mage.game; import mage.cards.Card; import mage.filter.FilterCard; import mage.util.Copyable; import java.io.Serializable; import java.util.*; import java.util.Map.Entry; import java.util.stream.Collectors; /** * @author BetaSteward_at_googlemail.com */ public class Exile implements Serializable, Copyable { private static final UUID PERMANENT = UUID.randomUUID(); private final Map exileZones = new HashMap<>(); public Exile() { createZone(PERMANENT, "Permanent"); } public Exile(final Exile exile) { for (Entry entry : exile.exileZones.entrySet()) { exileZones.put(entry.getKey(), entry.getValue().copy()); } } public Collection getExileZones() { return exileZones.values(); } public ExileZone getPermanentExile() { return exileZones.get(PERMANENT); } public void add(UUID id, String name, Card card) { createZone(id, name).add(card); } public void add(Card card) { exileZones.get(PERMANENT).add(card); } public ExileZone createZone(UUID id, String name) { return createZone(id, name + " - Exile", false); } private ExileZone createZone(UUID id, String name, boolean hidden) { exileZones.putIfAbsent(id, new ExileZone(id, name, hidden)); return exileZones.get(id); } public ExileZone getExileZone(UUID id) { return exileZones.get(id); } public Card getCard(UUID cardId, Game game) { for (ExileZone exile : exileZones.values()) { if (exile.contains(cardId)) { return game.getCard(cardId); } } return null; } public List getCards(FilterCard filter, Game game) { List allCards = getAllCards(game); return allCards.stream().filter(card -> filter.match(card, game)).collect(Collectors.toList()); } @Deprecated // TODO: must use related request due game range public List getAllCards(Game game) { return getAllCards(game, null); } /** * Return exiled cards from specific player. Use it in effects to find all cards in range. * * @param game * @param fromPlayerId * @return */ public List getAllCards(Game game, UUID fromPlayerId) { List res = new ArrayList<>(); for (ExileZone exile : exileZones.values()) { for (Card card : exile.getCards(game)) { if (fromPlayerId == null || card.getOwnerId().equals(fromPlayerId)) { res.add(card); } } } return res; } public List getAllCardsByRange(Game game, UUID controllerId) { List res = new ArrayList<>(); for (UUID playerId : game.getState().getPlayersInRange(controllerId, game)) { res.addAll(getAllCards(game, playerId)); } return res; } public boolean removeCard(Card card, Game game) { for (ExileZone exile : exileZones.values()) { if (exile.contains(card.getId())) { return exile.remove(card.getId()); } } return false; } /** * Move card from one exile zone to another. Use case example: create special zone for exiled and castable card. * * @param card * @param game * @param toZoneId */ public void moveToAnotherZone(Card card, Game game, ExileZone exileZone) { if (getCard(card.getId(), game) == null) { throw new IllegalArgumentException("Card must be in exile zone: " + card.getIdName()); } if (exileZone == null) { throw new IllegalArgumentException("Exile zone must exists: " + card.getIdName()); } removeCard(card, game); exileZone.add(card); } @Override public Exile copy() { return new Exile(this); } public boolean containsId(UUID cardId, Game game) { for (Card card : getAllCards(game)) { if (card.getId().equals(cardId)) { return true; } } return false; } public void clear() { for (ExileZone exile : exileZones.values()) { exile.clear(); } } public void cleanupEndOfTurnZones(Game game) { // moves cards from outdated zone to main exile zone ExileZone mainZone = getExileZone(PERMANENT); for (ExileZone zone : exileZones.values()) { if (zone.isCleanupOnEndTurn()) { for (Card card : zone.getCards(game)) { mainZone.add(card); zone.remove(card); } } } } @Override public String toString() { return "Cards: " + exileZones.values() .stream() .mapToInt(ExileZone::size) .sum(); } }