Plateau.py 125 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465
  1. #from __future__ import unicode_literals
  2. # -*- coding: utf-8 -*-
  3. from __future__ import division
  4. import os
  5. from sys import exit, argv, getsizeof, settrace
  6. from time import time, sleep, strftime, localtime
  7. from threading import Thread
  8. from PyQt4.QtCore import *
  9. from PyQt4.QtGui import *
  10. from PyQt4 import QtOpenGL
  11. ##from ui.ecran_editionAttaques import Ui_editionAttaques
  12. from Case import Case
  13. from Pion import Pion
  14. from PionDecor import PionDecor
  15. from Forme import Forme
  16. from Cache import Cache
  17. from EntreeSortie import EntreeSortie
  18. from EcranEditionCreature import EcranEditionCreature
  19. from EcranEditionDecors import EcranEditionDecors
  20. from EcranEditionTerrain import EcranEditionTerrain
  21. from EcranAffichageTexte import EcranAffichageTexte
  22. from EcranGestionCombat import EcranGestionCombat
  23. from EcranEditionAttaques import EcranEditionAttaques
  24. import regles as regles
  25. from outilsSvg import *
  26. from lancer import jet, estJetValide
  27. from operator import itemgetter, attrgetter
  28. from math import *
  29. class Plateau(QGraphicsScene):
  30. """plateau de jeu contenant les cases, decors et pions"""
  31. def __init__(self, fenetre, parent=None):
  32. super(Plateau, self).__init__()
  33. #parametres et variables
  34. self.id = ""
  35. self.nom = ""
  36. self.chapitre = 0
  37. self.fenetre = fenetre
  38. self.nbCasesX = 0
  39. self.nbCasesY = 0
  40. self.hCase = 0
  41. self.cases = {} #dict des cases du plateau (coordonnées: case)
  42. self.pions = {} #liste de pions positionnes sur le plateau
  43. self.decors = {} #liste des decors places sur le plateau
  44. self.ordreJeu = {} #numero du pion: ordre de jeu
  45. self.numPionEnCours = 0
  46. self.tour = 1
  47. self.cacheEnCours = 1
  48. self.caches = {}
  49. self.polygonesCaches = {}
  50. for i in range(1,5):
  51. self.caches[i] = {"actif":False,"listeCases":[]}
  52. self.polygonesCaches[i] = None
  53. self.listeCasesZonePlacement = []
  54. self.polygoneZonePlacement = None
  55. self.entreesSorties = []
  56. self.modePrincipal = "creation"
  57. self.modeActif = ""
  58. self.modeCombat = ""
  59. self.nbZoomActuel = 0
  60. self.epaisseurpinceau = 0
  61. self.enCours = False
  62. self.public = False
  63. self.dateCreation = ""
  64. self.dateSvg = ""
  65. self.notes = ""
  66. #note: la hauteur Z (qui gere l'empilement des objets graphiques est distribuee de cette maniere:
  67. #cases : 0 a 9
  68. #pions : 10 et +
  69. def __getstate__(self):
  70. self.dateSvg = time()
  71. state = {key:value for key, value in self.__dict__.items() if not key in ["fenetre", "modeActif", "listMode", "cacheEnCours", \
  72. "modeParam", "nbZoomActuel", "epaisseurPinceau", \
  73. "editionTerrain", "editionCreature", "editionDecor", \
  74. "polygoneZonePlacement", "gestionCombat", "polygonesCaches", \
  75. "editionAttaques"]}
  76. return (state)
  77. def __setstate__(self, state):
  78. self.__dict__ = state
  79. def creer(self, idPlateau, nom, chapitre, formeCases, nbCasesX, nbCasesY, couleur = QColor(0, 255, 0, 80)):
  80. """cree le plateau"""
  81. self.id = idPlateau
  82. self.nom = nom
  83. self.chapitre = chapitre
  84. self.dateCreation = time()
  85. self.hCase = 120 #hauteur d'une case
  86. self.nbCasesX = nbCasesX #nb cases en x
  87. self.nbCasesY = nbCasesY #nb cases en y
  88. self.formeCases = formeCases
  89. self.gestionCombat = None
  90. self.initialisationGraphique()
  91. self.connexions()
  92. #cree les cases hexagonales
  93. for x in range(nbCasesX):
  94. for y in range(nbCasesY):
  95. if formeCases == "H":
  96. if 1 == (x % 2):
  97. y += 0.5
  98. c = Case(self)
  99. c.creer(x, y, couleur)
  100. self.cases[(x,y)] = c
  101. def recreer(self, fenetre):
  102. self.fenetre = fenetre
  103. self.gestionCombat = None
  104. super(Plateau, self).__init__()
  105. self.connexions()
  106. self.initialisationGraphique()
  107. #recreation des cases
  108. for coord in self.cases:
  109. self.cases[coord].recreer(self)
  110. #recreation des pions
  111. for numPion in self.pions:
  112. self.pions[numPion].recreer(self)
  113. self.majOrdreJeu()
  114. #recreation des decors
  115. for num in self.decors:
  116. self.decors[num].recreer(self)
  117. #recreation des marqueurs entree/sortie
  118. for entreeSortie in self.entreesSorties:
  119. entreeSortie.recreer(self)
  120. #recreation de la zone de placement:
  121. if len(self.listeCasesZonePlacement) > 0:
  122. self.polygoneZonePlacement = None
  123. self.majZonePlacement(self.listeCasesZonePlacement)
  124. #recreation des caches
  125. self.polygonesCaches = {}
  126. for i in range(1,5):
  127. self.polygonesCaches[i] = None
  128. def fermer(self):
  129. """ferme le plateau 'proprement'"""
  130. self.miniature()
  131. for item in self.items():
  132. item.prepareGeometryChange()
  133. self.removeItem(item)
  134. if self.gestionCombat != None:
  135. del self.gestionCombat
  136. self.fenetre.reinitialiserPanneauxPlateau()
  137. def miniature(self):
  138. """renvoie une miniature du plateau (QPixMap compresse) qui sera enregistree avec les infos de la sauvegarde"""
  139. ## img = QImage(128, 128, QImage.Format_ARGB32_Premultiplied)
  140. img = QPixmap(1024, 768)
  141. img.fill(QColor("white"))
  142. peintre = QPainter(img)
  143. rendu = self.render(peintre)
  144. peintre.end()
  145. img.scaledToHeight(128, Qt.FastTransformation)
  146. def connexions(self):
  147. """connecte le plateau aux differents widgets de la fenetre principale"""
  148. self.fenetre.connect(self.fenetre.ui.modeCreationPlateau, SIGNAL("clicked()"), self.plateauModeCreation, Qt.UniqueConnection)
  149. self.fenetre.connect(self.fenetre.ui.modeCombatPlateau, SIGNAL("clicked()"), self.plateauModeCombat, Qt.UniqueConnection)
  150. self.fenetre.connect(self.fenetre.ui.afficherGestionCombat, SIGNAL("clicked()"), self.afficheEcranGestionCombat, Qt.UniqueConnection)
  151. self.fenetre.connect(self.fenetre.ui.terrainCouleur, SIGNAL("clicked()"), self.modePeintureCase, Qt.UniqueConnection)
  152. self.fenetre.connect(self.fenetre.ui.terrainCopie, SIGNAL("clicked()"), self.modeCopieTerrain, Qt.UniqueConnection)
  153. self.fenetre.connect(self.fenetre.ui.pionCouleur, SIGNAL("clicked()"), self.majCouleurPion, Qt.UniqueConnection)
  154. self.fenetre.connect(self.fenetre.ui.terrainEdit, SIGNAL("clicked()"), self.terrainEdit, Qt.UniqueConnection)
  155. self.fenetre.connect(self.fenetre.ui.terrainNouveau, SIGNAL("clicked()"), self.terrainNouveau, Qt.UniqueConnection)
  156. self.fenetre.connect(self.fenetre.ui.creatureEdit, SIGNAL("clicked()"), self.creatureEdit, Qt.UniqueConnection)
  157. self.fenetre.connect(self.fenetre.ui.creatureNouveau, SIGNAL("clicked()"), self.creatureNouveau, Qt.UniqueConnection)
  158. self.fenetre.connect(self.fenetre.ui.pionSimpleCreer, SIGNAL("clicked()"), self.modeCreationPion, Qt.UniqueConnection)
  159. self.fenetre.connect(self.fenetre.ui.pionSupprimer, SIGNAL("clicked()"), self.majModePionSupprimer, Qt.UniqueConnection)
  160. self.fenetre.connect(self.fenetre.ui.decorEdit, SIGNAL("clicked()"), self.decorEdit, Qt.UniqueConnection)
  161. self.fenetre.connect(self.fenetre.ui.decorNouveau, SIGNAL("clicked()"), self.decorNouveau, Qt.UniqueConnection)
  162. self.fenetre.connect(self.fenetre.ui.decorSupprimer, SIGNAL("clicked()"), self.majModePionSupprimer, Qt.UniqueConnection)
  163. self.fenetre.connect(self.fenetre.ui.definirEntree, SIGNAL("clicked()"), self.majModeDefinirEntree, Qt.UniqueConnection)
  164. self.fenetre.connect(self.fenetre.ui.definirSortie, SIGNAL("clicked()"), self.majModeDefinirSortie, Qt.UniqueConnection)
  165. self.fenetre.connect(self.fenetre.ui.definirZonePlacement, SIGNAL("clicked()"), self.majModeZonePlacement, Qt.UniqueConnection)
  166. self.fenetre.connect(self.fenetre.ui.formeSimple, SIGNAL("clicked()"), self.majModeForme, Qt.UniqueConnection)
  167. self.fenetre.connect(self.fenetre.ui.formeLigne, SIGNAL("clicked()"), self.majModeForme, Qt.UniqueConnection)
  168. self.fenetre.connect(self.fenetre.ui.formeLigneOrientee, SIGNAL("clicked()"), self.majModeForme, Qt.UniqueConnection)
  169. self.fenetre.connect(self.fenetre.ui.formeEllipseVide, SIGNAL("clicked()"), self.majModeForme, Qt.UniqueConnection)
  170. self.fenetre.connect(self.fenetre.ui.formeEllipsePlein, SIGNAL("clicked()"), self.majModeForme, Qt.UniqueConnection)
  171. self.fenetre.connect(self.fenetre.ui.formeRectVide, SIGNAL("clicked()"), self.majModeForme, Qt.UniqueConnection)
  172. self.fenetre.connect(self.fenetre.ui.formeRectPlein, SIGNAL("clicked()"), self.majModeForme, Qt.UniqueConnection)
  173. self.fenetre.connect(self.fenetre.ui.effFeu, SIGNAL("clicked()"), self.modeCaseEffet, Qt.UniqueConnection)
  174. self.fenetre.connect(self.fenetre.ui.effEau, SIGNAL("clicked()"), self.modeCaseEffet, Qt.UniqueConnection)
  175. self.fenetre.connect(self.fenetre.ui.effGlace, SIGNAL("clicked()"), self.modeCaseEffet, Qt.UniqueConnection)
  176. self.fenetre.connect(self.fenetre.ui.effPoison, SIGNAL("clicked()"), self.modeCaseEffet, Qt.UniqueConnection)
  177. self.fenetre.connect(self.fenetre.ui.effEffacer, SIGNAL("clicked()"), self.modeCaseEffet, Qt.UniqueConnection)
  178. self.fenetre.connect(self.fenetre.ui.cacheActif, SIGNAL("clicked()"), self.majEtatCacheEnCours, Qt.UniqueConnection)
  179. self.fenetre.connect(self.fenetre.ui.cacheVoir, SIGNAL("clicked()"), self.voirCacheEnCours, Qt.UniqueConnection)
  180. self.fenetre.connect(self.fenetre.ui.cachePlacer, SIGNAL("clicked()"), self.placerCacheEnCours, Qt.UniqueConnection)
  181. self.fenetre.connect(self.fenetre.ui.combatDeplacement, SIGNAL("clicked()"), self.majModeCombatDeplacement, Qt.UniqueConnection)
  182. self.fenetre.ui.combatDeplacement.setCheckable(True)
  183. self.fenetre.connect(self.fenetre.ui.combatAttaqueCaC, SIGNAL("clicked()"), self.majModeCombatAttaqueCaC, Qt.UniqueConnection)
  184. self.fenetre.ui.combatAttaqueCaC.setCheckable(True)
  185. self.fenetre.connect(self.fenetre.ui.combatAttaqueDist, SIGNAL("clicked()"), self.majModeCombatAttaqueDist, Qt.UniqueConnection)
  186. self.fenetre.ui.combatAttaqueDist.setCheckable(True)
  187. self.fenetre.connect(self.fenetre.ui.combatAttaqueZone, SIGNAL("clicked()"), self.fenetre.barreCombatZone, Qt.UniqueConnection)
  188. self.fenetre.connect(self.fenetre.ui.combatAttaqueZone, SIGNAL("clicked()"), self.majModeCombatZone, Qt.UniqueConnection)
  189. self.fenetre.ui.combatAttaqueZone.setCheckable(True)
  190. self.fenetre.connect(self.fenetre.ui.combatVol, SIGNAL("clicked()"), self.fenetre.barreCombatVol, Qt.UniqueConnection)
  191. self.fenetre.connect(self.fenetre.ui.combatVol_altitude, SIGNAL("valueChanged(int)"), self.majZPion, Qt.UniqueConnection)
  192. #self.fenetre.connect(self.fenetre.ui.combatVol_altitude, SIGNAL("editingFinished()"), self.majVisibiliteBarreCombat)
  193. self.fenetre.connect(self.fenetre.ui.combatZone_ligne, SIGNAL("clicked()"), self.majModeCombatZoneForme, Qt.UniqueConnection)
  194. self.fenetre.ui.combatZone_ligne.setCheckable(True)
  195. self.fenetre.connect(self.fenetre.ui.combatZone_disque, SIGNAL("clicked()"), self.majModeCombatZoneForme, Qt.UniqueConnection)
  196. self.fenetre.ui.combatZone_disque.setCheckable(True)
  197. self.fenetre.connect(self.fenetre.ui.combatZone_cone, SIGNAL("clicked()"), self.majModeCombatZoneForme, Qt.UniqueConnection)
  198. self.fenetre.ui.combatZone_cone.setCheckable(True)
  199. self.fenetre.connect(self.fenetre.ui.combatRetour, SIGNAL("clicked()"), self.fenetre.barreCombatMenu, Qt.UniqueConnection)
  200. self.fenetre.connect(self.fenetre.ui.combatPasserTour, SIGNAL("clicked()"), self.pionSuivant, Qt.UniqueConnection)
  201. #self.connect(self.ui.combatInventaire, SIGNAL("clicked()"), self.majModePionSupprimer)
  202. self.fenetre.connect(self.fenetre.ui.modeAffichagePlateau, SIGNAL("currentIndexChanged(int)"), self.majModeAffichage, Qt.UniqueConnection)
  203. self.fenetre.connect(self.fenetre.ui.listTerrains, SIGNAL("cellClicked(int,int)"), self.modeMajTerrainCase, Qt.UniqueConnection)
  204. self.fenetre.connect(self.fenetre.ui.listCreatures, SIGNAL("cellClicked(int,int)"), self.modeCreationCreaturePion, Qt.UniqueConnection)
  205. self.fenetre.connect(self.fenetre.ui.listDecors, SIGNAL("cellClicked(int,int)"), self.modeCreationDecor, Qt.UniqueConnection)
  206. self.fenetre.connect(self.fenetre.ui.epaisseurPinceau, SIGNAL("valueChanged(int)"), self.majEpaisseurPinceau, Qt.UniqueConnection)
  207. self.fenetre.connect(self.fenetre.ui.altitudeCase, SIGNAL("valueChanged(double)"), self.modeMajAltitudeCase, Qt.UniqueConnection)
  208. self.fenetre.connect(self.fenetre.ui.infoOrdreJeu, SIGNAL("cellClicked(int,int)"), self.clicListOrdreJeu, Qt.UniqueConnection)
  209. self.fenetre.connect(self.fenetre.ui.listeAttributs, SIGNAL("cellChanged(int,int)"), self.listeAttributCelluleModifiee, Qt.UniqueConnection)
  210. self.fenetre.connect(self.fenetre.ui.editerAttaques, SIGNAL("clicked()"), self.afficheEcranEditionAttaques, Qt.UniqueConnection)
  211. self.fenetre.connect(self.fenetre.ui.notesPion, SIGNAL("textChanged()"), self.majNotesPion, Qt.UniqueConnection)
  212. self.fenetre.connect(self.fenetre.ui.notesMjPlateau, SIGNAL("textChanged()"), self.majNotesPlateau, Qt.UniqueConnection)
  213. self.fenetre.connect(self.fenetre.ui.agrandirNotesMjPlateau, SIGNAL("clicked()"), self.agrandirNotesMjPlateau, Qt.UniqueConnection)
  214. def initialisationGraphique(self):
  215. """cree la scene graphique et les parametres necessaires a son fonctionnement, et met a jour l'interface"""
  216. #on cree la scene graphique
  217. kx = 1
  218. if self.formeCases == "H":
  219. kx = 0.866
  220. marge = 240
  221. self.setSceneRect(0 - marge, 0 - marge, (kx*self.hCase*(self.nbCasesX+2)) + marge, (self.hCase*(self.nbCasesY+2)) + marge)
  222. self.fenetre.ui.vuePlateau.setScene(self)
  223. self.fenetre.ui.vuePlateau.scale(0.25, 0.25)
  224. self.fenetre.ui.vuePlateau.centerOn(QPointF(0,0))
  225. self.fenetre.ui.vuePlateau.setDragMode(1)
  226. self.setItemIndexMethod(QGraphicsScene.BspTreeIndex)
  227. #self.setItemIndexMethod(QGraphicsScene.NoIndex)
  228. self.nbZoomActuel = 0
  229. self.epaisseurPinceau = 1
  230. self.polygoneZonePlacement = None
  231. self.polygonesCaches = {}
  232. for i in range(1,5):
  233. self.polygonesCaches[i] = None
  234. #gestion du mode d'interaction avec le plateau
  235. self.modeActif = "standard"
  236. self.listMode = ["standard", "caseMajTerrain", "caseCopie", "caseMajEffet", "caseMajAltitude", "pionCreation", \
  237. "pionSelectionne", "pionDecorCreation", "pionDecorSelectionne", "pionSupprimer", "cachePlacer", \
  238. "combatDeplacement", "combatAttaqueCaC", "combatAttaqueDist", \
  239. "placementEntreeSortie", "majZonePlacement"]
  240. self.modeParam = {"terrain": None, "numPionSelectionne": 0, "pionNouveau": None, "creature": None, "decor": None, "effet":"", "couleurPion": QColor("grey"), \
  241. "coordProjectionPosition": (0,0), "formeProjectionPosition": None, "nbRotations": 0, \
  242. "typeFormeDessin": "simple", "formeDessin": None, "origineFormeDessin": None, "point2FormeDessin": None, "listeCasesFormeDessin": [], \
  243. "projectionFormeDessin": [], "formeCoordEnCours": (0,0), \
  244. "zoneAttaqueCaC": [], "cibleAttaqueCaC": None, "cibleAttaqueDist": None, "pionCibleAttaqueDist": None, "ligneAttaqueDist": None, \
  245. "typeAttaqueZone": "", "formeAttaqueZone": None, "origineAttaqueZone": None, "point2AttaqueZone": None, "listeCasesAttaqueZone": [], "ligneMireAttaqueZone": None, \
  246. "entreeSortie": None}
  247. #mise a jour de l'interface de creation
  248. self.fenetre.majVisibilitePanneauxPlateau("creation")
  249. self.fenetre.ui.nomPlateau.setText(QString.fromUtf8(self.nom))
  250. self.fenetre.ui.nomPlateau.setText(QString.fromUtf8(self.nom))
  251. self.majBoutonsCouleursPerso()
  252. self.majBoutonsCaches()
  253. self.majListTerrains()
  254. self.majListCreatures()
  255. self.majListDecors()
  256. self.initListeOrdreJeu()
  257. self.initListeAttaques()
  258. self.fenetre.ui.notesMjPlateau.setText(QString.fromUtf8(self.notes))
  259. if self.modeParam["couleurPion"].isValid():
  260. self.fenetre.ui.pionCouleur.setStyleSheet("QPushButton {backGround:%s}" %(self.modeParam["couleurPion"].name()))
  261. #mise a jour de l'interface d'informations
  262. self.majInfosPion(None)
  263. self.majInfosDecor(None)
  264. def estCree(self):
  265. """renvoie vrai si des cases ont ete creees"""
  266. if len(self.cases) > 0:
  267. retour = True
  268. else:
  269. retour = False
  270. return retour
  271. def majBoutonsCouleursPerso(self):
  272. """met a jour l'affichage des couleurs customisees dans la boite de dialogue de selection de couleur"""
  273. for i in range(0,16):
  274. rgb = QColorDialog(self.fenetre.ui.vuePlateau).customColor(i)
  275. couleur = QColor(rgb)
  276. bouton = self.fenetre.ui.outilsEditionPlateau.findChild(QToolButton, "terrainCouleurPerso{}".format(i+1))
  277. if couleur.isValid():
  278. bouton.setStyleSheet("QToolButton {backGround:%s}"%(couleur.name()))
  279. self.fenetre.connect(bouton, SIGNAL("clicked()"), self.modePeintureCase_perso)
  280. def chercherCouleur(self):
  281. """ouvre la boite de dialogue de selection de couleur"""
  282. couleur = QColorDialog(self.fenetre.ui.vuePlateau).getColor(QColor("white"), self.fenetre.ui.vuePlateau)
  283. self.majBoutonsCouleursPerso()
  284. return couleur
  285. def majBoutonsCaches(self):
  286. """met a jour l'affichage et connecte les boutons de caches"""
  287. for i in range(1,5):
  288. bouton = self.fenetre.ui.outilsEditionPlateau.findChild(QToolButton, "plateauCache{}".format(i))
  289. self.fenetre.connect(bouton, SIGNAL("clicked()"), self.majAffichageMenuCache)
  290. def majListesPions(self, numPion = None):
  291. """met a jour les listes contenant des donnees liees aux pions"""
  292. ## self.majListeOrdreJeu()
  293. ##
  294. ## if numPion == None or numPion == self.pionSelectionne().numero:
  295. ## self.majListeAttributs()
  296. ## QApplication.processEvents()
  297. pass
  298. def majNotesPlateau(self):
  299. """les notes du plateau ont ete maj a l'ecran"""
  300. self.notes = str(self.fenetre.ui.notesMjPlateau.toPlainText().toUtf8())
  301. def agrandirNotesMjPlateau(self):
  302. """affiche les notes du plateau dans une QDialog, puis recupere les donnees qui y sont saisies"""
  303. affichageTexte = EcranAffichageTexte(self.notes)
  304. affichageTexte.setAttribute(Qt.WA_DeleteOnClose)
  305. r = affichageTexte.exec_()
  306. self.fenetre.ui.notesMjPlateau.setText(QString.fromUtf8(affichageTexte.recupererTexte()))
  307. ##### affichage de la liste des terrains enregistres, et fonctions d'acces aux donnees""""
  308. def afficheEcranEditionTerrains(self, terrain = None):
  309. """affiche l'ecran d'edition/creation de terrains"""
  310. self.editionTerrain = EcranEditionTerrain(terrain)
  311. self.connect(self.fenetre, SIGNAL("majListTerrains()"), self.majListTerrains)
  312. self.connect(self.editionTerrain, SIGNAL("majListTerrains()"), self.majListTerrains)
  313. self.editionTerrain.setAttribute(Qt.WA_DeleteOnClose)
  314. r = self.editionTerrain.exec_()
  315. def majListTerrains(self):
  316. """mise a jour de la liste des terrains depuis la sauvegarde"""
  317. dico = afficheSvg("lib\\biblio\\terrain")
  318. while self.fenetre.ui.listTerrains.rowCount() > 0:
  319. self.fenetre.ui.listTerrains.removeRow(0)
  320. index = 0
  321. #self.fenetre.ui.listTerrains.setColumnWidth(0,0)
  322. self.fenetre.ui.listTerrains.hideColumn(0)
  323. self.fenetre.ui.listTerrains.sizeHintForColumn(1)
  324. for elt in dico:
  325. terrain = dico[elt]
  326. self.fenetre.ui.listTerrains.insertRow(int(index))
  327. self.fenetre.ui.listTerrains.setItem(int(index),0,QTableWidgetItem(QString.fromUtf8(elt)))
  328. icon = QIcon("img\\"+terrain.imgTexture)
  329. item = QTableWidgetItem(icon,QString.fromUtf8(terrain.nom))
  330. self.fenetre.ui.listTerrains.setItem(int(index),1,item)
  331. index += 1
  332. self.fenetre.ui.listTerrains.setIconSize(QSize(30,20))
  333. self.fenetre.ui.listTerrains.sizeHintForColumn(1)
  334. self.fenetre.ui.listTerrains.sortItems(0)
  335. self.majMode("standard")
  336. def terrainEdit(self):
  337. """ouvre la fenetre 'terrains' en mode edition"""
  338. index = self.fenetre.ui.listTerrains.item(self.fenetre.ui.listTerrains.currentRow(), 0)
  339. if index > 0:
  340. terrain = charger("lib\\biblio\\terrain", str(index.text().toUtf8()))
  341. self.afficheEcranEditionTerrains(terrain)
  342. def terrainNouveau(self):
  343. """ouvre la fenetre 'terrains' en mode edition"""
  344. self.afficheEcranEditionTerrains()
  345. ###############
  346. ##### affichage de la liste des creatures enregistrees, et fonctions d'acces aux donnees""""
  347. def afficheEcranEditionCreatures(self, creature = None):
  348. """affiche l'ecran d'edition/creation de creatures"""
  349. self.editionCreature = EcranEditionCreature(creature, self.formeCases)
  350. self.editionCreature.setAttribute(Qt.WA_DeleteOnClose)
  351. self.connect(self.fenetre, SIGNAL("majListCreatures()"), self.majListCreatures)
  352. self.connect(self.editionCreature, SIGNAL("majListCreatures()"), self.majListCreatures)
  353. r = self.editionCreature.exec_()
  354. def majListCreatures(self):
  355. """mise a jour de la liste des creatures depuis la sauvegarde"""
  356. dico = afficheSvg("lib\\biblio\\creature")
  357. while self.fenetre.ui.listCreatures.rowCount() > 0:
  358. self.fenetre.ui.listCreatures.removeRow(0)
  359. index = 0
  360. #self.fenetre.ui.listTerrains.setColumnWidth(0,0)
  361. self.fenetre.ui.listCreatures.hideColumn(0)
  362. self.fenetre.ui.listCreatures.sizeHintForColumn(1)
  363. for elt in dico:
  364. creature = dico[elt]
  365. self.fenetre.ui.listCreatures.insertRow(int(index))
  366. self.fenetre.ui.listCreatures.setItem(int(index),0,QTableWidgetItem(QString.fromUtf8(elt)))
  367. icon = QIcon("img\\"+creature.logo)
  368. item = QTableWidgetItem(icon,QString.fromUtf8(creature.nom))
  369. self.fenetre.ui.listCreatures.setItem(int(index),1,item)
  370. index += 1
  371. self.fenetre.ui.listCreatures.setIconSize(QSize(30,20))
  372. self.fenetre.ui.listCreatures.sizeHintForColumn(1)
  373. ## self.fenetre.ui.listCreatures.sortItems(0)
  374. ## trierTable(self.fenetre.ui.listCreatures, 0, 0)
  375. self.majMode("standard")
  376. def creatureEdit(self):
  377. """ouvre la fenetre 'creatures' en mode edition"""
  378. index = self.fenetre.ui.listCreatures.item(self.fenetre.ui.listCreatures.currentRow(), 0)
  379. if index > 0:
  380. creature = charger("lib\\biblio\\creature", str(index.text().toUtf8()))
  381. self.afficheEcranEditionCreatures(creature)
  382. def creatureNouveau(self):
  383. """ouvre la fenetre 'creatures' en mode edition"""
  384. self.afficheEcranEditionCreatures()
  385. ###############
  386. ##### affichage de la liste des decors enregistrees, et fonctions d'acces aux donnees""""
  387. def afficheEcranEditionDecors(self, decor = None):
  388. """affiche l'ecran d'edition/creation de decors"""
  389. self.editionDecor = EcranEditionDecors(decor, self.formeCases)
  390. self.editionDecor.setAttribute(Qt.WA_DeleteOnClose)
  391. self.connect(self.fenetre, SIGNAL("majListDecors()"), self.majListDecors)
  392. self.connect(self.editionDecor, SIGNAL("majListDecors()"), self.majListDecors)
  393. r = self.editionDecor.exec_()
  394. def majListDecors(self):
  395. """mise a jour de la liste des decors depuis la sauvegarde"""
  396. dico = afficheSvg("lib\\biblio\\decor")
  397. while self.fenetre.ui.listDecors.rowCount() > 0:
  398. self.fenetre.ui.listDecors.removeRow(0)
  399. index = 0
  400. #self.fenetre.ui.listTerrains.setColumnWidth(0,0)
  401. self.fenetre.ui.listDecors.hideColumn(0)
  402. self.fenetre.ui.listDecors.sizeHintForColumn(1)
  403. for elt in dico:
  404. decor = dico[elt]
  405. self.fenetre.ui.listDecors.insertRow(int(index))
  406. self.fenetre.ui.listDecors.setItem(int(index),0,QTableWidgetItem(QString.fromUtf8(elt)))
  407. icon = QIcon("img\\"+decor.logo)
  408. item = QTableWidgetItem(icon,QString.fromUtf8(decor.nom))
  409. self.fenetre.ui.listDecors.setItem(int(index),1,item)
  410. index += 1
  411. self.fenetre.ui.listDecors.setIconSize(QSize(30,20))
  412. self.fenetre.ui.listDecors.sizeHintForColumn(1)
  413. ## self.fenetre.ui.listDecors.sortItems(0)
  414. ## trierTable(self.fenetre.ui.listDecors, 0, 0)
  415. self.majMode("standard")
  416. def decorEdit(self):
  417. """ouvre la fenetre 'decors' en mode edition"""
  418. index = self.fenetre.ui.listDecors.item(self.fenetre.ui.listDecors.currentRow(), 0)
  419. if index > 0:
  420. decor = charger("lib\\biblio\\decor", str(index.text().toUtf8()))
  421. self.afficheEcranEditionDecors(decor)
  422. def decorNouveau(self):
  423. """ouvre la fenetre 'decors' en mode edition"""
  424. self.afficheEcranEditionDecors()
  425. ###############
  426. ########### affichage des curseurs personnalises ########
  427. def curseurSelection(self):
  428. self.fenetre.ui.vuePlateau.setDragMode(0)
  429. curseurPix = QPixmap("img\\curseurEpee.png")#.scaledToHeight(55, Qt.FastTransformation)
  430. curseurPix.setMask(curseurPix.createHeuristicMask())
  431. curseurPointe = QCursor(curseurPix, 0, 0)
  432. self.fenetre.ui.vuePlateau.setCursor(curseurPointe)
  433. def curseurPinceau(self):
  434. self.fenetre.ui.vuePlateau.setDragMode(0)
  435. curseurPix = QPixmap("img\\curseurPinceau.png")
  436. curseurPinceau = QCursor(curseurPix, 0, curseurPix.height())
  437. self.fenetre.ui.vuePlateau.setCursor(curseurPinceau)
  438. def curseurGomme(self):
  439. self.fenetre.ui.vuePlateau.setDragMode(0)
  440. curseurPix = QPixmap("img\\gomme.png")#.scaledToHeight(55, Qt.FastTransformation)
  441. curseurGomme = QCursor(curseurPix, 0, 0)
  442. self.fenetre.ui.vuePlateau.setCursor(curseurGomme)
  443. def curseurSeringue(self):
  444. self.fenetre.ui.vuePlateau.setDragMode(0)
  445. curseurPix = QPixmap("img\\curseurSeringue.png")
  446. curseurSeringue = QCursor(curseurPix, 0, 0)
  447. self.fenetre.ui.vuePlateau.setCursor(curseurSeringue)
  448. def curseurEpee(self, valide = True):
  449. self.fenetre.ui.vuePlateau.setDragMode(0)
  450. if valide:
  451. curseurPix = QPixmap("img\\curseurEpee.png")
  452. curseurSeringue = QCursor(curseurPix, 0, 0)
  453. self.fenetre.ui.vuePlateau.setCursor(curseurSeringue)
  454. def curseurArc(self):
  455. self.fenetre.ui.vuePlateau.setDragMode(0)
  456. curseurPix = QPixmap("img\\arc.png")
  457. curseurSeringue = QCursor(curseurPix, curseurPix.width(), 0)
  458. self.fenetre.ui.vuePlateau.setCursor(curseurSeringue)
  459. def curseurBaguette(self):
  460. self.fenetre.ui.vuePlateau.setDragMode(0)
  461. curseurPix = QPixmap("img\\baguette.png")
  462. curseurBaguette = QCursor(curseurPix, curseurPix.width(), 0)
  463. self.fenetre.ui.vuePlateau.setCursor(curseurBaguette)
  464. ############
  465. ##### activation des differents modes d'interaction avec le plateau et mises a jour des principaux parametres #######
  466. def plateauModeCreation(self):
  467. self.fenetre.majVisibilitePanneauxPlateau("creation")
  468. self.modePrincipal = "creation"
  469. def plateauModeCombat(self):
  470. self.fenetre.majVisibilitePanneauxPlateau("combat")
  471. self.modePrincipal = "combat"
  472. def majModeAffichage(self, index):
  473. """met a jour le mode d'affichage"""
  474. nouveauMode = ""
  475. if index == 0:
  476. #passe a l'affichage standard
  477. pass
  478. elif index == 1:
  479. #passe en mode affichage de l'altitude
  480. nouveauMode = "altitude"
  481. elif index == 2:
  482. #affichage des terrains slt
  483. nouveauMode = "terrain"
  484. elif index == 3:
  485. #affichage tactique
  486. nouveauMode = "tactique"
  487. for coord in self.cases:
  488. self.cases[coord].majAffichageSpecial(nouveauMode)
  489. def modePeintureCase(self):
  490. """enclenche le mode peinture de case a partir de la couleur selectionnee"""
  491. self.majMode("standard")
  492. couleur = self.chercherCouleur()
  493. if couleur.isValid():
  494. self.majMode("caseMajTerrain", couleur)
  495. else:
  496. self.majMode("standard")
  497. def modePeintureCase_perso(self):
  498. origine = self.sender().objectName()
  499. index = int(origine.replace("terrainCouleurPerso",""))-1
  500. couleur = QColor(QColorDialog(self.fenetre.ui.vuePlateau).customColor(index))
  501. if couleur.isValid():
  502. self.majMode("caseMajTerrain", couleur)
  503. else:
  504. self.majMode("standard")
  505. def modeCopieTerrain(self):
  506. """enclenche le mode copie de case"""
  507. self.majMode("caseCopie")
  508. def modeCaseEffet(self):
  509. """enclenche le mode de mise a jour de l'ffet actif des cases"""
  510. self.majMode("standard")
  511. origine = self.sender().objectName()
  512. if origine == "effFeu":
  513. effet = "brule"
  514. elif origine == "effEau":
  515. effet = "eau"
  516. elif origine == "effGlace":
  517. effet = "glace"
  518. elif origine == "effPoison":
  519. effet = "poison"
  520. elif origine == "effEffacer":
  521. effet = "aucun"
  522. self.majMode("caseMajEffet", effet)
  523. self.fenetre.ui.vuePlateau.setFocus()
  524. def modeCreationPion(self):
  525. """enclenche le mode de creation de pions simples"""
  526. self.majMode("pionCreation")
  527. def majModePionSupprimer(self):
  528. """enclenche le mode suppression de pions sur clic gauche (creatures ou decors)"""
  529. self.majMode("pionSupprimer")
  530. def modeCreationDecor(self, ligne, col):
  531. """enclenche le mode de creation de decors depuis la liste des decors"""
  532. index = self.fenetre.ui.listDecors.item(ligne, 0)
  533. decor = charger("lib\\biblio\\decor", str(index.text().toUtf8()))
  534. self.majMode("pionDecorCreation", decor)
  535. self.fenetre.ui.vuePlateau.setFocus()
  536. def modeCreationCreaturePion(self, ligne, col):
  537. """enclenche le mode de creation de pions depuis la liste des creatures"""
  538. index = self.fenetre.ui.listCreatures.item(ligne, 0)
  539. creature = charger("lib\\biblio\\creature", str(index.text().toUtf8()))
  540. self.majMode("pionCreation", creature)
  541. self.fenetre.ui.vuePlateau.setFocus()
  542. def modeMajTerrainCase(self, ligne, col):
  543. """enclenche le mode permettant la mise a jour du terrain des cases"""
  544. index = self.fenetre.ui.listTerrains.item(ligne, 0)
  545. terrain = charger("lib\\biblio\\terrain", str(index.text().toUtf8()))
  546. self.majMode("caseMajTerrain", terrain)
  547. self.fenetre.ui.vuePlateau.setFocus()
  548. def majCouleurPion(self):
  549. """selectionne une nouvelle couleur"""
  550. self.majMode("standard")
  551. couleur = self.chercherCouleur()
  552. if couleur.isValid():
  553. self.modeParam["couleurPion"] = couleur
  554. self.fenetre.ui.pionCouleur.setStyleSheet("QPushButton {backGround:%s}" %(self.modeParam["couleurPion"].name()))
  555. def majEpaisseurPinceau(self, epaisseur):
  556. """met a jour l'epaisseur du pinceau (en cases)"""
  557. self.fenetre.ui.valeurEpaisseurPinceau.setText(QString.fromUtf8(str(epaisseur)))
  558. self.epaisseurPinceau = int(epaisseur)
  559. def modeMajAltitudeCase(self):
  560. self.majMode("caseMajAltitude")
  561. def majModeForme(self):
  562. """met a jour la forme utilisee pour la peinture"""
  563. bouton = self.sender().objectName()
  564. if bouton == "formeSimple":
  565. self.modeParam["typeFormeDessin"] = "simple"
  566. elif bouton == "formeLigne":
  567. self.modeParam["typeFormeDessin"] = "ligne"
  568. elif bouton == "formeLigneOrientee":
  569. self.modeParam["typeFormeDessin"] = "frontiere"
  570. elif bouton == "formeEllipseVide":
  571. self.modeParam["typeFormeDessin"] = "ellipseV"
  572. elif bouton == "formeEllipsePlein":
  573. self.modeParam["typeFormeDessin"] = "ellipseP"
  574. elif bouton == "formeRectVide":
  575. self.modeParam["typeFormeDessin"] = "rectV"
  576. elif bouton == "formeRectPlein":
  577. self.modeParam["typeFormeDessin"] = "rectP"
  578. if self.modeActif[0:7] != "caseMaj":
  579. self.majMode("caseMajTerrain", self.modeParam["terrain"])
  580. def majAffichageMenuCache(self):
  581. """un bouton 'cache' a ete clique, on met a jour les infos"""
  582. origine = self.sender().objectName()
  583. index = int(origine.replace("plateauCache",""))
  584. self.cacheEnCours = index
  585. for i in range(1,5):
  586. bouton = self.fenetre.ui.outilsEditionPlateau.findChild(QToolButton, "plateauCache{}".format(i))
  587. bouton.setCheckable(True)
  588. bouton.setChecked((i == index))
  589. self.fenetre.ui.cacheActif.setChecked(self.caches[index]["actif"])
  590. self.voirCacheEnCours()
  591. def majModeCombatDeplacement(self):
  592. """active le mode de combat 'deplacement' (mode standard)"""
  593. self.majModeCombat("combatDeplacement")
  594. def majModeCombatAttaqueCaC(self):
  595. """active le mode de combat 'corps-a-corps'"""
  596. self.majModeCombat("combatAttaqueCaC")
  597. def majModeCombatAttaqueDist(self):
  598. """active le mode de combat 'attaque a distance'"""
  599. self.majModeCombat("combatAttaqueDist")
  600. def majModeCombatZone(self):
  601. if not len(self.modeParam["typeAttaqueZone"]) > 0: self.modeParam["typeAttaqueZone"] = "ligne"
  602. self.majModeCombat("combatAttaqueZone")
  603. def majModeCombatZoneForme(self):
  604. if self.modeCombat == "combatAttaqueZone":
  605. origine = self.sender().objectName()
  606. self.modeParam["typeAttaqueZone"] = str(origine).split("_")[1]
  607. self.majModeCombat("combatAttaqueZone")
  608. def majModeDefinirEntree(self):
  609. self.modeParam["entreeSortie"] = EntreeSortie(self, "E")
  610. self.modeParam["entreeSortie"].creerForme()
  611. self.majMode("placementEntreeSortie")
  612. def majModeDefinirSortie(self):
  613. self.modeParam["entreeSortie"] = EntreeSortie(self, "S")
  614. self.modeParam["entreeSortie"].creerForme()
  615. self.majMode("placementEntreeSortie")
  616. def majModeZonePlacement(self):
  617. self.majMode("majZonePlacement")
  618. def majMode(self, mode = "standard", param = None):
  619. """modifie ou reinitialise le type d'interaction avec le plateau"""
  620. reinit = True #doit-on reinitialiser a la fin de l'operation
  621. ##on desactive le mode precedent
  622. if self.pionSelectionne() != None:
  623. self.pionSelectionne().afficheOmbreSelection(False)
  624. if self.modeActif == "pionSelectionne":
  625. self.majModeCombat("")
  626. self.modeParam["numPionSelectionne"] = 0
  627. elif self.pionDecorSelectionne() != None:
  628. self.pionDecorSelectionne().afficheOmbreSelection(False)
  629. self.modeParam["numPionSelectionne"] = 0
  630. if self.modeActif == "pionSelectionne" or self.modeActif == "pionCreation" or self.modeActif == "pionDecorSelectionne" or self.modeActif == "pionDecorCreation":
  631. self.majProjectionPosition(False)
  632. self.modeParam["formeProjectionPosition"] = None
  633. self.modeParam["coordProjectionPosition"] = None
  634. self.modeParam["nbRotations"] = 0
  635. if self.modeActif[0:7] == "caseMaj" or self.modeActif == "cachePlacer" or self.modeActif == "majZonePlacement":
  636. if self.modeParam["formeDessin"] != None:
  637. self.modeParam["formeDessin"].prepareGeometryChange()
  638. self.removeItem(self.modeParam["formeDessin"])
  639. self.modeParam["formeDessin"] = None
  640. self.modeParam["origineFormeDessin"] = None
  641. self.modeParam["point2FormeDessin"] = None
  642. self.afficherListeCases(self.modeParam["listeCasesFormeDessin"], False)
  643. self.modeParam["listeCasesFormeDessin"] = []
  644. if self.modeActif == "placementEntreeSortie":
  645. if self.modeParam["entreeSortie"] != None:
  646. self.modeParam["entreeSortie"].prepareGeometryChange()
  647. self.removeItem(self.modeParam["entreeSortie"])
  648. self.modeParam["entreeSortie"] = None
  649. ### definition du nouveau mode
  650. if mode in self.listMode and mode != "standard":
  651. self.modeActif = mode
  652. reinit = False
  653. if mode == "caseMajTerrain":
  654. #curseur pinceau, on ajoute des terrains au cases
  655. if param != None:
  656. self.curseurPinceau()
  657. self.fenetre.ui.vuePlateau.setDragMode(0)
  658. self.modeParam["terrain"] = param
  659. elif mode == "caseCopie":
  660. #curseur seringue, on 'preleve' le terrain sur la case cliquee, avant de passer en mode majTerrain
  661. self.curseurSeringue()
  662. self.fenetre.ui.vuePlateau.setDragMode(0)
  663. elif mode == "caseMajEffet":
  664. #on met a jour l'effet actif sur les cases
  665. if param != None:
  666. self.curseurPinceau()
  667. self.fenetre.ui.vuePlateau.setDragMode(0)
  668. self.modeParam["effet"] = param
  669. elif mode == "caseMajAltitude":
  670. self.curseurPinceau()
  671. self.fenetre.ui.vuePlateau.setDragMode(0)
  672. elif mode == "cachePlacer":
  673. self.curseurPinceau()
  674. self.fenetre.ui.vuePlateau.setDragMode(0)
  675. self.modeParam["typeFormeDessin"] = "rectP"
  676. elif mode == "majZonePlacement":
  677. self.curseurPinceau()
  678. self.fenetre.ui.vuePlateau.setDragMode(0)
  679. self.modeParam["typeFormeDessin"] = "rectP"
  680. elif mode == "placementEntreeSortie":
  681. self.fenetre.ui.vuePlateau.setDragMode(0)
  682. elif mode == "pionDecorCreation":
  683. #curseur 'plus', on cree de nouveaux decors
  684. self.modeParam["decor"] = None
  685. self.fenetre.ui.vuePlateau.setDragMode(0)
  686. self.modeParam["formeProjectionPosition"] = Forme(self.formeCases)
  687. self.modeParam["nbRotations"] = 0
  688. if param != None:
  689. if param.__class__.__name__ == "Decor":
  690. self.modeParam["decor"] = param
  691. self.modeParam["formeProjectionPosition"].definirForme(self.modeParam["decor"].formeDef[self.formeCases])
  692. elif mode == "pionDecorSelectionne":
  693. #un pion decor est selectionne
  694. if param != None:
  695. self.modeParam["numPionSelectionne"] = param
  696. self.pionDecorSelectionne().afficheOmbreSelection(True)
  697. self.fenetre.ui.vuePlateau.setCursor(QCursor(Qt.ArrowCursor))
  698. self.modeParam["formeProjectionPosition"] = Forme(self.formeCases)
  699. self.modeParam["formeProjectionPosition"].definirForme(self.pionDecorSelectionne().forme.formeDef)
  700. self.modeParam["nbRotations"] = self.pionDecorSelectionne().nbRotations
  701. elif mode == "pionCreation":
  702. #curseur 'plus', on cree de nouveaux pions
  703. self.modeParam["creature"] = None
  704. self.fenetre.ui.vuePlateau.setDragMode(0)
  705. self.modeParam["formeProjectionPosition"] = Forme(self.formeCases)
  706. self.modeParam["nbRotations"] = 0
  707. if param != None:
  708. if param.__class__.__name__ == "Creature":
  709. self.modeParam["creature"] = param
  710. self.modeParam["formeProjectionPosition"].definirForme(self.modeParam["creature"].formeDef[self.formeCases])
  711. elif mode == "pionSupprimer":
  712. #mode suppression de pions/pions decors
  713. self.curseurGomme()
  714. self.fenetre.ui.vuePlateau.setDragMode(0)
  715. elif mode == "pionSelectionne":
  716. #un pion est selectionne, on affiche les deplacements possibles
  717. if param != None:
  718. self.modeParam["numPionSelectionne"] = param
  719. self.pionSelectionne().afficheOmbreSelection(True)
  720. self.numPionEnCours = param
  721. for i in range(0, self.fenetre.ui.infoOrdreJeu.rowCount()):
  722. if str(self.fenetre.ui.infoOrdreJeu.item(i, 0).text().toUtf8()) == str(param):
  723. self.fenetre.ui.infoOrdreJeu.setCurrentCell(i,0)
  724. self.majAffichagePionSelectionne()
  725. self.majListeAttributs()
  726. self.afficherNotesPion()
  727. self.majListeAttaques()
  728. self.majModeCombat("aucun")
  729. elif not mode in self.listMode:
  730. print("mode non reconnu")
  731. ##on reinitialise si necessaire
  732. if reinit:
  733. #mode standard : pas d'interaction avec les cases, on deplace le plateau en le saisissant ou les pions en cliquant dessus
  734. self.modeActif = "standard"
  735. self.majAffichagePionSelectionne()
  736. self.majInfosPion()
  737. self.majInfosDecor()
  738. self.majListeAttributs()
  739. self.majListeAttaques()
  740. self.afficherNotesPion()
  741. self.fenetre.majVisibiliteBarreCombat("menu")
  742. QApplication.processEvents()
  743. self.fenetre.ui.vuePlateau.setCursor(QCursor(Qt.ArrowCursor))
  744. self.fenetre.ui.vuePlateau.setDragMode(1)
  745. else:
  746. self.fenetre.ui.vuePlateau.setFocus()
  747. def majModeCombat(self, mode = ""):
  748. """met a jour le mode de combat actif pour le pion selectionne"""
  749. ## on desactive le mode precedent si besoin
  750. #la projection de position
  751. if self.modeCombat == "combatDeplacement":
  752. #le champ de deplacement
  753. if self.modePrincipal == "combat":
  754. self.afficherChampDeplacement(False)
  755. self.majProjectionPosition(False)
  756. self.modeParam["formeProjectionPosition"] = None
  757. self.modeParam["coordProjectionPosition"] = None
  758. self.modeParam["nbRotations"] = 0
  759. elif self.modeCombat == "combatAttaqueCaC":
  760. #la zone d'attaque au cac
  761. self.materialiserPions(True)
  762. self.majZoneAttaqueCaC(False)
  763. self.modeParam["zoneAttaqueCaC"] = []
  764. if self.modeParam["cibleAttaqueCaC"] != None:
  765. self.modeParam["cibleAttaqueCaC"].estCibleAttaque(False)
  766. self.modeParam["cibleAttaqueCaC"] = None
  767. elif self.modeCombat == "combatAttaqueDist":
  768. self.materialiserPions(True)
  769. self.majLigneMireAttaqueDist()
  770. elif self.modeCombat == "combatAttaqueZone":
  771. #l'attaque de zone
  772. self.materialiserPions(True)
  773. QApplication.processEvents()
  774. self.reinitAttaqueZone()
  775. if mode != "combatAttaqueZone": self.fenetre.majVisibiliteBarreCombat("menu")
  776. self.modeParam["listeCasesAttaqueZone"] = []
  777. ## maj de l'aspect des boutons selon le mode demande
  778. for item in self.fenetre.ui.barreBasCombat.children():
  779. if item.__class__.__name__ == "QToolButton":
  780. if item.isCheckable():
  781. item.setChecked((item.objectName() == mode and self.pionSelectionne() != None))
  782. if mode == "combatAttaqueZone":
  783. item.setChecked((item.objectName() == "combatZone_{}".format(self.modeParam["typeAttaqueZone"])\
  784. and self.pionSelectionne() != None))
  785. ## definition du nouveau mode de combat
  786. if self.pionSelectionne() != None:
  787. self.modeCombat = mode
  788. self.fenetre.ui.vuePlateau.setDragMode(0)
  789. self.fenetre.ui.combatVol_altitude.setValue(self.pionSelectionne().z)
  790. if mode == "aucun":
  791. self.fenetre.ui.vuePlateau.setCursor(QCursor(Qt.ArrowCursor))
  792. if mode == "combatDeplacement":
  793. self.fenetre.ui.vuePlateau.setCursor(QCursor(Qt.ArrowCursor))
  794. self.modeParam["formeProjectionPosition"] = Forme(self.formeCases)
  795. self.modeParam["formeProjectionPosition"].definirForme(self.pionSelectionne().forme.formeDef)
  796. self.modeParam["nbRotations"] = self.pionSelectionne().nbRotations
  797. if self.modePrincipal == "combat":
  798. #si mode combat, affichage de la zone de deplacement
  799. self.afficherChampDeplacement(True)
  800. elif mode == "combatAttaqueCaC":
  801. self.curseurEpee()
  802. self.materialiserPions(False)
  803. self.modeParam["zoneAttaqueCaC"] = self.zone(self.pionSelectionne().position, self.pionSelectionne().creature.allonge, True, True)
  804. self.majZoneAttaqueCaC(True)
  805. elif mode == "combatAttaqueDist":
  806. self.curseurArc()
  807. self.materialiserPions(False)
  808. elif mode == "combatAttaqueZone":
  809. self.curseurBaguette()
  810. self.materialiserPions(False)
  811. self.majFormeAttaqueZone()
  812. else:
  813. self.fenetre.ui.combatVol_altitude.setValue(0)
  814. ###############
  815. ########## Gestion du combat ##############
  816. def majAffichageTour(self):
  817. """met a jour l'affichage du tour en cours"""
  818. self.fenetre.ui.combatTour.setText(QString.fromUtf8("Tour: {}".format(self.tour)))
  819. def majAffichagePionSelectionne(self):
  820. """affiche le nom et le logo du pion actuellement selectionne"""
  821. if self.pionSelectionne() != None:
  822. self.fenetre.ui.combatPionSelectionne.setText(QString.fromUtf8(self.pionSelectionne().txtId()))
  823. if len(self.pionSelectionne().creature.img["nom"]) > 0:
  824. pix = QPixmap(QString.fromUtf8("img\\"+self.pionSelectionne().creature.img["nom"]))
  825. pix = pix.scaled(44, 44, Qt.KeepAspectRatio, Qt.SmoothTransformation)
  826. self.fenetre.ui.combatPionSelectionne_img.setPixmap(pix)
  827. else:
  828. self.fenetre.ui.combatPionSelectionne_img.setPixmap(QPixmap())
  829. else:
  830. self.fenetre.ui.combatPionSelectionne_img.setPixmap(QPixmap())
  831. self.fenetre.ui.combatPionSelectionne.setText(QString.fromUtf8("Pas de pion selectionné"))
  832. def initListeOrdreJeu(self):
  833. """cree les colonnes et met en forme la table ordre jeu"""
  834. ## self.fenetre.ui.infoOrdreJeu.setColumnWidth(0, 20)
  835. self.fenetre.ui.infoOrdreJeu.setColumnWidth(2, 30)
  836. self.fenetre.ui.infoOrdreJeu.hideColumn(0)
  837. ## self.fenetre.ui.infoOrdreJeu.hideColumn(2)
  838. self.fenetre.ui.infoOrdreJeu.setIconSize(QSize(30,20))
  839. def majListeOrdreJeu(self):
  840. """met a jour la liste des pions infoOrdreJeu"""
  841. while self.fenetre.ui.infoOrdreJeu.rowCount() > 0:
  842. self.fenetre.ui.infoOrdreJeu.removeRow(0)
  843. index = 0
  844. for num in self.ordreJeu:
  845. self.fenetre.ui.infoOrdreJeu.insertRow(int(index))
  846. self.fenetre.ui.infoOrdreJeu.setItem(int(index), 0, QTableWidgetItem(QString.fromUtf8(str(num))))
  847. icon = QIcon("img\\"+self.pions[num].creature.logo)
  848. item = QTableWidgetItem(icon,QString.fromUtf8(self.pions[num].txtId()))
  849. self.fenetre.ui.infoOrdreJeu.setItem(int(index), 1, item)
  850. self.fenetre.ui.infoOrdreJeu.setItem(int(index), 2, QTableWidgetItem(QString.fromUtf8(str(self.ordreJeu[num]))))
  851. index += 1
  852. self.fenetre.ui.infoOrdreJeu.sizeHintForColumn(1)
  853. ## trierTable(self.fenetre.ui.infoOrdreJeu, 2, 0)
  854. def clicListOrdreJeu(self, ligne, col):
  855. """on a clique dans la liste d'ordre de jeu, le pion correspondant est selectionne et centre sur la carte"""
  856. numPion = int(self.fenetre.ui.infoOrdreJeu.item(ligne, 0).text().toUtf8())
  857. self.fenetre.ui.vuePlateau.centerOn(self.cases[self.pions[numPion].position].centreGraphique)
  858. self.pionSaisir(numPion)
  859. def pionSuivant(self):
  860. """selection du pion suivant dans la liste d'ordre de jeu"""
  861. if self.numPionEnCours in self.pions:
  862. suivant = self.ordreJeu[self.numPionEnCours] + 1
  863. else:
  864. suivant = 1
  865. if suivant > len(self.ordreJeu):
  866. self.tour += 1
  867. self.majAffichageTour()
  868. suivant = 1
  869. for num in self.ordreJeu:
  870. if self.ordreJeu[num] == suivant:
  871. numPion = num
  872. break
  873. for ligne in range(0, self.fenetre.ui.infoOrdreJeu.rowCount()):
  874. item = self.fenetre.ui.infoOrdreJeu.item(ligne, 0)
  875. item.setSelected(int(self.fenetre.ui.infoOrdreJeu.item(ligne, 0).text().toUtf8()) == numPion)
  876. if int(item.text().toUtf8()) == numPion:
  877. self.fenetre.ui.infoOrdreJeu.scrollToItem(item)
  878. self.fenetre.ui.vuePlateau.centerOn(self.cases[self.pions[numPion].position].centreGraphique)
  879. self.pionSaisir(numPion)
  880. def afficheEcranGestionCombat(self):
  881. #*egc
  882. """affiche l'ecran de gestion du combat"""
  883. self.gestionCombat = EcranGestionCombat(self)
  884. self.gestionCombat.show()
  885. self.connect(self.fenetre, SIGNAL("majListesPions"), self.majListesPions)
  886. self.connect(self.gestionCombat, SIGNAL("majListesPions"), self.majListesPions)
  887. QApplication.processEvents()
  888. ## r = self.gestionCombat.exec_()
  889. def majOrdreJeu(self):
  890. """met a jour l'ordre de jeu des pions en fonction de l'attribut prevu par les regles s'il existe,
  891. ou en fonction de l'ordre de jeu parametre sinon"""
  892. attribut = regles.attributOrdreJeu()
  893. if attribut != None:
  894. dico = {}
  895. for numPion in self.pions:
  896. dico[numPion] = int(self.pions[numPion].lstCarac[attribut])
  897. ordre = sorted(dico, key = dico.get, reverse=(regles.sensTriOrdreJeu() == 1))
  898. self.ordreJeu = {}
  899. for numPion in self.pions:
  900. self.ordreJeu[numPion] = ordre.index(numPion) + 1
  901. self.majListeOrdreJeu()
  902. def pionDeplacerDansOrdreJeu(self, numPion, nouvellePosition):
  903. """deplace un pion dans le dictionnaire gerant l'ordre de jeu de maniere a assurer sa coherence
  904. nouvellePosition = 0 supprime le pion de la liste"""
  905. if numPion in self.ordreJeu:
  906. if nouvellePosition == 0:
  907. del self.ordreJeu[numPion]
  908. if len(self.ordreJeu) > 0:
  909. i = 0
  910. tmp = sorted(self.ordreJeu, key=self.ordreJeu.get)
  911. if numPion in tmp:
  912. tmp.remove(numPion)
  913. for num in tmp:
  914. i += 1
  915. if i == nouvellePosition:
  916. self.ordreJeu[numPion] = i
  917. i += 1
  918. self.ordreJeu[num] = i
  919. if i < nouvellePosition:
  920. self.ordreJeu[numPion] = i + 1
  921. elif nouvellePosition > 0:
  922. self.ordreJeu[numPion] = 1
  923. self.majOrdreJeu()
  924. def majListeAttributs(self):
  925. """met a jour la liste des attributs dans le panneau de combat"""
  926. self.fenetre.ui.listeAttributs.setColumnWidth(0, (0.4*self.fenetre.ui.listeAttributs.width()))
  927. self.fenetre.ui.listeAttributs.setColumnWidth(1, (0.4*self.fenetre.ui.listeAttributs.width()))
  928. self.disconnect(self.fenetre.ui.listeAttributs, SIGNAL("cellChanged(int,int)"), self.listeAttributCelluleModifiee)
  929. #on vide la liste
  930. while self.fenetre.ui.listeAttributs.rowCount() > 0:
  931. self.fenetre.ui.listeAttributs.removeRow(0)
  932. self.fenetre.ui.listeAttributs.setVisible((self.pionSelectionne() != None))
  933. if self.pionSelectionne() != None:
  934. #creation des lignes de base
  935. lignesBase = ["Nom","Etat","Alt."] #attention: modifier aussi dans listeAttributCelluleModifiee
  936. for i in range(0, 10): #10 premieres colonnes reservees pour les infos de base
  937. self.fenetre.ui.listeAttributs.insertRow(i)
  938. item = QTableWidgetItem()
  939. if i < len(lignesBase):
  940. item.setText(QString.fromUtf8(lignesBase[i]))
  941. item.setFlags(Qt.NoItemFlags)
  942. self.fenetre.ui.listeAttributs.setItem(i, 0, item)
  943. self.fenetre.ui.listeAttributs.setRowHidden(i, (i >= len(lignesBase)))
  944. #maj des donnees de base
  945. self.fenetre.ui.listeAttributs.setItem(lignesBase.index("Nom"), 1, QTableWidgetItem(QString.fromUtf8(str(self.pionSelectionne().nom))))
  946. self.fenetre.ui.listeAttributs.setItem(lignesBase.index("Etat"), 1, QTableWidgetItem(QString.fromUtf8(str(self.pionSelectionne().etat))))
  947. self.fenetre.ui.listeAttributs.setItem(lignesBase.index("Alt."), 1, QTableWidgetItem(QString.fromUtf8(str(self.pionSelectionne().z))))
  948. #attributs issus des regles utilisees
  949. ordre = regles.ordreAttributs()
  950. for elt in ordre:
  951. ligne = 10 + ordre.index(elt)
  952. self.fenetre.ui.listeAttributs.insertRow(ligne)
  953. item = QTableWidgetItem(QString.fromUtf8(elt))
  954. item.setFlags(Qt.NoItemFlags)
  955. self.fenetre.ui.listeAttributs.setItem(ligne, 0, item)
  956. self.fenetre.ui.listeAttributs.setItem(ligne, 1, QTableWidgetItem(QString.fromUtf8(str(self.pionSelectionne().lstCarac[elt]))))
  957. self.connect(self.fenetre.ui.listeAttributs, SIGNAL("cellChanged(int,int)"), self.listeAttributCelluleModifiee)
  958. def listeAttributCelluleModifiee(self, ligne, colonne):
  959. """une cellule de la liste des attributs a ete modifiee"""
  960. if colonne != 1:
  961. print("valeur non modifiable")
  962. else:
  963. #on verifie la validite de la donnee entree
  964. lignesBase = ["Nom","Etat","Alt."] #attention: modifier aussi dans majListeAttribut
  965. if ligne < len(lignesBase):
  966. pass
  967. elif ligne >= 10:
  968. attribut = regles.ordreAttributs()[(ligne - 10)]
  969. nouvelleVal = str(self.fenetre.ui.listeAttributs.item(ligne, 1).text().toUtf8())
  970. valVerifiee = regles.listeControle()[attribut].controler(nouvelleVal)
  971. if valVerifiee != None:
  972. self.pionSelectionne().lstCarac[attribut] = valVerifiee
  973. if attribut == regles.attributOrdreJeu():
  974. print("maj ordre (a implementer)")
  975. else:
  976. self.fenetre.ui.listeAttributs.setItem(ligne, 1, QTableWidgetItem(QString.fromUtf8(str(self.pionSelectionne().lstCarac[attribut]))))
  977. def initListeAttaques(self):
  978. """met en forme et connecte la liste des attaques du pion"""
  979. self.fenetre.ui.listeAttaques.setColumnWidth(0, 0)
  980. self.fenetre.ui.listeAttaques.setColumnWidth(1, (0.3*self.fenetre.ui.listeAttaques.width()))
  981. self.fenetre.ui.listeAttaques.setColumnWidth(2, (0.7*self.fenetre.ui.listeAttaques.width()))
  982. self.connect(self.fenetre.ui.listeAttaques, SIGNAL("itemSelectionChanged()"), self.majInfosAttaqueEC)
  983. self.connect(self.fenetre.ui.listeAttaques, SIGNAL("cellClicked(int, int)"), self.listeAttaquesCelluleCliquee)
  984. self.connect(self.fenetre.ui.notesAttaqueEC, SIGNAL("textChanged()"), self.majNotesAttaqueEC)
  985. self.fenetre.ui.listeAttributsAttaqueEC.setColumnWidth(0, (0.49*self.fenetre.ui.listeAttributsAttaqueEC.width()))
  986. self.fenetre.ui.listeAttributsAttaqueEC.setColumnWidth(1, (0.5*self.fenetre.ui.listeAttributsAttaqueEC.width()))
  987. def majListeAttaques(self):
  988. #*eea
  989. """met a jour la liste des attaques du pion dans le panneau de combat"""
  990. #on vide la liste
  991. while self.fenetre.ui.listeAttaques.rowCount() > 0:
  992. self.fenetre.ui.listeAttaques.removeRow(0)
  993. self.fenetre.ui.listeAttaques.setVisible((self.pionSelectionne() != None))
  994. self.fenetre.ui.titreAttaques.setVisible((self.pionSelectionne() != None))
  995. self.fenetre.ui.editerAttaques.setVisible((self.pionSelectionne() != None))
  996. self.fenetre.ui.panneauAttaqueEC.setVisible((self.pionSelectionne() != None))
  997. i = 0
  998. if self.pionSelectionne() != None:
  999. for attaque in self.pionSelectionne().attaques:
  1000. self.fenetre.ui.listeAttaques.insertRow(i)
  1001. self.fenetre.ui.listeAttaques.setItem(i, 0, QTableWidgetItem(QString.fromUtf8(str(self.pionSelectionne().attaques.index(attaque)))))
  1002. icone = None
  1003. if attaque.typ == "cac":
  1004. icone = QIcon("img\\curseurEpee.png")
  1005. if attaque.typ == "dist":
  1006. icone = QIcon("img\\arc.png")
  1007. if attaque.typ == "zone":
  1008. icone = QIcon("img\\baguette.png")
  1009. if icone != None:
  1010. self.fenetre.ui.listeAttaques.setItem(i, 1, QTableWidgetItem(icone, QString.fromUtf8("")))
  1011. self.fenetre.ui.listeAttaques.setItem(i, 2, QTableWidgetItem(QString.fromUtf8(attaque.nom)))
  1012. self.majInfosAttaqueEC()
  1013. def listeAttaquesCelluleCliquee(self, ligne, colonne):
  1014. """on a clique sur une cellule de la liste des attaques"""
  1015. numAttaque = int(str(self.fenetre.ui.listeAttaques.item(ligne, 0).text().toUtf8()))
  1016. if numAttaque >= 0:
  1017. self.utiliserAttaque(numAttaque)
  1018. def utiliserAttaque(self, numAttaque):
  1019. """le pion selectionne utilise son attaque n"""
  1020. if self.pionSelectionne() != None:
  1021. if numAttaque < len(self.pionSelectionne().attaques):
  1022. attaque = self.pionSelectionne().attaques[numAttaque]
  1023. if attaque.typ == "cac":
  1024. self.majModeCombat("combatAttaqueCaC")
  1025. if attaque.typ == "dist":
  1026. self.majModeCombat("combatAttaqueDist")
  1027. if attaque.typ == "zone":
  1028. self.modeParam["typeAttaqueZone"] = attaque.formeZone
  1029. self.fenetre.ui.combatZone_disqueRayon.setValue(attaque.rayon)
  1030. self.majModeCombat("combatAttaqueZone")
  1031. def majInfosAttaqueEC(self):
  1032. #*aec
  1033. """met a jour les infos de l'attaque en cours (selectionnee)"""
  1034. selection = self.fenetre.ui.listeAttaques.selectedItems()
  1035. self.fenetre.ui.panneauAttaqueEC.setVisible(self.pionSelectionne() != None and len(selection) > 0)
  1036. if self.pionSelectionne() != None and len(selection) > 0:
  1037. ligne = selection[0].row()
  1038. numAttaque = int(str(self.fenetre.ui.listeAttaques.item(ligne, 0).text().toUtf8()))
  1039. self.disconnect(self.fenetre.ui.listeAttributsAttaqueEC, SIGNAL("cellChanged(int,int)"), self.attaqueECCelluleModifiee)
  1040. #on vide la liste
  1041. while self.fenetre.ui.listeAttributsAttaqueEC.rowCount() > 0:
  1042. self.fenetre.ui.listeAttributsAttaqueEC.removeRow(0)
  1043. self.fenetre.ui.listeAttributsAttaqueEC.insertRow(0)
  1044. self.fenetre.ui.listeAttributsAttaqueEC.setItem(0, 0, QTableWidgetItem(QString.fromUtf8("numAtt")))
  1045. self.fenetre.ui.listeAttributsAttaqueEC.setItem(0, 1, QTableWidgetItem(QString.fromUtf8(str(numAttaque))))
  1046. self.fenetre.ui.listeAttributsAttaqueEC.setRowHidden(0, True)
  1047. #attributs issus des regles utilisees
  1048. ordre = regles.ordreAttributsAttaques()
  1049. for elt in ordre:
  1050. ligne = 1 + ordre.index(elt)
  1051. self.fenetre.ui.listeAttributsAttaqueEC.insertRow(ligne)
  1052. item = QTableWidgetItem(QString.fromUtf8(elt))
  1053. item.setFlags(Qt.NoItemFlags)
  1054. self.fenetre.ui.listeAttributsAttaqueEC.setItem(ligne, 0, item)
  1055. self.fenetre.ui.listeAttributsAttaqueEC.setItem(ligne, 1, QTableWidgetItem(QString.fromUtf8(str(self.pionSelectionne().attaques[numAttaque].attributs[elt]))))
  1056. self.connect(self.fenetre.ui.listeAttributsAttaqueEC, SIGNAL("cellChanged(int,int)"), self.attaqueECCelluleModifiee)
  1057. #maj des notes
  1058. self.disconnect(self.fenetre.ui.notesAttaqueEC, SIGNAL("textChanged()"), self.majNotesAttaqueEC)
  1059. self.fenetre.ui.notesAttaqueEC.setText(QString.fromUtf8(self.pionSelectionne().attaques[numAttaque].notes))
  1060. self.connect(self.fenetre.ui.notesAttaqueEC, SIGNAL("textChanged()"), self.majNotesAttaqueEC)
  1061. def attaqueECCelluleModifiee(self, ligne, colonne):
  1062. """une cellule de la liste d'attributs de l'attaque en cours a ete modifiee"""
  1063. pass
  1064. def majNotesAttaqueEC(self):
  1065. """met a jour les notes de l'attaque en cours (selectionnee)"""
  1066. selection = self.fenetre.ui.listeAttaques.selectedItems()
  1067. self.fenetre.ui.panneauAttaqueEC.setVisible(self.pionSelectionne() != None and len(selection) > 0)
  1068. if self.pionSelectionne() != None and len(selection) > 0:
  1069. ligne = selection[0].row()
  1070. numAttaque = int(str(self.fenetre.ui.listeAttaques.item(ligne, 0).text().toUtf8()))
  1071. self.pionSelectionne().attaques[numAttaque].notes = str(self.fenetre.ui.notesAttaqueEC.toPlainText().toUtf8())
  1072. def afficheEcranEditionAttaques(self):
  1073. """affiche l'ecran d'edition/creation d'attaques"""
  1074. if self.pionSelectionne() != None:
  1075. self.editionAttaques = QDialog()
  1076. frame = EcranEditionAttaques(self.pionSelectionne())
  1077. frame.setParent(self.editionAttaques)
  1078. self.editionAttaques.setAttribute(Qt.WA_DeleteOnClose)
  1079. r = self.editionAttaques.exec_()
  1080. del self.editionAttaques
  1081. self.majListeAttaques()
  1082. def majNotesPion(self):
  1083. """les notes du pion ont ete mises a jour"""
  1084. if self.pionSelectionne() != None:
  1085. self.pionSelectionne().notes = str(self.fenetre.ui.notesPion.toPlainText().toUtf8())
  1086. else:
  1087. pass
  1088. def afficherNotesPion(self):
  1089. """affiche les notes du pion selectionne dans le QTextEdit dedie"""
  1090. self.fenetre.ui.notesPion.setVisible((self.pionSelectionne() != None))
  1091. if self.pionSelectionne() != None:
  1092. self.fenetre.ui.notesPion.setText(QString.fromUtf8(self.pionSelectionne().notes))
  1093. ###############
  1094. ############### fonctions de calcul ################
  1095. def zone(self, origine, distance, z=0, conditionFranchissable = False, conditionVisible = False):
  1096. """renvoie un dictionnaire representant la liste des coordonnees des cases comprises dans la zone
  1097. la zone en question est la liste des cases situees a une distance d des coordonnees d'origine
  1098. z = 0 -> hauteur z de l'origine par rapport a l'altitude de la case
  1099. conditionFranchissable = Vrai -> les cases infranchissables ne sont pas prises en compte
  1100. conditionVisible = Vrai -> les cases bloquant la visibilite ne sont pas prises en compte
  1101. [cf methode A* (A-star)]"""
  1102. aVerifier = []
  1103. aVerifier2 = []
  1104. resultat = {}
  1105. k = 0
  1106. #on part de la premiere case, puis on itere a partir de chaque nouveau depart
  1107. if origine in self.cases:
  1108. aVerifier.append(origine)
  1109. while k <= distance:
  1110. for depart in aVerifier:
  1111. for coord in self.cases[depart].voisins:
  1112. if not coord in aVerifier and not coord in aVerifier2 and not coord in resultat:
  1113. if conditionFranchissable and not conditionVisible:
  1114. if self.cases[coord].estFranchissable(z):
  1115. aVerifier2.append(coord)
  1116. elif not conditionFranchissable and conditionVisible:
  1117. if self.cases[coord].terrain.visibilite:
  1118. aVerifier2.append(coord)
  1119. elif conditionFranchissable and conditionVisible:
  1120. if self.cases[coord].estFranchissable(z) and self.cases[coord].terrain.visibilite:
  1121. aVerifier2.append(coord)
  1122. else:
  1123. aVerifier2.append(coord)
  1124. for elt in aVerifier:
  1125. resultat[elt] = k
  1126. aVerifier = aVerifier2
  1127. aVerifier2 = []
  1128. k += 1
  1129. return resultat
  1130. def coordonneesValides(self, coord):
  1131. """les coordonnees entrees en parametre sont elles celles d'une case du plateau"""
  1132. return (coord[0] >= 0 and coord[1] >= 0 and coord[0] < self.nbCasesX and coord[1] < self.nbCasesY)
  1133. def polygoneAgglo(self, listeCases):
  1134. """renvoie un polygone contruit par agglomeration des polygones des cases de la liste
  1135. les cases doivent etre adjacentes (cases hexagonales ou carrees)"""
  1136. pointsPolygone = []
  1137. segments = []
  1138. case = Case(self)
  1139. #on verifie que toutes les cases sont adjacentes les unes aux autres
  1140. valide = True
  1141. if len(listeCases) > 1:
  1142. for coord in listeCases:
  1143. if not len(set(listeCases).intersection(self.lstCoordAdjacentes(coord[0], coord[1]))) > 0: valide = False
  1144. if not len(listeCases) > 0:
  1145. valide = False
  1146. if valide:
  1147. #on parcourt les faces des polygones des cases, et on ne garde que ceux qui n'ont pas de case 'en face'
  1148. for coord in listeCases:
  1149. polygone = case.polygone(coord[0], coord[1])
  1150. voisins = self.lstCoordAdjacentes(coord[0], coord[1])
  1151. for i in range(0, len(voisins)):
  1152. if not voisins[i] in listeCases:
  1153. j = i+1
  1154. if j > len(voisins) - 1:
  1155. j = 0
  1156. segments.append(QLineF(polygone[i], polygone[j]))
  1157. #on 'accroche' les segments les uns aux autres, dans l'ordre
  1158. segments2 = [segments[0]]
  1159. for segment2 in segments2:
  1160. for segment in segments:
  1161. if not QLineF(segment.p1(), segment.p2()) in segments2 and not QLineF(segment.p2(), segment.p1()) in segments2:
  1162. if sqrt((segment.p1().x()-segment2.p2().x())**2+(segment.p1().y()-segment2.p2().y())**2) < 1:
  1163. segments2.append(QLineF(segment.p1(), segment.p2()))
  1164. elif sqrt((segment.p2().x()-segment2.p2().x())**2+(segment.p2().y()-segment2.p2().y())**2) < 1:
  1165. segments2.append(QLineF(segment.p2(), segment.p1()))
  1166. pointsPolygone = []
  1167. premierPoint = segments2[0].p1()
  1168. pointsPolygone.append(premierPoint)
  1169. for segment in segments2:
  1170. pointSuivant = segment.p2()
  1171. if pointSuivant != premierPoint:
  1172. pointsPolygone.append(pointSuivant)
  1173. #creation du polygone
  1174. polygone = QPolygonF()
  1175. for point in pointsPolygone:
  1176. polygone.append(point)
  1177. else:
  1178. polygone = None
  1179. return polygone
  1180. def lstCoordAdjacentes(self, x, y):
  1181. """renvoie la liste des coordonnees adjacentes, sans condition d'existence sur le plateau
  1182. attention: l'ordre est important"""
  1183. if self.formeCases == "H":
  1184. voisins = [(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)]
  1185. else:
  1186. voisins = [(x, y-1), (x+1, y), (x, y+1), (x-1, y)]
  1187. return voisins
  1188. def coordCentreListeCases(self, listeCases):
  1189. """renvoie les coordonnees centrales d'une liste de cases"""
  1190. retour = None
  1191. if len(listeCases) > 0:
  1192. listeTriee = sorted(listeCases)
  1193. posMilieu = int(len(listeCases)/2)
  1194. retour = listeTriee[posMilieu]
  1195. return retour
  1196. def coordonneesAuPoint(self, point):
  1197. """renvoie les coordonnees de la case situee au QPointF entre en parametre"""
  1198. coord = None
  1199. if point != None:
  1200. lstObjets = self.fenetre.ui.vuePlateau.scene().items(point)
  1201. for objet in lstObjets:
  1202. if objet:
  1203. if objet.__class__.__name__ == "Case":
  1204. coord = (objet.x, objet.y)
  1205. break
  1206. return coord
  1207. def creerOrigineFormeDessin(self, coord):
  1208. """place le point d'origine de la forme de selection"""
  1209. self.modeParam["listeCasesFormeDessin"] = []
  1210. self.modeParam["origineFormeDessin"] = self.cases[coord].centreGraphique
  1211. self.modeParam["point2FormeDessin"] = self.cases[coord].centreGraphique
  1212. self.modeParam["listeCasesFormeDessin"] = [coord]
  1213. self.afficherListeCases(self.modeParam["listeCasesFormeDessin"], True)
  1214. def majFormeDessin(self):
  1215. """cree/maj la forme utilisee pour selectionner les cases et met a jour la liste des cases selectionnes"""
  1216. self.afficherListeCases(self.modeParam["listeCasesFormeDessin"], False)
  1217. pinceau = QPen()
  1218. pinceau.setColor(QColor("black"))
  1219. point2 = self.modeParam["point2FormeDessin"]
  1220. if self.modeParam["typeFormeDessin"] == "simple":
  1221. #pas de forme: on ajoute les cases survolees a la liste des cases
  1222. coordPoint2 = self.coordonneesAuPoint(point2)
  1223. zone = self.zone(coordPoint2, self.epaisseurPinceau-1)
  1224. for coord in zone:
  1225. if not coord in self.modeParam["listeCasesFormeDessin"]:
  1226. self.modeParam["listeCasesFormeDessin"].append(coord)
  1227. elif self.modeParam["typeFormeDessin"] == "frontiere":
  1228. #droite qui selectionne toutes les cases situees au dessus d'elle
  1229. if self.modeParam["formeDessin"] == None:
  1230. self.modeParam["formeDessin"] = QGraphicsLineItem()
  1231. self.modeParam["formeDessin"].setPen(pinceau)
  1232. self.addItem(self.modeParam["formeDessin"])
  1233. ligne = QLineF(self.modeParam["origineFormeDessin"], point2)
  1234. orientation = int((1+int(ligne.angle()/22.5))/2)
  1235. ligne.setAngle(orientation*45)
  1236. self.modeParam["formeDessin"].setLine(ligne)
  1237. self.modeParam["listeCasesFormeDessin"] = self.selectionFrontiere()
  1238. else:
  1239. if self.modeParam["typeFormeDessin"] == "ligne":
  1240. #segment simple
  1241. if self.modeParam["formeDessin"] == None:
  1242. self.modeParam["formeDessin"] = QGraphicsLineItem()
  1243. self.modeParam["formeDessin"].setPen(pinceau)
  1244. self.modeParam["formeDessin"].prepareGeometryChange()
  1245. self.addItem(self.modeParam["formeDessin"])
  1246. ligne = QLineF(self.modeParam["origineFormeDessin"], point2)
  1247. self.modeParam["formeDessin"].setLine(ligne)
  1248. elif self.modeParam["typeFormeDessin"] == "rectV" or self.modeParam["typeFormeDessin"] == "rectP":
  1249. #rectangle
  1250. if self.modeParam["formeDessin"] == None:
  1251. self.modeParam["formeDessin"] = QGraphicsRectItem()
  1252. self.modeParam["formeDessin"].prepareGeometryChange()
  1253. self.modeParam["formeDessin"].setPen(pinceau)
  1254. self.addItem(self.modeParam["formeDessin"])
  1255. rect = self.rectFormeDessin(self.modeParam["origineFormeDessin"], point2)
  1256. self.modeParam["formeDessin"].setRect(rect)
  1257. elif self.modeParam["typeFormeDessin"] == "ellipseV" or self.modeParam["typeFormeDessin"] == "ellipseP":
  1258. #ellipse
  1259. if self.modeParam["formeDessin"] == None:
  1260. self.modeParam["formeDessin"] = QGraphicsEllipseItem()
  1261. self.modeParam["formeDessin"].prepareGeometryChange()
  1262. self.modeParam["formeDessin"].setPen(pinceau)
  1263. self.addItem(self.modeParam["formeDessin"])
  1264. rect = self.rectFormeDessin(self.modeParam["origineFormeDessin"], point2)
  1265. self.modeParam["formeDessin"].setRect(rect)
  1266. #on liste les cases intersectant la forme
  1267. plein = (self.modeParam["typeFormeDessin"][len(self.modeParam["typeFormeDessin"])-1:] == "P")
  1268. self.modeParam["listeCasesFormeDessin"] = self.casesSousForme(self.modeParam["formeDessin"], plein, self.epaisseurPinceau - 1)
  1269. self.afficherListeCases(self.modeParam["listeCasesFormeDessin"], True)
  1270. def casesSousForme(self, forme, plein = True, epaisseur = 0):
  1271. """renvoie la liste des cases en collision avec un QGraphicsItem en parametre
  1272. plein = False: pas le contenu de la forme
  1273. epaisseur = renvoie aussi les cases voisines jusqu'a la distance demandee"""
  1274. tmp = []
  1275. listeCases = []
  1276. point1 = None
  1277. point2 = None
  1278. #point 1 et 2
  1279. if forme.__class__.__name__ == "QGraphicsLineItem":
  1280. point1 = forme.line().p1()
  1281. point2 = forme.line().p2()
  1282. elif forme.__class__.__name__ == "QGraphicsRectItem" or forme.__class__.__name__ == "QGraphicsEllipseItem":
  1283. point1 = forme.rect().topLeft()
  1284. point2 = forme.rect().bottomRight()
  1285. else:
  1286. point1 = forme.boundingRect().topLeft()
  1287. point2 = forme.boundingRect().bottomRight()
  1288. #preselection des cases (meilleures perf)
  1289. if point1 != None and point2 != None and point1 != point2:
  1290. preSelection = self.preSelectionCollision(point1, point2)
  1291. else:
  1292. preSelection = []
  1293. for coord in self.cases:
  1294. preSelection.append(coord)
  1295. #on liste les cases en collision avec la forme
  1296. for coord in preSelection:
  1297. if self.cases[coord].collidesWithItem(forme, Qt.IntersectsItemShape):
  1298. if plein:
  1299. tmp.append(coord)
  1300. else:
  1301. contenu = True
  1302. for i in range(0,len(self.cases[coord].polygon())):
  1303. if not forme.contains(self.cases[coord].polygon().at(i)):
  1304. contenu = False
  1305. break
  1306. if contenu == False:
  1307. tmp.append(coord)
  1308. #on applique l'epaisseur du pinceau (lignes ou formes vides seulement) si necessaire
  1309. if not plein and epaisseur > 0:
  1310. for coord in tmp:
  1311. zone = self.zone(coord, epaisseur)
  1312. for coord2 in zone:
  1313. if not coord2 in listeCases:
  1314. listeCases.append(coord2)
  1315. else:
  1316. listeCases = tmp
  1317. #si la liste est vide, on ajoute l'origine de la forme
  1318. if len(listeCases) == 0:
  1319. listeCases = [self.coordonneesAuPoint(point1)]
  1320. return listeCases
  1321. def preSelectionCollision(self, point1, point2):
  1322. """renvoie une liste des cases qui peuvent etre concernees par une collision avec
  1323. un graphicsItem (pour des raisons de performance)"""
  1324. preSelection = []
  1325. coord1 = self.coordonneesAuPoint(point1)
  1326. coord2 = self.coordonneesAuPoint(point2)
  1327. if coord1 != None and coord2 != None:
  1328. minX = min(coord1[0], coord2[0]) - 1
  1329. maxX = max(coord1[0], coord2[0]) + 1
  1330. minY = min(coord1[1], coord2[1]) - 1
  1331. maxY = max(coord1[1], coord2[1]) + 1
  1332. for coord in self.cases:
  1333. if coord[0] >= minX and coord[0] <= maxX and coord[1] >= minY and coord[1] <= maxY :
  1334. preSelection.append(coord)
  1335. else:
  1336. preSelection = self.cases
  1337. return preSelection
  1338. def selectionFrontiere(self):
  1339. """retourne les cases selectionnees lors de l'utilisation de la forme 'frontiere'"""
  1340. listeCases = []
  1341. ligne = self.modeParam["formeDessin"].line()
  1342. normale = ligne.normalVector()
  1343. normale = normale.unitVector()
  1344. coordOrigine = self.coordonneesAuPoint(self.modeParam["origineFormeDessin"])
  1345. dx = normale.p2().x() - normale.p1().x()
  1346. dy = normale.p2().y() - normale.p1().y()
  1347. for coord in self.cases:
  1348. if dx < 0 and dy == 0: #normale pointe vers la gauche
  1349. if (coord[0] - coordOrigine[0]) <= 0:
  1350. listeCases.append(coord)
  1351. if dx > 0 and dy == 0: #normale pointe vers la droite
  1352. if (coord[0] - coordOrigine[0]) >= 0:
  1353. listeCases.append(coord)
  1354. if dx == 0 and dy < 0: #pointe vers le haut (rappel: axe y vers le bas)
  1355. if (coord[1] - coordOrigine[1]) <= 0:
  1356. listeCases.append(coord)
  1357. elif dx == 0 and dy > 0: #pointe vers le bas
  1358. if (coord[1] - coordOrigine[1]) >= 0:
  1359. listeCases.append(coord)
  1360. if dx > 0 and dy < 0: #pointe vers le haut droite
  1361. if (coord[0] - coordOrigine[0]) + -1*(coord[1] - coordOrigine[1]) >= 0:
  1362. listeCases.append(coord)
  1363. elif dx > 0 and dy > 0: #pointe vers le bas droite
  1364. if -1*(coord[0] - coordOrigine[0]) + -1*(coord[1] - coordOrigine[1]) <= 0:
  1365. listeCases.append(coord)
  1366. if dx < 0 and dy < 0: #pointe vers le haut gauche
  1367. if (coord[0] - coordOrigine[0]) + (coord[1] - coordOrigine[1]) <= 0:
  1368. listeCases.append(coord)
  1369. elif dx < 0 and dy > 0: #pointe vers le bas gauche
  1370. if -1*(coord[0] - coordOrigine[0]) + (coord[1] - coordOrigine[1]) >= 0:
  1371. listeCases.append(coord)
  1372. return listeCases
  1373. def validerFormeDessin(self):
  1374. """la projection des cases a peindre a ete acceptee, on peint ces cases"""
  1375. if self.modeActif[0:7] == "caseMaj" or self.modeActif == "caseCopie" or self.modeActif == "cachePlacer" or self.modeActif == "majZonePlacement":
  1376. self.majListeCases(self.modeParam["listeCasesFormeDessin"])
  1377. self.majMode(self.modeActif)
  1378. ## self.majMode("standard")
  1379. def rectFormeDessin(self, point1, point2):
  1380. """retourne le rectangle repesentant la forme de selection
  1381. determine pour ce faire les points A et B, cad le point haut-droit et le point bas-gauche"""
  1382. pointA = QPointF(min(point1.x(), point2.x()), min(point1.y(), point2.y()))
  1383. pointB = QPointF(max(point1.x(), point2.x()), max(point1.y(), point2.y()))
  1384. rect = QRectF(pointA, pointB)
  1385. return rect
  1386. def majLigneMireAttaqueDist(self, coordCible = None):
  1387. """met a jour la ligne de mire representant l'attaque a distance"""
  1388. if self.modeParam["cibleAttaqueDist"] != None and self.modeParam["cibleAttaqueDist"] != coordCible:
  1389. self.cases[self.modeParam["cibleAttaqueDist"]].majEstCibleCurseur(False)
  1390. ## if self.modeParam["pionCibleAttaqueDist"] != None and self.modeParam["pionCibleAttaqueDist"] != self.cases[coordCible].pionOccupant():
  1391. if self.modeParam["pionCibleAttaqueDist"] != None:
  1392. self.modeParam["pionCibleAttaqueDist"].estCibleAttaque(False)
  1393. if coordCible != None and self.pionSelectionne() != None and self.modeCombat == "combatAttaqueDist":
  1394. if coordCible == None:
  1395. coordCible = pion.position
  1396. if coordCible != self.modeParam["cibleAttaqueDist"]:
  1397. if self.modeParam["ligneAttaqueDist"] == None:
  1398. self.modeParam["ligneAttaqueDist"] = QGraphicsLineItem()
  1399. self.modeParam["ligneAttaqueDist"].setZValue(100)
  1400. pinceau = QPen()
  1401. pinceau.setWidth(6)
  1402. self.modeParam["ligneAttaqueDist"].setPen(pinceau)
  1403. self.modeParam["ligneAttaqueDist"].prepareGeometryChange()
  1404. self.addItem(self.modeParam["ligneAttaqueDist"])
  1405. z = 0
  1406. pionSurCase = self.cases[coordCible].pionOccupant()
  1407. if pionSurCase != None:
  1408. z = pionSurCase.z
  1409. cibleValide = self.estCibleAttaqueDistValide(self.pionSelectionne().position, coordCible, z)
  1410. if pionSurCase != None:
  1411. pionSurCase.estCibleAttaque(True, cibleValide)
  1412. self.modeParam["pionCibleAttaqueDist"] = pionSurCase
  1413. else:
  1414. self.cases[coordCible].majEstCibleCurseur(True, cibleValide)
  1415. self.modeParam["cibleAttaqueDist"] = coordCible
  1416. point1 = self.cases[self.pionSelectionne().position].centreGraphique
  1417. point2 = self.cases[coordCible].centreGraphique
  1418. ligne = QLineF(point1, point2)
  1419. self.modeParam["ligneAttaqueDist"].setLine(ligne)
  1420. else:
  1421. if self.modeParam["ligneAttaqueDist"] != None:
  1422. self.modeParam["ligneAttaqueDist"].prepareGeometryChange()
  1423. self.removeItem(self.modeParam["ligneAttaqueDist"])
  1424. self.modeParam["ligneAttaqueDist"] = None
  1425. self.modeParam["cibleAttaqueDist"] = None
  1426. ## if pion == None:
  1427. ## self.modeParam["pionCibleAttaqueDist"] = None
  1428. def estCibleAttaqueDistValide(self, coordOrigine, coordCible, zPion = 0):
  1429. """la case cible est elle valide pour une attaque a distance depuis la position et hauteur
  1430. du pion selectionne? on compare pour ce faire les altitudes des cases sur la ligne de mire"""
  1431. casesLigneMire = []
  1432. #on preselectionne les cases concernees
  1433. preSelection = self.preSelectionCollision(self.cases[coordOrigine].centreGraphique, self.cases[coordCible].centreGraphique)
  1434. if coordOrigine in preSelection:
  1435. preSelection.remove(coordOrigine)
  1436. if coordCible in preSelection:
  1437. preSelection.remove(coordCible)
  1438. for coord in preSelection:
  1439. ligne = QLineF(self.cases[coordOrigine].centreGraphique, self.cases[coordCible].centreGraphique)
  1440. ligneGraphique = QGraphicsLineItem(ligne)
  1441. ligneGraphique.prepareGeometryChange()
  1442. self.addItem(ligneGraphique)
  1443. if self.cases[coord].collidesWithItem(ligneGraphique, Qt.IntersectsItemShape):
  1444. casesLigneMire.append(coord)
  1445. self.removeItem(ligneGraphique)
  1446. del ligneGraphique
  1447. #on trie les cases par distance au point d'origine (de la plus proche a la plus eloignee)
  1448. casesLigneMireDistance = {} #distance: coord
  1449. for coord in casesLigneMire:
  1450. distance = sqrt((coordOrigine[0] - coord[0])**2 + (coordOrigine[1] - coord[1])**2)
  1451. casesLigneMireDistance[coord] = distance
  1452. #on compare enfin les altitudes de chaque case en fonction de la distance
  1453. zOrigine = self.cases[coordOrigine].altitude + self.pionSelectionne().z + self.pionSelectionne().creature.hauteur
  1454. zCible = self.cases[coordCible].altitude + zPion
  1455. distanceTot = sqrt((coordCible[0] - coordOrigine[0])**2 + (coordCible[1] - coordOrigine[1])**2)
  1456. valide = True
  1457. for coord in casesLigneMireDistance:
  1458. if self.cases[coord].terrain.visibilite == False:
  1459. valide = False
  1460. break
  1461. else:
  1462. if zOrigine >= zCible:
  1463. z = (zOrigine - zCible) * (casesLigneMireDistance[coord] / distanceTot)
  1464. else:
  1465. z = (zCible - zOrigine) * ((distanceTot - casesLigneMireDistance[coord]) / distanceTot)
  1466. if self.cases[coord].estOccupee(int(z)):
  1467. if self.modeParam["pionCibleAttaqueDist"] != None:
  1468. if self.cases[coord].estOccupeePar(int(z)) != self.modeParam["pionCibleAttaqueDist"]:
  1469. valide = False
  1470. else:
  1471. valide = False
  1472. break
  1473. return valide
  1474. def validerAttaqueDist(self):
  1475. """on essaie de valider une attaque a distance vers un pion ou une case"""
  1476. if self.modeParam["pionCibleAttaqueDist"] != None:
  1477. msg = "{} attaque a distance le pion {}".format(self.pionSelectionne().txtId(),
  1478. self.modeParam["pionCibleAttaqueDist"].txtId())
  1479. valide = self.estCibleAttaqueDistValide(self.pionSelectionne().position, self.modeParam["cibleAttaqueDist"], self.modeParam["pionCibleAttaqueDist"].z)
  1480. else:
  1481. msg = "{} attaque a distance la case {}".format(self.pionSelectionne().txtId(),
  1482. self.modeParam["cibleAttaqueDist"])
  1483. valide = self.estCibleAttaqueDistValide(self.pionSelectionne().position, self.modeParam["cibleAttaqueDist"])
  1484. if not valide:
  1485. msg += " [INVALIDE]"
  1486. print(msg)
  1487. self.majModeCombat("aucun")
  1488. def rectEllipseCirculaire(self, centre, rayon):
  1489. """renvoie le QRectF definissant une ellipse ayant le QPointF pour centre et le rayon en cases entres en param
  1490. attention: l'ellipse n'est pas tout a fait circulaire, elle couvre horizontalement et
  1491. verticalement le nombre de cases demandees"""
  1492. rect = QRectF()
  1493. if rayon > 0:
  1494. p1 = QPointF((centre.x() - (rayon*self.hCase)), (centre.y() - (rayon*self.hCase)))
  1495. p2 = QPointF((centre.x() + (rayon*self.hCase)), (centre.y() + (rayon*self.hCase)))
  1496. rect.setTopLeft(p1)
  1497. rect.setBottomRight(p2)
  1498. else:
  1499. rect = None
  1500. return rect
  1501. def polygoneCone(self, point1, point2):
  1502. """renvoie le polygone du cone defini par les deux points (origine, distance)"""
  1503. ligne1 = QLineF(point1, point2)
  1504. longueur = ligne1.length()
  1505. ligne1.setAngle(ligne1.angle() + 22.5)
  1506. ligne1.setLength(1.1547*longueur)
  1507. ligne2 = QLineF(point1, point2)
  1508. ligne2.setAngle(ligne2.angle() - 22.5)
  1509. ligne2.setLength(1.1547*longueur)
  1510. polygone = QPolygonF()
  1511. polygone.append(point1)
  1512. polygone.append(ligne1.p2())
  1513. polygone.append(ligne2.p2())
  1514. return polygone
  1515. def majFormeAttaqueZone(self):
  1516. """cree la forme de l'attaque de zone"""
  1517. for coord in self.modeParam["listeCasesAttaqueZone"]:
  1518. self.cases[coord].majEstCibleAttaque(False)
  1519. for numPion in self.pionsSurListeCase(self.modeParam["listeCasesAttaqueZone"]):
  1520. self.pions[numPion].estCibleAttaque(False)
  1521. if self.modeCombat == "combatAttaqueZone" and self.pionSelectionne() != None:
  1522. if self.modeParam["typeAttaqueZone"] == "ligne":
  1523. if self.modeParam["formeAttaqueZone"] == None:
  1524. self.modeParam["origineAttaqueZone"] = self.cases[self.pionSelectionne().position].centreGraphique
  1525. self.modeParam["point2AttaqueZone"] = self.cases[self.pionSelectionne().position].centreGraphique
  1526. self.modeParam["formeAttaqueZone"] = QGraphicsLineItem()
  1527. self.modeParam["formeAttaqueZone"].setPen(QPen(QColor("black")))
  1528. self.modeParam["formeAttaqueZone"].prepareGeometryChange()
  1529. self.addItem(self.modeParam["formeAttaqueZone"])
  1530. if self.modeParam["origineAttaqueZone"] != self.modeParam["point2AttaqueZone"]:
  1531. ligne = QLineF(self.modeParam["origineAttaqueZone"], self.modeParam["point2AttaqueZone"])
  1532. self.modeParam["formeAttaqueZone"].setLine(ligne)
  1533. self.modeParam["listeCasesAttaqueZone"] = []
  1534. lst = self.casesSousForme(self.modeParam["formeAttaqueZone"],False)
  1535. for coord in lst:
  1536. self.modeParam["listeCasesAttaqueZone"].append(coord)
  1537. #on retire la case du pion selectionne si besoin:
  1538. if self.pionSelectionne().position in self.modeParam["listeCasesAttaqueZone"]:
  1539. self.modeParam["listeCasesAttaqueZone"].remove(self.pionSelectionne().position)
  1540. elif self.modeParam["typeAttaqueZone"] == "disque":
  1541. if self.modeParam["ligneMireAttaqueZone"] == None:
  1542. self.modeParam["origineAttaqueZone"] = self.cases[self.pionSelectionne().position].centreGraphique
  1543. self.modeParam["point2AttaqueZone"] = self.cases[self.pionSelectionne().position].centreGraphique
  1544. self.modeParam["ligneMireAttaqueZone"] = QGraphicsLineItem()
  1545. self.modeParam["ligneMireAttaqueZone"].setPen(QPen(QColor("black")))
  1546. self.modeParam["ligneMireAttaqueZone"].prepareGeometryChange()
  1547. self.addItem(self.modeParam["ligneMireAttaqueZone"])
  1548. if self.modeParam["origineAttaqueZone"] != self.modeParam["point2AttaqueZone"]:
  1549. ligne = QLineF(self.modeParam["origineAttaqueZone"], self.modeParam["point2AttaqueZone"])
  1550. self.modeParam["ligneMireAttaqueZone"].setLine(ligne)
  1551. coordCible = self.coordonneesAuPoint(self.modeParam["point2AttaqueZone"])
  1552. cibleValide = self.estCibleAttaqueDistValide(self.pionSelectionne().position, coordCible)
  1553. if cibleValide:
  1554. rayon = self.fenetre.ui.combatZone_disqueRayon.value()
  1555. if self.modeParam["formeAttaqueZone"] == None:
  1556. self.modeParam["formeAttaqueZone"] = QGraphicsEllipseItem()
  1557. self.modeParam["formeAttaqueZone"].setPen(QPen(QColor("black")))
  1558. self.modeParam["formeAttaqueZone"].prepareGeometryChange()
  1559. self.addItem(self.modeParam["formeAttaqueZone"])
  1560. rect = self.rectEllipseCirculaire(self.modeParam["point2AttaqueZone"], rayon)
  1561. self.modeParam["listeCasesAttaqueZone"] = []
  1562. if rect != None and rect.bottomRight() != rect.topLeft():
  1563. self.modeParam["formeAttaqueZone"].setRect(rect)
  1564. lst = self.zone(coordCible, rayon, 0) #zone bcp plus rapide que casesSousforme
  1565. self.modeParam["listeCasesAttaqueZone"] = lst
  1566. else:
  1567. self.cases[coordCible].majEstCibleCurseur(True, False)
  1568. self.modeParam["listeCasesAttaqueZone"] = []
  1569. if self.modeParam["formeAttaqueZone"] != None:
  1570. self.modeParam["formeAttaqueZone"].setVisible(cibleValide == True and rect != None)
  1571. elif self.modeParam["typeAttaqueZone"] == "cone":
  1572. if self.modeParam["formeAttaqueZone"] == None:
  1573. self.modeParam["origineAttaqueZone"] = self.cases[self.pionSelectionne().position].centreGraphique
  1574. self.modeParam["point2AttaqueZone"] = self.cases[self.pionSelectionne().position].centreGraphique
  1575. self.modeParam["formeAttaqueZone"] = QGraphicsPolygonItem()
  1576. self.modeParam["formeAttaqueZone"].setPen(QPen(QColor("black")))
  1577. self.modeParam["formeAttaqueZone"].prepareGeometryChange()
  1578. self.addItem(self.modeParam["formeAttaqueZone"])
  1579. if self.modeParam["origineAttaqueZone"] != self.modeParam["point2AttaqueZone"]:
  1580. cone = self.polygoneCone(self.modeParam["origineAttaqueZone"], self.modeParam["point2AttaqueZone"])
  1581. self.modeParam["formeAttaqueZone"].setPolygon(cone)
  1582. lst = self.casesSousForme(self.modeParam["formeAttaqueZone"], True, True)
  1583. self.modeParam["listeCasesAttaqueZone"] = []
  1584. for coord in lst:
  1585. if coord != self.pionSelectionne().position:
  1586. self.modeParam["listeCasesAttaqueZone"].append(coord)
  1587. for coord in self.modeParam["listeCasesAttaqueZone"]:
  1588. self.cases[coord].majEstCibleAttaque(True)
  1589. for numPion in self.pionsSurListeCase(self.modeParam["listeCasesAttaqueZone"]):
  1590. self.pions[numPion].estCibleAttaque(True)
  1591. def validerAttaqueZone(self):
  1592. """l'attaque de zone est validee"""
  1593. if self.modeActif == "pionSelectionne" and self.modeCombat == "combatAttaqueZone":
  1594. for numPion in self.pionsSurListeCase(self.modeParam["listeCasesAttaqueZone"]):
  1595. print("{} a lance une attaque de zone sur {}".format(self.pionSelectionne().nom, self.pionSelectionne().numero , \
  1596. self.pions[numPion].nom, self.pions[numPion].numero))
  1597. self.majModeCombat("aucun")
  1598. def reinitAttaqueZone(self):
  1599. if self.modeParam["formeAttaqueZone"] != None:
  1600. self.modeParam["formeAttaqueZone"].prepareGeometryChange()
  1601. self.removeItem(self.modeParam["formeAttaqueZone"])
  1602. self.modeParam["formeAttaqueZone"] = None
  1603. if self.modeParam["ligneMireAttaqueZone"] != None:
  1604. self.modeParam["ligneMireAttaqueZone"].prepareGeometryChange()
  1605. self.removeItem(self.modeParam["ligneMireAttaqueZone"])
  1606. self.modeParam["ligneMireAttaqueZone"] = None
  1607. if self.modeParam["point2AttaqueZone"] != None:
  1608. self.cases[self.coordonneesAuPoint(self.modeParam["point2AttaqueZone"])].majEstCibleCurseur(False)
  1609. for coord in self.modeParam["listeCasesAttaqueZone"]:
  1610. self.cases[coord].majEstCibleAttaque(False)
  1611. for numPion in self.pionsSurListeCase(self.modeParam["listeCasesAttaqueZone"]):
  1612. self.pions[numPion].estCibleAttaque(False)
  1613. def pionSurCase(self, coord):
  1614. """renvoie le pion present sur la case, none sinon"""
  1615. retour = None
  1616. for num in self.pions:
  1617. if self.pions[num].position == coord:
  1618. retour = num
  1619. return retour
  1620. def pionsSurListeCase(self, listeCases):
  1621. """renvoie la liste des pions presents sur la liste de cases"""
  1622. retour = []
  1623. for coord in listeCases:
  1624. pion = self.cases[coord].pionOccupant()
  1625. if pion != None and not pion.numero in retour:
  1626. retour.append(pion.numero)
  1627. return retour
  1628. def majZonePlacement(self, listeCases):
  1629. """met a jour la forme et l'affichage de la zone de placement initale des joueurs"""
  1630. if len(listeCases) > 0:
  1631. if self.polygoneZonePlacement == None:
  1632. self.polygoneZonePlacement = QGraphicsPolygonItem(scene=self)
  1633. self.polygoneZonePlacement.setZValue(0)
  1634. qCouleurFond = QColor("white")
  1635. qCouleurFond.setAlpha(50)
  1636. self.polygoneZonePlacement.setBrush(qCouleurFond)
  1637. pinceau = QPen(QColor("orange"))
  1638. pinceau.setWidth(20)
  1639. self.polygoneZonePlacement.setPen(pinceau)
  1640. self.polygoneZonePlacement.setAcceptedMouseButtons(Qt.NoButton)
  1641. self.polygoneZonePlacement.setAcceptHoverEvents(False)
  1642. self.addItem(self.polygoneZonePlacement)
  1643. listeCases2 = []
  1644. for coord in listeCases:
  1645. if self.cases[coord].estFranchissable():
  1646. listeCases2.append(coord)
  1647. self.polygoneZonePlacement.setPolygon(self.polygoneAgglo(listeCases2))
  1648. self.listeCasesZonePlacement = listeCases
  1649. def materialiserPions(self,actif):
  1650. """avtive/desactive la reception par les pions (autres que le pion selectionne) des hover events"""
  1651. for numPion in self.pions:
  1652. if numPion != self.modeParam["numPionSelectionne"]:
  1653. self.pions[numPion].setAcceptsHoverEvents(actif)
  1654. self.pions[numPion].polygoneGraphique.setAcceptsHoverEvents(actif)
  1655. for numPion in self.decors:
  1656. self.decors[numPion].setAcceptsHoverEvents(actif)
  1657. self.decors[numPion].polygoneGraphique.setAcceptsHoverEvents(actif)
  1658. #######################
  1659. ######## interaction avec les cases, decors et pions #############
  1660. def pionSelectionne(self):
  1661. """renvoie le pion actuellement selectionne"""
  1662. if self.modeParam["numPionSelectionne"] in self.pions:
  1663. retour = self.pions[self.modeParam["numPionSelectionne"]]
  1664. else:
  1665. retour = None
  1666. return retour
  1667. def pionDecorSelectionne(self):
  1668. """renvoie le pion actuellement selectionne"""
  1669. if self.modeParam["numPionSelectionne"] in self.decors:
  1670. retour = self.decors[self.modeParam["numPionSelectionne"]]
  1671. else:
  1672. retour = None
  1673. return retour
  1674. def caseCliquee(self, x, y):
  1675. """on a clique sur la case (clic gauche)"""
  1676. coord = (x, y)
  1677. accepte = False
  1678. if coord in self.cases:
  1679. case = self.cases[coord]
  1680. if self.modeActif == "caseCopie":
  1681. if len(case.terrain.nom) > 0:
  1682. self.majMode("caseMajTerrain", case.terrain)
  1683. accepte = True
  1684. else:
  1685. self.majMode("caseMajTerrain", case.couleur)
  1686. accepte = True
  1687. elif self.modeActif[0:7] == "caseMaj" or self.modeActif == "cachePlacer" or self.modeActif == "majZonePlacement":
  1688. self.creerOrigineFormeDessin(coord)
  1689. accepte = True
  1690. elif self.modeActif == "pionDecorSelectionne":
  1691. if coord != self.pionDecorSelectionne().position:
  1692. self.pionDeposer(coord)
  1693. accepte = True
  1694. elif self.modeActif == "pionSelectionne":
  1695. if self.modeCombat == "combatDeplacement":
  1696. if self.pionSelectionne() != None:
  1697. if coord != self.pionSelectionne().position:
  1698. self.pionDeposer(coord)
  1699. accepte = True
  1700. elif self.modeCombat == "combatAttaqueCaC":
  1701. if self.modeParam["cibleAttaqueCaC"] in self.modeParam["zoneAttaqueCaC"]:
  1702. print("{} attaque {} au corps a corps".format(self.pionSelectionne().nom, self.modeParam["cibleAttaqueCaC"]))
  1703. self.majMode("pionSelectionne", None)
  1704. accepte = True
  1705. else:
  1706. print("cette case est invalide pour une corps a corps")
  1707. accepte = True
  1708. elif self.modeCombat == "combatAttaqueDist":
  1709. self.validerAttaqueDist()
  1710. accepte = True
  1711. elif self.modeCombat == "combatAttaqueZone":
  1712. self.validerAttaqueZone()
  1713. accepte = True
  1714. elif self.modeActif == "pionCreation":
  1715. self.pionCreer(coord)
  1716. accepte = True
  1717. elif self.modeActif == "pionDecorCreation":
  1718. self.pionDecorCreer(coord)
  1719. accepte = True
  1720. return accepte
  1721. def caseSurvolClicEnfonce(self, coordCase):
  1722. """une case est survolee par le curseur (le clic gauche est enfonce)"""
  1723. if self.modeActif[0:7] == "caseMaj" or self.modeActif == "cachePlacer" or self.modeActif == "majZonePlacement":
  1724. if self.cases[coordCase].centreGraphique != self.modeParam["point2FormeDessin"]:
  1725. if self.modeParam["origineFormeDessin"] == None: self.modeParam["origineFormeDessin"] = self.cases[coordCase].centreGraphique
  1726. self.modeParam["point2FormeDessin"] = self.cases[coordCase].centreGraphique
  1727. self.majFormeDessin()
  1728. def clicGaucheRelache(self):
  1729. """si une forme de selection etait affichee, elle est validee"""
  1730. if len(self.modeParam["listeCasesFormeDessin"]) > 0:
  1731. self.validerFormeDessin()
  1732. def caseSurvol(self, x, y):
  1733. """une case est survole par le curseur, on affiche ses informations dans la zone prevue"""
  1734. case = self.cases[(x, y)]
  1735. self.majInfosCase(case)
  1736. if (self.modeActif == "pionSelectionne" and self.modeCombat == "combatDeplacement") or self.modeActif == "pionCreation"\
  1737. or self.modeActif == "pionDecorSelectionne" or self.modeActif == "pionDecorCreation":
  1738. self.majProjectionPosition(False)
  1739. self.modeParam["coordProjectionPosition"] = (case.x, case.y)
  1740. self.majProjectionPosition(True)
  1741. elif self.modeActif == "pionSelectionne" and self.modeCombat == "combatAttaqueDist" and self.modeParam["cibleAttaqueDist"] != (case.x, case.y):
  1742. self.majLigneMireAttaqueDist((case.x,case.y))
  1743. elif self.modeActif == "pionSelectionne" and self.pionSelectionne() != None and self.modeCombat == "combatAttaqueZone":
  1744. if case.centreGraphique != self.modeParam["point2AttaqueZone"]:
  1745. self.cases[self.coordonneesAuPoint(self.modeParam["point2AttaqueZone"])].majEstCibleCurseur(False)
  1746. self.modeParam["point2AttaqueZone"] = case.centreGraphique
  1747. self.majFormeAttaqueZone()
  1748. def afficherListeCases(self, listeCases, actif):
  1749. """met ou non en evidence les cases selectionnees"""
  1750. for coord in listeCases:
  1751. self.cases[coord].majEstCibleCurseur(actif)
  1752. def majListeCases(self, listeCases):
  1753. """met a jour les cases dont les coordonnees sont dans la liste"""
  1754. self.afficherListeCases(listeCases, False)
  1755. if self.modeActif[0:7] == "caseMaj":
  1756. for coord in listeCases:
  1757. self.caseMaj(coord)
  1758. elif self.modeActif == "cachePlacer":
  1759. self.majAffichageCache(self.cacheEnCours, False)
  1760. self.caches[self.cacheEnCours]["listeCases"] = listeCases
  1761. self.majAffichageCache(self.cacheEnCours, True)
  1762. elif self.modeActif == "majZonePlacement":
  1763. self.majZonePlacement(listeCases)
  1764. def caseMaj(self, coordCase):
  1765. """met a jour la case selon le mode actif (et les cases adjacentes selon l'epaisseur du pinceau)"""
  1766. if self.modeActif == "caseMajTerrain":
  1767. if self.modeParam["terrain"] != None:
  1768. if self.modeParam["terrain"].__class__.__name__ == "Terrain":
  1769. self.cases[coordCase].majTerrain(self.modeParam["terrain"])
  1770. elif self.modeParam["terrain"].__class__.__name__ == "QColor":
  1771. if self.modeParam["terrain"].isValid():
  1772. self.cases[coordCase].majTerrain(None)
  1773. self.cases[coordCase].majCouleur(self.modeParam["terrain"])
  1774. self.cases[coordCase].majAltitude(int(self.fenetre.ui.altitudeCase.value()))
  1775. elif self.modeActif == "caseMajEffet":
  1776. self.cases[coordCase].majEffet(self.modeParam["effet"])
  1777. elif self.modeActif == "caseMajAltitude":
  1778. self.cases[coordCase].majAltitude(int(self.fenetre.ui.altitudeCase.value()))
  1779. def majInfosCase(self, case=None):
  1780. """met a jour les informations d'un pion dans la zone prevue"""
  1781. if case != None:
  1782. if len(case.terrain.nom) > 0:
  1783. self.fenetre.ui.infoCaseEnCours_terrain.setText(QString.fromUtf8(case.terrain.nom))
  1784. else:
  1785. self.fenetre.ui.infoCaseEnCours_terrain.setText(QString.fromUtf8("Case"))
  1786. self.fenetre.ui.infoCaseEnCours_coord.setText(QString.fromUtf8("X: {} Y: {}".format(case.x, case.y)))
  1787. self.fenetre.ui.infoCaseEnCours_altitude.setText(QString.fromUtf8("Alt.: {}".format(case.altitude)))
  1788. if case.effetActif != "":
  1789. pix = QPixmap(QString.fromUtf8("img\\"+case.imgEffet[case.effetActif]))
  1790. pix = pix.scaled(21, 21, Qt.KeepAspectRatio, Qt.SmoothTransformation)
  1791. self.fenetre.ui.infoCaseEnCours_effetImg.setPixmap(pix)
  1792. self.fenetre.ui.infoCaseEnCours_effetNom.setText(QString.fromUtf8(case.effetActif))
  1793. else:
  1794. self.fenetre.ui.infoCaseEnCours_effetImg.clear()
  1795. self.fenetre.ui.infoCaseEnCours_effetNom.clear()
  1796. self.fenetre.ui.infoCase.setVisible(True)
  1797. else:
  1798. self.fenetre.ui.infoCase.setVisible(False)
  1799. def pionClique(self, numPion):
  1800. """on a clique sur ce pion"""
  1801. accepte = False
  1802. if self.pionSelectionne() != None and self.modeCombat == "combatAttaqueDist":
  1803. self.validerAttaqueDist()
  1804. accepte = True
  1805. elif self.modeActif == "standard" and self.pionSelectionne() == None:
  1806. self.pionSaisir(numPion)
  1807. accepte = True
  1808. return accepte
  1809. def pionSurvol(self, numPion):
  1810. """le pion est survole par le curseur, on affiche ses informations dans la zone prevue"""
  1811. if numPion in self.pions:
  1812. pion = self.pions[numPion]
  1813. else:
  1814. pion = None
  1815. self.majInfosPion(pion)
  1816. if self.pionSelectionne() != None and self.modeCombat == "combatAttaqueCaC":
  1817. self.majProjectionAttaqueCaC(pion)
  1818. pass
  1819. if numPion in self.pions:
  1820. ## case = self.cases[self.pions[numPion].position]
  1821. self.caseSurvol(self.pions[numPion].position[0], self.pions[numPion].position[1])
  1822. def pionDoubleClic(self, numPion):
  1823. """on a double-clique sur le pion"""
  1824. accepte = False
  1825. if self.pionSelectionne() == self.pions[numPion] and self.modeCombat == "aucun":
  1826. ## self.pionSaisir(numPion)
  1827. self.majModeCombat("combatDeplacement")
  1828. accepte = True
  1829. return accepte
  1830. def pionCreer(self, coordCase):
  1831. """creer un jeton aux coordonnees indiquees"""
  1832. valide = True
  1833. for coord in self.modeParam["formeProjectionPosition"].listeCases(coordCase):
  1834. if not coord in self.cases:
  1835. valide = False
  1836. else:
  1837. if not self.cases[coord].estFranchissable():
  1838. valide = False
  1839. if valide:
  1840. numero = 1
  1841. if len(self.pions) > 0:
  1842. numero = max(self.pions) + 1
  1843. pion = Pion(self, numero)
  1844. if self.modeParam["creature"] != None:
  1845. nom = self.modeParam["creature"].nom
  1846. elif len(self.fenetre.ui.pionSimple_nom.text().toUtf8()) > 0:
  1847. nom = self.fenetre.ui.pionSimple_nom.text().toUtf8()
  1848. numTxt = self.numeroterNom(nom)
  1849. self.pions[numero] = pion
  1850. pion.creer(coordCase[0], coordCase[1], nom, numTxt, self.modeParam["creature"], self.modeParam["couleurPion"], self.modeParam["nbRotations"])
  1851. self.pionDeplacerDansOrdreJeu(numero, len(self.ordreJeu) + 2)
  1852. def numeroterNom(self, nom):
  1853. """renvoie le nom du pion avec un numero complementaire si necessaire """
  1854. i = 1
  1855. for numPion in self.pions:
  1856. if self.pions[numPion].nom == nom:
  1857. i += 1
  1858. if i == 1:
  1859. retour = ""
  1860. else:
  1861. retour = str(i)
  1862. return retour
  1863. def pionSaisir(self, numPion):
  1864. """on saisit un pion"""
  1865. if numPion != self.modeParam["numPionSelectionne"]:
  1866. self.majMode("pionSelectionne", numPion)
  1867. def pionDeposer(self, coordCase):
  1868. """on depose le pion sur la case voulue"""
  1869. if self.pionSelectionne() != None:
  1870. pion = self.pionSelectionne()
  1871. else:
  1872. pion = self.pionDecorSelectionne()
  1873. if pion != None:
  1874. valide = True
  1875. for coord in self.modeParam["formeProjectionPosition"].listeCases(coordCase, self.modeParam["nbRotations"]):
  1876. if coord in self.cases:
  1877. if not self.cases[coord].estFranchissable(pion.z):
  1878. if not self.cases[coord].estOccupeePar(pion.z) == pion:
  1879. #si la case est occupee par le pion qu'on depose, elle est donc valide pour un deplacement
  1880. valide = False
  1881. else:
  1882. valide = False
  1883. if valide:
  1884. positionPrecedente = pion.position
  1885. #mise a jour de la position du pion selon le mode
  1886. if self.modePrincipal == "combat":
  1887. ## if coordCase in pion.champDeplacement:
  1888. ## print("le pion a ete deplace de {} vers {} \n -> soit une distance de {} cases"\
  1889. ## "".format(positionPrecedente, coordCase, self.pionSelectionne().champDeplacement[coordCase]))
  1890. self.majInfosPion()
  1891. pion.majPosition(coordCase, self.modeParam["nbRotations"])
  1892. self.majModeCombat("aucun")
  1893. else:
  1894. print("Deplacement impossible!")
  1895. def afficherChampDeplacement(self, actif):
  1896. """cree et affiche ou efface et detruit le champ de deplacement du pion selectionne"""
  1897. if self.pionSelectionne() != None:
  1898. if actif:
  1899. self.pionSelectionne().champDeplacement = self.zone(self.pionSelectionne().position, \
  1900. self.pionSelectionne().deplacementRestant, \
  1901. True)
  1902. for coord in self.pionSelectionne().champDeplacement:
  1903. self.cases[coord].majEstDansChampDeplacement(actif)
  1904. if not actif:
  1905. self.pionSelectionne().champDeplacement = {}
  1906. def majInfosPion(self, pionSurvole=None):
  1907. """met a jour les informations d'un pion dans la zone prevue"""
  1908. if pionSurvole != None:
  1909. pion = pionSurvole
  1910. else:
  1911. if self.pionSelectionne() != None:
  1912. pion = self.pionSelectionne()
  1913. else:
  1914. pion = None
  1915. if pion != None:
  1916. self.fenetre.ui.infoPionEnCours_nom.setText(QString.fromUtf8(pion.nom))
  1917. self.fenetre.ui.infoPionEnCours_depRestant.setText(QString.fromUtf8("Dep.: {}".format(pion.deplacementRestant)))
  1918. if len(pion.creature.img["nom"]) > 0:
  1919. pix = QPixmap(QString.fromUtf8("img\\"+pion.creature.img["nom"]))
  1920. pix = pix.scaled(61, 51, Qt.KeepAspectRatio, Qt.SmoothTransformation)
  1921. self.fenetre.ui.infoPionEnCours_image.setPixmap(pix)
  1922. self.fenetre.ui.infoPion.setVisible(True)
  1923. self.fenetre.ui.infoPionEnCours_etourdi.setVisible(False)
  1924. self.fenetre.ui.infoPionEnCours_brule.setVisible(False)
  1925. self.fenetre.ui.infoPionEnCours_mouille.setVisible(False)
  1926. self.fenetre.ui.infoPionEnCours_vol.setVisible((pion.z != 0))
  1927. self.fenetre.ui.infoPionEnCours_endormi.setVisible(False)
  1928. self.fenetre.ui.infoPionEnCours_paralyse.setVisible(False)
  1929. self.fenetre.ui.infoPionEnCours_entrave.setVisible(False)
  1930. self.fenetre.ui.infoPionEnCours_mort.setVisible(False)
  1931. else:
  1932. self.fenetre.ui.infoPion.setVisible(False)
  1933. def majZPion(self, valeur):
  1934. """met a jour l'altitude du pion selectionne"""
  1935. if self.pionSelectionne() != None:
  1936. self.pionSelectionne().majZ(valeur)
  1937. def pionDecorSurvol(self, numPionDecor):
  1938. """le pion-decor est survole par le curseur, on affiche ses informations dans la zone prevue"""
  1939. if numPionDecor in self.decors:
  1940. self.majInfosDecor(self.decors[numPionDecor])
  1941. else:
  1942. self.majInfosDecor(None)
  1943. def pionDecorSaisir(self, numPionDecor):
  1944. """on saisit un pion decor (mode creation seulement)"""
  1945. if numPionDecor != self.modeParam["numPionSelectionne"]:
  1946. self.majMode("pionDecorSelectionne", numPionDecor)
  1947. def majInfosDecor(self, pionDecor=None):
  1948. """met a jour les informations d'un pion dans la zone prevue"""
  1949. if pionDecor != None:
  1950. self.fenetre.ui.infoDecorEnCours_nom.setText(QString.fromUtf8(pionDecor.decor.nom))
  1951. self.fenetre.ui.infoDecorEnCours_hauteur.setText(QString.fromUtf8("Haut.: {}".format(pionDecor.decor.hauteur)))
  1952. if len(pionDecor.decor.img["nom"]) > 0:
  1953. pix = QPixmap(QString.fromUtf8("img\\"+pionDecor.decor.img["nom"]))
  1954. pix = pix.scaled(61, 51, Qt.KeepAspectRatio, Qt.SmoothTransformation)
  1955. self.fenetre.ui.infoDecorEnCours_image.setPixmap(pix)
  1956. self.fenetre.ui.infoDecor.setVisible(True)
  1957. else:
  1958. self.fenetre.ui.infoDecor.setVisible(False)
  1959. def majProjectionPosition(self, activer, z=0):
  1960. """maj l'affichage de la projection de la position d'un objet avant creation ou deplacement (pion, decor...)"""
  1961. valide = True
  1962. possible = True
  1963. if self.modeParam["formeProjectionPosition"] != None and self.modeParam["coordProjectionPosition"] != None:
  1964. proj = self.modeParam["formeProjectionPosition"].listeCases(self.modeParam["coordProjectionPosition"], self.modeParam["nbRotations"])
  1965. for coord in proj:
  1966. if not coord in self.cases:
  1967. valide = False #deplacement valide? (pas d'occupation d'une case hors plateau)
  1968. if valide == True:
  1969. for coord in proj:
  1970. if self.cases[coord].estFranchissable(z) or self.cases[coord].estOccupeePar(z) == self.pionSelectionne():
  1971. #la case est franchissable, ou elle ne l'est pas mais est occupée par le pion meme qu'on deplace
  1972. self.cases[coord].majEstCibleCurseur(activer, True)
  1973. #on affiche la distance correspondante en bulle-info
  1974. if activer and self.modeActif == "pionSelectionne" and self.modePrincipal == "combat":
  1975. if coord in self.pionSelectionne().champDeplacement:
  1976. msg = "Distance {}".format(self.pionSelectionne().champDeplacement[coord])
  1977. QToolTip.showText(QCursor.pos(), msg, self.fenetre.ui.vuePlateau)
  1978. else:
  1979. self.cases[coord].majEstCibleCurseur(activer, False)
  1980. def majProjectionAttaqueCaC(self, pionCible):
  1981. """affiche ou non la cible de l'attaque au corps a corps, selon sa validite ou non"""
  1982. if self.modeActif == "pionSelectionne" and self.modeCombat == "combatAttaqueCaC":
  1983. if pionCible != self.pionSelectionne():
  1984. if pionCible != None:
  1985. conditionPossible = (pionCible.position in self.modeParam["zoneAttaqueCaC"])
  1986. pionCible.estCibleAttaque(True, conditionPossible)
  1987. else:
  1988. if self.modeParam["cibleAttaqueCaC"] != None:
  1989. self.modeParam["cibleAttaqueCaC"].estCibleAttaque(False)
  1990. self.modeParam["cibleAttaqueCaC"] = pionCible
  1991. def majZoneAttaqueCaC(self, actif = True):
  1992. """affiche ou non les cases a portee du pion selectionne pour une attaque au corps-a-corps"""
  1993. for coord in self.modeParam["zoneAttaqueCaC"]:
  1994. self.cases[coord].majEstDansChampDeplacement(actif)
  1995. def pionDecorCreer(self, coordCase):
  1996. """creer un jeton aux coordonnees indiquees"""
  1997. valide = True
  1998. for coord in self.modeParam["formeProjectionPosition"].listeCases(coordCase, self.modeParam["nbRotations"]):
  1999. if not self.cases[coord].estFranchissable():
  2000. valide = False
  2001. if valide:
  2002. numero = 10001
  2003. if len(self.decors) > 0:
  2004. numero = max(self.decors) + 10001
  2005. pionDecor = PionDecor(self, numero)
  2006. pionDecor.creer(coordCase[0], coordCase[1], self.modeParam["decor"], self.modeParam["nbRotations"])
  2007. self.decors[numero] = pionDecor
  2008. def pionSupprimer(self, num):
  2009. """supprime le pion entre en parametre"""
  2010. #settrace(trace_calls)
  2011. if num in self.pions:
  2012. for coord in self.pions[num].forme.listeCases(self.pions[num].position, self.pions[num].nbRotations):
  2013. self.cases[coord].majOccupation(self.pions[num])
  2014. self.pionSurvol(None)
  2015. self.pionDeplacerDansOrdreJeu(num, 0)
  2016. pionSuppr = self.pions.pop(num)
  2017. pionSuppr.supprimer()
  2018. else:
  2019. print("erreur: ce pion n'est pas dans la liste des pions")
  2020. def pionDecorSupprimer(self, num):
  2021. """supprime le pion entre en parametre"""
  2022. if num in self.decors:
  2023. for coord in self.decors[num].forme.listeCases(self.decors[num].position, self.decors[num].nbRotations):
  2024. self.cases[coord].majOccupation(self.decors[num])
  2025. self.pionDecorSurvol(None)
  2026. pionDecorASuppr = self.decors.pop(num)
  2027. pionDecorASuppr.supprimer()
  2028. else:
  2029. print("ce pion n'est pas dans la liste des decors")
  2030. ###############
  2031. ######### caches ###############
  2032. def voirCacheEnCours(self):
  2033. """centre la vue sur et met en evidence le cache actif"""
  2034. if len(self.caches[self.cacheEnCours]["listeCases"]) > 0:
  2035. #centre la vue sur le premier point du polygone d'une case du milieu de la liste
  2036. indiceMilieu = int((len(self.caches[self.cacheEnCours]["listeCases"])/2))
  2037. self.fenetre.ui.vuePlateau.centerOn(self.cases[self.caches[self.cacheEnCours]["listeCases"][indiceMilieu]].polygon()[0])
  2038. #mettre en surbrillance
  2039. #for coord in self.caches[self.cacheEnCours]["listeCases"]:
  2040. # self.
  2041. def placerCacheEnCours(self):
  2042. """active le mode de mise a jour de la liste des cases cachees par ce cache"""
  2043. self.majMode("cachePlacer")
  2044. self.fenetre.ui.cacheActif.setChecked(True)
  2045. self.majEtatCacheEnCours()
  2046. def majEtatCacheEnCours(self):
  2047. """met a jour l'etat (actif ou non) du cache selon l'etat de la case correspondante"""
  2048. self.caches[self.cacheEnCours]["actif"] = self.fenetre.ui.cacheActif.isChecked()
  2049. self.majAffichageCache(self.cacheEnCours, self.caches[self.cacheEnCours]["actif"])
  2050. def majAffichageCache(self, numCache, afficher = True):
  2051. """met a jour l'affichage des cases selon les parametres enregistres pour le cache"""
  2052. if self.polygonesCaches[numCache] == None:
  2053. self.polygonesCaches[numCache] = Cache(self, numCache)
  2054. self.polygonesCaches[numCache].afficher(afficher, self.caches[numCache]["listeCases"])
  2055. ###############"
  2056. ######### gestion des evenements souris et clavier ###############
  2057. #*ES
  2058. def wheelEvent(self, event):
  2059. """zoom/dezoom avec la molette de la souris"""
  2060. #on zoom/dezoom et on recentre sur la position du curseur
  2061. #super (Plateau, self).wheelEvent(event)
  2062. zoom = 1.00
  2063. if event.delta() > 0:
  2064. if self.nbZoomActuel <= 10:
  2065. self.nbZoomActuel += 1
  2066. zoom = 1.25
  2067. elif event.delta() < 0:
  2068. if self.nbZoomActuel >= -10:
  2069. zoom = 0.8
  2070. self.nbZoomActuel -= 1
  2071. if zoom != 0.00:
  2072. self.fenetre.ui.vuePlateau.scale(zoom, zoom)
  2073. self.fenetre.ui.vuePlateau.centerOn(event.scenePos())
  2074. event.accept() #pour considerer l'evenement comme resolu, sans ca les barres de defilement reagissent aussi
  2075. def mouseMoveEvent(self, event):
  2076. super(Plateau, self).mouseMoveEvent(event)
  2077. if event.buttons() == Qt.LeftButton and self.fenetre.ui.vuePlateau.dragMode() != QGraphicsView.ScrollHandDrag:
  2078. coord = self.coordonneesAuPoint(event.scenePos())
  2079. if coord != None:
  2080. self.caseSurvolClicEnfonce(coord)
  2081. else:
  2082. if self.modeActif == "placementEntreeSortie":
  2083. self.modeParam["entreeSortie"].majProjection(event.scenePos())
  2084. event.ignore()
  2085. def mousePressEvent(self, event):
  2086. super(Plateau, self).mousePressEvent(event)
  2087. if event.button() == 1 and self.fenetre.ui.vuePlateau.dragMode() != QGraphicsView.ScrollHandDrag:
  2088. if self.modeActif == "placementEntreeSortie" and not event.isAccepted():
  2089. self.modeParam["entreeSortie"].positionner()
  2090. self.entreesSorties.append(self.modeParam["entreeSortie"])
  2091. self.modeParam["entreeSortie"] = None
  2092. self.majMode("standard")
  2093. elif self.modeActif == "pionSupprimer":
  2094. for item in self.items(event.scenePos()):
  2095. if item.__class__.__name__ == "PionDecor":
  2096. self.pionDecorSupprimer(item.numero)
  2097. break
  2098. elif item.parentItem().__class__.__name__ == "Pion":
  2099. self.pionSupprimer(item.parentItem().numero)
  2100. break
  2101. else:
  2102. ## if not event.isAccepted():
  2103. ## self.caseCliquee(self.coordonneesAuPoint(event.scenePos()))
  2104. event.ignore()
  2105. elif event.button() == 2:
  2106. if self.pionSelectionne() != None and self.modeCombat != "" and self.modeCombat != "aucun":
  2107. self.majModeCombat("aucun")
  2108. else:
  2109. self.majMode("standard")
  2110. event.accept()
  2111. #event.ignore()
  2112. def mouseReleaseEvent(self, event):
  2113. super(Plateau, self).mouseReleaseEvent(event)
  2114. self.clicGaucheRelache()
  2115. def keyPressEvent(self, event):
  2116. """gestion des evenements clavier"""
  2117. toucheClavier = event.key()
  2118. if self.modeActif == "pionCreation" or self.modeActif == "pionSelectionne" or self.modeActif == "pionDecorSelectionne" or self.modeActif == "pionDecorCreation":
  2119. #pivots de la projection du deplacement
  2120. if toucheClavier == Qt.Key_Right:
  2121. self.majProjectionPosition(False)
  2122. self.modeParam["nbRotations"] += 1
  2123. self.majProjectionPosition(True)
  2124. event.accept()
  2125. elif toucheClavier == Qt.Key_Left:
  2126. self.majProjectionPosition(False)
  2127. self.modeParam["nbRotations"] -= 1
  2128. self.majProjectionPosition(True)
  2129. event.accept()
  2130. elif self.modeActif == "creerEntreeSortie":
  2131. if toucheClavier == Qt.Key_Right: self.modeParam["entreeSortie"].nbRotations += 1
  2132. if toucheClavier == Qt.Key_Left: self.modeParam["entreeSortie"].nbRotations -= 1
  2133. self.modeParam["entreeSortie"].majProjection()
  2134. if self.modeActif[0:7] == "caseMaj":
  2135. if toucheClavier == Qt.Key_Up:
  2136. self.fenetre.ui.epaisseurPinceau.setValue(self.fenetre.ui.epaisseurPinceau.value()+1)
  2137. elif toucheClavier == Qt.Key_Down:
  2138. self.fenetre.ui.epaisseurPinceau.setValue(self.fenetre.ui.epaisseurPinceau.value()-1)
  2139. if toucheClavier == Qt.Key_Delete:
  2140. if self.pionSelectionne() != None:
  2141. num = self.pionSelectionne().numero
  2142. if self.modeActif == "pionSelectionne":
  2143. self.pionSupprimer(num)
  2144. elif self.modeActif == "pionDecorSelectionne":
  2145. self.pionDecorSupprimer(num)
  2146. self.majMode("standard")
  2147. elif self.modeActif == "placementEntreeSortie":
  2148. self.modeParam["entreeSortie"].plateau = None
  2149. self.removeItem(self.modeParam["entreeSortie"])
  2150. self.entreesSorties.remove(self.modeParam["entreeSortie"])
  2151. self.modeParam["entreeSortie"] = None
  2152. self.majMode("standard")
  2153. ################