Pion.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. #from __future__ import unicode_literals
  2. # -*- coding: utf-8 -*-
  3. from __future__ import division
  4. import os
  5. from PyQt4.QtCore import *
  6. from PyQt4.QtGui import *
  7. from Forme import Forme
  8. from outilsSvg import *
  9. class Pion(QGraphicsItem):
  10. """pion du plateau de combat"""
  11. def __init__(self, parent=None):
  12. super(Pion, self).__init__()
  13. #caracteristiques du pion
  14. self.nom = nom
  15. self.couleur = couleur
  16. self.logo = logo
  17. self.img = ImgPion()
  18. self.etiquette = EtiquettePion()
  19. #infos liees au plateau (forme et position)
  20. self.plateau = None
  21. self.numComplementaire = "" #numero complementaire si necessaire
  22. #(si plusieurs pions portent le meme nom)
  23. self.position = (-1, -1)
  24. self.z = 0
  25. self.hauteur = 1
  26. self.forme = None
  27. self.formeDef = {"H":[], "C":[]}
  28. #objets et parametres graphiques
  29. self.pixGraphique = None
  30. self.etiquetteGraphique = None
  31. self.largeurOriginale = 0
  32. self.hauteurOriginale = 0
  33. self.polygoneGraphique = None
  34. self.nbRotations = 0
  35. def __getstate__(self):
  36. nePasSvg = ["plateau", "brillance", "shadow", "creature", "polygonesForme", "pixGraphique"]
  37. state = {key:value for key, value in self.__dict__.items() if not key in nePasSvg}
  38. return (state)
  39. def __setstate__(self, state):
  40. self.__dict__ = state
  41. super(Pion, self).__init__()
  42. def paint(self, painter, option, widget = None):
  43. """reimplemente de QGraphicsItem: on ne peint pas cet item, seulement ses items enfants"""
  44. pass
  45. def txtId(self):
  46. """renvoie le nom et le numero complementaire du pion"""
  47. return "{} {}".format(self.nom, self.numComplementaire)
  48. ###attributs du pion
  49. def position(self):
  50. """retourne la position actuelle du pion"""
  51. return self.position
  52. def casesOccupees(self):
  53. """retourne la liste des cases occupees sur le plateau par le pion (x,y,z)"""
  54. retour = []
  55. if self.plateau:
  56. for x, y in self.forme.listeCases(self.position, self.nbRotations):
  57. for z in range(1, self.hauteur):
  58. retour.append((x, y, (self.plateau.cases[(x, y)].altitude + self.z + z)))
  59. return retour
  60. def majZ(self, valeur):
  61. """met a jour l'altitude Z du pion"""
  62. if valeur != self.z:
  63. self.z = valeur
  64. ########### fonctions graphiques et geometriques ##############
  65. def ajouterAuPlateau(self, plateau):
  66. """cerre l'objet graphique representant le pion et le place sur le plateau"""
  67. self.plateau = plateau
  68. #definition de la forme (interpretation de formeDef)
  69. if len(self.formeDef[self.plateau.formeCases]) > 0:
  70. self.forme = Forme(self.plateau.formeCases)
  71. self.forme.definirForme(self.creature.formeDef[self.plateau.formeCases])
  72. #creation du polygone
  73. polygone = self.plateau.polygoneAgglo(self.forme.listeCases((0,0)))
  74. self.polygoneGraphique = QGraphicsPolygonItem()
  75. self.polygoneGraphique.setPolygon(polygone)
  76. self.polygoneGraphique.setAcceptHoverEvents(True)
  77. self.polygoneGraphique.setFlag(QGraphicsItem.ItemIsFocusable) #l'item peut recevoir des commandes souris/clavier
  78. self.polygoneGraphique.setParentItem(self)
  79. self.polygoneGraphique.setPos(QPointF(0,0))
  80. if self.plateau.formeCases == "H":
  81. self.polygoneGraphique.setTransformOriginPoint(QPointF(2*0.2886*self.plateau.hCase,0.5*self.plateau.hCase))
  82. else:
  83. self.polygoneGraphique.setTransformOriginPoint(QPointF(0.5*self.plateau.hCase,0.5*self.plateau.hCase))
  84. #parametres de l'objet graphique
  85. self.setZValue(10)
  86. self.setFlag(QGraphicsItem.ItemHasNoContents)
  87. self.setHandlesChildEvents(True)
  88. pinceau = QPen()
  89. pinceau.setColor(self.creature.couleur.darker(130))
  90. pinceau.setWidth(10)
  91. self.polygoneGraphique.setPen(pinceau)
  92. if self.creature.couleur.isValid():
  93. self.polygoneGraphique.setBrush(self.creature.couleur)
  94. else:
  95. self.polygoneGraphique.setBrush(QColor(255, 0, 0, 150))
  96. self.shadow = QGraphicsDropShadowEffect()
  97. self.shadow.setColor(QColor(50, 50, 50))
  98. self.shadow.setXOffset(1)
  99. self.shadow.setYOffset(2)
  100. self.shadow.setBlurRadius(3)
  101. self.shadow.setEnabled(True)
  102. self.polygoneGraphique.setGraphicsEffect(self.shadow)
  103. self.polygoneBrillance = QGraphicsPolygonItem()
  104. self.polygoneBrillance.setPolygon(self.polygoneGraphique.polygon())
  105. self.polygoneBrillance.setVisible(False)
  106. self.polygoneGraphique.setFlag(QGraphicsItem.ItemIsFocusable)
  107. self.setAcceptHoverEvents(True) #accepte les evenements survol souris
  108. self.polygoneBrillance.setParentItem(self.polygoneGraphique)
  109. #on ajoute l'objet au plateau
  110. self.plateau.addItem(self)
  111. self.majPosition(self.position, self.nbRotations)
  112. def majPosition(self, nouvellePosition, nbRotations = 0):
  113. """met a jour la position de l'objet graphique et de sa forme en fonction de sa position enregistree"""
  114. if self.plateau:
  115. #on met a jour l'occupation des cases
  116. if self.position != (-1,-1):
  117. for coord in self.forme.listeCases((self.position[0],self.position[1]), self.nbRotations):
  118. self.plateau.cases[coord].majOccupation(self)
  119. #on met a jour la position du pion
  120. self.position = nouvellePosition
  121. self.majNbRotation(nbRotations)
  122. #on replace
  123. if self.plateau.formeCases == "H":
  124. angleRotation = 60
  125. positionGraphique = QPointF(self.position[0] * 0.866 * self.plateau.hCase, self.position[1] * self.plateau.hCase)
  126. else:
  127. angleRotation = 90
  128. positionGraphique = QPointF(self.position[0] * self.plateau.hCase, self.position[1] * self.plateau.hCase)
  129. self.prepareGeometryChange()
  130. self.setPos(positionGraphique)
  131. self.polygoneGraphique.setRotation(self.nbRotations*angleRotation)
  132. #maj de l'image
  133. self.majImage()
  134. #on met a jour l'occupation des cases
  135. for coord in self.forme.listeCases((self.position[0],self.position[1]), self.nbRotations):
  136. self.plateau.cases[coord].majOccupation(self, self.z)
  137. def majImage(self):
  138. """met a jour la taille, la position et l'orientation de l'image"""
  139. if len(self.img.nom) > 0:
  140. pix = QPixmap(QString.fromUtf8("img\\"+self.img.nom))
  141. if self.img.masqueAuto:
  142. pix.setMask(pix.createHeuristicMask())
  143. if not pix.isNull():
  144. if not self.pixGraphique:
  145. self.pixGraphique = QGraphicsPixmapItem()
  146. self.pixGraphique.setZValue(10)
  147. if pix.height() >= pix.width():
  148. pix = pix.scaledToHeight(self.plateau.hCase*0.9, Qt.SmoothTransformation)
  149. else:
  150. pix = pix.scaledToWidth(self.plateau.hCase*0.9, Qt.SmoothTransformation)
  151. self.largeurOriginale = pix.width()
  152. self.hauteurOriginale = pix.height()
  153. pix = pix.scaled((self.img.kx/10)*pix.width(), \
  154. (self.img.ky/10)*pix.height(), \
  155. Qt.IgnoreAspectRatio, Qt.SmoothTransformation)
  156. self.pixGraphique.setPixmap(pix)
  157. deltaX = self.img.dx + 0.5*(self.plateau.hCase*1.1544 - self.largeurOriginale)
  158. deltaY = self.img.dy + 0.5*(self.plateau.hCase - self.hauteurOriginale)
  159. if self.img.nom != self.logo and self.pivote == True:
  160. self.pixGraphique.setParentItem(self.polygoneGraphique)
  161. else:
  162. self.pixGraphique.setParentItem(self)
  163. self.pixGraphique.setRotation(self.img.rotation)
  164. self.pixGraphique.setPos(QPointF(deltaX, deltaY))
  165. def majEtiquette(self):
  166. """met a jour la taille, le format et l'orientation de l'etiquette"""
  167. self.etiquetteGraphique = QGraphicsSimpleTextItem("{}".format(self.txtId()))
  168. self.etiquetteGraphique.setPos(QPointF(self.etiquette.dx - 0.112*self.plateau.hCase, \
  169. self.etiquette.dy - 0.275*self.plateau.hCase))
  170. police = QFont("Georgia", self.etiquette.taille_police)
  171. police.setBold(self.etiquette.gras)
  172. self.etiquetteGraphique.setFont(police)
  173. self.etiquetteGraphique.setRotation(self.etiquette.rotation)
  174. self.etiquetteGraphique.setParentItem(self)
  175. def majNbRotation(self, nbRotations):
  176. """ajoute/retranche le nombre au nombre total de rotations du pion"""
  177. self.nbRotations = nbRotations
  178. if self.plateau.formeCases == "H":
  179. rotationsTour = 6
  180. else:
  181. rotationsTour = 4
  182. if self.nbRotations >= 0:
  183. self.nbRotations = self.nbRotations % rotationsTour
  184. else:
  185. self.nbRotations = self.nbRotations % (-rotationsTour)
  186. def retirerDuPlateau(self):
  187. """'deconnecte' les items enfants avant de supprimer du pion du plateau"""
  188. self.polygoneBrillance.prepareGeometryChange()
  189. self.polygoneBrillance.setParentItem(None)
  190. if self.pixGraphique != None:
  191. self.pixGraphique.prepareGeometryChange()
  192. self.pixGraphique.setParentItem(None)
  193. self.polygoneGraphique.prepareGeometryChange()
  194. self.polygoneGraphique.setParentItem(None)
  195. if self.etiquetteGraphique:
  196. self.etiquetteGraphique.prepareGeometryChange()
  197. self.etiquetteGraphique.setParentItem(None)
  198. self.plateau.removeItem(self)
  199. self.plateau = None
  200. ###effets graphiques
  201. def afficheOmbreSelection(self, actif = False):
  202. """modifie l'ombre du pion en fonction de si celui-ci est selectionne ou non"""
  203. if actif:
  204. self.shadow.setXOffset(3)
  205. self.shadow.setYOffset(3)
  206. else:
  207. self.shadow.setXOffset(1)
  208. self.shadow.setYOffset(2)
  209. def surbrillance(self, active, opacite = 0.7, couleur = "white"):
  210. """active/desactive la surbrillance"""
  211. if active:
  212. self.polygoneBrillance.setOpacity(opacite)
  213. couleur = self.couleurSurbrillance(couleur)
  214. pinceau = self.polygoneGraphique.pen()
  215. self.polygoneBrillance.setBrush(couleur)
  216. self.polygoneBrillance.setPen(pinceau)
  217. self.polygoneBrillance.setVisible(active)
  218. def estCibleAttaque(self, estCible, possible = True):
  219. """le pion s'affiche comme etant cible d'une attaque"""
  220. if not possible:
  221. couleur = "red"
  222. else:
  223. couleur = "white"
  224. self.surbrillance(estCible, 0.8, couleur)
  225. def couleurSurbrillance(self, couleur = "white"):
  226. """renvoie une QColor visible pour la surbrillance, selon la couleur du pion"""
  227. retour = QColor(couleur)
  228. if self.polygoneGraphique.brush().color().lightness() > 220:
  229. retour = retour.darker(140)
  230. elif self.polygoneGraphique.brush().color().lightness() < 80:
  231. retour = retour.lighter(140)
  232. return retour
  233. ##################
  234. ############### evenements clavier et souris ##############
  235. def boundingRect(self):
  236. return QRectF()
  237. #######################
  238. class ImgPion():
  239. def __init__(self):
  240. self.nomFichier = "" #nom du fichier image source
  241. self.kx = 1 #coeff d'agrandissement horizontal
  242. self.ky = 1 #coeff d'agrandissement vertical
  243. self.dx = 0 #decalage horizontal
  244. self.dy = 0 #decalage vertical
  245. self.rotation = 0 #rotation(en degres)
  246. self.pivote = False #l'image pivote avec le pion?
  247. self.masqueAuto = False #creer un masque automatiquement
  248. class EtiquettePion():
  249. def __init__(self):
  250. self.taille_police = "" #taille de la police
  251. self.gras = False #en gras
  252. self.dx = 0 #decalage horizontal
  253. self.dy = 0 #decalage vertical
  254. self.rotation = 0 #rotation(en degres)