2011-11-07 13:46:58 -06:00
|
|
|
# -*- coding: utf-8 -*-
|
2011-11-07 16:52:18 -06:00
|
|
|
# Pluma External Tools plugin
|
2011-11-07 13:46:58 -06:00
|
|
|
# Copyright (C) 2005-2006 Steve Frécinaux <steve@istique.net>
|
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
|
|
|
|
__all__ = ('Manager', )
|
|
|
|
|
2011-11-07 16:52:18 -06:00
|
|
|
import pluma
|
2011-11-07 13:46:58 -06:00
|
|
|
import gtk
|
|
|
|
import gtksourceview2 as gsv
|
|
|
|
import os.path
|
|
|
|
from library import *
|
|
|
|
from functions import *
|
|
|
|
import hashlib
|
|
|
|
from xml.sax import saxutils
|
|
|
|
import gobject
|
|
|
|
|
|
|
|
class LanguagesPopup(gtk.Window):
|
|
|
|
COLUMN_NAME = 0
|
|
|
|
COLUMN_ID = 1
|
|
|
|
COLUMN_ENABLED = 2
|
|
|
|
|
|
|
|
def __init__(self, languages):
|
|
|
|
gtk.Window.__init__(self, gtk.WINDOW_POPUP)
|
|
|
|
|
|
|
|
self.set_default_size(200, 200)
|
|
|
|
self.props.can_focus = True
|
|
|
|
|
|
|
|
self.build()
|
|
|
|
self.init_languages(languages)
|
|
|
|
|
|
|
|
self.show()
|
|
|
|
self.map()
|
|
|
|
|
|
|
|
self.grab_add()
|
|
|
|
|
|
|
|
gtk.gdk.keyboard_grab(self.window, False, 0L)
|
|
|
|
gtk.gdk.pointer_grab(self.window, False, gtk.gdk.BUTTON_PRESS_MASK |
|
|
|
|
gtk.gdk.BUTTON_RELEASE_MASK |
|
|
|
|
gtk.gdk.POINTER_MOTION_MASK |
|
|
|
|
gtk.gdk.ENTER_NOTIFY_MASK |
|
|
|
|
gtk.gdk.LEAVE_NOTIFY_MASK |
|
|
|
|
gtk.gdk.PROXIMITY_IN_MASK |
|
|
|
|
gtk.gdk.PROXIMITY_OUT_MASK, None, None, 0L)
|
|
|
|
|
|
|
|
self.view.get_selection().select_path((0,))
|
|
|
|
|
|
|
|
def build(self):
|
|
|
|
self.model = gtk.ListStore(str, str, bool)
|
|
|
|
|
|
|
|
self.sw = gtk.ScrolledWindow()
|
|
|
|
self.sw.show()
|
|
|
|
|
|
|
|
self.sw.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
|
|
|
|
self.sw.set_shadow_type(gtk.SHADOW_ETCHED_IN)
|
|
|
|
|
|
|
|
self.view = gtk.TreeView(self.model)
|
|
|
|
self.view.show()
|
|
|
|
|
|
|
|
self.view.set_headers_visible(False)
|
|
|
|
|
|
|
|
column = gtk.TreeViewColumn()
|
|
|
|
|
|
|
|
renderer = gtk.CellRendererToggle()
|
|
|
|
column.pack_start(renderer, False)
|
|
|
|
column.set_attributes(renderer, active=self.COLUMN_ENABLED)
|
|
|
|
|
|
|
|
renderer.connect('toggled', self.on_language_toggled)
|
|
|
|
|
|
|
|
renderer = gtk.CellRendererText()
|
|
|
|
column.pack_start(renderer, True)
|
|
|
|
column.set_attributes(renderer, text=self.COLUMN_NAME)
|
|
|
|
|
|
|
|
self.view.append_column(column)
|
|
|
|
self.view.set_row_separator_func(self.on_separator)
|
|
|
|
|
|
|
|
self.sw.add(self.view)
|
|
|
|
|
|
|
|
self.add(self.sw)
|
|
|
|
|
|
|
|
def enabled_languages(self, model, path, piter, ret):
|
|
|
|
enabled = model.get_value(piter, self.COLUMN_ENABLED)
|
|
|
|
|
|
|
|
if path == (0,) and enabled:
|
|
|
|
return True
|
|
|
|
|
|
|
|
if enabled:
|
|
|
|
ret.append(model.get_value(piter, self.COLUMN_ID))
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
def languages(self):
|
|
|
|
ret = []
|
|
|
|
|
|
|
|
self.model.foreach(self.enabled_languages, ret)
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def on_separator(self, model, piter):
|
|
|
|
val = model.get_value(piter, self.COLUMN_NAME)
|
|
|
|
return val == '-'
|
|
|
|
|
|
|
|
def init_languages(self, languages):
|
|
|
|
manager = gsv.LanguageManager()
|
2011-11-07 16:52:18 -06:00
|
|
|
langs = pluma.language_manager_list_languages_sorted(manager, True)
|
2011-11-07 13:46:58 -06:00
|
|
|
|
|
|
|
self.model.append([_('All languages'), None, not languages])
|
|
|
|
self.model.append(['-', None, False])
|
|
|
|
self.model.append([_('Plain Text'), 'plain', 'plain' in languages])
|
|
|
|
self.model.append(['-', None, False])
|
|
|
|
|
|
|
|
for lang in langs:
|
|
|
|
self.model.append([lang.get_name(), lang.get_id(), lang.get_id() in languages])
|
|
|
|
|
|
|
|
def correct_all(self, model, path, piter, enabled):
|
|
|
|
if path == (0,):
|
|
|
|
return False
|
|
|
|
|
|
|
|
model.set_value(piter, self.COLUMN_ENABLED, enabled)
|
|
|
|
|
|
|
|
def on_language_toggled(self, renderer, path):
|
|
|
|
piter = self.model.get_iter(path)
|
|
|
|
|
|
|
|
enabled = self.model.get_value(piter, self.COLUMN_ENABLED)
|
|
|
|
self.model.set_value(piter, self.COLUMN_ENABLED, not enabled)
|
|
|
|
|
|
|
|
if path == '0':
|
|
|
|
self.model.foreach(self.correct_all, False)
|
|
|
|
else:
|
|
|
|
self.model.set_value(self.model.get_iter_first(), self.COLUMN_ENABLED, False)
|
|
|
|
|
|
|
|
def do_key_press_event(self, event):
|
|
|
|
if event.keyval == gtk.keysyms.Escape:
|
|
|
|
self.destroy()
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
event.window = self.view.get_bin_window()
|
|
|
|
return self.view.event(event)
|
|
|
|
|
|
|
|
def do_key_release_event(self, event):
|
|
|
|
event.window = self.view.get_bin_window()
|
|
|
|
return self.view.event(event)
|
|
|
|
|
|
|
|
def in_window(self, event, window=None):
|
|
|
|
if not window:
|
|
|
|
window = self.window
|
|
|
|
|
|
|
|
geometry = window.get_geometry()
|
|
|
|
origin = window.get_origin()
|
|
|
|
|
|
|
|
return event.x_root >= origin[0] and \
|
|
|
|
event.x_root <= origin[0] + geometry[2] and \
|
|
|
|
event.y_root >= origin[1] and \
|
|
|
|
event.y_root <= origin[1] + geometry[3]
|
|
|
|
|
|
|
|
def do_destroy(self):
|
|
|
|
gtk.gdk.keyboard_ungrab(0L)
|
|
|
|
gtk.gdk.pointer_ungrab(0L)
|
|
|
|
|
|
|
|
return gtk.Window.do_destroy(self)
|
|
|
|
|
|
|
|
def setup_event(self, event, window):
|
|
|
|
fr = event.window.get_origin()
|
|
|
|
to = window.get_origin()
|
|
|
|
|
|
|
|
event.window = window
|
|
|
|
event.x += fr[0] - to[0]
|
|
|
|
event.y += fr[1] - to[1]
|
|
|
|
|
|
|
|
def resolve_widgets(self, root):
|
|
|
|
res = [root]
|
|
|
|
|
|
|
|
if isinstance(root, gtk.Container):
|
|
|
|
root.forall(lambda x, y: res.extend(self.resolve_widgets(x)), None)
|
|
|
|
|
|
|
|
return res
|
|
|
|
|
|
|
|
def resolve_windows(self, window):
|
|
|
|
if not window:
|
|
|
|
return []
|
|
|
|
|
|
|
|
res = [window]
|
|
|
|
res.extend(window.get_children())
|
|
|
|
|
|
|
|
return res
|
|
|
|
|
|
|
|
def propagate_mouse_event(self, event):
|
|
|
|
allwidgets = self.resolve_widgets(self.get_child())
|
|
|
|
allwidgets.reverse()
|
|
|
|
|
|
|
|
orig = [event.x, event.y]
|
|
|
|
|
|
|
|
for widget in allwidgets:
|
|
|
|
windows = self.resolve_windows(widget.window)
|
|
|
|
windows.reverse()
|
|
|
|
|
|
|
|
for window in windows:
|
|
|
|
if not (window.get_events() & event.type):
|
|
|
|
continue
|
|
|
|
|
|
|
|
if self.in_window(event, window):
|
|
|
|
self.setup_event(event, window)
|
|
|
|
|
|
|
|
if widget.event(event):
|
|
|
|
return True
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
def do_button_press_event(self, event):
|
|
|
|
if not self.in_window(event):
|
|
|
|
self.destroy()
|
|
|
|
else:
|
|
|
|
return self.propagate_mouse_event(event)
|
|
|
|
|
|
|
|
def do_button_release_event(self, event):
|
|
|
|
if not self.in_window(event):
|
|
|
|
self.destroy()
|
|
|
|
else:
|
|
|
|
return self.propagate_mouse_event(event)
|
|
|
|
|
|
|
|
def do_scroll_event(self, event):
|
|
|
|
return self.propagate_mouse_event(event)
|
|
|
|
|
|
|
|
def do_motion_notify_event(self, event):
|
|
|
|
return self.propagate_mouse_event(event)
|
|
|
|
|
|
|
|
def do_enter_notify_event(self, event):
|
|
|
|
return self.propagate_mouse_event(event)
|
|
|
|
|
|
|
|
def do_leave_notify_event(self, event):
|
|
|
|
return self.propagate_mouse_event(event)
|
|
|
|
|
|
|
|
def do_proximity_in_event(self, event):
|
|
|
|
return self.propagate_mouse_event(event)
|
|
|
|
|
|
|
|
def do_proximity_out_event(self, event):
|
|
|
|
return self.propagate_mouse_event(event)
|
|
|
|
|
|
|
|
gobject.type_register(LanguagesPopup)
|
|
|
|
|
|
|
|
class Manager:
|
|
|
|
TOOL_COLUMN = 0 # For Tree
|
|
|
|
NAME_COLUMN = 1 # For Combo
|
|
|
|
|
|
|
|
def __init__(self, datadir):
|
|
|
|
self.datadir = datadir
|
|
|
|
self.dialog = None
|
|
|
|
self._languages = {}
|
|
|
|
self._tool_rows = {}
|
|
|
|
|
|
|
|
self.build()
|
|
|
|
|
|
|
|
def build(self):
|
|
|
|
callbacks = {
|
|
|
|
'on_new_tool_button_clicked' : self.on_new_tool_button_clicked,
|
|
|
|
'on_remove_tool_button_clicked' : self.on_remove_tool_button_clicked,
|
|
|
|
'on_tool_manager_dialog_response' : self.on_tool_manager_dialog_response,
|
|
|
|
'on_tool_manager_dialog_focus_out': self.on_tool_manager_dialog_focus_out,
|
|
|
|
'on_accelerator_key_press' : self.on_accelerator_key_press,
|
|
|
|
'on_accelerator_focus_in' : self.on_accelerator_focus_in,
|
|
|
|
'on_accelerator_focus_out' : self.on_accelerator_focus_out,
|
|
|
|
'on_languages_button_clicked' : self.on_languages_button_clicked
|
|
|
|
}
|
|
|
|
|
|
|
|
# Load the "main-window" widget from the ui file.
|
|
|
|
self.ui = gtk.Builder()
|
|
|
|
self.ui.add_from_file(os.path.join(self.datadir, 'ui', 'tools.ui'))
|
|
|
|
self.ui.connect_signals(callbacks)
|
|
|
|
self.dialog = self.ui.get_object('tool-manager-dialog')
|
|
|
|
|
|
|
|
self.view = self.ui.get_object('view')
|
|
|
|
|
|
|
|
self.__init_tools_model()
|
|
|
|
self.__init_tools_view()
|
|
|
|
|
|
|
|
for name in ['input', 'output', 'applicability', 'save-files']:
|
|
|
|
self.__init_combobox(name)
|
|
|
|
|
|
|
|
self.do_update()
|
|
|
|
|
|
|
|
def expand_from_doc(self, doc):
|
|
|
|
row = None
|
|
|
|
|
|
|
|
if doc:
|
|
|
|
if doc.get_language():
|
|
|
|
lid = doc.get_language().get_id()
|
|
|
|
|
|
|
|
if lid in self._languages:
|
|
|
|
row = self._languages[lid]
|
|
|
|
elif 'plain' in self._languages:
|
|
|
|
row = self._languages['plain']
|
|
|
|
|
|
|
|
if not row and None in self._languages:
|
|
|
|
row = self._languages[None]
|
|
|
|
|
|
|
|
if not row:
|
|
|
|
return
|
|
|
|
|
|
|
|
self.view.expand_row(row.get_path(), False)
|
|
|
|
self.view.get_selection().select_path(row.get_path())
|
|
|
|
|
|
|
|
def run(self, window):
|
|
|
|
if self.dialog == None:
|
|
|
|
self.build()
|
|
|
|
|
|
|
|
# Open up language
|
|
|
|
self.expand_from_doc(window.get_active_document())
|
|
|
|
|
|
|
|
self.dialog.set_transient_for(window)
|
|
|
|
window.get_group().add_window(self.dialog)
|
|
|
|
self.dialog.present()
|
|
|
|
|
|
|
|
def add_accelerator(self, item):
|
|
|
|
if not item.shortcut:
|
|
|
|
return
|
|
|
|
|
|
|
|
if item.shortcut in self.accelerators:
|
|
|
|
if not item in self.accelerators[item.shortcut]:
|
|
|
|
self.accelerators[item.shortcut].append(item)
|
|
|
|
else:
|
|
|
|
self.accelerators[item.shortcut] = [item]
|
|
|
|
|
|
|
|
def remove_accelerator(self, item, shortcut=None):
|
|
|
|
if not shortcut:
|
|
|
|
shortcut = item.shortcut
|
|
|
|
|
|
|
|
if not shortcut in self.accelerators:
|
|
|
|
return
|
|
|
|
|
|
|
|
self.accelerators[shortcut].remove(item)
|
|
|
|
|
|
|
|
if not self.accelerators[shortcut]:
|
|
|
|
del self.accelerators[shortcut]
|
|
|
|
|
|
|
|
def add_tool_to_language(self, tool, language):
|
|
|
|
if isinstance(language, gsv.Language):
|
|
|
|
lid = language.get_id()
|
|
|
|
else:
|
|
|
|
lid = language
|
|
|
|
|
|
|
|
if not lid in self._languages:
|
|
|
|
piter = self.model.append(None, [language])
|
|
|
|
|
|
|
|
parent = gtk.TreeRowReference(self.model, self.model.get_path(piter))
|
|
|
|
self._languages[lid] = parent
|
|
|
|
else:
|
|
|
|
parent = self._languages[lid]
|
|
|
|
|
|
|
|
piter = self.model.get_iter(parent.get_path())
|
|
|
|
child = self.model.append(piter, [tool])
|
|
|
|
|
|
|
|
if not tool in self._tool_rows:
|
|
|
|
self._tool_rows[tool] = []
|
|
|
|
|
|
|
|
self._tool_rows[tool].append(gtk.TreeRowReference(self.model, self.model.get_path(child)))
|
|
|
|
return child
|
|
|
|
|
|
|
|
def add_tool(self, tool):
|
|
|
|
manager = gsv.LanguageManager()
|
|
|
|
ret = None
|
|
|
|
|
|
|
|
for lang in tool.languages:
|
|
|
|
l = manager.get_language(lang)
|
|
|
|
|
|
|
|
if l:
|
|
|
|
ret = self.add_tool_to_language(tool, l)
|
|
|
|
elif lang == 'plain':
|
|
|
|
ret = self.add_tool_to_language(tool, 'plain')
|
|
|
|
|
|
|
|
if not ret:
|
|
|
|
ret = self.add_tool_to_language(tool, None)
|
|
|
|
|
|
|
|
self.add_accelerator(tool)
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def __init_tools_model(self):
|
|
|
|
self.tools = ToolLibrary()
|
|
|
|
self.current_node = None
|
|
|
|
self.script_hash = None
|
|
|
|
self.accelerators = dict()
|
|
|
|
|
|
|
|
self.model = gtk.TreeStore(object)
|
|
|
|
self.view.set_model(self.model)
|
|
|
|
|
|
|
|
for tool in self.tools.tree.tools:
|
|
|
|
self.add_tool(tool)
|
|
|
|
|
|
|
|
self.model.set_default_sort_func(self.sort_tools)
|
|
|
|
self.model.set_sort_column_id(-1, gtk.SORT_ASCENDING)
|
|
|
|
|
|
|
|
def sort_tools(self, model, iter1, iter2):
|
|
|
|
# For languages, sort All before everything else, otherwise alphabetical
|
|
|
|
t1 = model.get_value(iter1, self.TOOL_COLUMN)
|
|
|
|
t2 = model.get_value(iter2, self.TOOL_COLUMN)
|
|
|
|
|
|
|
|
if model.iter_parent(iter1) == None:
|
|
|
|
if t1 == None:
|
|
|
|
return -1
|
|
|
|
|
|
|
|
if t2 == None:
|
|
|
|
return 1
|
|
|
|
|
|
|
|
def lang_name(lang):
|
|
|
|
if isinstance(lang, gsv.Language):
|
|
|
|
return lang.get_name()
|
|
|
|
else:
|
|
|
|
return _('Plain Text')
|
|
|
|
|
|
|
|
n1 = lang_name(t1)
|
|
|
|
n2 = lang_name(t2)
|
|
|
|
else:
|
|
|
|
n1 = t1.name
|
|
|
|
n2 = t2.name
|
|
|
|
|
|
|
|
return cmp(n1.lower(), n2.lower())
|
|
|
|
|
|
|
|
def __init_tools_view(self):
|
|
|
|
# Tools column
|
|
|
|
column = gtk.TreeViewColumn('Tools')
|
|
|
|
renderer = gtk.CellRendererText()
|
|
|
|
column.pack_start(renderer, False)
|
|
|
|
renderer.set_property('editable', True)
|
|
|
|
self.view.append_column(column)
|
|
|
|
|
|
|
|
column.set_cell_data_func(renderer, self.get_cell_data_cb)
|
|
|
|
|
|
|
|
renderer.connect('edited', self.on_view_label_cell_edited)
|
|
|
|
renderer.connect('editing-started', self.on_view_label_cell_editing_started)
|
|
|
|
|
|
|
|
self.selection_changed_id = self.view.get_selection().connect('changed', self.on_view_selection_changed, None)
|
|
|
|
|
|
|
|
def __init_combobox(self, name):
|
|
|
|
combo = self[name]
|
|
|
|
combo.set_active(0)
|
|
|
|
|
|
|
|
# Convenience function to get an object from its name
|
|
|
|
def __getitem__(self, key):
|
|
|
|
return self.ui.get_object(key)
|
|
|
|
|
|
|
|
def set_active_by_name(self, combo_name, option_name):
|
|
|
|
combo = self[combo_name]
|
|
|
|
model = combo.get_model()
|
|
|
|
piter = model.get_iter_first()
|
|
|
|
while piter is not None:
|
|
|
|
if model.get_value(piter, self.NAME_COLUMN) == option_name:
|
|
|
|
combo.set_active_iter(piter)
|
|
|
|
return True
|
|
|
|
piter = model.iter_next(piter)
|
|
|
|
return False
|
|
|
|
|
|
|
|
def get_selected_tool(self):
|
|
|
|
model, piter = self.view.get_selection().get_selected()
|
|
|
|
|
|
|
|
if piter is not None:
|
|
|
|
tool = model.get_value(piter, self.TOOL_COLUMN)
|
|
|
|
|
|
|
|
if not isinstance(tool, Tool):
|
|
|
|
tool = None
|
|
|
|
|
|
|
|
return piter, tool
|
|
|
|
else:
|
|
|
|
return None, None
|
|
|
|
|
|
|
|
def compute_hash(self, string):
|
|
|
|
return hashlib.md5(string).hexdigest()
|
|
|
|
|
|
|
|
def save_current_tool(self):
|
|
|
|
if self.current_node is None:
|
|
|
|
return
|
|
|
|
|
|
|
|
if self.current_node.filename is None:
|
|
|
|
self.current_node.autoset_filename()
|
|
|
|
|
|
|
|
def combo_value(o, name):
|
|
|
|
combo = o[name]
|
|
|
|
return combo.get_model().get_value(combo.get_active_iter(), self.NAME_COLUMN)
|
|
|
|
|
|
|
|
self.current_node.input = combo_value(self, 'input')
|
|
|
|
self.current_node.output = combo_value(self, 'output')
|
|
|
|
self.current_node.applicability = combo_value(self, 'applicability')
|
|
|
|
self.current_node.save_files = combo_value(self, 'save-files')
|
|
|
|
|
|
|
|
buf = self['commands'].get_buffer()
|
|
|
|
script = buf.get_text(*buf.get_bounds())
|
|
|
|
h = self.compute_hash(script)
|
|
|
|
if h != self.script_hash:
|
|
|
|
# script has changed -> save it
|
|
|
|
self.current_node.save_with_script([line + "\n" for line in script.splitlines()])
|
|
|
|
self.script_hash = h
|
|
|
|
else:
|
|
|
|
self.current_node.save()
|
|
|
|
|
|
|
|
self.update_remove_revert()
|
|
|
|
|
|
|
|
def clear_fields(self):
|
|
|
|
self['accelerator'].set_text('')
|
|
|
|
|
|
|
|
buf = self['commands'].get_buffer()
|
|
|
|
buf.begin_not_undoable_action()
|
|
|
|
buf.set_text('')
|
|
|
|
buf.end_not_undoable_action()
|
|
|
|
|
|
|
|
for nm in ('input', 'output', 'applicability', 'save-files'):
|
|
|
|
self[nm].set_active(0)
|
|
|
|
|
|
|
|
self['languages_label'].set_text(_('All Languages'))
|
|
|
|
|
|
|
|
def fill_languages_button(self):
|
|
|
|
if not self.current_node or not self.current_node.languages:
|
|
|
|
self['languages_label'].set_text(_('All Languages'))
|
|
|
|
else:
|
|
|
|
manager = gsv.LanguageManager()
|
|
|
|
langs = []
|
|
|
|
|
|
|
|
for lang in self.current_node.languages:
|
|
|
|
if lang == 'plain':
|
|
|
|
langs.append(_('Plain Text'))
|
|
|
|
else:
|
|
|
|
l = manager.get_language(lang)
|
|
|
|
|
|
|
|
if l:
|
|
|
|
langs.append(l.get_name())
|
|
|
|
|
|
|
|
self['languages_label'].set_text(', '.join(langs))
|
|
|
|
|
|
|
|
def fill_fields(self):
|
|
|
|
node = self.current_node
|
|
|
|
self['accelerator'].set_text(default(node.shortcut, ''))
|
|
|
|
|
|
|
|
buf = self['commands'].get_buffer()
|
|
|
|
script = default(''.join(node.get_script()), '')
|
|
|
|
|
|
|
|
buf.begin_not_undoable_action()
|
|
|
|
buf.set_text(script)
|
|
|
|
buf.end_not_undoable_action()
|
|
|
|
|
|
|
|
self.script_hash = self.compute_hash(script)
|
|
|
|
contenttype = gio.content_type_guess(data=script)
|
2011-11-07 16:52:18 -06:00
|
|
|
lmanager = pluma.get_language_manager()
|
2011-11-07 13:46:58 -06:00
|
|
|
language = lmanager.guess_language(content_type=contenttype)
|
|
|
|
|
|
|
|
if language is not None:
|
|
|
|
buf.set_language(language)
|
|
|
|
buf.set_highlight_syntax(True)
|
|
|
|
else:
|
|
|
|
buf.set_highlight_syntax(False)
|
|
|
|
|
|
|
|
for nm in ('input', 'output', 'applicability', 'save-files'):
|
|
|
|
model = self[nm].get_model()
|
|
|
|
piter = model.get_iter_first()
|
|
|
|
|
|
|
|
self.set_active_by_name(nm,
|
|
|
|
default(node.__getattribute__(nm.replace('-', '_')),
|
|
|
|
model.get_value(piter, self.NAME_COLUMN)))
|
|
|
|
|
|
|
|
self.fill_languages_button()
|
|
|
|
|
|
|
|
def update_remove_revert(self):
|
|
|
|
piter, node = self.get_selected_tool()
|
|
|
|
|
|
|
|
removable = node is not None and node.is_local()
|
|
|
|
|
|
|
|
self['remove-tool-button'].set_sensitive(removable)
|
|
|
|
self['revert-tool-button'].set_sensitive(removable)
|
|
|
|
|
|
|
|
if node is not None and node.is_global():
|
|
|
|
self['remove-tool-button'].hide()
|
|
|
|
self['revert-tool-button'].show()
|
|
|
|
else:
|
|
|
|
self['remove-tool-button'].show()
|
|
|
|
self['revert-tool-button'].hide()
|
|
|
|
|
|
|
|
def do_update(self):
|
|
|
|
self.update_remove_revert()
|
|
|
|
|
|
|
|
piter, node = self.get_selected_tool()
|
|
|
|
self.current_node = node
|
|
|
|
|
|
|
|
if node is not None:
|
|
|
|
self.fill_fields()
|
|
|
|
self['tool-table'].set_sensitive(True)
|
|
|
|
else:
|
|
|
|
self.clear_fields()
|
|
|
|
self['tool-table'].set_sensitive(False)
|
|
|
|
|
|
|
|
def language_id_from_iter(self, piter):
|
|
|
|
if not piter:
|
|
|
|
return None
|
|
|
|
|
|
|
|
tool = self.model.get_value(piter, self.TOOL_COLUMN)
|
|
|
|
|
|
|
|
if isinstance(tool, Tool):
|
|
|
|
piter = self.model.iter_parent(piter)
|
|
|
|
tool = self.model.get_value(piter, self.TOOL_COLUMN)
|
|
|
|
|
|
|
|
if isinstance(tool, gsv.Language):
|
|
|
|
return tool.get_id()
|
|
|
|
elif tool:
|
|
|
|
return 'plain'
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
def selected_language_id(self):
|
|
|
|
# Find current language if there is any
|
|
|
|
model, piter = self.view.get_selection().get_selected()
|
|
|
|
|
|
|
|
return self.language_id_from_iter(piter)
|
|
|
|
|
|
|
|
def on_new_tool_button_clicked(self, button):
|
|
|
|
self.save_current_tool()
|
|
|
|
|
|
|
|
# block handlers while inserting a new item
|
|
|
|
self.view.get_selection().handler_block(self.selection_changed_id)
|
|
|
|
|
|
|
|
self.current_node = Tool(self.tools.tree);
|
|
|
|
self.current_node.name = _('New tool')
|
|
|
|
self.tools.tree.tools.append(self.current_node)
|
|
|
|
|
|
|
|
lang = self.selected_language_id()
|
|
|
|
|
|
|
|
if lang:
|
|
|
|
self.current_node.languages = [lang]
|
|
|
|
|
|
|
|
piter = self.add_tool(self.current_node)
|
|
|
|
|
|
|
|
self.view.set_cursor(self.model.get_path(piter), self.view.get_column(self.TOOL_COLUMN), True)
|
|
|
|
self.fill_fields()
|
|
|
|
|
|
|
|
self['tool-table'].set_sensitive(True)
|
|
|
|
self.view.get_selection().handler_unblock(self.selection_changed_id)
|
|
|
|
|
|
|
|
def tool_changed(self, tool, refresh=False):
|
|
|
|
for row in self._tool_rows[tool]:
|
|
|
|
self.model.row_changed(row.get_path(), self.model.get_iter(row.get_path()))
|
|
|
|
|
|
|
|
if refresh and tool == self.current_node:
|
|
|
|
self.fill_fields()
|
|
|
|
|
|
|
|
self.update_remove_revert()
|
|
|
|
|
|
|
|
def on_remove_tool_button_clicked(self, button):
|
|
|
|
piter, node = self.get_selected_tool()
|
|
|
|
|
|
|
|
if not node:
|
|
|
|
return
|
|
|
|
|
|
|
|
if node.is_global():
|
|
|
|
shortcut = node.shortcut
|
|
|
|
|
|
|
|
if node.parent.revert_tool(node):
|
|
|
|
self.remove_accelerator(node, shortcut)
|
|
|
|
self.add_accelerator(node)
|
|
|
|
|
|
|
|
self['revert-tool-button'].set_sensitive(False)
|
|
|
|
self.fill_fields()
|
|
|
|
|
|
|
|
self.tool_changed(node)
|
|
|
|
else:
|
|
|
|
parent = self.model.iter_parent(piter)
|
|
|
|
language = self.language_id_from_iter(parent)
|
|
|
|
|
|
|
|
self.model.remove(piter)
|
|
|
|
|
|
|
|
if language in node.languages:
|
|
|
|
node.languages.remove(language)
|
|
|
|
|
|
|
|
self._tool_rows[node] = filter(lambda x: x.valid(), self._tool_rows[node])
|
|
|
|
|
|
|
|
if not self._tool_rows[node]:
|
|
|
|
del self._tool_rows[node]
|
|
|
|
|
|
|
|
if node.parent.delete_tool(node):
|
|
|
|
self.remove_accelerator(node)
|
|
|
|
self.current_node = None
|
|
|
|
self.script_hash = None
|
|
|
|
|
|
|
|
if self.model.iter_is_valid(piter):
|
|
|
|
self.view.set_cursor(self.model.get_path(piter), self.view.get_column(self.TOOL_COLUMN), False)
|
|
|
|
|
|
|
|
self.view.grab_focus()
|
|
|
|
|
|
|
|
path = self._languages[language].get_path()
|
|
|
|
parent = self.model.get_iter(path)
|
|
|
|
|
|
|
|
if not self.model.iter_has_child(parent):
|
|
|
|
self.model.remove(parent)
|
|
|
|
del self._languages[language]
|
|
|
|
|
|
|
|
def on_view_label_cell_edited(self, cell, path, new_text):
|
|
|
|
if new_text != '':
|
|
|
|
piter = self.model.get_iter(path)
|
|
|
|
tool = self.model.get_value(piter, self.TOOL_COLUMN)
|
|
|
|
|
|
|
|
tool.name = new_text
|
|
|
|
|
|
|
|
self.save_current_tool()
|
|
|
|
self.tool_changed(tool)
|
|
|
|
|
|
|
|
def on_view_label_cell_editing_started(self, renderer, editable, path):
|
|
|
|
piter = self.model.get_iter(path)
|
|
|
|
tool = self.model.get_value(piter, self.TOOL_COLUMN)
|
|
|
|
|
|
|
|
if isinstance(editable, gtk.Entry):
|
|
|
|
editable.set_text(tool.name)
|
|
|
|
editable.grab_focus()
|
|
|
|
|
|
|
|
def on_view_selection_changed(self, selection, userdata):
|
|
|
|
self.save_current_tool()
|
|
|
|
self.do_update()
|
|
|
|
|
|
|
|
def accelerator_collision(self, name, node):
|
|
|
|
if not name in self.accelerators:
|
|
|
|
return []
|
|
|
|
|
|
|
|
ret = []
|
|
|
|
|
|
|
|
for other in self.accelerators[name]:
|
|
|
|
if not other.languages or not node.languages:
|
|
|
|
ret.append(other)
|
|
|
|
continue
|
|
|
|
|
|
|
|
for lang in other.languages:
|
|
|
|
if lang in node.languages:
|
|
|
|
ret.append(other)
|
|
|
|
continue
|
|
|
|
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def set_accelerator(self, keyval, mod):
|
|
|
|
# Check whether accelerator already exists
|
|
|
|
self.remove_accelerator(self.current_node)
|
|
|
|
|
|
|
|
name = gtk.accelerator_name(keyval, mod)
|
|
|
|
|
|
|
|
if name == '':
|
|
|
|
self.current_node.shorcut = None
|
|
|
|
self.save_current_tool()
|
|
|
|
return True
|
|
|
|
|
|
|
|
col = self.accelerator_collision(name, self.current_node)
|
|
|
|
|
|
|
|
if col:
|
|
|
|
dialog = gtk.MessageDialog(self.dialog,
|
|
|
|
gtk.DIALOG_MODAL,
|
|
|
|
gtk.MESSAGE_ERROR,
|
|
|
|
gtk.BUTTONS_OK,
|
|
|
|
_('This accelerator is already bound to %s') % (', '.join(map(lambda x: x.name, col)),))
|
|
|
|
|
|
|
|
dialog.run()
|
|
|
|
dialog.destroy()
|
|
|
|
|
|
|
|
self.add_accelerator(self.current_node)
|
|
|
|
return False
|
|
|
|
|
|
|
|
self.current_node.shortcut = name
|
|
|
|
self.add_accelerator(self.current_node)
|
|
|
|
self.save_current_tool()
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
def on_accelerator_key_press(self, entry, event):
|
|
|
|
mask = event.state & gtk.accelerator_get_default_mod_mask()
|
|
|
|
|
|
|
|
if event.keyval == gtk.keysyms.Escape:
|
|
|
|
entry.set_text(default(self.current_node.shortcut, ''))
|
|
|
|
self['commands'].grab_focus()
|
|
|
|
return True
|
|
|
|
elif event.keyval == gtk.keysyms.Delete \
|
|
|
|
or event.keyval == gtk.keysyms.BackSpace:
|
|
|
|
entry.set_text('')
|
|
|
|
self.remove_accelerator(self.current_node)
|
|
|
|
self.current_node.shortcut = None
|
|
|
|
self['commands'].grab_focus()
|
|
|
|
return True
|
|
|
|
elif event.keyval in range(gtk.keysyms.F1, gtk.keysyms.F12 + 1):
|
|
|
|
# New accelerator
|
|
|
|
if self.set_accelerator(event.keyval, mask):
|
|
|
|
entry.set_text(default(self.current_node.shortcut, ''))
|
|
|
|
self['commands'].grab_focus()
|
|
|
|
|
|
|
|
# Capture all `normal characters`
|
|
|
|
return True
|
|
|
|
elif gtk.gdk.keyval_to_unicode(event.keyval):
|
|
|
|
if mask:
|
|
|
|
# New accelerator
|
|
|
|
if self.set_accelerator(event.keyval, mask):
|
|
|
|
entry.set_text(default(self.current_node.shortcut, ''))
|
|
|
|
self['commands'].grab_focus()
|
|
|
|
# Capture all `normal characters`
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
def on_accelerator_focus_in(self, entry, event):
|
|
|
|
if self.current_node is None:
|
|
|
|
return
|
|
|
|
if self.current_node.shortcut:
|
|
|
|
entry.set_text(_('Type a new accelerator, or press Backspace to clear'))
|
|
|
|
else:
|
|
|
|
entry.set_text(_('Type a new accelerator'))
|
|
|
|
|
|
|
|
def on_accelerator_focus_out(self, entry, event):
|
|
|
|
if self.current_node is not None:
|
|
|
|
entry.set_text(default(self.current_node.shortcut, ''))
|
|
|
|
self.tool_changed(self.current_node)
|
|
|
|
|
|
|
|
def on_tool_manager_dialog_response(self, dialog, response):
|
|
|
|
if response == gtk.RESPONSE_HELP:
|
2011-11-07 16:52:18 -06:00
|
|
|
pluma.help_display(self.dialog, 'pluma', 'pluma-external-tools-plugin')
|
2011-11-07 13:46:58 -06:00
|
|
|
return
|
|
|
|
|
|
|
|
self.on_tool_manager_dialog_focus_out(dialog, None)
|
|
|
|
|
|
|
|
self.dialog.destroy()
|
|
|
|
self.dialog = None
|
|
|
|
self.tools = None
|
|
|
|
|
|
|
|
def on_tool_manager_dialog_focus_out(self, dialog, event):
|
|
|
|
self.save_current_tool()
|
|
|
|
|
2011-11-07 16:52:18 -06:00
|
|
|
for window in pluma.app_get_default().get_windows():
|
2011-11-07 13:46:58 -06:00
|
|
|
helper = window.get_data("ExternalToolsPluginWindowData")
|
|
|
|
helper.menu.update()
|
|
|
|
|
|
|
|
def get_cell_data_cb(self, column, cell, model, piter):
|
|
|
|
tool = model.get_value(piter, self.TOOL_COLUMN)
|
|
|
|
|
|
|
|
if tool == None or not isinstance(tool, Tool):
|
|
|
|
if tool == None:
|
|
|
|
label = _('All Languages')
|
|
|
|
elif not isinstance(tool, gsv.Language):
|
|
|
|
label = _('Plain Text')
|
|
|
|
else:
|
|
|
|
label = tool.get_name()
|
|
|
|
|
|
|
|
markup = saxutils.escape(label)
|
|
|
|
editable = False
|
|
|
|
else:
|
|
|
|
escaped = saxutils.escape(tool.name)
|
|
|
|
|
|
|
|
if tool.shortcut:
|
|
|
|
markup = '%s (<b>%s</b>)' % (escaped, saxutils.escape(tool.shortcut))
|
|
|
|
else:
|
|
|
|
markup = escaped
|
|
|
|
|
|
|
|
editable = True
|
|
|
|
|
|
|
|
cell.set_properties(markup=markup, editable=editable)
|
|
|
|
|
|
|
|
def tool_in_language(self, tool, lang):
|
|
|
|
if not lang in self._languages:
|
|
|
|
return False
|
|
|
|
|
|
|
|
ref = self._languages[lang]
|
|
|
|
parent = ref.get_path()
|
|
|
|
|
|
|
|
for row in self._tool_rows[tool]:
|
|
|
|
path = row.get_path()
|
|
|
|
|
|
|
|
if path[0] == parent[0]:
|
|
|
|
return True
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
def update_languages(self, popup):
|
|
|
|
self.current_node.languages = popup.languages()
|
|
|
|
self.fill_languages_button()
|
|
|
|
|
|
|
|
piter, node = self.get_selected_tool()
|
|
|
|
ret = None
|
|
|
|
|
|
|
|
if node:
|
|
|
|
ref = gtk.TreeRowReference(self.model, self.model.get_path(piter))
|
|
|
|
|
|
|
|
# Update languages, make sure to inhibit selection change stuff
|
|
|
|
self.view.get_selection().handler_block(self.selection_changed_id)
|
|
|
|
|
|
|
|
# Remove all rows that are no longer
|
|
|
|
for row in list(self._tool_rows[self.current_node]):
|
|
|
|
piter = self.model.get_iter(row.get_path())
|
|
|
|
language = self.language_id_from_iter(piter)
|
|
|
|
|
|
|
|
if (not language and not self.current_node.languages) or \
|
|
|
|
(language in self.current_node.languages):
|
|
|
|
continue
|
|
|
|
|
|
|
|
# Remove from language
|
|
|
|
self.model.remove(piter)
|
|
|
|
self._tool_rows[self.current_node].remove(row)
|
|
|
|
|
|
|
|
# If language is empty, remove it
|
|
|
|
parent = self.model.get_iter(self._languages[language].get_path())
|
|
|
|
|
|
|
|
if not self.model.iter_has_child(parent):
|
|
|
|
self.model.remove(parent)
|
|
|
|
del self._languages[language]
|
|
|
|
|
|
|
|
# Now, add for any that are new
|
|
|
|
manager = gsv.LanguageManager()
|
|
|
|
|
|
|
|
for lang in self.current_node.languages:
|
|
|
|
if not self.tool_in_language(self.current_node, lang):
|
|
|
|
l = manager.get_language(lang)
|
|
|
|
|
|
|
|
if not l:
|
|
|
|
l = 'plain'
|
|
|
|
|
|
|
|
self.add_tool_to_language(self.current_node, l)
|
|
|
|
|
|
|
|
if not self.current_node.languages and not self.tool_in_language(self.current_node, None):
|
|
|
|
self.add_tool_to_language(self.current_node, None)
|
|
|
|
|
|
|
|
# Check if we can still keep the current
|
|
|
|
if not ref or not ref.valid():
|
|
|
|
# Change selection to first language
|
|
|
|
path = self._tool_rows[self.current_node][0].get_path()
|
|
|
|
piter = self.model.get_iter(path)
|
|
|
|
parent = self.model.iter_parent(piter)
|
|
|
|
|
|
|
|
# Expand parent, select child and scroll to it
|
|
|
|
self.view.expand_row(self.model.get_path(parent), False)
|
|
|
|
self.view.get_selection().select_path(path)
|
|
|
|
self.view.set_cursor(path, self.view.get_column(self.TOOL_COLUMN), False)
|
|
|
|
|
|
|
|
self.view.get_selection().handler_unblock(self.selection_changed_id)
|
|
|
|
|
|
|
|
def on_languages_button_clicked(self, button):
|
|
|
|
popup = LanguagesPopup(self.current_node.languages)
|
|
|
|
popup.set_transient_for(self.dialog)
|
|
|
|
|
|
|
|
origin = button.window.get_origin()
|
|
|
|
popup.move(origin[0], origin[1] - popup.allocation.height)
|
|
|
|
|
|
|
|
popup.connect('destroy', self.update_languages)
|
|
|
|
|
|
|
|
# ex:et:ts=4:
|