ó
âdc           @` s   d  Z  d d l m Z m Z m Z m Z d d l Td d l m Z	 d d l m
 Z d d d	 g Z d
   Z d   Z e d  Z d   Z d S(   u   Class to generate GIMP PDB parameters out of settings and parse GIMP
procedure arguments to assign them as values to settings.
i    (   t   absolute_importt   divisiont   print_functiont   unicode_literals(   t   *i   (   t   group(   t   settingsu   create_paramsu	   iter_argsu   list_param_valuesc          G` sI   t  |   } g  } x0 | D]( } | j   r | j | j    q q W| S(   u_   
  Return a list of GIMP PDB parameters from the specified `Setting` or `Group`
  instances.
  (   t   _list_settingst   can_be_registered_to_pdbt   extendt   get_pdb_param(   t   settings_or_groupsR   t   paramst   setting(    (    sS   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/pygimplib/setting/pdbparams.pyt   create_params   s    c         c` s   t    } d } xH | D]@ } t | t j  rL | d 7} | j | d  n  | d 7} q Wx; t t t |   |   D] } | | k rv |  | Vqv qv Wd S(   u/  
  Iterate over arguments (`args`) passed to a GIMP PDB procedure.
  
  `settings` is a list of `Setting` instances that may modify the iteration. For
  example, if an argument is matched by a setting of type `ArraySetting`, the
  array argument causes the preceding argument to be skipped. The preceding
  argument is the array length and does not need to exist as a separate setting
  because the length can be obtained from the array itself in Python.
  
  If there are more settings than non-skipped arguments, the remaining settings
  will be ignored.
  i    i   N(   t   sett
   isinstancet	   settings_t   ArraySettingt   addt   ranget   mint   len(   t   argsR   t    indexes_of_array_length_settingst   indexR   t	   arg_index(    (    sS   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/pygimplib/setting/pdbparams.pyt	   iter_args$   s    	
"c         C` sr   t  |   } | rL x7 t |  D]& \ } } | j d k r | | =Pq q Wn  g  | D] } | j   rS | j ^ qS S(   u  
  Return a list of values of settings registrable to PDB.
  
  If `ignore_run_mode` is `True`, ignore setting(s) named `'run_mode'`. This
  makes it possible to call PDB functions with the setting values without
  manually omitting the `'run_mode'` setting.
  u   run_mode(   R   t	   enumeratet   nameR   t   value(   R   t   ignore_run_modeR   t   iR   (    (    sS   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/pygimplib/setting/pdbparams.pyt   list_param_values@   s    c         C` s   g  } xy |  D]q } t  | t j  r5 | j |  q t  | t j  r] | j | j    q t d j	 | t j t j    q W| S(   Nu$   {} is not an object of type {} or {}(
   R   R   t   Settingt   appendt   group_t   GroupR	   t   walkt	   TypeErrort   format(   R   R   t   setting_or_group(    (    sS   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/pygimplib/setting/pdbparams.pyR   S   s    N(   t   __doc__t
   __future__R    R   R   R   t   future.builtinst    R   R$   R   R   t   __all__R   R   t   TrueR!   R   (    (    (    sS   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/pygimplib/setting/pdbparams.pyt   <module>   s   "
			                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                               # -*- coding: utf-8 -*-

"""Wrapper of `setting.sources` module to allow easy loading/saving using
multiple setting sources.
"""

from __future__ import absolute_import, division, print_function, unicode_literals
from future.builtins import *

import collections

from . import _sources_errors

__all__ = [
  'Persistor',
]


class Persistor(object):
  """Loading and saving settings for multiple settings sources at once.
  
  Settings sources are `setting.sources.Source` instances.
  
  Apart from merely loading or saving settings, this class also triggers events
  before/after loading/saving for each setting and exits gracefully even if
  reading/writing settings to a setting source failed.
  """
  
  _STATUSES = SUCCESS, PARTIAL_SUCCESS, SOURCE_NOT_FOUND, FAIL, NO_SETTINGS = (0, 1, 2, 3, 4)
  
  _DEFAULT_SETTING_SOURCES = collections.OrderedDict()
  
  @classmethod
  def get_default_setting_sources(cls):
    """Returns a copy of a dictionary containing default setting sources.
    
    See `set_default_setting_sources()` for more information.
    """
    return collections.OrderedDict(cls._DEFAULT_SETTING_SOURCES.items())
  
  @classmethod
  def set_default_setting_sources(cls, sources):
    """Sets the dictionary of setting sources to use in methods of this class if
    no other setting sources in these methods are specified.
    
    The dictionary must contain pairs of (key, `setting.sources.Source` instance
    or list of `setting.sources.Source` instances).
    
    The key is a string that identifies a group of sources. The key can be
    specified in `setting.settings.Setting` instances within `setting_sources`
    to indicate which groups of sources the setting can be read from or written
    to. For example, if the `setting_sources` attribute of a setting contains
    [`'persistent'`], then only setting sources under the key `'persistent'`
    will be considered and other sources will be ignored. This is useful if you
    need to e.g. save settings to a different file while still ignoring settings
    not containing `'persistent'`.
    """
    if sources is None:
      sources = collections.OrderedDict()
    
    if not isinstance(sources, dict):
      raise TypeError('"sources" must be a dictionary')
    
    cls._DEFAULT_SETTING_SOURCES = sources
  
  @classmethod
  def load(cls, settings_or_groups, setting_sources=None, trigger_events=True):
    """Loads values from the specified settings or groups, or creates new
    settings within the specified groups if they do not exist.
    
    The order of setting sources in `settings_or_groups` indicates the
    preference of the sources, beginning with the first source. If not all
    settings could be found in the first source, the second source is read to
    assign values to the remaining settings. This continues until all sources
    are read or all settings are found.
    
    If the setting source(s) contain an invalid value for a setting, the default
    value for the setting will be assigned.
    Settings not found in any of the sources will also have their default values
    assigned.
    
    The following events are triggered for each `Setting` and `Group` instance
    within `settings_or_groups`, including child settings and groups:
    
    * `'before-load'` - invoked before loading settings. The event will not be
      triggered for settings present in the source but not in memory as they are
      not loaded yet.
    
    * `'after-load'` - invoked after loading settings. The event will also be
      triggered for settings originally not present in memory as they are now
      loaded. This event is triggered even if loading fails for any source.
    
    * events triggered in `Setting.set_value()` when a setting is being loaded.
    
    * events triggered in `Setting.reset()` when loading a setting was not
      successful (occurring when the loaded value was not valid).
    
    Parameters:
    
    * `settings_or_groups` - List of `settings.Setting` or `group.Group`
      instances whose values are loaded from `setting_sources`. For settings and
      groups that exist in `settings_or_groups`, only values are loaded.
      Child settings and groups not present in a group in memory but present in
      the source(s) are created within the group.
    
    * `setting_sources` - Dictionary or list of setting sources or `None`. If a
      dictionary, it must contain (key, setting source) pairs or
      (key, list of setting sources) pairs.
      See `set_default_setting_sources()` for more information on the key.
      If a list, it must contain keys and all keys must have a
      mapping to one of the default sources as returned by
      `get_default_setting_sources()`.
      If `None`, default setting sources are used as returned by
      `get_default_setting_sources()`.
    
    * `trigger_events` - If `True`, trigger `'before-load'` and `'after-load'`
      events for each setting. If `False`, these events are not triggered.
    
    Returns:
    
      A `PersistorResult` instance describing the result, particularly in the
      case of a failure. See `PersistorResult` for more information.
    """
    if not settings_or_groups:
      return cls._result(cls.NO_SETTINGS)
    
    if setting_sources is None:
      setting_sources = cls._DEFAULT_SETTING_SOURCES
    
    if not setting_sources:
      return cls._result(cls.FAIL)
    
    processed_setting_sources = cls._process_setting_sources(setting_sources)
    
    if not processed_setting_sources:
      return cls._result(cls.FAIL)
    
    cls._trigger_event(settings_or_groups, 'before-load', trigger_events)
    
    settings_not_loaded, statuses_per_source, messages_per_source = cls._load(
      settings_or_groups, processed_setting_sources)
    
    cls._trigger_event(settings_or_groups, 'after-load', trigger_events)
    
    return cls._get_return_result(settings_not_loaded, statuses_per_source, messages_per_source)
  
  @classmethod
  def _load(cls, settings_or_groups, setting_sources):
    settings_not_loaded = settings_or_groups
    
    statuses_per_source = {}
    messages_per_source = {}
    
    for unused_, sources in setting_sources.items():
      for source in sources:
        try:
          source.read(settings_not_loaded)
        except _sources_errors.SourceNotFoundError as e:
          statuses_per_source[source] = cls.SOURCE_NOT_FOUND
          messages_per_source[source] = str(e)
        except _sources_errors.SourceError as e:
          statuses_per_source[source] = cls.FAIL
          messages_per_source[source] = str(e)
        else:
          statuses_per_source[source] = cls.SUCCESS
          messages_per_source[source] = ''
          
          if source.settings_not_loaded:
            settings_not_loaded = source.settings_not_loaded
          else:
            return [], statuses_per_source, messages_per_source
    
    return settings_not_loaded, statuses_per_source, messages_per_source
  
  @classmethod
  def save(cls, settings_or_groups, setting_sources=None, trigger_events=True):
    """Saves settings to the specified setting sources.
    
    The following events for each `Setting` instance (including child settings
    in `Group` instances) within `settings_or_groups` are triggered:
    
    * `'before-save'` - invoked before saving settings.
    
    * `'after-save'` - invoked after saving settings. This event is triggered
      even if saving fails for any source.
    
    Parameters:
    
    * `settings_or_groups` - List of `settings.Setting` or `group.Group`
      instances whose values are saved to `setting_sources`.
    
    * `setting_sources` - Dictionary or list of setting sources or `None`. See
      `load()` for more information.
    
    * `trigger_events` - If `True`, trigger `'before-save'` and `'after-save'`
      events for each setting. If `False`, these events are not triggered.
    
    Returns:
    
      A `PersistorResult` instance describing the result, particularly in the
      case of a failure. See `PersistorResult` for more information.
    """
    if not settings_or_groups:
      return cls._result(cls.NO_SETTINGS)
    
    if setting_sources is None:
      setting_sources = cls._DEFAULT_SETTING_SOURCES
    
    if not setting_sources:
      return cls._result(cls.FAIL)
    
    processed_setting_sources = cls._process_setting_sources(setting_sources)
    
    if not processed_setting_sources:
      return cls._result(cls.FAIL)
    
    cls._trigger_event(settings_or_groups, 'before-save', trigger_events)
    
    statuses_per_source, messages_per_source = cls._save(
      settings_or_groups, processed_setting_sources)
    
    cls._trigger_event(settings_or_groups, 'after-save', trigger_events)
    
    return cls._get_return_result([], statuses_per_source, messages_per_source)
  
  @classmethod
  def _save(cls, settings_or_groups, setting_sources):
    statuses_per_source = {}
    messages_per_source = {}
    
    for unused_, sources in setting_sources.items():
      for source in sources:
        try:
          source.write(settings_or_groups)
        except _sources_errors.SourceError as e:
          statuses_per_source[source] = cls.FAIL
          messages_per_source[source] = str(e)
        else:
          statuses_per_source[source] = cls.SUCCESS
          messages_per_source[source] = ''
    
    return statuses_per_source, messages_per_source
  
  @classmethod
  def clear(cls, setting_sources=None):
    """Removes all settings from all specified setting sources.
    
    Parameters:
    
    * `setting_sources` - Dictionary or list of setting sources or `None`. See
      `load()` for more information. If there are no sources to clear, this
      method has no effect.
    """
    if setting_sources is None:
      setting_sources = cls._DEFAULT_SETTING_SOURCES
    
    processed_setting_sources = cls._process_setting_sources(setting_sources)
    
    if setting_sources is not None:
      for sources in processed_setting_sources.values():
        for source in sources:
          source.clear()
  
  @classmethod
  def _trigger_event(cls, settings_or_groups, event_name, trigger_events):
    if trigger_events:
      for setting in cls._list_settings(settings_or_groups, include_groups=True):
        setting.invoke_event(event_name)
  
  @classmethod
  def _set_up_settings_to_ignore(cls, ignore_tag, source_name, filtered_settings):
    settings_to_ignore = []
    for setting in filtered_settings:
      if setting.setting_sources is not None and source_name not in setting.setting_sources:
        setting.tags.add(ignore_tag)
        settings_to_ignore.append(setting)
    
    return settings_to_ignore
  
  @classmethod
  def _clean_up_settings_to_ignore(cls, settings_to_ignore, ignore_tag):
    for setting in settings_to_ignore:
      setting.tags.discard(ignore_tag)
  
  @staticmethod
  def _result(
        status, settings_not_loaded=None, statuses_per_source=None, messages_per_source=None):
    if settings_not_loaded is None:
      settings_not_loaded = []
    
    if statuses_per_source is None:
      statuses_per_source = {}
    
    if messages_per_source is None:
      messages_per_source = {}
    
    return PersistorResult(status, settings_not_loaded, statuses_per_source, messages_per_source)
  
  @classmethod
  def _get_return_result(cls, settings_not_loaded, statuses_per_source, messages_per_source):
    if (all(status == cls.SUCCESS for status in statuses_per_source.values())
        and not settings_not_loaded):
      return cls._result(cls.SUCCESS)
    elif all(status == cls.FAIL for status in statuses_per_source.values()):
      return cls._result(cls.FAIL, settings_not_loaded, statuses_per_source, messages_per_source)
    else:
      return cls._result(
        cls.PARTIAL_SUCCESS, settings_not_loaded, statuses_per_source, messages_per_source)
  
  @classmethod
  def _process_setting_sources(cls, setting_sources):
    processed_setting_sources = collections.OrderedDict()
    
    if not isinstance(setting_sources, dict):
      for key in setting_sources:
        if key not in processed_setting_sources:
          processed_setting_sources[key] = []
        
        try:
          source = cls._DEFAULT_SETTING_SOURCES[key]
        except KeyError:
          # Causes `Persistor.load()` or `Persistor.save()` to return `FAIL`
          return []
        else:
          if isinstance(source, collections.Iterable):
            for item in source:
              processed_setting_sources[key].append(item)
          else:
            processed_setting_sources[key].append(source)
    else:
      for key, source in setting_sources.items():
        if key not in processed_setting_sources:
          processed_setting_sources[key] = []
        
        if isinstance(source, collections.Iterable):
          for item in source:
            processed_setting_sources[key].append(item)
        else:
          processed_setting_sources[key].append(source)
    
    return processed_setting_sources
  
  @staticmethod
  def _list_settings(settings_or_groups, **walk_kwargs):
    settings = []
    for setting_or_group in settings_or_groups:
      if isinstance(setting_or_group, collections.Iterable):
        group = setting_or_group
        settings.append(group)
        settings.extend(list(group.walk(**walk_kwargs)))
      else:
        setting = setting_or_group
        settings.append(setting)
    return settings


PersistorResult = collections.namedtuple(
  'PersistorResult',
  ['status', 'settings_not_loaded', 'statuses_per_source', 'messages_per_source'])
"""Data describing the result of `Persistor.load()` or `Persistor.save()`.

The names of parameters below are parameters passed to `Persistor.load()` or
`Persistor.save()`.

Attributes:

* `status` - Value indicating whether loading or saving proceeded with success
  or some form of failure.

  * `SUCCESS` - All settings were successfully loaded or saved.
  
  * `PARTIAL_SUCCESS` - This status is returned when at least one of the
    following occurs:
    
    * Only some settings were successfully loaded.
    
    * At least one source was not found when attempting to read from it.
    
    * Reading from at least one source was successful and failed for at least
      one other source.
    
    * Writing to at least one source was successful and failed for at least one
      other source.
    
    The `settings_not_loaded` attribute contains all settings not loaded and the
    `statuses` attribute contains statuses for each source individually.
  
  * `FAIL` - This status is returned when at least one of the following occurs:
  
    * Reading from/writing to all sources failed.
    
    * The `setting_sources` parameter is `None` and the default sources returned
      by `Persistor.get_default_setting_sources()` is an empty dictionary.
    
    * The `setting_sources` parameter is a list of keys (source names) and there
      is at least one key not present in the default sources as returned by
      `Persistor.get_default_setting_sources()`.
  
  * `NO_SETTINGS` - Used when the `settings_or_groups` parameter is empty.

* `settings_not_loaded` - List of settings and groups that were not loaded when
  calling `Persistor.load()`. These settings were not found in any setting
  source or they were skipped if their `setting_sources` attribute is not `None`
  and does not contain the key identifying the group of sources (see
  `set_default_setting_sources()` for more information on the key.).
  For `Persistor.save()`, this attribute is always empty.

* `statuses_per_source` - `status` values for each setting source passed to
  `Persistor.load()` or `Persistor.save()`. It is a dictionary of
  (setting source, status) pairs. Beside values in the `status` attribute, the
  following values can also appear in `statuses_per_source`:
  
  * `SOURCE_NOT_FOUND` - The source was not found when attempting to load
    settings, either because the corresponding file is missing or the source
    name does not exist.

* `messages_per_source` - Messages for each setting source passed to
  `Persistor.load()` or `Persistor.save()`. It is a dictionary of
  (setting source, message) pairs.
"""
                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                   