fixed progress monitor error, enhanced GUI error display

This commit is contained in:
k4yt3x 2020-05-08 22:12:24 -04:00
parent e0e42b11c8
commit 0d9d5c4f43
2 changed files with 82 additions and 38 deletions

View File

@ -142,8 +142,12 @@ class Upscaler:
raise ArgumentError('input output path type mismatch') raise ArgumentError('input output path type mismatch')
for input_path in self.input: for input_path in self.input:
if not input_path.is_file() and not input_path.is_dir(): if not input_path.is_file() and not input_path.is_dir():
Avalon.error(_('Input path {} is neither a file nor a directory'.format(input_path))) Avalon.error(_('Input path {} is neither a file nor a directory').format(input_path))
raise FileNotFoundError(f'{input_path} is neither file nor directory') raise FileNotFoundError(f'{input_path} is neither file nor directory')
with contextlib.suppress(FileNotFoundError):
if input_path.samefile(self.output):
Avalon.error(_('Input directory and output directory cannot be the same'))
raise FileExistsError('input directory and output directory are the same')
# if input is a file # if input is a file
elif self.input.is_file(): elif self.input.is_file():
@ -162,6 +166,10 @@ class Upscaler:
Avalon.error(_('Input and output path type mismatch')) Avalon.error(_('Input and output path type mismatch'))
Avalon.error(_('Input is directory but output is existing single file')) Avalon.error(_('Input is directory but output is existing single file'))
raise ArgumentError('input output path type mismatch') raise ArgumentError('input output path type mismatch')
with contextlib.suppress(FileNotFoundError):
if self.input.samefile(self.output):
Avalon.error(_('Input directory and output directory cannot be the same'))
raise FileExistsError('input directory and output directory are the same')
# if input is neither # if input is neither
else: else:
@ -461,7 +469,7 @@ class Upscaler:
Avalon.info(_('Reading video information')) Avalon.info(_('Reading video information'))
video_info = fm.get_video_info(self.current_input_video) video_info = fm.get_video_info(self.current_input_video)
# analyze original video with ffprobe and retrieve framerate # analyze original video with FFprobe and retrieve framerate
# width, height = info['streams'][0]['width'], info['streams'][0]['height'] # width, height = info['streams'][0]['width'], info['streams'][0]['height']
# find index of video stream # find index of video stream

View File

