Bonjour,
Dans le cadre d'un stage en développement de plugins Python pour Qgis 2.14, je suis bloqué sur la gestion de fenêtres multiples dans PyQT.
J'ai utilisé le plugin Builder et QT designer pour dessiner mes interfaces, sans soucis apparent. sans entrer dans les actions qui seront effectuées par le plugin, j'ai un soucis lors de l'appel de ma fenêtre d "ajout d'espèces".
Le premier appel se passe très bien. Mais si je ferme la fenêtre principale de mon Widget et que je la rouvre les ennuis commencent...ma fenêtre de choix despèce va réapparaître normalement lors de l'appel, mais à sa fermeture, une nouvelle instance de celle-ci se relance systématiquement. Si je relance mon plugin, ça sera 3 fois, puis 4, ainsi de suite.
http://hpics.li/d4ff7d2
![]()
Je précise que le plugin est en début de développement, et vous prie de tenir compte uniquement de la question d'ouverture multiples de fenêtres.
En remerciant la potentielle courageuse qui lira ce post :)
Bien à vous
Dans le cadre d'un stage en développement de plugins Python pour Qgis 2.14, je suis bloqué sur la gestion de fenêtres multiples dans PyQT.
J'ai utilisé le plugin Builder et QT designer pour dessiner mes interfaces, sans soucis apparent. sans entrer dans les actions qui seront effectuées par le plugin, j'ai un soucis lors de l'appel de ma fenêtre d "ajout d'espèces".
Le premier appel se passe très bien. Mais si je ferme la fenêtre principale de mon Widget et que je la rouvre les ennuis commencent...ma fenêtre de choix despèce va réapparaître normalement lors de l'appel, mais à sa fermeture, une nouvelle instance de celle-ci se relance systématiquement. Si je relance mon plugin, ça sera 3 fois, puis 4, ainsi de suite.
http://hpics.li/d4ff7d2

