gem-graph-client/libide/terminal/ide-terminal-page.c

747 lines
22 KiB
C

/* ide-terminal-page.c
*
* Copyright 2015-2019 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/>.
*
* SPDX-License-Identifier: GPL-3.0-or-later
*/
#define G_LOG_DOMAIN "ide-terminal-page"
#include "config.h"
#include <fcntl.h>
#include <glib/gi18n.h>
#include <libide-foundry.h>
#include <libide-gui.h>
#include <libide-terminal.h>
#include <stdlib.h>
#include <vte/vte.h>
#include <unistd.h>
#define PCRE2_CODE_UNIT_WIDTH 0
#include <pcre2.h>
#include "ide-terminal-page.h"
#include "ide-terminal-page-private.h"
#include "ide-terminal-page-actions.h"
#define FLAPPING_DURATION_USEC (G_USEC_PER_SEC / 20)
G_DEFINE_FINAL_TYPE (IdeTerminalPage, ide_terminal_page, IDE_TYPE_PAGE)
enum {
PROP_0,
PROP_CLOSE_ON_EXIT,
PROP_LAUNCHER,
PROP_MANAGE_SPAWN,
PROP_RESPAWN_ON_EXIT,
PROP_PTY,
N_PROPS
};
enum {
TEXT_INSERTED,
N_SIGNALS
};
static GParamSpec *properties [N_PROPS];
static guint signals [N_SIGNALS];
static void ide_terminal_page_connect_terminal (IdeTerminalPage *self,
VteTerminal *terminal);
static gboolean
terminal_has_notification_signal (void)
{
GQuark quark;
guint signal_id;
return g_signal_parse_name ("notification-received",
VTE_TYPE_TERMINAL,
&signal_id,
&quark,
FALSE);
}
static gboolean
destroy_widget_in_idle (GtkWidget *widget)
{
IdeTerminalPage *self = (IdeTerminalPage *)widget;
IDE_ENTRY;
g_assert (IDE_IS_TERMINAL_PAGE (self));
if (!self->destroyed)
gtk_widget_destroy (widget);
g_assert (self->destroyed);
IDE_RETURN (G_SOURCE_REMOVE);
}
static void
ide_terminal_page_spawn_cb (GObject *object,
GAsyncResult *result,
gpointer user_data)
{
IdeTerminalLauncher *launcher = (IdeTerminalLauncher *)object;
g_autoptr(IdeTerminalPage) self = user_data;
g_autoptr(GError) error = NULL;
g_autofree gchar *title = NULL;
gboolean maybe_flapping;
gint64 now;
IDE_ENTRY;
g_assert (IDE_IS_TERMINAL_LAUNCHER (launcher));
g_assert (G_IS_ASYNC_RESULT (result));
g_assert (IDE_IS_TERMINAL_PAGE (self));
self->exited = TRUE;
ide_terminal_launcher_spawn_finish (launcher, result, &error);
if (self->destroyed)
IDE_EXIT;
if (error != NULL)
{
g_autofree gchar *format = NULL;
format = g_strdup_printf ("%s\r\n%s\r\n", _("Failed to launch subprocess. You may need to rebuild your project."), error->message);
ide_terminal_page_feed (self, format);
}
title = g_strdup_printf ("%s (%s)",
ide_page_get_title (IDE_PAGE (self)) ?: _("Untitled terminal"),
/* translators: exited describes that the terminal shell process has exited */
_("Exited"));
ide_page_set_title (IDE_PAGE (self), title);
now = g_get_monotonic_time ();
maybe_flapping = ABS (now - self->last_respawn) < FLAPPING_DURATION_USEC;
if (!self->respawn_on_exit)
{
if (self->close_on_exit && !maybe_flapping)
gdk_threads_add_idle_full (G_PRIORITY_LOW + 1000,
(GSourceFunc) destroy_widget_in_idle,
g_object_ref (self),
g_object_unref);
else
vte_terminal_set_input_enabled (VTE_TERMINAL (self->terminal_top), FALSE);
IDE_EXIT;
}
if (maybe_flapping)
{
ide_terminal_page_feed (self, _("Subprocess launcher failed too quickly, will not respawn."));
ide_terminal_page_feed (self, "\r\n");
IDE_EXIT;
}
g_clear_object (&self->pty);
vte_terminal_reset (VTE_TERMINAL (self->terminal_top), TRUE, TRUE);
self->pty = vte_pty_new_sync (VTE_PTY_DEFAULT, NULL, NULL);
vte_terminal_set_pty (VTE_TERMINAL (self->terminal_top), self->pty);
/* Spawn our terminal and wait for it to exit */
self->last_respawn = now;
self->exited = FALSE;
ide_page_set_title (IDE_PAGE (self), _("Untitled terminal"));
ide_terminal_launcher_spawn_async (self->launcher,
self->pty,
NULL,
ide_terminal_page_spawn_cb,
g_object_ref (self));
IDE_EXIT;
}
static gboolean
ide_terminal_page_do_spawn_in_idle (IdeTerminalPage *self)
{
IDE_ENTRY;
g_assert (IDE_IS_TERMINAL_PAGE (self));
if (self->destroyed)
IDE_RETURN (G_SOURCE_REMOVE);
self->last_respawn = g_get_monotonic_time ();
if (self->pty == NULL)
{
g_autoptr(GError) error = NULL;
if (!(self->pty = vte_pty_new_sync (VTE_PTY_DEFAULT, NULL, &error)))
{
g_critical ("Failed to create PTY for terminal: %s", error->message);
IDE_RETURN (G_SOURCE_REMOVE);
}
}
vte_terminal_set_pty (VTE_TERMINAL (self->terminal_top), self->pty);
if (!self->manage_spawn)
IDE_RETURN (G_SOURCE_REMOVE);
/* Spawn our terminal and wait for it to exit */
ide_terminal_launcher_spawn_async (self->launcher,
self->pty,
NULL,
ide_terminal_page_spawn_cb,
g_object_ref (self));
IDE_RETURN (G_SOURCE_REMOVE);
}
static void
ide_terminal_page_realize (GtkWidget *widget)
{
IdeTerminalPage *self = (IdeTerminalPage *)widget;
g_assert (IDE_IS_TERMINAL_PAGE (self));
GTK_WIDGET_CLASS (ide_terminal_page_parent_class)->realize (widget);
if (self->did_defered_setup_in_realize)
return;
self->did_defered_setup_in_realize = TRUE;
/* We don't want to process this in realize as it could be holding things
* up from being mapped. Instead, wait until the GDK backend has finished
* reacting to realize/etc and then spawn from idle.
*/
g_idle_add_full (G_PRIORITY_LOW,
(GSourceFunc)ide_terminal_page_do_spawn_in_idle,
g_object_ref (self),
g_object_unref);
}
static void
ide_terminal_page_get_preferred_width (GtkWidget *widget,
gint *min_width,
gint *nat_width)
{
/*
* Since we are placing the terminal in a GtkStack, we need
* to fake the size a bit. Otherwise, GtkStack tries to keep the
* widget at its natural size (which prevents us from getting
* appropriate size requests.
*/
GTK_WIDGET_CLASS (ide_terminal_page_parent_class)->get_preferred_width (widget, min_width, nat_width);
*nat_width = *min_width;
}
static void
ide_terminal_page_get_preferred_height (GtkWidget *widget,
gint *min_height,
gint *nat_height)
{
/*
* Since we are placing the terminal in a GtkStack, we need
* to fake the size a bit. Otherwise, GtkStack tries to keep the
* widget at its natural size (which prevents us from getting
* appropriate size requests.
*/
GTK_WIDGET_CLASS (ide_terminal_page_parent_class)->get_preferred_height (widget, min_height, nat_height);
*nat_height = *min_height;
}
static void
ide_terminal_page_set_needs_attention (IdeTerminalPage *self,
gboolean needs_attention)
{
GtkWidget *parent;
g_assert (IDE_IS_TERMINAL_PAGE (self));
parent = gtk_widget_get_parent (GTK_WIDGET (self));
if (GTK_IS_STACK (parent) &&
!gtk_widget_in_destruction (GTK_WIDGET (self)) &&
!gtk_widget_in_destruction (parent))
{
if (!gtk_widget_in_destruction (GTK_WIDGET (self->terminal_top)))
self->needs_attention = !!needs_attention;
gtk_container_child_set (GTK_CONTAINER (parent), GTK_WIDGET (self),
"needs-attention", needs_attention,
NULL);
}
}
static void
notification_received_cb (VteTerminal *terminal,
const gchar *summary,
const gchar *body,
IdeTerminalPage *self)
{
g_assert (VTE_IS_TERMINAL (terminal));
g_assert (IDE_IS_TERMINAL_PAGE (self));
if (self->destroyed)
return;
if (!gtk_widget_has_focus (GTK_WIDGET (terminal)))
ide_terminal_page_set_needs_attention (self, TRUE);
}
static gboolean
focus_in_event_cb (VteTerminal *terminal,
GdkEvent *event,
IdeTerminalPage *self)
{
g_assert (VTE_IS_TERMINAL (terminal));
g_assert (IDE_IS_TERMINAL_PAGE (self));
self->needs_attention = FALSE;
ide_terminal_page_set_needs_attention (self, FALSE);
gtk_revealer_set_reveal_child (self->search_revealer_top, FALSE);
return GDK_EVENT_PROPAGATE;
}
static void
window_title_changed_cb (VteTerminal *terminal,
IdeTerminalPage *self)
{
const gchar *title;
g_assert (VTE_IS_TERMINAL (terminal));
g_assert (IDE_IS_TERMINAL_PAGE (self));
if (self->destroyed)
return;
title = vte_terminal_get_window_title (VTE_TERMINAL (self->terminal_top));
if (title == NULL || title[0] == '\0')
title = _("Untitled terminal");
ide_page_set_title (IDE_PAGE (self), title);
}
static void
style_context_changed (GtkStyleContext *style_context,
IdeTerminalPage *self)
{
GtkStateFlags state;
GdkRGBA fg;
GdkRGBA bg;
g_assert (GTK_IS_STYLE_CONTEXT (style_context));
g_assert (IDE_IS_TERMINAL_PAGE (self));
state = gtk_style_context_get_state (style_context);
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
gtk_style_context_get_color (style_context, state, &fg);
gtk_style_context_get_background_color (style_context, state, &bg);
G_GNUC_END_IGNORE_DEPRECATIONS;
if (bg.alpha == 0.0)
gdk_rgba_parse (&bg, "#f6f7f8");
ide_page_set_primary_color_fg (IDE_PAGE (self), &fg);
ide_page_set_primary_color_bg (IDE_PAGE (self), &bg);
}
static IdePage *
ide_terminal_page_create_split (IdePage *page)
{
IdeTerminalPage *self = (IdeTerminalPage *)page;
g_assert (IDE_IS_TERMINAL_PAGE (self));
return g_object_new (IDE_TYPE_TERMINAL_PAGE,
"close-on-exit", self->close_on_exit,
"launcher", self->launcher,
"manage-spawn", self->manage_spawn,
"pty", NULL,
"respawn-on-exit", self->respawn_on_exit,
"visible", TRUE,
NULL);
}
static void
ide_terminal_page_grab_focus (GtkWidget *widget)
{
IdeTerminalPage *self = (IdeTerminalPage *)widget;
g_assert (IDE_IS_TERMINAL_PAGE (self));
gtk_widget_grab_focus (GTK_WIDGET (self->terminal_top));
}
static void
ide_terminal_page_connect_terminal (IdeTerminalPage *self,
VteTerminal *terminal)
{
g_assert (IDE_IS_TERMINAL_PAGE (self));
g_assert (VTE_IS_TERMINAL (terminal));
if (self->destroyed)
return;
g_signal_connect_object (terminal,
"focus-in-event",
G_CALLBACK (focus_in_event_cb),
self,
0);
g_signal_connect_object (terminal,
"window-title-changed",
G_CALLBACK (window_title_changed_cb),
self,
0);
if (terminal_has_notification_signal ())
{
g_signal_connect_object (terminal,
"notification-received",
G_CALLBACK (notification_received_cb),
self,
0);
}
}
static void
ide_terminal_page_context_set (GtkWidget *widget,
IdeContext *context)
{
IdeTerminalPage *self = (IdeTerminalPage *)widget;
g_assert (IDE_IS_TERMINAL_PAGE (self));
g_assert (!context || IDE_IS_CONTEXT (context));
if (self->launcher == NULL && context != NULL)
self->launcher = ide_terminal_launcher_new (context);
}
static void
ide_terminal_page_on_text_inserted_cb (IdeTerminalPage *self,
VteTerminal *terminal)
{
g_assert (IDE_IS_TERMINAL_PAGE (self));
g_assert (VTE_IS_TERMINAL (terminal));
g_signal_emit (self, signals [TEXT_INSERTED], 0);
}
static GFile *
ide_terminal_page_get_file_or_directory (IdePage *page)
{
IdeTerminalPage *self = (IdeTerminalPage *)page;
const char *uri;
g_assert (IDE_IS_TERMINAL_PAGE (self));
if (self->destroyed)
return NULL;
if (!(uri = vte_terminal_get_current_file_uri (VTE_TERMINAL (self->terminal_top))))
uri = vte_terminal_get_current_directory_uri (VTE_TERMINAL (self->terminal_top));
if (uri != NULL)
return g_file_new_for_uri (uri);
return NULL;
}
static void
ide_terminal_page_destroy (GtkWidget *widget)
{
IdeTerminalPage *self = (IdeTerminalPage *)widget;
self->destroyed = TRUE;
GTK_WIDGET_CLASS (ide_terminal_page_parent_class)->destroy (widget);
}
static void
ide_terminal_page_finalize (GObject *object)
{
IdeTerminalPage *self = IDE_TERMINAL_PAGE (object);
g_clear_object (&self->launcher);
g_clear_object (&self->save_as_file_top);
g_clear_pointer (&self->selection_buffer, g_free);
g_clear_object (&self->pty);
G_OBJECT_CLASS (ide_terminal_page_parent_class)->finalize (object);
}
static void
ide_terminal_page_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
IdeTerminalPage *self = IDE_TERMINAL_PAGE (object);
switch (prop_id)
{
case PROP_CLOSE_ON_EXIT:
g_value_set_boolean (value, self->close_on_exit);
break;
case PROP_LAUNCHER:
g_value_set_object (value, self->launcher);
break;
case PROP_MANAGE_SPAWN:
g_value_set_boolean (value, self->manage_spawn);
break;
case PROP_PTY:
g_value_set_object (value, self->pty);
break;
case PROP_RESPAWN_ON_EXIT:
g_value_set_boolean (value, self->respawn_on_exit);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
static void
ide_terminal_page_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
IdeTerminalPage *self = IDE_TERMINAL_PAGE (object);
switch (prop_id)
{
case PROP_CLOSE_ON_EXIT:
self->close_on_exit = g_value_get_boolean (value);
break;
case PROP_MANAGE_SPAWN:
self->manage_spawn = g_value_get_boolean (value);
break;
case PROP_PTY:
self->pty = g_value_dup_object (value);
break;
case PROP_RESPAWN_ON_EXIT:
self->respawn_on_exit = g_value_get_boolean (value);
break;
case PROP_LAUNCHER:
ide_terminal_page_set_launcher (self, g_value_get_object (value));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
static void
ide_terminal_page_class_init (IdeTerminalPageClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
IdePageClass *page_class = IDE_PAGE_CLASS (klass);
object_class->finalize = ide_terminal_page_finalize;
object_class->get_property = ide_terminal_page_get_property;
object_class->set_property = ide_terminal_page_set_property;
widget_class->realize = ide_terminal_page_realize;
widget_class->get_preferred_width = ide_terminal_page_get_preferred_width;
widget_class->get_preferred_height = ide_terminal_page_get_preferred_height;
widget_class->grab_focus = ide_terminal_page_grab_focus;
widget_class->destroy = ide_terminal_page_destroy;
page_class->create_split = ide_terminal_page_create_split;
page_class->get_file_or_directory = ide_terminal_page_get_file_or_directory;
gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/libide-terminal/ui/ide-terminal-page.ui");
gtk_widget_class_bind_template_child (widget_class, IdeTerminalPage, terminal_top);
gtk_widget_class_bind_template_child (widget_class, IdeTerminalPage, terminal_overlay_top);
properties [PROP_CLOSE_ON_EXIT] =
g_param_spec_boolean ("close-on-exit",
"Close on Exit",
"Close on Exit",
TRUE,
(G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
properties [PROP_MANAGE_SPAWN] =
g_param_spec_boolean ("manage-spawn",
"Manage Spawn",
"Manage Spawn",
TRUE,
(G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
properties [PROP_RESPAWN_ON_EXIT] =
g_param_spec_boolean ("respawn-on-exit",
"Respawn on Exit",
"Respawn on Exit",
TRUE,
(G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
properties [PROP_PTY] =
g_param_spec_object ("pty",
"Pty",
"The pseudo terminal to use",
VTE_TYPE_PTY,
(G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
properties [PROP_LAUNCHER] =
g_param_spec_object ("launcher",
"Launcher",
"The launcher to use for spawning",
IDE_TYPE_TERMINAL_LAUNCHER,
(G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
g_object_class_install_properties (object_class, N_PROPS, properties);
signals [TEXT_INSERTED] =
g_signal_new ("text-inserted",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_LAST,
0,
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
g_signal_set_va_marshaller (signals [TEXT_INSERTED],
G_TYPE_FROM_CLASS (klass),
g_cclosure_marshal_VOID__VOIDv);
}
static void
ide_terminal_page_init (IdeTerminalPage *self)
{
GtkStyleContext *style_context;
self->close_on_exit = TRUE;
self->respawn_on_exit = TRUE;
self->manage_spawn = TRUE;
self->tsearch = g_object_new (IDE_TYPE_TERMINAL_SEARCH,
"visible", TRUE,
NULL);
self->search_revealer_top = ide_terminal_search_get_revealer (self->tsearch);
gtk_widget_init_template (GTK_WIDGET (self));
g_signal_connect_object (self->terminal_top,
"text-inserted",
G_CALLBACK (ide_terminal_page_on_text_inserted_cb),
self,
G_CONNECT_SWAPPED);
ide_page_set_icon_name (IDE_PAGE (self), "builder-terminal-symbolic");
ide_page_set_can_split (IDE_PAGE (self), TRUE);
ide_page_set_menu_id (IDE_PAGE (self), "ide-terminal-page-document-menu");
gtk_overlay_add_overlay (self->terminal_overlay_top, GTK_WIDGET (self->tsearch));
ide_terminal_page_connect_terminal (self, VTE_TERMINAL (self->terminal_top));
ide_terminal_search_set_terminal (self->tsearch, VTE_TERMINAL (self->terminal_top));
ide_terminal_page_actions_init (self);
style_context = gtk_widget_get_style_context (GTK_WIDGET (self->terminal_top));
gtk_style_context_add_class (style_context, "terminal");
g_signal_connect_object (style_context,
"changed",
G_CALLBACK (style_context_changed),
self,
0);
style_context_changed (style_context, self);
gtk_widget_set_can_focus (GTK_WIDGET (self->terminal_top), TRUE);
ide_widget_set_context_handler (self, ide_terminal_page_context_set);
}
void
ide_terminal_page_set_pty (IdeTerminalPage *self,
VtePty *pty)
{
g_return_if_fail (IDE_IS_TERMINAL_PAGE (self));
g_return_if_fail (VTE_IS_PTY (pty));
if (self->destroyed)
return;
if (g_set_object (&self->pty, pty))
{
vte_terminal_reset (VTE_TERMINAL (self->terminal_top), TRUE, TRUE);
vte_terminal_set_pty (VTE_TERMINAL (self->terminal_top), pty);
}
}
void
ide_terminal_page_feed (IdeTerminalPage *self,
const gchar *message)
{
g_return_if_fail (IDE_IS_TERMINAL_PAGE (self));
g_return_if_fail (self->destroyed == FALSE);
if (self->terminal_top != NULL)
vte_terminal_feed (VTE_TERMINAL (self->terminal_top), message, -1);
}
void
ide_terminal_page_set_launcher (IdeTerminalPage *self,
IdeTerminalLauncher *launcher)
{
g_return_if_fail (IDE_IS_TERMINAL_PAGE (self));
g_return_if_fail (!launcher || IDE_IS_TERMINAL_LAUNCHER (launcher));
g_return_if_fail (self->destroyed == FALSE);
if (g_set_object (&self->launcher, launcher))
{
gboolean can_split;
if (launcher != NULL)
{
const gchar *title = ide_terminal_launcher_get_title (launcher);
ide_page_set_title (IDE_PAGE (self), title);
can_split = ide_terminal_launcher_can_respawn (launcher);
}
else
{
self->manage_spawn = FALSE;
can_split = FALSE;
}
ide_page_set_can_split (IDE_PAGE (self), can_split);
}
}
const gchar *
ide_terminal_page_get_current_directory_uri (IdeTerminalPage *self)
{
g_return_val_if_fail (IDE_IS_TERMINAL_PAGE (self), NULL);
g_return_val_if_fail (self->destroyed == FALSE, NULL);
return vte_terminal_get_current_directory_uri (VTE_TERMINAL (self->terminal_top));
}