
ddc           @` s  d  Z  d d l m Z m Z m Z m Z d d l Td d l Z e j d  d d l	 Z	 d d l
 Z
 d d l Z d d l m Z d d l Z d d l m Z d d l m Z d d	 l m Z d
 e j j f d     YZ d e j j f d     YZ d e j f d     YZ d S(   u?   Widgets to interactively edit actions (procedures/constraints).i    (   t   absolute_importt   divisiont   print_functiont   unicode_literals(   t   *Nu   2.0(   t   pdb(   t	   pygimplib(   t   actions(   t   messagest	   ActionBoxc           B` s  e  Z d  Z i e j d! e j f f d 6e j d! e j e j f f d 6e j d! e j f f d 6Z d Z	 d Z
 d! d! d! e d! e j j j d  Z d   Z d   Z d	   Z d
   Z d   Z d   Z d   Z d   Z d   Z d   Z d   Z d   Z d   Z d   Z d   Z d   Z  d   Z! d   Z" d   Z# d   Z$ d! d! d  Z% d   Z& d   Z' d   Z( d   Z) d    Z* RS("   u  
  This class defines a scrollable box that allows the user to add, edit and
  remove actions interactively. Each action has an associated widget
  (item) displayed in the box.
  
  The box connects events to the passed actions that keeps the actions and
  the box in sync. For example, when adding an action via `actions.add()`,
  the item for the action is automatically added to the box. Conversely, when
  calling `add_item()` from this class, both the action and the item are
  added to the actions and the GUI, respectively.
  
  Signals:
  
  * `'action-box-item-added'` - An item was added via `add_item()`.
    
    Arguments:
    
    * `item` - The added item.
    
  * `'action-box-item-reordered'` - An item was reordered via
    `reorder_item()`.
    
    Arguments:
    
    * `item` - The reordered item.
    * `new_position` - The new position of the reordered item (starting from 0).
    
  * `'action-box-item-removed'` - An item was removed via `remove_item()`.
    
    Arguments:
    
    * `item` - The removed item.
  s   action-box-item-addeds   action-box-item-reordereds   action-box-item-removedi   i  c   
      ` s   t    j d | | |	  |   _ | d  k	 r4 | n i    _ |   _ |   _ |   _ |   _ d    _	   j
     j j d   f d      _   j j d   f d      _   j j d   f d      _   j j d   f d	      _ d  S(
   Nt   item_spacingu   after-add-actionc         ` s     j  |  S(   N(   t   _add_item_from_action(   R   t   actiont   orig_action_dict(   t   self(    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyt   <lambda>b   t    u   after-reorder-actionc         ` s     j  | |  S(   N(   t   _reorder_action(   R   R   t   current_positiont   new_position(   R   (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR   f   s    u   before-remove-actionc         ` s     j  |  S(   N(   t   _remove_action(   R   R   (   R   (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR   k   R   u   before-clear-actionsc         ` s
     j    S(   N(   t   _clear(   R   (   R   (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR   n   R   (   t   supert   __init__t   _actionst   Nonet   _builtin_actionst   _add_action_textt   _edit_action_textt   _allow_custom_actionst   _add_custom_action_textt   _pdb_procedure_browser_dialogt	   _init_guit   connect_eventt   _after_add_action_event_idt   _after_reorder_action_event_idt   _before_remove_action_event_idt   _before_clear_actions_event_id(
   R   R   t   builtin_actionst   add_action_textt   edit_action_textt   allow_custom_actionst   add_custom_action_textR
   t   argst   kwargs(    (   R   sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR   G   s(    						
				c         C` sd   |  j  j |  j t  t j |  j  |  } |  j  j |  j t  |  j |  } |  j d |  | S(   Nu   action-box-item-added(	   R   t   set_event_enabledR"   t   Falset   actions_t   addt   TrueR   t   emit(   R   t   action_dict_or_functionR   t   item(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyt   add_itemp   s    c         C` sq   |  j  | |  } |  j j |  j t  t j |  j | j j |  |  j j |  j t	  |  j
 d | |  d  S(   Nu   action-box-item-reordered(   t   _reorder_itemR   R-   R#   R.   R/   t   reorderR   t   nameR1   R2   (   R   R4   R   t   processed_new_position(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyt   reorder_item{   s
    c         C` sf   |  j  |  |  j j |  j t  t j |  j | j j  |  j j |  j t	  |  j
 d |  d  S(   Nu   action-box-item-removed(   t   _remove_itemR   R-   R$   R.   R/   t   removeR   R8   R1   R2   (   R   R4   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyt   remove_item   s
    c         C` s9  |  j  d  k	 r t j   |  _ t j   } | j |  j  | j t j	 t j
 t j  d t d t t j t j j |  j    } | j t  | j | d t d t |  j j |  n t j d t j
  |  _ |  j j t j  |  j j d |  j  |  j j |  j d t d t t j   |  _ |  j   d  S(   Nt   expandt   fillt   stocku   clicked(   R   R   t   gtkt   Buttont   _button_addt   HBoxt   set_spacingt   _ADD_BUTTON_HBOX_SPACINGt
   pack_startt   image_new_from_stockt	   STOCK_ADDt   ICON_SIZE_MENUR.   t   Labelt   pgt   utilst   safe_encode_gtkt   set_use_underlineR1   R0   t
   set_relieft   RELIEF_NONEt   connectt   _on_button_add_clickedt   _vboxt   Menut   _actions_menut   _init_actions_menu_popup(   R   t   button_hboxt	   label_add(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR       s     %c         C` sl   |  j  |  t | | d j j  } t   j |  | j j d |  j |  | j	 j d |  j
 |  | S(   Nu   enabledu   clicked(   t   _init_action_item_guit   _ActionBoxItemt   guit   elementR   R5   t   button_editRR   t   _on_item_button_edit_clickedt   button_removet8   _on_item_button_remove_clicked_remove_action_edit_dialog(   R   R   R4   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR      s    	c         C` s   | j    t | d j t j j j  r | d j j j d d  | d j j j	   j
 t j  | d j j j	   j |  j  | d j j j	   j d |  j | d j j  n  d  S(   Nu   enabledu   width-requesti   u   size-allocate(   t   initialize_guit
   isinstanceR\   RL   t   settingt   SettingGuiTypest   check_buttonR]   t   set_propertyt	   get_childt   set_ellipsizet   pangot   ELLIPSIZE_ENDt   set_max_width_charst$   _ACTION_ENABLED_LABEL_MAX_CHAR_WIDTHRR   t'   _on_action_item_gui_label_size_allocate(   R   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyRZ      s    
 
c         C` s9   t  j j |  r" | j d   n | j | j    d  S(   N(   RL   R\   t   label_fits_textt   set_tooltip_textR   t   get_text(   R   t   item_gui_labelt
   allocationt   item_gui(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyRn      s    c         ` s`   t    f d   |  j D d   } | d  k	 rA |  j | |  n t d j   j |     d  S(   Nc         3` s*   |  ]  } | j  j   j k r | Vq d  S(   N(   R   R8   (   t   .0t   item_(   R   (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pys	   <genexpr>   s    u+   action "{}" does not match any item in "{}"(   t   nextt   _itemsR   R6   t
   ValueErrort   formatR8   (   R   R   R   R4   (    (   R   sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR      s    "c         C` s   t    j | |  S(   N(   R   R:   (   R   R4   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR6      s    c         ` s`   t    f d   |  j D d   } | d  k	 r> |  j |  n t d j   j   |     d  S(   Nc         3` s*   |  ]  } | j  j   j k r | Vq d  S(   N(   R   R8   (   Ru   Rv   (   R   (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pys	   <genexpr>   s    u+   action "{}" does not match any item in "{}"(   Rw   Rx   R   R;   Ry   Rz   t   get_path(   R   R   R4   (    (   R   sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR      s
    "	c         C` sF   |  j  |  t |  j  d k r2 |  j j   n  t   j |  d  S(   Ni   (   t   _get_item_positiont   lenRx   RC   t
   grab_focusR   R=   (   R   R4   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR;      s    "c         C` s8   x1 t  t |  j   D] } |  j |  j d  q Wd  S(   Ni    (   t   rangeR}   Rx   R;   (   R   t   unused_(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR      s    c         C` sd   x$ |  j  j   D] } |  j |  q W|  j rS |  j j t j    |  j   n  |  j j	   d  S(   N(
   R   t   valuest   _add_action_to_menu_popupR   RV   t   appendRA   t   SeparatorMenuItemt$   _add_add_custom_action_to_menu_popupt   show_all(   R   t   action_dict(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyRW      s    	c         C` s    |  j  j d  d  d  d d  d  S(   Ni    (   RV   t   popupR   (   R   t   button(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyRS      s    c         C` sR   t  j d t j j | d  d t  } | j d |  j |  |  j j	 |  d  S(   Nt   labelu   display_namet   use_underlineu   activate(
   RA   t   MenuItemRL   RM   RN   R.   RR   t   _on_actions_menu_item_activateRV   R   (   R   R   t	   menu_item(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR      s
    		c         C` s5   |  j  |  } | j d t  r1 |  j |  n  d  S(   Nu   display_options_on_create(   R5   t   getR.   t   _display_action_edit_dialog(   R   R   R   R4   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR      s    c         C` sB   t  j d |  j d t  } | j d |  j  |  j j |  d  S(   NR   R   u   activate(   RA   R   R   R.   RR   t(   _on_add_custom_action_menu_item_activateRV   R   (   R   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR      s    c         C` s,   |  j  r |  j  j   n |  j   |  _  d  S(   N(   R   t   showt$   _create_pdb_procedure_browser_dialog(   R   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR     s    	c      	   C` s   t  j t d  d t j j d t j t j t j	 t j
 f } | j t j  | j t j t j
 f  | j d |  j  | j   | S(   Nu   Procedure Browsert   rolet   buttonsu   response(   t   gimpuit   ProcBrowserDialogt   _RL   t   configt   PLUGIN_NAMERA   RI   t   RESPONSE_OKt   STOCK_CANCELt   RESPONSE_CANCELt   set_default_responset   set_alternative_button_orderRR   t)   _on_pdb_procedure_browser_dialog_responseR   (   R   t   dialog(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR   	  s    !
c         C` s  | t  j k r| j   } | rt t j j |  } y t j |  } Wn t j	 k
 r } t
 j d t d j | j   d d d t d  j | j | j  d t j j |   d t d	  d
 t  | j   d  SXt | d <|  j |  } |  j | |  qn  | j   d  S(   Nt   main_messageu.   An error occurred while adding procedure "{}".t   failure_messageu    t   detailsuN   Could not add procedure "{}" because the parameter type "{}" is not supported.t   parentt   report_descriptionu]   You can help fix this issue by sending a report with the text above to one of the sites belowt   display_details_initiallyu   enabled(   RA   R   t   get_selectedR   RL   RM   t   safe_encode_gimpR/   t!   get_action_dict_for_pdb_proceduret   UnsupportedPdbProcedureErrort	   messages_t   display_failure_messageR   Rz   t   procedure_namet   unsupported_param_typeR\   t   get_toplevel_windowR1   t   hideR.   R5   R   (   R   R   t   response_idR   t   pdb_proceduret   pdb_proc_action_dictt   eR4   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR     s.    		

c         C` s   t  | j | d |  j |  d t j j } | | _ | d  k r[ | j d |  j	 |  n | j d |  j
 | |  | j   d  S(   Nt   titleR   u   response(   t   _ActionEditDialogR   t   _get_action_edit_dialog_titleRL   R   R   t   action_edit_dialogR   RR   t.   _on_action_edit_dialog_for_new_action_responset3   _on_action_edit_dialog_for_existing_action_responseR   (   R   R4   R   t   action_values_before_dialogR   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR   9  s"    	
c         ` s     j    r   j j   d  S  j d j d  rV t t j j   j d j	  } n d  }   f d     j j   D } |  j   | |  d  S(   Nu   originu   gimp_pdbu   functionc         ` s(   i  |  ] } | j  | j   j   q S(    (   t   valueR{   R   (   Ru   Rd   (   R4   (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pys
   <dictcomp>\  s   	(   t   is_being_editedR   t   presentR   t   is_itemR   RL   RM   R   R   R   t   walkR   (   R   t   edit_buttonR4   R   R   (    (   R4   sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR_   Q  s    #c         C` s[   | j    | t j k rA | j d j   | j d j t  n |  j |  d  | _	 d  S(   Nu	   argumentsu   enabled(
   t   destroyRA   R   R   t   apply_gui_values_to_settingst	   set_valueR1   R=   R   R   (   R   R   R   R4   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR   a  s    
c         C` sJ   | j    | t j k r- | j d j   n | j j |  d  | _ d  S(   Nu	   arguments(   R   RA   R   R   R   t
   set_valuesR   R   (   R   R   R   R4   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR   m  s
    
c         C` s&   | j    r" | j j t j  n  d  S(   N(   R   R   t   responseRA   R   (   R   R`   R4   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyRa   x  s    c         C` s4   |  j  d  k	 r, d j |  j  | j d j  Sd  Sd  S(   Nu   {}: {}u   display_name(   R   R   Rz   R   R   (   R   R4   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR   }  s    N(+   t   __name__t
   __module__t   __doc__t   gobjectt   SIGNAL_RUN_FIRSTR   t   TYPE_PYOBJECTt   TYPE_INTt   __gsignals__RF   Rm   R1   RL   R\   t   ItemBoxt   ITEM_SPACINGR   R5   R:   R=   R    R   RZ   Rn   R   R6   R   R;   R   RW   RS   R   R   R   R   R   R   R   R_   R   R   Ra   R   (    (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR	      sN   !!																
							"				R[   c           B` sq   e  Z d    Z e d    Z e d    Z d   Z d   Z d   Z d   Z	 d	 d	 d	 d	 d  Z d   Z RS(
   c         C` s   t    j |  d  |  _ | |  _ t j   |  _ |  j |  j t j	 d d t j   |  _
 |  j |  j
 t j d d |  j
 j   d  |  _ d  S(   Nt   positioni    (   R   R   R   R   t   _actionRA   RB   t   _button_editt   _setup_item_buttont
   STOCK_EDITt   _button_warningt   _setup_item_indicator_buttont   STOCK_DIALOG_WARNINGR   t!   _display_warning_message_event_id(   R   R   t   item_widget(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR     s    		c         C` s   |  j  S(   N(   R   (   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR     s    c         C` s   |  j  S(   N(   R   (   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR^     s    c         C` s   |  j  d  k	 S(   N(   R   R   (   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR     s    c         C` s)   |  j  d  k	 r% |  j  j t j  n  d  S(   N(   R   R   R   RA   R   (   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyt   close_edit_dialog  s    c         C` s   |  j  j |  d  S(   N(   t   widgetRp   (   R   t   text(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyt   set_tooltip  s    c         C` s   |  j  j   S(   N(   R   t   get_visible(   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyt   has_warning  s    c         C` s   | r~ |  j  |  | d  k	 rn |  j d  k	 rD |  j j |  j  n  |  j j d |  j | | | |  |  _ n  |  j j   nH |  j j   |  j  d   |  j d  k	 r |  j j |  j  d  |  _ n  d  S(   Nu   clicked(	   R   R   R   R   t
   disconnectRR   t   _on_button_warning_clickedR   R   (   R   R   R   R   R   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyt   set_warning  s    	c         C` s   t  j | | | d | d  S(   NR   (   R   R   (   R   R   R   t   short_messaget   full_messageR   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR     s    N(   R   R   R   t   propertyR   R^   R   R   R   R   R   R   R   (    (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR[     s   					R   c           B` s   e  Z d  Z d  Z d Z d  Z d Z d Z d Z d Z	 d Z
 d   Z d d  Z d   Z d   Z d	   Z d
   Z d   Z RS(   i   i   i,  i   i   c         O` s  t    j | |   |  j   |  j t  |  j t j t j  |  _	 |  j t j
 t j  |  _ |  j t j t j g  t j   |  _ |  j j t d   |  j j |  j d t d t |  j j |  j t  t j j   |  _ |  j j j t  |  j j j t j  |  j j j d j  t! j" | d j#    |  j j$ d |  j% |  d  |  _' | d j# r|  j( | d j#  |  _' n* | d  k	 r|  j( | j) | j*  |  _' n  t j+ d t  |  _, |  j, j- |  j.  |  j, j/ |  j0  t j1   |  _2 |  j2 j3 |  j4  |  j2 j5 |  j6  |  j2 j | d	 j j7 d t d t d
 | k rz|  j2 j | d
 j j7 d t d t n  | d j j7 j8 |  j2  t j1   |  _9 |  j9 j5 |  j:  |  j9 j3 |  j;  |  j9 j |  j d t d t |  j' d  k	 r|  j9 j |  j' d t d t n  |  j9 j |  j, d t d t |  j9 j | d j j7 d t d t |  j< j |  j9 d t d t |  j= | |  |  j> |  j	  |  j j$ d |  j? |  |  j$ d |  j@ |  d  S(   Nu   _ResetR>   R?   u	   <b>{}</b>u   display_nameu   changedu   descriptiont   homogeneousu   enabled_for_previewsu   also_apply_to_parent_foldersu   more_options_expandedu   clickedu   response(A   R   R   t   set_transientt   set_resizableR.   t
   add_buttonRA   t   STOCK_OKR   t
   _button_okR   R   t   _button_cancelR   RB   t   _button_resett	   set_labelR   t   action_areaRG   t   set_child_secondaryR1   RL   R\   t   EditableLabelt   _label_procedure_nameR   t   set_use_markupRi   Rj   Rk   t
   set_markupRz   R   t   markup_escape_textR   RR   t    _on_label_procedure_name_changedR   t   _label_procedure_descriptiont   _create_label_descriptiont
   proc_blurbt	   proc_helpt   Tablet   _table_action_argumentst   set_row_spacingst   _TABLE_ROW_SPACINGt   set_col_spacingst   _TABLE_COLUMN_SPACINGt   VBoxt   _vbox_more_optionsRE   t   _MORE_OPTIONS_SPACINGt   set_border_widtht   _MORE_OPTIONS_BORDER_WIDTHR]   R0   RT   t   _DIALOG_BORDER_WIDTHt   _DIALOG_VBOX_SPACINGt   vboxt   _set_argumentst	   set_focust   _on_button_reset_clickedt   _on_action_edit_dialog_response(   R   R   R   R+   R,   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR     sb    
 				 "&c         C` sP   t  j   } | j t  | j d d  | j |  | rL | j |  n  | S(   Ng        g      ?(   RA   RK   t   set_line_wrapR1   t   set_alignmentR   Rp   (   R   t   summaryt   full_descriptiont   label_description(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR     s    c         C` sZ  xSt  | d  D]A\ } } | j j   s2 q n  t j | j  } | j d d  | d  k	 r{ | j | j	 | d  n  |  j
 j | d d | | d  | j j } t | j t j j j  s&t | t j j  r2| j j   r| j j j d |  j  |  j | j j _ q#|  j   } q2n |  j   } |  j
 j | d d | | d  q Wd  S(   Nu	   argumentsg        g      ?i   i    i   u   width-request(   t	   enumerateR\   R   RA   RK   t   display_nameR  R   Rp   t   paramsR   t   attachR]   Rc   RL   Rd   Re   t   nullt   ArraySettingt   element_typet   get_allowed_gui_typesRg   t   _ARRAY_PARAMETER_GUI_WIDTHt   _ARRAY_PARAMETER_GUI_MAX_HEIGHTt
   max_heightt   _create_placeholder_widget(   R   R   R   t   iRd   R   t   gui_element_to_attach(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR  !  s"     c         C` s   | d j    d  S(   Nu	   arguments(   t   reset(   R   R   R   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR  ;  s    c         C` sI   | d j  | j j    | j j d j t j | j j      d  S(   Nu   display_nameu	   <b>{}</b>(   R   R   Rq   R   Rz   R   R   (   R   t   editable_labelR   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR   >  s    	c         C` s   x- t  |  j j    D] } |  j j |  q Wx- t  |  j j    D] } |  j j |  qF W| d j j j |  j  |  j j | d j j  d  S(   Nu   more_options_expanded(   t   listR   t   get_childrenR<   R  R\   R]   RT   (   R   R   R   R   t   child(    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR  D  s    c         C` s   t  j   } | j |  j  | j t  j t  j t  j  d t d t t  j	   } | j
 t  | j d j t j t d     | j | d t d t | S(   NR>   R?   u!   <span font_size="small">{}</span>u   Cannot modify this parameter(   RA   RD   RE   t7   _PLACEHOLDER_WIDGET_HORIZONTAL_SPACING_BETWEEN_ELEMENTSRG   RH   R   t   ICON_SIZE_BUTTONR.   RK   R   R1   R   Rz   R   R   R   (   R   t   hboxR   (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR  N  s    N(   R   R   R  R	  R   R  R  R  R'  R  R  R   R   R   R  R  R   R  R  (    (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyR     s    	C
				
(   R   t
   __future__R    R   R   R   t   future.builtinst   pygtkt   requireRA   R   Rj   t   gimpR   R   t   export_layersR   RL   R   R/   t   export_layers.guiR   R   R\   R   R	   t   ItemBoxItemR[   t   DialogR   (    (    (    sC   /home/josie/.config/GIMP/2.10/plug-ins/export_layers/gui/actions.pyt   <module>   s    "
 o?                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  # -*- coding: utf-8 -*-

"""Main module for the plug-in GUI."""

from __future__ import absolute_import, division, print_function, unicode_literals
from future.builtins import *

import contextlib
import functools
import os
import traceback

try:
  import webbrowser
except ImportError:
  _webbrowser_module_found = False
else:
  _webbrowser_module_found = True

import pygtk
pygtk.require('2.0')
import gtk
import gobject
import pango

import gimp
from gimp import pdb
import gimpenums
import gimpui

from export_layers import pygimplib as pg

from export_layers import actions
from export_layers import batcher as batcher_
from export_layers import builtin_constraints
from export_layers import builtin_procedures
from export_layers import exceptions
from export_layers import renamer as renamer_
from export_layers import update
from export_layers import utils as utils_

from export_layers.gui import actions as actions_
from export_layers.gui import message_label as message_label_
from export_layers.gui import messages as messages_
from export_layers.gui import preview_image as preview_image_
from export_layers.gui import preview_name as preview_name_
from export_layers.gui import previews_controller as previews_controller_
from export_layers.gui import progress as progress_

if hasattr(pg.setting.sources, 'json'):
  _json_module_found = True
else:
  _json_module_found = False


def display_reset_prompt(parent=None, more_settings_shown=False):
  dialog = gtk.MessageDialog(
    parent=parent,
    type=gtk.MESSAGE_WARNING,
    flags=gtk.DIALOG_MODAL | gtk.DIALOG_DESTROY_WITH_PARENT,
    buttons=gtk.BUTTONS_YES_NO)
  dialog.set_transient_for(parent)
  dialog.set_title(pg.config.PLUGIN_TITLE)
  
  dialog.set_markup(
    gobject.markup_escape_text(_('Are you sure you want to reset settings?')))
  
  if more_settings_shown:
    checkbutton_reset_actions = gtk.CheckButton(
      label=_('Remove procedures and constraints'), use_underline=False)
    checkbutton_reset_actions.set_active(True)
    
    dialog.vbox.pack_start(checkbutton_reset_actions, expand=False, fill=False)
  
  dialog.set_focus(dialog.get_widget_for_response(gtk.RESPONSE_NO))
  
  dialog.show_all()
  response_id = dialog.run()
  dialog.destroy()
  
  clear_actions = checkbutton_reset_actions.get_active() if more_settings_shown else False
  
  return response_id, clear_actions


@contextlib.contextmanager
def handle_gui_in_export(run_mode, image, layer, output_filepath, window):
  should_manipulate_window = run_mode == gimpenums.RUN_INTERACTIVE
  
  if should_manipulate_window:
    window_position = window.get_position()
    window.hide()
  while gtk.events_pending():
    gtk.main_iteration()
  
  try:
    yield
  finally:
    if should_manipulate_window:
      window.move(*window_position)
      window.show()
    while gtk.events_pending():
      gtk.main_iteration()


def stop_batcher(batcher):
  if batcher is not None:
    batcher.stop()
    return True
  else:
    return False


def _set_settings(func):
  """
  This is a decorator for `Group.apply_gui_values_to_settings()` that prevents
  the decorated function from being invoked if there are invalid setting
  values. For the invalid values, an error message is displayed.
  
  This decorator is meant to be used in the `ExportLayersDialog` class.
  """
  
  @functools.wraps(func)
  def func_wrapper(self, *args, **kwargs):
    try:
      self._settings['main'].apply_gui_values_to_settings()
      self._settings['gui'].apply_gui_values_to_settings()
      
      self._settings['gui/current_directory'].gui.update_setting_value()
      
      self._settings['main/output_directory'].set_value(
        self._settings['gui/current_directory'].value)
      
      self._settings['main/selected_layers'].value[self._image.ID] = (
        self._name_preview.selected_items)
      self._settings['gui/name_preview_layers_collapsed_state'].value[self._image.ID] = (
        self._name_preview.collapsed_items)
      self._settings['gui/image_preview_displayed_layers'].value[self._image.ID] = (
        [self._image_preview.item.raw.ID] if self._image_preview.item is not None else [])
    except pg.setting.SettingValueError as e:
      self._display_inline_message(str(e), gtk.MESSAGE_ERROR, e.setting)
      return
    else:
      func(self, *args, **kwargs)
  
  return func_wrapper


def _setup_image_ids_and_directories_and_initial_directory(
      settings, current_directory_setting, current_image):
  """
  Set up the initial directory path for the current image according to the
  following priority list:
  
    1. Last export directory path of the current image
    2. Import directory path of the current image
    3. Last export directory path of any image (i.e. the current value of
       `'main/output_directory'`)
    4. The default directory path (default value) for `'main/output_directory'`
  
  Notes:
  
    Directory 3. is set upon loading `'main/output_directory'` from a persistent
    source.
    Directory 4. is set upon the instantiation of `'main/output_directory'`.
  """
  settings['gui/image_ids_and_directories'].update_image_ids_and_dirpaths()
  
  update_performed = _update_directory(
    current_directory_setting,
    current_image,
    settings['gui/image_ids_and_directories'].value[current_image.ID])
  
  if not update_performed:
    current_directory_setting.set_value(settings['main/output_directory'].value)


def _update_directory(setting, current_image, current_image_dirpath):
  """
  Set the directory path to the setting according to the priority list below:
  
  1. `current_image_dirpath` if not `None`
  2. `current_image` - import path of the current image if not `None`
  
  If update was performed, return `True`, otherwise return `False`.
  """
  if current_image_dirpath is not None:
    if isinstance(current_image_dirpath, bytes):
      current_image_dirpath = pg.utils.safe_decode_gimp(current_image_dirpath)
    
    setting.set_value(current_image_dirpath)
    return True
  
  if current_image.filename is not None:
    setting.set_value(os.path.dirname(pg.utils.safe_decode_gimp(current_image.filename)))
    return True
  
  return False


def _setup_output_directory_changed(settings, current_image):
  def on_output_directory_changed(
        output_directory, image_ids_and_directories, current_image_id):
    image_ids_and_directories.update_dirpath(current_image_id, output_directory.value)
  
  settings['main/output_directory'].connect_event(
    'value-changed',
    on_output_directory_changed,
    settings['gui/image_ids_and_directories'],
    current_image.ID)


#===============================================================================


class ExportLayersDialog(object):
  
  _DIALOG_SIZE = (900, 610)
  _DIALOG_BORDER_WIDTH = 5
  _DIALOG_CONTENTS_BORDER_WIDTH = 5
  _DIALOG_VBOX_SPACING = 5
  
  _SAVE_IN_FOLDER_LABEL_PADDING = 3
  _PREVIEW_LABEL_BORDER_WIDTH = 5
  
  _HBOX_EXPORT_LABELS_NAME_SPACING = 10
  _HBOX_EXPORT_NAME_ENTRIES_SPACING = 3
  _HBOX_EXPORT_NAME_AND_MESSAGE_HORIZONTAL_SPACING = 8
  _HBOX_EXPORT_NAME_AND_MESSAGE_BORDER_WIDTH = 2
  
  _MORE_SETTINGS_HORIZONTAL_SPACING = 12
  _MORE_SETTINGS_BORDER_WIDTH = 3
  
  _IMPORT_SETTINGS_CUSTOM_WIDGETS_BORDER_WIDTH = 3
  
  _FILE_EXTENSION_ENTRY_MIN_WIDTH_CHARS = 4
  _FILE_EXTENSION_ENTRY_MAX_WIDTH_CHARS = 10
  _FILENAME_PATTERN_ENTRY_MIN_WIDTH_CHARS = 12
  _FILENAME_PATTERN_ENTRY_MAX_WIDTH_CHARS = 40
  
  _DELAY_NAME_PREVIEW_UPDATE_TEXT_ENTRIES_MILLISECONDS = 100
  _DELAY_CLEAR_LABEL_MESSAGE_MILLISECONDS = 10000
  
  _MAXIMUM_IMAGE_PREVIEW_AUTOMATIC_UPDATE_DURATION_SECONDS = 1.0
  
  def __init__(self, initial_layer_tree, settings, run_gui_func=None):
    self._initial_layer_tree = initial_layer_tree
    self._settings = settings
    
    self._image = self._initial_layer_tree.image
    self._message_setting = None
    self._batcher = None
    self._batcher_for_previews = batcher_.Batcher(
      gimpenums.RUN_NONINTERACTIVE,
      self._image,
      self._settings['main/procedures'],
      self._settings['main/constraints'],
      overwrite_chooser=pg.overwrite.NoninteractiveOverwriteChooser(
        self._settings['main/overwrite_mode'].items['replace']),
      item_tree=self._initial_layer_tree)
    
    if gimp.version[:2] == (2, 8):
      pg.pdbutils.suppress_gimp_progress()
    
    self._init_settings()
    
    self._init_gui_elements()
    self._assign_gui_to_settings()
    self._connect_events()
    
    self._init_actions()
    
    self._finish_init_and_show()
    
    pg.gui.set_gui_excepthook_additional_callback(
      self._display_inline_message_on_setting_value_error)
    
    if not run_gui_func:
      gtk.main()
    else:
      run_gui_func(self, self._dialog, self._settings)
  
  @property
  def name_preview(self):
    return self._name_preview
  
  @property
  def image_preview(self):
    return self._image_preview
  
  @property
  def folder_chooser(self):
    return self._folder_chooser
  
  def _init_settings(self):
    self._settings['main/procedures'].tags.add('ignore_load')
    self._settings['main/constraints'].tags.add('ignore_load')
    
    load_result = self._settings.load()
    load_messages = '\n\n'.join(
      message for message in load_result.messages_per_source.values() if message)
    if pg.setting.Persistor.FAIL in load_result.statuses_per_source.values():
      messages_.display_message(load_messages, gtk.MESSAGE_WARNING)
    
    _setup_image_ids_and_directories_and_initial_directory(
      self._settings, self._settings['gui/current_directory'], self._image)
    _setup_output_directory_changed(self._settings, self._image)
  
  def _init_actions(self):
    self._settings['main/procedures'].tags.discard('ignore_load')
    result = self._settings['main/procedures'].load()
    
    if self._settings['main/procedures'] in result.settings_not_loaded:
      actions.clear(self._settings['main/procedures'], add_initial_actions=True)
    
    self._settings['main/constraints'].tags.discard('ignore_load')
    result = self._settings['main/constraints'].load()
    
    if self._settings['main/constraints'] in result.settings_not_loaded:
      actions.clear(self._settings['main/constraints'], add_initial_actions=True)
  
  def _init_gui_elements(self):
    self._dialog = gimpui.Dialog(title=pg.config.PLUGIN_TITLE, role=pg.config.PLUGIN_NAME)
    self._dialog.set_transient()
    self._dialog.set_default_size(*self._DIALOG_SIZE)
    self._dialog.set_border_width(self._DIALOG_BORDER_WIDTH)
    self._dialog.set_default_response(gtk.RESPONSE_CANCEL)
    
    pg.gui.set_gui_excepthook_parent(self._dialog)
    
    self._folder_chooser_label = gtk.Label()
    self._folder_chooser_label.set_markup('<b>' + _('Save in folder:') + '</b>')
    self._folder_chooser_label.set_alignment(0.0, 0.5)
    
    self._folder_chooser = gtk.FileChooserWidget(action=gtk.FILE_CHOOSER_ACTION_SELECT_FOLDER)
    
    self._vbox_folder_chooser = gtk.VBox(homogeneous=False)
    self._vbox_folder_chooser.set_spacing(self._DIALOG_VBOX_SPACING)
    self._vbox_folder_chooser.pack_start(
      self._folder_chooser_label,
      expand=False,
      fill=False,
      padding=self._SAVE_IN_FOLDER_LABEL_PADDING)
    self._vbox_folder_chooser.pack_start(self._folder_chooser)
    
    self._init_gui_previews()
    
    self._preview_label = gtk.Label()
    self._preview_label.set_markup('<b>' + _('Preview') + '</b>')
    self._preview_label.set_alignment(0.0, 0.5)
    
    self._hbox_preview_label = gtk.HBox()
    self._hbox_preview_label.set_border_width(self._PREVIEW_LABEL_BORDER_WIDTH)
    self._hbox_preview_label.pack_start(self._preview_label)
    
    self._vpaned_previews = gtk.VPaned()
    self._vpaned_previews.pack1(self._name_preview, resize=True, shrink=True)
    self._vpaned_previews.pack2(self._image_preview, resize=True, shrink=True)
    
    self._vbox_previews = gtk.VBox()
    self._vbox_previews.pack_start(self._hbox_preview_label, expand=False, fill=False)
    self._vbox_previews.pack_start(self._vpaned_previews, expand=True, fill=True)
    
    self._frame_previews = gtk.Frame()
    self._frame_previews.set_shadow_type(gtk.SHADOW_ETCHED_OUT)
    self._frame_previews.add(self._vbox_previews)
    
    self._file_extension_label = gtk.Label()
    self._file_extension_label.set_markup(
      '<b>{}:</b>'.format(
        gobject.markup_escape_text(self._settings['main/file_extension'].display_name)))
    self._file_extension_label.set_alignment(0.0, 0.5)
    
    self._file_extension_entry = pg.gui.FileExtensionEntry(
      minimum_width_chars=self._FILE_EXTENSION_ENTRY_MIN_WIDTH_CHARS,
      maximum_width_chars=self._FILE_EXTENSION_ENTRY_MAX_WIDTH_CHARS)
    self._file_extension_entry.set_activates_default(True)
    
    self._save_as_label = gtk.Label()
    self._save_as_label.set_markup('<b>{}:</b>'.format(gobject.markup_escape_text(_('Save as'))))
    self._save_as_label.set_alignment(0.0, 0.5)
    
    self._dot_label = gtk.Label('.')
    self._dot_label.set_alignment(0.0, 1.0)
    
    self._filename_pattern_entry = pg.gui.FilenamePatternEntry(
      renamer_.get_field_descriptions(renamer_.FIELDS),
      minimum_width_chars=self._FILENAME_PATTERN_ENTRY_MIN_WIDTH_CHARS,
      maximum_width_chars=self._FILENAME_PATTERN_ENTRY_MAX_WIDTH_CHARS,
      default_item=self._settings['main/layer_filename_pattern'].default_value)
    self._filename_pattern_entry.set_activates_default(True)
    
    self._label_message = message_label_.MessageLabel()
    
    self._hbox_export_name_labels = gtk.HBox(homogeneous=False)
    self._hbox_export_name_labels.pack_start(self._file_extension_label, expand=False, fill=False)
    self._hbox_export_name_labels.pack_start(self._save_as_label, expand=False, fill=False)
    
    self._hbox_export_name_entries = gtk.HBox(homogeneous=False)
    self._hbox_export_name_entries.set_spacing(self._HBOX_EXPORT_NAME_ENTRIES_SPACING)
    self._hbox_export_name_entries.pack_start(
      self._filename_pattern_entry, expand=False, fill=False)
    self._hbox_export_name_entries.pack_start(self._dot_label, expand=False, fill=False)
    self._hbox_export_name_entries.pack_start(self._file_extension_entry, expand=False, fill=False)
    
    self._hbox_export_name = gtk.HBox(homogeneous=False)
    self._hbox_export_name.set_spacing(self._HBOX_EXPORT_LABELS_NAME_SPACING)
    self._hbox_export_name.pack_start(self._hbox_export_name_labels, expand=False, fill=False)
    self._hbox_export_name.pack_start(self._hbox_export_name_entries, expand=False, fill=False)
    
    self._hbox_export_name_and_message = gtk.HBox(homogeneous=False)
    self._hbox_export_name_and_message.set_spacing(
      self._HBOX_EXPORT_NAME_AND_MESSAGE_HORIZONTAL_SPACING)
    self._hbox_export_name_and_message.set_border_width(
      self._HBOX_EXPORT_NAME_AND_MESSAGE_BORDER_WIDTH)
    self._hbox_export_name_and_message.pack_start(self._hbox_export_name, expand=False, fill=False)
    self._hbox_export_name_and_message.pack_start(self._label_message, expand=True, fill=True)
    
    self._box_procedures = actions_.ActionBox(
      self._settings['main/procedures'],
      builtin_procedures.BUILTIN_PROCEDURES,
      _('Add P_rocedure...'),
      _('Edit Procedure'),
      add_custom_action_text=_('Add Custom Procedure...'))
    
    self._box_constraints = actions_.ActionBox(
      self._settings['main/constraints'],
      builtin_constraints.BUILTIN_CONSTRAINTS,
      _('Add C_onstraint...'),
      _('Edit Constraint'),
      allow_custom_actions=False)
    
    self._hbox_actions = gtk.HBox(homogeneous=True)
    self._hbox_actions.set_spacing(self._MORE_SETTINGS_HORIZONTAL_SPACING)
    self._hbox_actions.set_border_width(self._MORE_SETTINGS_BORDER_WIDTH)
    self._hbox_actions.pack_start(self._box_procedures, expand=True, fill=True)
    self._hbox_actions.pack_start(self._box_constraints, expand=True, fill=True)
    
    self._label_message_for_edit_mode = message_label_.MessageLabel()
    
    self._vbox_actions_and_message_for_edit_mode = gtk.VBox(homogeneous=False)
    self._vbox_actions_and_message_for_edit_mode.pack_start(
      self._hbox_actions, expand=True, fill=True)
    self._vbox_actions_and_message_for_edit_mode.pack_start(
      self._label_message_for_edit_mode, expand=False, fill=False)
    
    self._vbox_chooser_and_settings = gtk.VBox()
    self._vbox_chooser_and_settings.set_spacing(self._DIALOG_VBOX_SPACING)
    self._vbox_chooser_and_settings.pack_start(self._vbox_folder_chooser, expand=True, fill=True)
    self._vbox_chooser_and_settings.pack_start(
      self._hbox_export_name_and_message, expand=False, fill=False)
    
    self._vpaned_chooser_and_actions = gtk.VPaned()
    self._vpaned_chooser_and_actions.pack1(
      self._vbox_chooser_and_settings, resize=True, shrink=False)
    self._vpaned_chooser_and_actions.pack2(
      self._vbox_actions_and_message_for_edit_mode, resize=False, shrink=True)
    
    self._hpaned_settings_and_previews = gtk.HPaned()
    self._hpaned_settings_and_previews.pack1(
      self._vpaned_chooser_and_actions, resize=True, shrink=False)
    self._hpaned_settings_and_previews.pack2(
      self._frame_previews, resize=True, shrink=True)
    
    self._button_run = self._dialog.add_button(_('_Export'), gtk.RESPONSE_OK)
    self._button_run.set_flags(gtk.CAN_DEFAULT)
    self._button_run.hide()
    
    self._button_close = self._dialog.add_button(_('_Cancel'), gtk.RESPONSE_CANCEL)
    self._button_close.hide()
    
    self._dialog.set_alternative_button_order([gtk.RESPONSE_OK, gtk.RESPONSE_CANCEL])
    
    self._button_stop = gtk.Button()
    self._button_stop.set_label(_('_Stop'))
    self._button_stop.set_no_show_all(True)
    
    self._label_button_settings = gtk.Label(_('_Settings'))
    self._label_button_settings.set_use_underline(True)
    
    self._hbox_button_settings = gtk.HBox()
    self._hbox_button_settings.pack_start(self._label_button_settings, expand=True, fill=True)
    self._hbox_button_settings.pack_start(
      gtk.Arrow(gtk.ARROW_DOWN, gtk.SHADOW_IN), expand=False, fill=False)
    
    self._button_settings = gtk.Button()
    self._button_settings.add(self._hbox_button_settings)
    
    self._menu_item_show_more_settings = gtk.CheckMenuItem(_('Show More Settings'))
    self._menu_item_edit_mode = gtk.CheckMenuItem(_('Batch Editing'))
    self._menu_item_save_settings = gtk.MenuItem(_('Save Settings'))
    self._menu_item_reset_settings = gtk.MenuItem(_('Reset settings'))
    self._menu_item_import_settings = gtk.MenuItem(_('Import Settings...'))
    self._menu_item_export_settings = gtk.MenuItem(_('Export Settings...'))
    
    self._menu_settings = gtk.Menu()
    self._menu_settings.append(self._menu_item_show_more_settings)
    self._menu_settings.append(self._menu_item_edit_mode)
    self._menu_settings.append(self._menu_item_save_settings)
    self._menu_settings.append(self._menu_item_reset_settings)
    self._menu_settings.append(self._menu_item_import_settings)
    self._menu_settings.append(self._menu_item_export_settings)
    self._menu_settings.show_all()
    
    self._dialog.action_area.pack_end(self._button_stop, expand=False, fill=False)
    self._dialog.action_area.pack_start(self._button_settings, expand=False, fill=False)
    self._dialog.action_area.set_child_secondary(self._button_settings, True)
    
    if _webbrowser_module_found:
      self._button_help = gtk.Button()
      self._button_help.set_label(_('_Help'))
      self._dialog.action_area.pack_start(self._button_help, expand=False, fill=False)
      self._dialog.action_area.set_child_secondary(self._button_help, True)
    
    self._progress_bar = gtk.ProgressBar()
    self._progress_bar.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
    self._progress_bar.set_no_show_all(True)
    
    self._hbox_contents = gtk.HBox()
    self._hbox_contents.pack_start(self._hpaned_settings_and_previews, expand=True, fill=True)
    self._hbox_contents.set_border_width(self._DIALOG_CONTENTS_BORDER_WIDTH)
    
    self._dialog.vbox.set_spacing(self._DIALOG_VBOX_SPACING)
    self._dialog.vbox.pack_start(self._hbox_contents, expand=True, fill=True)
    self._dialog.vbox.pack_end(self._progress_bar, expand=False, fill=False)
    
    # Move the action area above the progress bar.
    self._dialog.vbox.reorder_child(self._dialog.action_area, -1)
  
  def _connect_events(self):
    self._box_procedures.connect(
      'action-box-item-added', self._on_box_procedures_item_added)
    
    self._button_run.connect('clicked', self._on_button_run_clicked, 'processing')
    self._button_close.connect('clicked', self._on_button_close_clicked)
    self._button_stop.connect('clicked', self._on_button_stop_clicked)
    
    if _webbrowser_module_found:
      self._button_help.connect('clicked', self._on_button_help_clicked)
    
    self._button_settings.connect('clicked', self._on_button_settings_clicked)
    self._menu_item_show_more_settings.connect(
      'toggled', self._on_menu_item_show_more_settings_toggled)
    self._menu_item_edit_mode.connect('toggled', self._on_menu_item_edit_mode_toggled)
    self._menu_item_save_settings.connect('activate', self._on_save_settings_activate)
    self._menu_item_reset_settings.connect('activate', self._on_reset_settings_activate)
    self._menu_item_import_settings.connect('activate', self._on_import_settings_activate)
    self._menu_item_export_settings.connect('activate', self._on_export_settings_activate)
    
    self._file_extension_entry.connect(
      'changed',
      self._on_text_entry_changed,
      self._settings['main/file_extension'],
      'invalid_file_extension')
    self._file_extension_entry.connect(
      'focus-out-event',
      self._on_file_extension_entry_focus_out_event,
      self._settings['main/file_extension'])
    
    self._filename_pattern_entry.connect(
      'changed',
      self._on_text_entry_changed,
      self._settings['main/layer_filename_pattern'],
      'invalid_layer_filename_pattern')
    
    self._dialog.connect('key-press-event', self._on_dialog_key_press_event)
    self._dialog.connect('delete-event', self._on_dialog_delete_event)
    self._dialog.connect('notify::is-active', self._on_dialog_notify_is_active)
    
    self._hpaned_settings_and_previews.connect(
      'notify::position',
      self._previews_controller.on_paned_outside_previews_notify_position)
    self._vpaned_previews.connect(
      'notify::position',
      self._previews_controller.on_paned_between_previews_notify_position)
    
    self._previews_controller.connect_setting_changes_to_previews()
    self._previews_controller.connect_name_preview_events()
    
    self._image_preview.connect('preview-updated', self._on_image_preview_updated)
    self._name_preview.connect('preview-updated', self._on_name_preview_updated)
  
  def _finish_init_and_show(self):
    while gtk.events_pending():
      gtk.main_iteration()
    
    self._dialog.vbox.show_all()
    self._show_hide_more_settings()
    self._update_gui_for_edit_mode(update_name_preview=False)
    
    if not self._settings['main/edit_mode'].value:
      self._dialog.set_focus(self._file_extension_entry)
    
    self._button_run.grab_default()
    # Place the cursor at the end of the text entry.
    self._file_extension_entry.set_position(-1)
    
    self._dialog.show()
  
  def _assign_gui_to_settings(self):
    self._settings.initialize_gui({
      'main/file_extension': [
        pg.setting.SettingGuiTypes.extended_entry, self._file_extension_entry],
      'main/layer_filename_pattern': [
        pg.setting.SettingGuiTypes.extended_entry, self._filename_pattern_entry],
      'main/edit_mode': [
        pg.setting.SettingGuiTypes.check_menu_item, self._menu_item_edit_mode],
      'gui/show_more_settings': [
        pg.setting.SettingGuiTypes.check_menu_item, self._menu_item_show_more_settings],
      'gui/image_preview_automatic_update': [
        pg.setting.SettingGuiTypes.check_menu_item,
        self._image_preview.menu_item_update_automatically],
      'gui/size/dialog_position': [
        pg.setting.SettingGuiTypes.window_position, self._dialog],
      'gui/size/dialog_size': [
        pg.setting.SettingGuiTypes.window_size, self._dialog],
      'gui/size/paned_outside_previews_position': [
        pg.setting.SettingGuiTypes.paned_position, self._hpaned_settings_and_previews],
      'gui/size/paned_between_previews_position': [
        pg.setting.SettingGuiTypes.paned_position, self._vpaned_previews],
      'gui/size/settings_vpane_position': [
        pg.setting.SettingGuiTypes.paned_position, self._vpaned_chooser_and_actions],
      'gui/current_directory': [
        pg.setting.SettingGuiTypes.folder_chooser_widget, self._folder_chooser],
    })
  
  def _init_gui_previews(self):
    self._name_preview = preview_name_.NamePreview(
      self._batcher_for_previews,
      self._settings,
      self._initial_layer_tree,
      self._settings['gui/name_preview_layers_collapsed_state'].value[self._image.ID],
      self._settings['main/selected_layers'].value[self._image.ID],
      'selected_in_preview',
      self._settings['main/available_tags'])
    
    self._image_preview = preview_image_.ImagePreview(
      self._batcher_for_previews, self._settings)
    
    self._previews_controller = previews_controller_.PreviewsController(
      self._name_preview, self._image_preview, self._settings, self._image)
  
  def _load_settings(self, filepath, file_format, load_size_settings=True):
    if file_format == 'pkl' or not _json_module_found:
      source = pg.setting.sources.PickleFileSource(pg.config.SOURCE_NAME, filepath)
    else:
      source = pg.setting.sources.JsonFileSource(pg.config.SOURCE_NAME, filepath)
    
    actions.clear(self._settings['main/procedures'], add_initial_actions=False)
    actions.clear(self._settings['main/constraints'], add_initial_actions=False)
    
    settings_to_ignore_for_reset = []
    for setting in self._settings.walk(lambda s: 'ignore_reset' not in s.tags):
      if ((setting.setting_sources is not None and list(setting.setting_sources) == ['session'])
          or setting.get_path('root').startswith('gui/size')):
        setting.tags.add('ignore_reset')
        settings_to_ignore_for_reset.append(setting)
    
    self._reset_settings()
    
    for setting in settings_to_ignore_for_reset:
      setting.tags.discard('ignore_reset')
    
    status, message = update.update(
      self._settings, handle_invalid='abort', sources={'persistent': source})
    if status == update.ABORT:
      messages_.display_import_export_settings_failure_message(
        _(('Failed to import settings from file "{}".'
           ' Settings must be reset completely.').format(filepath)),
        details=message,
        parent=self._dialog)
      
      self._reset_settings()
      actions.clear(self._settings['main/procedures'])
      actions.clear(self._settings['main/constraints'])
      return False
    
    size_settings_to_ignore_for_load = []
    if not load_size_settings:
      for setting in self._settings['gui'].walk(lambda s: 'ignore_load' not in s.tags):
        if setting.get_path('root').startswith('gui/size'):
          setting.tags.add('ignore_load')
          size_settings_to_ignore_for_load.append(setting)
    
    load_result = self._settings.load({'persistent': source})
    
    for setting in size_settings_to_ignore_for_load:
      setting.tags.discard('ignore_load')
    
    if any(status in load_result.statuses_per_source.values()
           for status in [pg.setting.Persistor.SOURCE_NOT_FOUND, pg.setting.Persistor.FAIL]):
      messages_.display_import_export_settings_failure_message(
        _('Failed to import settings from file "{}"'.format(filepath)),
        details='\n\n'.join(
          message for message in load_result.messages_per_source.values() if message),
        parent=self._dialog)
      return False
    else:
      return True
  
  def _save_settings(self, filepath=None, file_format='json'):
    if filepath is None:
      save_result = self._settings.save()
    else:
      if file_format == 'pkl' or not _json_module_found:
        source = pg.setting.sources.PickleFileSource(pg.config.SOURCE_NAME, filepath)
      else:
        source = pg.setting.sources.JsonFileSource(pg.config.SOURCE_NAME, filepath)
      
      save_result = self._settings.save({'persistent': source})
    
    if pg.setting.Persistor.FAIL in save_result.statuses_per_source.values():
      messages_.display_import_export_settings_failure_message(
        _('Failed to export settings to file "{}"'.format(filepath)),
        details='\n\n'.join(
          message for message in save_result.messages_per_source.values() if message),
        parent=self._dialog)
      return False
    else:
      return True
  
  @_set_settings
  def _save_settings_to_default_location(self):
    save_successful = self._save_settings()
    if save_successful:
      self._display_inline_message(_('Settings successfully saved.'), gtk.MESSAGE_INFO)
  
  def _reset_settings(self):
    self._settings.reset()
  
  def _get_setting_filepath(self, action, add_file_extension_if_missing=True):
    if action == 'import':
      dialog_action = gtk.FILE_CHOOSER_ACTION_OPEN
      button_ok = gtk.STOCK_OPEN
      title = _('Import Settings')
    elif action == 'export':
      dialog_action = gtk.FILE_CHOOSER_ACTION_SAVE
      button_ok = gtk.STOCK_SAVE
      title = _('Export Settings')
    else:
      raise ValueError('invalid action; valid values: "import", "export"')
    
    file_dialog = gtk.FileChooserDialog(
      title=title,
      parent=self._dialog,
      action=dialog_action)
    
    file_dialog.set_do_overwrite_confirmation(True)
    
    file_dialog.add_buttons(
      button_ok, gtk.RESPONSE_OK,
      gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL)
    
    file_dialog.set_alternative_button_order([gtk.RESPONSE_OK, gtk.RESPONSE_CANCEL])
    
    if action == 'import':
      check_button_load_size_settings = gtk.CheckButton(_('Import size-related settings'))
      check_button_load_size_settings.set_border_width(
        self._IMPORT_SETTINGS_CUSTOM_WIDGETS_BORDER_WIDTH)
      file_dialog.vbox.pack_start(check_button_load_size_settings, expand=False, fill=False)
    else:
      check_button_load_size_settings = None
    
    json_file_ext = '.json'
    pickle_file_ext = '.pkl'
    
    if _json_module_found:
      filter_json = gtk.FileFilter()
      filter_json.set_name(_('JSON file ({})').format(json_file_ext))
      filter_json.add_mime_type('application/json')
      file_dialog.add_filter(filter_json)
      
      default_file_ext = json_file_ext
      default_file_format = json_file_ext[1:]
    else:
      default_file_ext = pickle_file_ext
      default_file_format = pickle_file_ext[1:]
    
    filter_pickle = gtk.FileFilter()
    filter_pickle.set_name(_('Pickle file ({})').format(pickle_file_ext))
    filter_pickle.add_pattern('*' + pickle_file_ext)
    file_dialog.add_filter(filter_pickle)
    
    filter_any = gtk.FileFilter()
    filter_any.set_name(_('Any file'))
    filter_any.add_pattern('*')
    file_dialog.add_filter(filter_any)
    
    file_dialog.show_all()
    
    response_id = file_dialog.run()
    
    if response_id == gtk.RESPONSE_OK:
      filepath = pg.utils.safe_decode_gtk(file_dialog.get_filename())
      
      file_ext = os.path.splitext(filepath)[1]
      if add_file_extension_if_missing and not file_ext:
        if file_dialog.get_filter() == filter_pickle:
          filepath += pickle_file_ext
        else:
          filepath += default_file_ext
      
      file_ext = os.path.splitext(filepath)[1]
      if file_ext:
        file_format = file_ext[1:]
      else:
        file_format = default_file_format
    else:
      filepath = None
      file_format = None
    
    if check_button_load_size_settings:
      load_size_settings = check_button_load_size_settings.get_active()
    else:
      load_size_settings = False
    
    file_dialog.destroy()
    
    return filepath, file_format, load_size_settings
  
  def _on_text_entry_changed(self, entry, setting, name_preview_lock_update_key=None):
    try:
      setting.gui.update_setting_value()
    except pg.setting.SettingValueError as e:
      pg.invocation.timeout_add_strict(
        self._DELAY_NAME_PREVIEW_UPDATE_TEXT_ENTRIES_MILLISECONDS,
        self._name_preview.set_sensitive, False)
      self._display_inline_message(str(e), gtk.MESSAGE_ERROR, setting)
      self._name_preview.lock_update(True, name_preview_lock_update_key)
    else:
      self._name_preview.lock_update(False, name_preview_lock_update_key)
      if self._message_setting == setting:
        self._display_inline_message(None)
      
      self._name_preview.add_function_at_update(
        self._name_preview.set_sensitive, True)
      
      pg.invocation.timeout_add_strict(
        self._DELAY_NAME_PREVIEW_UPDATE_TEXT_ENTRIES_MILLISECONDS,
        self._name_preview.update)
  
  def _on_file_extension_entry_focus_out_event(self, entry, event, setting):
    setting.apply_to_gui()
  
  def _on_box_procedures_item_added(self, box_procedures, item):
    if any(item.action['orig_name'].value == name
           for name in ['insert_background_layers', 'insert_foreground_layers']):
      actions.reorder(self._settings['main/procedures'], item.action.name, 0)
  
  def _on_menu_item_show_more_settings_toggled(self, menu_item):
    self._show_hide_more_settings()
  
  def _show_hide_more_settings(self):
    if self._menu_item_show_more_settings.get_active():
      self._vbox_actions_and_message_for_edit_mode.show()
      
      self._file_extension_label.hide()
      self._save_as_label.show()
      self._dot_label.show()
      self._filename_pattern_entry.show()
    else:
      self._settings['main/edit_mode'].set_value(False)
      
      self._vbox_actions_and_message_for_edit_mode.hide()
      
      self._file_extension_label.show()
      self._save_as_label.hide()
      self._dot_label.hide()
      self._filename_pattern_entry.hide()
  
  def _on_menu_item_edit_mode_toggled(self, menu_item):
    self._update_gui_for_edit_mode()
  
  def _update_gui_for_edit_mode(self, update_name_preview=True):
    if self._menu_item_edit_mode.get_active():
      self._settings['gui/show_more_settings'].set_value(True)
      
      self._vbox_chooser_and_settings.hide()
      self._label_message_for_edit_mode.show()
      
      self._button_run.set_label(_('Run'))
      self._button_close.set_label(_('Close'))
    else:
      self._vbox_chooser_and_settings.show()
      self._label_message_for_edit_mode.hide()
      
      self._button_run.set_label(_('Export'))
      self._button_close.set_label(_('Cancel'))
    
    if update_name_preview:
      self._name_preview.update()
  
  def _on_dialog_notify_is_active(self, dialog, property_spec):
    if not pdb.gimp_image_is_valid(self._image):
      gtk.main_quit()
      return
    
    if self._initial_layer_tree is not None:
      self._initial_layer_tree = None
      return
  
  def _on_image_preview_updated(self, preview, error, update_duration_seconds):
    self._display_warnings_and_tooltips_for_actions()
    
    if (self._settings[
         'gui/image_preview_automatic_update_if_below_maximum_duration'].value
        and (update_duration_seconds
             >= self._MAXIMUM_IMAGE_PREVIEW_AUTOMATIC_UPDATE_DURATION_SECONDS)):
      self._settings['gui/image_preview_automatic_update'].set_value(False)
      
      self._display_inline_message(
        '{}\n\n{}'.format(
          _('Disabling automatic preview update.'),
          _('The preview takes too long to update.'
            ' You may turn automatic updates back on from the menu above the previewed image.')),
        gtk.MESSAGE_INFO)
  
  def _on_name_preview_updated(self, preview, error):
    self._display_warnings_and_tooltips_for_actions(clear_previous=False)
  
  def _display_warnings_and_tooltips_for_actions(self, clear_previous=True):
    self._set_warning_on_actions(self._batcher_for_previews, clear_previous=clear_previous)
    
    self._set_action_skipped_tooltips(
      self._box_procedures,
      self._batcher_for_previews.skipped_procedures,
      _('This procedure is skipped. Reason: {}'),
      clear_previous=clear_previous)
    
    self._set_action_skipped_tooltips(
      self._box_constraints,
      self._batcher_for_previews.skipped_constraints,
      _('This constraint is skipped. Reason: {}'),
      clear_previous=clear_previous)
  
  def _on_dialog_key_press_event(self, dialog, event):
    if gtk.gdk.keyval_name(event.keyval) == 'Escape':
      stopped = stop_batcher(self._batcher)
      return stopped
    
    # Ctrl + S is pressed
    if ((event.state & gtk.accelerator_get_default_mod_mask()) == gtk.gdk.CONTROL_MASK
        and gtk.gdk.keyval_name(gtk.gdk.keyval_to_lower(event.keyval)) == 's'):
      self._save_settings_to_default_location()
      return True
    
    return False
  
  def _on_button_settings_clicked(self, button):
    pg.gui.menu_popup_below_widget(self._menu_settings, button)
  
  def _on_save_settings_activate(self, menu_item):
    self._save_settings_to_default_location()
  
  def _on_import_settings_activate(self, menu_item):
    filepath, file_format, load_size_settings = self._get_setting_filepath(action='import')
    
    if filepath is not None:
      import_successful = self._load_settings(filepath, file_format, load_size_settings)
      # Also override default setting sources so that the imported settings actually persist.
      self._save_settings()
      
      if import_successful:
        self._display_inline_message(_('Settings successfully imported.'), gtk.MESSAGE_INFO)
  
  @_set_settings
  def _on_export_settings_activate(self, menu_item):
    filepath, file_format, unused_ = self._get_setting_filepath(action='export')
    
    if filepath is not None:
      export_successful = self._save_settings(filepath, file_format)
      if export_successful:
        self._display_inline_message(_('Settings successfully exported.'), gtk.MESSAGE_INFO)
  
  def _on_reset_settings_activate(self, menu_item):
    response_id, clear_actions = display_reset_prompt(
      parent=self._dialog,
      more_settings_shown=self._settings['gui/show_more_settings'].value)
    
    if response_id == gtk.RESPONSE_YES:
      if clear_actions:
        actions.clear(self._settings['main/procedures'])
        actions.clear(self._settings['main/constraints'])
      else:
        self._settings['main/procedures'].tags.add('ignore_reset')
        self._settings['main/constraints'].tags.add('ignore_reset')
      
      self._reset_settings()
      self._save_settings()
      
      if clear_actions:
        utils_.clear_setting_sources(self._settings)
      else:
        self._settings['main/procedures'].tags.remove('ignore_reset')
        self._settings['main/constraints'].tags.remove('ignore_reset')
      
      self._display_inline_message(_('Settings reset.'), gtk.MESSAGE_INFO)
  
  @_set_settings
  def _on_button_run_clicked(self, button, lock_update_key):
    self._setup_gui_before_batch_run()
    self._batcher, overwrite_chooser, progress_updater = self._setup_batcher()
    
    item_progress_indicator = progress_.ItemProgressIndicator(
      self._progress_bar, progress_updater)
    item_progress_indicator.install_progress_for_status(
      self._progress_set_value_and_show_dialog)
    
    should_quit = True
    self._name_preview.lock_update(True, lock_update_key)
    self._image_preview.lock_update(True, lock_update_key)
    
    try:
      self._batcher.run(**utils_.get_settings_for_batcher(self._settings['main']))
    except exceptions.BatcherCancelError as e:
      should_quit = False
    except exceptions.ActionError as e:
      messages_.display_failure_message(
        messages_.get_failing_action_message(e),
        failure_message=str(e),
        details=traceback.format_exc(),
        parent=self._dialog)
      should_quit = False
    except exceptions.BatcherError as e:
      messages_.display_processing_failure_message(e, parent=self._dialog)
      should_quit = False
    except Exception as e:
      if pdb.gimp_image_is_valid(self._image):
        raise
      else:
        messages_.display_invalid_image_failure_message(parent=self._dialog)
    else:
      self._settings['special/first_plugin_run'].set_value(False)
      self._settings['special/first_plugin_run'].save()
      
      if self._settings['main/edit_mode'].value or not self._batcher.exported_raw_items:
        should_quit = False
      
      if not self._settings['main/edit_mode'].value and not self._batcher.exported_raw_items:
        messages_.display_message(
          _('No layers were exported.'), gtk.MESSAGE_INFO, parent=self._dialog)
    finally:
      item_progress_indicator.uninstall_progress_for_status()
      self._name_preview.lock_update(False, lock_update_key)
      self._image_preview.lock_update(False, lock_update_key)
      
      if self._settings['main/edit_mode'].value:
        self._image_preview.update()
        self._name_preview.update(reset_items=True)
      
      self._set_warning_on_actions(self._batcher)
      
      self._batcher = None
    
    if overwrite_chooser.overwrite_mode in self._settings['main/overwrite_mode'].items.values():
      self._settings['main/overwrite_mode'].set_value(overwrite_chooser.overwrite_mode)
    
    self._settings['main'].save(['session'])
    self._settings['gui'].save(['session'])
    
    if should_quit:
      gtk.main_quit()
    else:
      self._restore_gui_after_batch_run()
      progress_updater.reset()
  
  def _setup_gui_before_batch_run(self):
    self._display_inline_message(None)
    self._reset_action_tooltips_and_indicators()
    self._close_action_edit_dialogs()
    self._set_gui_enabled(False)
  
  def _restore_gui_after_batch_run(self):
    self._set_gui_enabled(True)
  
  def _setup_batcher(self):
    overwrite_chooser = pg.gui.GtkDialogOverwriteChooser(
      self._get_overwrite_dialog_items(),
      default_value=self._settings['main/overwrite_mode'].items['replace'],
      default_response=pg.overwrite.OverwriteModes.CANCEL,
      title=pg.config.PLUGIN_TITLE,
      parent=self._dialog)
    
    progress_updater = pg.gui.GtkProgressUpdater(self._progress_bar)
    
    batcher = batcher_.Batcher(
      gimpenums.RUN_INTERACTIVE,
      self._image,
      self._settings['main/procedures'],
      self._settings['main/constraints'],
      overwrite_chooser=overwrite_chooser,
      progress_updater=progress_updater,
      export_context_manager=handle_gui_in_export,
      export_context_manager_args=[self._dialog])
    
    return batcher, overwrite_chooser, progress_updater
  
  def _get_overwrite_dialog_items(self):
    return list(zip(
      self._settings['main/overwrite_mode'].items.values(),
      self._settings['main/overwrite_mode'].items_display_names.values()))
  
  def _set_gui_enabled(self, enabled):
    self._progress_bar.set_visible(not enabled)
    self._button_stop.set_visible(not enabled)
    self._button_close.set_visible(enabled)
    
    for child in self._dialog.vbox:
      if child not in (self._dialog.action_area, self._progress_bar):
        child.set_sensitive(enabled)
    
    self._button_settings.set_sensitive(enabled)
    
    for button in self._dialog.action_area:
      if button != self._button_stop:
        button.set_sensitive(enabled)
    
    if enabled:
      self._dialog.set_focus(self._file_extension_entry)
      self._file_extension_entry.set_position(-1)
    else:
      self._dialog.set_focus(self._button_stop)
  
  def _progress_set_value_and_show_dialog(self, fraction):
    self._progress_bar.set_fraction(fraction)
    
    # Without this workaround, the main dialog would not appear until the export
    # of the second layer.
    if not self._dialog.get_mapped():
      self._dialog.show()
    
    while gtk.events_pending():
      gtk.main_iteration()
  
  def _on_dialog_delete_event(self, dialog, event):
    gtk.main_quit()
  
  def _on_button_close_clicked(self, button):
    gtk.main_quit()
  
  def _on_button_stop_clicked(self, button):
    stop_batcher(self._batcher)
  
  def _on_button_help_clicked(self, button):
    if os.path.isfile(pg.config.LOCAL_DOCS_PATH):
      docs_url = pg.config.LOCAL_DOCS_PATH
    else:
      docs_url = pg.config.DOCS_URL
    
    webbrowser.open_new_tab(docs_url)
  
  def _set_action_skipped_tooltips(self, action_box, skipped_actions, message, clear_previous=True):
    for box_item in action_box.items:
      if not box_item.has_warning():
        if box_item.action.name in skipped_actions:
          skipped_message = skipped_actions[box_item.action.name][0][1]
          box_item.set_tooltip(message.format(skipped_message))
        else:
          if clear_previous:
            box_item.set_tooltip(None)
  
  def _set_warning_on_actions(self, batcher, clear_previous=True):
    action_boxes = [self._box_procedures, self._box_constraints]
    failed_actions_dict = [batcher.failed_procedures, batcher.failed_constraints]
    
    for action_box, failed_actions in zip(action_boxes, failed_actions_dict):
      for box_item in action_box.items:
        if box_item.action.name in failed_actions:
          box_item.set_warning(
            True,
            messages_.get_failing_action_message(
              (box_item.action, failed_actions[box_item.action.name][0][0])),
            failed_actions[box_item.action.name][0][1],
            failed_actions[box_item.action.name][0][2],
            parent=self._dialog)
        else:
          if clear_previous:
            box_item.set_warning(False)
  
  def _reset_action_tooltips_and_indicators(self):
    for action_box in [self._box_procedures, self._box_constraints]:
      for box_item in action_box.items:
        box_item.set_tooltip(None)
        box_item.set_warning(False)
  
  def _close_action_edit_dialogs(self):
    for action_box in [self._box_procedures, self._box_constraints]:
      for box_item in action_box.items:
        box_item.close_edit_dialog()
  
  def _display_inline_message(self, text, message_type=gtk.MESSAGE_ERROR, setting=None):
    self._message_setting = setting
    
    if self._settings['main/edit_mode'].value:
      label_message = self._label_message_for_edit_mode
    else:
      label_message = self._label_message
    
    label_message.set_text(text, message_type, self._DELAY_CLEAR_LABEL_MESSAGE_MILLISECONDS)
  
  def _display_inline_message_on_setting_value_error(
        self, exc_type, exc_value, exc_traceback):
    if issubclass(exc_type, pg.setting.SettingValueError):
      self._display_inline_message(str(exc_value), gtk.MESSAGE_ERROR)
      return True
    else:
      return False


class ExportLayersRepeatDialog(object):
  
  _BORDER_WIDTH = 8
  _HBOX_HORIZONTAL_SPACING = 8
  _DIALOG_WIDTH = 500
  
  def __init__(self, layer_tree, settings):
    self._layer_tree = layer_tree
    self._settings = settings
    
    self._image = self._layer_tree.image
    self._batcher = None
    
    self._settings.load(['session'])
    
    self._init_gui()
    
    pg.gui.set_gui_excepthook_parent(self._dialog)
    
    gtk.main_iteration()
    self.show()
    self.run_batcher()
  
  def _init_gui(self):
    self._dialog = gimpui.Dialog(title=pg.config.PLUGIN_TITLE, role=None)
    self._dialog.set_transient()
    self._dialog.set_border_width(self._BORDER_WIDTH)
    self._dialog.set_default_size(self._DIALOG_WIDTH, -1)
    
    self._button_stop = gtk.Button()
    self._button_stop.set_label(_('_Stop'))
    
    self._buttonbox = gtk.HButtonBox()
    self._buttonbox.pack_start(self._button_stop, expand=False, fill=False)
    
    self._progress_bar = gtk.ProgressBar()
    self._progress_bar.set_ellipsize(pango.ELLIPSIZE_MIDDLE)
    
    self._hbox_action_area = gtk.HBox(homogeneous=False)
    self._hbox_action_area.set_spacing(self._HBOX_HORIZONTAL_SPACING)
    self._hbox_action_area.pack_start(self._progress_bar, expand=True, fill=True)
    self._hbox_action_area.pack_end(self._buttonbox, expand=False, fill=False)
    
    self._dialog.vbox.pack_end(self._hbox_action_area, expand=False, fill=False)
    
    self._button_stop.connect('clicked', self._on_button_stop_clicked)
    self._dialog.connect('delete-event', self._on_dialog_delete_event)
  
  def run_batcher(self):
    progress_updater = pg.gui.GtkProgressUpdater(self._progress_bar)
    item_progress_indicator = progress_.ItemProgressIndicator(
      self._progress_bar, progress_updater)
    item_progress_indicator.install_progress_for_status()
    
    self._batcher = batcher_.Batcher(
      gimpenums.RUN_WITH_LAST_VALS,
      self._image,
      self._settings['main/procedures'],
      self._settings['main/constraints'],
      overwrite_chooser=pg.overwrite.NoninteractiveOverwriteChooser(
        self._settings['main/overwrite_mode'].value),
      progress_updater=progress_updater,
      export_context_manager=handle_gui_in_export,
      export_context_manager_args=[self._dialog])
    try:
      self._batcher.run(
        item_tree=self._layer_tree,
        **utils_.get_settings_for_batcher(self._settings['main']))
    except exceptions.BatcherCancelError:
      pass
    except exceptions.BatcherError as e:
      messages_.display_processing_failure_message(e, parent=self._dialog)
    except Exception as e:
      if pdb.gimp_image_is_valid(self._image):
        raise
      else:
        messages_.display_invalid_image_failure_message(parent=self._dialog)
    else:
      if not self._settings['main/edit_mode'].value and not self._batcher.exported_raw_items:
        messages_.display_message(
          _('No layers were exported.'), gtk.MESSAGE_INFO, parent=self._dialog)
    finally:
      item_progress_indicator.uninstall_progress_for_status()
  
  def show(self):
    self._dialog.vbox.show_all()
    self._dialog.action_area.hide()
    self._dialog.show()
  
  def hide(self):
    self._dialog.hide()
  
  def _on_button_stop_clicked(self, button):
    stop_batcher(self._batcher)
  
  def _on_dialog_delete_event(self, dialog, event):
    stop_batcher(self._batcher)
                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                    