Source code for pymodaq_gui.parameter.ioxml

from __future__ import annotations

from typing import Union
from pathlib import Path

import importlib
import json
from pathlib import Path
from xml.etree import ElementTree as ET
from collections import OrderedDict
from qtpy import QtGui
from qtpy.QtCore import QDateTime, QTime
from pymodaq_gui.parameter import Parameter

from pyqtgraph.parametertree.Parameter import PARAM_TYPES, PARAM_NAMES


VALID_FOR_CONFIGURATION = 'valid_for_configuration'


[docs] def walk_parameters_to_xml(parent_elt=None, param=None): """ To convert a parameter object (and children) to xml data tree. =============== ================================ ================================== **Parameters** **Type** **Description** *parent_elt* XML element the root element *param* instance of pyqtgraph parameter Parameter object to be converted =============== ================================ ================================== Returns ------- XML element : parent_elt XML element with subelements from Parameter object See Also -------- add_text_to_elt, walk_parameters_to_xml, dict_from_param """ if type(param) is None: raise TypeError('No valid param input') if parent_elt is None: opts = dict_from_param(param) parent_elt = ET.Element(param.name(), **opts) if 'value' in param.opts: add_text_to_elt(parent_elt, param) params_list = param.children() for param in params_list: opts = dict_from_param(param) elt = ET.Element(param.name(), **opts) if param.hasChildren(): walk_parameters_to_xml(elt, param) if 'value' in param.opts: add_text_to_elt(elt, param) parent_elt.append(elt) return parent_elt
[docs] def add_text_to_elt(elt, param): """Add a text filed in a xml element corresponding to the parameter value Parameters ---------- elt: XML elt param: Parameter See Also -------- add_text_to_elt, walk_parameters_to_xml, dict_from_param """ param_type = str(param.type()) param_val = param.value() if 'bool' in param_type or 'led' in param_type: if param_val: text = '1' else: text = '0' elif param_type == 'itemselect': if param_val is not None: elt.set('all_items', str(param_val['all_items'])) # use list(eval(val_str[1:-1])) to get back a list of strings text = str(param_val['selected']) # use list(eval(val_str[1:-1])) to get back a list of strings else: text = str(None) elif param_type == 'color': text = str([param_val.red(), param_val.green(), param_val.blue(), param_val.alpha()]) elif param_type == 'list': if isinstance(param_val, str): text = f"str({param_val!r})" # Export repr() for hangling non-printable characters elif isinstance(param_val, int): text = f"int({param_val})" elif isinstance(param_val, float): text = f"float({param_val})" else: text = str(param_val) elif param_type == 'int': if param_val is True: # known bug is True should be clearly specified here val = 1 else: val = param_val text = str(val) elif param_type == 'date_time': text = str(param_val.toMSecsSinceEpoch()) elif param_type == 'date': text = str(QDateTime(param_val, QTime()).toMSecsSinceEpoch()) elif param_type == 'table_view': try: data = dict(classname=param_val.__class__.__name__, module=param_val.__class__.__module__, data=param_val.get_data_all(), header=param_val.header) text = json.dumps(data) except Exception: text = '' else: text = str(param_val) elt.text = text
[docs] def dict_from_param(param): """Get Parameter properties as a dictionary Parameters ---------- param: Parameter Returns ------- opts: dict See Also -------- add_text_to_elt, walk_parameters_to_xml, dict_from_param """ opts = dict([]) param_type = str(param.type()) opts.update(dict(type=param_type)) title = param.opts['title'] if title is None: title = param.name() opts.update(dict(title=title)) visible = '1' if 'visible' in param.opts: if param.opts['visible']: visible = '1' else: visible = '0' opts.update(dict(visible=visible)) removable = '1' if 'removable' in param.opts: if param.opts['removable']: removable = '1' else: removable = '0' opts.update(dict(removable=removable)) readonly = '0' if 'readonly' in param.opts: if param.opts['readonly']: readonly = '1' else: readonly = '0' opts.update(dict(readonly=readonly)) if VALID_FOR_CONFIGURATION in param.opts: opts.update({VALID_FOR_CONFIGURATION: '1' if param.opts[VALID_FOR_CONFIGURATION] else '0'}) # if 'limits' in param.opts: # values = str(param.opts['limits']) # opts.update(dict(values=values)) if 'limits' in param.opts: limits = str(param.opts['limits']) opts.update(dict(limits=limits)) # opts.update(dict(values=limits)) if 'addList' in param.opts: addList = str(param.opts['addList']) opts.update(dict(addList=addList)) if 'addText' in param.opts: addText = str(param.opts['addText']) opts.update(dict(addText=addText)) if 'detlist' in param.opts: detlist = str(param.opts['detlist']) opts.update(dict(detlist=detlist)) if 'movelist' in param.opts: movelist = str(param.opts['movelist']) opts.update(dict(movelist=movelist)) if 'label' in param.opts: label = str(param.opts['label']) opts.update(dict(label=label)) if 'suffix' in param.opts: suffix = str(param.opts['suffix']) opts.update(dict(suffix=suffix)) if 'show_pb' in param.opts: if param.opts['show_pb']: show_pb = '1' else: show_pb = '0' opts.update(dict(show_pb=show_pb)) if 'filetype' in param.opts: if param.opts['filetype']: filetype = '1' else: filetype = '0' opts.update(dict(filetype=filetype)) return opts
[docs] def elt_to_dict(el): """Convert xml element attributes to a dictionnary Parameters ---------- el Returns ------- """ param = dict([]) # name=el.tag, title=title, type=param_type, value=param_value, values=[param_value], # visible=visible, removable=removable, readonly=readonly, show_pb=show_pb) param.update(dict(name=el.tag)) param_type = el.get('type') param.update(dict(type=param_type)) title = el.get('title') if title == 'None': title = el.tag param.update(dict(title=title)) if 'visible' not in el.attrib.keys(): visible = True else: visible = bool(int(el.get('visible'))) param.update(dict(visible=visible)) if 'removable' not in el.attrib.keys(): removable = False else: removable = bool(int(el.get('removable'))) param.update(dict(removable=removable)) if 'readonly' not in el.attrib.keys(): readonly = False else: readonly = bool(int(el.get('readonly'))) param.update(dict(readonly=readonly)) if VALID_FOR_CONFIGURATION in el.attrib.keys(): valid = bool(int(el.get(VALID_FOR_CONFIGURATION))) param.update({VALID_FOR_CONFIGURATION: valid}) if 'show_pb' in el.attrib.keys(): show_pb = bool(int(el.get('show_pb'))) else: show_pb = False param.update(dict(show_pb=show_pb)) if 'readonly' not in el.attrib.keys(): readonly = False else: readonly = bool(int(el.get('readonly'))) param.update(dict(readonly=readonly)) if 'filetype' in el.attrib.keys(): filetype = bool(int(el.get('filetype'))) param.update(dict(filetype=filetype)) if 'detlist' in el.attrib.keys(): detlist = eval(el.get('detlist')) param.update(dict(detlist=detlist)) if 'movelist' in el.attrib.keys(): movelist = eval(el.get('movelist')) param.update(dict(movelist=movelist)) if 'addList' in el.attrib.keys(): addList = eval(el.get('addList')) param.update(dict(addList=addList)) if 'addText' in el.attrib.keys(): addText = str(el.get('addText')) param.update(dict(addText=addText)) if 'label' in el.attrib.keys(): label = str(el.get('label')) param.update(dict(label=label)) if 'suffix' in el.attrib.keys(): suffix = str(el.get('suffix')) param.update(dict(suffix=suffix)) # if 'limits' in el.attrib.keys(): # try: # values = list(eval(el.get('limits'))) # make sure the evaluated values are returned as list (in case another # # iterator type has been used # param.update(dict(values=values)) # except: # pass if 'limits' in el.attrib.keys(): try: limits = eval(el.get('limits')) param.update(dict(limits=limits)) except: pass return param
[docs] def parameter_to_xml_string(param): """ Convert a Parameter to a XML string. Parameters ---------- param: Parameter Returns ------- str: XMl string See Also -------- add_text_to_elt, walk_parameters_to_xml, dict_from_param Examples -------- >>> from pyqtgraph.parametertree import Parameter >>> #Create an instance of Parameter >>> settings=Parameter(name='settings') >>> converted_xml=parameter_to_xml_string(settings) >>> # The converted Parameter >>> print(converted_xml) b'<settings title="settings" type="None" />' """ xml_elt = walk_parameters_to_xml(param=param) return ET.tostring(xml_elt)
[docs] def parameter_to_xml_file(param, filename: Union[str, Path], overwrite=True): """ Convert the given parameter to XML element and update the given XML file. =============== ================================= ================================ **Parameters** **Type** **Description** *param* instance of pyqtgraph parameter the parameter to be added *filename* string the filename of the XML file *overwrite* boolean raise Error is False and file exists =============== ================================= ================================ See Also -------- walk_parameters_to_xml Examples -------- """ if not isinstance(filename, Path): filename = Path(filename) parent = filename.parent filename = filename.stem fname = parent.joinpath(filename + ".xml") # forcing the right extension on the filename xml_elt = walk_parameters_to_xml(param=param) tree = ET.ElementTree(xml_elt) if not overwrite and fname.exists() and fname.is_file(): raise FileExistsError tree.write(str(fname))
[docs] def walk_xml_to_parameter(params=[], XML_elt=None): """ To convert an XML element (and children) to list of dict enabling creation of parameter object. =============== ================== ======================================= **Parameters** **Type** **Description** *params* dictionnary list the list to create parameter object *XML_elt* XML object the XML object to be converted =============== ================== ======================================= Returns ------- params : dictionnary list list of dict to create parameter object Examples ------- >>> from pyqtgraph.parametertree import Parameter, ParameterItem >>> import xml.etree.ElementTree as ET >>> tree = ET.parse('text_bis.xml') >>> root = tree.getroot() >>> params=walk_xml_to_parameter(XML_elt=root) >>> settings_xml=Parameter.create(name='Settings XML', type='group', children=params) >>> settings=Parameter.create(name='Settings', type='group', children=params) See Also -------- walk_parameters_to_xml """ try: if type(XML_elt) is not ET.Element: raise TypeError('not valid XML element') if len(XML_elt) == 0: param_dict = set_dict_from_el(XML_elt) params.append(param_dict) for el in XML_elt: param_dict = set_dict_from_el(el) if param_dict['type'] not in PARAM_TYPES: param_dict['type'] = 'group' # in case the custom group has been defined somewhere but not # registered again in this session if len(el) == 0: children = [] else: subparams = [] children = walk_xml_to_parameter(subparams, el) param_dict['children'] = children params.append(param_dict) except Exception as e: # to be able to debug when there's an issue raise e return params
[docs] def set_dict_from_el(el): """Convert an element into a dict ---------- el: xml element param_dict: dictionnary from which the parameter will be constructed """ param_dict = elt_to_dict(el) set_txt_from_elt(el, param_dict) return param_dict
[docs] def set_txt_from_elt(el, param_dict): """ get the value of the parameter from the text value of the xml element Parameters ---------- el: xml element param_dict: dictionnary from which the parameter will be constructed """ val_text = el.text param_type = param_dict['type'] # param_type = el.get('type') # Redundancy (param_dict already has this attribute) if val_text is not None: if param_type == 'float': param_value = float(val_text) elif param_type == 'int': param_value = int(float(val_text)) elif param_type == 'slide': param_value = float(val_text) elif param_type == 'itemselect': if val_text == 'None': param_value = dict(all_items=[], selected=[]) else: param_value = dict(all_items=eval(el.get('all_items', val_text)), selected=eval(val_text)) elif 'bool' in param_type or 'led' in param_type: # covers 'bool' 'bool_push', 'led' and 'led_push'types param_value = bool(int(val_text)) elif param_type == 'date_time': param_value = QDateTime.fromMSecsSinceEpoch(int(val_text)) elif param_type == 'date': param_value = QDateTime.fromMSecsSinceEpoch(int(val_text)).date() elif param_type == 'table': param_value = eval(val_text) elif param_type == 'color': param_value = QtGui.QColor(*eval(val_text)) elif param_type == 'list': try: param_value = eval(val_text) except Exception: param_value = val_text # for back compatibility elif param_type == 'table_view': data_dict = json.loads(val_text) mod = importlib.import_module(data_dict['module']) _cls = getattr(mod, data_dict['classname']) param_value = _cls(data_dict['data'], header=data_dict['header']) elif param_type == 'action': if val_text == 'None': param_value = None else: param_value = eval(val_text) else: param_value = val_text param_dict.update(dict(value=param_value)) else: if param_type == 'str': param_dict.update(dict(value=''))
[docs] def XML_file_to_parameter(file_name: Union[str, Path]) -> list: """ Convert a xml file into pyqtgraph parameter object. Returns ------- params : list of dictionary a list of dictionary defining a Parameter object and its children See Also -------- walk_parameters_to_xml Examples -------- """ tree = ET.parse(str(file_name)) root = tree.getroot() params = walk_xml_to_parameter(params=[], XML_elt=root) return params
[docs] def xml_file_to_parameter_dict(file_name: Union[str, Path]) -> dict: tree = ET.parse(str(file_name)) root = tree.getroot() param_dict = set_dict_from_el(root) if len(root) > 0: param_dict['children'] = walk_xml_to_parameter(params=[], XML_elt=root) return param_dict
[docs] def XML_string_to_parameter(xml_string): """ Convert a xml string into a list of dict for initialize pyqtgraph parameter object. """ root = ET.fromstring(xml_string) params = walk_xml_to_parameter(params=[], XML_elt=root) return params
[docs] def xml_string_to_parameter_dict(xml_string) -> dict: """ Convert a xml string into a dict to initialize pyqtgraph parameter object. """ root = ET.fromstring(xml_string) tree = ET.ElementTree(root) param_dict = set_dict_from_el(root) if len(root) > 0: param_dict['children'] = walk_xml_to_parameter(params=[], XML_elt=root) return param_dict
[docs] def xml_string_to_parameter(xml_string) -> Parameter: return Parameter.create(**xml_string_to_parameter_dict(xml_string))
[docs] def XML_string_to_pobject(xml_string) -> Parameter: """ return a Parameter object from its deserialized version from a XML string Deprecated as not symetric with parameter_to_xml_string Parameters ---------- xml_string: (str) string representation of a Parameter Object """ return Parameter.create(name='settings', type='group', children=XML_string_to_parameter(xml_string))