mirror of
				https://github.com/saitohirga/WSJT-X.git
				synced 2025-10-22 16:40:24 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			474 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			474 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/env python
 | |
| # -*- coding: utf-8 -*-
 | |
| 
 | |
| """QDarkStyle is a dark stylesheet for Python and Qt applications.
 | |
| 
 | |
| This module provides a function to transparently load the stylesheets
 | |
| with the correct rc file.
 | |
| 
 | |
| First, start importing our module
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     import qdarkstyle
 | |
| 
 | |
| Then you can get stylesheet provided by QDarkStyle for various Qt wrappers
 | |
| as shown bellow
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     # PySide
 | |
|     dark_stylesheet = qdarkstyle.load_stylesheet_pyside()
 | |
|     # PySide 2
 | |
|     dark_stylesheet = qdarkstyle.load_stylesheet_pyside2()
 | |
|     # PyQt4
 | |
|     dark_stylesheet = qdarkstyle.load_stylesheet_pyqt()
 | |
|     # PyQt5
 | |
|     dark_stylesheet = qdarkstyle.load_stylesheet_pyqt5()
 | |
| 
 | |
| Or from environment variables provided for QtPy or PyQtGraph, see
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     # QtPy
 | |
|     dark_stylesheet = qdarkstyle.load_stylesheet_from_environment()
 | |
|     # PyQtGraph
 | |
|     dark_stylesheet = qdarkstyle.load_stylesheet_from_environment(is_pyqtgraph)
 | |
| 
 | |
| Finally, set your QApplication with it
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     app.setStyleSheet(dark_stylesheet)
 | |
| 
 | |
| Enjoy!
 | |
| 
 | |
| """
 | |
| 
 | |
| import logging
 | |
| import os
 | |
| import platform
 | |
| import sys
 | |
| import warnings
 | |
| import copy
 | |
| 
 | |
| if sys.version_info >= (3, 4):
 | |
|     import importlib
 | |
| 
 | |
| __version__ = "2.6.5"
 | |
| 
 | |
| 
 | |
| QT_BINDINGS = ['PyQt4', 'PyQt5', 'PySide', 'PySide2']
 | |
| """list: values of all Qt bindings to import."""
 | |
| 
 | |
| QT_ABSTRACTIONS = ['qtpy', 'pyqtgraph', 'Qt']
 | |
| """list: values of all Qt abstraction layers to import."""
 | |
| 
 | |
| QT4_IMPORT_API = ['QtCore', 'QtGui']
 | |
| """list: which subpackage to import for Qt4 API."""
 | |
| 
 | |
| QT5_IMPORT_API = ['QtCore', 'QtGui', 'QtWidgets']
 | |
| """list: which subpackage to import for Qt5 API."""
 | |
| 
 | |
| QT_API_VALUES = ['pyqt', 'pyqt5', 'pyside', 'pyside2']
 | |
| """list: values for QT_API environment variable used by QtPy."""
 | |
| 
 | |
| QT_LIB_VALUES = ['PyQt', 'PyQt5', 'PySide', 'PySide2']
 | |
| """list: values for PYQTGRAPH_QT_LIB environment variable used by PyQtGraph."""
 | |
| 
 | |
| QT_BINDING = 'Not set or nonexistent'
 | |
| """str: Qt binding in use."""
 | |
| 
 | |
| QT_ABSTRACTION = 'Not set or nonexistent'
 | |
| """str: Qt abstraction layer in use."""
 | |
| 
 | |
| 
 | |
| def _logger():
 | |
|     return logging.getLogger('qdarkstyle')
 | |
| 
 | |
| 
 | |
| def _qt_wrapper_import(qt_api):
 | |
|     """
 | |
|     Check if Qt API defined can be imported.
 | |
| 
 | |
|     :param qt_api: Qt API string to test import
 | |
| 
 | |
|     :return load function fot given qt_api, otherwise empty string
 | |
| 
 | |
|     """
 | |
|     qt_wrapper = ''
 | |
|     loader = ""
 | |
| 
 | |
|     try:
 | |
|         if qt_api == 'PyQt' or qt_api == 'pyqt':
 | |