@ -53,14 +53,14 @@ def resource_path(relative_path: str) -> pathlib.Path:
class WorkerSignals(QObject): class WorkerSignals(QObject):
progress = pyqtSignal(tuple) progress = pyqtSignal(tuple)
error = pyqtSignal(str) error = pyqtSignal(Exception)
interrupted = pyqtSignal() interrupted = pyqtSignal()
finished = pyqtSignal() finished = pyqtSignal()
class ProgressBarWorker(QRunnable): class ProgressMonitorWorkder(QRunnable):
def __init__(self, fn, *args, **kwargs): def __init__(self, fn, *args, **kwargs):
super(ProgressBarWorker, self).__init__() super(ProgressMonitorWorkder, self).__init__()
self.fn = fn self.fn = fn
self.args = args self.args = args
self.kwargs = kwargs self.kwargs = kwargs
@ -94,10 +94,9 @@ class UpscalerWorker(QRunnable):
self.fn(*self.args, **self.kwargs) self.fn(*self.args, **self.kwargs)
except (KeyboardInterrupt, SystemExit): except (KeyboardInterrupt, SystemExit):
self.signals.interrupted.emit() self.signals.interrupted.emit()
except Exception: except Exception as e:
error_message = traceback.format_exc() traceback.print_exc()
print(error_message, file=sys.stderr) self.signals.error.emit(e)
self.signals.error.emit(error_message)
else: else:
self.signals.finished.emit() self.signals.finished.emit()
@ -164,7 +163,7 @@ class Video2XMainWindow(QMainWindow):
self.action_exit.triggered.connect(sys.exit) self.action_exit.triggered.connect(sys.exit)
self.action_about = self.findChild(QAction, 'actionAbout') self.action_about = self.findChild(QAction, 'actionAbout')
self.action_about.triggered.connect(lambda: self.show_message(LEGAL_INFO, custom_icon=QtGui.QPixmap(self.video2x_icon_path))) self.action_about.triggered.connect(self.show_about)
# main tab # main tab
# select input file/folder # select input file/folder
@ -617,20 +616,46 @@ class Video2XMainWindow(QMainWindow):
return return
driver_line_edit.setText(str(driver_binary_path.absolute())) driver_line_edit.setText(str(driver_binary_path.absolute()))
def show_error(self, message: str): def show_about(self, message: str):
QErrorMessage(self).showMessage(message.replace('\n', '<br>')) message_box = QMessageBox(self)
message_box.setWindowTitle('About Video2X')
message_box.setIconPixmap(QtGui.QPixmap(self.video2x_icon_path).scaled(64, 64))
message_box.setText(LEGAL_INFO)
message_box.exec_()
def show_message(self, message: str, custom_icon=None): def show_information(self, message: str):
message_box = QMessageBox() message_box = QMessageBox(self)
message_box.setWindowTitle('Message') message_box.setWindowTitle('Information')
if custom_icon: message_box.setIcon(QMessageBox.Information)
message_box.setIconPixmap(custom_icon.scaled(64, 64))
else:
message_box.setIcon(QMessageBox.Information)
message_box.setText(message) message_box.setText(message)
message_box.exec_() message_box.exec_()
def start_progress_bar(self, progress_callback): def show_warning(self, message: str):
message_box = QMessageBox(self)
message_box.setWindowTitle('Warning')
message_box.setIcon(QMessageBox.Warning)
message_box.setText(message)
message_box.exec_()
def show_error(self, exception: Exception):
# QErrorMessage(self).showMessage(message.replace('\n', '<br>'))
message_box = QMessageBox(self)
message_box.setWindowTitle('Error')
message_box.setIcon(QMessageBox.Critical)
error_message = '''Upscaler ran into an error:
{}
Check the console output for details.
When reporting an error, please include console output.'''
try:
message_box.setText(error_message.format(exception.args[0]))
except (AttributeError, IndexError):
message_box.setText(error_message.format(exception))
message_box.exec_()
def progress_monitor(self, progress_callback):
# initialize progress bar values # initialize progress bar values
upscale_begin_time = time.time() upscale_begin_time = time.time()
@ -702,16 +727,18 @@ class Video2XMainWindow(QMainWindow):
# resolve input and output directories from GUI # resolve input and output directories from GUI
if len(self.input_table_data) == 0: if len(self.input_table_data) == 0:
self.show_error('Input path unspecified') self.show_warning('Input path unspecified', standard_icon=QMessageBox.Warning)
return return
if self.output_line_edit.text().strip() == '': if self.output_line_edit.text().strip() == '':
self.show_error('Output path unspecified') self.show_warning('Output path unspecified', standard_icon=QMessageBox.Warning)
return return
if len(self.input_table_data) == 1: if len(self.input_table_data) == 1:
input_directory = self.input_table_data[0] input_directory = self.input_table_data[0]
else: else:
input_directory = self.input_table_data input_directory = self.input_table_data
# resolve output directory
output_directory = pathlib.Path(os.path.expandvars(self.output_line_edit.text())) output_directory = pathlib.Path(os.path.expandvars(self.output_line_edit.text()))
# load driver settings from GUI # load driver settings from GUI
@ -733,33 +760,35 @@ class Video2XMainWindow(QMainWindow):
self.upscaler.image_format = self.config['video2x']['image_format'].lower() self.upscaler.image_format = self.config['video2x']['image_format'].lower()
self.upscaler.preserve_frames = bool(self.preserve_frames_check_box.isChecked()) self.upscaler.preserve_frames = bool(self.preserve_frames_check_box.isChecked())
# start progress bar
if AVAILABLE_DRIVERS[self.driver_combo_box.currentText()] != 'anime4kcpp':
progress_bar_worker = ProgressBarWorker(self.start_progress_bar)
progress_bar_worker.signals.progress.connect(self.set_progress)
self.threadpool.start(progress_bar_worker)
# run upscaler # run upscaler
worker = UpscalerWorker(self.upscaler.run) worker = UpscalerWorker(self.upscaler.run)
worker.signals.error.connect(self.upscale_errored) worker.signals.error.connect(self.upscale_errored)
worker.signals.interrupted.connect(self.upscale_interrupted) worker.signals.interrupted.connect(self.upscale_interrupted)
worker.signals.finished.connect(self.upscale_successful) worker.signals.finished.connect(self.upscale_successful)
self.threadpool.start(worker) self.threadpool.start(worker)
# start progress monitoring
if AVAILABLE_DRIVERS[self.driver_combo_box.currentText()] != 'anime4kcpp':
progress_bar_worker = ProgressMonitorWorkder(self.progress_monitor)
progress_bar_worker.signals.progress.connect(self.set_progress)
self.threadpool.start(progress_bar_worker)
self.start_button.setEnabled(False) self.start_button.setEnabled(False)
self.stop_button.setEnabled(True) self.stop_button.setEnabled(True)
except Exception: except Exception as e:
self.upscale_errored(traceback.format_exc()) traceback.print_exc()
self.upscale_errored(e)
def upscale_errored(self, error_message): def upscale_errored(self, exception: Exception):
self.show_error(f'Upscaler ran into an error:\n{error_message}') self.show_error(exception)
self.threadpool.waitForDone(5) self.threadpool.waitForDone(5)
self.start_button.setEnabled(True) self.start_button.setEnabled(True)
self.stop_button.setEnabled(False) self.stop_button.setEnabled(False)
self.reset_progress_display() self.reset_progress_display()
def upscale_interrupted(self): def upscale_interrupted(self):
self.show_message('Upscale has been interrupted') self.show_information('Upscale has been interrupted')
self.threadpool.waitForDone(5) self.threadpool.waitForDone(5)
self.start_button.setEnabled(True) self.start_button.setEnabled(True)
self.stop_button.setEnabled(False) self.stop_button.setEnabled(False)
@ -768,7 +797,7 @@ class Video2XMainWindow(QMainWindow):
def upscale_successful(self): def upscale_successful(self):
# if all threads have finished # if all threads have finished
self.threadpool.waitForDone(5) self.threadpool.waitForDone(5)
self.show_message('Upscale finished successfully, taking {} seconds'.format(round((time.time() - self.begin_time), 5))) self.show_information('Upscale finished successfully, taking {} seconds'.format(round((time.time() - self.begin_time), 5)))
self.start_button.setEnabled(True) self.start_button.setEnabled(True)
self.stop_button.setEnabled(False) self.stop_button.setEnabled(False)
self.reset_progress_display() self.reset_progress_display()
@ -785,7 +814,14 @@ class Video2XMainWindow(QMainWindow):
# this file shouldn't be imported # this file shouldn't be imported
if __name__ == '__main__': if __name__ == '__main__':
app = QApplication(sys.argv) try:
window = Video2XMainWindow() app = QApplication(sys.argv)
window.show() window = Video2XMainWindow()
app.exec_() window.show()
app.exec_()
# on GUI exception, print error message in console
# and hold window open using input()
except Exception:
traceback.print_exc()
input('Press enter to close')