# -*- coding: utf-8 -*- # Pluma External Tools plugin # 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', ) import pluma 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() langs = pluma.language_manager_list_languages_sorted(manager, True) 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) lmanager = pluma.get_language_manager() 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: pluma.help_display(self.dialog, 'pluma', 'pluma-external-tools-plugin') 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() for window in pluma.app_get_default().get_windows(): 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: