[gnome-builder] build: add IdeTerminal to libide



commit 85bef40d1bcd5b00a92a1525416df8ae2b306c6d
Author: Christian Hergert <chergert redhat com>
Date:   Tue Nov 21 20:36:09 2017 -0800

    build: add IdeTerminal to libide
    
    This starts porting the terminal into core libide by adding
    IdeTerminal. Some source cleanup occurred as part of this
    transition.

 meson.build                        |    3 +-
 src/libide/ide.h                   |    1 +
 src/libide/meson.build             |    9 +-
 src/libide/terminal/ide-terminal.c |  378 ++++++++++++++++++++++++++++++++++++
 src/libide/terminal/ide-terminal.h |   49 +++++
 src/libide/terminal/meson.build    |   12 ++
 6 files changed, 448 insertions(+), 4 deletions(-)
---
diff --git a/meson.build b/meson.build
index de12191..dc2bbeb 100644
--- a/meson.build
+++ b/meson.build
@@ -181,7 +181,8 @@ libm_dep = cc.find_library('m', required: false)
 libpangoft2_dep = dependency('pangoft2', version: '>= 1.38.0')
 libpeas_dep = dependency('libpeas-1.0', version: '>= 1.22.0')
 libtemplate_glib_dep = dependency('template-glib-1.0', version: '>= 3.27.2')
-libxml2_dep = dependency('libxml-2.0', version: '>=2.9.0')
+libvte_dep = dependency('vte-2.91', version: '>= 0.40.2')
+libxml2_dep = dependency('libxml-2.0', version: '>= 2.9.0')
 
 if get_option('with_flatpak') or get_option('with_git')
   libgit_dep = dependency('libgit2-glib-1.0', version: '>= 0.25.0')
diff --git a/src/libide/ide.h b/src/libide/ide.h
index 320ce4a..cc92ff7 100644
--- a/src/libide/ide.h
+++ b/src/libide/ide.h
@@ -172,6 +172,7 @@ G_BEGIN_DECLS
 #include "testing/ide-test-manager.h"
 #include "testing/ide-test-provider.h"
 #include "threading/ide-thread-pool.h"
+#include "terminal/ide-terminal.h"
 #include "transfers/ide-pkcon-transfer.h"
 #include "transfers/ide-transfer.h"
 #include "transfers/ide-transfer-button.h"
diff --git a/src/libide/meson.build b/src/libide/meson.build
index b5fcc8b..b69b817 100644
--- a/src/libide/meson.build
+++ b/src/libide/meson.build
@@ -87,6 +87,7 @@ subdir('sourceview')
 subdir('subprocess')
 subdir('symbols')
 subdir('template')
+subdir('terminal')
 subdir('testing')
 subdir('threading')
 subdir('transfers')
@@ -204,6 +205,7 @@ libide_deps = [
   libpangoft2_dep,
   libpeas_dep,
   libtemplate_glib_dep,
+  libvte_dep,
   libxml2_dep,
 ]
 
@@ -249,7 +251,8 @@ libide_plugin_dep = declare_dependency(
                          libgtksource_dep,
                          libtemplate_glib_dep,
                          libjson_glib_dep,
-                         libjsonrpc_glib_dep ],
+                         libjsonrpc_glib_dep,
+                         libvte_dep ],
 )
 
 pkgg = import('pkgconfig')
@@ -261,7 +264,7 @@ pkgg.generate(
          name: 'Libide',
      filebase: 'libide-1.0',
   description: 'Libide contains the components used to build the GNOME Builder IDE.',
-     requires: [ 'gtk+-3.0', 'gtksourceview-3.0', 'libdazzle-1.0', 'template-glib-1.0', 'jsonrpc-glib-1.0', 
'libpeas-1.0' ],
+     requires: [ 'gtk+-3.0', 'gtksourceview-3.0', 'libdazzle-1.0', 'template-glib-1.0', 'jsonrpc-glib-1.0', 
'libpeas-1.0', 'vte-2.91' ],
   install_dir: join_paths(pkglibdir, 'pkgconfig'),
 )
 
