747 lines
22 KiB
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));
|
||
|
}
|