--- a/src/eric7/Plugins/VcsPlugins/vcsGit/GitBisectLogBrowserDialog.py Wed Jul 13 11:16:20 2022 +0200 +++ b/src/eric7/Plugins/VcsPlugins/vcsGit/GitBisectLogBrowserDialog.py Wed Jul 13 14:55:47 2022 +0200 @@ -11,8 +11,12 @@ from PyQt6.QtCore import pyqtSlot, Qt, QPoint, QProcess, QTimer from PyQt6.QtWidgets import ( - QWidget, QDialogButtonBox, QHeaderView, QTreeWidgetItem, QApplication, - QLineEdit + QWidget, + QDialogButtonBox, + QHeaderView, + QTreeWidgetItem, + QApplication, + QLineEdit, ) from EricWidgets import EricMessageBox @@ -28,72 +32,71 @@ """ Class implementing a dialog to browse the bisect log history. """ + CommitIdColumn = 0 OperationColumn = 1 SubjectColumn = 2 - + def __init__(self, vcs, parent=None): """ Constructor - + @param vcs reference to the vcs object @param parent reference to the parent widget (QWidget) """ super().__init__(parent) self.setupUi(self) - + self.__position = QPoint() - - self.buttonBox.button( - QDialogButtonBox.StandardButton.Close).setEnabled(False) - self.buttonBox.button( - QDialogButtonBox.StandardButton.Cancel).setDefault(True) - + + self.buttonBox.button(QDialogButtonBox.StandardButton.Close).setEnabled(False) + self.buttonBox.button(QDialogButtonBox.StandardButton.Cancel).setDefault(True) + self.logTree.headerItem().setText(self.logTree.columnCount(), "") - + self.refreshButton = self.buttonBox.addButton( - self.tr("&Refresh"), QDialogButtonBox.ButtonRole.ActionRole) - self.refreshButton.setToolTip( - self.tr("Press to refresh the list of commits")) + self.tr("&Refresh"), QDialogButtonBox.ButtonRole.ActionRole + ) + self.refreshButton.setToolTip(self.tr("Press to refresh the list of commits")) self.refreshButton.setEnabled(False) - + self.vcs = vcs - + self.repodir = "" self.__currentCommitId = "" - + self.__initData() self.__resetUI() - + self.__process = EricOverrideCursorProcess() self.__process.finished.connect(self.__procFinished) self.__process.readyReadStandardOutput.connect(self.__readStdout) self.__process.readyReadStandardError.connect(self.__readStderr) - + def __initData(self): """ Private method to (re-)initialize some data. """ - self.buf = [] # buffer for stdout - + self.buf = [] # buffer for stdout + def closeEvent(self, e): """ Protected slot implementing a close event handler. - + @param e close event (QCloseEvent) """ if ( - self.__process is not None and - self.__process.state() != QProcess.ProcessState.NotRunning + self.__process is not None + and self.__process.state() != QProcess.ProcessState.NotRunning ): self.__process.terminate() QTimer.singleShot(2000, self.__process.kill) self.__process.waitForFinished(3000) - + self.__position = self.pos() - + e.accept() - + def show(self): """ Public slot to show the dialog. @@ -101,27 +104,26 @@ if not self.__position.isNull(): self.move(self.__position) self.__resetUI() - + super().show() - + def __resetUI(self): """ Private method to reset the user interface. """ self.logTree.clear() - + def __resizeColumnsLog(self): """ Private method to resize the log tree columns. """ - self.logTree.header().resizeSections( - QHeaderView.ResizeMode.ResizeToContents) + self.logTree.header().resizeSections(QHeaderView.ResizeMode.ResizeToContents) self.logTree.header().setStretchLastSection(True) - + def __generateLogItem(self, commitId, operation, subject): """ Private method to generate a bisect log tree entry. - + @param commitId commit id info (string) @param operation bisect operation (string) @param subject subject of the bisect log entry (string) @@ -134,115 +136,110 @@ ] itm = QTreeWidgetItem(self.logTree, columnLabels) return itm - + def __getLogEntries(self): """ Private method to retrieve bisect log entries from the repository. """ - self.buttonBox.button( - QDialogButtonBox.StandardButton.Close).setEnabled(False) - self.buttonBox.button( - QDialogButtonBox.StandardButton.Cancel).setEnabled(True) - self.buttonBox.button( - QDialogButtonBox.StandardButton.Cancel).setDefault(True) - + self.buttonBox.button(QDialogButtonBox.StandardButton.Close).setEnabled(False) + self.buttonBox.button(QDialogButtonBox.StandardButton.Cancel).setEnabled(True) + self.buttonBox.button(QDialogButtonBox.StandardButton.Cancel).setDefault(True) + self.inputGroup.setEnabled(True) self.inputGroup.show() self.refreshButton.setEnabled(False) - + self.buf = [] self.cancelled = False self.errors.clear() self.intercept = False - + args = self.vcs.initCommand("bisect") args.append("log") - + self.__process.kill() - + self.__process.setWorkingDirectory(self.repodir) - + self.inputGroup.setEnabled(True) self.inputGroup.show() - - self.__process.start('git', args) + + self.__process.start("git", args) procStarted = self.__process.waitForStarted(5000) if not procStarted: self.inputGroup.setEnabled(False) self.inputGroup.hide() EricMessageBox.critical( self, - self.tr('Process Generation Error'), + self.tr("Process Generation Error"), self.tr( - 'The process {0} could not be started. ' - 'Ensure, that it is in the search path.' - ).format('git')) - + "The process {0} could not be started. " + "Ensure, that it is in the search path." + ).format("git"), + ) + def start(self, projectdir): """ Public slot to start the git bisect log command. - + @param projectdir directory name of the project (string) """ self.errorGroup.hide() QApplication.processEvents() - + self.__initData() - + # find the root of the repo self.repodir = projectdir while not os.path.isdir(os.path.join(self.repodir, self.vcs.adminDir)): self.repodir = os.path.dirname(self.repodir) if os.path.splitdrive(self.repodir)[1] == os.sep: return - + self.activateWindow() self.raise_() - + self.logTree.clear() self.__getLogEntries() - + def __procFinished(self, exitCode, exitStatus): """ Private slot connected to the finished signal. - + @param exitCode exit code of the process (integer) @param exitStatus exit status of the process (QProcess.ExitStatus) """ self.__processBuffer() self.__finish() - + def __finish(self): """ Private slot called when the process finished or the user pressed the button. """ if ( - self.__process is not None and - self.__process.state() != QProcess.ProcessState.NotRunning + self.__process is not None + and self.__process.state() != QProcess.ProcessState.NotRunning ): self.__process.terminate() QTimer.singleShot(2000, self.__process.kill) self.__process.waitForFinished(3000) - - self.buttonBox.button( - QDialogButtonBox.StandardButton.Close).setEnabled(True) - self.buttonBox.button( - QDialogButtonBox.StandardButton.Cancel).setEnabled(False) - self.buttonBox.button( - QDialogButtonBox.StandardButton.Close).setDefault(True) - + + self.buttonBox.button(QDialogButtonBox.StandardButton.Close).setEnabled(True) + self.buttonBox.button(QDialogButtonBox.StandardButton.Cancel).setEnabled(False) + self.buttonBox.button(QDialogButtonBox.StandardButton.Close).setDefault(True) + self.inputGroup.setEnabled(False) self.inputGroup.hide() self.refreshButton.setEnabled(True) - + def __processBuffer(self): """ Private method to process the buffered output of the git log command. """ for line in self.buf: line = line.rstrip() - + if line.startswith("# "): operation, commitId, subject = line[2:].split(None, 2) # commit is surrounded by [...], abbreviate to 20 chars @@ -250,74 +247,74 @@ # operation is followed by ':' operation = operation[:-1] self.__generateLogItem(commitId, operation, subject) - + self.__resizeColumnsLog() self.logTree.setCurrentItem(self.logTree.topLevelItem(0)) - + # restore current item if self.__currentCommitId: items = self.logTree.findItems( - self.__currentCommitId, Qt.MatchFlag.MatchExactly, - self.CommitIdColumn) + self.__currentCommitId, Qt.MatchFlag.MatchExactly, self.CommitIdColumn + ) if items: self.logTree.setCurrentItem(items[0]) self.__currentCommitId = "" - + def __readStdout(self): """ Private slot to handle the readyReadStandardOutput signal. - + It reads the output of the process and inserts it into a buffer. """ self.__process.setReadChannel(QProcess.ProcessChannel.StandardOutput) - + while self.__process.canReadLine(): - line = str(self.__process.readLine(), - Preferences.getSystem("IOEncoding"), - 'replace') + line = str( + self.__process.readLine(), + Preferences.getSystem("IOEncoding"), + "replace", + ) self.buf.append(line) - + def __readStderr(self): """ Private slot to handle the readyReadStandardError signal. - + It reads the error output of the process and inserts it into the error pane. """ if self.__process is not None: - s = str(self.__process.readAllStandardError(), - Preferences.getSystem("IOEncoding"), - 'replace') + s = str( + self.__process.readAllStandardError(), + Preferences.getSystem("IOEncoding"), + "replace", + ) self.__showError(s) - + def __showError(self, out): """ Private slot to show some error. - + @param out error to be shown (string) """ self.errorGroup.show() self.errors.insertPlainText(out) self.errors.ensureCursorVisible() - + def on_buttonBox_clicked(self, button): """ Private slot called by a button of the button box clicked. - + @param button button that was clicked (QAbstractButton) """ - if button == self.buttonBox.button( - QDialogButtonBox.StandardButton.Close - ): + if button == self.buttonBox.button(QDialogButtonBox.StandardButton.Close): self.close() - elif button == self.buttonBox.button( - QDialogButtonBox.StandardButton.Cancel - ): + elif button == self.buttonBox.button(QDialogButtonBox.StandardButton.Cancel): self.cancelled = True self.__finish() elif button == self.refreshButton: self.on_refreshButton_clicked() - + @pyqtSlot() def on_refreshButton_clicked(self): """ @@ -329,20 +326,20 @@ self.__currentCommitId = itm.text(self.CommitIdColumn) else: self.__currentCommitId = "" - + self.start(self.repodir) - + def on_passwordCheckBox_toggled(self, isOn): """ Private slot to handle the password checkbox toggled. - + @param isOn flag indicating the status of the check box (boolean) """ if isOn: self.input.setEchoMode(QLineEdit.EchoMode.Password) else: self.input.setEchoMode(QLineEdit.EchoMode.Normal) - + @pyqtSlot() def on_sendButton_clicked(self): """ @@ -350,7 +347,7 @@ """ inputTxt = self.input.text() inputTxt += os.linesep - + if self.passwordCheckBox.isChecked(): self.errors.insertPlainText(os.linesep) self.errors.ensureCursorVisible() @@ -358,23 +355,23 @@ self.errors.insertPlainText(inputTxt) self.errors.ensureCursorVisible() self.errorGroup.show() - + self.__process.write(strToQByteArray(inputTxt)) - + self.passwordCheckBox.setChecked(False) self.input.clear() - + def on_input_returnPressed(self): """ Private slot to handle the press of the return key in the input field. """ self.intercept = True self.on_sendButton_clicked() - + def keyPressEvent(self, evt): """ Protected slot to handle a key press event. - + @param evt the key press event (QKeyEvent) """ if self.intercept: