Outils pour utilisateurs

Outils du site


pyqt4_console_python

Intégrer une console Python dans un programme

[Python v2.7.3, PyQt4 v4.9.5, Qt4 v4.8.3]

[modification le 28/1/2013 16h]

[modification le 7/2/2013]

Objectif

On est en train d'utiliser un programme, et on a un petit calcul à faire. Oui, je sais, tous les PC ont une calculatrice, mais le calcul dont on a besoin ne s'y trouve pas forcément: calcul de crédit, calcul en nombres complexes, calcul arithmétique (PGCD, loi de Poisson, …), etc… Selon le domaine dans lequel on est, il suffira d'avoir le module supplémentaire à importer.

Le but de ce projet est de permettre à un programme quelconque Python-PyQt4 d'ouvrir une fenêtre à l'intérieur de laquelle on disposera de l'interpréteur Python, comme la partie interactive de IDLE.

Et ça marchera, même dans un programme autonome traité par cx_freeze, et exécuté sur un PC qui n'a pas Python!

Autre variante: utiliser cette fenêtre lancée par une icone située dans la zone de notification! Il y a une solution sur ce site dans le chapitre concernant PyQt4.

Principe de la solution proposée

On utilise le module “code” qui est fait pour ça.

La fenêtre est toute simple: QMainWindow avec à l'intérieur un QTextEdit. Mais le QTextEdit est particulier, parce qu'il a fallu traiter l'interface avec l'interpréteur, et traiter certains caractères comme la touche “entrée” pour envoyer la dernière ligne tapée à l'interpréteur. J'ai donc sous-classé le QTextEdit pour ça: classe “Visu” dans le code ci-dessous. Cette classe devra aussi lancer le thread qui portera l'interpréteur, et recevoir ses réponses pour les afficher.

L'interpréteur sera porté par un thread. Pourquoi un thread? Parce que sinon, l'interpréteur aurait figé le graphique à chaque exécution et attente.

Mais cela posera un problème particulier: un code s'exécutant dans un thread ne doit pas toucher directement au graphique PyQt4. Les échanges d'information se feront exclusivement par échange de messages. J'ai donc choisi pour ça un thread “QThread”.

La partie interpréteur utilisée est la classe “InteractiveConsole” du module code. Elle est ici sous-classée et certaines méthodes surchargées pour assurer les échanges d'entrée/sortie avec la partie graphique, y compris avec un détournement des sys.stdout et sys.stderr.

Concernant le détournement des sys.stdout et sys.stderr. Ce sont des objets de type “fichier”. On va les détourner sur un “StringIO.StringIO”, pour avoir aussi un type fichier, mais on va rediriger sa méthode write sur la méthode write de l'interpréteur.

Voilà pour les principes. le reste est largement commenté dans le code lui-même.

Ce code fonctionne bien, mais il n'est pas encore parfait: il faut le considérer en version “beta”. Les remarques sont bienvenues!