|             import PyQt4
 | |
|             qt_wrapper = 'PyQt4'
 | |
|             loader = load_stylesheet_pyqt()
 | |
|         elif qt_api == 'PyQt5' or qt_api == 'pyqt5':
 | |
|             import PyQt5
 | |
|             qt_wrapper = 'PyQt5'
 | |
|             loader = load_stylesheet_pyqt5()
 | |
|         elif qt_api == 'PySide' or qt_api == 'pyside':
 | |
|             import PySide
 | |
|             qt_wrapper = 'PySide'
 | |
|             loader = load_stylesheet_pyside()
 | |
|         elif qt_api == 'PySide2' or qt_api == 'pyside2':
 | |
|             import PySide2
 | |
|             qt_wrapper = 'PySide2'
 | |
|             loader = load_stylesheet_pyside2()
 | |
|     except ImportError as err:
 | |
|         _logger().error("Impossible import Qt wrapper.\n %s", str(err))
 | |
|     else:
 | |
|         _logger().info("Using Qt wrapper = %s ", qt_wrapper)
 | |
|         QT_BINDING = qt_wrapper
 | |
|     finally:
 | |
|         return loader
 | |
| 
 | |
| 
 | |
| def load_stylesheet_from_environment(is_pyqtgraph=False):
 | |
|     """
 | |
|     Load the stylesheet from QT_API (or PYQTGRAPH_QT_LIB) environment variable.
 | |
| 
 | |
|     :param is_pyqtgraph: True if it is to be set using PYQTGRAPH_QT_LIB
 | |
| 
 | |
|     :raise KeyError: if QT_API/PYQTGRAPH_QT_LIB does not exist
 | |
| 
 | |
|     :return the stylesheet string
 | |
|     """
 | |
|     warnings.warn(
 | |
|         "load_stylesheet_from_environment() will be deprecated in version 3,"
 | |
|         "use load_stylesheet()",
 | |
|         PendingDeprecationWarning
 | |
|     )
 | |
|     qt_api = ''
 | |
|     pyqtgraph_qt_lib = ''
 | |
| 
 | |
|     loader = ""
 | |
| 
 | |
|     # Get values from QT_API
 | |
|     try:
 | |
|         qt_api = os.environ['QT_API']
 | |
|     except KeyError as err:
 | |
|         # Log this error just if using QT_API
 | |
|         if not is_pyqtgraph:
 | |
|             _logger().error("QT_API does not exist, do os.environ['QT_API']= "
 | |
|                             "and choose one option from %s", QT_API_VALUES)
 | |
|     else:
 | |
|         if not is_pyqtgraph:
 | |
|             if qt_api in QT_API_VALUES:
 | |
|                 QT_ABSTRACTION = "qtpy"
 | |
|                 _logger().info("Found QT_API='%s'", qt_api)
 | |
|                 loader = _qt_wrapper_import(qt_api)
 | |
|             else:
 | |
|                 # Raise this error because the function need this key/value
 | |
|                 raise KeyError("QT_API=%s is unknown, please use a value "
 | |
|                                "from %s",
 | |
|                                (qt_api, QT_API_VALUES))
 | |
| 
 | |
|     # Get values from PYQTGRAPH_QT_LIB
 | |
|     try:
 | |
|         pyqtgraph_qt_lib = os.environ['PYQTGRAPH_QT_LIB']
 | |
|     except KeyError as err:
 | |
|         # Log this error just if using PYQTGRAPH_QT_LIB
 | |
|         if is_pyqtgraph:
 | |
|             _logger().error("PYQTGRAP_QT_API does not exist, do "
 | |
|                             "os.environ['PYQTGRAPH_QT_LIB']= "
 | |
|                             "and choose one option from %s",
 | |
|                             QT_LIB_VALUES)
 | |
|     else:
 | |
| 
 | |
|         if is_pyqtgraph:
 | |
|             if pyqtgraph_qt_lib in QT_LIB_VALUES:
 | |
|                 QT_ABSTRACTION = "pyqtgraph"
 | |
|                 _logger().info("Found PYQTGRAPH_QT_LIB='%s'", pyqtgraph_qt_lib)
 | |
