1
0
Fork 0
mirror of https://github.com/shlomif/PySolFC.git synced 2025-04-05 00:02:29 -04:00
PySolFC/pysollib/move.py
skomoroh 7d832033b2 * fixed bug with flipAllMove
* improved toolbar: set `pause' button as checkbutton


git-svn-id: file:///home/shlomif/Backup/svn-dumps/PySolFC/svnsync-repos/pysolfc/PySolFC/trunk@25 efabe8c0-fbe8-4139-b769-b5e6d273206e
2006-07-23 21:05:29 +00:00

522 lines
18 KiB
Python

## vim:ts=4:et:nowrap
##
##---------------------------------------------------------------------------##
##
## PySol -- a Python Solitaire game
##
## Copyright (C) 2003 Markus Franz Xaver Johannes Oberhumer
## Copyright (C) 2002 Markus Franz Xaver Johannes Oberhumer
## Copyright (C) 2001 Markus Franz Xaver Johannes Oberhumer
## Copyright (C) 2000 Markus Franz Xaver Johannes Oberhumer
## Copyright (C) 1999 Markus Franz Xaver Johannes Oberhumer
## Copyright (C) 1998 Markus Franz Xaver Johannes Oberhumer
## All Rights Reserved.
##
## This program is free software; you can redistribute it and/or modify
## it under the terms of the GNU General Public License as published by
## the Free Software Foundation; either version 2 of the License, or
## (at your option) any later version.
##
## This program is distributed in the hope that it will be useful,
## but WITHOUT ANY WARRANTY; without even the implied warranty of
## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
## GNU General Public License for more details.
##
## You should have received a copy of the GNU General Public License
## along with this program; see the file COPYING.
## If not, write to the Free Software Foundation, Inc.,
## 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
##
## Markus F.X.J. Oberhumer
## <markus@oberhumer.com>
## http://www.oberhumer.com/pysol
##
##---------------------------------------------------------------------------##
# imports
import sys
# /***********************************************************************
# // moves (undo / redo)
# ************************************************************************/
## Currently we have the following atomic moves:
## - move the top cards from one stack on the top of another
## - flip the top card of a stack
## - turn a whole stack onto another stack
## - update the model or complete view a stack
## - increase the round (the number of redeals)
## - save the seed of game.random
## - shuffle a stack
class AtomicMove:
def do(self, game):
self.redo(game)
def __repr__(self):
return str(self.__dict__)
def __str__(self):
return str(self.__dict__)
# Custom comparision for detecting redo moves. See Game.finishMove().
def cmpForRedo(self, other):
return -1
# /***********************************************************************
# // Move the top N cards from a stack to another stack.
# ************************************************************************/
class AMoveMove(AtomicMove):
def __init__(self, ncards, from_stack, to_stack, frames, shadow=-1):
assert from_stack is not to_stack
self.ncards = ncards
self.from_stack_id = from_stack.id
self.to_stack_id = to_stack.id
self.frames = frames
self.shadow = shadow
# do the actual move
def __doMove(self, game, ncards, from_stack, to_stack):
if game.moves.state == game.S_PLAY:
assert to_stack.acceptsCards(from_stack, from_stack.cards[-ncards:])
cards = []
for i in range(ncards):
card = from_stack.removeCard()
cards.append(card)
cards.reverse()
if self.frames != 0:
x, y = to_stack.getPositionFor(cards[0])
game.animatedMoveTo(from_stack, to_stack, cards, x, y, frames=self.frames, shadow=self.shadow)
for c in cards:
to_stack.addCard(c)
def redo(self, game):
self.__doMove(game, self.ncards, game.allstacks[self.from_stack_id],
game.allstacks[self.to_stack_id])
def undo(self, game):
self.__doMove(game, self.ncards, game.allstacks[self.to_stack_id],
game.allstacks[self.from_stack_id])
def cmpForRedo(self, other):
return (cmp(self.ncards, other.ncards) or
cmp(self.from_stack_id, other.from_stack_id) or
cmp(self.to_stack_id, other.to_stack_id))
# /***********************************************************************
# // Flip the top card of a stack.
# ************************************************************************/
class AFlipMove(AtomicMove):
def __init__(self, stack):
self.stack_id = stack.id
# do the actual move
def __doMove(self, game, stack):
card = stack.cards[-1]
if card.face_up:
card.showBack()
else:
card.showFace()
def redo(self, game):
self.__doMove(game, game.allstacks[self.stack_id])
def undo(self, game):
self.__doMove(game, game.allstacks[self.stack_id])
def cmpForRedo(self, other):
return cmp(self.stack_id, other.stack_id)
# /***********************************************************************
# // Flip all cards
# ************************************************************************/
class AFlipAllMove(AtomicMove):
def __init__(self, stack):
self.stack_id = stack.id
def redo(self, game):
stack = game.allstacks[self.stack_id]
for card in stack.cards:
if card.face_up:
card.showBack()
def undo(self, game):
stack = game.allstacks[self.stack_id]
for card in stack.cards:
if not card.face_up:
card.showFace()
def cmpForRedo(self, other):
return cmp(self.stack_id, other.stack_id)
# /***********************************************************************
# // Turn the Waste stack onto the empty Talon.
# ************************************************************************/
class ATurnStackMove(AtomicMove):
def __init__(self, from_stack, to_stack, update_flags=1):
assert from_stack is not to_stack
self.from_stack_id = from_stack.id
self.to_stack_id = to_stack.id
self.update_flags = update_flags
def redo(self, game):
from_stack = game.allstacks[self.from_stack_id]
to_stack = game.allstacks[self.to_stack_id]
assert len(from_stack.cards) > 0
assert len(to_stack.cards) == 0
l = len(from_stack.cards)
for i in range(l):
##unhide = (i >= l - 2)
unhide = 1
##print 1, unhide, from_stack.getCard().__dict__
card = from_stack.removeCard(unhide=unhide, update=0)
##print 2, unhide, card.__dict__
assert card.face_up
to_stack.addCard(card, unhide=unhide, update=0)
card.showBack(unhide=unhide)
##print 3, unhide, to_stack.getCard().__dict__
if self.update_flags & 2:
### not used yet
assert 0
from_stack.round = from_stack.round + 1
if self.update_flags & 1:
assert to_stack is game.s.talon
assert to_stack.round < to_stack.max_rounds or to_stack.max_rounds < 0
to_stack.round = to_stack.round + 1
from_stack.updateText()
to_stack.updateText()
def undo(self, game):
from_stack = game.allstacks[self.to_stack_id]
to_stack = game.allstacks[self.from_stack_id]
assert len(from_stack.cards) > 0
assert len(to_stack.cards) == 0
l = len(from_stack.cards)
for i in range(l):
##unhide = (i >= l - 2)
unhide = 1
card = from_stack.removeCard(unhide=unhide, update=0)
assert not card.face_up
card.showFace(unhide=unhide)
to_stack.addCard(card, unhide=unhide, update=0)
if self.update_flags & 2:
### not used yet
assert 0
assert to_stack.round > 1
to_stack.round = to_stack.round - 1
if self.update_flags & 1:
assert from_stack is game.s.talon
assert from_stack.round > 1
from_stack.round = from_stack.round - 1
from_stack.updateText()
to_stack.updateText()
def cmpForRedo(self, other):
return (cmp(self.from_stack_id, other.from_stack_id) or
cmp(self.to_stack_id, other.to_stack_id) or
cmp(self.update_flags, other.update_flags))
# /***********************************************************************
# // ATurnStackMove is somewhat optimized to avoid unnecessary
# // unhide and hide operations.
# // FIXME: doesn't work yet
# ************************************************************************/
class NEW_ATurnStackMove(AtomicMove):
def __init__(self, from_stack, to_stack, update_flags=1):
assert from_stack is not to_stack
self.from_stack_id = from_stack.id
self.to_stack_id = to_stack.id
self.update_flags = update_flags
# do the actual turning move
def __doMove(self, from_stack, to_stack, show_face):
assert len(from_stack.cards) > 0
assert len(to_stack.cards) == 0
for card in from_stack.cards:
card.item.dtag(from_stack.group)
card.item.addtag(to_stack.group)
if show_face:
assert not card.face_up
card.showFace(unhide=0)
else:
assert card.face_up
card.showBack(unhide=0)
to_stack.cards = from_stack.cards
from_stack.cards = []
from_stack.refreshView()
from_stack.updateText()
to_stack.refreshView()
to_stack.updateText()
def redo(self, game):
from_stack = game.allstacks[self.from_stack_id]
to_stack = game.allstacks[self.to_stack_id]
if self.update_flags & 1:
assert to_stack is game.s.talon
assert to_stack.round < to_stack.max_rounds or to_stack.max_rounds < 0
to_stack.round = to_stack.round + 1
self.__doMove(from_stack, to_stack, 0)
def undo(self, game):
from_stack = game.allstacks[self.from_stack_id]
to_stack = game.allstacks[self.to_stack_id]
if self.update_flags & 1:
assert to_stack is game.s.talon
assert to_stack.round > 1
to_stack.round = to_stack.round - 1
self.__doMove(to_stack, from_stack, 1)
def cmpForRedo(self, other):
return (cmp(self.from_stack_id, other.from_stack_id) or
cmp(self.to_stack_id, other.to_stack_id) or
cmp(self.update_flags, other.update_flags))
# /***********************************************************************
# // Update the view or model of a stack. Only needed for complex
# // games in combination with undo.
# ************************************************************************/
class AUpdateStackMove(AtomicMove):
def __init__(self, stack, flags):
self.stack_id = stack.id
self.flags = flags
# do the actual move
def __doMove(self, game, stack, undo):
if self.flags & 64:
# model
stack.updateModel(undo, self.flags)
else:
# view
if self.flags & 16:
stack.updateText()
if self.flags & 32:
stack.refreshView()
def redo(self, game):
if (self.flags & 3) in (1, 3):
self.__doMove(game, game.allstacks[self.stack_id], 0)
def undo(self, game):
if (self.flags & 3) in (2, 3):
self.__doMove(game, game.allstacks[self.stack_id], 1)
def cmpForRedo(self, other):
return cmp(self.stack_id, other.stack_id) or cmp(self.flags, other.flags)
AUpdateStackModelMove = AUpdateStackMove
AUpdateStackViewMove = AUpdateStackMove
# /***********************************************************************
# // Increase the `round' member variable of a Talon stack.
# ************************************************************************/
class ANextRoundMove(AtomicMove):
def __init__(self, stack):
self.stack_id = stack.id
def redo(self, game):
stack = game.allstacks[self.stack_id]
assert stack is game.s.talon
assert stack.round < stack.max_rounds or stack.max_rounds < 0
stack.round = stack.round + 1
stack.updateText()
def undo(self, game):
stack = game.allstacks[self.stack_id]
assert stack is game.s.talon
assert stack.round > 1
stack.round = stack.round - 1
stack.updateText()
def cmpForRedo(self, other):
return cmp(self.stack_id, other.stack_id)
# /***********************************************************************
# // Save the current state (needed for undo in some games).
# ************************************************************************/
class ASaveSeedMove(AtomicMove):
def __init__(self, game):
self.state = game.random.getstate()
def redo(self, game):
game.random.setstate(self.state)
def undo(self, game):
game.random.setstate(self.state)
def cmpForRedo(self, other):
return cmp(self.state, other.state)
# /***********************************************************************
# // Save game variables
# ************************************************************************/
class ASaveStateMove(AtomicMove):
def __init__(self, game, flags):
self.state = game.getState()
self.flags = flags
def redo(self, game):
if (self.flags & 3) in (1, 3):
game.setState(self.state)
def undo(self, game):
if (self.flags & 3) in (2, 3):
game.setState(self.state)
def cmpForRedo(self, other):
return cmp(self.state, other.state)
# /***********************************************************************
# // Shuffle all cards of a stack. Saves the seed. Does not flip any cards.
# ************************************************************************/
class AShuffleStackMove(AtomicMove):
def __init__(self, stack, game):
self.stack_id = stack.id
# save cards and state
self.card_ids = tuple(map(lambda c: c.id, stack.cards))
self.state = game.random.getstate()
def redo(self, game):
stack = game.allstacks[self.stack_id]
# paranoia
assert stack is game.s.talon
assert self.card_ids == tuple(map(lambda c: c.id, stack.cards))
# shuffle (see random)
game.random.setstate(self.state)
seq = stack.cards
n = len(seq) - 1
while n > 0:
j = game.random.randint(0, n)
seq[n], seq[j] = seq[j], seq[n]
n = n - 1
stack.refreshView()
def undo(self, game):
stack = game.allstacks[self.stack_id]
# restore cards
cards = []
for id in self.card_ids:
c = game.cards[id]
assert c.id == id
cards.append(c)
stack.cards = cards
# restore the state
game.random.setstate(self.state)
stack.refreshView()
def cmpForRedo(self, other):
return (cmp(self.stack_id, other.stack_id) or
cmp(self.card_ids, other.card_ids) or
cmp(self.state, other.state))
# /***********************************************************************
# // ACloseStackMove
# ************************************************************************/
class ACloseStackMove(AtomicMove):
def __init__(self, stack):
self.stack_id = stack.id
def redo(self, game):
stack = game.allstacks[self.stack_id]
assert stack.cards
stack.is_filled = True
stack._shadeStack()
def undo(self, game):
stack = game.allstacks[self.stack_id]
assert stack.cards
stack.is_filled = False
stack._unshadeStack()
def cmpForRedo(self, other):
return cmp(self.stack_id, other.stack_id)
# /***********************************************************************
# // ASingleCardMove - move single card from *anyone* position
# ************************************************************************/
class ASingleCardMove(AtomicMove):
def __init__(self, from_stack, to_stack, from_pos, frames, shadow=-1):
self.from_stack_id = from_stack.id
self.to_stack_id = to_stack.id
self.from_pos = from_pos
self.frames = frames
self.shadow = shadow
def redo(self, game):
from_stack = game.allstacks[self.from_stack_id]
to_stack = game.allstacks[self.to_stack_id]
from_pos = self.from_pos
if game.moves.state == game.S_PLAY:
assert to_stack.acceptsCards(from_stack, [from_stack.cards[from_pos]])
card = from_stack.cards[from_pos]
card = from_stack.removeCard(card, update_positions=1)
if self.frames != 0:
x, y = to_stack.getPositionFor(card)
game.animatedMoveTo(from_stack, to_stack, [card], x, y,
frames=self.frames, shadow=self.shadow)
to_stack.addCard(card)
def undo(self, game):
from_stack = game.allstacks[self.from_stack_id]
to_stack = game.allstacks[self.to_stack_id]
from_pos = self.from_pos
card = to_stack.removeCard()
## if self.frames != 0:
## x, y = to_stack.getPositionFor(card)
## game.animatedMoveTo(from_stack, to_stack, [card], x, y,
## frames=self.frames, shadow=self.shadow)
from_stack.insertCard(card, from_pos)
def cmpForRedo(self, other):
return cmp((self.from_stack_id, self.to_stack_id, self.from_pos),
(other.from_stack_id, other.to_stack_id, other.from_pos))
# /***********************************************************************
# // AInnerMove - change position of single card in stack
# ************************************************************************/
class AInnerMove(AtomicMove):
def __init__(self, stack, from_pos, to_pos):
self.stack_id = stack.id
self.from_pos, self.to_pos = from_pos, to_pos
def redo(self, game):
pass
def undo(self, game):
pass
def cmpForRedo(self, other):
return cmp((self.stack_id, self.from_pos, self.to_pos),
(other.stack_id, other.from_pos, other.to_pos))