Case.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446
  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 Terrain import Terrain
  8. class Case(QGraphicsPolygonItem):
  9. """objet graphique representant une case du plateau"""
  10. def __init__(self, plateau, parent=None):
  11. super(Case, self).__init__()
  12. #plateau
  13. self.plateau = plateau
  14. #attributs
  15. self.x = 0
  16. self.y = 0
  17. self.altitude = 0
  18. self.terrain = Terrain() #terrain par defaut
  19. self.couleur = None #couleur du fond par defaut
  20. self.bordure = QColor(85, 85, 85, 85) #couleur de la bordure par defaut
  21. self.estDansChampDeplacement = False #la case est dans le champ de deplacement du pion selectionne
  22. self.centreGraphique = None
  23. self.formeCases = "H"
  24. self.occupeePar = {} #objet: altitudes occupees (sous forme de tuple, ex: (0,1,2) pour une creature occupant les cases d'altitude 0, 1 et 2)
  25. self.estCache = False #est sous un cache place par le MJ (cache le terrain, les decors, les pions aux joueurs...)
  26. self.ombre = True #ombre (cache les pions uniquement)
  27. #effet sur la case
  28. self.effetActif = ""
  29. #polygones d'affichage
  30. self.polygoneEffet = None
  31. self.polygoneAffichageSpecial = None
  32. self.etiquetteAltitude = None
  33. def __getstate__(self):
  34. """selectionne les attributs qui seront sauvegardes"""
  35. self.idTerrain = self.terrain.id
  36. state = {key:value for key, value in self.__dict__.items() if key in ["x", "y", "altitude","idTerrain","bordure","couleur", \
  37. "estCache", "ombre", "effetActif"]}
  38. return (state)
  39. def __setstate__(self, state):
  40. """recupere les attributs sauvegardes"""
  41. self.__dict__ = state
  42. if self.idTerrain == "00":
  43. self.terrain = Terrain()
  44. else:
  45. self.terrain = charger("librairie\\terrain", self.idTerrain)
  46. ######## fonctions de base : geometrie, aspect graphique ######
  47. def creer(self, x, y, couleur = QColor(0, 255, 0, 80)):
  48. """creation du polygone et enregistrement des donnees geometriques"""
  49. self.x = x
  50. self.y = y
  51. self.couleur = couleur
  52. #creation de l'objet graphique sur le plateau
  53. self.creerGraphique()
  54. def creerGraphique(self):
  55. """cree les objets graphiques representant la case"""
  56. #reinitialisation des variables
  57. self.couleurEffet = {"brule":QColor("orange"), "glace":QColor("white"), "poison":QColor("green"), "eau":QColor("blue")}
  58. self.imgEffet = {"brule":"effFeu.jpg", "glace":"effGlace.jpg", "poison":"effPoison.png", "eau":"effEau.png"}
  59. #enregistrement des cases voisines:
  60. self.voisins = self.lstVoisins(self.x, self.y)
  61. self.occupeePar = {}
  62. #enregistrement du centre
  63. if self.plateau.formeCases == "H":
  64. self.centreGraphique = QPointF(((self.x*0.866)+0.5773)*self.plateau.hCase, (self.y+0.5)*self.plateau.hCase)
  65. else:
  66. self.centreGraphique = QPointF((self.x+0.5)*self.plateau.hCase, (self.y+0.5)*self.plateau.hCase)
  67. #cree le polygone de la case
  68. self.setPolygon(self.polygone(self.x, self.y))
  69. self.plateau.addItem(self)
  70. #interactions graphiques:
  71. self.setFlag(QGraphicsItem.ItemIsFocusable)
  72. self.setAcceptHoverEvents(True)
  73. self.setAcceptDrops(True)
  74. self.setZValue(0)
  75. self.setFiltersChildEvents(True)
  76. #pour afficher les coordonnees des cases:
  77. #text = QGraphicsSimpleTextItem("{}-{}".format(self.x,self.y), parent=self)
  78. #if self.plateau.formeCases == "H":
  79. # text.setPos(QPointF(((self.x*0.866)+0.2886)*self.plateau.hCase, self.y*self.plateau.hCase))
  80. #else:
  81. # text.setPos(QPointF(self.x*self.plateau.hCase, self.y*self.plateau.hCase))
  82. #polygone utilise lorsque dans champ de deplacement
  83. self.polygoneChampDep = QGraphicsPolygonItem(self.polygone(self.x, self.y), parent=self)
  84. self.polygoneChampDep.setVisible(False)
  85. self.polygoneChampDep.setAcceptHoverEvents(False)
  86. #apparence initiale de la case
  87. self.majCouleur(self.couleur)
  88. #polygone utilise pour afficher la cible du curseur
  89. self.polygoneCible = QGraphicsPolygonItem(self.polygone(self.x, self.y), parent=self)
  90. self.polygoneCible.setVisible(False)
  91. self.polygoneCible.setAcceptHoverEvents(False)
  92. #polygone test (pour capter les evts souris par dessus d'autres objets)
  93. ## self.polygoneTest = QGraphicsPolygonItem(self.polygone(self.x, self.y), parent=self)
  94. ## self.polygoneTest.setVisible(True)
  95. ## self.polygoneTest.setAcceptHoverEvents(True)
  96. ## self.polygoneTest.setFlag(QGraphicsItem.ItemIsFocusable)
  97. ## self.polygoneTest.setBrush(QColor(0,0,0,0))
  98. ## self.polygoneTest.setPen(QPen(QColor(0,0,0,0)))
  99. ## self.polygoneTest.setAcceptedMouseButtons(Qt.NoButton)
  100. ## self.polygoneTest.setZValue(200)
  101. ## self.polygoneTest.setParentItem(self)
  102. def majGraphique(self):
  103. """met a jour l'aspect graphique de la acse en fonction de son etat"""
  104. if self.terrain:
  105. if len(self.terrain.nom) > 0 :
  106. self.majTerrain(self.terrain)
  107. self.majAffichageSpecial("")
  108. self.majEffet(self.effetActif)
  109. def recreer(self, plateau):
  110. self.plateau = plateau
  111. super(Case, self).__init__()
  112. #polygones d'affichage
  113. self.polygoneEffet = None
  114. self.polygoneAffichageSpecial = None
  115. self.etiquetteAltitude = None
  116. self.creerGraphique()
  117. self.majGraphique()
  118. def polygone(self, x, y):
  119. """renvoie l'objet graphique hexagone de la case"""
  120. polygone = QPolygonF()
  121. if self.plateau.formeCases == "H":
  122. polygone << QPointF(((x*0.866)+0.2886)*self.plateau.hCase, y*self.plateau.hCase) \
  123. << QPointF(((x*0.866)+0.866)*self.plateau.hCase, y*self.plateau.hCase) \
  124. << QPointF(((x*0.866)+1.1547)*self.plateau.hCase, (y+0.5)*self.plateau.hCase) \
  125. << QPointF(((x*0.866)+0.866)*self.plateau.hCase, (y+1)*self.plateau.hCase) \
  126. << QPointF(((x*0.866)+0.2886)*self.plateau.hCase, (y+1)*self.plateau.hCase) \
  127. << QPointF( (x*0.866)*self.plateau.hCase, (y+0.5)*self.plateau.hCase)
  128. else:
  129. polygone << QPointF(x*self.plateau.hCase, y*self.plateau.hCase) \
  130. << QPointF((x+1)*self.plateau.hCase, y*self.plateau.hCase) \
  131. << QPointF((x+1)*self.plateau.hCase, (y+1)*self.plateau.hCase) \
  132. << QPointF(x*self.plateau.hCase, (y+1)*self.plateau.hCase)
  133. return polygone
  134. def lstVoisins(self, x, y):
  135. """renvoie la liste des cases voisines
  136. seulement cases existantes sur le plateau / seulement cases adjacentes (cas des cases carrees)"""
  137. voisins = []
  138. if self.plateau.formeCases == "H":
  139. lst = [(x, y-1), (x+1, y-0.5), (x+1, y+0.5), (x, y+1), (x-1, y+0.5), (x-1, y-0.5)]
  140. else:
  141. lst = [(x, y-1), (x+1, y-1), (x+1, y), (x+1, y+1), (x, y+1), (x-1, y+1), (x-1, y), (x-1, y-1)]
  142. for coord in lst:
  143. if self.plateau.coordonneesValides(coord): voisins.append(coord)
  144. return voisins
  145. ########################
  146. ########## fonctions liees a l'etat de la case ###########
  147. def majCouleur(self, couleur):
  148. self.couleur = couleur
  149. self.setBrush(couleur)
  150. self.polygoneChampDep.setBrush(self.couleurDep())
  151. pinceau = QPen()
  152. pinceau.setColor(self.couleurBordure())
  153. pinceau.setWidth(1)
  154. self.setPen(pinceau)
  155. def majTerrain(self, terrain):
  156. if terrain != None:
  157. self.terrain = terrain
  158. if len(self.terrain.imgTexture) > 0:
  159. self.setBrush(QBrush(QImage("img\\"+self.terrain.imgTexture)))
  160. else:
  161. if self.terrain.couleur.isValid():
  162. self.setBrush(QBrush(self.terrain.couleur))
  163. self.couleur = self.terrain.couleur
  164. self.polygoneChampDep.setBrush(self.couleurDep())
  165. else:
  166. self.terrain = Terrain()
  167. def majOccupation(self, objet, nouveauZ = None):
  168. """met a jour l'occupation de la case par les pions, decors..."""
  169. if objet != None:
  170. if not objet in self.occupeePar and nouveauZ != None:
  171. #on ajoute l'objet a la liste des objets occupant la case ou on met a jour son altitude
  172. casesOccupees = []
  173. for i in range(0, objet.hauteur):
  174. casesOccupees.append(self.altitude + nouveauZ + i)
  175. self.occupeePar[objet] = casesOccupees
  176. elif objet in self.occupeePar and nouveauZ == None:
  177. #on supprime l'objet de la liste des objets occupant la case
  178. del self.occupeePar[objet]
  179. else:
  180. pass
  181. def estOccupee(self, z=0):
  182. """renvoie vrai si la case correspondant a la hauteur z est occupee"""
  183. retour = False
  184. for objet in self.occupeePar:
  185. if objet.hauteur > 0 and z in self.occupeePar[objet]:
  186. retour = True
  187. return retour
  188. def estOccupeePar(self, z=0):
  189. """si la case correspondant a la hauteur z est occupee, renvoie l'objet en question
  190. sinon renvoie None"""
  191. retour = None
  192. for objet in self.occupeePar:
  193. if objet.hauteur > 0 and z in self.occupeePar[objet]:
  194. retour = objet
  195. return retour
  196. def pionOccupant(self):
  197. """si un pion occupe cette case, le renvoie"""
  198. retour = None
  199. for objet in self.occupeePar:
  200. if objet.__class__.__name__ == "Pion":
  201. retour = objet
  202. return retour
  203. def estFranchissable(self, z=0):
  204. """la case est-elle franchissable?"""
  205. retour = True
  206. if self.terrain.franchissable == False:
  207. retour = False
  208. else:
  209. if self.estOccupee(z) == True:
  210. retour = False
  211. return retour
  212. def estObstacleVision(self, hauteurObs):
  213. """renvoie vrai si la case et l'eventuel decor qui l'occupe bloquent le champ de
  214. vision d'un observateur situe a la hauteur precisee"""
  215. retour = False
  216. if self.altitude > hauteurObs:
  217. retour = True
  218. else:
  219. hauteurObstacle = 0
  220. for objet in self.occupeePar:
  221. if objet.hauteur > hauteurObstacle:
  222. hauteurObstacle = objet.hauteur
  223. if (self.altitude + hauteurObstacle) > hauteurObs:
  224. retour = True
  225. return retour
  226. def majEstDansChampDeplacement(self, actif):
  227. """la case apparait ou pas comme etant dans le champ de deplacement d'un pion"""
  228. self.polygoneChampDep.setVisible(actif)
  229. def majEstCibleAttaque(self, actif):
  230. """la case apparait ou pas comme etant dans le champ de deplacement d'un pion"""
  231. self.polygoneChampDep.setVisible(actif)
  232. def majEstCibleCurseur(self, actif, valide=True):
  233. """affiche la case comme etant la cible du curseur (lors d'une creation de pion, d'un deplacement...)"""
  234. if actif:
  235. pinceau = QPen()
  236. pinceau.setWidth(5)
  237. brush = QBrush()
  238. brush.setStyle(13)
  239. pinceau.setColor(self.couleurProj(valide))
  240. brush.setColor(self.couleurProj(valide))
  241. self.polygoneCible.setPen(pinceau)
  242. self.polygoneCible.setBrush(brush)
  243. self.setZValue(99)
  244. else:
  245. self.setZValue(0)
  246. self.polygoneCible.setVisible(actif)
  247. def majEffet(self, effet):
  248. """met a jour l'effet actif sur la case"""
  249. if self.polygoneEffet == None:
  250. #on cree le polygone utilise pour afficher les effets
  251. self.polygoneEffet = QGraphicsPolygonItem(self.polygone(self.x, self.y), parent=self)
  252. pinceau = QPen()
  253. pinceau.setColor(self.bordure)
  254. pinceau.setWidth(1)
  255. self.polygoneEffet.setPen(pinceau)
  256. self.polygoneEffet.setVisible(False)
  257. if len(effet) > 0 and effet != "aucun":
  258. #gradient de couleur
  259. gradient = QRadialGradient(self.centreGraphique, 0.5*self.plateau.hCase)
  260. couleur0 = QColor(0,0,0,0)
  261. couleur20 = self.couleurEffet[effet]
  262. couleur20.setAlpha(70)
  263. couleur10 = self.couleurEffet[effet]
  264. couleur10.setAlpha(255)
  265. gradient.setColorAt(0, couleur0)
  266. gradient.setColorAt(0.8, couleur0)
  267. gradient.setColorAt(0.95, couleur10)
  268. gradient.setColorAt(0.98, couleur20)
  269. gradient.setColorAt(1, couleur10)
  270. self.polygoneEffet.setBrush(gradient)
  271. self.polygoneEffet.setVisible(True)
  272. self.effetActif = effet
  273. else:
  274. self.polygoneEffet.setVisible(False)
  275. def majAltitude(self, altitude):
  276. """met a jour l'altitude de la case"""
  277. self.altitude = altitude
  278. def majAffichageSpecial(self, affichage=""):
  279. """donne a la case l'aspect demande en rendant visible/invisible le polygone d'affichage special"""
  280. #polygone d'affichage special (altitude, tactique...)
  281. if self.polygoneAffichageSpecial == None:
  282. self.polygoneAffichageSpecial = QGraphicsPolygonItem(self.polygone(self.x, self.y), parent=self)
  283. pinceau = QPen()
  284. pinceau.setColor(self.bordure)
  285. pinceau.setWidth(1)
  286. self.polygoneAffichageSpecial.setPen(pinceau)
  287. if affichage != "altitude" and self.etiquetteAltitude:
  288. self.etiquetteAltitude.setVisible(False)
  289. if affichage == "tactique":
  290. if self.terrain.franchissable:
  291. self.polygoneAffichageSpecial.setBrush(QColor(255,255,255))
  292. else:
  293. self.polygoneAffichageSpecial.setBrush(QColor(50,50,50))
  294. self.polygoneAffichageSpecial.setVisible(True)
  295. elif affichage == "altitude":
  296. if self.etiquetteAltitude == None:
  297. self.etiquetteAltitude = QGraphicsSimpleTextItem("{}".format(self.altitude), parent=self)
  298. police = QFont("Georgia", 18)
  299. police.setItalic(True)
  300. self.etiquetteAltitude.setFont(police)
  301. self.etiquetteAltitude.setPos(QPointF(((self.x*0.866)+0.65)*self.plateau.hCase, (self.y+0.7)*self.plateau.hCase))
  302. if self.altitude >= 0:
  303. couleur = QColor("red").lighter(200-(5*self.altitude))
  304. else:
  305. couleur = QColor("purple").lighter(200+(5*self.altitude))
  306. self.polygoneAffichageSpecial.setBrush(couleur)
  307. self.polygoneAffichageSpecial.setZValue(5)
  308. self.polygoneAffichageSpecial.setVisible(True)
  309. self.etiquetteAltitude.setText(QString.fromUtf8("{}".format(self.altitude)))
  310. self.etiquetteAltitude.setVisible(True)
  311. self.etiquetteAltitude.setZValue(6)
  312. else:
  313. self.polygoneAffichageSpecial.setVisible(False)
  314. def couleurDep(self):
  315. """renvoie une couleur secondaire utilisee pour les champs de deplacements"""
  316. luminositeActuelle = self.couleurEffective().lightness()
  317. if luminositeActuelle < 150:
  318. retour = QColor(210,255,255,100)
  319. else:
  320. retour = QColor(210,255,255,100)
  321. return retour
  322. def couleurProj(self, valide):
  323. """renvoie une couleur secondaire utilisee pour les projections de deplacement"""
  324. luminositeActuelle = self.couleurEffective().lightness()
  325. if valide:
  326. retour = QColor("white")
  327. else:
  328. retour = QColor("red")
  329. if luminositeActuelle > 220:
  330. retour = retour.darker(120)
  331. return retour
  332. def couleurBordure(self):
  333. """renvoie la couleur utilisee pour les bordures de la case"""
  334. luminositeActuelle = self.couleurEffective().lightness()
  335. if luminositeActuelle > 150:
  336. retour = QColor(85, 85, 85, 130)
  337. elif luminositeActuelle > 100 and luminositeActuelle <= 150:
  338. retour = QColor(10, 10, 10, 130)
  339. else:
  340. retour = QColor(255, 255, 255, 180)
  341. return retour
  342. def couleurEffective(self):
  343. """renvoie la couleur effective de la case (utile quand la texture est une image)"""
  344. texture = self.brush()
  345. if texture.textureImage().isNull():
  346. couleurFond = texture.color()
  347. else:
  348. couleurFond = self.couleurDominante(texture.textureImage())
  349. return couleurFond
  350. def couleurDominante(self, img):
  351. """retourne la couleur dominante d'une QImage"""
  352. r = 0
  353. v = 0
  354. b = 0
  355. for i in range(0, img.width()):
  356. for j in range(0, img.height()):
  357. pixel = img.pixel(i,j)
  358. r += qRed(pixel)
  359. v += qGreen(pixel)
  360. b += qBlue(pixel)
  361. nb_pix = img.width() * img.height()
  362. r_moy = int(r / nb_pix)
  363. v_moy = int(v / nb_pix)
  364. b_moy = int(b / nb_pix)
  365. return QColor(r_moy, v_moy, b_moy)
  366. ######### evebements souris et clavier #################
  367. #*EC
  368. def mousePressEvent(self, event):
  369. """evenement lors du clic souris"""
  370. super(Case, self).mousePressEvent(event)
  371. if event.button() == 1: #sur clic gauche
  372. accepte = self.plateau.caseCliquee((self.x, self.y))
  373. if accepte: event.accept()
  374. else:
  375. event.ignore()
  376. def hoverEnterEvent(self, event):
  377. """met a jour l'affichage de la case au survol de la souris, si un pion est selectionne"""
  378. super(Case, self).hoverEnterEvent(event)
  379. self.plateau.caseSurvol(self)
  380. def hoverLeaveEvent(self, event):
  381. """met a jour l'affichage de la case au survol de la souris, si un pion est selectionne"""
  382. super(Case, self).hoverLeaveEvent(event)
  383. #self.plateau.caseSurvol()
  384. ########################