# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et: # Copyright 2014-2021 Florian Bruhin (The Compiler) # # This file is part of qutebrowser. # # qutebrowser is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # qutebrowser is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with qutebrowser. If not, see . """Get arguments to pass to Qt.""" import os import sys import argparse import pathlib from typing import Any, Dict, Iterator, List, Optional, Sequence, Tuple from PyQt5.QtCore import QLibraryInfo, QLocale from qutebrowser.config import config from qutebrowser.misc import objects from qutebrowser.utils import usertypes, qtutils, utils, log, version _ENABLE_FEATURES = '--enable-features=' _DISABLE_FEATURES = '--disable-features=' _BLINK_SETTINGS = '--blink-settings=' def qt_args(namespace: argparse.Namespace) -> List[str]: """Get the Qt QApplication arguments based on an argparse namespace. Args: namespace: The argparse namespace. Return: The argv list to be passed to Qt. """ argv = [sys.argv[0]] if namespace.qt_flag is not None: argv += ['--' + flag[0] for flag in namespace.qt_flag] if namespace.qt_arg is not None: for name, value in namespace.qt_arg: argv += ['--' + name, value] argv += ['--' + arg for arg in config.val.qt.args] if objects.backend != usertypes.Backend.QtWebEngine: assert objects.backend == usertypes.Backend.QtWebKit, objects.backend return argv try: # pylint: disable=unused-import from qutebrowser.browser.webengine import webenginesettings except ImportError: # This code runs before a QApplication is available, so before # backendproblem.py is run to actually inform the user of the missing # backend. Thus, we could end up in a situation where we're here, but # QtWebEngine isn't actually available. # We shouldn't call _qtwebengine_args() in this case as it relies on # QtWebEngine actually being importable, e.g. in # version.qtwebengine_versions(). log.init.debug("QtWebEngine requested, but unavailable...") return argv special_prefixes = (_ENABLE_FEATURES, _DISABLE_FEATURES, _BLINK_SETTINGS) special_flags = [flag for flag in argv if flag.startswith(special_prefixes)] argv = [flag for flag in argv if not flag.startswith(special_prefixes)] argv += list(_qtwebengine_args(namespace, special_flags)) return argv def _qtwebengine_features( versions: version.WebEngineVersions, special_flags: Sequence[str], ) -> Tuple[Sequence[str], Sequence[str]]: """Get a tuple of --enable-features/--disable-features flags for QtWebEngine. Args: versions: The WebEngineVersions to get flags for. special_flags: Existing flags passed via the commandline. """ enabled_features = [] disabled_features = [] for flag in special_flags: if flag.startswith(_ENABLE_FEATURES): flag = flag[len(_ENABLE_FEATURES):] enabled_features += flag.split(',') elif flag.startswith(_DISABLE_FEATURES): flag = flag[len(_DISABLE_FEATURES):] disabled_features += flag.split(',') elif flag.startswith(_BLINK_SETTINGS): pass else: raise utils.Unreachable(flag) if versions.webengine >= utils.VersionNumber(5, 15, 1) and utils.is_linux: # Enable WebRTC PipeWire for screen capturing on Wayland. # # This is disabled in Chromium by default because of the "dialog hell": # https://bugs.chromium.org/p/chromium/issues/detail?id=682122#c50 # https://github.com/flatpak/xdg-desktop-portal-gtk/issues/204 # # However, we don't have Chromium's confirmation dialog in qutebrowser, # so we should only get qutebrowser's permission dialog. # # In theory this would be supported with Qt 5.13 already, but # QtWebEngine only started picking up PipeWire correctly with Qt # 5.15.1. # # This only should be enabled on Wayland, but it's too early to check # that, as we don't have a QApplication available at this point. Thus, # just turn it on unconditionally on Linux, which shouldn't hurt. enabled_features.append('WebRTCPipeWireCapturer') if not utils.is_mac: # Enable overlay scrollbars. # # There are two additional flags in Chromium: # # - OverlayScrollbarFlashAfterAnyScrollUpdate # - OverlayScrollbarFlashWhenMouseEnter # # We don't expose/activate those, but the changes they introduce are # quite subtle: The former seems to show the scrollbar handle even if # there was a 0px scroll (though no idea how that can happen...). The # latter flashes *all* scrollbars when a scrollable area was entered, # which doesn't seem to make much sense. if config.val.scrolling.bar == 'overlay': enabled_features.append('OverlayScrollbar') if (versions.webengine >= utils.VersionNumber(5, 14) and config.val.content.headers.referer == 'same-domain'): # Handling of reduced-referrer-granularity in Chromium 76+ # https://chromium-review.googlesource.com/c/chromium/src/+/1572699 # # Note that this is removed entirely (and apparently the default) starting with # Chromium 89 (presumably arriving with Qt 6.2): # https://chromium-review.googlesource.com/c/chromium/src/+/2545444 enabled_features.append('ReducedReferrerGranularity') if versions.webengine == utils.VersionNumber(5, 15, 2): # WORKAROUND for https://bugreports.qt.io/browse/QTBUG-89740 disabled_features.append('InstalledApp') if not config.val.input.media_keys: disabled_features.append('HardwareMediaKeyHandling') return (enabled_features, disabled_features) def _get_locale_pak_path(locales_path: pathlib.Path, locale_name: str) -> pathlib.Path: """Get the path for a locale .pak file.""" return locales_path / (locale_name + '.pak') def _get_pak_name(locale_name: str) -> str: """Get the Chromium .pak name for a locale name. Based on Chromium's behavior in l10n_util::CheckAndResolveLocale: https://source.chromium.org/chromium/chromium/src/+/master:ui/base/l10n/l10n_util.cc;l=344-428;drc=43d5378f7f363dab9271ca37774c71176c9e7b69 """ if locale_name in {'en', 'en-PH', 'en-LR'}: return 'en-US' elif locale_name.startswith('en-'): return 'en-GB' elif locale_name.startswith('es-'): return 'es-419' elif locale_name == 'pt': return 'pt-BR' elif locale_name.startswith('pt-'): # pragma: no cover return 'pt-PT' elif locale_name in {'zh-HK', 'zh-MO'}: return 'zh-TW' elif locale_name == 'zh' or locale_name.startswith('zh-'): return 'zh-CN' return locale_name.split('-')[0] def _webengine_locales_path() -> pathlib.Path: """Get the path of the QtWebEngine locales.""" if version.is_flatpak(): # TranslationsPath is /usr/translations on Flatpak, i.e. the path for qtbase, # not QtWebEngine. base = pathlib.Path('/app/translations') else: base = pathlib.Path(QLibraryInfo.location(QLibraryInfo.TranslationsPath)) return base / 'qtwebengine_locales' def _get_lang_override( webengine_version: utils.VersionNumber, locale_name: str ) -> Optional[str]: """Get a --lang switch to override Qt's locale handling. This is needed as a WORKAROUND for https://bugreports.qt.io/browse/QTBUG-91715 There is no fix yet, but we assume it'll be fixed with QtWebEngine 5.15.4. """ if not config.val.qt.workarounds.locale: return None if webengine_version != utils.VersionNumber(5, 15, 3) or not utils.is_linux: return None locales_path = _webengine_locales_path() if not locales_path.exists(): log.init.debug(f"{locales_path} not found, skipping workaround!") return None pak_path = _get_locale_pak_path(locales_path, locale_name) if pak_path.exists(): log.init.debug(f"Found {pak_path}, skipping workaround") return None pak_name = _get_pak_name(locale_name) pak_path = _get_locale_pak_path(locales_path, pak_name) if pak_path.exists(): log.init.debug(f"Found {pak_path}, applying workaround") return pak_name log.init.debug(f"Can't find pak in {locales_path} for {locale_name} or {pak_name}") return 'en-US' def _qtwebengine_args( namespace: argparse.Namespace, special_flags: Sequence[str], ) -> Iterator[str]: """Get the QtWebEngine arguments to use based on the config.""" versions = version.qtwebengine_versions(avoid_init=True) qt_514_ver = utils.VersionNumber(5, 14) qt_515_ver = utils.VersionNumber(5, 15) if qt_514_ver <= versions.webengine < qt_515_ver: # WORKAROUND for https://bugreports.qt.io/browse/QTBUG-82105 yield '--disable-shared-workers' # WORKAROUND equivalent to # https://codereview.qt-project.org/c/qt/qtwebengine/+/256786 # also see: # https://codereview.qt-project.org/c/qt/qtwebengine-chromium/+/265753 if versions.webengine >= utils.VersionNumber(5, 12, 3): if 'stack' in namespace.debug_flags: # Only actually available in Qt 5.12.5, but let's save another # check, as passing the option won't hurt. yield '--enable-in-process-stack-traces' elif 'stack' not in namespace.debug_flags: yield '--disable-in-process-stack-traces' lang_override = _get_lang_override( webengine_version=versions.webengine, locale_name=QLocale().bcp47Name(), ) if lang_override is not None: yield f'--lang={lang_override}' if 'chromium' in namespace.debug_flags: yield '--enable-logging' yield '--v=1' if 'wait-renderer-process' in namespace.debug_flags: yield '--renderer-startup-dialog' from qutebrowser.browser.webengine import darkmode darkmode_settings = darkmode.settings( versions=versions, special_flags=special_flags, ) for switch_name, values in darkmode_settings.items(): # If we need to use other switches (say, --enable-features), we might need to # refactor this so values still get combined with existing ones. assert switch_name in ['dark-mode-settings', 'blink-settings'], switch_name yield f'--{switch_name}=' + ','.join(f'{k}={v}' for k, v in values) enabled_features, disabled_features = _qtwebengine_features(versions, special_flags) if enabled_features: yield _ENABLE_FEATURES + ','.join(enabled_features) if disabled_features: yield _DISABLE_FEATURES + ','.join(disabled_features) yield from _qtwebengine_settings_args(versions) def _qtwebengine_settings_args(versions: version.WebEngineVersions) -> Iterator[str]: settings: Dict[str, Dict[Any, Optional[str]]] = { 'qt.force_software_rendering': { 'software-opengl': None, 'qt-quick': None, 'chromium': '--disable-gpu', 'none': None, }, 'content.canvas_reading': { True: None, False: '--disable-reading-from-canvas', }, 'content.webrtc_ip_handling_policy': { 'all-interfaces': None, 'default-public-and-private-interfaces': '--force-webrtc-ip-handling-policy=' 'default_public_and_private_interfaces', 'default-public-interface-only': '--force-webrtc-ip-handling-policy=' 'default_public_interface_only', 'disable-non-proxied-udp': '--force-webrtc-ip-handling-policy=' 'disable_non_proxied_udp', }, 'qt.process_model': { 'process-per-site-instance': None, 'process-per-site': '--process-per-site', 'single-process': '--single-process', }, 'qt.low_end_device_mode': { 'auto': None, 'always': '--enable-low-end-device-mode', 'never': '--disable-low-end-device-mode', }, 'content.headers.referer': { 'always': None, }, 'content.prefers_reduced_motion': { True: '--force-prefers-reduced-motion', False: None, }, } qt_514_ver = utils.VersionNumber(5, 14) if qt_514_ver <= versions.webengine < utils.VersionNumber(5, 15, 2): # In Qt 5.14 to 5.15.1, `--force-dark-mode` is used to set the # preferred colorscheme. In Qt 5.15.2, this is handled by a # blink-setting in browser/webengine/darkmode.py instead. settings['colors.webpage.preferred_color_scheme'] = { 'dark': '--force-dark-mode', 'light': None, 'auto': None, } referrer_setting = settings['content.headers.referer'] if versions.webengine >= qt_514_ver: # Starting with Qt 5.14, this is handled via --enable-features referrer_setting['same-domain'] = None else: referrer_setting['same-domain'] = '--reduced-referrer-granularity' # WORKAROUND for https://bugreports.qt.io/browse/QTBUG-60203 can_override_referer = ( versions.webengine >= utils.VersionNumber(5, 12, 4) and versions.webengine != utils.VersionNumber(5, 13) ) referrer_setting['never'] = None if can_override_referer else '--no-referrers' for setting, args in sorted(settings.items()): arg = args[config.instance.get(setting)] if arg is not None: yield arg def _warn_qtwe_flags_envvar() -> None: """Warn about the QTWEBENGINE_CHROMIUM_FLAGS envvar if it is set.""" qtwe_flags_var = 'QTWEBENGINE_CHROMIUM_FLAGS' qtwe_flags = os.environ.get(qtwe_flags_var) if qtwe_flags is not None: log.init.warning( f"You have {qtwe_flags_var}={qtwe_flags!r} set in your environment. " "This is currently unsupported and interferes with qutebrowser's own " "flag handling (including workarounds for certain crashes). " "Consider using the qt.args qutebrowser setting instead.") def init_envvars() -> None: """Initialize environment variables which need to be set early.""" if objects.backend == usertypes.Backend.QtWebEngine: software_rendering = config.val.qt.force_software_rendering if software_rendering == 'software-opengl': os.environ['QT_XCB_FORCE_SOFTWARE_OPENGL'] = '1' elif software_rendering == 'qt-quick': os.environ['QT_QUICK_BACKEND'] = 'software' elif software_rendering == 'chromium': os.environ['QT_WEBENGINE_DISABLE_NOUVEAU_WORKAROUND'] = '1' _warn_qtwe_flags_envvar() else: assert objects.backend == usertypes.Backend.QtWebKit, objects.backend if config.val.qt.force_platform is not None: os.environ['QT_QPA_PLATFORM'] = config.val.qt.force_platform if config.val.qt.force_platformtheme is not None: os.environ['QT_QPA_PLATFORMTHEME'] = config.val.qt.force_platformtheme if config.val.window.hide_decoration: os.environ['QT_WAYLAND_DISABLE_WINDOWDECORATION'] = '1' if config.val.qt.highdpi: env_var = ('QT_ENABLE_HIGHDPI_SCALING' if qtutils.version_check('5.14', compiled=False) else 'QT_AUTO_SCREEN_SCALE_FACTOR') os.environ[env_var] = '1' for var, val in config.val.qt.environ.items(): if val is None and var in os.environ: del os.environ[var] elif val is not None: os.environ[var] = val