/* sysprof-tracks-view.c * * Copyright 2023 Christian Hergert * * 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 . * * SPDX-License-Identifier: GPL-3.0-or-later */ #include "config.h" #include "sysprof-css-private.h" #include "sysprof-track-view.h" #include "sysprof-tracks-view.h" #include "sysprof-time-ruler.h" struct _SysprofTracksView { GtkWidget parent_instance; SysprofSession *session; GtkWidget *top_left; GtkListView *list_view; double motion_x; double motion_y; double drag_start_x; double drag_start_y; double drag_offset_x; double drag_offset_y; guint in_drag_selection : 1; }; enum { PROP_0, PROP_SESSION, N_PROPS }; G_DEFINE_FINAL_TYPE (SysprofTracksView, sysprof_tracks_view, GTK_TYPE_WIDGET) static GParamSpec *properties [N_PROPS]; static void sysprof_tracks_view_motion_enter_cb (SysprofTracksView *self, double x, double y, GtkEventControllerMotion *motion) { g_assert (SYSPROF_IS_TRACKS_VIEW (self)); g_assert (GTK_IS_EVENT_CONTROLLER_MOTION (motion)); self->motion_x = x; self->motion_y = y; gtk_widget_queue_draw (GTK_WIDGET (self)); } static void sysprof_tracks_view_motion_leave_cb (SysprofTracksView *self, GtkEventControllerMotion *motion) { g_assert (SYSPROF_IS_TRACKS_VIEW (self)); g_assert (GTK_IS_EVENT_CONTROLLER_MOTION (motion)); self->motion_x = -1; self->motion_y = -1; gtk_widget_queue_draw (GTK_WIDGET (self)); } static void sysprof_tracks_view_motion_cb (SysprofTracksView *self, double x, double y, GtkEventControllerMotion *motion) { g_assert (SYSPROF_IS_TRACKS_VIEW (self)); g_assert (GTK_IS_EVENT_CONTROLLER_MOTION (motion)); self->motion_x = x; self->motion_y = y; gtk_widget_queue_draw (GTK_WIDGET (self)); } static void sysprof_tracks_view_drag_begin_cb (SysprofTracksView *self, double start_x, double start_y, GtkGestureDrag *drag) { g_assert (SYSPROF_IS_TRACKS_VIEW (self)); g_assert (GTK_IS_GESTURE_DRAG (drag)); self->drag_start_x = start_x; self->drag_start_y = start_y; self->drag_offset_x = 0; self->drag_offset_y = 0; self->in_drag_selection = TRUE; gtk_widget_queue_draw (GTK_WIDGET (self)); } static void sysprof_tracks_view_drag_end_cb (SysprofTracksView *self, double offset_x, double offset_y, GtkGestureDrag *drag) { g_assert (SYSPROF_IS_TRACKS_VIEW (self)); g_assert (GTK_IS_GESTURE_DRAG (drag)); self->drag_offset_x = offset_x, self->drag_offset_y = offset_y; self->in_drag_selection = FALSE; gtk_widget_queue_draw (GTK_WIDGET (self)); } static void sysprof_tracks_view_drag_update_cb (SysprofTracksView *self, double offset_x, double offset_y, GtkGestureDrag *drag) { g_assert (SYSPROF_IS_TRACKS_VIEW (self)); g_assert (GTK_IS_GESTURE_DRAG (drag)); self->drag_offset_x = offset_x, self->drag_offset_y = offset_y; gtk_widget_queue_draw (GTK_WIDGET (self)); } static void sysprof_tracks_view_snapshot (GtkWidget *widget, GtkSnapshot *snapshot) { SysprofTracksView *self = (SysprofTracksView *)widget; GdkRGBA shadow_color; GdkRGBA line_color; GdkRGBA color; g_assert (SYSPROF_IS_TRACKS_VIEW (self)); g_assert (GTK_IS_SNAPSHOT (snapshot)); GTK_WIDGET_CLASS (sysprof_tracks_view_parent_class)->snapshot (widget, snapshot); if (self->motion_x == -1 && self->motion_y == -1) return; if (self->motion_x < gtk_widget_get_width (self->top_left)) return; G_GNUC_BEGIN_IGNORE_DEPRECATIONS { GtkStyleContext *style_context = gtk_widget_get_style_context (GTK_WIDGET (self)); gtk_style_context_get_color (style_context, &color); shadow_color = color; shadow_color.alpha *= .1; line_color = color; line_color.alpha *= .5; } G_GNUC_END_IGNORE_DEPRECATIONS if (self->in_drag_selection && self->drag_offset_x != .0) { graphene_rect_t area; graphene_rect_t selection; area = GRAPHENE_RECT_INIT (gtk_widget_get_width (self->top_left), 0, gtk_widget_get_width (GTK_WIDGET (self)) - gtk_widget_get_width (self->top_left), gtk_widget_get_height (GTK_WIDGET (self))); selection = GRAPHENE_RECT_INIT (self->drag_start_x, 0, self->drag_offset_x, gtk_widget_get_height (GTK_WIDGET (self))); graphene_rect_normalize (&selection); graphene_rect_intersection (&area, &selection, &selection); gtk_snapshot_append_color (snapshot, &shadow_color, &GRAPHENE_RECT_INIT (area.origin.x, area.origin.y, selection.origin.x - area.origin.x, area.size.height)); gtk_snapshot_append_color (snapshot, &shadow_color, &GRAPHENE_RECT_INIT (selection.origin.x + selection.size.width, area.origin.y, (area.origin.x + area.size.width) - (selection.origin.x + selection.size.width), area.size.height)); } gtk_snapshot_append_color (snapshot, &line_color, &GRAPHENE_RECT_INIT (self->motion_x, 0, 1, gtk_widget_get_height (GTK_WIDGET (self)))); } static void sysprof_tracks_view_dispose (GObject *object) { SysprofTracksView *self = (SysprofTracksView *)object; GtkWidget *child; g_clear_object (&self->session); gtk_widget_dispose_template (GTK_WIDGET (self), SYSPROF_TYPE_TRACKS_VIEW); while ((child = gtk_widget_get_first_child (GTK_WIDGET (self)))) gtk_widget_unparent (child); G_OBJECT_CLASS (sysprof_tracks_view_parent_class)->dispose (object); } static void sysprof_tracks_view_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { SysprofTracksView *self = SYSPROF_TRACKS_VIEW (object); switch (prop_id) { case PROP_SESSION: g_value_set_object (value, sysprof_tracks_view_get_session (self)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void sysprof_tracks_view_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { SysprofTracksView *self = SYSPROF_TRACKS_VIEW (object); switch (prop_id) { case PROP_SESSION: sysprof_tracks_view_set_session (self, g_value_get_object (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void sysprof_tracks_view_class_init (SysprofTracksViewClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); object_class->dispose = sysprof_tracks_view_dispose; object_class->get_property = sysprof_tracks_view_get_property; object_class->set_property = sysprof_tracks_view_set_property; widget_class->snapshot = sysprof_tracks_view_snapshot; properties[PROP_SESSION] = g_param_spec_object ("session", NULL, NULL, SYSPROF_TYPE_SESSION, (G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS)); g_object_class_install_properties (object_class, N_PROPS, properties); gtk_widget_class_set_template_from_resource (widget_class, "/libsysprof-gtk/sysprof-tracks-view.ui"); gtk_widget_class_set_layout_manager_type (widget_class, GTK_TYPE_BIN_LAYOUT); gtk_widget_class_set_css_name (widget_class, "tracks"); gtk_widget_class_bind_template_child (widget_class, SysprofTracksView, list_view); gtk_widget_class_bind_template_child (widget_class, SysprofTracksView, top_left); gtk_widget_class_bind_template_callback (widget_class, sysprof_tracks_view_motion_enter_cb); gtk_widget_class_bind_template_callback (widget_class, sysprof_tracks_view_motion_leave_cb); gtk_widget_class_bind_template_callback (widget_class, sysprof_tracks_view_motion_cb); gtk_widget_class_bind_template_callback (widget_class, sysprof_tracks_view_drag_begin_cb); gtk_widget_class_bind_template_callback (widget_class, sysprof_tracks_view_drag_end_cb); gtk_widget_class_bind_template_callback (widget_class, sysprof_tracks_view_drag_update_cb); g_type_ensure (SYSPROF_TYPE_TIME_RULER); g_type_ensure (SYSPROF_TYPE_TRACK_VIEW); } static void sysprof_tracks_view_init (SysprofTracksView *self) { _sysprof_css_init (); gtk_widget_init_template (GTK_WIDGET (self)); } GtkWidget * sysprof_tracks_view_new (void) { return g_object_new (SYSPROF_TYPE_TRACKS_VIEW, NULL); } /** * sysprof_tracks_view_get_session: * @self: a #SysprofTracksView * * Gets the session for the tracks. * * Returns: (transfer none) (nullable): a #SysprofSession or %NULL */ SysprofSession * sysprof_tracks_view_get_session (SysprofTracksView *self) { g_return_val_if_fail (SYSPROF_IS_TRACKS_VIEW (self), NULL); return self->session; } static GListModel * sysprof_tracks_view_create_model_func (gpointer item, gpointer user_data) { g_assert (SYSPROF_IS_TRACK (item)); return sysprof_track_list_subtracks (item); } void sysprof_tracks_view_set_session (SysprofTracksView *self, SysprofSession *session) { g_return_if_fail (SYSPROF_IS_TRACKS_VIEW (self)); g_return_if_fail (!session || SYSPROF_IS_SESSION (session)); if (self->session == session) return; if (self->session) { gtk_list_view_set_model (self->list_view, NULL); g_clear_object (&self->session); } if (session) { g_autoptr(GtkTreeListModel) tree_list_model = NULL; g_autoptr(GtkNoSelection) no = NULL; g_autoptr(GListModel) tracks = NULL; self->session = g_object_ref (session); tracks = sysprof_session_list_tracks (session); tree_list_model = gtk_tree_list_model_new (g_object_ref (tracks), FALSE, FALSE, sysprof_tracks_view_create_model_func, self, NULL); no = gtk_no_selection_new (g_object_ref (G_LIST_MODEL (tree_list_model))); gtk_list_view_set_model (self->list_view, GTK_SELECTION_MODEL (no)); } g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_SESSION]); }