@@ -271,7 +274,7 @@ libide_gir = gnome.generate_gir(libide,
             namespace: 'Ide',
         symbol_prefix: 'ide',
     identifier_prefix: 'Ide',
-             includes: [ 'Gio-2.0', 'GtkSource-3.0', 'Peas-1.0', 'Dazzle-1.0', 'Json-1.0', 'Template-1.0'],
+             includes: [ 'Gio-2.0', 'GtkSource-3.0', 'Peas-1.0', 'Dazzle-1.0', 'Json-1.0', 'Template-1.0', 
'Vte-2.91' ],
              install: true,
       install_dir_gir: pkggirdir,
   install_dir_typelib: pkgtypelibdir,
diff --git a/src/libide/terminal/ide-terminal.c b/src/libide/terminal/ide-terminal.c
new file mode 100644
index 0000000..4e3696f
--- /dev/null
+++ b/src/libide/terminal/ide-terminal.c
@@ -0,0 +1,378 @@
+/* ide-terminal.c
+ *
+ * Copyright © 2016-2017 Christian Hergert <christian hergert me>
+ *
+ * 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 3 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <dazzle.h>
+#include <glib/gi18n.h>
+#include <ide.h>
+
+#include "terminal/ide-terminal.h"
+
+#define BUILDER_PCRE2_MULTILINE 0x00000400u
+
+typedef struct
+{
+  GtkWidget *popup_menu;
+  gchar     *url;
+} IdeTerminalPrivate;
+
+typedef struct
+{
+  IdeTerminal *terminal;
+  GdkEvent   *event;
+} PopupInfo;
+
+G_DEFINE_TYPE_WITH_PRIVATE (IdeTerminal, ide_terminal, VTE_TYPE_TERMINAL)
+
+enum {
+  COPY_LINK_ADDRESS,
+  OPEN_LINK,
+  POPULATE_POPUP,
+  SELECT_ALL,
+  SEARCH_REVEAL,
+  N_SIGNALS
+};
+
+/* From vteapp.c */
+#define DINGUS1 
"(((gopher|news|telnet|nntp|file|http|ftp|https)://)|(www|ftp)[-A-Za-z0-9]*\\.)[-A-Za-z0-9\\.]+(:[0-9]*)?"
+#define DINGUS2 DINGUS1 "/[-A-Za-z0-9_\\$\\.\\+\\!\\*\\(\\),;:@&=\\?/~\\#\\%]*[^]'\\.}>\\) ,\\\"]"
+
+static guint signals[N_SIGNALS];
+static const gchar *url_regexes[] = { DINGUS1, DINGUS2 };
+
+static void
+popup_menu_detach (GtkWidget *attach_widget,
+                   GtkMenu   *menu)
+{
+  IdeTerminal *self = IDE_TERMINAL (attach_widget);
+  IdeTerminalPrivate *priv = ide_terminal_get_instance_private (self);
+
+  g_assert (IDE_IS_TERMINAL (self));
+  g_assert (GTK_IS_MENU (menu));
+  g_assert (priv->popup_menu == NULL || GTK_IS_WIDGET (priv->popup_menu));
+
+  g_clear_pointer (&priv->url, g_free);
+
+  if (priv->popup_menu == GTK_WIDGET (menu))
+    g_clear_pointer (&priv->popup_menu, gtk_widget_destroy);
+}
+
+static void
+popup_targets_received (GtkClipboard     *clipboard,
+                        GtkSelectionData *data,
+                        gpointer          user_data)
+{
+  PopupInfo *popup_info = user_data;
+  g_autoptr(IdeTerminal) self = NULL;
+  g_autoptr(GdkEvent) event = NULL;
+  IdeTerminalPrivate *priv;
+
+  g_assert (popup_info != NULL);
+  g_assert (IDE_IS_TERMINAL (popup_info->terminal));
+
+  self = g_steal_pointer (&popup_info->terminal);
+  priv = ide_terminal_get_instance_private (self);
+  event = g_steal_pointer (&popup_info->event);
+
+  if (gtk_widget_get_realized (GTK_WIDGET (self)))
+    {
+      DzlWidgetActionGroup *group;
+      GMenu *menu;
+      gboolean clipboard_contains_text;
+      gboolean have_selection;
+
+      clipboard_contains_text = gtk_selection_data_targets_include_text (data);
+      have_selection = vte_terminal_get_has_selection (VTE_TERMINAL (self));
+
+      g_clear_pointer (&priv->popup_menu, gtk_widget_destroy);
+
+      priv->url = vte_terminal_match_check_event (VTE_TERMINAL (self), event, NULL);
+
+      menu = dzl_application_get_menu_by_id (DZL_APPLICATION_DEFAULT, "ide-terminal-view-popup-menu");
+      priv->popup_menu = gtk_menu_new_from_model (G_MENU_MODEL (menu));
+
+      group = DZL_WIDGET_ACTION_GROUP (gtk_widget_get_action_group (GTK_WIDGET (self), "terminal"));
+
+      dzl_widget_action_group_set_action_enabled (group, "copy-link-address", priv->url != NULL);
+      dzl_widget_action_group_set_action_enabled (group, "open-link", priv->url != NULL);
+      dzl_widget_action_group_set_action_enabled (group, "copy-clipboard", have_selection);
+      dzl_widget_action_group_set_action_enabled (group, "paste-clipboard", clipboard_contains_text);
+
+      dzl_gtk_widget_add_style_class (priv->popup_menu, GTK_STYLE_CLASS_CONTEXT_MENU);
+      gtk_menu_attach_to_widget (GTK_MENU (priv->popup_menu), GTK_WIDGET (self), popup_menu_detach);
+
+      g_signal_emit (self, signals[POPULATE_POPUP], 0, priv->popup_menu);
+
+      gtk_menu_popup_at_pointer (GTK_MENU (priv->popup_menu), event);
+    }
+
+  g_slice_free (PopupInfo, popup_info);
+}
+
+static void
+ide_terminal_do_popup (IdeTerminal    *self,
+                       const GdkEvent *event)
+{
+  PopupInfo *popup_info;
+  GtkClipboard *clipboard;
+
+  g_assert (IDE_IS_TERMINAL (self));
+
+  popup_info = g_slice_new0 (PopupInfo);
+  popup_info->event = event ? gdk_event_copy (event) : gtk_get_current_event ();
+  popup_info->terminal = g_object_ref (self);
+
+  clipboard = gtk_widget_get_clipboard (GTK_WIDGET (self), GDK_SELECTION_CLIPBOARD);
+
+  gtk_clipboard_request_contents (clipboard,
+                                  gdk_atom_intern_static_string ("TARGETS"),
+                                  popup_targets_received,
+                                  popup_info);
+}
+
+static gboolean
+ide_terminal_popup_menu (GtkWidget *widget)
+{
+  IdeTerminal *self = (IdeTerminal *)widget;
+
+  g_assert (IDE_IS_TERMINAL (self));
+
+  ide_terminal_do_popup (self, NULL);
+
+  return TRUE;
+}
+
+static gboolean
+ide_terminal_button_press_event (GtkWidget      *widget,
+                                 GdkEventButton *button)
+{
+  IdeTerminal *self = (IdeTerminal *)widget;
+
+  g_assert (IDE_IS_TERMINAL (self));
+  g_assert (button != NULL);
+
+  if ((button->type == GDK_BUTTON_PRESS) && (button->button == GDK_BUTTON_SECONDARY))
+    {
+      if (!gtk_widget_has_focus (GTK_WIDGET (self)))
+        gtk_widget_grab_focus (GTK_WIDGET (self));
+
+      ide_terminal_do_popup (self, (GdkEvent *)button);
+
+      return GDK_EVENT_STOP;
+    }
+  else if ((button->type == GDK_BUTTON_PRESS) && (button->button == GDK_BUTTON_PRIMARY)
+            && ((button->state & GDK_CONTROL_MASK) == GDK_CONTROL_MASK))
+    {
+      g_autofree gchar *pattern = NULL;
+
+      pattern = vte_terminal_match_check_event (VTE_TERMINAL (self), (GdkEvent *)button, NULL);
+
+      if (pattern != NULL)
+        {
+          GtkApplication *app;
+          GtkWindow *focused_window;
+
+          if (NULL != (app = GTK_APPLICATION (g_application_get_default ())) &&
+              NULL != (focused_window = gtk_application_get_active_window (app)))
+            gtk_show_uri_on_window (focused_window,
+                                    pattern,
+                                    gtk_get_current_event_time (),
+                                    NULL);
+        }
+
+      return GDK_EVENT_STOP;
+    }
+
+  return GTK_WIDGET_CLASS (ide_terminal_parent_class)->button_press_event (widget, button);
+}
+
+static void
+ide_terminal_real_select_all (IdeTerminal *self,
+                              gboolean     all)
+{
+  g_assert (IDE_IS_TERMINAL (self));
+
+  if (all)
+    vte_terminal_select_all (VTE_TERMINAL (self));
+  else
+    vte_terminal_unselect_all (VTE_TERMINAL (self));
+}
+
+static gboolean
+ide_terminal_copy_link_address (IdeTerminal *self)
+{
+  IdeTerminalPrivate *priv = ide_terminal_get_instance_private (self);
+
+  g_assert (IDE_IS_TERMINAL (self));
+  g_assert (priv->url != NULL);
+
+  if (ide_str_empty0 (priv->url))
+    return FALSE;
+
+  gtk_clipboard_set_text (gtk_widget_get_clipboard (GTK_WIDGET (self), GDK_SELECTION_CLIPBOARD),
+                          priv->url, strlen (priv->url));
+
+  return TRUE;
+}
+
+static gboolean
+ide_terminal_open_link (IdeTerminal *self)
+{
+  IdeTerminalPrivate *priv = ide_terminal_get_instance_private (self);
+  GtkApplication *app;
+  GtkWindow *focused_window;
+
+  g_assert (IDE_IS_TERMINAL (self));
+  g_assert (priv->url != NULL);
+
+  if (ide_str_empty0 (priv->url))
+    return FALSE;
+
+  if (NULL != (app = GTK_APPLICATION (g_application_get_default ())) &&
+      NULL != (focused_window = gtk_application_get_active_window (app)))
+    return gtk_show_uri_on_window (focused_window,
+                                   priv->url,
+                                   gtk_get_current_event_time (),
+                                   NULL);
+
+  return FALSE;
+}
+
+static void
+ide_terminal_real_search_reveal (IdeTerminal *self)
+{
+  GtkWidget *parent_overlay;
+
+  g_assert (IDE_IS_TERMINAL (self));
+
+  parent_overlay = gtk_widget_get_ancestor (GTK_WIDGET (self), GTK_TYPE_OVERLAY);
+
+  if (parent_overlay != NULL)
+    {
+      GtkRevealer *revealer = dzl_gtk_widget_find_child_typed (parent_overlay, GTK_TYPE_REVEALER);
+
+      if (revealer != NULL && !gtk_revealer_get_child_revealed (revealer))
+        gtk_revealer_set_reveal_child (revealer, TRUE);
+    }
+}
+
+static void
+ide_terminal_class_init (IdeTerminalClass *klass)
+{
+  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
+  GtkBindingSet *binding_set;
+
+  widget_class->button_press_event = ide_terminal_button_press_event;
+  widget_class->popup_menu = ide_terminal_popup_menu;
+
+  klass->copy_link_address = ide_terminal_copy_link_address;
+  klass->open_link = ide_terminal_open_link;
+  klass->select_all = ide_terminal_real_select_all;
+  klass->search_reveal = ide_terminal_real_search_reveal;
+
+  signals [COPY_LINK_ADDRESS] =
+    g_signal_new ("copy-link-address",
+                  G_TYPE_FROM_CLASS (klass),
+                  G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
+                  G_STRUCT_OFFSET (IdeTerminalClass, copy_link_address),
+                  NULL, NULL, NULL,
+                  G_TYPE_BOOLEAN,
+                  0);
+
+  signals [SEARCH_REVEAL] =
+    g_signal_new ("search-reveal",
+                  G_TYPE_FROM_CLASS (klass),
+                  G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
+                  G_STRUCT_OFFSET (IdeTerminalClass, search_reveal),
+                  NULL, NULL, NULL,
+                  G_TYPE_NONE,
+                  0);
+
+  signals [OPEN_LINK] =
+    g_signal_new ("open-link",
+                  G_TYPE_FROM_CLASS (klass),
+                  G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
+                  G_STRUCT_OFFSET (IdeTerminalClass, open_link),
+                  NULL, NULL, NULL,
+                  G_TYPE_BOOLEAN,
+                  0);
+
+  signals [POPULATE_POPUP] =
+    g_signal_new ("populate-popup",
+                  G_TYPE_FROM_CLASS (klass),
+                  G_SIGNAL_RUN_LAST,
+                  G_STRUCT_OFFSET (IdeTerminalClass, populate_popup),
+                  NULL, NULL, NULL,
+                  G_TYPE_NONE,
+                  1,
+                  GTK_TYPE_WIDGET);
+
+  signals [SELECT_ALL] =
+    g_signal_new ("select-all",
+                  G_TYPE_FROM_CLASS (klass),
+                  G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
+                  G_STRUCT_OFFSET (IdeTerminalClass, select_all),
+                  NULL, NULL, NULL,
+                  G_TYPE_NONE,
+                  1,
+                  G_TYPE_BOOLEAN);
+
+  binding_set = gtk_binding_set_by_class (klass);
+
+  gtk_binding_entry_add_signal (binding_set,
+                                GDK_KEY_c,
+                                GDK_SHIFT_MASK | GDK_CONTROL_MASK,
+                                "copy-clipboard",
+                                0);
+
+  gtk_binding_entry_add_signal (binding_set,
+                                GDK_KEY_v,
+                                GDK_SHIFT_MASK | GDK_CONTROL_MASK,
+                                "paste-clipboard",
+                                0);
+
+  gtk_binding_entry_add_signal (binding_set,
+                                GDK_KEY_f,
+                                GDK_SHIFT_MASK | GDK_CONTROL_MASK,
+                                "search-reveal",
+                                0);
+}
+
+static void
+ide_terminal_init (IdeTerminal *self)
+{
+  dzl_widget_action_group_attach (self, "terminal");
+
+  for (guint i = 0; i < G_N_ELEMENTS (url_regexes); i++)
+    {
+      g_autoptr(VteRegex) regex = NULL;
+      const gchar *pattern = url_regexes[i];
+      gint tag;
+
+      regex = vte_regex_new_for_match (pattern, IDE_LITERAL_LENGTH (pattern),
+                                       VTE_REGEX_FLAGS_DEFAULT | BUILDER_PCRE2_MULTILINE,
+                                       NULL);
+      tag = vte_terminal_match_add_regex (VTE_TERMINAL (self), regex, 0);
+      vte_terminal_match_set_cursor_type (VTE_TERMINAL (self), tag, GDK_HAND2);
+    }
+}
+
+GtkWidget *
+ide_terminal_new (void)
+{
+  return g_object_new (IDE_TYPE_TERMINAL, NULL);
+}
diff --git a/src/libide/terminal/ide-terminal.h b/src/libide/terminal/ide-terminal.h
new file mode 100644
index 0000000..6ff1547
--- /dev/null
+++ b/src/libide/terminal/ide-terminal.h
@@ -0,0 +1,49 @@
+/* ide-terminal.h
+ *
+ * Copyright (C) 2017 Christian Hergert <chergert redhat com>
+ *
+ * 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 3 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#pragma once
+
+#include <vte/vte.h>
+
+#include "ide-version-macros.h"
+
+G_BEGIN_DECLS
+
+#define IDE_TYPE_TERMINAL (ide_terminal_get_type())
+
+G_DECLARE_DERIVABLE_TYPE (IdeTerminal, ide_terminal, IDE, TERMINAL, VteTerminal)
+
+struct _IdeTerminalClass
+{
+  VteTerminalClass parent_class;
+
+  void     (*populate_popup)      (IdeTerminal *self,
+                                   GtkWidget   *widget);
+  void     (*select_all)          (IdeTerminal *self,
+                                   gboolean     all);
+  void     (*search_reveal)       (IdeTerminal *self);
+  gboolean (*open_link)           (IdeTerminal *self);
+  gboolean (*copy_link_address)   (IdeTerminal *self);
+
+  gpointer padding[16];
+};
+
+IDE_AVAILABLE_IN_3_28
+GtkWidget *ide_terminal_new (void);
+
+G_END_DECLS
diff --git a/src/libide/terminal/meson.build b/src/libide/terminal/meson.build
new file mode 100644
index 0000000..d232763
--- /dev/null
+++ b/src/libide/terminal/meson.build
@@ -0,0 +1,12 @@
+terminal_headers = [
+  'ide-terminal.h',
+]
+
+terminal_sources = [
+  'ide-terminal.c',
+]
+
+libide_public_headers += files(terminal_headers)
+libide_public_sources += files(terminal_sources)
+
+install_headers(terminal_headers, subdir: join_paths(libide_header_subdir, 'terminal'))


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]