Quantcast
Channel: Forum SIG - Systèmes d'Information Géographique et Géomatique - 87,88
Viewing all articles
Browse latest Browse all 3118

[QGIS 2.x] Plugin en PyQT et fenêtres multiples

$
0
0
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 d’espè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

Viewing all articles
Browse latest Browse all 3118

Trending Articles