L'idée initiale de ce code m'a été soufflée sur un forum (http://www.developpez.net/forums/d1301688/autres-langages/python-zope/gui/pyside-pyqt/console-python-qtextedit/): merci à wiztricks!

On peut ajouter des lignes de code dans le fichier consolepy_init.py afin d'initialiser l'interpréteur: ces lignes seront exécutées comme si l'utilisateur les avait tapées au clavier. On peut dans ce fichier importer des modules spécialisés, et donc de préparer la console à traiter des problèmes spécifiques au domaine dans lequel on est (scientifique, financier, technologique, construction, etc…).

Le menu “Aide” de la fenêtre appelle le fichier “consolepy.html” (à créer) qui s'affiche dans le navigateur internet par défaut. On peut l'écrire facilement avec, par exemple, le logiciel gratuit “kompozer”, ce qui permettra d'avoir l'aide en ligne sur les modules spécifiques qu'on aura ajoutés.

Il faudra aussi lui ajouter l'icone 'icone.png' que vous souhaitez.

Voilà ce que ça va donner:

Voilà le code. Il est en Python 2.7, mais il est facile à convertir en Python 3.2 grâce à 2to3.py (+ quelques modifications manuelles supplémentaires).

Code

#!/usr/bin/python
# -*- coding: utf-8 -*-
from __future__ import division
# Python 2.7
 
__programme__ = u"Console Python"
__logiciel__ = "consolepy"
__version__ = "1.00"
__date__ = "01/2013"
 
import sys, os
import code
import webbrowser
from io import BytesIO as StringIO 
from Queue import Queue
 
from PyQt4 import QtCore, QtGui
 
#############################################################################
class Interpy(code.InteractiveConsole):
    """Interpréteur Python"""
 
    #========================================================================
    def __init__(self, tlanceur, locals=None, filename="<console>"): 
        """initialisation"""
        code.InteractiveConsole.__init__(self, locals=None, filename="<console>")
        self.tlanceur = tlanceur # adresse du thread qui a lancé l'interpréteur
 
        # prépa de l'arrêt d'exécution de l'interpréteur sur demande
        self.stopexecflag = False
 
        # rediriger sur le thread lanceur pour gagner du temps d'exécution
        self.write = self.tlanceur.write
        self.read = self.tlanceur.read
        self.quit = self.tlanceur.quit
 
    #========================================================================
    def runcode(self, code):
        """surcharge de runcode pour rediriger sys.stdout et sys.stderr"""
        # redirection de la sortie d'affichage
        self.std_sav = sys.stdout, sys.stderr
        sys.stdout = sys.stderr = sio = StringIO()
        sio.write = self.write #2 rediriger l'écriture du fichier en RAM: sio
 
        # exécution de l'instruction Python compilée
        try:
            sys.settrace(self.trace) # mise en place du traçage
            exec code in self.locals # exécution du code
            sys.settrace(None) # arrêt du traçage
        except SystemExit:
            sys.settrace(None) # arrêt du traçage
            self.quit() # quitter le programme
        except:
            sys.settrace(None) # arrêt du traçage 
            self.showtraceback() # affichage de l'exception rencontrée
        finally:
            # remettre la sortie d'affichage initiale
            sys.stdout, sys.stderr = self.std_sav
            sio.close()
 
    #========================================================================
    def trace(self, frame, event, arg):
        """méthode appelée à chaque ligne de code exécutée par exec"""
        if event == 'line':
            if self.stopexecflag:
                self.stopexecflag = False
                raise  KeyboardInterrupt ("Arret d'execution sur demande")
        return self.trace
 
    #========================================================================
    def stopexec(self):
        """ méthode qu'on appelle pour demander l'arrêt de l'exécution"""
        self.stopexecflag = True
 
    #========================================================================
    def raw_input(self, prompt):
        """lire la prochaine ligne d'instruction Python"""
        return self.read(prompt)
 
    #========================================================================
    def write(self, data):
        """(redirigé) affiche data par l'intermédiaire du thread"""
        pass
 
    #========================================================================
    def read(self, prompt):
        """(redirigé) lit la chaine à interpréter par l'intermédiaire du thread"""
        pass
 
    #========================================================================
    def quit(self):
        """(redirigé) ferme l'application par l'intermédiaire du thread"""
        pass
 
#############################################################################
class Tipy(QtCore.QThread):
    """thread Qt qui porte l'interpréteur Python"""
 
    finconsole = QtCore.pyqtSignal()
    pourafficher = QtCore.pyqtSignal(unicode) 
 
    #========================================================================
    def __init__(self, parent=None): 
        super(Tipy, self).__init__(parent)
        # lancement de l'interpréteur Python (argument: l'adresse du thread)
        self.interpy = Interpy(self)
        # initialisation de la pile qui contiendra l'instruction à exécuter
        self.instruction = Queue(maxsize=1)
        # initialisation du drapeau pour synchroniser l'affichage du QTextEdit
        self.okwrite = False
        self.mutexokwrite = QtCore.QMutex()
 
    #========================================================================
    def run(self):
        """partie exécutée en asynchrone: la boucle de l'interpréteur Python"""
        self.interpy.interact()
 
    #========================================================================
    def write(self, data):
        """affiche data en envoyant un message à la fenêtre"""
        # pour être sûr que data est en unicode
        if not isinstance(data, unicode): 
            data = unicode(data)
        # débarrasse data des éventuels fins de ligne des 2 côtés
        while data!=u"" and data[0] in ['\n', '\r']: data = data[1:]
        while data!=u"" and data[-1] in ['\n', '\r']: data = data[:-1]
        if data!=u"":
            # envoie la réponse sous forme de message (avec shake-hand)
            self.mutexokwrite.lock()
            self.okwrite = False
            self.mutexokwrite.unlock()
            # envoie data avec le signal 'pourafficher'
            self.pourafficher.emit(data)
            # attend jusqu'à ce que le message soit effectivement affiché
            while not self.okwrite:
                pass 
 
    #========================================================================
    def read(self, prompt):
        """lit l'instruction Python à interpréter"""
        # envoi l'invite pour affichage
        self.write(prompt)
        # prend l'instruction dans la pile dès qu'il y en a une
        return self.instruction.get()
 
    #========================================================================
    def stop(self):
        """appelé quand on veut arrêter l'exécution de l'interpréteur"""
        self.interpy.stopexec()
 
    #========================================================================
    def quit(self):
        """méthode appelée quand on veut fermer l'application"""
        # arrête l'interpréteur s'il est en cours d'exécution
        self.stop()
        # émet le signal de fin pour la fenêtre
        self.finconsole.emit()
 
#############################################################################
class Visu(QtGui.QTextEdit):
    """sous-classement de QTextEdit pour communiquer avec l'interpréteur
       Python via un QThread
    """
 
    # signal pour recevoir du texte à afficher
    pourafficher = QtCore.pyqtSignal(unicode) 
    # signal pour recevoir une demande d'arrêt
    finconsole = QtCore.pyqtSignal()
    # signal pour émettre une demande de fermeture de la fenêtre
    finfenetre = QtCore.pyqtSignal()
 
    #========================================================================
    def __init__(self, parent=None, initpy=u"consolepy_init.py"):
        super(Visu, self).__init__(parent)
        # stockage de l'argument passé
        self.initpy = initpy
 
        # prépa de l'initialisation de l'interpréteur si demandé au lancement
        if isinstance(self.initpy, (str, unicode)):
            # si c'est une chaine: elle doit représenter un fichier à charger
            nfc = self.initpy
            self.initpy = []
            if os.path.exists(nfc):
                with open(nfc, 'r') as f:
                    for ligne in f:
                        self.initpy.append(ligne.rstrip())
        else:
            if not isinstance(self.initpy, (list, tuple)):
                # mauvais type pour self.initpy: on n'en tient pas compte
                self.initpy = []
        self.lginitpy = len(self.initpy)
 
        # configuration du QTextEdit
        self.setAcceptRichText(False)
        self.setLineWrapMode(QtGui.QTextEdit.NoWrap) 
        # Changer la police de caractères et sa taille
        font = QtGui.QFont()
        font.setFamily(u"DejaVu Sans Mono")
        font.setPointSize(10)
        self.setFont(font)
 
        # lancement du thread qui porte l'interpréteur Python
        self.tipy = Tipy()
        # prépa pour recevoir de tipy un signal de fermeture de la fenêtre 
        self.tipy.finconsole.connect(self.quitter)
        # prépa pour recevoir de tipy du texte à afficher 
        self.tipy.pourafficher.connect(self.affiche)
        # démarrage du thread
        self.tipy.start()
 
        # initialisation de la position courante du curseur après invite
        self.pos1 = self.textCursor().position()
        # portera la position du curseur à la 1ère invite
        self.pos0 = self.pos1
 
        # initialisation de l'historique des lignes d'instruction Python
        self.historique = []
        self.ih = 0
 
        # compteur du nombre d'invites affichées  
        self.nbinvites = -1 
 
    #========================================================================
    @QtCore.pyqtSlot(unicode)
    def affiche(self, texte):
        """Affiche la chaine 'texte' dans le widget QTextEdit"""
 
        # ajoute la chaine unicode à la fin du QTextEdit
        self.append(texte)
        # déplace le curseur à la fin du texte
        self.moveCursor(QtGui.QTextCursor.End, QtGui.QTextCursor.MoveAnchor)
        # force le rafraichissement pour affichage en temps réel
        QtCore.QCoreApplication.processEvents() 
        # met à jour la position du curseur dans le texte
        self.pos1 = self.textCursor().position()
 
        # renvoie un accusé de réception de fin d'affichage (shake-hand)
        self.tipy.mutexokwrite.lock()
        self.tipy.okwrite = True
        self.tipy.mutexokwrite.unlock()
 
        # envoie les lignes de code d'initialisation après la 1ère invite
        if self.nbinvites < self.lginitpy:
            if self.nbinvites==0:
                # 1ère invite
                self.pos0 = self.textCursor().position() 
            if self.nbinvites>=0:
                # il y a encore des lignes d'instruction initiales à exécuter
                self.tipy.instruction.put(self.initpy[self.nbinvites])
            self.nbinvites += 1
 
    # =======================================================================
    def keyPressEvent(self, event):
        """traitement des évènements clavier du QTextEdit"""
 
        #--------------------------------------------------------------------
        if event.key() in [QtCore.Qt.Key_Return, QtCore.Qt.Key_Enter]:
            # touche "retour"
            # déplacer le curseur à la fin du texte si ce n'est pas le cas
            self.moveCursor(QtGui.QTextCursor.End, QtGui.QTextCursor.MoveAnchor)
            # texte situé entre la précédente position du curseur et l'actuelle
            pos1 = self.pos1
            pos2 = self.textCursor().position()
            texte = unicode(self.toPlainText())[pos1:pos2]
            # débarrasse l'instruction 'texte' des éventuels fins de ligne
            while texte!=u"" and texte[0] in ['\n', '\r']: texte = texte[1:]
            texte = texte.rstrip() # retire à droite fin de ligne et espaces
            # empile l'instruction pour qu'elle soit exécutée
            self.tipy.instruction.put(texte)
            # conserver la ligne d'instruction dans l'historique
            if texte != u"":
                self.historique.append(texte)
                self.ih = len(self.historique)-1 # pointe sur le dernier élément
            # sauvegarde la position du curseur de début d'instruction
            self.pos1 = pos2
            event.accept()
 
        #--------------------------------------------------------------------
        elif event.key()==QtCore.Qt.Key_Z and (event.modifiers() & QtCore.Qt.ControlModifier):
            # Ctrl-Z: annule ce qui vient d'être tapé
            event.accept()
            # exécute le Ctrl-Z normal du QTextEdit
            QtGui.QTextEdit.keyPressEvent(self, event)
            # déplace le curseur à la fin du texte
            self.moveCursor(QtGui.QTextCursor.End, QtGui.QTextCursor.MoveAnchor)
            # force le rafraichissement pour affichage en temps réel
            QtCore.QCoreApplication.processEvents() 
            # met à jour la position du curseur dans le texte
            self.pos1 = self.textCursor().position()
 
        #--------------------------------------------------------------------
        elif event.key()==QtCore.Qt.Key_S and \
                         (event.modifiers() & QtCore.Qt.ControlModifier):
            # Ctrl-S arrête l'interpréteur
            self.tipy.stop()            
            event.accept()
 
        #--------------------------------------------------------------------
        elif event.key()==QtCore.Qt.Key_E and \
                         (event.modifiers() & QtCore.Qt.AltModifier):
            # Alt-E remet l'affichage au début à l'emplacement de la 1ère invite
            texte = unicode(self.toPlainText())
            texte = texte[:self.pos0]
            self.clear()
            self.affiche(texte)
            self.pos1 = self.pos0
            event.accept()
 
        #--------------------------------------------------------------------
        elif event.key()==QtCore.Qt.Key_Up:
            # traitement touche flèche en haut: instruction précédente
            if self.historique!=[] and self.ih>=0:
                pos1 = self.pos1
                pos2 = self.textCursor().position()
                instruction = self.historique[self.ih]
                texte = unicode(self.toPlainText())
                texte = texte[:pos1] + instruction + texte[pos2:]
                self.clear()
                self.affiche(texte)
                self.pos1 = pos1
                if self.ih>0:
                    self.ih -= 1
            event.accept()
 
        #--------------------------------------------------------------------
        elif event.key()==QtCore.Qt.Key_Down:
            # traitement touche flèche en bas, instruction suivante
            if self.historique!=[] and self.ih<len(self.historique)-1:
                self.ih += 1
                pos1 = self.pos1
                pos2 = self.textCursor().position()
                instruction = self.historique[self.ih]
                texte = unicode(self.toPlainText())
                texte = texte[:pos1] + instruction + texte[pos2:]
                self.clear()
                self.affiche(texte)
                self.pos1 = pos1
            event.accept()
 
        #--------------------------------------------------------------------
        elif event.key()==QtCore.Qt.Key_Backspace:
            # empêche le backspace de revenir sur l'invite
            pos = self.textCursor().position()
            if pos>self.pos1:
                event.ignore()
                # évènement transmis à l'ancêtre
                QtGui.QTextEdit.keyPressEvent(self, event)
            else:
                event.accept()    
 
        #--------------------------------------------------------------------
        elif event.key()==QtCore.Qt.Key_Tab:
            """Tabulation: insérer 4 espaces à l'emplacement du curseur"""
            self.insertPlainText(u"    ")
 
        #--------------------------------------------------------------------
        elif event.key() in [QtCore.Qt.Key_Up, 
                             QtCore.Qt.Key_Down,
                             QtCore.Qt.Key_PageUp, 
                             QtCore.Qt.Key_PageDown]:
            """neutralisation de touches de QTextEdit inutile pour 
               l'interpréteur
            """
            event.accept() 
 
        #--------------------------------------------------------------------
        else:
            # n'importe quel autre caractère que ceux ci-dessus
            event.ignore()
            # évènement transmis à l'ancêtre QTextEdit
            QtGui.QTextEdit.keyPressEvent(self, event)
 
    #========================================================================
    @QtCore.pyqtSlot()
    def quitter(self):
        """la fenêtre a reçu le signal de fermeture de l'interpréteur """
        # on ré-émet le signal de fermeture pour la fenêtre
        #self.emit(QtCore.SIGNAL("finfenetre()"))
        self.finfenetre.emit()
 
#############################################################################
class Consolepy(QtGui.QMainWindow):
 
    #========================================================================
    def __init__(self, initpy=u"", parent=None):
        super(Consolepy, self).__init__(parent)
 
        # instructions pour la fenêtre 
        self.setWindowTitle(__programme__)
        self.resize(700, 700) # definit la taille de la fenetre
        icone = QtGui.QIcon(u'icone.png')
        self.setWindowIcon(icone)
 
        # créer le QTextEdit personnalisé
        self.visu = Visu(self, initpy)
 
        self.setCentralWidget(QtGui.QFrame())
        posit = QtGui.QGridLayout()
        posit.addWidget(self.visu, 0, 0)
        self.centralWidget().setLayout(posit)
 
        #--------------------------------------------------------------------
        # créer le menu principal de la fenêtre
        menubar = self.menuBar()
 
        aideMenu = menubar.addMenu('Aide')
 
        aideAction = QtGui.QAction('&Manuel', self)
        aideAction.setShortcut(QtCore.Qt.Key_F1)        
        aideAction.setStatusTip(u"Manuel du programme")
        aideAction.triggered.connect(self.manuel_m)
        aideMenu.addAction(aideAction)
 
        aproposAction = QtGui.QAction('&A propos', self)
        aproposAction.setStatusTip(u"A propos du programme")
        aproposAction.triggered.connect(self.apropos)
        aideMenu.addAction(aproposAction)
 
        #--------------------------------------------------------------------
        # initialisation de la barre de status (affiche le nom de page html)
        self.statusBar().showMessage(u"")
 
        # se préparer à recevoir de self.visu un signal de fermeture
        self.connect(self.visu, QtCore.SIGNAL("finfenetre()"), self.quitter)
 
        # mettre le focus sur le QTextEdit
        self.visu.setFocus()
 
    #========================================================================
    def manuel_m(self):
        # affiche le manuel dans le navigateur web par défaut
        manuel = os.path.abspath("consolepy.html")
        if os.path.exists(manuel):
            webbrowser.open(manuel)
 
    #========================================================================
    def apropos(self):
        """Fenêtre 'à propos' """
        pf = sys.platform
        if pf=='win32': pf='Windows'
        elif pf=='linux2': pf = 'Linux'
        elif pf=='sunos5': pf='Sun OS'
        else: pass
        QtGui.QMessageBox.about(self, u"A propos du logiciel",
        u"""%s version %s  (%s)
    Copyright Jean-Paul Vidal 2013
    Licence GPL3
    Source sur demande ici: http://www.jpvweb.com
 
    Contexte d'exécution en cours:
    Système d'exploitation: %s
    Python version: %s
    Qt version: %s
    PyQt4 version: %s""" % (
            __logiciel__,
            __version__,
            __date__,
            pf,
            sys.version, 
            QtCore.QT_VERSION_STR, 
            QtCore.PYQT_VERSION_STR)
            )
 
    #========================================================================
    def quitter(self):
        """fermeture de la fenêtre demandée par l'interpréteur (par signal)"""
        self.close()
 
    #========================================================================
    def closeEvent(self, event):
        """ferme la fenêtre, même avec la croix ou le menu système"""
        event.accept()
 
#############################################################################
if __name__ == '__main__':
 
    #========================================================================
    app = QtGui.QApplication(sys.argv)
    # définition du style
    app.setStyle(QtGui.QStyleFactory.create(u"Plastique"))
    app.setPalette(QtGui.QApplication.style().standardPalette())
 
    #========================================================================
    # pour assurer la traduction automatique du conversationnel à la locale
    locale = QtCore.QLocale.system().name()
    translator = QtCore.QTranslator ()
 
    if os.path.splitext(sys.argv[0])[1] in ['.py', '.pyw']:
        # exécution par l'interpréteur normal
        reptrad = unicode(QtCore.QLibraryInfo.location(QtCore.QLibraryInfo.TranslationsPath))
    else:
        # exécution de la version exécutable après cx_freeze (ou équivalent): 
        #=>les fichiers de traduction doivent se trouver dans "translations"
        reptrad = unicode("translations")
 
    translator.load(QtCore.QString("qt_") + locale, reptrad)
    app.installTranslator(translator)
 
    #========================================================================
    # paramétrage: instructions Python à exécuter par la console au lancement
    """
    initpy = [u"# -*- coding: utf-8 -*-",
              u"from __future__ import division",
              u"from math import *"]
    """
    initpy = "consolepy_init.py"
 
    #========================================================================
    fen = Consolepy(initpy)
    fen.show()
 
    #========================================================================
    sys.exit(app.exec_())


Amusez-vous bien!

pyqt4_console_python.txt · Dernière modification: 2013/02/07 13:11 de tyrtamos

Outils de la page