|                 loader = _qt_wrapper_import(pyqtgraph_qt_lib)
 | |
|             else:
 | |
|                 # Raise this error because the function need this key/value
 | |
|                 raise KeyError("PYQTGRAPH_QT_LIB=%s is unknown, please use a "
 | |
|                                "value from %s", (
 | |
|                                    pyqtgraph_qt_lib,
 | |
|                                    QT_LIB_VALUES))
 | |
| 
 | |
|     # Just a warning if both are set but differs each other
 | |
|     if qt_api and pyqtgraph_qt_lib:
 | |
|         if qt_api != pyqtgraph_qt_lib.lower():
 | |
|             _logger().warning("Both QT_API=%s and PYQTGRAPH_QT_LIB=%s are set, "
 | |
|                               "but with different values, this could cause "
 | |
|                               "some issues if using them in the same project!",
 | |
|                               qt_api, pyqtgraph_qt_lib)
 | |
| 
 | |
|     return loader
 | |
| 
 | |
| 
 | |
| def load_stylesheet(pyside=True):
 | |
|     """
 | |
|     Load the stylesheet. Takes care of importing the rc module.
 | |
| 
 | |
|     :param pyside: True to load the pyside rc file, False to load the PyQt rc file
 | |
| 
 | |
|     :return the stylesheet string
 | |
|     """
 | |
|     warnings.warn(
 | |
|         "load_stylesheet() will not receive pyside parameter in version 3. "
 | |
|         "Set QtPy environment variable to specify the Qt binding insteady.",
 | |
|         FutureWarning
 | |
|     )
 | |
|     # Smart import of the rc file
 | |
| 
 | |
|     pyside_ver = None
 | |
| 
 | |
|     if pyside:
 | |
| 
 | |
|         # Detect the PySide version available
 | |
|         try:
 | |
|             import PySide
 | |
|         except ImportError: # Compatible with py27
 | |
|             import PySide2
 | |
|             pyside_ver = 2
 | |
|         else:
 | |
|             pyside_ver = 1
 | |
| 
 | |
|         if pyside_ver == 1:
 | |
|             import qdarkstyle.pyside_style_rc
 | |
|         else:
 | |
|             import qdarkstyle.pyside2_style_rc
 | |
|     else:
 | |
|         import qdarkstyle.pyqt_style_rc
 | |
| 
 | |
|     # Load the stylesheet content from resources
 | |
|     if not pyside:
 | |
|         from PyQt4.QtCore import QFile, QTextStream
 | |
|     else:
 | |
|         if pyside_ver == 1:
 | |
|             from PySide.QtCore import QFile, QTextStream
 | |
|         else:
 | |
|             from PySide2.QtCore import QFile, QTextStream
 | |
| 
 | |
|     f = QFile(":qdarkstyle/style.qss")
 | |
|     if not f.exists():
 | |
|         _logger().error("Unable to load stylesheet, file not found in "
 | |
|                         "resources")
 | |
|         return ""
 | |
|     else:
 | |
|         f.open(QFile.ReadOnly | QFile.Text)
 | |
|         ts = QTextStream(f)
 | |
|         stylesheet = ts.readAll()
 | |
|         if platform.system().lower() == 'darwin':  # see issue #12 on github
 | |
|             mac_fix = '''
 | |
|             QDockWidget::title
 | |
|             {
 | |
|                 background-color: #32414B;
 | |
|                 text-align: center;
 | |
|                 height: 12px;
 | |
|             }
 | |
|             '''
 | |
|             stylesheet += mac_fix
 | |
|         return stylesheet
 | |
| 
 | |
| 
 | |
| def load_stylesheet_pyside():
 | |
|     """
 | |
|     Load the stylesheet for use in a pyside application.
 | |
| 
 | |
|     :return the stylesheet string
 | |
|     """
 | |
|     warnings.warn(
 | |
|         "load_stylesheet_pyside() will be deprecated in version 3,"
 | |
|         "set QtPy environment variable to specify the Qt binding and "
 | |
|         "use load_stylesheet()",
 | |
|         PendingDeprecationWarning
 | |
|     )
 | |
