Source code for wolfhece.PyConfig

"""
Author: HECE - University of Liege, Pierre Archambeau
Date: 2024

Copyright (c) 2024 University of Liege. All rights reserved.

This script and its content are protected by copyright law. Unauthorized
copying or distribution of this file, via any medium, is strictly prohibited.
"""

import os
import json
from enum import Enum
from pathlib import Path
import logging

import wx
from .PyTranslate import _

[docs] class ConfigurationKeys(Enum): """ Using enumerated keys make sure we can check value names at code write time (i.e. we don't use string which are brittle) """
[docs] VERSION = "Version"
[docs] PLAY_WELCOME_SOUND = "PlayWelcomeSound"
[docs] TICKS_SIZE = "TicksSize"
[docs] TICKS_BOUNDS = "TicksBounds"
[docs] COLOR_BACKGROUND = "ColorBackground"
[docs] ACTIVE_ARRAY_PALETTE_FOR_IMAGE = "Use active array palette for image"
[docs] ACTIVE_RES2D_PALETTE_FOR_IMAGE = "Use active result palette for image"
[docs] ASSEMBLY_IMAGES = "AssemblyImages"
[docs] class WolfConfiguration: """ Holds the PyWolf configuration """ def __init__(self, path=None): # We make sure we use a standard location # to store the configuration if path is None: if os.name == "nt": # On Windows NT, LOCALAPPDATA is expected to be defined. # (might not be true in the future, who knows) self._options_file_path = Path(os.getenv("LOCALAPPDATA")) / "wolf.conf" else: self._options_file_path = Path("wolf.conf") else: self._options_file_path = path #Set default -- useful if new options are inserted # --> ensuring that default values are created even if not stored in the options file self.set_default_config() if self._options_file_path.exists(): self.load() else: # self.set_default_config() # This save is not 100% necessary but it helps # to make sure a config file exists. self.save() @property
[docs] def path(self) -> Path: """ Where the configuration is read/saved.""" return self._options_file_path
[docs] def set_default_config(self): self._config = { ConfigurationKeys.VERSION.value: 1, ConfigurationKeys.PLAY_WELCOME_SOUND.value: True, ConfigurationKeys.TICKS_SIZE.value: 500., ConfigurationKeys.ACTIVE_ARRAY_PALETTE_FOR_IMAGE.value: True, ConfigurationKeys.ACTIVE_RES2D_PALETTE_FOR_IMAGE.value: False, ConfigurationKeys.TICKS_BOUNDS.value: True, ConfigurationKeys.COLOR_BACKGROUND.value: [255, 255, 255, 255], ConfigurationKeys.ASSEMBLY_IMAGES.value: 0 } self._types = { ConfigurationKeys.VERSION.value: int, ConfigurationKeys.PLAY_WELCOME_SOUND.value: bool, ConfigurationKeys.TICKS_SIZE.value: float, ConfigurationKeys.ACTIVE_ARRAY_PALETTE_FOR_IMAGE.value: bool, ConfigurationKeys.ACTIVE_RES2D_PALETTE_FOR_IMAGE.value: bool, ConfigurationKeys.TICKS_BOUNDS.value: bool, ConfigurationKeys.COLOR_BACKGROUND.value: list, ConfigurationKeys.ASSEMBLY_IMAGES.value: int } self._check_config()
[docs] def _check_config(self): assert self._config.keys() == self._types.keys() for idx, (key,val) in enumerate(self._config.items()): assert type(val) == self._types[key]
[docs] def load(self): with open(self._options_file_path, "r", encoding="utf-8") as configfile: filecfg = json.loads(configfile.read()) for curkey in filecfg.keys(): if curkey in self._config.keys(): self._config[curkey] = filecfg[curkey] self._check_config()
[docs] def save(self): # Make sure to write the config file only if it can # be dumped by JSON. txt = json.dumps(self._config, indent=1) with open(self._options_file_path, "w", encoding="utf-8") as configfile: configfile.write(txt)
def __getitem__(self, key: ConfigurationKeys): assert isinstance(key, ConfigurationKeys), "Please only use enum's for configuration keys." return self._config[key.value] def __setitem__(self, key: ConfigurationKeys, value): # A half-measure to ensure the config structure # can be somehow validated before run time. assert isinstance(key, ConfigurationKeys), "Please only use enum's for configuration keys." self._config[key.value] = value self._check_config()
[docs] class GlobalOptionsDialog(wx.Dialog): """ A dialog to set global options for a WolfMapViewer. """ def __init__(self, *args, **kw): super(GlobalOptionsDialog, self).__init__(*args, **kw) self.InitUI() self.SetSize((600, 400)) self.SetTitle(_("Global options"))
[docs] def push_configuration(self, configuration): self.cfg_welcome_voice.SetValue(configuration[ConfigurationKeys.PLAY_WELCOME_SOUND]) self.cfg_ticks_size.SetValue(str(configuration[ConfigurationKeys.TICKS_SIZE])) self.cfg_ticks_bounds.SetValue(configuration[ConfigurationKeys.TICKS_BOUNDS]) self.cfg_bkg_color.SetColour(configuration[ConfigurationKeys.COLOR_BACKGROUND]) self.cfg_active_array_pal.SetValue(configuration[ConfigurationKeys.ACTIVE_ARRAY_PALETTE_FOR_IMAGE]) self.cfg_active_res_pal.SetValue(configuration[ConfigurationKeys.ACTIVE_RES2D_PALETTE_FOR_IMAGE]) self.cfg_assembly_images.SetSelection(configuration[ConfigurationKeys.ASSEMBLY_IMAGES])
[docs] def pull_configuration(self, configuration): configuration[ConfigurationKeys.PLAY_WELCOME_SOUND] = self.cfg_welcome_voice.IsChecked() configuration[ConfigurationKeys.TICKS_SIZE] = float(self.cfg_ticks_size.Value) configuration[ConfigurationKeys.TICKS_BOUNDS] = self.cfg_ticks_bounds.IsChecked() configuration[ConfigurationKeys.COLOR_BACKGROUND] = list(self.cfg_bkg_color.GetColour()) configuration[ConfigurationKeys.ACTIVE_ARRAY_PALETTE_FOR_IMAGE] = self.cfg_active_array_pal.IsChecked() configuration[ConfigurationKeys.ACTIVE_RES2D_PALETTE_FOR_IMAGE] = self.cfg_active_res_pal.IsChecked() configuration[ConfigurationKeys.ASSEMBLY_IMAGES] = self.cfg_assembly_images.GetSelection()
[docs] def InitUI(self): vbox = wx.BoxSizer(wx.VERTICAL) # Panel 'Miscellaneous' pnl = wx.ScrolledWindow(self) sb = wx.StaticBox(pnl, label=_('Miscellaneous')) sbs = wx.StaticBoxSizer(sb , orient=wx.VERTICAL) self.cfg_welcome_voice = wx.CheckBox(pnl, label=_('Welcome voice')) self.cfg_welcome_voice.SetToolTip(_('Play a welcome message when opening the application')) sbs.Add(self.cfg_welcome_voice) sbs.AddSpacer(5) hsizer = wx.BoxSizer(wx.HORIZONTAL) self.label_background_color = wx.StaticText(pnl, label=_('Background color')) self.cfg_bkg_color = wx.ColourPickerCtrl(pnl, colour=(255,255,255,255)) self.cfg_bkg_color.SetToolTip(_('Background color for the viewer')) hsizer.Add(self.label_background_color, 1, wx.EXPAND) hsizer.Add(self.cfg_bkg_color, 1, wx.EXPAND) sbs.Add(hsizer, 1, wx.EXPAND) sbs.AddSpacer(5) pnl.SetSizer(sbs) pnl.Layout() vbox.Add(pnl, proportion=1, flag=wx.ALL|wx.EXPAND, border=5) ### Panel 'Copy to clipboard' pnl = wx.ScrolledWindow(self) sb = wx.StaticBox(pnl, label=_('Copy to clipboard')) sbs = wx.StaticBoxSizer(sb, orient=wx.VERTICAL) hboxticks = wx.BoxSizer(wx.HORIZONTAL) self.label_ticks_size = wx.StaticText(pnl, label=_('Default ticks size [m]')) self.cfg_ticks_size = wx.TextCtrl(pnl, value='500.',style = wx.TE_CENTRE ) hboxticks.Add(self.label_ticks_size, 1, wx.EXPAND) hboxticks.Add(self.cfg_ticks_size, 1, wx.EXPAND, 5) sbs.Add(hboxticks, 1, wx.EXPAND,5) self.cfg_ticks_bounds = wx.CheckBox(pnl, label=_('Add bounds to ticks')) self.cfg_ticks_bounds.SetToolTip(_('If not checked, the extreme values of the ticks will not be displayed')) sbs.Add(self.cfg_ticks_bounds, 1, wx.EXPAND, 5) self.cfg_active_array_pal = wx.CheckBox(pnl, label=_('Use active array palette for image')) self.cfg_active_array_pal.SetToolTip(_('If checked, the active array palette will be used for the image')) sbs.Add(self.cfg_active_array_pal, 1, wx.EXPAND, 5) self.cfg_active_res_pal = wx.CheckBox(pnl, label=_('Use active result palette for image')) self.cfg_active_res_pal.SetToolTip(_('If checked, the active result palette will be used for the image (but priority to active array palette if checked)')) sbs.Add(self.cfg_active_res_pal, 1, wx.EXPAND, 5) locsizer = wx.BoxSizer(wx.HORIZONTAL) self.label_assembly_images = wx.StaticText(pnl, label=_('Assembly mode for images (if linked viewers)')) self.cfg_assembly_images = wx.ListBox(pnl, choices=['horizontal', 'vertical', 'square'], style=wx.LB_SINGLE) self.cfg_assembly_images.SetToolTip(_('Choose the assembly mode for images -- horizontal, vertical or square')) locsizer.Add(self.label_assembly_images, 1, wx.EXPAND, 2) locsizer.Add(self.cfg_assembly_images, 1, wx.EXPAND,5) sbs.Add(locsizer, 3, wx.EXPAND, 5) pnl.SetSizer(sbs) pnl.Layout() vbox.Add(pnl, proportion=1, flag=wx.ALL|wx.EXPAND, border=5) # Buttons hbox2 = wx.BoxSizer(wx.HORIZONTAL) okButton = wx.Button(self, wx.ID_OK, label=_('Ok')) okButton.SetDefault() closeButton = wx.Button(self, label=_('Close')) hbox2.Add(okButton) hbox2.Add(closeButton, flag=wx.LEFT, border=5) vbox.Add(hbox2, flag=wx.ALIGN_CENTER|wx.TOP|wx.BOTTOM, border=10) self.SetSizer(vbox) self.Layout() okButton.Bind(wx.EVT_BUTTON, self.OnOk) closeButton.Bind(wx.EVT_BUTTON, self.OnClose)
[docs] def OnOk(self, e): if self.IsModal(): self.EndModal(wx.ID_OK) else: self.Close()
[docs] def OnClose(self, e): self.Destroy()
[docs] def handle_configuration_dialog(wxparent, configuration): dlg = GlobalOptionsDialog(wxparent) try: dlg.push_configuration(configuration) if dlg.ShowModal() == wx.ID_OK: # do something here dlg.pull_configuration(configuration) configuration.save() logging.info(_('Configuration saved in {}').format(str(configuration.path))) else: # handle dialog being cancelled or ended by some other button pass finally: # explicitly cause the dialog to destroy itself dlg.Destroy()
if __name__ == "__main__":
[docs] cfg = WolfConfiguration(Path("test.conf"))
cfg[ConfigurationKeys.PLAY_WELCOME_SOUND] = False print(cfg._config) cfg.save() cfg = WolfConfiguration(Path("test.conf")) cfg.load() print(cfg[ConfigurationKeys.PLAY_WELCOME_SOUND])