Je précise que le plugin est en début de développement, et vous prie de tenir compte uniquement de la question d'ouverture multiples de fenêtres.
Code:
# -*- coding: utf-8 -*-
"""
/***************************************************************************
ModuleExport
A QGIS plugin
export
-------------------
begin : 2016-04-14
git sha : $Format:%H$
copyright : (C) 2016 by ******
email : *********@outlook.com
***************************************************************************/
/***************************************************************************
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
***************************************************************************/
"""
from PyQt4.QtCore import QSettings, QTranslator, qVersion, QCoreApplication
from PyQt4.QtGui import QAction, QIcon, QMessageBox
# Initialize Qt resources from file resources.py
import resources
# Import the code for the dialog
from ModuleExport_dialog import ModuleExportDialog
from DialogSelectFaune import DialogSelectFaune
import os.path
import psycopg2
from functools import partial #Pour pouvoir passer un argument de parametre en signal.
from PyQt4.Qt import QPushButton
class ModuleExport:
"""QGIS Plugin Implementation."""
def __init__(self, iface):
# Save reference to the QGIS interface
self.iface = iface
# initialize plugin directory
self.plugin_dir = os.path.dirname(__file__)
# initialize locale
locale = QSettings().value('locale/userLocale')[0:2]
locale_path = os.path.join(
self.plugin_dir,
'i18n',
'ModuleExport_{}.qm'.format(locale))
if os.path.exists(locale_path):
self.translator = QTranslator()
self.translator.load(locale_path)
if qVersion() > '4.3.3':
QCoreApplication.installTranslator(self.translator)
# Create the dialog (after translation) and keep reference
self.dlg = ModuleExportDialog()
# Declare instance attributes
self.actions = []
self.menu = self.tr(u'&ModuleExport')
# TODO: We are going to let the user set this up in a future iteration
self.toolbar = self.iface.addToolBar(u'ModuleExport')
self.toolbar.setObjectName(u'ModuleExport')
# noinspection PyMethodMayBeStatic
def tr(self, message):
"""Get the translation for a string using Qt translation API.
We implement this ourselves since we do not inherit QObject.
:param message: String for translation.
:type message: str, QString
:returns: Translated version of message.
:rtype: QString
"""
# noinspection PyTypeChecker,PyArgumentList,PyCallByClass
return QCoreApplication.translate('ModuleExport', message)
def add_action(
self,
icon_path,
text,
callback,
enabled_flag=True,
add_to_menu=True,
add_to_toolbar=True,
status_tip=None,
whats_this=None,
parent=None):
"""Add a toolbar icon to the toolbar.
:param icon_path: Path to the icon for this action. Can be a resource
path (e.g. ':/plugins/foo/bar.png') or a normal file system path.
:type icon_path: str
:param text: Text that should be shown in menu items for this action.
:type text: str
:param callback: Function to be called when the action is triggered.
:type callback: function
:param enabled_flag: A flag indicating if the action should be enabled
by default. Defaults to True.
:type enabled_flag: bool
:param add_to_menu: Flag indicating whether the action should also
be added to the menu. Defaults to True.
:type add_to_menu: bool
:param add_to_toolbar: Flag indicating whether the action should also
be added to the toolbar. Defaults to True.
:type add_to_toolbar: bool
:param status_tip: Optional text to show in a popup when mouse pointer
hovers over the action.
:type status_tip: str
:param parent: Parent widget for the new action. Defaults None.
:type parent: QWidget
:param whats_this: Optional text to show in the status bar when the
mouse pointer hovers over the action.
:returns: The action that was created. Note that the action is also
added to self.actions list.
:rtype: QAction
"""
icon = QIcon(self.plugin_dir+"/icon.png")
action = QAction(icon, "Module d'export", parent)
action.triggered.connect(callback)
action.setEnabled(enabled_flag)
if status_tip is not None:
action.setStatusTip(status_tip)
if whats_this is not None:
action.setWhatsThis(whats_this)
if add_to_toolbar:
self.toolbar.addAction(action)
if add_to_menu:
self.iface.addPluginToMenu(
self.menu,
action)
self.actions.append(action)
#Ajout de notre action a la liste d actions.
return action
def initGui(self):
"""Create the menu entries and toolbar icons inside the QGIS GUI."""
icon_path = self.plugin_dir+"/icon.png"
self.add_action(
icon_path,
text=self.tr(u'Module d export'),
callback=self.run,
whats_this = self.tr(u'Module permettant d\'exporter des données Faune / Flore vers une Shapefile ou un CSV.'),
parent=self.iface.mainWindow())
def unload(self):
"""Removes the plugin menu item and icon from QGIS GUI."""
for action in self.actions:
self.iface.removePluginMenu(
self.tr(u'&ModuleExport'),
action)
self.iface.removeToolBarIcon(action)
# remove the toolbar
del self.toolbar
####################################################################################
####################################################################################
############## ###################
############## ###################
############## FONCTIONS DE RECUP DE DONNEES ###################
############## DANS PGSQL ###################
############## ###################
####################################################################################
####################################################################################
def get_table_name(self):
#MODIF LOIC
#Fonction pour obtenir le nom réel de la couche selectionnée. "Schema.table"
table_name = str(self.iface.activeLayer().source())
table_name = table_name.replace(table_name[0:table_name.find('table=')+6], "").replace('"','').split()
return table_name[0]
def get_db_name(self):
#MODIF LOIC
#Fonction pour obtenir le nom de la base de données source
madb = str(self.iface.activeLayer().source())
madb = madb.replace(madb[0:madb.find('dbname=')+7], "").replace('"','').replace("'","").split()
return madb[0]
def get_host_ip(self):
#MODIF LOIC
#Fonction pour obtenir le contenu de Host
hname = str(self.iface.activeLayer().source())
hname = hname.replace(hname[0:hname.find('host=')+5], "").replace('"','').replace("'","").split()
return hname[0]
def get_dbuser_name(self):
#MODIF LOIC
#Fonction pour obtenir le nom du User
usname = str(self.iface.activeLayer().source())
usname = usname.replace(usname[0:usname.find('user=')+5], "").replace('"','').replace("'","").split()
return usname[0]
def get_pswrd(self):
#MODIF LOIC
#Fonction pour obtenir le mot de passe
pwrd = str(self.iface.activeLayer().source())
pwrd = pwrd.replace(pwrd[0:pwrd.find('password=')+9], "").replace('"','').replace("'","").split()
return pwrd[0]
def get_primary_key(self):
#MODIF LOIC
#Fonction pour receperer le nom de la clé primaire
#Attention: Ne fonctionne pas sur des clés primaires multiples.
idconnxion = ("dbname=%s host=%s user=%s password=%s") % (self.get_db_name(), self.get_host_ip(), self.get_dbuser_name(), self.get_pswrd())
conn = psycopg2.connect(idconnxion)
cur = conn.cursor()
marequete = """
SELECT a.attname
FROM pg_index i
JOIN pg_attribute a ON a.attrelid = i.indrelid
AND a.attnum = ANY(i.indkey)
WHERE i.indrelid = '%s'::regclass
AND i.indisprimary;
"""% self.get_table_name()
cur.execute(marequete)
c = cur.fetchall()
conn.close()
return str(c).replace("[", "").replace("]","").replace("(", "").replace(")", "").replace("'", "").replace(",", "")[1:]
# [1:] : lecture de la chaine du 2eme caractere jusquau dernier de la chaine.
def recup_id(self):
Mesobjet = self.iface.activeLayer().selectedFeatures ()
Mesid = ', '.join([unicode(f['id']) for f in Mesobjet ])
return Mesid
####################################################################################
####################################################################################
############## ###################
############## ###################
############## FENETRES DE FAUNE FLORE ###################
############## ###################
############## ###################
####################################################################################
####################################################################################
def fenetre_select_faune(self):
self.dlg_F = DialogSelectFaune()
self.dlg_F.btn_ajoutfaune.setIcon(QIcon(self.plugin_dir+"/btn_add.png"))
self.dlg_F.btn_rmfaune.setIcon(QIcon(self.plugin_dir+"/btn_remove.png"))
#Lacement de la suppression en mettant un parametre de type (flore ou faune : fl/fa)
#Utilisation de /functools/.partial pour ne pas perturber QT et passer un parametre. <> gestion slot/signal
self.dlg_F.vider_faune.clicked.connect(partial(self.vider_la_table, 'fa'))
self.dlg_F.btn_ajoutfaune.clicked.connect(partial(self.ajout_ls_espece, 'fa'))
idconnexion = ("dbname=%s host=%s user=%s password=%s") % (self.get_db_name(), self.get_host_ip(), self.get_dbuser_name(), self.get_pswrd())
conn = psycopg2.connect(idconnexion)
cur = conn.cursor()
marequete = """
select w.taxon, w.cd_nom, w.cd_ref
from bdfauneflore.w_ref_faune w
group by cd_nom, cd_ref, taxon
limit 20"""
cur.execute(marequete)
for i in cur:
self.dlg_F.ls_faune_wref.addItem (cur.fetchone()[0])
marequete2 = """
select *
from export.faune_select
limit 20
"""
cur.execute(marequete2)
for i in cur:
self.dlg_F.ls_faune_selected.addItem (cur.fetchone()[0])
conn.close()
#self.dlg_F.show()
#Si on appuie sur ok ou si l'utilisateur ferme la fenetre alors les listes sont vidées graphiquement.
if self.dlg_F.exec_() or self.dlg_F.close():
#self.dlg_F.ls_faune_selected.clear()
#self.dlg_F.ls_faune_wref.clear()
self.dlg_F.destroy()
####################################################################################
############## FONCTIONS POUR FAUNE FLORE ###################
####################################################################################
#Fonction pour vider les tables de flore select et faune select.
def vider_la_table(self,typesuppr):
#CONTROLE: transformation du parametre en nom de table.
if typesuppr == 'fa':
matable = 'export.faune_select'
elif typesuppr =='fl':
matable = 'export.flore_select'
else:
print 'erreur sur la table à vider.'
idconnexion = ("dbname=%s host=%s user=%s password=%s") % (self.get_db_name(), self.get_host_ip(), self.get_dbuser_name(), self.get_pswrd())
conn = psycopg2.connect(idconnexion)
cur = conn.cursor()
marequete = """
DELETE FROM %s
""" % self.tr(matable)
print marequete
cur.execute(marequete)
#passage du parametre pour taper dans daune ou flore.
conn.commit()
conn.close()
#On vide la boite de dialogue.
self.dlg_F.ls_faune_selected.clear()
def ajout_ls_espece(self,typeadd):
#CONTROLE: transformation du parametre en noms de tables.
if typeadd == 'fa':
matable_select = self.tr('export.faune_select')
ma_wref = self.tr('bdfauneflore.w_ref_faune')
elif typeadd =='fl':
matable_select = self.tr('export.flore_select')
ma_wref = self.tr('bdfauneflore.w_ref_flore')
else:
print 'erreur sur la table à vider.'
#Création d'une liste + guillemets + parentheses pour la passer en parametre dans la requeet SQL.
malisteqt = "', '".join([str(x.text()) for x in self.dlg_F.ls_faune_wref.selectedItems()])
malisteqt = self.tr("('" + malisteqt + "')")
print malisteqt
print type(malisteqt)
idconnexion = ("dbname=%s host=%s user=%s password=%s") % (self.get_db_name(), self.get_host_ip(), self.get_dbuser_name(), self.get_pswrd())
conn = psycopg2.connect(idconnexion)
cur = conn.cursor()
marequete ="""
insert into %s %s, cd_nom, cd_ref %s
select bdf.taxon, cd_nom, cd_ref
from %s bdf
where bdf.taxon in %s
""" % (matable_select, self.tr('(taxon'), self.tr(')'), ma_wref, malisteqt)
print marequete
cur.execute(marequete)
conn.commit()
conn.close
####################################################################################
####################################################################################
############## ###################
############## ###################
############## LANCE-REQUETE ET RUN ###################
############## ###################
############## ###################
####################################################################################
####################################################################################
def lance_requete(self):
idconnxion = ("dbname=%s host=%s user=%s password=%s") % (self.get_db_name(), self.get_host_ip(), self.get_dbuser_name(), self.get_pswrd())
conn = psycopg2.connect(idconnxion)
cur = conn.cursor()
marequete = """
SELECT export.taxons_zoneexport('{%s}'::int[]) ;
""" % self.recup_id()
cur.execute(marequete)
conn.commit()
#DEBUG
#Affichage des log PG
for notice in conn.notices:
print notice
conn.close()
print marequete
def run(self):
# show the dialog
self.dlg.show()
#Bouton pour appeler le formulaire de selection des especes faune.
self.dlg.ButtonFaune.clicked.connect(self.fenetre_select_faune)
# Run the dialog event loop
result = self.dlg.exec_()
# See if OK was pressed
if result:
self.lance_requete()
print self.get_db_name()
print self.get_host_ip()
print self.get_table_name()
print self.get_dbuser_name()
print self.get_pswrd()
En remerciant la potentielle courageuse qui lira ce post :)
Bien à vous