|     return load_stylesheet(pyside=True)
 | |
| 
 | |
| 
 | |
| def load_stylesheet_pyside2():
 | |
|     """
 | |
|     Load the stylesheet for use in a pyside2 application.
 | |
| 
 | |
|     :raise NotImplementedError: Because it is not supported yet
 | |
|     """
 | |
|     warnings.warn(
 | |
|         "load_stylesheet_pyside2() will be deprecated in version 3,"
 | |
|         "set QtPy environment variable to specify the Qt binding and "
 | |
|         "use load_stylesheet()",
 | |
|         PendingDeprecationWarning
 | |
|     )
 | |
|     return load_stylesheet(pyside=True)
 | |
| 
 | |
| 
 | |
| def load_stylesheet_pyqt():
 | |
|     """
 | |
|     Load the stylesheet for use in a pyqt4 application.
 | |
| 
 | |
|     :return the stylesheet string
 | |
|     """
 | |
|     warnings.warn(
 | |
|         "load_stylesheet_pyqt() will be deprecated in version 3,"
 | |
|         "set QtPy environment variable to specify the Qt binding and "
 | |
|         "use load_stylesheet()",
 | |
|         PendingDeprecationWarning
 | |
|     )
 | |
|     return load_stylesheet(pyside=False)
 | |
| 
 | |
| 
 | |
| def load_stylesheet_pyqt5():
 | |
|     """
 | |
|     Load the stylesheet for use in a pyqt5 application.
 | |
| 
 | |
|     :param pyside: True to load the pyside rc file, False to load the PyQt rc file
 | |
| 
 | |
|     :return the stylesheet string
 | |
|     """
 | |
|     warnings.warn(
 | |
|         "load_stylesheet_pyqt5() will be deprecated in version 3,"
 | |
|         "set QtPy environment variable to specify the Qt binding and "
 | |
|         "use load_stylesheet()",
 | |
|         PendingDeprecationWarning
 | |
|     )
 | |
|     # Smart import of the rc file
 | |
|     import qdarkstyle.pyqt5_style_rc
 | |
| 
 | |
|     # Load the stylesheet content from resources
 | |
|     from PyQt5.QtCore import QFile, QTextStream
 | |
| 
 | |
|     f = QFile(":qdarkstyle/style.qss")
 | |
|     if not f.exists():
 | |
|         _logger().error("Unable to load stylesheet, file not found in "
 | |
|                         "resources")
 | |
|         return ""
 | |
|     else:
 | |
|         f.open(QFile.ReadOnly | QFile.Text)
 | |
|         ts = QTextStream(f)
 | |
|         stylesheet = ts.readAll()
 | |
|         if platform.system().lower() == 'darwin':  # see issue #12 on github
 | |
|             mac_fix = '''
 | |
|             QDockWidget::title
 | |
|             {
 | |
|                 background-color: #32414B;
 | |
|                 text-align: center;
 | |
|                 height: 12px;
 | |
|             }
 | |
|             '''
 | |
|             stylesheet += mac_fix
 | |
|         return stylesheet
 | |
| 
 | |
| 
 | |
| def information():
 | |
|     """Get system and runtime information."""
 | |
|     info = []
 | |
|     qt_api = ''
 | |
|     qt_lib = ''
 | |
|     qt_bin = ''
 | |
| 
 | |
|     try:
 | |
|         qt_api = os.environ['QT_API']
 | |
|     except KeyError:
 | |
|         qt_api = 'Not set or nonexistent'
 | |
| 
 | |
|     try:
 | |
|         from Qt import __binding__
 | |
|     except Exception:
 | |
|         # It should be (KeyError, ModuleNotFoundError, ImportError)
 | |
|         # but each python version have a different one, and not define others
 | |
|         qt_lib = 'Not set or nonexistent'
 | |
|     else:
 | |
|         qt_lib = __binding__
 | |
| 
 | |
|     try:
 | |
|         qt_bin = os.environ['PYQTGRAPH_QT_LIB']
 | |
|     except KeyError:
 | |
