ó
dňôdc           @` sŇ   d  Z  d d l m Z m Z m Z m Z d d l Td d l Z d e	 f d     YZ
 d e
 f d     YZ d	 e
 f d
     YZ d e
 f d     YZ e j j d e
 f d     Y Z d e f d     YZ d S(   u?   Custom exception classes related to batch processing or export.i    (   t   absolute_importt   divisiont   print_functiont   unicode_literals(   t   *Nt   BatcherErrorc           B` s   e  Z RS(    (   t   __name__t
   __module__(    (    (    sB   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/exceptions.pyR   
   s   t   BatcherCancelErrorc           B` s   e  Z RS(    (   R   R   (    (    (    sB   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/exceptions.pyR      s   t   ActionErrorc           B` s   e  Z d    Z RS(   c         C` s8   t    j |  | |  _ | |  _ | |  _ | |  _ d  S(   N(   t   supert   __init__t   messaget   actiont   itemt	   traceback(   t   selfR   R   R   R   (    (    sB   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/exceptions.pyR      s
    			(   R   R   R   (    (    (    sB   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/exceptions.pyR	      s   t
   SkipActionc           B` s   e  Z RS(    (   R   R   (    (    (    sB   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/exceptions.pyR      s   t   ExportErrorc           B` s#   e  Z d  d d d  Z d   Z RS(   u    c         C` s,   t    j   | |  _ | |  _ | |  _ d  S(   N(   R
   R   t   _messaget	   item_namet   file_extension(   R   R   R   R   (    (    sB   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/exceptions.pyR   $   s    		c         C` sc   |  j  } |  j r4 | d t d  d |  j 7} n  |  j r_ | d t d  d |  j 7} n  | S(   Nu   
u   Layer:u    u   File extension:(   R   R   t   _R   (   R   t   str_(    (    sB   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/exceptions.pyt   __str__+   s    		"	"N(   R   R   t   NoneR   R   (    (    (    sB   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/exceptions.pyR   !   s   t   InvalidOutputDirectoryErrorc           B` s   e  Z RS(    (   R   R   (    (    (    sB   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/exceptions.pyR   6   s   (   t   __doc__t
   __future__R    R   R   R   t   future.builtinst   future.utilst   futuret	   ExceptionR   R   R	   R   t   utilst   python_2_unicode_compatibleR   R   (    (    (    sB   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/exceptions.pyt   <module>   s   "
                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          # -*- coding: utf-8 -*-

"""Built-in procedure to export a given item as an image."""

from __future__ import absolute_import, division, print_function, unicode_literals
from future.builtins import *

import collections
import os

import gimp
from gimp import pdb
import gimpenums

from export_layers import pygimplib as pg

from export_layers import exceptions
from export_layers import renamer as renamer_
from export_layers import uniquifier


class ExportModes(object):
  
  EXPORT_MODES = (
    EACH_LAYER,
    EACH_TOP_LEVEL_LAYER_OR_GROUP,
    ENTIRE_IMAGE_AT_ONCE,
  ) = 0, 1, 2


def export(
      batcher,
      output_directory=gimp.user_directory(1),  # `Documents` directory
      file_extension='png',
      export_mode=ExportModes.EACH_LAYER,
      single_image_filename_pattern=None,
      use_file_extension_in_item_name=False,
      convert_file_extension_to_lowercase=False,
      preserve_layer_name_after_export=False):
  item_uniquifier = uniquifier.ItemUniquifier()
  file_extension_properties = _FileExtensionProperties()
  processed_parent_names = set()
  default_file_extension = file_extension
  
  if export_mode == ExportModes.ENTIRE_IMAGE_AT_ONCE and single_image_filename_pattern is not None:
    renamer_for_image = renamer_.ItemRenamer(single_image_filename_pattern)
  else:
    renamer_for_image = None
  
  if export_mode != ExportModes.EACH_LAYER and batcher.process_export:
    multi_layer_image = pg.pdbutils.create_image_from_metadata(batcher.input_image)
    pdb.gimp_image_undo_freeze(multi_layer_image)
    batcher.invoker.add(_delete_image_on_cleanup, ['cleanup_contents'], [multi_layer_image])
  else:
    multi_layer_image = None
  
  if batcher.edit_mode and batcher.process_export:
    image_copy = pg.pdbutils.create_image_from_metadata(batcher.input_image)
    pdb.gimp_image_undo_freeze(image_copy)
    batcher.invoker.add(_delete_image_on_cleanup, ['cleanup_contents'], [image_copy])
  else:
    image_copy = batcher.current_image
  
  while True:
    item = batcher.current_item
    current_file_extension = default_file_extension
    
    item_to_process = item
    raw_item_to_process = batcher.current_raw_item
    
    if batcher.edit_mode and batcher.process_export:
      raw_item_to_process = _copy_layer(raw_item_to_process, image_copy, item)
    
    if multi_layer_image is None:
      image_to_process = image_copy
    else:
      image_to_process = multi_layer_image
    
    if export_mode == ExportModes.ENTIRE_IMAGE_AT_ONCE:
      if batcher.process_export:
        raw_item_to_process = _merge_and_resize_image(batcher, image_copy, raw_item_to_process)
        raw_item_to_process = _copy_layer(raw_item_to_process, image_to_process, item)
      
      if batcher.item_tree.next(item, with_folders=False) is not None:
        _refresh_image_copy_for_edit_mode(batcher, image_copy)
        yield
        continue
      else:
        item_to_process = pg.itemtree.Item(item.raw, pg.itemtree.TYPE_ITEM, [], [], None, None)
        if single_image_filename_pattern is not None:
          item_to_process.name = renamer_for_image.rename(batcher, item_to_process)
        else:
          item_to_process.name = item.name
    elif export_mode == ExportModes.EACH_TOP_LEVEL_LAYER_OR_GROUP:
      if batcher.process_export:
        raw_item_to_process = _merge_and_resize_image(batcher, image_copy, raw_item_to_process)
        raw_item_to_process = _copy_layer(raw_item_to_process, image_to_process, item)
      
      current_top_level_item = _get_top_level_item(item)
      next_top_level_item = _get_top_level_item(batcher.item_tree.next(item, with_folders=False))
      
      if current_top_level_item == next_top_level_item:
        _refresh_image_copy_for_edit_mode(batcher, image_copy)
        yield
        continue
      else:
        item_to_process = current_top_level_item
    
    if preserve_layer_name_after_export:
      item_to_process.push_state()
    
    if batcher.process_names:
      if use_file_extension_in_item_name:
        current_file_extension = _get_current_file_extension(
          item_to_process, default_file_extension, file_extension_properties)
      
      if convert_file_extension_to_lowercase:
        current_file_extension = current_file_extension.lower()
      
      _process_parent_folder_names(item_to_process, item_uniquifier, processed_parent_names)
      _process_item_name(
        batcher, item_to_process, item_uniquifier,
        current_file_extension, default_file_extension, force_default_file_extension=False)
    
    if batcher.process_export:
      if export_mode == ExportModes.EACH_LAYER:
        raw_item_to_process = _merge_and_resize_image(batcher, image_copy, raw_item_to_process)
      
      overwrite_mode, export_status = _export_item(
        batcher, item_to_process, image_to_process, raw_item_to_process,
        output_directory, default_file_extension, file_extension_properties)
      
      if export_status == ExportStatuses.USE_DEFAULT_FILE_EXTENSION:
        if batcher.process_names:
          _process_item_name(
            batcher, item_to_process, item_uniquifier,
            current_file_extension, default_file_extension, force_default_file_extension=True)
        
        if batcher.process_export:
          overwrite_mode, unused_ = _export_item(
            batcher, item_to_process, image_to_process, raw_item_to_process,
            output_directory, default_file_extension, file_extension_properties)
      
      if overwrite_mode != pg.overwrite.OverwriteModes.SKIP:
        file_extension_properties[
          pg.path.get_file_extension(item_to_process.name)].processed_count += 1
        # Append the original raw item
        batcher._exported_raw_items.append(item_to_process.raw)
    
    if preserve_layer_name_after_export:
      item_to_process.pop_state()
    
    _sync_raw_item_name(batcher, item_to_process)
    
    if multi_layer_image is not None:
      _refresh_image(multi_layer_image)
    
    _refresh_image_copy_for_edit_mode(batcher, image_copy)
    
    yield


def _delete_image_on_cleanup(batcher, image):
  if batcher.process_export:
    if image is not None:
      pg.pdbutils.try_delete_image(image)


def _get_top_level_item(item):
  if item is not None and item.parents:
    return item.parents[0]
  else:
    return item


def _process_parent_folder_names(item, item_uniquifier, processed_parent_names):
  for parent in item.parents:
    if parent not in processed_parent_names:
      _validate_name(parent)
      item_uniquifier.uniquify(parent)
      
      processed_parent_names.add(parent)


def _process_item_name(
      batcher, item, item_uniquifier,
      current_file_extension, default_file_extension, force_default_file_extension):
  if not force_default_file_extension:
    if current_file_extension == default_file_extension:
      item.name += '.' + default_file_extension
    else:
      item.name = pg.path.get_filename_with_new_file_extension(
        item.name, current_file_extension, keep_extra_trailing_periods=True)
  else:
    item.name = pg.path.get_filename_with_new_file_extension(
      item.name, default_file_extension, keep_extra_trailing_periods=True)
  
  _validate_name(item)
  item_uniquifier.uniquify(
    item,
    position=_get_unique_substring_position(item.name, pg.path.get_file_extension(item.name)))


def _get_current_file_extension(item, default_file_extension, file_extension_properties):
  item_file_extension = pg.path.get_file_extension(item.name)
  
  if item_file_extension and file_extension_properties[item_file_extension].is_valid:
    return item_file_extension
  else:
    return default_file_extension


def _merge_and_resize_image(batcher, image, raw_item):
  """Merges all layers in the current image into one.
  
  Merging is necessary for:
  * custom procedures inserting layers (background, foreground). Some file
    formats may discard all but one layer.
  * multi-layer images, with each layer containing background or foreground
    which are originally separate layers.
  """
  raw_item_name = raw_item.name
  
  raw_item_merged = pdb.gimp_image_merge_visible_layers(image, gimpenums.EXPAND_AS_NECESSARY)
  pdb.gimp_layer_resize_to_image_size(raw_item_merged)
  
  raw_item_merged.name = raw_item_name
  image.active_layer = raw_item_merged
  
  if not batcher.edit_mode:
    batcher.current_raw_item = raw_item_merged
  
  return raw_item_merged


def _copy_layer(raw_item, dest_image, item):
  raw_item_copy = pg.pdbutils.copy_and_paste_layer(
    raw_item, dest_image, None, len(dest_image.layers), True, True, True)
  pdb.gimp_item_set_name(raw_item_copy, item.name)
  
  return raw_item_copy

def _validate_name(item):
  item.name = pg.path.FilenameValidator.validate(item.name)


def _get_unique_substring_position(str_, file_extension):
  return len(str_) - len('.' + file_extension)


def _export_item(
      batcher, item, image, raw_item,
      output_directory, default_file_extension, file_extension_properties):
  output_filepath = _get_item_filepath(item, output_directory)
  file_extension = pg.path.get_file_extension(item.name)
  export_status = ExportStatuses.NOT_EXPORTED_YET
  
  batcher.progress_updater.update_text(_('Saving "{}"').format(output_filepath))
  
  overwrite_mode, output_filepath = pg.overwrite.handle_overwrite(
    output_filepath, batcher.overwrite_chooser,
    _get_unique_substring_position(output_filepath, file_extension))
  
  if overwrite_mode == pg.overwrite.OverwriteModes.CANCEL:
    raise exceptions.BatcherCancelError('cancelled')
  
  if overwrite_mode != pg.overwrite.OverwriteModes.SKIP:
    _make_dirs(item, os.path.dirname(output_filepath), default_file_extension)
    
    export_status = _export_item_once_wrapper(
      batcher,
      _get_export_func(file_extension),
      _get_run_mode(batcher, file_extension, file_extension_properties),
      image,
      raw_item,
      output_filepath,
      file_extension,
      default_file_extension,
      file_extension_properties)
    
    if export_status == ExportStatuses.FORCE_INTERACTIVE:
      export_status = _export_item_once_wrapper(
        batcher,
        _get_export_func(file_extension),
        gimpenums.RUN_INTERACTIVE,
        image,
        raw_item,
        output_filepath,
        file_extension,
        default_file_extension,
        file_extension_properties)
  
  return overwrite_mode, export_status


def _get_item_filepath(item, dirpath):
  """Returns a file path based on the specified directory path and the name of
  the item and its parents.
  
  The file path created has the following format:
    
    <directory path>/<item path components>/<item name>
  
  If the directory path is not an absolute path or is `None`, the
  current working directory is prepended.
  
  Item path components consist of parents' item names, starting with the
  topmost parent.
  """
  if dirpath is None:
    dirpath = ''
  
  path = os.path.abspath(dirpath)
  
  path_components = [parent.name for parent in item.parents]
  if path_components:
    path = os.path.join(path, os.path.join(*path_components))
  
  return os.path.join(path, item.name)


def _make_dirs(item, dirpath, default_file_extension):
  try:
    pg.path.make_dirs(dirpath)
  except OSError as e:
    try:
      message = e.args[1]
      if e.filename is not None:
        message += ': "{}"'.format(e.filename)
    except (IndexError, AttributeError):
      message = str(e)
    
    raise exceptions.InvalidOutputDirectoryError(message, item.name, default_file_extension)


def _export_item_once_wrapper(
      batcher, export_func, run_mode, image, raw_item, output_filepath, file_extension,
      default_file_extension, file_extension_properties):
  with batcher.export_context_manager(
         run_mode, image, raw_item, output_filepath,
         *batcher.export_context_manager_args, **batcher.export_context_manager_kwargs):
    export_status = _export_item_once(
      batcher, export_func, run_mode, image, raw_item, output_filepath, file_extension,
      default_file_extension, file_extension_properties)
  
  return export_status


def _get_run_mode(batcher, file_extension, file_extension_properties):
  file_extension_property = file_extension_properties[file_extension]
  if file_extension_property.is_valid and file_extension_property.processed_count > 0:
    return gimpenums.RUN_WITH_LAST_VALS
  else:
    return batcher.initial_run_mode


def _get_export_func(file_extension):
  return pg.fileformats.get_save_procedure(file_extension)


def _export_item_once(
      batcher, export_func, run_mode, image, raw_item, output_filepath, file_extension,
      default_file_extension, file_extension_properties):
  export_status = ExportStatuses.NOT_EXPORTED_YET
  
  try:
    export_func(
      run_mode,
      image,
      raw_item,
      pg.utils.safe_encode_gimp(output_filepath),
      pg.utils.safe_encode_gimp(os.path.basename(output_filepath)))
  except RuntimeError as e:
    # HACK: Examining the exception message seems to be the only way to determine
    # some specific cases of export failure.
    if _was_export_canceled_by_user(str(e)):
      raise exceptions.BatcherCancelError(str(e))
    elif _should_export_again_with_interactive_run_mode(str(e), run_mode):
      export_status = ExportStatuses.FORCE_INTERACTIVE
    elif _should_export_again_with_default_file_extension(file_extension, default_file_extension):
      file_extension_properties[file_extension].is_valid = False
      export_status = ExportStatuses.USE_DEFAULT_FILE_EXTENSION
    else:
      raise exceptions.ExportError(str(e), raw_item.name, default_file_extension)
  else:
    export_status = ExportStatuses.EXPORT_SUCCESSFUL
  
  return export_status


def _was_export_canceled_by_user(exception_message):
  return any(message in exception_message.lower() for message in ['cancelled', 'canceled'])


def _should_export_again_with_interactive_run_mode(exception_message, current_run_mode):
  return (
    'calling error' in exception_message.lower()
    and current_run_mode in [gimpenums.RUN_WITH_LAST_VALS, gimpenums.RUN_NONINTERACTIVE])


def _should_export_again_with_default_file_extension(file_extension, default_file_extension):
  return file_extension != default_file_extension


def _sync_raw_item_name(batcher, item_to_process):
  if batcher.current_item == item_to_process and batcher.process_names and not batcher.is_preview:
    batcher.current_raw_item.name = batcher.current_item.name


def _refresh_image_copy_for_edit_mode(batcher, image_copy):
  if batcher.edit_mode and batcher.process_export:
    _refresh_image(image_copy)


def _refresh_image(image):
  for layer in image.layers:
    pdb.gimp_image_remove_layer(image, layer)


class _FileExtension(object):
  """
  This class defines additional properties for a file extension.
  
  Attributes:
  
  * `is_valid` - If `True`, file extension is valid and can be used in filenames
    for file export procedures.
  
  * `processed_count` - Number of items with the specific file extension that
    have already been exported.
  """
  
  def __init__(self):
    self.is_valid = True
    self.processed_count = 0


class _FileExtensionProperties(object):
  """Mapping of file extensions from `pygimplib.fileformats.file_formats` to
  `_FileExtension` instances.
  
  File extension as a key is always converted to lowercase.
  """
  def __init__(self):
    self._properties = collections.defaultdict(_FileExtension)
    
    for file_format in pg.fileformats.file_formats:
      # This ensures that the file format dialog will be displayed only once per
      # file format if multiple file extensions for the same format are used
      # (e.g. 'jpg', 'jpeg' or 'jpe' for the JPEG format).
      extension_properties = _FileExtension()
      for file_extension in file_format.file_extensions:
        self._properties[file_extension.lower()] = extension_properties
  
  def __getitem__(self, key):
    return self._properties[key.lower()]


class ExportStatuses(object):
  EXPORT_STATUSES = (
    NOT_EXPORTED_YET, EXPORT_SUCCESSFUL, FORCE_INTERACTIVE, USE_DEFAULT_FILE_EXTENSION
  ) = (0, 1, 2, 3)
                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                     