dm.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557
  1. #from __future__ import unicode_literals
  2. # -*- coding: utf-8 -*-
  3. """ensemble des widgets surchargés"""
  4. import os
  5. from PyQt4.QtCore import QString, SIGNAL, QVariant, QSize, pyqtSignal, \
  6. QPersistentModelIndex, Qt, QEvent, QModelIndex
  7. from PyQt4.QtGui import QLabel, QPixmap, QTextEdit, QLineEdit, QGraphicsView, \
  8. QComboBox, QSpinBox, QTabWidget, QTableWidget, QTableWidgetItem, \
  9. QItemSelectionModel, QIcon, QColor, QBrush, QGroupBox, QFrame
  10. from lib.mat import chargerMat
  11. from lib.outilsSvg import afficheSvg
  12. try:
  13. from lib.commun import rep, charger
  14. except:
  15. try:
  16. from commun import rep, charger
  17. except:
  18. print "dm: impossible de charger commun"
  19. try:
  20. import lib.regles as regles
  21. except:
  22. try:
  23. import regles as regles
  24. except:
  25. print "dm: impossible de charger regles"
  26. class Biblio():
  27. """cette classe fournit une liste spreciale"""
  28. def __init__(self):
  29. self.dico = {}
  30. self.ordre = []
  31. def trierPar(self, param):
  32. """trie en fonction du parametre des objets"""
  33. ordre = []
  34. return ordre
  35. def elt(self, index):
  36. return self.dict[index]
  37. class DmLabel(QLabel):
  38. """surcharge de QLabel"""
  39. def __init__(self, parent = None):
  40. super(DmLabel, self).__init__(parent)
  41. self._image = None
  42. def majTexte(self, txt):
  43. self.clear()
  44. self.setText(QString.fromUtf8(txt))
  45. def chargerImage(self, img = None):
  46. """prend une RImage en parametre"""
  47. self.clear()
  48. if img:
  49. self._image = img
  50. pix = img.pix(self.width(), self.height())
  51. else:
  52. pix = QPixmap()
  53. self.setPixmap(pix)
  54. def texte(self):
  55. return str(self.text().toUtf8())
  56. def image(self):
  57. return self._image
  58. def mousePressEvent(self, event):
  59. if event.button() == 1:
  60. self.emit(SIGNAL("clicked()"))
  61. def mouseDoubleClickEvent(self, event):
  62. if event.button() == 1:
  63. self.emit(SIGNAL("doubleClicked()"))
  64. class DmTextEdit(QTextEdit):
  65. def __init__(self, parent = None):
  66. super(DmTextEdit, self).__init__(parent)
  67. def majTexte(self, txt):
  68. self.clear()
  69. self.setText(QString.fromUtf8(str(txt)))
  70. def texte(self):
  71. return str(self.toPlainText().toUtf8())
  72. class DmLineEdit(QLineEdit):
  73. def __init__(self, parent = None):
  74. super(DmLineEdit, self).__init__(parent)
  75. def majTexte(self, txt):
  76. self.clear()
  77. self.setText(QString.fromUtf8(str(txt)))
  78. def texte(self):
  79. return str(self.text().toUtf8())
  80. class DmGraphicsView(QGraphicsView):
  81. """surcharge de QGraphicsView"""
  82. def __init__(self, parent = None):
  83. super(DmGraphicsView, self).__init__(parent)
  84. self.nbZoomActuel = 0
  85. def resizeEvent(self, event):
  86. super(DmGraphicsView, self).resizeEvent(event)
  87. self.emit(SIGNAL("resizeEvent()"))
  88. def wheelEvent(self, event):
  89. """zoom/dezoom avec la molette de la souris"""
  90. #on zoom/dezoom et on recentre sur la position du curseur
  91. zoom = 1.00
  92. if event.delta() > 0:
  93. if self.nbZoomActuel <= 10:
  94. self.nbZoomActuel += 1
  95. zoom = 1.25
  96. elif event.delta() < 0:
  97. if self.nbZoomActuel >= -10:
  98. zoom = 0.8
  99. self.nbZoomActuel -= 1
  100. if zoom != 0.00:
  101. self.scale(zoom, zoom)
  102. self.centerOn(self.mapToScene(event.pos()))
  103. self.emit(SIGNAL("zoomChange(int)"), self.nbZoomActuel)
  104. event.accept() #pour considerer l'evenement comme resolu, sans ca les barres de defilement reagissent aussi
  105. class DmComboBox(QComboBox):
  106. """surcharge de QComboBox"""
  107. def __init__(self, parent = None):
  108. super(DmComboBox, self).__init__(parent)
  109. def allerA(self, donnee):
  110. """cherche la donnee demandee et affiche la ligne"""
  111. index = self.findData(QVariant(donnee))
  112. self.setCurrentIndex(index)
  113. def valeurActuelle(self):
  114. """renvoie sous forme de QVariant la valeur en cours"""
  115. return self.itemData(self.currentIndex())
  116. class DmSpinBox(QSpinBox):
  117. """surcharge de QSpinBox"""
  118. def __init__(self, parent = None):
  119. super(DmSpinBox, self).__init__(parent)
  120. def focusInEvent(self, event):
  121. super(DmSpinBox, self).focusInEvent(event)
  122. self.selectAll()
  123. class DmTabInventaire(QTabWidget):
  124. """surcharge de QTabWidget stylisee et
  125. utilisee pour filtrer les types d'objets affiches"""
  126. def __init__(self, parent = None):
  127. super(DmTabInventaire, self).__init__(parent)
  128. self.setStyleSheet(QString.fromUtf8("""
  129. QTabWidget::pane { border-top: 2px solid #C2C7CB; }
  130. QTabWidget::tab-bar { alignment: right; }
  131. QTabBar::tab {
  132. background: qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1,
  133. stop: 0 #E1E1E1, stop: 0.4 #DDDDDD,
  134. stop: 0.5 #D8D8D8, stop: 1.0 #D3D3D3);
  135. border: 2px solid #C4C4C3;
  136. border-bottom-color: #C2C7CB; border-top-left-radius: 4px;
  137. border-top-right-radius: 8px;
  138. border-top-left-radius: 8px;
  139. padding-left: 15px; }
  140. QTabBar::tab:selected, QTabBar::tab:hover {
  141. background: qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1,
  142. stop: 0 #fafafa, stop: 0.4 #f4f4f4,
  143. stop: 0.5 #e7e7e7, stop: 1.0 #fafafa); }
  144. QTabBar::tab:selected { border-color: #9B9B9B;
  145. border-bottom-color: #C2C7CB; }
  146. """))
  147. class DmTableWidget(QTableWidget):
  148. """surcharge de QTableWidget"""
  149. def __init__(self, parent = None):
  150. super(DmTableWidget, self).__init__(parent)
  151. self.majEnCours = False
  152. def vider(self):
  153. """supprime toutes les lignes"""
  154. self.majEnCours = True
  155. while self.rowCount() > 0:
  156. self.removeRow(0)
  157. self.majEnCours = False
  158. def nouvelleLigneFin(self):
  159. """ajoute une ligne en derniere position"""
  160. pos = self.rowCount()
  161. self.insertRow(pos)
  162. return pos
  163. def lignes(self):
  164. """retourne la liste des index de lignes de la table"""
  165. return range(0, self.rowCount())
  166. def colonnes(self):
  167. """retourne la liste des index de lignes de la table"""
  168. return range(0, self.columnCount())
  169. def masquerColonneId(self):
  170. """masque la colonne en position 0 qui contient les identifiants"""
  171. self.setColumnWidth(0, 0)
  172. def majData(self, ligne, colonne, var):
  173. """met a jour la cellule avec une donnee de type QVariant"""
  174. var = QVariant(var)
  175. if not self.item(ligne, colonne):
  176. item = QTableWidgetItem()
  177. item.setData(0, var)
  178. self.setItem(ligne, colonne, item)
  179. else:
  180. self.item(ligne, colonne).setData(0, var)
  181. return True
  182. def majTexte(self, ligne, colonne, texte):
  183. """met a jour la cellule avec du texte"""
  184. if not self.item(ligne, colonne):
  185. self.setItem(ligne, colonne, QTableWidgetItem(QString.fromUtf8(str(texte))))
  186. else:
  187. self.item(ligne, colonne).setText(QString.fromUtf8(str(texte)))
  188. return True
  189. def majEnt(self, ligne, colonne, valeur):
  190. """met a jour la cellule avec une valeur numerique entiere
  191. si valeur ne peut etre converti en numerique, on retourne Faux"""
  192. retour = False
  193. try:
  194. entier = QVariant(long(valeur))
  195. self.item(ligne, colonne).setData(0, entier)
  196. retour = True
  197. except:
  198. pass
  199. return retour
  200. def chercherLigne(self, colonne, valeur):
  201. """cherche la valeur dans la colonne demandee
  202. renvoie la ligne correspondante """
  203. ligne = None
  204. for i in range(0, self.rowCount()):
  205. if str(self.item(i, colonne).text().toUtf8()) == str(valeur):
  206. ligne = i
  207. break
  208. return ligne
  209. def majLigne(self, ligne, dico, flags = []):
  210. """rempli la ligne avec les donnees trouvees dans le dictionnaire"""
  211. for colonne in range(0, self.columnCount()):
  212. nomChamp = ""
  213. item = self.horizontalHeaderItem(colonne)
  214. if item:
  215. nomChamp = str(item.text())
  216. if nomChamp in dico:
  217. item = QTableWidgetItem()
  218. for f in flags:
  219. item.setFlag(f)
  220. var = QVariant(dico[nomChamp])
  221. item.setData(0, var)
  222. self.setItem(ligne, colonne, item)
  223. def texte(self, ligne, colonne):
  224. return str(self.item(ligne, colonne).text().toUtf8())
  225. def data(self, ligne, colonne):
  226. retour = None
  227. item = self.item(ligne, colonne)
  228. if item:
  229. retour = item.data(0)
  230. return retour
  231. def ligneSelectionnee(self):
  232. """renvoie la ligne selectionnee (la premiere si plusieurs le sont)"""
  233. return self.currentRow()
  234. def selectionner(self, ligne, colonne):
  235. self.setCurrentCell(ligne, colonne, QItemSelectionModel.Select)
  236. class DmTableMat(DmTableWidget):
  237. """table utilisee pour afficher les bibliotheques d'objets:
  238. terrains, decors, creatures"""
  239. def __init__(self, parent = None):
  240. super(DmTableMat, self).__init__(parent)
  241. def charger(self, filtre = ""):
  242. """remplit la table avec les donnees contenues dans les fichiers de materiel
  243. (filtre selon le prefixe)"""
  244. self.setIconSize(QSize(30,20))
  245. self.horizontalHeader().setStretchLastSection(True)
  246. self.masquerColonneId()
  247. self.majEnCours = True
  248. self.setSortingEnabled(False)
  249. for attributsFichier in os.walk(rep("jeu")):
  250. for f in attributsFichier[2]:
  251. if f[:2] == filtre or len(filtre) == 0:
  252. ligne = self.nouvelleLigneFin()
  253. mat = charger(os.path.join(attributsFichier[0], f))
  254. self.majTexte(ligne, 0, mat.idM())
  255. icon = QIcon() if not mat.icone() else QIcon(mat.icone().fichier())
  256. item = QTableWidgetItem(icon, QString.fromUtf8(mat.nom()))
  257. self.setItem(ligne, 1, item)
  258. self.sizeHintForColumn(1)
  259. self.sortItems(1)
  260. self.setSortingEnabled(True)
  261. self.majEnCours = False
  262. def maj(self):
  263. self.vider()
  264. self.remplir()
  265. def selection(self):
  266. """renvoie l'objet actuellement selectionne"""
  267. mat = chargerMat(self.texte(self.currentRow(), 0))
  268. return mat
  269. class DmTableAttributsPi(DmTableWidget):
  270. """table utilisee pour afficher les attributs d'un pion dans le panneau Pi"""
  271. def __init__(self, parent = None):
  272. super(DmTableMat, self).__init__(parent)
  273. self.fichier = ""
  274. self.masquerColonneId()
  275. def defFichier(self, fichier):
  276. """definit le fichier de sauvegarde qui la source des donnees de la table"""
  277. self.fichier = fichier
  278. def initAffichage(self):
  279. """fait les differents reglages relatifs a l'apparence"""
  280. self.setColumnWidth(0, 0)
  281. self.setIconSize(QSize(30,20))
  282. self.horizontalHeader().setStretchLastSection(True)
  283. def remplir(self):
  284. """remplit la table avec les donnees contenues dans le dictionnaire de la savuvegarde"""
  285. dico = afficheSvg(self.fichier)
  286. self.majEnCours = True
  287. self.setSortingEnabled(False)
  288. index = 0
  289. for elt in dico:
  290. objet = dico[elt]
  291. self.insertRow(int(index))
  292. #code de l'objet
  293. self.setItem(int(index),0,QTableWidgetItem(QString.fromUtf8(elt)))
  294. #icone et nom
  295. icon = QIcon("img\\"+objet.icone())
  296. item = QTableWidgetItem(icon,QString.fromUtf8(objet.nom))
  297. self.setItem(int(index),1,item)
  298. index += 1
  299. self.sizeHintForColumn(1)
  300. self.setSortingEnabled(True)
  301. self.sortItems(1)
  302. self.majEnCours = False
  303. def maj(self):
  304. self.vider()
  305. self.remplir()
  306. def actuel(self):
  307. """renvoie l'objet actuellement selectionne"""
  308. objet = None
  309. index = self.item(self.currentRow(), 0)
  310. if index > 0:
  311. objet = charger(self.fichier, str(index.text().toUtf8()))
  312. return objet
  313. class DmTableInventaire(DmTableWidget):
  314. """table utilisee pour afficher l'inventaire d'un combattant"""
  315. itemExited = pyqtSignal(QTableWidgetItem)
  316. def __init__(self, parent = None):
  317. super(DmTableInventaire, self).__init__(parent)
  318. self._inventaire = []
  319. #pour la gestion des survols
  320. self.setMouseTracking(True)
  321. self._last_index = QPersistentModelIndex()
  322. self.viewport().installEventFilter(self)
  323. self.connect(self, SIGNAL("itemEntered(QTableWidgetItem*)"), self.itemSurvole)
  324. self.connect(self, SIGNAL("itemExited(QTableWidgetItem*)"), self.itemSurvoleFin)
  325. self.connect(self, SIGNAL("itemClicked(QTableWidgetItem*)"), self.itemClique)
  326. def construire(self):
  327. self.setColumnWidth(0, 20);self.setColumnWidth(1, 70)
  328. self.setColumnWidth(2, 241);self.setColumnWidth(4, 60)
  329. self.setColumnWidth(6, 20)
  330. def ajouterObjet(self, objet):
  331. ligne = self.rowCount() - 1 #avant la ligne 'ajouter'
  332. self.insertRow(ligne)
  333. self.remplirLigne(ligne, objet)
  334. self._inventaire.append(objet)
  335. def charger(self, inventaire):
  336. """charge l'inventaire en parametre"""
  337. for objet in inventaire:
  338. self.ajouterObjet(objet)
  339. def majObjet(self, ancien, nouveau):
  340. """maj la ligne correspondant a l'objet"""
  341. index = self._inventaire.index(ancien)
  342. self._inventaire[index] = nouveau
  343. self.remplirLigne(index, nouveau)
  344. def supprimerObjet(self, objet):
  345. """ajoute la ligne correspondant a l'objet"""
  346. index = self._inventaire.index(objet)
  347. self._inventaire.remove(objet)
  348. self.removeRow(index)
  349. def remplirLigne(self, ligne, objet):
  350. """remplit les cellules de la ligne avec les donnees de l'objet"""
  351. imgType = ["pieces.png", "btn_ModeCombat.png", "chimie.png", "diamant.png", "boite_outils.png"]
  352. self.creerItem(ligne, 0, "", imgType[objet.typeObjet])
  353. self.creerItem(ligne, 1, objet.quantite)
  354. self.creerItem(ligne, 2, objet.nom)
  355. self.creerItem(ligne, 3, objet.poidsTotal())
  356. self.creerItem(ligne, 4, "")
  357. def inventaire(self):
  358. """cree un inventaire a partir des donnees de la table"""
  359. return self._inventaire
  360. def itemSurvole(self, item):
  361. ligne = item.row()
  362. self.majCouleurLigne(ligne, QColor(150, 50, 50, 150))
  363. def itemSurvoleFin(self, item):
  364. ligne = item.row()
  365. self.majCouleurLigne(ligne, QColor(248, 248, 248, 150))
  366. def itemClique(self, item):
  367. ligne = item.row()
  368. self.emit(SIGNAL("objetClique(int)"), ligne)
  369. def majCouleurLigne(self, ligne, couleur):
  370. """modifie la couleur de fond des items de la lignes"""
  371. for col in self.colonnes():
  372. item = self.item(ligne, col)
  373. if item:
  374. item.setBackground(QBrush(couleur))
  375. def creerItem(self, ligne, colonne, txt, img = None):
  376. item = QTableWidgetItem()
  377. item.setFlags(Qt.NoItemFlags)
  378. item.setText(QString().fromUtf8(" {}".format(txt)))
  379. if img:
  380. pix = QPixmap("img\\{}".format(img))
  381. item.setIcon(QIcon(pix))
  382. self.setItem(ligne, colonne, item)
  383. def filtrer(self, typeObjet):
  384. """filtre l'affichage des lignes par type d'objet"""
  385. for ligne in range(0, self.rowCount() - 1):
  386. self.setRowHidden(ligne, typeObjet >= 0 and typeObjet != self._inventaire[ligne].typeObjet )
  387. def eventFilter(self, widget, event):
  388. if widget is self.viewport():
  389. index = self._last_index
  390. if event.type() == QEvent.MouseMove:
  391. index = self.indexAt(event.pos())
  392. elif event.type() == QEvent.Leave:
  393. index = QModelIndex()
  394. if index != self._last_index:
  395. row = self._last_index.row()
  396. column = self._last_index.column()
  397. item = self.item(row, column)
  398. if item is not None:
  399. self.itemExited.emit(item)
  400. self._last_index = QPersistentModelIndex(index)
  401. return QTableWidget.eventFilter(self, widget, event)
  402. class DmTableInventaireCombat(DmTableInventaire):
  403. """inventaire affiche dans le panneau de combat"""
  404. def __init__(self, parent = None):
  405. super(DmTableInventaireCombat, self).__init__(parent)
  406. class DmTableMenu(QTableWidget):
  407. """table utilisee comme barre d'onglets verticale"""
  408. def __init__(self, parent = None):
  409. super(DmTableMenu, self).__init__(parent)
  410. def setItem(self, ligne, colonne, item):
  411. super(DmTableMenu, self).setItem(ligne, colonne, item)
  412. if ligne == 0:
  413. self.setItemSelected(item, True)
  414. class DmFrameInf_Combattant(QGroupBox):
  415. """frame d'information (combattant)"""
  416. def __init__(self, parent = None):
  417. super(DmFrameInf_Combattant, self).__init__(parent)
  418. def maj(self, combattant):
  419. self.setVisible((combattant != None))
  420. if combattant:
  421. self.findChild(DmLabel, "inf_pionNom").majTexte(combattant.nom())
  422. self.findChild(DmLabel, "inf_pionImage").chargerImage(combattant.img.rimage)
  423. self.findChild(DmLabel, "inf_pionEffet").setVisible(False)
  424. class DmFrameInf_Decor(QGroupBox):
  425. """frame d'information (decor)"""
  426. def __init__(self, parent = None):
  427. super(DmFrameInf_Decor, self).__init__(parent)
  428. def maj(self, decor):
  429. self.setVisible((decor != None))
  430. if decor:
  431. self.findChild(DmLabel, "inf_decorNom").majTexte(decor.nom())
  432. self.findChild(DmLabel, "inf_decorImage").chargerImage(decor.img.rimage)
  433. class DmFrameInf_Case(QGroupBox):
  434. """frame d'information (case)"""
  435. def __init__(self, parent = None):
  436. super(DmFrameInf_Case, self).__init__(parent)
  437. def maj(self, case):
  438. self.setVisible((case != None))
  439. if len(case.terrain.nom()) > 0:
  440. self.findChild(DmLabel, "inf_caseTerrain").majTexte(case.terrain.nom())
  441. else:
  442. self.findChild(DmLabel, "inf_caseTerrain").majTexte("Case")
  443. self.findChild(DmLabel, "inf_caseCoord").majTexte("X: {} Y: {}".format(case.x, case.y))
  444. self.findChild(DmLabel, "inf_caseAltitude").majTexte("Alt.: {}".format(case.altitude))
  445. if case.effetActif != "":
  446. self.findChild(DmLabel, "inf_caseEffet").chargerImage("img\\"+case.imgEffet[case.effetActif])
  447. else:
  448. self.findChild(DmLabel, "inf_caseEffet").clear()
  449. class DmFrame(QFrame):
  450. """surcharge de QFrame"""
  451. def __init__(self, parent = None):
  452. super(DmFrame, self).__init__(parent)