|         qt_bin = 'Not set or nonexistent'
 | |
| 
 | |
|     info.append('QDarkStyle: %s' % __version__)
 | |
|     info.append('OS: %s %s %s' % (platform.system(), platform.release(), platform.machine()))
 | |
|     info.append('Platform: %s' % sys.platform)
 | |
|     info.append('Python: %s' % '.'.join(str(e) for e in sys.version_info[:]))
 | |
|     info.append('Python API: %s' % sys.api_version)
 | |
| 
 | |
|     info.append('Binding in use:     %s' % QT_BINDING)
 | |
|     info.append('Abstraction in use: %s' % QT_ABSTRACTION)
 | |
| 
 | |
|     info.append('qtpy (QT_API):                %s' % qt_api)
 | |
|     info.append('pyqtgraph (PYQTGRAPH_QT_LIB): %s' % qt_lib)
 | |
|     info.append('Qt.py (__binding__):          %s' % qt_bin)
 | |
| 
 | |
|     return info
 | |
| 
 | |
| 
 | |
| def qt_bindings():
 | |
|     """Return a list of qt bindings available."""
 | |
|     return _check_imports(import_list=QT_BINDINGS)
 | |
| 
 | |
| 
 | |
| def qt_abstractions():
 | |
|     """Return a list of qt abstraction layers available."""
 | |
|     return _check_imports(import_list=QT_ABSTRACTIONS)
 | |
| 
 | |
| 
 | |
| def _check_imports(import_list):
 | |
|     """Return a list of imports available."""
 | |
| 
 | |
|     # Disable warnings here
 | |
|     warnings.filterwarnings("ignore")
 | |
| 
 | |
|     import_list_return = copy.deepcopy(import_list)
 | |
|     # Using import_list_return var in for, does not work in py2.7
 | |
|     # when removing the element, it reflects on for list
 | |
|     # so it skips next element
 | |
|     for current_import in import_list:
 | |
| 
 | |
|         spec = True
 | |
|         # Copy the sys path to make sure to not insert anything
 | |
|         sys_path = sys.path
 | |
| 
 | |
|         # Check import
 | |
|         if sys.version_info >= (3, 4):
 | |
|             spec = importlib.util.find_spec(current_import)
 | |
|         else:
 | |
|             try:
 | |
|                 __import__(current_import)
 | |
|             except RuntimeWarning:
 | |
|                 spec = True
 | |
|             except Exception:
 | |
|                 spec = None
 | |
|             else:
 | |
|                 spec = True
 | |
| 
 | |
|         if spec is None:
 | |
|             # Remove if not available
 | |
|             import_list_return.remove(current_import)
 | |
| 
 | |
|         # Restore sys path
 | |
|         sys.path = sys_path
 | |
| 
 | |
|     # Restore warnings
 | |
|     warnings.resetwarnings()
 | |
| 
 | |
|     return import_list_return
 | |
| 
 | |
| 
 | |
| def _import_qt_modules_from(use_binding='pyqt5', use_abstraction='qtpy'):
 | |
|     """New approach to import modules using importlib."""
 | |
| 
 | |
|     if not sys.version_info >= (3, 4):
 | |
|         print('Function not available for Python < 3.4')
 | |
| 
 | |
|     spec_binding = importlib.util.find_spec(use_binding)
 | |
|     spec_abstraction = importlib.util.find_spec(use_abstraction)
 | |
| 
 | |
|     if spec_binding is None:
 | |
|         print("Cannot find Qt binding: ", use_binding)
 | |
|     else:
 | |
|         module = importlib.util.module_from_spec(spec_binding)
 | |
|         spec.loader.exec_module(module)
 | |
|         # Adding the module to sys.modules is optional.
 | |
|         sys.modules[name] = module
 | |
| 
 | |
|     if spec_abstraction is None:
 | |
|         print("Cannot find Qt abstraction layer: ", use_abstraction)
 | |
|     else:
 | |
|         module = importlib.util.module_from_spec(spec)
 | |
|         spec.loader.exec_module(module)
 | |
|         # Adding the module to sys.modules is optional.
 | |
|         sys.modules[name] = module
 |