|  | *setSpacing*: le due precedenti, combinate. |  | *setSpacing*: le due precedenti, combinate. | 
            |  | *setColumnMinimumWidth*: larghezza minima della co |  | *setColumnMinimumWidth*: larghezza minima della co | 
            |  | lonna specificata come primo parametro. |  | lonna specificata come primo parametro. | 
            |  | *setRowMinimumWidth*: stessa cosa per una riga. |  | *setRowMinimumWidth*: stessa cosa per una riga. | 
            |  | *int columnCount*: ritorna il numero di colonne de |  | *int columnCount*: ritorna il numero di colonne de | 
            |  | lla griglia. |  | lla griglia. | 
            |  | *int rowCount*: ritorna il numero di righe. |  | *int rowCount*: ritorna il numero di righe. | 
            | t |  | t |  | 
            |  |  |  | Menu | 
            |  |  |  | ##### | 
            |  |  |  |  | 
            |  |  |  | Andiamo a definire un semplice menu.:: | 
            |  |  |  |  | 
            |  |  |  | #!/usr/bin/python | 
            |  |  |  |  | 
            |  |  |  | import sys | 
            |  |  |  | from PyQt4 import QtGui, QtCore | 
            |  |  |  |  | 
            |  |  |  | class MainWindow(QtGui.QMainWindow): | 
            |  |  |  |  | 
            |  |  |  | def mySlot(self): | 
            |  |  |  | print "Menu voice has been hovered | 
            |  |  |  | " | 
            |  |  |  |  | 
            |  |  |  | def __init__(self): | 
            |  |  |  | QtGui.QMainWindow.__init__(self) | 
            |  |  |  | self.setWindowTitle('Menu') | 
            |  |  |  |  | 
            |  |  |  | quit = QtGui.QAction(QtGui.QIcon(" | 
            |  |  |  | icons/cancel.png"), "Quit", self) | 
            |  |  |  | quit.setShortcut("Ctrl+Q") | 
            |  |  |  | quit.setStatusTip("Quit applicatio | 
            |  |  |  | n") | 
            |  |  |  | self.connect(quit, QtCore.SIGNAL(' | 
            |  |  |  | triggered()'), QtCore.SLOT('close()')) | 
            |  |  |  |  | 
            |  |  |  | sep = QtGui.QAction(self) | 
            |  |  |  | sep.setSeparator(True) | 
            |  |  |  |  | 
            |  |  |  | info = QtGui.QAction(QtGui.QIcon(" | 
            |  |  |  | icons/information.png"), "Information", self) | 
            |  |  |  | info.setShortcut("Ctrl+I") | 
            |  |  |  | info.setStatusTip("Show informatio | 
            |  |  |  | n") | 
            |  |  |  | self.connect(info, QtCore.SIGNAL(' | 
            |  |  |  | hovered()'), self.mySlot) | 
            |  |  |  |  | 
            |  |  |  | self.statusBar().show() | 
            |  |  |  |  | 
            |  |  |  | menuB = self.menuBar() | 
            |  |  |  | file = menuB.addMenu('&File') | 
            |  |  |  | file.addAction(quit) | 
            |  |  |  | file.addAction(sep) | 
            |  |  |  | file.addAction(info) | 
            |  |  |  |  | 
            |  |  |  | app = QtGui.QApplication(sys.argv) | 
            |  |  |  | main = MainWindow() | 
            |  |  |  | main.show() | 
            |  |  |  | sys.exit(app.exec_()) | 
            |  |  |  |  | 
            |  |  |  | .. img:: 12 | 
            |  |  |  | :nolink: | 
            |  |  |  |  | 
            |  |  |  | In ordine di “importanza”, i passi sono: | 
            |  |  |  |  | 
            |  |  |  | * creare una menuBar (la nostra menuB) | 
            |  |  |  | * aggiungere tutti i menu che vogliamo creare con | 
            |  |  |  | il metodo *addMenu()*. Questo metodo prende come a | 
            |  |  |  | rgomento una stringa. Il carattere preceduto dal s | 
            |  |  |  | imbolo & funge da scorciatoia: premendo Alt+caratt | 
            |  |  |  | ere attiveremo il menu corrispondente. | 
            |  |  |  | * Ad ogni menu vanno poi aggiunge le singole *QAct | 
            |  |  |  | ion*, ovvero le voci di menu.:: | 
            |  |  |  |  | 
            |  |  |  | quit = QtGui.QAction(QtGui.QIcon(" | 
            |  |  |  | icons/cancel.png"), "Quit", self) | 
            |  |  |  | quit.setShortcut("Ctrl+Q") | 
            |  |  |  | quit.setStatusTip("Quit applicatio | 
            |  |  |  | n") | 
            |  |  |  | self.connect(quit, QtCore.SIGNAL(' | 
            |  |  |  | triggered()'), QtCore.SLOT('close()')) | 
            |  |  |  |  | 
            |  |  |  | Il costruttore di *QAction* permette di specificar | 
            |  |  |  | e subito il path dell'icona e il testo della voce | 
            |  |  |  | di menu, oppure di ometterlo e di indicarlo dopo c | 
            |  |  |  | on i metodi appositi::: | 
            |  |  |  |  | 
            |  |  |  | quit = QtGui.QAction(self) | 
            |  |  |  | quit.setText(“Quit”) | 
            |  |  |  | quit.setIcon(QtGui.QIcon(“icons/qu | 
            |  |  |  | it.png”) | 
            |  |  |  |  | 
            |  |  |  | Questi ultimi due codici danno lo stesso identico | 
            |  |  |  | risultato. | 
            |  |  |  | Il metodo *setText()* merita particolare attenzion | 
            |  |  |  | e: infatti esso si può trovare in diversi widget c | 
            |  |  |  | he comprendono un testo (pulsanti, etichette, etc. | 
            |  |  |  | ..) per modificare il testo che accompagna il widg | 
            |  |  |  | et stesso. Il corrispondente che invece ritorna il | 
            |  |  |  | testo corrente sotto forma di *QString* è semplic | 
            |  |  |  | emente *text()*. | 
            |  |  |  | Eccezione: per le *textEdit*, che possono contener | 
            |  |  |  | e non solo testo semplice ma anche HTML, invece di | 
            |  |  |  | *text()* si ha *toPlainText()*. | 
            |  |  |  | Per ritornare la shortcut o lo statustip correnti | 
            |  |  |  | potete usare i metodi *statusTip()* e *shortcut()* | 
            |  |  |  | . | 
            |  |  |  | Il secondo oggetto non rappresenta una voce di men | 
            |  |  |  | u, ma semplicemente un separatore: per questo nel | 
            |  |  |  | costruttore non passiamo nessun argomento. | 
            |  |  |  |  | 
            |  |  |  | Nel codice precedente sono mostrati due segnali ti | 
            |  |  |  | pici dei menu: *triggered()* viene emesso quando l | 
            |  |  |  | 'utente clicca su una voce di menu. ATTENZIONE: no | 
            |  |  |  | n viene dunque emesso il segnale *clicked()* come | 
            |  |  |  | si potrebbe pensare! Invece *hovered()* è emesso q | 
            |  |  |  | uando il mouse passa sulla voce di menu. | 
            |  |  |  | Ultima nota: per le voci di menu è presente anche | 
            |  |  |  | il metodo *setCheckable()* (e *setChecked()*, per | 
            |  |  |  | specificare l'azione iniziale), identico a quello | 
            |  |  |  | usato per i toggle button in precedenza: se viene | 
            |  |  |  | passato il parametro True, l'icona della voce si c | 
            |  |  |  | omporterà proprio come un toggle button. | 
            |  |  |  |  | 
            |  |  |  | Toolbar | 
            |  |  |  | ######## | 
            |  |  |  |  | 
            |  |  |  | Una toolbar è un insieme di pulsanti che assolvono | 
            |  |  |  | ad alcune funzioni di base. In alcuni casi può es | 
            |  |  |  | sere più comoda di un menu perché permette all'ute | 
            |  |  |  | nte di scegliere l'azione più velocemente.:: | 
            |  |  |  |  | 
            |  |  |  | #!/usr/bin/python | 
            |  |  |  |  | 
            |  |  |  | import sys | 
            |  |  |  | from PyQt4 import QtGui, QtCore | 
            |  |  |  |  | 
            |  |  |  | class MainWindow(QtGui.QMainWindow): | 
            |  |  |  |  | 
            |  |  |  | def __init__(self): | 
            |  |  |  | QtGui.QMainWindow.__init__(self) | 
            |  |  |  |  | 
            |  |  |  | self.setWindowTitle('Toolbar') | 
            |  |  |  |  | 
            |  |  |  | quit = QtGui.QAction(QtGui.QIcon(" | 
            |  |  |  | icons/cancel.png"), "Quit", self) | 
            |  |  |  | quit.setShortcut("Ctrl+Q") | 
            |  |  |  | quit.setStatusTip("Quit applicatio | 
            |  |  |  | n") | 
            |  |  |  | self.connect(quit, QtCore.SIGNAL(' | 
            |  |  |  | triggered()'), QtCore.SLOT('close()')) | 
            |  |  |  |  | 
            |  |  |  | sep = QtGui.QAction(self) | 
            |  |  |  | sep.setSeparator(True) | 
            |  |  |  |  | 
            |  |  |  | info = QtGui.QAction(QtGui.QIcon(" | 
            |  |  |  | icons/information.png"), "Info", self) | 
            |  |  |  | info.setShortcut("Ctrl+I") | 
            |  |  |  | info.setStatusTip("Show informatio | 
            |  |  |  | n") | 
            |  |  |  |  | 
            |  |  |  | self.statusBar().show() | 
            |  |  |  |  | 
            |  |  |  | toolbar = self.addToolBar('My tool | 
            |  |  |  | ') | 
            |  |  |  | toolbar.addAction(quit) | 
            |  |  |  | toolbar.addAction(info) | 
            |  |  |  | toolbar.setToolButtonStyle(QtCore. | 
            |  |  |  | Qt.ToolButtonTextUnderIcon) | 
            |  |  |  |  | 
            |  |  |  | app = QtGui.QApplication(sys.argv) | 
            |  |  |  | main = MainWindow() | 
            |  |  |  | main.show() | 
            |  |  |  | sys.exit(app.exec_()) | 
            |  |  |  |  | 
            |  |  |  | .. img:: 15 | 
            |  |  |  | :nolink: | 
            |  |  |  |  | 
            |  |  |  | Come è possibile notare, il codice non è troppo di | 
            |  |  |  | verso da quello usato per i menu. Le varie “action | 
            |  |  |  | ” si dichiarano sempre allo stesso modo, ma alla f | 
            |  |  |  | ine invece di dichiarare barre e menu, ci limitiam | 
            |  |  |  | o ad una toolbar. | 
            |  |  |  | Il metodo *setToolButtonStyle()* permette di speci | 
            |  |  |  | ficare la presenza del testo e nel caso, la sua po | 
            |  |  |  | sizione. Gli argomenti che possiamo passargli (olt | 
            |  |  |  | re a quello indicato, che posiziona il testo sotto | 
            |  |  |  | l'icona) sono: *Qt.ToolButtonTextBesideIcon* (tes | 
            |  |  |  | to accanto all'icona), *Qt.ToolButtonTextOnly* (so | 
            |  |  |  | lo testo), *Qt.ToolButtonIconOnly* (solo l'icona, | 
            |  |  |  | l'azione predefinita). | 
            |  |  |  | Altro metodo è *setOrientation()*: accetta come ar | 
            |  |  |  | gomenti *Qt.Horizontal* o *Qt.Vertical*. | 
            |  |  |  |  | 
            |  |  |  | Dialogs | 
            |  |  |  | ######## | 
            |  |  |  |  | 
            |  |  |  | Le dialog window sono “popup”, piccole finestrelle | 
            |  |  |  | che possono essere dipendenti o meno da quella pr | 
            |  |  |  | incipale. Vengono utilizzate per visualizzare brev | 
            |  |  |  | i messaggi o per prendere input. | 
            |  |  |  | PyQt4 mette a disposizione diversi dialogs standar | 
            |  |  |  | d per le operazioni più comuni: vediamo in un unic | 
            |  |  |  | o codice come si dichiarano, e come vengono ritorn | 
            |  |  |  | ate le informazioni.:: | 
            |  |  |  |  | 
            |  |  |  | #!/usr/bin/python | 
            |  |  |  |  | 
            |  |  |  | import sys | 
            |  |  |  | from PyQt4 import QtGui, QtCore | 
            |  |  |  |  | 
            |  |  |  | class MainWindow(QtGui.QMainWindow): | 
            |  |  |  | def __init__(self): | 
            |  |  |  | QtGui.QMainWindow.__init__(self) | 
            |  |  |  | buttons = [0] * 13 | 
            |  |  |  | self.titles = ["Get integer", "Get | 
            |  |  |  | double", "Get item", "Get text", "Set color", "Se | 
            |  |  |  | t font", "Set directory", "Open file", "Save File" | 
            |  |  |  | , "Critical message", "Info message", "Question", | 
            |  |  |  | "Warning"] | 
            |  |  |  | slots = [self.getInt, self.getDoub | 
            |  |  |  | le, self.getItem, self.getText, self.getColor, sel | 
            |  |  |  | f.getFont, self.getDirectory, self.openFile, self. | 
            |  |  |  | saveFile, self.Critical, self.Info, self.Question, | 
            |  |  |  | self.Warning] | 
            |  |  |  |  | 
            |  |  |  | self.resize(350, 250) | 
            |  |  |  | self.setWindowTitle('Dialogs') | 
            |  |  |  |  | 
            |  |  |  | widget = QtGui.QWidget(self) | 
            |  |  |  |  | 
            |  |  |  | grid = QtGui.QGridLayout(widget) | 
            |  |  |  | grid.setVerticalSpacing(10) | 
            |  |  |  | grid.setHorizontalSpacing(8) | 
            |  |  |  |  | 
            |  |  |  | row = 0 | 
            |  |  |  | col = 0 | 
            |  |  |  |  | 
            |  |  |  | for i in range(13): | 
            |  |  |  | buttons[i] = QtGui.QPushBu | 
            |  |  |  | tton(self.titles[i], widget) | 
            |  |  |  | self.connect(buttons[i], Q | 
            |  |  |  | tCore.SIGNAL('clicked()'), slots[i]) | 
            |  |  |  |  | 
            |  |  |  | grid.addWidget(buttons[i], | 
            |  |  |  | row, col) | 
            |  |  |  |  | 
            |  |  |  | if col == 2: | 
            |  |  |  | col = 0 | 
            |  |  |  | row += 1 | 
            |  |  |  | else: | 
            |  |  |  | col += 1 | 
            |  |  |  |  | 
            |  |  |  | self.textEdit = QtGui.QTextEdit(wi | 
            |  |  |  | dget) | 
            |  |  |  | self.textEdit.setReadOnly(True) | 
            |  |  |  | grid.addWidget(self.textEdit, 5, 0 | 
            |  |  |  | , 1, 3) | 
            |  |  |  |  | 
            |  |  |  | self.setLayout(grid) | 
            |  |  |  | self.setCentralWidget(widget) | 
            |  |  |  |  | 
            |  |  |  | def getInt(self): | 
            |  |  |  | integer, ok = QtGui.QInputDialog.g | 
            |  |  |  | etInteger(self, self.titles[0], "Integer: ", 9, 0, | 
            |  |  |  | 1000, 1) | 
            |  |  |  |  | 
            |  |  |  | if ok: | 
            |  |  |  | self.textEdit.append(self. | 
            |  |  |  | tr("%1").arg(integer)) | 
            |  |  |  |  | 
            |  |  |  | def getDouble(self): | 
            |  |  |  | double, ok = QtGui.QInputDialog.ge | 
            |  |  |  | tDouble(self, self.titles[1], self.tr("Amount:"), | 
            |  |  |  | 37.56, -10000, 10000, 1) | 
            |  |  |  |  | 
            |  |  |  | if ok: | 
            |  |  |  | self.textEdit.append(self. | 
            |  |  |  | tr("%1").arg(double)) | 
            |  |  |  |  | 
            |  |  |  | def getItem(self): | 
            |  |  |  | items = QtCore.QStringList() | 
            |  |  |  | items << "GNU/Linux" << "Windows" | 
            |  |  |  | << "Macintosh" << "QNX" | 
            |  |  |  |  | 
            |  |  |  | item, ok = QtGui.QInputDialog.getI | 
            |  |  |  | tem(self, self.titles[2], "OS", items, 0, False) | 
            |  |  |  |  | 
            |  |  |  | if ok and item.isEmpty() == False: | 
            |  |  |  |  | 
            |  |  |  | self.textEdit.append(item) | 
            |  |  |  |  | 
            |  |  |  |  | 
            |  |  |  | def getText(self): | 
            |  |  |  | text, ok = QtGui.QInputDialog.getT | 
            |  |  |  | ext(self, self.titles[3], "Enter your name:") | 
            |  |  |  |  | 
            |  |  |  | if ok and text.isEmpty() == False: | 
            |  |  |  |  | 
            |  |  |  | self.textEdit.append(text) | 
            |  |  |  |  | 
            |  |  |  |  | 
            |  |  |  | def getColor(self): | 
            |  |  |  | color = QtGui.QColorDialog.getColo | 
            |  |  |  | r(QtCore.Qt.blue, self) | 
            |  |  |  |  | 
            |  |  |  | if color.isValid(): | 
            |  |  |  | self.textEdit.setTextColor | 
            |  |  |  | (color) | 
            |  |  |  | self.textEdit.append(color | 
            |  |  |  | .name()) | 
            |  |  |  |  | 
            |  |  |  | def getFont(self): | 
            |  |  |  | font, ok = QtGui.QFontDialog.getFo | 
            |  |  |  | nt() | 
            |  |  |  |  | 
            |  |  |  | if ok: | 
            |  |  |  | self.textEdit.setFont(font | 
            |  |  |  | ) | 
            |  |  |  | self.textEdit.append(font. | 
            |  |  |  | key()) | 
            |  |  |  |  | 
            |  |  |  | def getDirectory(self): | 
            |  |  |  | directory = QtGui.QFileDialog.getE | 
            |  |  |  | xistingDirectory(self, self.titles[6], "Select dir | 
            |  |  |  | ectory:", QtGui.QFileDialog.ShowDirsOnly) | 
            |  |  |  |  | 
            |  |  |  | if directory.isEmpty() == False: | 
            |  |  |  | self.textEdit.append(direc | 
            |  |  |  | tory) | 
            |  |  |  |  | 
            |  |  |  | def openFile(self): | 
            |  |  |  | fName = QtGui.QFileDialog.getOpenF | 
            |  |  |  | ileName(self, self.titles[7], "Open new file", sel | 
            |  |  |  | f.tr("All Files (*);;Text Files (*txt)")) | 
            |  |  |  |  | 
            |  |  |  | if fName.isEmpty() == False: | 
            |  |  |  | self.textEdit.append(fName | 
            |  |  |  | ) | 
            |  |  |  |  | 
            |  |  |  | def saveFile(self): | 
            |  |  |  | fName = QtGui.QFileDialog.getSaveF | 
            |  |  |  | ileName(self, self.titles[8], "Save a new file", s | 
            |  |  |  | elf.tr("All Files(*)")) | 
            |  |  |  |  | 
            |  |  |  | if fName.isEmpty() == False: | 
            |  |  |  | self.textEdit.append(fName | 
            |  |  |  | ) | 
            |  |  |  |  | 
            |  |  |  | def Critical(self): | 
            |  |  |  | reply = QtGui.QMessageBox.critical | 
            |  |  |  | (self, self.titles[9], "Critical message!", QtGui. | 
            |  |  |  | QMessageBox.Abort, QtGui.QMessageBox.Ignore, QtGui | 
            |  |  |  | .QMessageBox.Retry) | 
            |  |  |  |  | 
            |  |  |  | if reply == QtGui.QMessageBox.Abor | 
            |  |  |  | t: | 
            |  |  |  | self.textEdit.append("Abor | 
            |  |  |  | t") | 
            |  |  |  | elif reply == QtGui.QMessageBox.Re | 
            |  |  |  | try: | 
            |  |  |  | self.textEdit.append("Retr | 
            |  |  |  | y") | 
            |  |  |  | else: | 
            |  |  |  | self.textEdit.append("Igno | 
            |  |  |  | re") | 
            |  |  |  |  | 
            |  |  |  | def Info(self): | 
            |  |  |  | QtGui.QMessageBox.information(self | 
            |  |  |  | , self.titles[10], "Information message") | 
            |  |  |  |  | 
            |  |  |  | def Question(self): | 
            |  |  |  | reply = QtGui.QMessageBox.question | 
            |  |  |  | (self, self.titles[11], "Are you sure?", QtGui.QMe | 
            |  |  |  | ssageBox.Yes, QtGui.QMessageBox.No) | 
            |  |  |  |  | 
            |  |  |  | if reply == QtGui.QMessageBox.Yes: | 
            |  |  |  |  | 
            |  |  |  | self.textEdit.append("Yes" | 
            |  |  |  | ) | 
            |  |  |  | else: | 
            |  |  |  | self.textEdit.append("No") | 
            |  |  |  |  | 
            |  |  |  |  | 
            |  |  |  | def Warning(self): | 
            |  |  |  | reply = QtGui.QMessageBox.warning( | 
            |  |  |  | self, self.titles[12], "Warning!", "Try again", "C | 
            |  |  |  | ontinue") | 
            |  |  |  |  | 
            |  |  |  | if reply == 0: | 
            |  |  |  | self.textEdit.append("Try | 
            |  |  |  | again") | 
            |  |  |  | else: | 
            |  |  |  | self.textEdit.append("Cont | 
            |  |  |  | inue") | 
            |  |  |  |  | 
            |  |  |  | app = QtGui.QApplication(sys.argv) | 
            |  |  |  | main = MainWindow() | 
            |  |  |  | main.show() | 
            |  |  |  | sys.exit(app.exec_()) | 
            |  |  |  |  | 
            |  |  |  | Per guadagnare in chiarezza, abbiamo posto i pulsa | 
            |  |  |  | nti, i testi degli stessi e gli slots in degli arr | 
            |  |  |  | ay. Il codice all'interno del ciclo for si occupa | 
            |  |  |  | di inizializzarli e sistemarli all'interno della g | 
            |  |  |  | riglia.:: | 
            |  |  |  |  | 
            |  |  |  | integer, ok = QtGui.QInputDialog.g | 
            |  |  |  | etInteger(self, self.titles[0], "Integer: ", 9, 0, | 
            |  |  |  | 1000, 1) | 
            |  |  |  |  | 
            |  |  |  | Il metodo *getInteger()* richiede i seguenti param | 
            |  |  |  | etri: *self*, una stringa da usare come titolo, un | 
            |  |  |  | 'altra da inserire all'interno del dialog, e quatt | 
            |  |  |  | ro valori numerici. Il primo indica il valore di d | 
            |  |  |  | efault, il secondo il valore minimo selezionabile, | 
            |  |  |  | il terzo il valore massimo, e infine l'ultimo val | 
            |  |  |  | ore indica di quanti numeri si deve andare avanti/ | 
            |  |  |  | indietro quando si premono le freccette a lato. Ad | 
            |  |  |  | esempio se mettete 2, e poi cliccate sulla frecce | 
            |  |  |  | tta in alto, si passerà da 9 direttamente ad 11.:: | 
            |  |  |  |  | 
            |  |  |  |  | 
            |  |  |  | double, ok = QtGui.QInputDialog.ge | 
            |  |  |  | tDouble(self, self.titles[1], self.tr("Amount:"), | 
            |  |  |  | 37.56, -10000, 10000, 1) | 
            |  |  |  |  | 
            |  |  |  | Cosa simile vale per l'input di valori double, ovv | 
            |  |  |  | ero con virgola mobile.:: | 
            |  |  |  |  | 
            |  |  |  | items << "GNU/Linux" << "Windows" | 
            |  |  |  | << "Macintosh" << "QNX" | 
            |  |  |  |  | 
            |  |  |  | item, ok = QtGui.QInputDialog.getI | 
            |  |  |  | tem(self, self.titles[2], "OS", items, 0, False) | 
            |  |  |  |  | 
            |  |  |  | In questo dialog, ci viene proposta una scelta fra | 
            |  |  |  | degli elementi predefiniti. La variabile *items* | 
            |  |  |  | li contiene. Gli ultimi due parametri di *getItem( | 
            |  |  |  | )* indicano rispettivamente l'indice della stringa | 
            |  |  |  | da visualizzare di default e un'indicazione rispe | 
            |  |  |  | tto al layout. Provare a sostituire False con True | 
            |  |  |  | per vedere la differenza.:: | 
            |  |  |  |  | 
            |  |  |  | def getColor(self): | 
            |  |  |  | color = QtGui.QColorDialog.getColo | 
            |  |  |  | r(QtCore.Qt.blue, self) | 
            |  |  |  |  | 
            |  |  |  | Qui l'unico parametro interessante è il primo: il | 
            |  |  |  | colore di default. È un parametro opzionale, e se | 
            |  |  |  | viene omesso è settato a *QtCore.Qt.white*. | 
            |  |  |  |  | 
            |  |  |  | Altri widgets | 
            |  |  |  | ############## | 
            |  |  |  |  | 
            |  |  |  | textEdit | 
            |  |  |  | ========= | 
            |  |  |  |  | 
            |  |  |  | Anche se abbiamo già incontrato questo widget in p | 
            |  |  |  | recedenza, in realtà esistono altri metodi di cui | 
            |  |  |  | non abbiamo parlato. Il prossimo codice ne usa una | 
            |  |  |  | buona parte per creare un rudimentale editor, di | 
            |  |  |  | nome Tiny Editor.:: | 
            |  |  |  |  | 
            |  |  |  | #!/usr/bin/python | 
            |  |  |  |  | 
            |  |  |  | import sys | 
            |  |  |  | from PyQt4 import QtGui, QtCore | 
            |  |  |  |  | 
            |  |  |  | class MainWindow(QtGui.QMainWindow): | 
            |  |  |  | def createMenuVoice(self, iconPath, name, | 
            |  |  |  | shortcut, tip, slot): | 
            |  |  |  | voice = QtGui.QAction(QtGui.QIcon( | 
            |  |  |  | iconPath), name, self) | 
            |  |  |  | voice.setShortcut(shortcut) | 
            |  |  |  | voice.setStatusTip(tip) | 
            |  |  |  | self.connect(voice, QtCore.SIGNAL( | 
            |  |  |  | 'triggered()'), slot) | 
            |  |  |  |  | 
            |  |  |  | return voice | 
            |  |  |  |  | 
            |  |  |  | def createSeparator(self): | 
            |  |  |  | sVoice = QtGui.QAction(self) | 
            |  |  |  | sVoice.setSeparator(True) | 
            |  |  |  |  | 
            |  |  |  | return sVoice | 
            |  |  |  |  | 
            |  |  |  | def createMenu(self): | 
            |  |  |  | tinyMenu = self.menuBar() | 
            |  |  |  |  | 
            |  |  |  | file = tinyMenu.addMenu("&File") | 
            |  |  |  | edit = tinyMenu.addMenu("&Edit") | 
            |  |  |  | font = tinyMenu.addMenu("F&ont") | 
            |  |  |  |  | 
            |  |  |  | new = self.createMenuVoice("icons/ | 
            |  |  |  | new.png", "New", "Ctrl+N", "New file", self.textEd | 
            |  |  |  | it.clear) | 
            |  |  |  | open = self.createMenuVoice("icons | 
            |  |  |  | /open.png", "Open file", "Ctrl+O", "Open a new fil | 
            |  |  |  | e", self.openNewFile) | 
            |  |  |  | save = self.createMenuVoice("icons | 
            |  |  |  | /save.png", "Save file", "Ctrl+S", "Save file", se | 
            |  |  |  | lf.saveNewFile) | 
            |  |  |  | sep = self.createSeparator() | 
            |  |  |  | quit = self.createMenuVoice("icons | 
            |  |  |  | /quit.png", "Quit", "Ctrl+Q", "Quit TinyEditor", Q | 
            |  |  |  | tCore.SLOT('close()')) | 
            |  |  |  | file.addAction(new) | 
            |  |  |  | file.addAction(open) | 
            |  |  |  | file.addAction(save) | 
            |  |  |  | file.addAction(sep) | 
            |  |  |  | file.addAction(quit) | 
            |  |  |  |  | 
            |  |  |  | undo = self.createMenuVoice("icons | 
            |  |  |  | /undo.png", "Undo", "Ctrl+U", "Undo operation", se | 
            |  |  |  | lf.textEdit.undo) | 
            |  |  |  | redo = self.createMenuVoice("icons | 
            |  |  |  | /redo.png", "Redo", "Ctrl+R", "Redo operation", se | 
            |  |  |  | lf.textEdit.redo) | 
            |  |  |  | sep1 = self.createSeparator() | 
            |  |  |  | cut = self.createMenuVoice("icons/ | 
            |  |  |  | cut.png", "Cut", "Ctrl+X", "Cut selected text", se | 
            |  |  |  | lf.textEdit.cut) | 
            |  |  |  | copy = self.createMenuVoice("icons | 
            |  |  |  | /copy.png", "Copy", "Ctrl+C", "Copy selected text" | 
            |  |  |  | , self.textEdit.copy) | 
            |  |  |  | paste = self.createMenuVoice("icon | 
            |  |  |  | s/paste.png", "Paste", "Ctrl+V", "Paste text", sel | 
            |  |  |  | f.paste) | 
            |  |  |  | sep2 = self.createSeparator() | 
            |  |  |  | selectAll = self.createMenuVoice(" | 
            |  |  |  | icons/selectAll.png", "Select all", "Ctrl+A", "Sel | 
            |  |  |  | ect all text", self.textEdit.selectAll) | 
            |  |  |  | edit.addAction(undo) | 
            |  |  |  | edit.addAction(redo) | 
            |  |  |  | edit.addAction(sep1) | 
            |  |  |  | edit.addAction(cut) | 
            |  |  |  | edit.addAction(copy) | 
            |  |  |  | edit.addAction(paste) | 
            |  |  |  | edit.addAction(sep2) | 
            |  |  |  | edit.addAction(selectAll) | 
            |  |  |  |  | 
            |  |  |  | setFont = self.createMenuVoice("ic | 
            |  |  |  | ons/setfont.png", "Set font", "Ctrl+F", "Set font" | 
            |  |  |  | , self.setFont) | 
            |  |  |  | underline = self.createMenuVoice(" | 
            |  |  |  | icons/underline.png", "Underline", "Ctrl+L", "Unde | 
            |  |  |  | rline text", self.underline) | 
            |  |  |  | italic = self.createMenuVoice("ico | 
            |  |  |  | ns/italic.png", "Italic", "Ctrl+I", "Set text to i | 
            |  |  |  | talic", self.italic) | 
            |  |  |  | tColor = self.createMenuVoice("ico | 
            |  |  |  | ns/color.png", "Set text color", "Ctrl+R", "Set te | 
            |  |  |  | xt color", self.setColor) | 
            |  |  |  | delete = self.createMenuVoice("ico | 
            |  |  |  | ns/delete.png", "Delete format", "Ctrl+D", "Delete | 
            |  |  |  | format", self.deleteFormat) | 
            |  |  |  | font.addAction(setFont) | 
            |  |  |  | font.addAction(underline) | 
            |  |  |  | font.addAction(italic) | 
            |  |  |  | font.addAction(tColor) | 
            |  |  |  | font.addAction(delete) | 
            |  |  |  |  | 
            |  |  |  | def __init__(self): | 
            |  |  |  | QtGui.QMainWindow.__init__(self) | 
            |  |  |  |  | 
            |  |  |  | self.setWindowTitle('Tiny editor') | 
            |  |  |  |  | 
            |  |  |  |  | 
            |  |  |  | self.textEdit = QtGui.QTextEdit() | 
            |  |  |  | self.textEdit.setReadOnly(False) | 
            |  |  |  |  | 
            |  |  |  | if self.textEdit.isUndoRedoEnabled | 
            |  |  |  | () == False: | 
            |  |  |  | self.textEdit.setUndoRedoE | 
            |  |  |  | nabled(True) | 
            |  |  |  |  | 
            |  |  |  | self.createMenu() | 
            |  |  |  | self.statusBar() | 
            |  |  |  | self.setCentralWidget(self.textEdi | 
            |  |  |  | t) | 
            |  |  |  |  | 
            |  |  |  | def openNewFile(self): | 
            |  |  |  | fName = QtGui.QFileDialog.getOpenF | 
            |  |  |  | ileName(self, "Open text file", "Open new file", s | 
            |  |  |  | elf.tr("Text Files (*.txt)")) | 
            |  |  |  |  | 
            |  |  |  | if fName.isEmpty() == False: | 
            |  |  |  | fptr = open(fName, 'r') | 
            |  |  |  |  | 
            |  |  |  | content = fptr.read() | 
            |  |  |  | self.textEdit.append(conte | 
            |  |  |  | nt) | 
            |  |  |  | fptr.close() | 
            |  |  |  |  | 
            |  |  |  | def saveNewFile(self): | 
            |  |  |  | fName = QtGui.QFileDialog.getSaveF | 
            |  |  |  | ileName(self, "Save text file", "Save a new file", | 
            |  |  |  | self.tr("Text Files (*.txt)")) | 
            |  |  |  |  | 
            |  |  |  | if fName.isEmpty() == False: | 
            |  |  |  | fptr = open(fName, 'w') | 
            |  |  |  |  | 
            |  |  |  | fptr.write(self.textEdit.t | 
            |  |  |  | oPlainText()) | 
            |  |  |  | fptr.close() | 
            |  |  |  |  | 
            |  |  |  | def paste(self): | 
            |  |  |  | if self.textEdit.canPaste(): | 
            |  |  |  | self.textEdit.paste() | 
            |  |  |  | else: | 
            |  |  |  | QtGui.QMessageBox.critical | 
            |  |  |  | (self, "Error", "Impossible to paste text", QtGui. | 
            |  |  |  | QMessageBox.Ok) | 
            |  |  |  |  | 
            |  |  |  |  | 
            |  |  |  | def setFont(self): | 
            |  |  |  | font, ok = QtGui.QFontDialog.getFo | 
            |  |  |  | nt() | 
            |  |  |  |  | 
            |  |  |  | if ok: | 
            |  |  |  | self.textEdit.setFont(font | 
            |  |  |  | ) | 
            |  |  |  |  | 
            |  |  |  | def underline(self): | 
            |  |  |  | self.textEdit.setFontUnderline(Tru | 
            |  |  |  | e) | 
            |  |  |  |  | 
            |  |  |  | def italic(self): | 
            |  |  |  | self.textEdit.setFontItalic(True) | 
            |  |  |  |  | 
            |  |  |  | def setColor(self): | 
            |  |  |  | color = QtGui.QColorDialog.getColo | 
            |  |  |  | r(QtCore.Qt.blue, self) | 
            |  |  |  |  | 
            |  |  |  | if color.isValid(): | 
            |  |  |  | self.textEdit.setTextColor | 
            |  |  |  | (color) | 
            |  |  |  |  | 
            |  |  |  | def deleteFormat(self): | 
            |  |  |  | self.textEdit.setFontUnderline(Fal | 
            |  |  |  | se) | 
            |  |  |  | self.textEdit.setFontItalic(False) | 
            |  |  |  |  | 
            |  |  |  | self.textEdit.setTextColor(QtCore. | 
            |  |  |  | Qt.black) | 
            |  |  |  |  | 
            |  |  |  | app = QtGui.QApplication(sys.argv) | 
            |  |  |  | main = MainWindow() | 
            |  |  |  | main.show() | 
            |  |  |  | sys.exit(app.exec_()) | 
            |  |  |  |  | 
            |  |  |  | .. img:: 14 | 
            |  |  |  | :nolink: | 
            |  |  |  |  | 
            |  |  |  | Il tutto è piuttosto intuitivo. | 
            |  |  |  | Come notate, la maggior parte delle azioni tipiche | 
            |  |  |  | di un editor (taglia, copia, incolla, seleziona t | 
            |  |  |  | utto, annulla operazione...) non hanno bisogno di | 
            |  |  |  | un'implementazione a parte del programmatore, ma c | 
            |  |  |  | i si può avvalere dei metodi predefiniti del widge | 
            |  |  |  | t. | 
            |  |  |  | Come noterete, sono riapparsi anche precedenti dia | 
            |  |  |  | log predefiniti. | 
            |  |  |  | Come esercizio potreste tentare di migliorare ques | 
            |  |  |  | to editor! | 
            |  |  |  |  | 
            |  |  |  | GroupBox e SpinBox | 
            |  |  |  | =================== | 
            |  |  |  |  | 
            |  |  |  | Vediamo altri due widgets utili:  GroupBox e  Spin | 
            |  |  |  | Box. Il primo serve a dividere diverse sezioni di | 
            |  |  |  | programma tramite dei titoletti in “grassetto”, me | 
            |  |  |  | ntre la seconda è utile per l'input di dati numeri | 
            |  |  |  | ci.:: | 
            |  |  |  |  | 
            |  |  |  | #!/usr/bin/python | 
            |  |  |  |  | 
            |  |  |  | import sys | 
            |  |  |  | from PyQt4 import QtGui, QtCore | 
            |  |  |  |  | 
            |  |  |  | class MainWindow(QtGui.QMainWindow): | 
            |  |  |  | def __init__(self): | 
            |  |  |  | QtGui.QMainWindow.__init__(self) | 
            |  |  |  |  | 
            |  |  |  | self.resize(350, 250) | 
            |  |  |  | self.setWindowTitle('GroupBox e Sp | 
            |  |  |  | inBox') | 
            |  |  |  |  | 
            |  |  |  | widget = QtGui.QWidget(self) | 
            |  |  |  |  | 
            |  |  |  | self.gBox = QtGui.QGroupBox("SpinB | 
            |  |  |  | oxes", widget) | 
            |  |  |  | self.gBox.setCheckable(True) | 
            |  |  |  | self.gBox.setChecked(True) | 
            |  |  |  | self.connect(self.gBox, QtCore.SIG | 
            |  |  |  | NAL('clicked()'), self.showSpin) | 
            |  |  |  |  | 
            |  |  |  | self.iSpin = QtGui.QSpinBox(widget | 
            |  |  |  | ) | 
            |  |  |  | self.iSpin.setRange(0, 100) | 
            |  |  |  | self.iSpin.setSuffix("%") | 
            |  |  |  | self.iSpin.setValue(50) | 
            |  |  |  | self.iSpin.setSingleStep(5) | 
            |  |  |  |  | 
            |  |  |  | self.dSpin = QtGui.QDoubleSpinBox( | 
            |  |  |  | widget) | 
            |  |  |  | self.dSpin.setDecimals(3) | 
            |  |  |  | self.dSpin.setPrefix("$") | 
            |  |  |  | self.dSpin.setRange(0, 1000.0) | 
            |  |  |  | self.dSpin.setSingleStep(0.5) | 
            |  |  |  | self.dSpin.setValue(100) | 
            |  |  |  |  | 
            |  |  |  | vBox = QtGui.QVBoxLayout(widget) | 
            |  |  |  | vBox.setSpacing(3) | 
            |  |  |  |  | 
            |  |  |  | vBox.addWidget(self.iSpin) | 
            |  |  |  | vBox.addWidget(self.dSpin) | 
            |  |  |  |  | 
            |  |  |  | widget.setLayout(vBox) | 
            |  |  |  | self.setCentralWidget(widget) | 
            |  |  |  |  | 
            |  |  |  | def showSpin(self): | 
            |  |  |  | if self.gBox.isChecked(): | 
            |  |  |  | self.dSpin.show() | 
            |  |  |  | self.iSpin.show() | 
            |  |  |  | else: | 
            |  |  |  | self.dSpin.hide() | 
            |  |  |  | self.iSpin.hide() | 
            |  |  |  |  | 
            |  |  |  | app = QtGui.QApplication(sys.argv) | 
            |  |  |  | main = MainWindow() | 
            |  |  |  | main.show() | 
            |  |  |  | sys.exit(app.exec_()) | 
            |  |  |  |  | 
            |  |  |  | .. img:: 11 | 
            |  |  |  | :nolink: | 
            |  |  |  |  | 
            |  |  |  | In questo caso, associamo alla GroupBox una checkb | 
            |  |  |  | ox che ci permette di decidere fra due comportamen | 
            |  |  |  | ti. In realtà una GroupBox di default appare solo | 
            |  |  |  | come un titoletto in grassetto, senza alcuna check | 
            |  |  |  | box. Quando si cambia lo stato della checkbox vien | 
            |  |  |  | e emesso un segnale *clicked()*.:: | 
            |  |  |  |  | 
            |  |  |  | if self.gBox.isChecked(): | 
            |  |  |  |  | 
            |  |  |  | Se questo metodo ritorna True, la checkbox è stata | 
            |  |  |  | attivata, altrimenti è stata disattivata. | 
            |  |  |  | Come vediamo, esistono due principali tipi di spin | 
            |  |  |  | box: la classica *QSpinBox*, che contiene interi, | 
            |  |  |  | e la *QDoubleSpinBox*, che contiene valori double. | 
            |  |  |  |  | 
            |  |  |  | È possibile specificare un range di valori accetta | 
            |  |  |  | bili, un valore di default, il “salto” fra due val | 
            |  |  |  | ori contigui, e stringhe da usare come suffissi o | 
            |  |  |  | prefissi dei valori numerici. | 
            |  |  |  |  | 
            |  |  |  | ProgressBar | 
            |  |  |  | ============ | 
            |  |  |  | Esempio di utilizzo di una barra di avanzamento.:: | 
            |  |  |  |  | 
            |  |  |  |  | 
            |  |  |  | #!/usr/bin/python | 
            |  |  |  |  | 
            |  |  |  | import sys | 
            |  |  |  | from PyQt4 import QtGui, QtCore | 
            |  |  |  |  | 
            |  |  |  | class MainWindow(QtGui.QMainWindow): | 
            |  |  |  | def __init__(self): | 
            |  |  |  | QtGui.QMainWindow.__init__(self) | 
            |  |  |  |  | 
            |  |  |  | self.resize(350, 250) | 
            |  |  |  | self.setWindowTitle('ProgressBar') | 
            |  |  |  |  | 
            |  |  |  | widget = QtGui.QWidget() | 
            |  |  |  |  | 
            |  |  |  | grid = QtGui.QGridLayout(widget) | 
            |  |  |  | self.progressBar = QtGui.QProgress | 
            |  |  |  | Bar(widget) | 
            |  |  |  | self.progressBar.setRange(0, 100) | 
            |  |  |  | self.progressBar.setValue(0) | 
            |  |  |  | self.progressBar.setTextVisible(Tr | 
            |  |  |  | ue) | 
            |  |  |  |  | 
            |  |  |  | self.button = QtGui.QPushButton('S | 
            |  |  |  | tart', widget) | 
            |  |  |  | self.connect(self.button, QtCore.S | 
            |  |  |  | IGNAL('clicked()'), self.StartProgress) | 
            |  |  |  |  | 
            |  |  |  | self.horiz = QtGui.QPushButton('Ve | 
            |  |  |  | rtical', widget) | 
            |  |  |  | self.horiz.setCheckable(True) | 
            |  |  |  | self.connect(self.horiz, QtCore.SI | 
            |  |  |  | GNAL('clicked()'), self.changeOrientation) | 
            |  |  |  |  | 
            |  |  |  | self.direction = QtGui.QPushButton | 
            |  |  |  | ('Reverse', widget) | 
            |  |  |  | self.direction.setCheckable(True) | 
            |  |  |  | self.connect(self.direction, QtCor | 
            |  |  |  | e.SIGNAL('clicked()'), self.Reverse) | 
            |  |  |  |  | 
            |  |  |  | grid.addWidget(self.progressBar, 0 | 
            |  |  |  | , 0, 1, 3) | 
            |  |  |  | grid.addWidget(self.button, 1, 0) | 
            |  |  |  | grid.addWidget(self.horiz, 1, 1) | 
            |  |  |  | grid.addWidget(self.direction, 1, | 
            |  |  |  | 2) | 
            |  |  |  |  | 
            |  |  |  | self.timer = QtCore.QBasicTimer() | 
            |  |  |  | self.step = 0 | 
            |  |  |  |  | 
            |  |  |  | widget.setLayout(grid) | 
            |  |  |  | self.setCentralWidget(widget) | 
            |  |  |  |  | 
            |  |  |  | def Reverse(self): | 
            |  |  |  | if self.direction.isChecked(): | 
            |  |  |  | self.progressBar.setInvert | 
            |  |  |  | edAppearance(True) | 
            |  |  |  | else: | 
            |  |  |  | self.progressBar.setInvert | 
            |  |  |  | edAppearance(False) | 
            |  |  |  |  | 
            |  |  |  | def changeOrientation(self): | 
            |  |  |  | if self.horiz.isChecked(): | 
            |  |  |  | self.progressBar.setOrient | 
            |  |  |  | ation(QtCore.Qt.Vertical) | 
            |  |  |  | else: | 
            |  |  |  | self.progressBar.setOrient | 
            |  |  |  | ation(QtCore.Qt.Horizontal) | 
            |  |  |  |  | 
            |  |  |  | def timerEvent(self, event): | 
            |  |  |  | if self.step >= 100: | 
            |  |  |  | self.timer.stop() | 
            |  |  |  | return | 
            |  |  |  | self.step = self.step+1 | 
            |  |  |  | self.progressBar.setValue(self.ste | 
            |  |  |  | p) | 
            |  |  |  |  | 
            |  |  |  | def StartProgress(self): | 
            |  |  |  | if self.timer.isActive(): | 
            |  |  |  | self.timer.stop() | 
            |  |  |  | self.button.setText('Start | 
            |  |  |  | ') | 
            |  |  |  | else: | 
            |  |  |  | self.timer.start(100, self | 
            |  |  |  | ) | 
            |  |  |  | self.button.setText('Stop' | 
            |  |  |  | ) | 
            |  |  |  |  | 
            |  |  |  | app = QtGui.QApplication(sys.argv) | 
            |  |  |  | main = MainWindow() | 
            |  |  |  | main.show() | 
            |  |  |  | sys.exit(app.exec_()) | 
            |  |  |  |  | 
            |  |  |  | .. img:: 13 | 
            |  |  |  | :nolink: | 
            |  |  |  |  | 
            |  |  |  | La dichiarazione della progress bar è questa::: | 
            |  |  |  |  | 
            |  |  |  | self.progressBar = QtGui.QProgress | 
            |  |  |  | Bar(widget) | 
            |  |  |  | self.progressBar.setRange(0, 100) | 
            |  |  |  | self.progressBar.setValue(0) | 
            |  |  |  | self.progressBar.setTextVisible(Tr | 
            |  |  |  | ue) | 
            |  |  |  |  | 
            |  |  |  | Nota: i tre metodi che chiamo dopo il costruttore | 
            |  |  |  | in realtà potrebbero essere omessi, in quanto quel | 
            |  |  |  | li specificati sono già i comportamenti di default | 
            |  |  |  | . Li ho inseriti solamente a scopo dimostrativo.:: | 
            |  |  |  |  | 
            |  |  |  |  | 
            |  |  |  | self.timer = QtCore.QBasicTimer() | 
            |  |  |  | self.step = 0 | 
            |  |  |  |  | 
            |  |  |  | Qui dichiariamo un timer, un oggetto di basso live | 
            |  |  |  | llo molto semplice e adatto alle nostre esigenze. | 
            |  |  |  | Ogni tot millisecondi (il valore precisato nel met | 
            |  |  |  | odo *start()*), il timer genera un evento che noi | 
            |  |  |  | intercettiamo con la nostra funzione. Ogni 100 mil | 
            |  |  |  | lisecondi dunque avanziamo di 1 nella progress bar | 
            |  |  |  | . | 
            |  |  |  | Se vogliamo che essa sia più veloce, dobbiamo ovvi | 
            |  |  |  | amente diminuire il valore numerico passato a *sta | 
            |  |  |  | rt()*. | 
            |  |  |  | Le due funzioni importanti sono *Reverse* e *chang | 
            |  |  |  | eOrientation*: essi, a seconda del valore del togg | 
            |  |  |  | le button corrispondente, “rovesciano” il testo de | 
            |  |  |  | lla progress bar oppure il suo orientamento. | 
            |  |  |  |  | 
            |  |  |  | Tooltip, password e combo box | 
            |  |  |  | ============================== | 
            |  |  |  |  | 
            |  |  |  | Finiamo con tre widget.:: | 
            |  |  |  |  | 
            |  |  |  | #!/usr/bin/python | 
            |  |  |  |  | 
            |  |  |  | import sys | 
            |  |  |  | from PyQt4 import QtGui, QtCore | 
            |  |  |  |  | 
            |  |  |  | class MainWindow(QtGui.QMainWindow): | 
            |  |  |  | def __init__(self): | 
            |  |  |  | QtGui.QMainWindow.__init__(self) | 
            |  |  |  |  | 
            |  |  |  | self.resize(350, 250) | 
            |  |  |  | self.setWindowTitle('Tooltip e pas | 
            |  |  |  | sword') | 
            |  |  |  | widget = QtGui.QWidget() | 
            |  |  |  | hbox = QtGui.QHBoxLayout(widget) | 
            |  |  |  | hbox.setSpacing(10) | 
            |  |  |  |  | 
            |  |  |  | self.label = QtGui.QLabel("Enter y | 
            |  |  |  | our username", widget) | 
            |  |  |  | self.line = QtGui.QLineEdit(widget | 
            |  |  |  | ) | 
            |  |  |  | comboBox = QtGui.QComboBox() | 
            |  |  |  | comboBox.addItem("Username") | 
            |  |  |  | comboBox.addItem("Password") | 
            |  |  |  | self.connect(comboBox, QtCore.SIGN | 
            |  |  |  | AL("activated(int)"), self.changeEcho) | 
            |  |  |  |  | 
            |  |  |  | hbox.addWidget(self.label) | 
            |  |  |  | hbox.addWidget(self.line) | 
            |  |  |  | hbox.addWidget(comboBox) | 
            |  |  |  | self.setToolTip('This is a tooltip | 
            |  |  |  | ') | 
            |  |  |  | self.setCentralWidget(widget) | 
            |  |  |  |  | 
            |  |  |  | def changeEcho(self, index): | 
            |  |  |  | if index == 0: | 
            |  |  |  | self.line.setEchoMode(QtGu | 
            |  |  |  | i.QLineEdit.Normal) | 
            |  |  |  | self.label.setText("Enter | 
            |  |  |  | your username") | 
            |  |  |  | else: | 
            |  |  |  | self.line.setEchoMode(QtGu | 
            |  |  |  | i.QLineEdit.Password) | 
            |  |  |  | self.label.setText("Enter | 
            |  |  |  | your password") | 
            |  |  |  |  | 
            |  |  |  | app = QtGui.QApplication(sys.argv) | 
            |  |  |  | main = MainWindow() | 
            |  |  |  | main.show() | 
            |  |  |  | sys.exit(app.exec_()) | 
            |  |  |  |  | 
            |  |  |  | .. img:: 16 | 
            |  |  |  | :nolink: | 
            |  |  |  |  | 
            |  |  |  | Il widget *QLineEdit* è comodo per inserire piccol | 
            |  |  |  | e porzioni di testo. Se vogliamo una maggiore sicu | 
            |  |  |  | rezza possiamo anche impostare il tipo di “echo” a | 
            |  |  |  | password, come viene fatto nello slot. | 
            |  |  |  | La *combo box* permette all'utente di scegliere di | 
            |  |  |  | verse opzioni: la selezione è catturata dal segnal | 
            |  |  |  | e *activated()*, che manda al proprio slot un para | 
            |  |  |  | metro intero, ovvero l'indice della voce momentane | 
            |  |  |  | amente attivata. | 
            |  |  |  | Il tooltip può essere visto soffermando il cursore | 
            |  |  |  | su un punto qualsiasi della finestra. | 
            |  |  |  |  | 
            |  |  |  | Conclusioni. | 
            |  |  |  | ############# | 
            |  |  |  |  | 
            |  |  |  | Se avete qualsiasi suggerimento, correzione ai cod | 
            |  |  |  | ici, miglioria che volete vedere in questo tutoria | 
            |  |  |  | l, scrivetemi pure a syn.shainer@gmail.com | 
            |  |  |  | Lisa |