def new_qt_application(app_config: dict, logger: Logger, quit_on_last_closed: bool = False, name: str = None) -> QApplication: app = QApplication(sys.argv) app.setQuitOnLastWindowClosed( quit_on_last_closed ) # otherwise windows opened through the tray icon kill the application when closed app.setApplicationName(name if name else __app_name__) app.setApplicationVersion(__version__) app.setWindowIcon(util.get_default_icon()[1]) if app_config['ui']['qt_style']: app.setStyle(str(app_config['ui']['qt_style'])) else: app.setStyle('fusion') app.setProperty('qt_style', app.style().objectName().lower()) theme_key = app_config['ui']['theme'].strip( ) if app_config['ui']['theme'] else None set_theme(theme_key=theme_key, app=app, logger=logger) if not app_config['ui']['system_theme']: app.setPalette(app.style().standardPalette()) return app
def new_manage_panel(app_args: Namespace, app_config: dict, logger: logging.Logger) -> Tuple[QApplication, QWidget]: i18n = generate_i18n(app_config, resource.get_path('locale')) cache_cleaner = CacheCleaner() cache_factory = DefaultMemoryCacheFactory(expiration_time=int(app_config['memory_cache']['data_expiration']), cleaner=cache_cleaner) icon_cache = cache_factory.new(int(app_config['memory_cache']['icon_expiration'])) http_client = HttpClient(logger) context = ApplicationContext(i18n=i18n, http_client=http_client, download_icons=bool(app_config['download']['icons']), app_root_dir=ROOT_DIR, cache_factory=cache_factory, disk_loader_factory=DefaultDiskCacheLoaderFactory(logger), logger=logger, distro=util.get_distro(), file_downloader=AdaptableFileDownloader(logger, bool(app_config['download']['multithreaded']), i18n, http_client, app_config['download']['multithreaded_client']), app_name=__app_name__, internet_checker=InternetChecker(offline=app_args.offline)) managers = gems.load_managers(context=context, locale=i18n.current_key, config=app_config, default_locale=DEFAULT_I18N_KEY) if app_args.reset: util.clean_app_files(managers) exit(0) manager = GenericSoftwareManager(managers, context=context, config=app_config) app = new_qt_application(app_config=app_config, logger=logger, quit_on_last_closed=True) if app_args.settings: # only settings window manager.cache_available_managers() return app, SettingsWindow(manager=manager, i18n=i18n, screen_size=app.primaryScreen().size(), window=None) else: manage_window = ManageWindow(i18n=i18n, manager=manager, icon_cache=icon_cache, screen_size=app.primaryScreen().size(), config=app_config, context=context, http_client=http_client, icon=util.get_default_icon()[1], logger=logger) prepare = PreparePanel(screen_size=app.primaryScreen().size(), context=context, manager=manager, i18n=i18n, manage_window=manage_window, app_config=app_config) cache_cleaner.start() return app, prepare
def __init__(self, i18n: I18n, max_tries: int = 3): super(RootDialog, self).__init__(flags=Qt.CustomizeWindowHint | Qt.WindowTitleHint) self.i18n = i18n self.max_tries = max_tries self.tries = 0 self.setSizePolicy(QSizePolicy.Minimum, QSizePolicy.Preferred) self.setWindowIcon(util.get_default_icon()[1]) self.setWindowTitle(i18n['popup.root.title']) self.setLayout(QVBoxLayout()) self.setMinimumWidth(300) self.label_msg = QLabel(i18n['popup.root.msg']) self.label_msg.setObjectName('message') self.layout().addWidget(self.label_msg) self.input_password = QLineEdit() self.input_password.setObjectName('password') self.layout().addWidget(self.input_password) self.label_error = QLabel() self.label_error.setProperty('error', 'true') self.layout().addWidget(self.label_error) self.label_error.hide() self.lower_bar = QToolBar() self.layout().addWidget(self.lower_bar) self.lower_bar.addWidget(new_spacer()) self.bt_ok = QPushButton(i18n['popup.root.continue']) self.bt_ok.setDefault(True) self.bt_ok.setAutoDefault(True) self.bt_ok.setCursor(QCursor(Qt.PointingHandCursor)) self.bt_ok.setSizePolicy(QSizePolicy.Minimum, QSizePolicy.Minimum) self.bt_ok.setObjectName('ok') self.bt_ok.clicked.connect(self._validate_password) self.lower_bar.addWidget(self.bt_ok) self.bt_cancel = QPushButton() self.bt_cancel.setText(i18n['popup.button.cancel']) self.bt_cancel.setCursor(QCursor(Qt.PointingHandCursor)) self.bt_cancel.setSizePolicy(QSizePolicy.Minimum, QSizePolicy.Minimum) self.bt_cancel.setObjectName('bt_cancel') self.bt_cancel.clicked.connect(self.close) self.lower_bar.addWidget(self.bt_cancel) self.lower_bar.addWidget(new_spacer()) self.valid = False self.password = None self.validate_password = ValidatePassword() self.validate_password.signal_valid.connect( self._handle_password_validated)
def ask_root_password(context: ApplicationContext, i18n: I18n, app_config: dict = None) -> Tuple[str, bool]: cur_config = read_config() if not app_config else app_config store_password = bool(cur_config['store_root_password']) if store_password and context.root_password and validate_password(context.root_password): return context.root_password, True diag = QInputDialog() diag.setStyleSheet("""QLineEdit { border-radius: 5px; font-size: 16px; border: 1px solid lightblue }""") diag.setInputMode(QInputDialog.TextInput) diag.setTextEchoMode(QLineEdit.Password) diag.setWindowIcon(util.get_default_icon()[1]) diag.setWindowTitle(i18n['popup.root.title']) diag.setLabelText('') diag.setOkButtonText(i18n['popup.root.continue'].capitalize()) diag.setCancelButtonText(i18n['popup.button.cancel'].capitalize()) diag.resize(400, 200) for attempt in range(3): ok = diag.exec_() if ok: if not validate_password(diag.textValue()): body = i18n['popup.root.bad_password.body'] if attempt == 2: body += '. ' + i18n['popup.root.bad_password.last_try'] show_message(title=i18n['popup.root.bad_password.title'], body=body, type_=MessageType.ERROR) ok = False diag.setTextValue('') if ok: if store_password: context.root_password = diag.textValue() return diag.textValue(), ok else: break return '', False
def new_qt_application(app_config: dict, quit_on_last_closed: bool = False, name: str = None) -> QApplication: app = QApplication(sys.argv) app.setQuitOnLastWindowClosed( quit_on_last_closed ) # otherwise windows opened through the tray icon kill the application when closed app.setApplicationName(name if name else __app_name__) app.setApplicationVersion(__version__) app.setWindowIcon(util.get_default_icon()[1]) if app_config['ui']['style']: app.setStyle(str(app_config['ui']['style'])) else: if app.style().objectName().lower() not in {'fusion', 'breeze'}: app.setStyle('Fusion') return app
def main(): if not os.getenv('PYTHONUNBUFFERED'): os.environ['PYTHONUNBUFFERED'] = '1' urllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning) args = app_args.read() logger = logs.new_logger(__app_name__, bool(args.logs)) local_config = config.read_config(update_file=True) if local_config['ui']['auto_scale']: os.environ['QT_AUTO_SCREEN_SCALE_FACTOR'] = '1' logger.info("Auto screen scale factor activated") if local_config['ui']['hdpi']: logger.info("HDPI settings activated") QCoreApplication.setAttribute(Qt.AA_UseHighDpiPixmaps) QCoreApplication.setAttribute(Qt.AA_EnableHighDpiScaling) i18n_key, current_i18n = translation.get_locale_keys( local_config['locale']) default_i18n = translation.get_locale_keys( DEFAULT_I18N_KEY)[1] if i18n_key != DEFAULT_I18N_KEY else {} i18n = I18n(i18n_key, current_i18n, DEFAULT_I18N_KEY, default_i18n) cache_cleaner = CacheCleaner() cache_factory = DefaultMemoryCacheFactory(expiration_time=int( local_config['memory_cache']['data_expiration']), cleaner=cache_cleaner) icon_cache = cache_factory.new( int(local_config['memory_cache']['icon_expiration'])) http_client = HttpClient(logger) context = ApplicationContext( i18n=i18n, http_client=http_client, disk_cache=bool(local_config['disk_cache']['enabled']), download_icons=bool(local_config['download']['icons']), app_root_dir=ROOT_DIR, cache_factory=cache_factory, disk_loader_factory=DefaultDiskCacheLoaderFactory( disk_cache_enabled=bool(local_config['disk_cache']['enabled']), logger=logger), logger=logger, distro=util.get_distro(), file_downloader=AdaptableFileDownloader( logger, bool(local_config['download']['multithreaded']), i18n, http_client), app_name=__app_name__) managers = gems.load_managers(context=context, locale=i18n_key, config=local_config, default_locale=DEFAULT_I18N_KEY) if args.reset: util.clean_app_files(managers) exit(0) manager = GenericSoftwareManager(managers, context=context, config=local_config) manager.prepare() app = QApplication(sys.argv) app.setQuitOnLastWindowClosed( False ) # otherwise windows opened through the tray icon kill the aplication when closed app.setApplicationName(__app_name__) app.setApplicationVersion(__version__) app_icon = util.get_default_icon()[1] app.setWindowIcon(app_icon) if local_config['ui']['style']: app.setStyle(str(local_config['ui']['style'])) else: if app.style().objectName().lower() not in {'fusion', 'breeze'}: app.setStyle('Fusion') manage_window = ManageWindow(i18n=i18n, manager=manager, icon_cache=icon_cache, screen_size=app.primaryScreen().size(), config=local_config, context=context, http_client=http_client, icon=app_icon, logger=logger) if args.tray: tray_icon = TrayIcon(i18n=i18n, manager=manager, manage_window=manage_window, screen_size=app.primaryScreen().size(), config=local_config) manage_window.set_tray_icon(tray_icon) tray_icon.show() if args.show_panel: tray_icon.show_manage_window() else: manage_window.refresh_apps() manage_window.show() cache_cleaner.start() Thread(target=config.remove_old_config, args=(logger, ), daemon=True).start() sys.exit(app.exec_())
def ask_root_password(context: ApplicationContext, i18n: I18n, app_config: dict = None) -> Tuple[str, bool]: cur_config = read_config() if not app_config else app_config store_password = bool(cur_config['store_root_password']) if store_password and context.root_password and validate_password( context.root_password): return context.root_password, True diag = QInputDialog(flags=Qt.CustomizeWindowHint | Qt.WindowTitleHint) diag.setStyleSheet( """QLineEdit { border-radius: 5px; font-size: 16px; border: 1px solid lightblue }""" ) diag.setInputMode(QInputDialog.TextInput) diag.setTextEchoMode(QLineEdit.Password) diag.setWindowIcon(util.get_default_icon()[1]) diag.setWindowTitle(i18n['popup.root.title']) diag.setLabelText('') diag.setOkButtonText(i18n['popup.root.continue'].capitalize()) diag.setCancelButtonText(i18n['popup.button.cancel'].capitalize()) bt_box = [c for c in diag.children() if isinstance(c, QDialogButtonBox)][0] for bt in bt_box.buttons(): if bt_box.buttonRole(bt) == QDialogButtonBox.AcceptRole: bt.setStyleSheet(css.OK_BUTTON) bt.setCursor(QCursor(Qt.PointingHandCursor)) bt.setIcon(QIcon()) diag.resize(400, 200) for attempt in range(3): ok = diag.exec_() if ok: if not validate_password(diag.textValue()): body = i18n['popup.root.bad_password.body'] if attempt == 2: body += '. ' + i18n['popup.root.bad_password.last_try'] show_message(title=i18n['popup.root.bad_password.title'], body=body, type_=MessageType.ERROR) ok = False diag.setTextValue('') if ok: if store_password: context.root_password = diag.textValue() return diag.textValue(), ok else: break return '', False
def __init__(self, i18n: I18n): super(AboutDialog, self).__init__() self.setWindowTitle(i18n['tray.action.about']) layout = QVBoxLayout() self.setLayout(layout) label_logo = QLabel() icon = util.get_default_icon()[1].pixmap(64, 64) label_logo.setPixmap(icon) label_logo.setAlignment(Qt.AlignCenter) layout.addWidget(label_logo) label_name = QLabel(__app_name__) label_name.setStyleSheet('font-weight: bold; font-size: 14px') label_name.setAlignment(Qt.AlignCenter) layout.addWidget(label_name) label_version = QLabel(i18n['version'].lower() + ' ' + __version__) label_version.setStyleSheet( 'QLabel { font-size: 11px; font-weight: bold }') label_version.setAlignment(Qt.AlignCenter) layout.addWidget(label_version) layout.addWidget(QLabel('')) line_desc = QLabel(i18n['about.info.desc']) line_desc.setStyleSheet('font-size: 12px; font-weight: bold;') line_desc.setAlignment(Qt.AlignCenter) line_desc.setMinimumWidth(400) layout.addWidget(line_desc) layout.addWidget(QLabel('')) available_gems = [ f for f in glob('{}/gems/*'.format(ROOT_DIR)) if not f.endswith('.py') and not f.endswith('__pycache__') ] available_gems.sort() gems_widget = QWidget() gems_widget.setLayout(QHBoxLayout()) gems_widget.layout().addWidget(QLabel()) for gem_path in available_gems: icon = QLabel() icon_path = gem_path + '/resources/img/{}.svg'.format( gem_path.split('/')[-1]) icon.setPixmap(QIcon(icon_path).pixmap(QSize(25, 25))) gems_widget.layout().addWidget(icon) gems_widget.layout().addWidget(QLabel()) layout.addWidget(gems_widget) layout.addWidget(QLabel('')) label_more_info = QLabel() label_more_info.setStyleSheet('font-size: 11px;') label_more_info.setText(i18n['about.info.link'] + " <a href='{url}'>{url}</a>".format( url=PROJECT_URL)) label_more_info.setOpenExternalLinks(True) label_more_info.setAlignment(Qt.AlignCenter) layout.addWidget(label_more_info) label_license = QLabel() label_license.setStyleSheet('font-size: 11px;') label_license.setText("<a href='{}'>{}</a>".format( LICENSE_URL, i18n['about.info.license'])) label_license.setOpenExternalLinks(True) label_license.setAlignment(Qt.AlignCenter) layout.addWidget(label_license) layout.addWidget(QLabel('')) label_trouble_question = QLabel(i18n['about.info.trouble.question']) label_trouble_question.setStyleSheet( 'font-size: 10px; font-weight: bold') label_trouble_question.setAlignment(Qt.AlignCenter) layout.addWidget(label_trouble_question) label_trouble_answer = QLabel(i18n['about.info.trouble.answer']) label_trouble_answer.setStyleSheet('font-size: 10px;') label_trouble_answer.setAlignment(Qt.AlignCenter) layout.addWidget(label_trouble_answer) layout.addWidget(QLabel('')) label_rate_question = QLabel(i18n['about.info.rate.question']) label_rate_question.setStyleSheet( 'font-size: 10px; font-weight: bold;') label_rate_question.setAlignment(Qt.AlignCenter) layout.addWidget(label_rate_question) label_rate_answer = QLabel(i18n['about.info.rate.answer']) label_rate_answer.setStyleSheet('font-size: 10px;') label_rate_answer.setAlignment(Qt.AlignCenter) layout.addWidget(label_rate_answer) layout.addWidget(QLabel('')) self.adjustSize() self.setFixedSize(self.size())
def new_manage_panel(app_args: Namespace, app_config: dict, logger: logging.Logger) -> Tuple[QApplication, QWidget]: i18n = generate_i18n(app_config, resource.get_path('locale')) cache_cleaner = CacheCleaner() cache_factory = DefaultMemoryCacheFactory(expiration_time=int(app_config['memory_cache']['data_expiration']), cleaner=cache_cleaner) icon_cache = cache_factory.new(int(app_config['memory_cache']['icon_expiration'])) http_client = HttpClient(logger) downloader = AdaptableFileDownloader(logger=logger, multithread_enabled=app_config['download']['multithreaded'], multithread_client=app_config['download']['multithreaded_client'], i18n=i18n, http_client=http_client, check_ssl=app_config['download']['check_ssl']) context = ApplicationContext(i18n=i18n, http_client=http_client, download_icons=bool(app_config['download']['icons']), app_root_dir=ROOT_DIR, cache_factory=cache_factory, disk_loader_factory=DefaultDiskCacheLoaderFactory(logger), logger=logger, distro=util.get_distro(), file_downloader=downloader, app_name=__app_name__, app_version=__version__, internet_checker=InternetChecker(offline=app_args.offline), suggestions_mapping=read_suggestions_mapping(), root_user=user.is_root()) managers = gems.load_managers(context=context, locale=i18n.current_key, config=app_config, default_locale=DEFAULT_I18N_KEY, logger=logger) if app_args.reset: util.clean_app_files(managers) exit(0) force_suggestions = bool(app_args.suggestions) manager = GenericSoftwareManager(managers, context=context, config=app_config, force_suggestions=force_suggestions) app = new_qt_application(app_config=app_config, logger=logger, quit_on_last_closed=True) screen_size = app.primaryScreen().size() context.screen_width, context.screen_height = screen_size.width(), screen_size.height() logger.info(f"Screen: {screen_size.width()} x {screen_size.height()} " f"(DPI: {int(app.primaryScreen().logicalDotsPerInch())})") if app_args.settings: # only settings window manager.cache_available_managers() return app, SettingsWindow(manager=manager, i18n=i18n, window=None) else: manage_window = ManageWindow(i18n=i18n, manager=manager, icon_cache=icon_cache, screen_size=screen_size, config=app_config, context=context, http_client=http_client, icon=util.get_default_icon()[1], force_suggestions=force_suggestions, logger=logger) prepare = PreparePanel(screen_size=screen_size, context=context, manager=manager, i18n=i18n, manage_window=manage_window, app_config=app_config, force_suggestions=force_suggestions) cache_cleaner.start() return app, prepare