From 0104c15730421f49e6c39d68506bad307bc5db76 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Wed, 22 Sep 2021 20:37:06 -0300 Subject: [PATCH 01/13] Snippets plugin: initial version. --- plugins/meson.build | 2 + plugins/snippets/meson.build | 35 ++++ plugins/snippets/plugin.vala | 315 +++++++++++++++++++++++++++++++ plugins/snippets/snippets.json | 11 ++ plugins/snippets/snippets.plugin | 9 + 5 files changed, 372 insertions(+) create mode 100644 plugins/snippets/meson.build create mode 100644 plugins/snippets/plugin.vala create mode 100644 plugins/snippets/snippets.json create mode 100644 plugins/snippets/snippets.plugin diff --git a/plugins/meson.build b/plugins/meson.build index 2e8855ee1d..ca19626854 100644 --- a/plugins/meson.build +++ b/plugins/meson.build @@ -9,7 +9,9 @@ subdir('outline') subdir('pastebin') subdir('preserve-indent') subdir('spell') +subdir('snippets') subdir('strip-trailing-save') subdir('terminal') subdir('vim-emulation') subdir('word-completion') + diff --git a/plugins/snippets/meson.build b/plugins/snippets/meson.build new file mode 100644 index 0000000000..487f2cc1e2 --- /dev/null +++ b/plugins/snippets/meson.build @@ -0,0 +1,35 @@ +module_name = 'snippets' + +module_files = [ + 'plugin.vala' +] + +json_dep = dependency('json-glib-1.0') + +module_deps = [ + codecore_dep, + json_dep +] + +shared_module( + module_name, + module_files, + dependencies: module_deps, + install: true, + install_dir: join_paths(pluginsdir, module_name), +) + +custom_target(module_name + '.plugin_merge', + input: module_name + '.plugin', + output: module_name + '.plugin', + command : [msgfmt, + '--desktop', + '--keyword=Description', + '--keyword=Name', + '-d' + join_paths(meson.source_root (), 'po', 'plugins'), + '--template=@INPUT@', + '-o@OUTPUT@', + ], + install : true, + install_dir: join_paths(pluginsdir, module_name), +) diff --git a/plugins/snippets/plugin.vala b/plugins/snippets/plugin.vala new file mode 100644 index 0000000000..43b98a7d86 --- /dev/null +++ b/plugins/snippets/plugin.vala @@ -0,0 +1,315 @@ +/* + * Copyright (c) 2011 Lucas Baudin + * + * This is a 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 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; see the file COPYING. If not, + * write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301 USA. + * + */ + + + private class Code.Plugins.Snippets.Snippet : Object { + public string name; + public string tag; + public string language; + + public Gee.ArrayList tabstops; + public uint n_tabstops { + get { + return tabstops.size; + } + } + + private string _body; + public string body { + get { return _body; } + set { + _body = value; + + // PARSE $1 - $n + int next_tabstop = 1; + int next_tabstop_idx = 1; + while ((next_tabstop_idx = _body.index_of ("$%d".printf(next_tabstop), 0)) >= 0) { + for (int i = 0; i < next_tabstop - 1; i++) { + if (tabstops[i] > next_tabstop_idx) { + tabstops[i] -= 2; + } + } + + tabstops.add (next_tabstop_idx); + _body = _body.splice (next_tabstop_idx, next_tabstop_idx + 2); + next_tabstop++; + } + + // PARSE $0 + next_tabstop_idx = _body.index_of ("$0", 0); + if (next_tabstop_idx >= 0) { + for (int i = 0; i < next_tabstop - 1; i++) { + if (tabstops[i] > next_tabstop_idx) { + tabstops[i] -= 2; + } + } + tabstops.add (next_tabstop_idx); + _body = _body.splice (next_tabstop_idx, next_tabstop_idx + 2); + } else { + tabstops.add (_body.length); + } + } + } + + public Snippet (string name, string tag, string body, string language) { + tabstops = new Gee.ArrayList (); + this.name = name; + this.tag = tag; + this.body = body; + this.language = language; + } +} + +private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Object { + public string name; + public Code.Plugins.Snippets.Plugin snippets; + public Gee.HashMultiMap snippet_map; + + private int current_tabstop = 0; + private Snippet current_editing_snippet; + + public Provider () { + snippet_map = new Gee.HashMultiMap (); + + var par = new Json.Parser (); + string user_snippets_filename = Environment.get_user_data_dir () + "/code/snippets/snippets.json"; + par.load_from_file (user_snippets_filename); + var reader = new Json.Reader (par.get_root ()); + + foreach (string language in reader.list_members ()) { + reader.read_member (language); + + var n_snippets = reader.count_elements (); + for (int i = 0; i < n_snippets; i++) { + reader.read_element (i); + + reader.read_member ("name"); + var name = reader.get_string_value (); + reader.end_member (); + + reader.read_member ("tag"); + var tag = reader.get_string_value (); + reader.end_member (); + + reader.read_member ("body"); + var body = reader.get_string_value (); + reader.end_member (); + + var snippet = new Snippet (name, tag, body, language); + + snippet_map.@set (language, snippet); + reader.end_element (); + } + reader.end_member (); + } + + } + + public string get_name () { + return this.name; + } + + private string word_prefix (Gtk.SourceCompletionContext context) { + Gtk.TextIter word; + context.get_iter (out word); + var word_start = word.copy (); + word_start.backward_word_start (); + + return word.get_buffer ().get_text (word_start, word, false); + } + + public bool match (Gtk.SourceCompletionContext context) { + var word = word_prefix (context); + var snips = snippet_map.@get(snippets.current_document.get_language_name ()); + var found = false; + snips.foreach ((sni) => { + if (sni.tag.has_prefix (word)) found = true; + return !found; + }); + return found; + } + + public void populate (Gtk.SourceCompletionContext context) { + var word = word_prefix (context); + var snips = snippet_map.@get(snippets.current_document.get_language_name ()); + var props = new List (); + + snips.foreach ((sni) => { + if (sni.tag.has_prefix (word)) { + var item = new Gtk.SourceCompletionItem () { + label = sni.name, + text = sni.body + }; + item.set_data ("snippet", sni); + props.append (item); + } + return true; + }); + context.add_proposals (this, props, true); + } + + public void start_editing_placeholders () { + var current_view = snippets.current_view; + var buffer = current_view.buffer; + Gtk.TextIter snippet_start; + buffer.get_iter_at_mark (out snippet_start, buffer.get_mark ("SNIPPET_START")); + + for (int i = 0; i < current_editing_snippet.n_tabstops; i++) { + Gtk.TextIter tab_i = snippet_start.copy (); + tab_i.forward_chars (current_editing_snippet.tabstops[i]); + current_view.buffer.create_mark ("SNIPPET_TAB_%d".printf(i), tab_i, true); + } + + current_tabstop = 0; + place_cursor_at_tabstop (buffer, current_tabstop); + if (current_editing_snippet.n_tabstops > 1) { + current_view.completion.block_interactive (); + current_view.key_press_event.connect (next_placeholder); + } else { + end_editing_placeholders (); + } + } + + public bool next_placeholder (Gdk.EventKey evt) { + if (evt.keyval == Gdk.Key.Tab) { + var current_view = snippets.current_view; + var buffer = current_view.buffer; + + current_tabstop++; + place_cursor_at_tabstop (buffer, current_tabstop); + if (current_tabstop == current_editing_snippet.n_tabstops - 1) { + end_editing_placeholders (); + } + return true; + } + return false; + } + + public void place_cursor_at_tabstop (Gtk.TextBuffer buffer, int tabstop) { + Gtk.TextIter iter_start; + var mark = buffer.get_mark ("SNIPPET_TAB_%d".printf(tabstop)); + buffer.get_iter_at_mark (out iter_start, mark); + buffer.place_cursor (iter_start); + } + + public void end_editing_placeholders () { + if (current_editing_snippet.n_tabstops > 0) { + snippets.current_view.completion.unblock_interactive (); + snippets.current_view.key_press_event.disconnect (next_placeholder); + } + for (int i = 0; i < current_editing_snippet.n_tabstops; i++) { + snippets.current_view.buffer.delete_mark_by_name ("SNIPPET_TAB_%d".printf(i)); + } + snippets.current_view.buffer.delete_mark_by_name ("SNIPPET_START"); + } + + public bool activate_proposal (Gtk.SourceCompletionProposal proposal, Gtk.TextIter iter) { + current_editing_snippet = proposal.get_data ("snippet"); + var iter_start = iter.copy(); + iter_start.backward_word_start (); + + var current_buffer = iter.get_buffer (); + current_buffer.delete (ref iter_start, ref iter); + current_buffer.create_mark ("SNIPPET_START", iter_start, true); + iter.get_buffer ().insert (ref iter, current_editing_snippet.body, current_editing_snippet.body.length); + start_editing_placeholders (); + return true; + } +} + + +public class Code.Plugins.Snippets.Plugin : Peas.ExtensionBase, Peas.Activatable { + public Object object { owned get; construct; } + + private List text_view_list = new List (); + // public Euclide.Completion.Parser parser {get; private set;} + public Gtk.SourceView? current_view {get; private set;} + public Scratch.Services.Document current_document {get; private set;} + + private Scratch.MainWindow main_window; + private Scratch.Services.Interface plugins; + + + public void activate () { + plugins = (Scratch.Services.Interface) object; + plugins.hook_window.connect ((w) => { + this.main_window = w; + }); + + plugins.hook_document.connect (on_new_source_view); + } + + public void deactivate () { + text_view_list.@foreach (cleanup); + } + + public void update_state () { + + } + + public void on_new_source_view (Scratch.Services.Document doc) { + if (current_view != null) { + if (current_view == doc.source_view) + return; + + cleanup (current_view); + } + + current_document = doc; + current_view = doc.source_view; + + if (text_view_list.find (current_view) == null) + text_view_list.append (current_view); + + var comp_provider = new Code.Plugins.Snippets.Provider () { + snippets = this, + name = "Snippets" + }; + + try { + current_view.completion.add_provider (comp_provider); + current_view.completion.show_headers = true; + current_view.completion.show_icons = true; + } catch (Error e) { + warning (e.message); + } + } + + private void cleanup (Gtk.SourceView view) { + current_view.completion.get_providers ().foreach ((p) => { + try { + /* Only remove provider added by this plug in */ + if (p.get_name () == "Snippets") { + debug ("removing provider %s", p.get_name ()); + current_view.completion.remove_provider (p); + } + } catch (Error e) { + warning (e.message); + } + }); + } +} + +[ModuleInit] +public void peas_register_types (GLib.TypeModule module) { + var objmodule = module as Peas.ObjectModule; + objmodule.register_extension_type (typeof (Peas.Activatable), + typeof (Code.Plugins.Snippets.Plugin)); +} diff --git a/plugins/snippets/snippets.json b/plugins/snippets/snippets.json new file mode 100644 index 0000000000..6bdd87d243 --- /dev/null +++ b/plugins/snippets/snippets.json @@ -0,0 +1,11 @@ +{ + "Markdown": [ + + ], + + "HTML": [ + {"name": "html", "tag": "html", "body": "\n$0\n"}, + {"name": "div", "tag": "div", "body": "
\n$0\n
"}, + {"name": "ul+li3", "tag": "ul+li", "body": "
    \n
  • $1
  • \n
  • $2
  • \n
  • $3
  • \n
"} + ] +} \ No newline at end of file diff --git a/plugins/snippets/snippets.plugin b/plugins/snippets/snippets.plugin new file mode 100644 index 0000000000..edb4385959 --- /dev/null +++ b/plugins/snippets/snippets.plugin @@ -0,0 +1,9 @@ +[Plugin] +Name=Snippets +Module=snippets +Loader=C +IAge=2 +Description=Turbo charge productivity using snippets +Authors=Igor Montagner +Copyright=Copyright © Igor Montagner +Website=https://github.com/elementary/code From a1c907ba8dd19b3086d3868db3b226f18d37ac31 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Wed, 6 Oct 2021 11:00:20 -0300 Subject: [PATCH 02/13] Timeout for editing. --- plugins/snippets/plugin.vala | 18 +++++++++++++++++- 1 file changed, 17 insertions(+), 1 deletion(-) diff --git a/plugins/snippets/plugin.vala b/plugins/snippets/plugin.vala index 43b98a7d86..bbe2ed0b70 100644 --- a/plugins/snippets/plugin.vala +++ b/plugins/snippets/plugin.vala @@ -1,5 +1,5 @@ /* - * Copyright (c) 2011 Lucas Baudin + * Copyright (c) 2021 Igor Montagner * * This is a free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as @@ -84,6 +84,8 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj private int current_tabstop = 0; private Snippet current_editing_snippet; + private uint placeholder_edit_timeout = -1; + private bool still_editing_placeholder = false; public Provider () { snippet_map = new Gee.HashMultiMap (); @@ -182,12 +184,23 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj if (current_editing_snippet.n_tabstops > 1) { current_view.completion.block_interactive (); current_view.key_press_event.connect (next_placeholder); + + still_editing_placeholder = true; + placeholder_edit_timeout = Timeout.add (500, () => { + if (!still_editing_placeholder) { + end_editing_placeholders (); + return false; + } + still_editing_placeholder = false; + return true; + }); } else { end_editing_placeholders (); } } public bool next_placeholder (Gdk.EventKey evt) { + still_editing_placeholder = true; if (evt.keyval == Gdk.Key.Tab) { var current_view = snippets.current_view; var buffer = current_view.buffer; @@ -210,6 +223,7 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj } public void end_editing_placeholders () { + still_editing_placeholder = false; if (current_editing_snippet.n_tabstops > 0) { snippets.current_view.completion.unblock_interactive (); snippets.current_view.key_press_event.disconnect (next_placeholder); @@ -218,6 +232,8 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj snippets.current_view.buffer.delete_mark_by_name ("SNIPPET_TAB_%d".printf(i)); } snippets.current_view.buffer.delete_mark_by_name ("SNIPPET_START"); + Source.remove (placeholder_edit_timeout); + placeholder_edit_timeout = -1; } public bool activate_proposal (Gtk.SourceCompletionProposal proposal, Gtk.TextIter iter) { From 2534a6f5a524dabc9cc09fd6ddf1041b0a93dfdc Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Fri, 15 Oct 2021 22:32:02 -0300 Subject: [PATCH 03/13] Code quality fixes. --- plugins/snippets/plugin.vala | 59 +++++++++++++++++++----------------- 1 file changed, 32 insertions(+), 27 deletions(-) diff --git a/plugins/snippets/plugin.vala b/plugins/snippets/plugin.vala index bbe2ed0b70..046595bb2d 100644 --- a/plugins/snippets/plugin.vala +++ b/plugins/snippets/plugin.vala @@ -20,9 +20,9 @@ private class Code.Plugins.Snippets.Snippet : Object { - public string name; - public string tag; - public string language; + public string name {get; construct; } + public string tag {get; construct; } + public string language {get; construct; } public Gee.ArrayList tabstops; public uint n_tabstops { @@ -40,7 +40,7 @@ // PARSE $1 - $n int next_tabstop = 1; int next_tabstop_idx = 1; - while ((next_tabstop_idx = _body.index_of ("$%d".printf(next_tabstop), 0)) >= 0) { + while ((next_tabstop_idx = _body.index_of ("$%d".printf (next_tabstop), 0)) >= 0) { for (int i = 0; i < next_tabstop - 1; i++) { if (tabstops[i] > next_tabstop_idx) { tabstops[i] -= 2; @@ -69,16 +69,21 @@ } public Snippet (string name, string tag, string body, string language) { + Object ( + name: name, + tag: tag, + body: body, + language: language + ); + } + + construct { tabstops = new Gee.ArrayList (); - this.name = name; - this.tag = tag; - this.body = body; - this.language = language; } } private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Object { - public string name; + public string name {get; set; } public Code.Plugins.Snippets.Plugin snippets; public Gee.HashMultiMap snippet_map; @@ -87,17 +92,17 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj private uint placeholder_edit_timeout = -1; private bool still_editing_placeholder = false; - public Provider () { + construct { snippet_map = new Gee.HashMultiMap (); var par = new Json.Parser (); string user_snippets_filename = Environment.get_user_data_dir () + "/code/snippets/snippets.json"; par.load_from_file (user_snippets_filename); var reader = new Json.Reader (par.get_root ()); - + foreach (string language in reader.list_members ()) { reader.read_member (language); - + var n_snippets = reader.count_elements (); for (int i = 0; i < n_snippets; i++) { reader.read_element (i); @@ -121,11 +126,7 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj } reader.end_member (); } - - } - public string get_name () { - return this.name; } private string word_prefix (Gtk.SourceCompletionContext context) { @@ -139,7 +140,7 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj public bool match (Gtk.SourceCompletionContext context) { var word = word_prefix (context); - var snips = snippet_map.@get(snippets.current_document.get_language_name ()); + var snips = snippet_map.@get (snippets.current_document.get_language_name ()); var found = false; snips.foreach ((sni) => { if (sni.tag.has_prefix (word)) found = true; @@ -149,8 +150,8 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj } public void populate (Gtk.SourceCompletionContext context) { - var word = word_prefix (context); - var snips = snippet_map.@get(snippets.current_document.get_language_name ()); + var word = word_prefix (context); + var snips = snippet_map.@get (snippets.current_document.get_language_name ()); var props = new List (); snips.foreach ((sni) => { @@ -175,16 +176,16 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj for (int i = 0; i < current_editing_snippet.n_tabstops; i++) { Gtk.TextIter tab_i = snippet_start.copy (); - tab_i.forward_chars (current_editing_snippet.tabstops[i]); - current_view.buffer.create_mark ("SNIPPET_TAB_%d".printf(i), tab_i, true); + tab_i.forward_chars (current_editing_snippet.tabstops[i]); + current_view.buffer.create_mark ("SNIPPET_TAB_%d".printf (i), tab_i, true); } current_tabstop = 0; place_cursor_at_tabstop (buffer, current_tabstop); if (current_editing_snippet.n_tabstops > 1) { - current_view.completion.block_interactive (); + current_view.completion.block_interactive (); current_view.key_press_event.connect (next_placeholder); - + still_editing_placeholder = true; placeholder_edit_timeout = Timeout.add (500, () => { if (!still_editing_placeholder) { @@ -209,15 +210,17 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj place_cursor_at_tabstop (buffer, current_tabstop); if (current_tabstop == current_editing_snippet.n_tabstops - 1) { end_editing_placeholders (); - } + } + return true; } + return false; } public void place_cursor_at_tabstop (Gtk.TextBuffer buffer, int tabstop) { Gtk.TextIter iter_start; - var mark = buffer.get_mark ("SNIPPET_TAB_%d".printf(tabstop)); + var mark = buffer.get_mark ("SNIPPET_TAB_%d".printf (tabstop)); buffer.get_iter_at_mark (out iter_start, mark); buffer.place_cursor (iter_start); } @@ -228,9 +231,11 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj snippets.current_view.completion.unblock_interactive (); snippets.current_view.key_press_event.disconnect (next_placeholder); } + for (int i = 0; i < current_editing_snippet.n_tabstops; i++) { - snippets.current_view.buffer.delete_mark_by_name ("SNIPPET_TAB_%d".printf(i)); + snippets.current_view.buffer.delete_mark_by_name ("SNIPPET_TAB_%d".printf (i)); } + snippets.current_view.buffer.delete_mark_by_name ("SNIPPET_START"); Source.remove (placeholder_edit_timeout); placeholder_edit_timeout = -1; @@ -238,7 +243,7 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj public bool activate_proposal (Gtk.SourceCompletionProposal proposal, Gtk.TextIter iter) { current_editing_snippet = proposal.get_data ("snippet"); - var iter_start = iter.copy(); + var iter_start = iter.copy (); iter_start.backward_word_start (); var current_buffer = iter.get_buffer (); From bb4fd20c8b20ac5d878a8e8c1e94098538338257 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Mon, 18 Oct 2021 19:26:53 -0300 Subject: [PATCH 04/13] Install snippets.json --- plugins/snippets/meson.build | 5 +++++ plugins/snippets/plugin.vala | 12 ++++++------ 2 files changed, 11 insertions(+), 6 deletions(-) diff --git a/plugins/snippets/meson.build b/plugins/snippets/meson.build index 487f2cc1e2..04d076b645 100644 --- a/plugins/snippets/meson.build +++ b/plugins/snippets/meson.build @@ -19,6 +19,11 @@ shared_module( install_dir: join_paths(pluginsdir, module_name), ) +install_data( + 'snippets.json', + install_dir: join_paths(pluginsdir, module_name), +) + custom_target(module_name + '.plugin_merge', input: module_name + '.plugin', output: module_name + '.plugin', diff --git a/plugins/snippets/plugin.vala b/plugins/snippets/plugin.vala index 046595bb2d..feb59bf786 100644 --- a/plugins/snippets/plugin.vala +++ b/plugins/snippets/plugin.vala @@ -83,7 +83,6 @@ } private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Object { - public string name {get; set; } public Code.Plugins.Snippets.Plugin snippets; public Gee.HashMultiMap snippet_map; @@ -96,7 +95,7 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj snippet_map = new Gee.HashMultiMap (); var par = new Json.Parser (); - string user_snippets_filename = Environment.get_user_data_dir () + "/code/snippets/snippets.json"; + string user_snippets_filename = GLib.Path.build_filename (Constants.PLUGINDIR, "snippets", "snippets.json"); par.load_from_file (user_snippets_filename); var reader = new Json.Reader (par.get_root ()); @@ -129,6 +128,10 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj } + public string get_name () { + return "Snippets"; + } + private string word_prefix (Gtk.SourceCompletionContext context) { Gtk.TextIter word; context.get_iter (out word); @@ -260,14 +263,12 @@ public class Code.Plugins.Snippets.Plugin : Peas.ExtensionBase, Peas.Activatable public Object object { owned get; construct; } private List text_view_list = new List (); - // public Euclide.Completion.Parser parser {get; private set;} public Gtk.SourceView? current_view {get; private set;} public Scratch.Services.Document current_document {get; private set;} private Scratch.MainWindow main_window; private Scratch.Services.Interface plugins; - public void activate () { plugins = (Scratch.Services.Interface) object; plugins.hook_window.connect ((w) => { @@ -300,8 +301,7 @@ public class Code.Plugins.Snippets.Plugin : Peas.ExtensionBase, Peas.Activatable text_view_list.append (current_view); var comp_provider = new Code.Plugins.Snippets.Provider () { - snippets = this, - name = "Snippets" + snippets = this }; try { From 929695349a619ab315e1c839072f998955bfaa45 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Mon, 18 Oct 2021 20:10:01 -0300 Subject: [PATCH 05/13] Add Vala snippets and respect indent. --- plugins/preserve-indent/preserve-indent.vala | 121 +------------------ plugins/snippets/plugin.vala | 16 ++- plugins/snippets/snippets.json | 19 ++- src/Utils.vala | 114 +++++++++++++++++ 4 files changed, 145 insertions(+), 125 deletions(-) diff --git a/plugins/preserve-indent/preserve-indent.vala b/plugins/preserve-indent/preserve-indent.vala index 6f07e0c3d3..a63779ecfa 100644 --- a/plugins/preserve-indent/preserve-indent.vala +++ b/plugins/preserve-indent/preserve-indent.vala @@ -55,29 +55,7 @@ public class Scratch.Plugins.PreserveIndent : Peas.ExtensionBase, Peas.Activatab public void update_state () { } - // determine how many characters precede a given iterator position - private int measure_indent_at_iter (Widgets.SourceView view, Gtk.TextIter iter) { - Gtk.TextIter line_begin, pos; - - view.buffer.get_iter_at_line (out line_begin, iter.get_line ()); - - pos = line_begin; - int indent = 0; - int tabwidth = Scratch.settings.get_int ("indent-width"); - - unichar ch = pos.get_char (); - while (pos.get_offset () < iter.get_offset () && ch != '\n' && ch.isspace ()) { - if (ch == '\t') { - indent += tabwidth; - } else { - ++indent; - } - - pos.forward_char (); - ch = pos.get_char (); - } - return indent; - } + private void on_cut_or_copy_clipboard () { Widgets.SourceView view = this.active_document.source_view; @@ -90,7 +68,7 @@ public class Scratch.Plugins.PreserveIndent : Peas.ExtensionBase, Peas.Activatab var buffer = view.buffer; if (buffer.get_selection_bounds (out select_begin, out select_end)) { - int indent = this.measure_indent_at_iter (view, select_begin); + int indent = Scratch.Utils.measure_indent_at_iter (view, select_begin); this.last_clipboard_indent_level = indent; } else { this.last_clipboard_indent_level = 0; @@ -135,110 +113,21 @@ public class Scratch.Plugins.PreserveIndent : Peas.ExtensionBase, Peas.Activatab view.buffer.get_iter_at_mark (out paste_begin, view.buffer.get_mark ("paste_start")); view.buffer.get_iter_at_mark (out paste_end, view.buffer.get_insert ()); - int indent_level = this.measure_indent_at_iter (view, paste_begin); + int indent_level = Scratch.Utils.measure_indent_at_iter (view, paste_begin); int indent_diff = indent_level - this.last_clipboard_indent_level; paste_begin.forward_line (); if (indent_diff > 0) { - this.increase_indent_in_region (view, paste_begin, paste_end, indent_diff); + Scratch.Utils.increase_indent_in_region (view, paste_begin, paste_end, indent_diff); } else if (indent_diff < 0) { - this.decrease_indent_in_region (view, paste_begin, paste_end, indent_diff.abs ()); + Scratch.Utils.decrease_indent_in_region (view, paste_begin, paste_end, indent_diff.abs ()); } view.buffer.delete_mark_by_name ("paste_start"); view.buffer.end_user_action (); this.waiting_for_clipboard_text = false; } - - private void increase_indent_in_region ( - Widgets.SourceView view, - Gtk.TextIter region_begin, - Gtk.TextIter region_end, - int nchars - ) { - int first_line = region_begin.get_line (); - int last_line = region_end.get_line (); - int buf_last_line = view.buffer.get_line_count () - 1; - - int nlines = (first_line - last_line).abs () + 1; - if (nlines < 1 || nchars < 1 || last_line < first_line || !view.editable - || first_line == buf_last_line - ) { - return; - } - - // add a string of whitespace to each line after the first pasted line - string indent_str; - - if (view.insert_spaces_instead_of_tabs) { - indent_str = string.nfill (nchars, ' '); - } else { - int tabwidth = Scratch.settings.get_int ("indent-width"); - int tabs = nchars / tabwidth; - int spaces = nchars % tabwidth; - - indent_str = string.nfill (tabs, '\t'); - if (spaces > 0) { - indent_str += string.nfill (spaces, ' '); - } - } - - Gtk.TextIter itr; - for (var i = first_line; i <= last_line; ++i) { - view.buffer.get_iter_at_line (out itr, i); - view.buffer.insert (ref itr, indent_str, indent_str.length); - } - } - - private void decrease_indent_in_region ( - Widgets.SourceView view, - Gtk.TextIter region_begin, - Gtk.TextIter region_end, - int nchars - ) { - int first_line = region_begin.get_line (); - int last_line = region_end.get_line (); - - int nlines = (first_line - last_line).abs () + 1; - if (nlines < 1 || nchars < 1 || last_line < first_line || !view.editable) { - return; - } - - Gtk.TextBuffer buffer = view.buffer; - int tabwidth = Scratch.settings.get_int ("indent-width"); - Gtk.TextIter del_begin, del_end, itr; - - for (var line = first_line; line <= last_line; ++line) { - buffer.get_iter_at_line (out itr, line); - // crawl along the line and tally indentation as we go, - // when requested number of chars is hit, or if we run out of whitespace (eg. find glyphs or newline), - // delete the segment from line start to where we are now - int chars_to_delete = 0; - int indent_chars_found = 0; - unichar ch = itr.get_char (); - while (ch != '\n' && !ch.isgraph () && indent_chars_found < nchars) { - if (ch == ' ') { - ++chars_to_delete; - ++indent_chars_found; - } else if (ch == '\t') { - ++chars_to_delete; - indent_chars_found += tabwidth; - } - itr.forward_char (); - ch = itr.get_char (); - } - - if (ch == '\n' || chars_to_delete < 1) { - continue; - } - - buffer.get_iter_at_line (out del_begin, line); - buffer.get_iter_at_line_offset (out del_end, line, chars_to_delete); - buffer.delete (ref del_begin, ref del_end); - } - - } } [ModuleInit] diff --git a/plugins/snippets/plugin.vala b/plugins/snippets/plugin.vala index feb59bf786..df411a61d6 100644 --- a/plugins/snippets/plugin.vala +++ b/plugins/snippets/plugin.vala @@ -91,6 +91,8 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj private uint placeholder_edit_timeout = -1; private bool still_editing_placeholder = false; + private const int FINISH_EDITING_TIMEOUT = 1000; + construct { snippet_map = new Gee.HashMultiMap (); @@ -174,8 +176,9 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj public void start_editing_placeholders () { var current_view = snippets.current_view; var buffer = current_view.buffer; - Gtk.TextIter snippet_start; + Gtk.TextIter snippet_start, snippet_end; buffer.get_iter_at_mark (out snippet_start, buffer.get_mark ("SNIPPET_START")); + buffer.get_iter_at_mark (out snippet_end, buffer.get_mark ("SNIPPET_END")); for (int i = 0; i < current_editing_snippet.n_tabstops; i++) { Gtk.TextIter tab_i = snippet_start.copy (); @@ -183,6 +186,11 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj current_view.buffer.create_mark ("SNIPPET_TAB_%d".printf (i), tab_i, true); } + var indent_start = Scratch.Utils.measure_indent_at_iter (current_view, snippet_start); + var snippet_line2 = snippet_start.copy (); + snippet_line2.forward_line (); + Scratch.Utils.increase_indent_in_region (current_view, snippet_line2, snippet_end, indent_start); + current_tabstop = 0; place_cursor_at_tabstop (buffer, current_tabstop); if (current_editing_snippet.n_tabstops > 1) { @@ -190,7 +198,7 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj current_view.key_press_event.connect (next_placeholder); still_editing_placeholder = true; - placeholder_edit_timeout = Timeout.add (500, () => { + placeholder_edit_timeout = Timeout.add (FINISH_EDITING_TIMEOUT, () => { if (!still_editing_placeholder) { end_editing_placeholders (); return false; @@ -240,6 +248,7 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj } snippets.current_view.buffer.delete_mark_by_name ("SNIPPET_START"); + snippets.current_view.buffer.delete_mark_by_name ("SNIPPET_END"); Source.remove (placeholder_edit_timeout); placeholder_edit_timeout = -1; } @@ -252,6 +261,7 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj var current_buffer = iter.get_buffer (); current_buffer.delete (ref iter_start, ref iter); current_buffer.create_mark ("SNIPPET_START", iter_start, true); + current_buffer.create_mark ("SNIPPET_END", iter_start, false); iter.get_buffer ().insert (ref iter, current_editing_snippet.body, current_editing_snippet.body.length); start_editing_placeholders (); return true; @@ -263,7 +273,7 @@ public class Code.Plugins.Snippets.Plugin : Peas.ExtensionBase, Peas.Activatable public Object object { owned get; construct; } private List text_view_list = new List (); - public Gtk.SourceView? current_view {get; private set;} + public Scratch.Widgets.SourceView? current_view {get; private set;} public Scratch.Services.Document current_document {get; private set;} private Scratch.MainWindow main_window; diff --git a/plugins/snippets/snippets.json b/plugins/snippets/snippets.json index 6bdd87d243..9aa91a5e02 100644 --- a/plugins/snippets/snippets.json +++ b/plugins/snippets/snippets.json @@ -1,11 +1,18 @@ { - "Markdown": [ + "Vala": [ + {"name": "Namespace", "tag": "name", "body": "namespace $1 {\n $0\n}"}, + {"name": "Class", "tag": "class", "body": "class $1 {\n $0\n}"}, + {"name": "Construct", "tag": "constr", "body": "construct {\n $0\n}"}, - ], + {"name": "While", "tag": "while", "body": "while ($1) {\n $0\n}"}, + {"name": "ForEach", "tag": "foreach", "body": "foreach ($1 in $2) {\n $0\n}"}, + {"name": "For", "tag": "for", "body": "for ($1; $2; $3) { $0\n}"}, + + {"name": "If", "tag": "if", "body": "if ($1) {\n $0\n}"}, + {"name": "If", "tag": "ifelse", "body": "if ($1) {\n $0\n} else {\n \n}"}, + + {"name": "Public Method", "tag": "pubfunc", "body": "public $1 $2 ($3) {\n $0\n}"}, + {"name": "Private Method", "tag": "privfunc", "body": "public $1 $2 ($3) {\n $0\n}"} - "HTML": [ - {"name": "html", "tag": "html", "body": "\n$0\n"}, - {"name": "div", "tag": "div", "body": "
\n$0\n
"}, - {"name": "ul+li3", "tag": "ul+li", "body": "
    \n
  • $1
  • \n
  • $2
  • \n
  • $3
  • \n
"} ] } \ No newline at end of file diff --git a/src/Utils.vala b/src/Utils.vala index 52919bec03..95c027fd50 100644 --- a/src/Utils.vala +++ b/src/Utils.vala @@ -142,4 +142,118 @@ namespace Scratch.Utils { return path; } } + + // determine how many characters precede a given iterator position + public int measure_indent_at_iter (Widgets.SourceView view, Gtk.TextIter iter) { + Gtk.TextIter line_begin, pos; + + view.buffer.get_iter_at_line (out line_begin, iter.get_line ()); + + pos = line_begin; + int indent = 0; + int tabwidth = Scratch.settings.get_int ("indent-width"); + + unichar ch = pos.get_char (); + while (pos.get_offset () < iter.get_offset () && ch != '\n' && ch.isspace ()) { + if (ch == '\t') { + indent += tabwidth; + } else { + ++indent; + } + + pos.forward_char (); + ch = pos.get_char (); + } + return indent; + } + + public void increase_indent_in_region ( + Widgets.SourceView view, + Gtk.TextIter region_begin, + Gtk.TextIter region_end, + int nchars + ) { + int first_line = region_begin.get_line (); + int last_line = region_end.get_line (); + int buf_last_line = view.buffer.get_line_count () - 1; + + int nlines = (first_line - last_line).abs () + 1; + if (nlines < 1 || nchars < 1 || last_line < first_line || !view.editable + || first_line == buf_last_line + ) { + return; + } + + // add a string of whitespace to each line after the first pasted line + string indent_str; + + if (view.insert_spaces_instead_of_tabs) { + indent_str = string.nfill (nchars, ' '); + } else { + int tabwidth = Scratch.settings.get_int ("indent-width"); + int tabs = nchars / tabwidth; + int spaces = nchars % tabwidth; + + indent_str = string.nfill (tabs, '\t'); + if (spaces > 0) { + indent_str += string.nfill (spaces, ' '); + } + } + + Gtk.TextIter itr; + for (var i = first_line; i <= last_line; ++i) { + view.buffer.get_iter_at_line (out itr, i); + view.buffer.insert (ref itr, indent_str, indent_str.length); + } + } + + public void decrease_indent_in_region ( + Widgets.SourceView view, + Gtk.TextIter region_begin, + Gtk.TextIter region_end, + int nchars + ) { + int first_line = region_begin.get_line (); + int last_line = region_end.get_line (); + + int nlines = (first_line - last_line).abs () + 1; + if (nlines < 1 || nchars < 1 || last_line < first_line || !view.editable) { + return; + } + + Gtk.TextBuffer buffer = view.buffer; + int tabwidth = Scratch.settings.get_int ("indent-width"); + Gtk.TextIter del_begin, del_end, itr; + + for (var line = first_line; line <= last_line; ++line) { + buffer.get_iter_at_line (out itr, line); + // crawl along the line and tally indentation as we go, + // when requested number of chars is hit, or if we run out of whitespace (eg. find glyphs or newline), + // delete the segment from line start to where we are now + int chars_to_delete = 0; + int indent_chars_found = 0; + unichar ch = itr.get_char (); + while (ch != '\n' && !ch.isgraph () && indent_chars_found < nchars) { + if (ch == ' ') { + ++chars_to_delete; + ++indent_chars_found; + } else if (ch == '\t') { + ++chars_to_delete; + indent_chars_found += tabwidth; + } + itr.forward_char (); + ch = itr.get_char (); + } + + if (ch == '\n' || chars_to_delete < 1) { + continue; + } + + buffer.get_iter_at_line (out del_begin, line); + buffer.get_iter_at_line_offset (out del_end, line, chars_to_delete); + buffer.delete (ref del_begin, ref del_end); + } + + } + } From 9a96bc6f15da0164db6026e98047ca8c1adeace5 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Mon, 18 Oct 2021 20:12:13 -0300 Subject: [PATCH 06/13] lint --- plugins/preserve-indent/preserve-indent.vala | 2 -- 1 file changed, 2 deletions(-) diff --git a/plugins/preserve-indent/preserve-indent.vala b/plugins/preserve-indent/preserve-indent.vala index a63779ecfa..5e83214f70 100644 --- a/plugins/preserve-indent/preserve-indent.vala +++ b/plugins/preserve-indent/preserve-indent.vala @@ -55,8 +55,6 @@ public class Scratch.Plugins.PreserveIndent : Peas.ExtensionBase, Peas.Activatab public void update_state () { } - - private void on_cut_or_copy_clipboard () { Widgets.SourceView view = this.active_document.source_view; if (!view.auto_indent) { From 7708635c3e3bb9d32da079da02e1f5d2f05e7076 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Mon, 18 Oct 2021 20:37:31 -0300 Subject: [PATCH 07/13] Update snippets --- plugins/snippets/snippets.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/plugins/snippets/snippets.json b/plugins/snippets/snippets.json index 9aa91a5e02..ffcb2db956 100644 --- a/plugins/snippets/snippets.json +++ b/plugins/snippets/snippets.json @@ -6,10 +6,10 @@ {"name": "While", "tag": "while", "body": "while ($1) {\n $0\n}"}, {"name": "ForEach", "tag": "foreach", "body": "foreach ($1 in $2) {\n $0\n}"}, - {"name": "For", "tag": "for", "body": "for ($1; $2; $3) { $0\n}"}, + {"name": "For", "tag": "for", "body": "for ($1; $2; $3) {\n $0\n}"}, {"name": "If", "tag": "if", "body": "if ($1) {\n $0\n}"}, - {"name": "If", "tag": "ifelse", "body": "if ($1) {\n $0\n} else {\n \n}"}, + {"name": "IfElse", "tag": "ifelse", "body": "if ($1) {\n $0\n} else {\n \n}"}, {"name": "Public Method", "tag": "pubfunc", "body": "public $1 $2 ($3) {\n $0\n}"}, {"name": "Private Method", "tag": "privfunc", "body": "public $1 $2 ($3) {\n $0\n}"} From a6e70db5e9c3828cec0885765adb1e29142b3689 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Sun, 7 Nov 2021 15:01:02 -0300 Subject: [PATCH 08/13] Avoid crashing when used with WordCompletion. Add Details to each snippet --- plugins/snippets/plugin.vala | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/plugins/snippets/plugin.vala b/plugins/snippets/plugin.vala index df411a61d6..5db89cb157 100644 --- a/plugins/snippets/plugin.vala +++ b/plugins/snippets/plugin.vala @@ -23,7 +23,7 @@ public string name {get; construct; } public string tag {get; construct; } public string language {get; construct; } - + public Gee.ArrayList tabstops; public uint n_tabstops { get { @@ -135,8 +135,9 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj } private string word_prefix (Gtk.SourceCompletionContext context) { - Gtk.TextIter word; - context.get_iter (out word); + var ins_mark = snippets.current_view.buffer.get_insert (); + Gtk.TextIter? word; + snippets.current_view.buffer.get_iter_at_mark (out word, ins_mark); var word_start = word.copy (); word_start.backward_word_start (); @@ -163,7 +164,8 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj if (sni.tag.has_prefix (word)) { var item = new Gtk.SourceCompletionItem () { label = sni.name, - text = sni.body + text = sni.body, + info = sni.body }; item.set_data ("snippet", sni); props.append (item); From a1b13d371dcdd3832ef0ad6a65b44c26cf1c51e5 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Sun, 7 Nov 2021 17:12:55 -0300 Subject: [PATCH 09/13] vala lint --- plugins/snippets/plugin.vala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/plugins/snippets/plugin.vala b/plugins/snippets/plugin.vala index 5db89cb157..032b8d7fa1 100644 --- a/plugins/snippets/plugin.vala +++ b/plugins/snippets/plugin.vala @@ -23,7 +23,7 @@ public string name {get; construct; } public string tag {get; construct; } public string language {get; construct; } - + public Gee.ArrayList tabstops; public uint n_tabstops { get { From 0f08c24b52271d6fa2578103ecb82e97a62b5896 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Sat, 13 Nov 2021 09:25:50 -0300 Subject: [PATCH 10/13] Do not suggest snippets if editing placeholders. --- plugins/snippets/plugin.vala | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/plugins/snippets/plugin.vala b/plugins/snippets/plugin.vala index 032b8d7fa1..c5a382ce0a 100644 --- a/plugins/snippets/plugin.vala +++ b/plugins/snippets/plugin.vala @@ -145,6 +145,10 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj } public bool match (Gtk.SourceCompletionContext context) { + if (still_editing_placeholder) { + return false; + } + var word = word_prefix (context); var snips = snippet_map.@get (snippets.current_document.get_language_name ()); var found = false; @@ -196,7 +200,6 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj current_tabstop = 0; place_cursor_at_tabstop (buffer, current_tabstop); if (current_editing_snippet.n_tabstops > 1) { - current_view.completion.block_interactive (); current_view.key_press_event.connect (next_placeholder); still_editing_placeholder = true; @@ -241,7 +244,6 @@ private class Code.Plugins.Snippets.Provider : Gtk.SourceCompletionProvider, Obj public void end_editing_placeholders () { still_editing_placeholder = false; if (current_editing_snippet.n_tabstops > 0) { - snippets.current_view.completion.unblock_interactive (); snippets.current_view.key_press_event.disconnect (next_placeholder); } From 5a9182d39ee6b43d79818c05e17b77b52b5449a3 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Sat, 13 Nov 2021 09:19:17 -0300 Subject: [PATCH 11/13] Rewrite word completion to play nice with other completion providers. --- .../word-completion/completion-provider.vala | 55 +++------ plugins/word-completion/engine.vala | 54 +++----- plugins/word-completion/meson.build | 1 + plugins/word-completion/plugin.vala | 54 +++----- plugins/word-completion/prefix-tree.vala | 116 ++++++++++++++++++ 5 files changed, 175 insertions(+), 105 deletions(-) create mode 100644 plugins/word-completion/prefix-tree.vala diff --git a/plugins/word-completion/completion-provider.vala b/plugins/word-completion/completion-provider.vala index 7a9071a793..8fcf79815e 100644 --- a/plugins/word-completion/completion-provider.vala +++ b/plugins/word-completion/completion-provider.vala @@ -28,7 +28,6 @@ public class Scratch.Plugins.CompletionProvider : Gtk.SourceCompletionProvider, private Gtk.TextView? view; private Gtk.TextBuffer? buffer; private Euclide.Completion.Parser parser; - private bool proposals_found = false; private Gtk.TextMark completion_end_mark; private Gtk.TextMark completion_start_mark; @@ -53,40 +52,36 @@ public class Scratch.Plugins.CompletionProvider : Gtk.SourceCompletionProvider, } public bool match (Gtk.SourceCompletionContext context) { - return true; + Gtk.TextIter start, end; + buffer.get_iter_at_offset (out end, buffer.cursor_position); + start = end.copy(); + start.backward_word_start (); + string text = buffer.get_text (start, end, true); + + return parser.match (text); } public void populate (Gtk.SourceCompletionContext context) { /*Store current insertion point for use in activate_proposal */ GLib.List? file_props; bool no_minimum = (context.get_activation () == Gtk.SourceCompletionActivation.USER_REQUESTED); - proposals_found = get_proposals (out file_props, no_minimum); - - if (proposals_found) - context.add_proposals (this, file_props, true); - - /* Signal to plugin whether proposals are available - * If none, the completion will be active but not visible */ - can_propose (proposals_found); + get_proposals (out file_props, no_minimum); + context.add_proposals (this, file_props, true); } public bool activate_proposal (Gtk.SourceCompletionProposal proposal, Gtk.TextIter iter) { - if (proposals_found) { - /* Count backward from completion_mark instead of iter - * (avoids wrong insertion if the user is typing fast) */ - Gtk.TextIter start; - Gtk.TextIter end; - Gtk.TextMark mark; + Gtk.TextIter start; + Gtk.TextIter end; + Gtk.TextMark mark; - mark = buffer.get_mark (COMPLETION_END_MARK_NAME); - buffer.get_iter_at_mark (out end, mark); + mark = buffer.get_mark (COMPLETION_END_MARK_NAME); + buffer.get_iter_at_mark (out end, mark); - mark = buffer.get_mark (COMPLETION_START_MARK_NAME); - buffer.get_iter_at_mark (out start, mark); + mark = buffer.get_mark (COMPLETION_START_MARK_NAME); + buffer.get_iter_at_mark (out start, mark); - buffer.@delete (ref start, ref end); - buffer.insert (ref start, proposal.get_text (), proposal.get_text ().length); - } + buffer.@delete (ref start, ref end); + buffer.insert (ref start, proposal.get_text (), proposal.get_text ().length); return true; } @@ -95,11 +90,6 @@ public class Scratch.Plugins.CompletionProvider : Gtk.SourceCompletionProvider, Gtk.SourceCompletionActivation.USER_REQUESTED; } - public unowned Gtk.Widget? get_info_widget (Gtk.SourceCompletionProposal proposal) { - /* As no additional info is provided no widget is needed */ - return null; - } - public int get_interactive_delay () { return 0; } @@ -116,11 +106,6 @@ public class Scratch.Plugins.CompletionProvider : Gtk.SourceCompletionProvider, return true; } - public void update_info (Gtk.SourceCompletionProposal proposal, Gtk.SourceCompletionInfo info) { - /* No additional info provided on proposals */ - return; - } - private bool get_proposals (out GLib.List? props, bool no_minimum) { string to_find = ""; Gtk.TextIter iter; @@ -158,13 +143,13 @@ public class Scratch.Plugins.CompletionProvider : Gtk.SourceCompletionProvider, /* There is no minimum length of word to find if the user requested a completion */ if (no_minimum || to_find.length >= Euclide.Completion.Parser.MINIMUM_WORD_LENGTH) { /* Get proposals, if any */ - Gee.TreeSet prop_word_list; + List prop_word_list; if (parser.get_for_word (to_find, out prop_word_list)) { foreach (var word in prop_word_list) { var item = new Gtk.SourceCompletionItem (); item.label = word; item.text = word; - props.prepend (item); + props.append (item); } return true; diff --git a/plugins/word-completion/engine.vala b/plugins/word-completion/engine.vala index 55273ffa29..6fa58a1fc2 100644 --- a/plugins/word-completion/engine.vala +++ b/plugins/word-completion/engine.vala @@ -22,74 +22,58 @@ public class Euclide.Completion.Parser : GLib.Object { public const int MINIMUM_WORD_LENGTH = 1; public const int MAX_TOKENS = 1000000; + private Scratch.Plugins.PrefixTree prefix_tree; + public const string DELIMITERS = " .,;:?{}[]()0123456789+-=&|-<>*\\/\n\t\'\""; public bool is_delimiter (unichar c) { return DELIMITERS.index_of_char (c) >= 0; } - public Gee.HashMap> text_view_words; + public Gee.HashMap text_view_words; public bool parsing_cancelled = false; - private Gee.ArrayList words; - private string last_word = ""; - public Parser () { - text_view_words = new Gee.HashMap> (); + text_view_words = new Gee.HashMap (); + prefix_tree = new Scratch.Plugins.PrefixTree (); } - public void add_last_word () { - add_word (last_word); + public bool match (string to_find) { + return prefix_tree.find_prefix (to_find); } - public bool get_for_word (string to_find, out Gee.TreeSet list) { - uint length = to_find.length; - list = new Gee.TreeSet (); - last_word = to_find; - if (words != null) { - lock (words) { - foreach (var word in words) { - if (word.length > length && word.slice (0, length) == to_find) { - list.add (word); - } - } - } - } - - return !list.is_empty; + public bool get_for_word (string to_find, out List list) { + list = prefix_tree.get_all_matches (to_find); + return list.length () > 0; } public void rebuild_word_list (Gtk.TextView view) { - lock (words) { - words.clear (); - } + prefix_tree.clear (); parse_text_view (view); } public void parse_text_view (Gtk.TextView view) { /* If this view has already been parsed, restore the word list */ - lock (words) { + lock (prefix_tree) { if (text_view_words.has_key (view)) { - words = text_view_words.@get (view); + prefix_tree = text_view_words.@get (view); } else { - /* Else create a new word list and parse the buffer text */ - words = new Gee.ArrayList (); + /* Else create a new word list and parse the buffer text */ + prefix_tree = new Scratch.Plugins.PrefixTree (); } } if (view.buffer.text.length > 0) { parse_string (view.buffer.text); - text_view_words.@set (view, words); + text_view_words.@set (view, prefix_tree); } } - private void add_word (string word) { + public void add_word (string word) { if (word.length < MINIMUM_WORD_LENGTH) return; - if (!(word in words)) { - lock (words) { - words.add (word); - } + lock (prefix_tree) { + prefix_tree.insert (word); } } diff --git a/plugins/word-completion/meson.build b/plugins/word-completion/meson.build index 6aa1ed0225..2887aef312 100644 --- a/plugins/word-completion/meson.build +++ b/plugins/word-completion/meson.build @@ -1,6 +1,7 @@ module_name = 'word-completion' module_files = [ + 'prefix-tree.vala', 'completion-provider.vala', 'engine.vala', 'plugin.vala' diff --git a/plugins/word-completion/plugin.vala b/plugins/word-completion/plugin.vala index 0debc5c6b1..c6f34b66bd 100644 --- a/plugins/word-completion/plugin.vala +++ b/plugins/word-completion/plugin.vala @@ -41,7 +41,6 @@ public class Scratch.Plugins.Completion : Peas.ExtensionBase, Peas.Activatable { private const uint USER_REQUESTED_KEY = Gdk.Key.backslash; private uint timeout_id = 0; - private bool completion_visible = false; public void activate () { plugins = (Scratch.Services.Interface) object; @@ -77,8 +76,6 @@ public class Scratch.Plugins.Completion : Peas.ExtensionBase, Peas.Activatable { current_document = doc; current_view = doc.source_view; current_view.key_press_event.connect (on_key_press); - current_view.completion.show.connect (on_completion_shown); - current_view.completion.hide.connect (on_completion_hidden); if (text_view_list.find (current_view) == null) text_view_list.append (current_view); @@ -86,11 +83,10 @@ public class Scratch.Plugins.Completion : Peas.ExtensionBase, Peas.Activatable { var comp_provider = new Scratch.Plugins.CompletionProvider (this); comp_provider.priority = 1; comp_provider.name = provider_name_from_document (doc); - comp_provider.can_propose.connect (on_propose); try { current_view.completion.add_provider (comp_provider); - current_view.completion.show_headers = false; + current_view.completion.show_headers = true; current_view.completion.show_icons = true; /* Wait a bit to allow text to load then run parser*/ timeout_id = Timeout.add (1000, on_timeout_update); @@ -119,64 +115,54 @@ public class Scratch.Plugins.Completion : Peas.ExtensionBase, Peas.Activatable { private bool on_key_press (Gtk.Widget view, Gdk.EventKey event) { uint kv = event.keyval; unichar uc = (unichar)(Gdk.keyval_to_unicode (kv)); + print ("key key %s\n", uc.to_string ()); + + print ("is print %s is delimiter %s\n", + uc.isprint ()?"Y":"N", + parser.is_delimiter (uc)?"Y":"N"); /* Pass through any modified keypress except Shift or Capslock */ Gdk.ModifierType mods = event.state & Gdk.ModifierType.MODIFIER_MASK & ~Gdk.ModifierType.SHIFT_MASK & ~Gdk.ModifierType.LOCK_MASK; if (mods > 0 ) { + print ("enter mod\n"); /* Default key for USER_REQUESTED completion is ControlSpace * but this is trapped elsewhere. Control + USER_REQUESTED_KEY acts as an * alternative and also purges spelling mistakes and unused words from the list. * If used when a word or part of a word is selected, the selection will be * used as the word to find. */ if ((mods & Gdk.ModifierType.CONTROL_MASK) > 0 && (kv == USER_REQUESTED_KEY)) { + print ("REBUILD\n"); parser.rebuild_word_list (current_view); current_view.show_completion (); return true; - } else - return false; + } } - bool activating = kv in ACTIVATE_KEYS; + if (kv in ACTIVATE_KEYS || + (parser.is_delimiter (uc) && uc.isprint ()) ) { + var buffer = current_view.buffer; + var mark = buffer.get_insert (); + Gtk.TextIter cursor_iter; + buffer.get_iter_at_mark (out cursor_iter, mark); - if (completion_visible && activating) { - current_view.completion.activate_proposal (); - parser.add_last_word (); - return true; - } + var word_start = cursor_iter; + word_start.backward_word_start (); - if (activating || (uc.isprint () && parser.is_delimiter (uc) )) { - parser.add_last_word (); - current_view.completion.hide (); + string word = buffer.get_text (word_start, cursor_iter, false); + parser.add_word (word); } return false; } - private void on_completion_shown () { - completion_visible = true; - } - - private void on_completion_hidden () { - completion_visible = false; - } - - private void on_propose (bool can_propose) { - if (!can_propose) - current_view.completion.hide (); - - completion_visible = can_propose; - } - private string provider_name_from_document (Scratch.Services.Document doc) { return _("%s - Word Completion").printf (doc.get_basename ()); } private void cleanup (Gtk.SourceView view) { current_view.key_press_event.disconnect (on_key_press); - current_view.completion.show.disconnect (on_completion_shown); - current_view.completion.hide.disconnect (on_completion_hidden); current_view.completion.get_providers ().foreach ((p) => { try { @@ -189,8 +175,6 @@ public class Scratch.Plugins.Completion : Peas.ExtensionBase, Peas.Activatable { warning (e.message); } }); - - completion_visible = false; } } diff --git a/plugins/word-completion/prefix-tree.vala b/plugins/word-completion/prefix-tree.vala new file mode 100644 index 0000000000..84e972d7e5 --- /dev/null +++ b/plugins/word-completion/prefix-tree.vala @@ -0,0 +1,116 @@ + +namespace Scratch.Plugins { + private class PrefixNode : Object { + public GLib.List children; + public unichar value { get; set; } + + construct { + children = new List (); + } + } + + public class PrefixTree : Object { + private PrefixNode root; + + construct { + clear (); + } + + public void clear () { + root = new PrefixNode () { + value = '\0' + }; + } + + public void insert (string word) { + if (word.length == 0) { + return; + } + + this.insert_at (word, this.root); + } + + private void insert_at (string word, PrefixNode node, int i = 0) { + unichar curr; + + word.get_next_char (ref i, out curr); + + foreach (var child in node.children) { + if (child.value == curr) { + if (curr != '\0') { + insert_at (word, child, i); + } + return; + } + } + + if (!curr.isprint () && curr != '\0') { + return; + } + + var new_child = new PrefixNode () { + value = curr + }; + node.children.insert_sorted (new_child, (c1, c2) => { + if (c1.value > c2.value) { + return 1; + } else if (c1.value == c2.value) { + return 0; + } + return -1; + }); + if (curr != '\0') { + insert_at (word, new_child, i); + } + } + + public bool find_prefix (string prefix) { + return find_prefix_at (prefix, root) != null? true : false; + } + + private PrefixNode? find_prefix_at (string prefix, PrefixNode node, int i = 0) { + unichar curr; + + prefix.get_next_char (ref i, out curr); + if (curr == '\0') { + return node; + } + + foreach (var child in node.children) { + if (child.value == curr) { + return find_prefix_at (prefix, child, i); + } + } + + return null; + } + + public List get_all_matches (string prefix) { + var list = new List (); + var node = find_prefix_at (prefix, root, 0); + if (node != null) { + var sb = new StringBuilder (prefix); + get_all_matches_rec (node, ref sb, ref list); + } + + return list; + } + + private void get_all_matches_rec ( + PrefixNode node, + ref StringBuilder sbuilder, + ref List matches) { + + foreach (var child in node.children) { + if (child.value == '\0') { + matches.append (sbuilder.str); + } else { + sbuilder.append_unichar (child.value); + get_all_matches_rec (child, ref sbuilder, ref matches); + var length = child.value.to_string ().length; + sbuilder.erase (sbuilder.len - length, -1); + } + } + } + } +} \ No newline at end of file From 0a650e65919457676403106463bfc0b18be05861 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Sat, 13 Nov 2021 09:34:41 -0300 Subject: [PATCH 12/13] lint --- plugins/word-completion/completion-provider.vala | 2 +- plugins/word-completion/plugin.vala | 9 +-------- plugins/word-completion/prefix-tree.vala | 16 ++++++++-------- 3 files changed, 10 insertions(+), 17 deletions(-) diff --git a/plugins/word-completion/completion-provider.vala b/plugins/word-completion/completion-provider.vala index 8fcf79815e..af3e4cba70 100644 --- a/plugins/word-completion/completion-provider.vala +++ b/plugins/word-completion/completion-provider.vala @@ -54,7 +54,7 @@ public class Scratch.Plugins.CompletionProvider : Gtk.SourceCompletionProvider, public bool match (Gtk.SourceCompletionContext context) { Gtk.TextIter start, end; buffer.get_iter_at_offset (out end, buffer.cursor_position); - start = end.copy(); + start = end.copy (); start.backward_word_start (); string text = buffer.get_text (start, end, true); diff --git a/plugins/word-completion/plugin.vala b/plugins/word-completion/plugin.vala index c6f34b66bd..29fe371915 100644 --- a/plugins/word-completion/plugin.vala +++ b/plugins/word-completion/plugin.vala @@ -115,32 +115,25 @@ public class Scratch.Plugins.Completion : Peas.ExtensionBase, Peas.Activatable { private bool on_key_press (Gtk.Widget view, Gdk.EventKey event) { uint kv = event.keyval; unichar uc = (unichar)(Gdk.keyval_to_unicode (kv)); - print ("key key %s\n", uc.to_string ()); - - print ("is print %s is delimiter %s\n", - uc.isprint ()?"Y":"N", - parser.is_delimiter (uc)?"Y":"N"); /* Pass through any modified keypress except Shift or Capslock */ Gdk.ModifierType mods = event.state & Gdk.ModifierType.MODIFIER_MASK & ~Gdk.ModifierType.SHIFT_MASK & ~Gdk.ModifierType.LOCK_MASK; if (mods > 0 ) { - print ("enter mod\n"); /* Default key for USER_REQUESTED completion is ControlSpace * but this is trapped elsewhere. Control + USER_REQUESTED_KEY acts as an * alternative and also purges spelling mistakes and unused words from the list. * If used when a word or part of a word is selected, the selection will be * used as the word to find. */ if ((mods & Gdk.ModifierType.CONTROL_MASK) > 0 && (kv == USER_REQUESTED_KEY)) { - print ("REBUILD\n"); parser.rebuild_word_list (current_view); current_view.show_completion (); return true; } } - if (kv in ACTIVATE_KEYS || + if (kv in ACTIVATE_KEYS || (parser.is_delimiter (uc) && uc.isprint ()) ) { var buffer = current_view.buffer; var mark = buffer.get_insert (); diff --git a/plugins/word-completion/prefix-tree.vala b/plugins/word-completion/prefix-tree.vala index 84e972d7e5..341ea24602 100644 --- a/plugins/word-completion/prefix-tree.vala +++ b/plugins/word-completion/prefix-tree.vala @@ -3,7 +3,7 @@ namespace Scratch.Plugins { private class PrefixNode : Object { public GLib.List children; public unichar value { get; set; } - + construct { children = new List (); } @@ -26,15 +26,15 @@ namespace Scratch.Plugins { if (word.length == 0) { return; } - - this.insert_at (word, this.root); + + this.insert_at (word, this.root); } private void insert_at (string word, PrefixNode node, int i = 0) { unichar curr; word.get_next_char (ref i, out curr); - + foreach (var child in node.children) { if (child.value == curr) { if (curr != '\0') { @@ -92,15 +92,15 @@ namespace Scratch.Plugins { var sb = new StringBuilder (prefix); get_all_matches_rec (node, ref sb, ref list); } - + return list; } - private void get_all_matches_rec ( - PrefixNode node, + private void get_all_matches_rec ( + PrefixNode node, ref StringBuilder sbuilder, ref List matches) { - + foreach (var child in node.children) { if (child.value == '\0') { matches.append (sbuilder.str); From 95840ab0840d07c5f93acfcedcce01f5e444c904 Mon Sep 17 00:00:00 2001 From: Igor Montagner Date: Sat, 13 Nov 2021 09:38:19 -0300 Subject: [PATCH 13/13] lint 2 --- plugins/word-completion/prefix-tree.vala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/plugins/word-completion/prefix-tree.vala b/plugins/word-completion/prefix-tree.vala index 341ea24602..1b437efc4f 100644 --- a/plugins/word-completion/prefix-tree.vala +++ b/plugins/word-completion/prefix-tree.vala @@ -113,4 +113,4 @@ namespace Scratch.Plugins { } } } -} \ No newline at end of file +}