/*
* Nautilus
*
* Copyright (C) 1999, 2000, 2004 Red Hat, Inc.
* Copyright (C) 1999, 2000, 2001 Eazel, Inc.
*
* Nautilus 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 2 of the License, or (at your option) any later version.
*
* Nautilus 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 .
*
* Authors: Elliot Lee
* John Sullivan
* Alexander Larsson
*/
/* nautilus-window.c: Implementation of the main window object */
#include "nautilus-window.h"
#include
#include
#include
#include
#include
#include
#include
#include
#include
#ifdef GDK_WINDOWING_WAYLAND
#include
#endif
#ifdef GDK_WINDOWING_X11
#include
#endif
#define DEBUG_FLAG NAUTILUS_DEBUG_WINDOW
#include "nautilus-debug.h"
#include "nautilus-application.h"
#include "nautilus-bookmark-list.h"
#include "nautilus-clipboard.h"
#include "nautilus-dnd.h"
#include "nautilus-enums.h"
#include "nautilus-file-operations.h"
#include "nautilus-file-undo-manager.h"
#include "nautilus-file-utilities.h"
#include "nautilus-global-preferences.h"
#include "nautilus-list-view.h"
#include "nautilus-location-entry.h"
#include "nautilus-metadata.h"
#include "nautilus-mime-actions.h"
#include "nautilus-notebook.h"
#include "nautilus-other-locations-window-slot.h"
#include "nautilus-pathbar.h"
#include "nautilus-profile.h"
#include "nautilus-properties-window.h"
#include "nautilus-signaller.h"
#include "nautilus-toolbar.h"
#include "nautilus-trash-monitor.h"
#include "nautilus-ui-utilities.h"
#include "nautilus-window-slot.h"
/* Forward and back buttons on the mouse */
static gboolean mouse_extra_buttons = TRUE;
static int mouse_forward_button = 9;
static int mouse_back_button = 8;
static void mouse_back_button_changed (gpointer callback_data);
static void mouse_forward_button_changed (gpointer callback_data);
static void use_extra_mouse_buttons_changed (gpointer callback_data);
static void nautilus_window_initialize_actions (NautilusWindow *window);
static GtkWidget *nautilus_window_ensure_location_entry (NautilusWindow *window);
static void close_slot (NautilusWindow *window,
NautilusWindowSlot *slot,
gboolean remove_from_notebook);
/* Sanity check: highest mouse button value I could find was 14. 5 is our
* lower threshold (well-documented to be the one of the button events for the
* scrollwheel), so it's hardcoded in the functions below. However, if you have
* a button that registers higher and want to map it, file a bug and
* we'll move the bar. Makes you wonder why the X guys don't have
* defined values for these like the XKB stuff, huh?
*/
#define UPPER_MOUSE_LIMIT 14
#define NOTIFICATION_TIMEOUT 6 /*s */
struct _NautilusWindow
{
GtkApplicationWindow parent_instance;
GtkWidget *notebook;
/* available slots, and active slot.
* Both of them may never be NULL.
*/
GList *slots;
NautilusWindowSlot *active_slot;
GtkWidget *content_paned;
/* Side Pane */
int side_pane_width;
GtkWidget *sidebar; /* container for the GtkPlacesSidebar */
GtkWidget *places_sidebar; /* the actual GtkPlacesSidebar */
GVolume *selected_volume; /* the selected volume in the sidebar popup callback */
GFile *selected_file; /* the selected file in the sidebar popup callback */
/* Main view */
GtkWidget *main_view;
/* Notifications */
GtkWidget *in_app_notification_undo;
GtkWidget *in_app_notification_undo_label;
GtkWidget *in_app_notification_undo_close_button;
GtkWidget *in_app_notification_undo_undo_button;
guint in_app_notification_undo_timeout_id;
GtkWidget *notification_operation;
GtkWidget *notification_operation_label;
GtkWidget *notification_operation_close;
GtkWidget *notification_operation_open;
guint notification_operation_timeout_id;
GFile *folder_to_open;
/* Toolbar */
GtkWidget *toolbar;
gboolean temporary_navigation_bar;
/* focus widget before the location bar has been shown temporarily */
GtkWidget *last_focus_widget;
/* Handle when exported */
gchar *export_handle;
guint sidebar_width_handler_id;
gulong bookmarks_id;
GQueue *tab_data_queue;
GtkPadController *pad_controller;
GtkGesture *multi_press_gesture;
GtkGesture *notebook_multi_press_gesture;
};
enum
{
SLOT_ADDED,
SLOT_REMOVED,
ACTIVE_SELECTION_CHANGED,
LAST_SIGNAL
};
static guint signals[LAST_SIGNAL] = { 0 };
G_DEFINE_TYPE (NautilusWindow, nautilus_window, GTK_TYPE_APPLICATION_WINDOW);
static const struct
{
unsigned int keyval;
const char *action;
} extra_window_keybindings [] =
{
/* Window actions */
{ GDK_KEY_AddFavorite, "bookmark-current-location" },
{ GDK_KEY_Favorites, "bookmarks" },
{ GDK_KEY_Go, "enter-location" },
{ GDK_KEY_HomePage, "go-home" },
{ GDK_KEY_OpenURL, "enter-location" },
{ GDK_KEY_Refresh, "reload" },
{ GDK_KEY_Reload, "reload" },
{ GDK_KEY_Search, "search" },
{ GDK_KEY_Start, "go-home" },
{ GDK_KEY_Stop, "stop" },
{ GDK_KEY_Back, "back" },
{ GDK_KEY_Forward, "forward" },
};
static const GtkPadActionEntry pad_actions[] =
{
{ GTK_PAD_ACTION_BUTTON, 0, -1, N_("Parent folder"), "up" },
{ GTK_PAD_ACTION_BUTTON, 1, -1, N_("Home"), "go-home" },
{ GTK_PAD_ACTION_BUTTON, 2, -1, N_("New tab"), "new-tab" },
{ GTK_PAD_ACTION_BUTTON, 3, -1, N_("Close current view"), "close-current-view" },
{ GTK_PAD_ACTION_BUTTON, 4, -1, N_("Back"), "back" },
{ GTK_PAD_ACTION_BUTTON, 5, -1, N_("Forward"), "forward" },
};
static void
action_close_current_view (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window;
NautilusWindowSlot *slot;
window = NAUTILUS_WINDOW (user_data);
slot = nautilus_window_get_active_slot (window);
nautilus_window_slot_close (window, slot);
}
static void
action_go_home (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window;
GFile *home;
window = NAUTILUS_WINDOW (user_data);
home = g_file_new_for_path (g_get_home_dir ());
nautilus_window_open_location_full (window, home, nautilus_event_get_window_open_flags (), NULL, NULL);
g_object_unref (home);
}
static void
action_reload (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindowSlot *slot;
slot = nautilus_window_get_active_slot (NAUTILUS_WINDOW (user_data));
nautilus_window_slot_queue_reload (slot);
}
static void
action_stop (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window;
NautilusWindowSlot *slot;
window = NAUTILUS_WINDOW (user_data);
slot = nautilus_window_get_active_slot (window);
nautilus_window_slot_stop_loading (slot);
}
static void
action_up (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindowSlot *slot;
GFile *parent, *location;
slot = nautilus_window_get_active_slot (NAUTILUS_WINDOW (user_data));
location = nautilus_window_slot_get_location (slot);
if (location != NULL)
{
parent = g_file_get_parent (location);
if (parent != NULL)
{
nautilus_window_open_location_full (NAUTILUS_WINDOW (user_data),
parent,
nautilus_event_get_window_open_flags (),
NULL, NULL);
}
g_clear_object (&parent);
}
}
static void
action_back (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
nautilus_window_back_or_forward (NAUTILUS_WINDOW (user_data),
TRUE, 0, nautilus_event_get_window_open_flags ());
}
static void
action_forward (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
nautilus_window_back_or_forward (NAUTILUS_WINDOW (user_data),
FALSE, 0, nautilus_event_get_window_open_flags ());
}
static void
action_bookmark_current_location (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window = user_data;
NautilusApplication *app = NAUTILUS_APPLICATION (g_application_get_default ());
NautilusWindowSlot *slot;
slot = nautilus_window_get_active_slot (window);
nautilus_bookmark_list_append (nautilus_application_get_bookmarks (app),
nautilus_window_slot_get_bookmark (slot));
}
static void
action_new_tab (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
nautilus_window_new_tab (NAUTILUS_WINDOW (user_data));
}
static void
action_enter_location (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window = user_data;
nautilus_window_ensure_location_entry (window);
}
static void
action_tab_previous (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window = user_data;
nautilus_notebook_prev_page (NAUTILUS_NOTEBOOK (window->notebook));
}
static void
action_tab_next (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window = user_data;
nautilus_notebook_next_page (NAUTILUS_NOTEBOOK (window->notebook));
}
static void
action_tab_move_left (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window = user_data;
nautilus_notebook_reorder_current_child_relative (NAUTILUS_NOTEBOOK (window->notebook), -1);
}
static void
action_tab_move_right (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window = user_data;
nautilus_notebook_reorder_current_child_relative (NAUTILUS_NOTEBOOK (window->notebook), 1);
}
static void
action_go_to_tab (GSimpleAction *action,
GVariant *value,
gpointer user_data)
{
NautilusWindow *window = NAUTILUS_WINDOW (user_data);
GtkNotebook *notebook;
gint16 num;
notebook = GTK_NOTEBOOK (window->notebook);
num = g_variant_get_int32 (value);
if (num < gtk_notebook_get_n_pages (notebook))
{
gtk_notebook_set_current_page (notebook, num);
}
}
static void
action_prompt_for_location_root (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window = user_data;
GFile *location;
GtkWidget *entry;
location = g_file_new_for_path ("/");
entry = nautilus_window_ensure_location_entry (window);
nautilus_location_entry_set_location (NAUTILUS_LOCATION_ENTRY (entry), location);
g_object_unref (location);
}
static void
action_prompt_for_location_home (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
GtkWidget *entry;
entry = nautilus_window_ensure_location_entry (NAUTILUS_WINDOW (user_data));
nautilus_location_entry_set_special_text (NAUTILUS_LOCATION_ENTRY (entry),
"~");
gtk_editable_set_position (GTK_EDITABLE (entry), -1);
}
static void
action_redo (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window = user_data;
nautilus_file_undo_manager_redo (GTK_WINDOW (window), NULL);
}
static void
action_undo (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window = user_data;
nautilus_file_undo_manager_undo (GTK_WINDOW (window), NULL);
}
static void
action_toggle_state_view_button (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
GVariant *current_state;
current_state = g_action_get_state (G_ACTION (action));
g_action_change_state (G_ACTION (action),
g_variant_new_boolean (!g_variant_get_boolean (current_state)));
g_variant_unref (current_state);
}
static void
on_location_changed (NautilusWindow *window)
{
gtk_places_sidebar_set_location (GTK_PLACES_SIDEBAR (window->places_sidebar),
nautilus_window_slot_get_location (nautilus_window_get_active_slot (window)));
}
static void
on_slot_location_changed (NautilusWindowSlot *slot,
GParamSpec *pspec,
NautilusWindow *window)
{
if (nautilus_window_get_active_slot (window) == slot)
{
on_location_changed (window);
}
}
static void
on_slot_selection_changed (NautilusWindowSlot *slot,
GParamSpec *pspec,
NautilusWindow *window)
{
if (nautilus_window_get_active_slot (window) == slot)
{
g_signal_emit (window, signals[ACTIVE_SELECTION_CHANGED], 0);
}
}
static void
notebook_switch_page_cb (GtkNotebook *notebook,
GtkWidget *page,
unsigned int page_num,
NautilusWindow *window)
{
NautilusWindowSlot *slot;
GtkWidget *widget;
widget = gtk_notebook_get_nth_page (GTK_NOTEBOOK (window->notebook), page_num);
g_assert (widget != NULL);
/* find slot corresponding to the target page */
slot = NAUTILUS_WINDOW_SLOT (widget);
g_assert (slot != NULL);
nautilus_window_set_active_slot (nautilus_window_slot_get_window (slot),
slot);
}
static void
connect_slot (NautilusWindow *window,
NautilusWindowSlot *slot)
{
g_signal_connect (slot, "notify::location",
G_CALLBACK (on_slot_location_changed), window);
g_signal_connect (slot, "notify::selection",
G_CALLBACK (on_slot_selection_changed), window);
}
static void
disconnect_slot (NautilusWindow *window,
NautilusWindowSlot *slot)
{
g_signal_handlers_disconnect_by_data (slot, window);
}
static NautilusWindowSlot *
nautilus_window_create_slot (NautilusWindow *window,
GFile *location)
{
NautilusFile *file = NULL;
NautilusWindowSlot *slot;
if (location)
{
file = nautilus_file_get (location);
}
/* If not file, assume we open the home directory. We will switch eventually
* to a different location if not.
*/
if (file && nautilus_file_is_other_locations (file))
{
slot = NAUTILUS_WINDOW_SLOT (nautilus_other_locations_window_slot_new (window));
}
else
{
slot = nautilus_window_slot_new (window);
}
nautilus_file_unref (file);
return slot;
}
static NautilusWindowSlot *
nautilus_window_create_and_init_slot (NautilusWindow *window,
GFile *location,
NautilusWindowOpenFlags flags)
{
NautilusWindowSlot *slot;
slot = nautilus_window_create_slot (window, location);
nautilus_window_initialize_slot (window, slot, flags);
return slot;
}
static NautilusWindowSlot *
replace_active_slot (NautilusWindow *window,
GFile *location,
NautilusWindowOpenFlags flags)
{
NautilusWindowSlot *new_slot;
NautilusWindowSlot *active_slot;
new_slot = nautilus_window_create_and_init_slot (window, location, flags);
active_slot = nautilus_window_get_active_slot (window);
if (active_slot)
{
close_slot (window, active_slot, TRUE);
}
return new_slot;
}
void
nautilus_window_initialize_slot (NautilusWindow *window,
NautilusWindowSlot *slot,
NautilusWindowOpenFlags flags)
{
g_assert (NAUTILUS_IS_WINDOW (window));
g_assert (NAUTILUS_IS_WINDOW_SLOT (slot));
connect_slot (window, slot);
g_signal_handlers_block_by_func (window->notebook,
G_CALLBACK (notebook_switch_page_cb),
window);
nautilus_notebook_add_tab (NAUTILUS_NOTEBOOK (window->notebook),
slot,
(flags & NAUTILUS_WINDOW_OPEN_SLOT_APPEND) != 0 ?
-1 :
gtk_notebook_get_current_page (GTK_NOTEBOOK (window->notebook)) + 1,
FALSE);
g_signal_handlers_unblock_by_func (window->notebook,
G_CALLBACK (notebook_switch_page_cb),
window);
window->slots = g_list_append (window->slots, slot);
g_signal_emit (window, signals[SLOT_ADDED], 0, slot);
}
void
nautilus_window_open_location_full (NautilusWindow *window,
GFile *location,
NautilusWindowOpenFlags flags,
GList *selection,
NautilusWindowSlot *target_slot)
{
NautilusWindowSlot *active_slot;
gboolean new_tab_at_end;
NautilusNavigationState *navigation_state = NULL;
/* The location owner can be one of the slots requesting to handle an
* unhandled location. But this slot can be destroyed when switching to
* a new slot. So keep the location alive.
*/
g_object_ref (location);
/* Assert that we are not managing new windows */
g_assert (!(flags & NAUTILUS_WINDOW_OPEN_FLAG_NEW_WINDOW));
/* if the flags say we want a new tab, open a slot in the current window */
if ((flags & NAUTILUS_WINDOW_OPEN_FLAG_NEW_TAB) != 0)
{
new_tab_at_end = g_settings_get_enum (nautilus_preferences, NAUTILUS_PREFERENCES_NEW_TAB_POSITION) == NAUTILUS_NEW_TAB_POSITION_END;
if (new_tab_at_end)
{
flags |= NAUTILUS_WINDOW_OPEN_SLOT_APPEND;
}
}
active_slot = nautilus_window_get_active_slot (window);
if (!target_slot)
{
target_slot = active_slot;
}
if (target_slot == NULL || (flags & NAUTILUS_WINDOW_OPEN_FLAG_NEW_TAB) != 0)
{
target_slot = nautilus_window_create_and_init_slot (window, location, flags);
}
else if (!nautilus_window_slot_handles_location (target_slot, location))
{
navigation_state = nautilus_window_slot_get_navigation_state (active_slot);
target_slot = replace_active_slot (window, location, flags);
}
/* Make the opened location the one active if we weren't ask for the
* oposite, since it's the most usual use case */
if (!(flags & NAUTILUS_WINDOW_OPEN_FLAG_DONT_MAKE_ACTIVE))
{
gtk_window_present (GTK_WINDOW (window));
nautilus_window_set_active_slot (window, target_slot);
}
if (navigation_state != NULL)
{
nautilus_window_slot_open_location_set_navigation_state (target_slot,
location, flags, selection,
NAUTILUS_LOCATION_CHANGE_STANDARD,
navigation_state, 0);
free_navigation_state (navigation_state);
}
else
{
nautilus_window_slot_open_location_full (target_slot, location, flags, selection);
}
g_object_unref (location);
}
static void
unset_focus_widget (NautilusWindow *window)
{
if (window->last_focus_widget != NULL)
{
g_object_remove_weak_pointer (G_OBJECT (window->last_focus_widget),
(gpointer *) &window->last_focus_widget);
window->last_focus_widget = NULL;
}
}
static void
remember_focus_widget (NautilusWindow *window)
{
GtkWidget *focus_widget;
focus_widget = gtk_window_get_focus (GTK_WINDOW (window));
if (focus_widget != NULL)
{
unset_focus_widget (window);
window->last_focus_widget = focus_widget;
g_object_add_weak_pointer (G_OBJECT (focus_widget),
(gpointer *) &(window->last_focus_widget));
}
}
static void
nautilus_window_grab_focus (GtkWidget *widget)
{
NautilusWindowSlot *slot;
slot = nautilus_window_get_active_slot (NAUTILUS_WINDOW (widget));
GTK_WIDGET_CLASS (nautilus_window_parent_class)->grab_focus (widget);
if (slot)
{
gtk_widget_grab_focus (GTK_WIDGET (slot));
}
}
static void
restore_focus_widget (NautilusWindow *window)
{
if (window->last_focus_widget != NULL)
{
gtk_widget_grab_focus (window->last_focus_widget);
unset_focus_widget (window);
}
}
static void
location_entry_cancel_callback (GtkWidget *widget,
NautilusWindow *window)
{
nautilus_toolbar_set_show_location_entry (NAUTILUS_TOOLBAR (window->toolbar), FALSE);
restore_focus_widget (window);
}
static void
location_entry_location_changed_callback (GtkWidget *widget,
GFile *location,
NautilusWindow *window)
{
nautilus_toolbar_set_show_location_entry (NAUTILUS_TOOLBAR (window->toolbar), FALSE);
restore_focus_widget (window);
nautilus_window_open_location_full (window, location, 0, NULL, NULL);
}
static void
close_slot (NautilusWindow *window,
NautilusWindowSlot *slot,
gboolean remove_from_notebook)
{
int page_num;
GtkNotebook *notebook;
g_assert (NAUTILUS_IS_WINDOW_SLOT (slot));
DEBUG ("Closing slot %p", slot);
disconnect_slot (window, slot);
window->slots = g_list_remove (window->slots, slot);
g_signal_emit (window, signals[SLOT_REMOVED], 0, slot);
notebook = GTK_NOTEBOOK (window->notebook);
if (remove_from_notebook)
{
page_num = gtk_notebook_page_num (notebook, GTK_WIDGET (slot));
g_assert (page_num >= 0);
/* this will call gtk_widget_destroy on the slot */
gtk_notebook_remove_page (notebook, page_num);
}
}
void
nautilus_window_new_tab (NautilusWindow *window)
{
NautilusWindowSlot *current_slot;
GFile *location;
g_autofree gchar *uri = NULL;
current_slot = nautilus_window_get_active_slot (window);
location = nautilus_window_slot_get_location (current_slot);
if (location != NULL)
{
uri = g_file_get_uri (location);
if (eel_uri_is_search (uri))
{
location = g_file_new_for_path (g_get_home_dir ());
}
else
{
g_object_ref (location);
}
nautilus_window_open_location_full (window, location,
NAUTILUS_WINDOW_OPEN_FLAG_NEW_TAB,
NULL, NULL);
g_object_unref (location);
}
}
static void
update_cursor (NautilusWindow *window)
{
NautilusWindowSlot *slot;
slot = nautilus_window_get_active_slot (window);
if (slot != NULL &&
nautilus_window_slot_get_allow_stop (slot))
{
GdkDisplay *display;
g_autoptr (GdkCursor) cursor = NULL;
display = gtk_widget_get_display (GTK_WIDGET (window));
cursor = gdk_cursor_new_from_name (display, "progress");
gdk_window_set_cursor (gtk_widget_get_window (GTK_WIDGET (window)), cursor);
}
else
{
gdk_window_set_cursor (gtk_widget_get_window (GTK_WIDGET (window)), NULL);
}
}
void
nautilus_window_reset_menus (NautilusWindow *window)
{
nautilus_window_sync_allow_stop (window, nautilus_window_get_active_slot (window));
}
void
nautilus_window_sync_allow_stop (NautilusWindow *window,
NautilusWindowSlot *slot)
{
GAction *stop_action;
GAction *reload_action;
gboolean allow_stop, slot_is_active, slot_allow_stop;
stop_action = g_action_map_lookup_action (G_ACTION_MAP (window),
"stop");
reload_action = g_action_map_lookup_action (G_ACTION_MAP (window),
"reload");
allow_stop = g_action_get_enabled (stop_action);
slot_allow_stop = nautilus_window_slot_get_allow_stop (slot);
slot_is_active = (slot == nautilus_window_get_active_slot (window));
if (!slot_is_active ||
allow_stop != slot_allow_stop)
{
if (slot_is_active)
{
g_simple_action_set_enabled (G_SIMPLE_ACTION (stop_action), slot_allow_stop);
g_simple_action_set_enabled (G_SIMPLE_ACTION (reload_action), !slot_allow_stop);
}
if (gtk_widget_get_realized (GTK_WIDGET (window)))
{
update_cursor (window);
}
/* Avoid updating the notebook if we are calling on dispose or
* on removal of a notebook tab */
if (nautilus_notebook_contains_slot (NAUTILUS_NOTEBOOK (window->notebook), slot))
{
nautilus_notebook_sync_loading (NAUTILUS_NOTEBOOK (window->notebook), slot);
}
}
}
GtkWidget *
nautilus_window_get_notebook (NautilusWindow *window)
{
g_return_val_if_fail (NAUTILUS_IS_WINDOW (window), NULL);
return window->notebook;
}
static gboolean
save_sidebar_width_cb (gpointer user_data)
{
NautilusWindow *window = user_data;
window->sidebar_width_handler_id = 0;
DEBUG ("Saving sidebar width: %d", window->side_pane_width);
g_settings_set_int (nautilus_window_state,
NAUTILUS_WINDOW_STATE_SIDEBAR_WIDTH,
window->side_pane_width);
return FALSE;
}
/* side pane helpers */
static void
side_pane_size_allocate_callback (GtkWidget *widget,
GtkAllocation *allocation,
gpointer user_data)
{
NautilusWindow *window = user_data;
if (window->sidebar_width_handler_id != 0)
{
g_source_remove (window->sidebar_width_handler_id);
window->sidebar_width_handler_id = 0;
}
if (allocation->width != window->side_pane_width &&
allocation->width > 1)
{
window->side_pane_width = allocation->width;
window->sidebar_width_handler_id =
g_idle_add (save_sidebar_width_cb, window);
}
}
static void
setup_side_pane_width (NautilusWindow *window)
{
g_return_if_fail (window->sidebar != NULL);
window->side_pane_width =
g_settings_get_int (nautilus_window_state,
NAUTILUS_WINDOW_STATE_SIDEBAR_WIDTH);
gtk_paned_set_position (GTK_PANED (window->content_paned),
window->side_pane_width);
}
/* Callback used when the places sidebar changes location; we need to change the displayed folder */
static void
open_location_cb (NautilusWindow *window,
GFile *location,
GtkPlacesOpenFlags open_flags)
{
NautilusWindowOpenFlags flags;
NautilusApplication *application;
switch (open_flags)
{
case GTK_PLACES_OPEN_NEW_TAB:
{
flags = NAUTILUS_WINDOW_OPEN_FLAG_NEW_TAB |
NAUTILUS_WINDOW_OPEN_FLAG_DONT_MAKE_ACTIVE;
}
break;
case GTK_PLACES_OPEN_NEW_WINDOW:
{
flags = NAUTILUS_WINDOW_OPEN_FLAG_NEW_WINDOW;
}
break;
case GTK_PLACES_OPEN_NORMAL: /* fall-through */
default:
{
flags = 0;
}
break;
}
application = NAUTILUS_APPLICATION (g_application_get_default ());
/* FIXME: We shouldn't need to provide the window, but seems gtk_application_get_active_window
* is not working properly in GtkApplication, so we cannot rely on that...
*/
nautilus_application_open_location_full (application, location, flags,
NULL, window, NULL);
}
static void
places_sidebar_unmount_operation_cb (NautilusWindow *window,
GMountOperation *mount_operation)
{
g_signal_connect (mount_operation, "show-unmount-progress",
G_CALLBACK (show_unmount_progress_cb), NULL);
g_signal_connect (mount_operation, "aborted",
G_CALLBACK (show_unmount_progress_aborted_cb), NULL);
}
/* Callback used when the places sidebar needs us to present an error message */
static void
places_sidebar_show_error_message_cb (GtkPlacesSidebar *sidebar,
const char *primary,
const char *secondary,
gpointer user_data)
{
NautilusWindow *window = NAUTILUS_WINDOW (user_data);
show_dialog (primary, secondary, GTK_WINDOW (window), GTK_MESSAGE_ERROR);
}
static void
places_sidebar_show_other_locations_with_flags (NautilusWindow *window,
GtkPlacesOpenFlags open_flags)
{
GFile *location;
location = g_file_new_for_uri ("other-locations:///");
open_location_cb (window, location, open_flags);
g_object_unref (location);
}
static void
places_sidebar_show_starred_location (NautilusWindow *window,
GtkPlacesOpenFlags open_flags)
{
GFile *location;
location = g_file_new_for_uri ("starred:///");
open_location_cb (window, location, open_flags);
g_object_unref (location);
}
static GList *
build_selection_list_from_gfile_list (GList *gfile_list)
{
GList *result;
GList *l;
result = NULL;
for (l = gfile_list; l; l = l->next)
{
GFile *file;
NautilusDragSelectionItem *item;
file = l->data;
item = nautilus_drag_selection_item_new ();
item->uri = g_file_get_uri (file);
item->file = nautilus_file_get_existing (file);
item->got_icon_position = FALSE;
result = g_list_prepend (result, item);
}
return g_list_reverse (result);
}
void
nautilus_window_start_dnd (NautilusWindow *window,
GdkDragContext *context)
{
g_return_if_fail (NAUTILUS_IS_WINDOW (window));
gtk_places_sidebar_set_drop_targets_visible (GTK_PLACES_SIDEBAR (window->places_sidebar),
TRUE,
context);
}
void
nautilus_window_end_dnd (NautilusWindow *window,
GdkDragContext *context)
{
g_return_if_fail (NAUTILUS_IS_WINDOW (window));
gtk_places_sidebar_set_drop_targets_visible (GTK_PLACES_SIDEBAR (window->places_sidebar),
FALSE,
context);
}
/* Callback used when the places sidebar needs to know the drag action to suggest */
static GdkDragAction
places_sidebar_drag_action_requested_cb (GtkPlacesSidebar *sidebar,
GdkDragContext *context,
GFile *dest_file,
GList *source_file_list,
gpointer user_data)
{
GList *items;
char *uri;
int action = 0;
NautilusDragInfo *info;
guint32 source_actions;
info = nautilus_drag_get_source_data (context);
if (info != NULL)
{
items = info->selection_cache;
source_actions = info->source_actions;
}
else
{
items = build_selection_list_from_gfile_list (source_file_list);
source_actions = 0;
}
uri = g_file_get_uri (dest_file);
if (items == NULL)
{
goto out;
}
nautilus_drag_default_drop_action_for_icons (context, uri, items, source_actions, &action);
out:
if (info == NULL)
{
nautilus_drag_destroy_selection_list (items);
}
g_free (uri);
return action;
}
/* Callback used when the places sidebar needs us to pop up a menu with possible drag actions */
static GdkDragAction
places_sidebar_drag_action_ask_cb (GtkPlacesSidebar *sidebar,
GdkDragAction actions,
gpointer user_data)
{
return nautilus_drag_drop_action_ask (GTK_WIDGET (sidebar), actions);
}
static GList *
build_uri_list_from_gfile_list (GList *file_list)
{
GList *result;
GList *l;
result = NULL;
for (l = file_list; l; l = l->next)
{
GFile *file = l->data;
char *uri;
uri = g_file_get_uri (file);
result = g_list_prepend (result, uri);
}
return g_list_reverse (result);
}
/* Callback used when the places sidebar has URIs dropped into it. We do a normal file operation for them. */
static void
places_sidebar_drag_perform_drop_cb (GtkPlacesSidebar *sidebar,
GFile *dest_file,
GList *source_file_list,
GdkDragAction action,
gpointer user_data)
{
char *dest_uri;
GList *source_uri_list;
dest_uri = g_file_get_uri (dest_file);
source_uri_list = build_uri_list_from_gfile_list (source_file_list);
nautilus_file_operations_copy_move (source_uri_list, dest_uri, action, GTK_WIDGET (sidebar), NULL, NULL, NULL);
g_free (dest_uri);
g_list_free_full (source_uri_list, g_free);
}
/* Callback used in the "empty trash" menu item from the places sidebar */
static void
action_empty_trash (GSimpleAction *action,
GVariant *variant,
gpointer user_data)
{
NautilusWindow *window = NAUTILUS_WINDOW (user_data);
nautilus_file_operations_empty_trash (GTK_WIDGET (window), TRUE, NULL);
}
/* Callback used for the "properties" menu item from the places sidebar */
static void
action_properties (GSimpleAction *action,
GVariant *variant,
gpointer user_data)
{
NautilusWindow *window = NAUTILUS_WINDOW (user_data);
GList *list;
NautilusFile *file;
file = nautilus_file_get (window->selected_file);
list = g_list_append (NULL, file);
nautilus_properties_window_present (list, GTK_WIDGET (window), NULL, NULL,
NULL);
nautilus_file_list_free (list);
g_clear_object (&window->selected_file);
}
static gboolean
check_have_gnome_disks (void)
{
gchar *disks_path;
gboolean res;
disks_path = g_find_program_in_path ("gnome-disks");
res = (disks_path != NULL);
g_free (disks_path);
return res;
}
static gboolean
should_show_format_command (GVolume *volume)
{
gchar *unix_device_id;
gboolean show_format;
unix_device_id = g_volume_get_identifier (volume, G_VOLUME_IDENTIFIER_KIND_UNIX_DEVICE);
show_format = (unix_device_id != NULL) && check_have_gnome_disks ();
g_free (unix_device_id);
return show_format;
}
static void
action_restore_tab (GSimpleAction *action,
GVariant *state,
gpointer user_data)
{
NautilusWindow *window = NAUTILUS_WINDOW (user_data);
NautilusWindowOpenFlags flags;
g_autoptr (GFile) location = NULL;
NautilusWindowSlot *slot;
NautilusNavigationState *data;
if (g_queue_get_length (window->tab_data_queue) == 0)
{
return;
}
flags = NAUTILUS_WINDOW_OPEN_FLAG_NEW_TAB | NAUTILUS_WINDOW_OPEN_FLAG_DONT_MAKE_ACTIVE;
data = g_queue_pop_head (window->tab_data_queue);
location = nautilus_file_get_location (data->file);
slot = nautilus_window_create_and_init_slot (window, location, flags);
nautilus_window_slot_open_location_full (slot, location, flags, NULL);
nautilus_window_slot_restore_navigation_state (slot, data);
free_navigation_state (data);
}
static guint
get_window_xid (NautilusWindow *window)
{
#ifdef GDK_WINDOWING_X11
if (GDK_IS_X11_DISPLAY (gtk_widget_get_display (GTK_WIDGET (window))))
{
GdkWindow *gdk_window = gtk_widget_get_window (GTK_WIDGET (window));
return (guint) gdk_x11_window_get_xid (gdk_window);
}
#endif
return 0;
}
static void
action_format (GSimpleAction *action,
GVariant *variant,
gpointer user_data)
{
NautilusWindow *window = NAUTILUS_WINDOW (user_data);
GAppInfo *app_info;
gchar *cmdline, *device_identifier, *xid_string;
device_identifier = g_volume_get_identifier (window->selected_volume,
G_VOLUME_IDENTIFIER_KIND_UNIX_DEVICE);
xid_string = g_strdup_printf ("%x", get_window_xid (window));
cmdline = g_strconcat ("gnome-disks ",
"--block-device ", device_identifier, " ",
"--format-device ",
"--xid ", xid_string,
NULL);
app_info = g_app_info_create_from_commandline (cmdline, NULL, 0, NULL);
g_app_info_launch (app_info, NULL, NULL, NULL);
g_free (cmdline);
g_free (device_identifier);
g_free (xid_string);
g_clear_object (&app_info);
g_clear_object (&window->selected_volume);
}
static void
add_menu_separator (GtkWidget *menu)
{
GtkWidget *separator;
separator = gtk_separator_new (GTK_ORIENTATION_HORIZONTAL);
gtk_container_add (GTK_CONTAINER (menu), separator);
gtk_widget_show (separator);
}
static void
places_sidebar_populate_popup_cb (GtkPlacesSidebar *sidebar,
GtkWidget *menu,
GFile *selected_file,
GVolume *selected_volume,
gpointer user_data)
{
NautilusWindow *window = NAUTILUS_WINDOW (user_data);
GFile *trash;
GtkWidget *menu_item;
GAction *action;
g_clear_object (&window->selected_file);
g_clear_object (&window->selected_volume);
if (selected_file)
{
trash = g_file_new_for_uri ("trash:///");
if (g_file_equal (trash, selected_file))
{
add_menu_separator (menu);
menu_item = gtk_model_button_new ();
gtk_actionable_set_action_name (GTK_ACTIONABLE (menu_item),
"win.empty-trash");
g_object_set (menu_item, "text", _("Empty _Trash"), NULL);
gtk_container_add (GTK_CONTAINER (menu), menu_item);
gtk_widget_show (menu_item);
action = g_action_map_lookup_action (G_ACTION_MAP (window),
"empty-trash");
g_simple_action_set_enabled (G_SIMPLE_ACTION (action),
!nautilus_trash_monitor_is_empty ());
}
g_object_unref (trash);
if (g_file_is_native (selected_file))
{
window->selected_file = g_object_ref (selected_file);
add_menu_separator (menu);
menu_item = gtk_model_button_new ();
gtk_actionable_set_action_name (GTK_ACTIONABLE (menu_item),
"win.properties");
g_object_set (menu_item, "text", _("_Properties"), NULL);
gtk_container_add (GTK_CONTAINER (menu), menu_item);
gtk_widget_show (menu_item);
}
}
if (selected_volume)
{
if (should_show_format_command (selected_volume))
{
menu_item = gtk_model_button_new ();
gtk_actionable_set_action_name (GTK_ACTIONABLE (menu_item),
"win.format");
g_object_set (menu_item, "text", _("_Format…"), NULL);
if (selected_volume != NULL && G_IS_VOLUME (selected_volume))
{
window->selected_volume = g_object_ref (selected_volume);
}
gtk_container_add (GTK_CONTAINER (menu), menu_item);
gtk_widget_show (menu_item);
action = g_action_map_lookup_action (G_ACTION_MAP (window),
"format");
g_simple_action_set_enabled (G_SIMPLE_ACTION (action),
selected_volume != NULL &&
G_IS_VOLUME (selected_volume));
}
}
}
static void
nautilus_window_set_up_sidebar (NautilusWindow *window)
{
setup_side_pane_width (window);
g_signal_connect (window->sidebar,
"size-allocate",
G_CALLBACK (side_pane_size_allocate_callback),
window);
gtk_places_sidebar_set_open_flags (GTK_PLACES_SIDEBAR (window->places_sidebar),
(GTK_PLACES_OPEN_NORMAL
| GTK_PLACES_OPEN_NEW_TAB
| GTK_PLACES_OPEN_NEW_WINDOW));
g_signal_connect_swapped (window->places_sidebar, "open-location",
G_CALLBACK (open_location_cb), window);
g_signal_connect (window->places_sidebar, "show-error-message",
G_CALLBACK (places_sidebar_show_error_message_cb), window);
g_signal_connect (window->places_sidebar, "drag-action-requested",
G_CALLBACK (places_sidebar_drag_action_requested_cb), window);
g_signal_connect (window->places_sidebar, "drag-action-ask",
G_CALLBACK (places_sidebar_drag_action_ask_cb), window);
g_signal_connect (window->places_sidebar, "drag-perform-drop",
G_CALLBACK (places_sidebar_drag_perform_drop_cb), window);
g_signal_connect (window->places_sidebar, "populate-popup",
G_CALLBACK (places_sidebar_populate_popup_cb), window);
g_signal_connect (window->places_sidebar, "unmount",
G_CALLBACK (places_sidebar_unmount_operation_cb), window);
}
void
nautilus_window_hide_sidebar (NautilusWindow *window)
{
DEBUG ("Called hide_sidebar()");
g_return_if_fail (NAUTILUS_IS_WINDOW (window));
gtk_widget_hide (window->sidebar);
}
void
nautilus_window_show_sidebar (NautilusWindow *window)
{
DEBUG ("Called show_sidebar()");
g_return_if_fail (NAUTILUS_IS_WINDOW (window));
gtk_widget_show (window->sidebar);
setup_side_pane_width (window);
}
void
nautilus_window_slot_close (NautilusWindow *window,
NautilusWindowSlot *slot)
{
NautilusNavigationState *data;
DEBUG ("Requesting to remove slot %p from window %p", slot, window);
if (window == NULL)
{
return;
}
data = nautilus_window_slot_get_navigation_state (slot);
if (data != NULL)
{
g_queue_push_head (window->tab_data_queue, data);
}
close_slot (window, slot, TRUE);
/* If that was the last slot in the window, close the window. */
if (window->slots == NULL)
{
DEBUG ("Last slot removed, closing the window");
nautilus_window_close (window);
}
}
static void
nautilus_window_sync_bookmarks (NautilusWindow *window)
{
gboolean can_bookmark = FALSE;
NautilusWindowSlot *slot;
NautilusBookmarkList *bookmarks;
GAction *action;
GFile *location;
slot = window->active_slot;
location = nautilus_window_slot_get_location (slot);
if (location != NULL)
{
bookmarks = nautilus_application_get_bookmarks
(NAUTILUS_APPLICATION (gtk_window_get_application (GTK_WINDOW (window))));
can_bookmark = nautilus_bookmark_list_can_bookmark_location (bookmarks, location);
}
action = g_action_map_lookup_action (G_ACTION_MAP (window), "bookmark-current-location");
g_simple_action_set_enabled (G_SIMPLE_ACTION (action), can_bookmark);
}
void
nautilus_window_sync_location_widgets (NautilusWindow *window)
{
NautilusWindowSlot *slot;
GFile *location;
GAction *action;
gboolean enabled;
slot = window->active_slot;
location = nautilus_window_slot_get_location (slot);
/* Change the location bar and path bar to match the current location. */
if (location != NULL)
{
GtkWidget *location_entry;
GtkWidget *path_bar;
location_entry = nautilus_toolbar_get_location_entry (NAUTILUS_TOOLBAR (window->toolbar));
nautilus_location_entry_set_location (NAUTILUS_LOCATION_ENTRY (location_entry), location);
path_bar = nautilus_toolbar_get_path_bar (NAUTILUS_TOOLBAR (window->toolbar));
nautilus_path_bar_set_path (NAUTILUS_PATH_BAR (path_bar), location);
}
action = g_action_map_lookup_action (G_ACTION_MAP (window), "back");
enabled = nautilus_window_slot_get_back_history (slot) != NULL;
g_simple_action_set_enabled (G_SIMPLE_ACTION (action), enabled);
action = g_action_map_lookup_action (G_ACTION_MAP (window), "forward");
enabled = nautilus_window_slot_get_forward_history (slot) != NULL;
g_simple_action_set_enabled (G_SIMPLE_ACTION (action), enabled);
nautilus_window_sync_bookmarks (window);
}
static GtkWidget *
nautilus_window_ensure_location_entry (NautilusWindow *window)
{
GtkWidget *location_entry;
remember_focus_widget (window);
nautilus_toolbar_set_show_location_entry (NAUTILUS_TOOLBAR (window->toolbar), TRUE);
location_entry = nautilus_toolbar_get_location_entry (NAUTILUS_TOOLBAR (window->toolbar));
gtk_widget_grab_focus (location_entry);
return location_entry;
}
static void
remove_notifications (NautilusWindow *window)
{
GtkRevealerTransitionType transition_type;
/* Hide it inmediatily so we can animate the new notification. */
transition_type = gtk_revealer_get_transition_type (GTK_REVEALER (window->in_app_notification_undo));
gtk_revealer_set_transition_type (GTK_REVEALER (window->in_app_notification_undo),
GTK_REVEALER_TRANSITION_TYPE_NONE);
gtk_revealer_set_reveal_child (GTK_REVEALER (window->in_app_notification_undo),
FALSE);
gtk_revealer_set_transition_type (GTK_REVEALER (window->in_app_notification_undo),
transition_type);
if (window->in_app_notification_undo_timeout_id != 0)
{
g_source_remove (window->in_app_notification_undo_timeout_id);
window->in_app_notification_undo_timeout_id = 0;
}
transition_type = gtk_revealer_get_transition_type (GTK_REVEALER (window->notification_operation));
gtk_revealer_set_transition_type (GTK_REVEALER (window->notification_operation),
GTK_REVEALER_TRANSITION_TYPE_NONE);
gtk_revealer_set_reveal_child (GTK_REVEALER (window->notification_operation),
FALSE);
gtk_revealer_set_transition_type (GTK_REVEALER (window->notification_operation),
transition_type);
if (window->notification_operation_timeout_id != 0)
{
g_source_remove (window->notification_operation_timeout_id);
window->notification_operation_timeout_id = 0;
}
}
static void
hide_in_app_notification_undo (NautilusWindow *window)
{
if (window->in_app_notification_undo_timeout_id != 0)
{
g_source_remove (window->in_app_notification_undo_timeout_id);
window->in_app_notification_undo_timeout_id = 0;
}
gtk_revealer_set_reveal_child (GTK_REVEALER (window->in_app_notification_undo), FALSE);
}
static void
on_in_app_notification_undo_undo_button_clicked (GtkWidget *notification,
NautilusWindow *window)
{
hide_in_app_notification_undo (window);
nautilus_file_undo_manager_undo (GTK_WINDOW (window), NULL);
}
static void
on_in_app_notification_undo_close_button_clicked (GtkWidget *notification,
NautilusWindow *window)
{
hide_in_app_notification_undo (window);
}
static gboolean
on_in_app_notification_undo_timeout (NautilusWindow *window)
{
hide_in_app_notification_undo (window);
return G_SOURCE_REMOVE;
}
static gchar *
in_app_notification_undo_deleted_get_label (NautilusFileUndoInfo *undo_info)
{
GList *files;
gchar *file_label;
gchar *label;
gint length;
files = nautilus_file_undo_info_trash_get_files (NAUTILUS_FILE_UNDO_INFO_TRASH (undo_info));
length = g_list_length (files);
if (length == 1)
{
file_label = g_file_get_basename (files->data);
/* Translators: only one item has been deleted and %s is its name. */
label = g_markup_printf_escaped (_("“%s” deleted"), file_label);
g_free (file_label);
}
else
{
/* Translators: one or more items might have been deleted, and %d
* is the count. */
label = g_markup_printf_escaped (ngettext ("%d file deleted", "%d files deleted", length), length);
}
return label;
}
static gchar *
in_app_notification_undo_unstar_get_label (NautilusFileUndoInfo *undo_info)
{
GList *files;
gchar *label;
gint length;
files = nautilus_file_undo_info_starred_get_files (NAUTILUS_FILE_UNDO_INFO_STARRED (undo_info));
length = g_list_length (files);
if (length == 1)
{
g_autofree gchar *file_label = NULL;
file_label = nautilus_file_get_display_name (files->data);
/* Translators: one item has been unstarred and %s is its name. */
label = g_markup_printf_escaped (_("“%s” unstarred"), file_label);
}
else
{
/* Translators: one or more items have been unstarred, and %d
* is the count. */
label = g_markup_printf_escaped (ngettext ("%d file unstarred", "%d files unstarred", length), length);
}
return label;
}
static void
nautilus_window_on_undo_changed (NautilusFileUndoManager *manager,
NautilusWindow *window)
{
NautilusFileUndoInfo *undo_info;
NautilusFileUndoManagerState state;
undo_info = nautilus_file_undo_manager_get_action ();
state = nautilus_file_undo_manager_get_state ();
if (undo_info != NULL &&
state == NAUTILUS_FILE_UNDO_MANAGER_STATE_UNDO)
{
gboolean popup_notification = FALSE;
g_autofree gchar *label = NULL;
if (nautilus_file_undo_info_get_op_type (undo_info) == NAUTILUS_FILE_UNDO_OP_MOVE_TO_TRASH)
{
g_autoptr (GList) files = NULL;
files = nautilus_file_undo_info_trash_get_files (NAUTILUS_FILE_UNDO_INFO_TRASH (undo_info));
/* Don't pop up a notification if user canceled the operation or the focus
* is not in the this window. This is an easy way to know from which window
* was the delete operation made */
if (files != NULL && gtk_window_has_toplevel_focus (GTK_WINDOW (window)))
{
popup_notification = TRUE;
label = in_app_notification_undo_deleted_get_label (undo_info);
}
}
else if (nautilus_file_undo_info_get_op_type (undo_info) == NAUTILUS_FILE_UNDO_OP_STARRED)
{
NautilusWindowSlot *active_slot;
GFile *location;
active_slot = nautilus_window_get_active_slot (window);
location = nautilus_window_slot_get_location (active_slot);
/* Don't pop up a notification if the focus is not in the this
* window. This is an easy way to know from which window was the
* unstart operation made */
if (eel_uri_is_starred (g_file_get_uri (location)) &&
gtk_window_has_toplevel_focus (GTK_WINDOW (window)) &&
!nautilus_file_undo_info_starred_is_starred (NAUTILUS_FILE_UNDO_INFO_STARRED (undo_info)))
{
popup_notification = TRUE;
label = in_app_notification_undo_unstar_get_label (undo_info);
}
}
if (popup_notification)
{
remove_notifications (window);
gtk_label_set_markup (GTK_LABEL (window->in_app_notification_undo_label),
label);
gtk_revealer_set_reveal_child (GTK_REVEALER (window->in_app_notification_undo),
TRUE);
window->in_app_notification_undo_timeout_id = g_timeout_add_seconds (NOTIFICATION_TIMEOUT,
(GSourceFunc) on_in_app_notification_undo_timeout,
window);
}
}
else
{
hide_in_app_notification_undo (window);
}
}
static void
hide_notification_operation (NautilusWindow *window)
{
if (window->notification_operation_timeout_id != 0)
{
g_source_remove (window->notification_operation_timeout_id);
window->notification_operation_timeout_id = 0;
}
gtk_revealer_set_reveal_child (GTK_REVEALER (window->notification_operation), FALSE);
g_clear_object (&window->folder_to_open);
}
static void
on_notification_operation_open_clicked (GtkWidget *notification,
NautilusWindow *window)
{
nautilus_window_open_location_full (window, window->folder_to_open,
0, NULL, NULL);
hide_notification_operation (window);
}
static void
on_notification_operation_close_clicked (GtkWidget *notification,
NautilusWindow *window)
{
hide_notification_operation (window);
}
static gboolean
on_notification_operation_timeout (NautilusWindow *window)
{
hide_notification_operation (window);
return FALSE;
}
void
nautilus_window_show_operation_notification (NautilusWindow *window,
gchar *main_label,
GFile *folder_to_open)
{
gchar *button_label;
gchar *folder_name;
NautilusFile *folder;
GFile *current_location;
current_location = nautilus_window_slot_get_location (window->active_slot);
if (gtk_window_has_toplevel_focus (GTK_WINDOW (window)))
{
remove_notifications (window);
gtk_label_set_text (GTK_LABEL (window->notification_operation_label),
main_label);
if (g_file_equal (folder_to_open, current_location))
{
gtk_widget_hide (window->notification_operation_open);
}
else
{
gtk_widget_show (window->notification_operation_open);
window->folder_to_open = g_object_ref (folder_to_open);
folder = nautilus_file_get (folder_to_open);
folder_name = nautilus_file_get_display_name (folder);
button_label = g_strdup_printf (_("Open %s"), folder_name);
gtk_button_set_label (GTK_BUTTON (window->notification_operation_open),
button_label);
nautilus_file_unref (folder);
g_free (folder_name);
g_free (button_label);
}
gtk_revealer_set_reveal_child (GTK_REVEALER (window->notification_operation), TRUE);
window->notification_operation_timeout_id = g_timeout_add_seconds (NOTIFICATION_TIMEOUT,
(GSourceFunc) on_notification_operation_timeout,
window);
}
}
static void
path_bar_location_changed_callback (GtkWidget *widget,
GFile *location,
NautilusWindow *window)
{
nautilus_window_open_location_full (window, location, 0, NULL, NULL);
}
static void
notebook_popup_menu_new_tab_cb (GtkMenuItem *menuitem,
gpointer user_data)
{
NautilusWindow *window = user_data;
nautilus_window_new_tab (window);
}
static void
notebook_popup_menu_move_left_cb (GtkMenuItem *menuitem,
gpointer user_data)
{
NautilusWindow *window = user_data;
nautilus_notebook_reorder_current_child_relative (NAUTILUS_NOTEBOOK (window->notebook), -1);
}
static void
notebook_popup_menu_move_right_cb (GtkMenuItem *menuitem,
gpointer user_data)
{
NautilusWindow *window = user_data;
nautilus_notebook_reorder_current_child_relative (NAUTILUS_NOTEBOOK (window->notebook), 1);
}
static void
notebook_popup_menu_close_cb (GtkMenuItem *menuitem,
gpointer user_data)
{
NautilusWindow *window = user_data;
NautilusWindowSlot *slot;
slot = window->active_slot;
nautilus_window_slot_close (window, slot);
}
static void
notebook_popup_menu_show (NautilusWindow *window,
const GdkEvent *event)
{
GtkWidget *popup;
GtkWidget *item;
gboolean can_move_left, can_move_right;
NautilusNotebook *notebook;
notebook = NAUTILUS_NOTEBOOK (window->notebook);
can_move_left = nautilus_notebook_can_reorder_current_child_relative (notebook, -1);
can_move_right = nautilus_notebook_can_reorder_current_child_relative (notebook, 1);
popup = gtk_menu_new ();
item = gtk_menu_item_new_with_mnemonic (_("_New Tab"));
g_signal_connect (item, "activate",
G_CALLBACK (notebook_popup_menu_new_tab_cb),
window);
gtk_menu_shell_append (GTK_MENU_SHELL (popup),
item);
gtk_menu_shell_append (GTK_MENU_SHELL (popup),
gtk_separator_menu_item_new ());
item = gtk_menu_item_new_with_mnemonic (_("Move Tab _Left"));
g_signal_connect (item, "activate",
G_CALLBACK (notebook_popup_menu_move_left_cb),
window);
gtk_menu_shell_append (GTK_MENU_SHELL (popup),
item);
gtk_widget_set_sensitive (item, can_move_left);
item = gtk_menu_item_new_with_mnemonic (_("Move Tab _Right"));
g_signal_connect (item, "activate",
G_CALLBACK (notebook_popup_menu_move_right_cb),
window);
gtk_menu_shell_append (GTK_MENU_SHELL (popup),
item);
gtk_widget_set_sensitive (item, can_move_right);
gtk_menu_shell_append (GTK_MENU_SHELL (popup),
gtk_separator_menu_item_new ());
item = gtk_menu_item_new_with_mnemonic (_("_Close Tab"));
g_signal_connect (item, "activate",
G_CALLBACK (notebook_popup_menu_close_cb), window);
gtk_menu_shell_append (GTK_MENU_SHELL (popup),
item);
gtk_widget_show_all (popup);
gtk_menu_popup_at_pointer (GTK_MENU (popup), event);
}
/* emitted when the user clicks the "close" button of tabs */
static void
notebook_tab_close_requested (NautilusNotebook *notebook,
NautilusWindowSlot *slot,
NautilusWindow *window)
{
nautilus_window_slot_close (window, slot);
}
static void
notebook_button_press_cb (GtkGestureMultiPress *gesture,
gint n_press,
gdouble x,
gdouble y,
gpointer user_data)
{
NautilusWindow *window;
GdkEventSequence *sequence;
const GdkEvent *event;
window = NAUTILUS_WINDOW (user_data);
if (nautilus_notebook_content_area_hit (NAUTILUS_NOTEBOOK (window->notebook), x, y))
{
return;
}
sequence = gtk_gesture_single_get_current_sequence (GTK_GESTURE_SINGLE (gesture));
event = gtk_gesture_get_last_event (GTK_GESTURE (gesture), sequence);
notebook_popup_menu_show (window, event);
}
static gboolean
notebook_popup_menu_cb (GtkWidget *widget,
gpointer user_data)
{
NautilusWindow *window = user_data;
notebook_popup_menu_show (window, NULL);
return TRUE;
}
GtkWidget *
nautilus_window_get_toolbar (NautilusWindow *window)
{
g_return_val_if_fail (NAUTILUS_IS_WINDOW (window), NULL);
return window->toolbar;
}
static void
setup_toolbar (NautilusWindow *window)
{
GtkWidget *path_bar;
GtkWidget *location_entry;
g_object_set (window->toolbar, "window", window, NULL);
/* connect to the pathbar signals */
path_bar = nautilus_toolbar_get_path_bar (NAUTILUS_TOOLBAR (window->toolbar));
g_signal_connect_object (path_bar, "path-clicked",
G_CALLBACK (path_bar_location_changed_callback), window, 0);
g_signal_connect_swapped (path_bar, "open-location",
G_CALLBACK (open_location_cb), window);
/* connect to the location entry signals */
location_entry = nautilus_toolbar_get_location_entry (NAUTILUS_TOOLBAR (window->toolbar));
g_signal_connect_object (location_entry, "location-changed",
G_CALLBACK (location_entry_location_changed_callback), window, 0);
g_signal_connect_object (location_entry, "cancel",
G_CALLBACK (location_entry_cancel_callback), window, 0);
gtk_window_set_titlebar (GTK_WINDOW (window), window->toolbar);
}
static void
notebook_page_removed_cb (GtkNotebook *notebook,
GtkWidget *page,
guint page_num,
gpointer user_data)
{
NautilusWindow *window = user_data;
NautilusWindowSlot *slot = NAUTILUS_WINDOW_SLOT (page);
gboolean dnd_slot;
dnd_slot = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (slot), "dnd-window-slot"));
if (!dnd_slot)
{
return;
}
close_slot (window, slot, FALSE);
}
static void
notebook_page_added_cb (GtkNotebook *notebook,
GtkWidget *page,
guint page_num,
gpointer user_data)
{
NautilusWindow *window = user_data;
NautilusWindowSlot *slot = NAUTILUS_WINDOW_SLOT (page);
NautilusWindowSlot *dummy_slot;
gboolean dnd_slot;
dnd_slot = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (slot), "dnd-window-slot"));
if (!dnd_slot)
{
return;
}
g_object_set_data (G_OBJECT (page), "dnd-window-slot",
GINT_TO_POINTER (FALSE));
nautilus_window_slot_set_window (slot, window);
window->slots = g_list_append (window->slots, slot);
g_signal_emit (window, signals[SLOT_ADDED], 0, slot);
nautilus_window_set_active_slot (window, slot);
dummy_slot = g_list_nth_data (window->slots, 0);
if (dummy_slot != NULL)
{
close_slot (window, dummy_slot, TRUE);
}
gtk_widget_show (GTK_WIDGET (window));
}
static GtkNotebook *
notebook_create_window_cb (GtkNotebook *notebook,
GtkWidget *page,
gint x,
gint y,
gpointer user_data)
{
NautilusApplication *app;
NautilusWindow *new_window;
NautilusWindowSlot *slot;
if (!NAUTILUS_IS_WINDOW_SLOT (page))
{
return NULL;
}
app = NAUTILUS_APPLICATION (g_application_get_default ());
new_window = nautilus_application_create_window
(app, gtk_widget_get_screen (GTK_WIDGET (notebook)));
slot = NAUTILUS_WINDOW_SLOT (page);
g_object_set_data (G_OBJECT (slot), "dnd-window-slot",
GINT_TO_POINTER (TRUE));
gtk_window_set_position (GTK_WINDOW (new_window), GTK_WIN_POS_MOUSE);
return GTK_NOTEBOOK (new_window->notebook);
}
static void
setup_notebook (NautilusWindow *window)
{
g_signal_connect (window->notebook, "tab-close-request",
G_CALLBACK (notebook_tab_close_requested),
window);
g_signal_connect (window->notebook, "popup-menu",
G_CALLBACK (notebook_popup_menu_cb),
window);
g_signal_connect (window->notebook, "switch-page",
G_CALLBACK (notebook_switch_page_cb),
window);
g_signal_connect (window->notebook, "create-window",
G_CALLBACK (notebook_create_window_cb),
window);
g_signal_connect (window->notebook, "page-added",
G_CALLBACK (notebook_page_added_cb),
window);
g_signal_connect (window->notebook, "page-removed",
G_CALLBACK (notebook_page_removed_cb),
window);
g_signal_connect (window->notebook_multi_press_gesture, "pressed",
G_CALLBACK (notebook_button_press_cb),
window);
}
const GActionEntry win_entries[] =
{
{ "back", action_back },
{ "forward", action_forward },
{ "up", action_up },
{ "view-menu", action_toggle_state_view_button, NULL, "false", NULL },
{ "reload", action_reload },
{ "stop", action_stop },
{ "new-tab", action_new_tab },
{ "enter-location", action_enter_location },
{ "bookmark-current-location", action_bookmark_current_location },
{ "undo", action_undo },
{ "redo", action_redo },
/* Only accesible by shorcuts */
{ "close-current-view", action_close_current_view },
{ "go-home", action_go_home },
{ "tab-previous", action_tab_previous },
{ "tab-next", action_tab_next },
{ "tab-move-left", action_tab_move_left },
{ "tab-move-right", action_tab_move_right },
{ "prompt-root-location", action_prompt_for_location_root },
{ "prompt-home-location", action_prompt_for_location_home },
{ "go-to-tab", NULL, "i", "0", action_go_to_tab },
{ "empty-trash", action_empty_trash },
{ "properties", action_properties },
{ "format", action_format },
{ "restore-tab", action_restore_tab },
};
static void
nautilus_window_initialize_actions (NautilusWindow *window)
{
GApplication *app;
GAction *action;
GVariant *state;
gchar detailed_action[80];
gchar accel[80];
gint i;
const gchar *reload_accels[] =
{
"F5",
"r",
NULL
};
const gchar *prompt_root_location_accels[] =
{
"slash",
"KP_Divide",
NULL
};
const gchar *prompt_home_location_accels[] =
{
"asciitilde",
"dead_tilde",
NULL
};
g_action_map_add_action_entries (G_ACTION_MAP (window),
win_entries, G_N_ELEMENTS (win_entries),
window);
app = g_application_get_default ();
nautilus_application_set_accelerator (app, "win.back", "Left");
nautilus_application_set_accelerator (app, "win.forward", "Right");
nautilus_application_set_accelerator (app, "win.enter-location", "l");
nautilus_application_set_accelerator (app, "win.new-tab", "t");
nautilus_application_set_accelerator (app, "win.close-current-view", "w");
/* Special case reload, since users are used to use two shortcuts instead of one */
nautilus_application_set_accelerators (app, "win.reload", reload_accels);
nautilus_application_set_accelerator (app, "win.undo", "z");
nautilus_application_set_accelerator (app, "win.redo", "z");
/* Only accesible by shorcuts */
nautilus_application_set_accelerator (app, "win.bookmark-current-location", "d");
nautilus_application_set_accelerator (app, "win.up", "Up");
nautilus_application_set_accelerator (app, "win.go-home", "Home");
nautilus_application_set_accelerator (app, "win.tab-previous", "Page_Up");
nautilus_application_set_accelerator (app, "win.tab-next", "Page_Down");
nautilus_application_set_accelerator (app, "win.tab-move-left", "Page_Up");
nautilus_application_set_accelerator (app, "win.tab-move-right", "Page_Down");
nautilus_application_set_accelerators (app, "win.prompt-root-location", prompt_root_location_accels);
/* Support keyboard layouts which have a dead tilde key but not a tilde key. */
nautilus_application_set_accelerators (app, "win.prompt-home-location", prompt_home_location_accels);
nautilus_application_set_accelerator (app, "win.view-menu", "F10");
nautilus_application_set_accelerator (app, "win.restore-tab", "t");
/* Alt+N for the first 9 tabs */
for (i = 0; i < 9; ++i)
{
g_snprintf (detailed_action, sizeof (detailed_action), "win.go-to-tab(%i)", i);
g_snprintf (accel, sizeof (accel), "%i", i + 1);
nautilus_application_set_accelerator (app, detailed_action, accel);
}
action = g_action_map_lookup_action (G_ACTION_MAP (app), "show-hide-sidebar");
state = g_action_get_state (action);
if (g_variant_get_boolean (state))
{
nautilus_window_show_sidebar (window);
}
g_variant_unref (state);
}
static void
nautilus_window_constructed (GObject *self)
{
NautilusWindow *window;
NautilusWindowSlot *slot;
NautilusApplication *application;
window = NAUTILUS_WINDOW (self);
nautilus_profile_start (NULL);
G_OBJECT_CLASS (nautilus_window_parent_class)->constructed (self);
application = NAUTILUS_APPLICATION (g_application_get_default ());
gtk_window_set_application (GTK_WINDOW (window), GTK_APPLICATION (application));
setup_toolbar (window);
gtk_window_set_default_size (GTK_WINDOW (window),
NAUTILUS_WINDOW_DEFAULT_WIDTH,
NAUTILUS_WINDOW_DEFAULT_HEIGHT);
setup_notebook (window);
nautilus_window_set_up_sidebar (window);
g_signal_connect_object (nautilus_file_undo_manager_get (), "undo-changed",
G_CALLBACK (nautilus_window_on_undo_changed), self,
G_CONNECT_AFTER);
/* Is required that the UI is constructed before initializating the actions, since
* some actions trigger UI widgets to show/hide. */
nautilus_window_initialize_actions (window);
slot = nautilus_window_create_and_init_slot (window, NULL, 0);
nautilus_window_set_active_slot (window, slot);
window->bookmarks_id =
g_signal_connect_swapped (nautilus_application_get_bookmarks (application), "changed",
G_CALLBACK (nautilus_window_sync_bookmarks), window);
nautilus_toolbar_on_window_constructed (NAUTILUS_TOOLBAR (window->toolbar));
nautilus_profile_end (NULL);
}
static gint
sort_slots_active_last (NautilusWindowSlot *a,
NautilusWindowSlot *b,
NautilusWindow *window)
{
if (window->active_slot == a)
{
return 1;
}
if (window->active_slot == b)
{
return -1;
}
return 0;
}
static void
destroy_slots_foreach (gpointer data,
gpointer user_data)
{
NautilusWindowSlot *slot = data;
NautilusWindow *window = user_data;
close_slot (window, slot, TRUE);
}
static void
nautilus_window_destroy (GtkWidget *object)
{
NautilusWindow *window;
NautilusApplication *application;
GList *slots_copy;
window = NAUTILUS_WINDOW (object);
application = NAUTILUS_APPLICATION (gtk_window_get_application (GTK_WINDOW (window)));
DEBUG ("Destroying window");
/* close all slots safely */
slots_copy = g_list_copy (window->slots);
if (window->active_slot != NULL)
{
/* Make sure active slot is last one to be closed, to avoid default activation
* of others slots when closing the active one, see bug #741952 */
slots_copy = g_list_sort_with_data (slots_copy, (GCompareDataFunc) sort_slots_active_last, window);
}
g_list_foreach (slots_copy, (GFunc) destroy_slots_foreach, window);
g_list_free (slots_copy);
/* the slots list should now be empty */
g_assert (window->slots == NULL);
window->active_slot = NULL;
g_clear_signal_handler (&window->bookmarks_id, nautilus_application_get_bookmarks (application));
g_clear_handle_id (&window->in_app_notification_undo_timeout_id, g_source_remove);
nautilus_window_unexport_handle (window);
GTK_WIDGET_CLASS (nautilus_window_parent_class)->destroy (object);
}
static void
nautilus_window_dispose (GObject *object)
{
NautilusWindow *window;
window = NAUTILUS_WINDOW (object);
g_clear_object (&window->notebook_multi_press_gesture);
G_OBJECT_CLASS (nautilus_window_parent_class)->dispose (object);
}
static void
nautilus_window_finalize (GObject *object)
{
NautilusWindow *window;
window = NAUTILUS_WINDOW (object);
if (window->sidebar_width_handler_id != 0)
{
g_source_remove (window->sidebar_width_handler_id);
window->sidebar_width_handler_id = 0;
}
if (window->in_app_notification_undo_timeout_id != 0)
{
g_source_remove (window->in_app_notification_undo_timeout_id);
window->in_app_notification_undo_timeout_id = 0;
}
if (window->notification_operation_timeout_id != 0)
{
g_source_remove (window->notification_operation_timeout_id);
window->notification_operation_timeout_id = 0;
}
g_clear_object (&window->selected_file);
g_clear_object (&window->selected_volume);
g_signal_handlers_disconnect_by_func (nautilus_file_undo_manager_get (),
G_CALLBACK (nautilus_window_on_undo_changed),
window);
g_queue_free_full (window->tab_data_queue, free_navigation_state);
g_object_unref (window->pad_controller);
/* nautilus_window_close() should have run */
g_assert (window->slots == NULL);
G_OBJECT_CLASS (nautilus_window_parent_class)->finalize (object);
}
static void
nautilus_window_save_geometry (NautilusWindow *window)
{
GdkWindow *gdk_window;
GdkWindowState window_state;
gboolean is_maximized;
g_assert (NAUTILUS_IS_WINDOW (window));
gdk_window = gtk_widget_get_window (GTK_WIDGET (window));
if (!gdk_window)
{
return;
}
window_state = gdk_window_get_state (gtk_widget_get_window (GTK_WIDGET (window)));
/* Don't save the window state for tiled windows. This is a special case,
* where the geometry only makes sense in combination with other tiled
* windows, that we can't possibly restore. */
if (window_state & GDK_WINDOW_STATE_TILED)
{
return;
}
is_maximized = window_state & GDK_WINDOW_STATE_MAXIMIZED;
/* Only save the initial size when the window is not maximized. If the
* window is maximized, a previously stored initial size will be more
* appropriate when unmaximizing the window in the future. */
if (!is_maximized)
{
gint width;
gint height;
GVariant *initial_size;
gtk_window_get_size (GTK_WINDOW (window), &width, &height);
initial_size = g_variant_new_parsed ("(%i, %i)", width, height);
g_settings_set_value (nautilus_window_state,
NAUTILUS_WINDOW_STATE_INITIAL_SIZE,
initial_size);
}
g_settings_set_boolean (nautilus_window_state,
NAUTILUS_WINDOW_STATE_MAXIMIZED,
is_maximized);
}
void
nautilus_window_close (NautilusWindow *window)
{
g_return_if_fail (NAUTILUS_IS_WINDOW (window));
nautilus_window_save_geometry (window);
nautilus_window_set_active_slot (window, NULL);
gtk_widget_destroy (GTK_WIDGET (window));
}
void
nautilus_window_set_active_slot (NautilusWindow *window,
NautilusWindowSlot *new_slot)
{
NautilusWindowSlot *old_slot;
g_assert (NAUTILUS_IS_WINDOW (window));
if (new_slot)
{
g_assert ((window == nautilus_window_slot_get_window (new_slot)));
}
old_slot = nautilus_window_get_active_slot (window);
if (old_slot == new_slot)
{
return;
}
DEBUG ("Setting new slot %p as active, old slot inactive %p", new_slot, old_slot);
/* make old slot inactive if it exists (may be NULL after init, for example) */
if (old_slot != NULL)
{
/* inform slot & view */
nautilus_window_slot_set_active (old_slot, FALSE);
nautilus_toolbar_set_window_slot (NAUTILUS_TOOLBAR (window->toolbar), NULL);
}
window->active_slot = new_slot;
/* make new slot active, if it exists */
if (new_slot)
{
/* inform slot & view */
nautilus_window_slot_set_active (new_slot, TRUE);
nautilus_toolbar_set_window_slot (NAUTILUS_TOOLBAR (window->toolbar), new_slot);
on_location_changed (window);
g_signal_emit (window, signals[ACTIVE_SELECTION_CHANGED], 0);
}
}
static void
nautilus_window_realize (GtkWidget *widget)
{
GTK_WIDGET_CLASS (nautilus_window_parent_class)->realize (widget);
update_cursor (NAUTILUS_WINDOW (widget));
}
static gboolean
nautilus_window_key_press_event (GtkWidget *widget,
GdkEventKey *event)
{
NautilusWindow *window;
guint keyval;
GtkWidget *focus_widget;
window = NAUTILUS_WINDOW (widget);
if (G_UNLIKELY (!gdk_event_get_keyval ((GdkEvent *) event, &keyval)))
{
g_return_val_if_reached (GDK_EVENT_PROPAGATE);
}
focus_widget = gtk_window_get_focus (GTK_WINDOW (window));
if (focus_widget != NULL && GTK_IS_EDITABLE (focus_widget))
{
/* if we have input focus on a GtkEditable (e.g. a GtkEntry), forward
* the event to it before activating accelerator bindings too.
*/
if (gtk_window_propagate_key_event (GTK_WINDOW (window),
(GdkEventKey *) event))
{
return GDK_EVENT_STOP;
}
}
for (int i = 0; i < G_N_ELEMENTS (extra_window_keybindings); i++)
{
if (extra_window_keybindings[i].keyval == keyval)
{
GAction *action;
action = g_action_map_lookup_action (G_ACTION_MAP (window), extra_window_keybindings[i].action);
g_assert (action != NULL);
if (g_action_get_enabled (action))
{
g_action_activate (action, NULL);
return GDK_EVENT_STOP;
}
break;
}
}
if (GTK_WIDGET_CLASS (nautilus_window_parent_class)->key_press_event (widget, event))
{
return GDK_EVENT_STOP;
}
if (nautilus_window_slot_handle_event (window->active_slot, (GdkEvent *) event))
{
return GDK_EVENT_STOP;
}
return GDK_EVENT_PROPAGATE;
}
void
nautilus_window_sync_title (NautilusWindow *window,
NautilusWindowSlot *slot)
{
g_return_if_fail (NAUTILUS_IS_WINDOW (window));
g_return_if_fail (NAUTILUS_IS_WINDOW_SLOT (slot));
if (slot == nautilus_window_get_active_slot (window))
{
gtk_window_set_title (GTK_WINDOW (window), nautilus_window_slot_get_title (slot));
}
nautilus_notebook_sync_tab_label (NAUTILUS_NOTEBOOK (window->notebook), slot);
}
#ifdef GDK_WINDOWING_WAYLAND
typedef struct
{
NautilusWindow *window;
NautilusWindowHandleExported callback;
gpointer user_data;
} WaylandWindowHandleExportedData;
static void
wayland_window_handle_exported (GdkWindow *window,
const char *wayland_handle_str,
gpointer user_data)
{
WaylandWindowHandleExportedData *data = user_data;
data->window->export_handle = g_strdup_printf ("wayland:%s", wayland_handle_str);
data->callback (data->window, data->window->export_handle, 0, data->user_data);
}
#endif
gboolean
nautilus_window_export_handle (NautilusWindow *window,
NautilusWindowHandleExported callback,
gpointer user_data)
{
guint xid = get_window_xid (window);
if (window->export_handle != NULL)
{
callback (window, window->export_handle, xid, user_data);
return TRUE;
}
#ifdef GDK_WINDOWING_X11
if (GDK_IS_X11_DISPLAY (gtk_widget_get_display (GTK_WIDGET (window))))
{
window->export_handle = g_strdup_printf ("x11:%x", xid);
callback (window, window->export_handle, xid, user_data);
return TRUE;
}
#endif
#ifdef GDK_WINDOWING_WAYLAND
if (GDK_IS_WAYLAND_DISPLAY (gtk_widget_get_display (GTK_WIDGET (window))))
{
GdkWindow *gdk_window = gtk_widget_get_window (GTK_WIDGET (window));
WaylandWindowHandleExportedData *data;
data = g_new0 (WaylandWindowHandleExportedData, 1);
data->window = window;
data->callback = callback;
data->user_data = user_data;
if (!gdk_wayland_window_export_handle (gdk_window,
wayland_window_handle_exported,
data,
g_free))
{
g_free (data);
return FALSE;
}
else
{
return TRUE;
}
}
#endif
g_warning ("Couldn't export handle, unsupported windowing system");
return FALSE;
}
void
nautilus_window_unexport_handle (NautilusWindow *window)
{
if (window->export_handle == NULL)
{
return;
}
#ifdef GDK_WINDOWING_WAYLAND
if (GDK_IS_WAYLAND_DISPLAY (gtk_widget_get_display (GTK_WIDGET (window))))
{
GdkWindow *gdk_window = gtk_widget_get_window (GTK_WIDGET (window));
if (gdk_window != NULL)
{
gdk_wayland_window_unexport_handle (gdk_window);
}
}
#endif
g_clear_pointer (&window->export_handle, g_free);
}
/**
* nautilus_window_show:
* @widget: GtkWidget
*
* Call parent and then show/hide window items
* base on user prefs.
*/
static void
nautilus_window_show (GtkWidget *widget)
{
GTK_WIDGET_CLASS (nautilus_window_parent_class)->show (widget);
}
NautilusWindowSlot *
nautilus_window_get_active_slot (NautilusWindow *window)
{
g_assert (NAUTILUS_IS_WINDOW (window));
return window->active_slot;
}
GList *
nautilus_window_get_slots (NautilusWindow *window)
{
g_assert (NAUTILUS_IS_WINDOW (window));
return window->slots;
}
static void
on_is_maximized_changed (GObject *object,
GParamSpec *pspec,
gpointer user_data)
{
gboolean is_maximized;
is_maximized = gtk_window_is_maximized (GTK_WINDOW (object));
g_settings_set_boolean (nautilus_window_state,
NAUTILUS_WINDOW_STATE_MAXIMIZED, is_maximized);
}
static gboolean
nautilus_window_delete_event (GtkWidget *widget,
GdkEventAny *event)
{
nautilus_window_close (NAUTILUS_WINDOW (widget));
return FALSE;
}
static void
on_multi_press_gesture_pressed (GtkGestureMultiPress *gesture,
gint n_press,
gdouble x,
gdouble y,
gpointer user_data)
{
GtkWidget *widget;
NautilusWindow *window;
guint button;
widget = gtk_event_controller_get_widget (GTK_EVENT_CONTROLLER (gesture));
window = NAUTILUS_WINDOW (widget);
button = gtk_gesture_single_get_current_button (GTK_GESTURE_SINGLE (gesture));
if (mouse_extra_buttons && (button == mouse_back_button))
{
nautilus_window_back_or_forward (window, TRUE, 0, 0);
}
else if (mouse_extra_buttons && (button == mouse_forward_button))
{
nautilus_window_back_or_forward (window, FALSE, 0, 0);
}
}
static void
mouse_back_button_changed (gpointer callback_data)
{
int new_back_button;
new_back_button = g_settings_get_int (nautilus_preferences, NAUTILUS_PREFERENCES_MOUSE_BACK_BUTTON);
/* Bounds checking */
if (new_back_button < 6 || new_back_button > UPPER_MOUSE_LIMIT)
{
return;
}
mouse_back_button = new_back_button;
}
static void
mouse_forward_button_changed (gpointer callback_data)
{
int new_forward_button;
new_forward_button = g_settings_get_int (nautilus_preferences, NAUTILUS_PREFERENCES_MOUSE_FORWARD_BUTTON);
/* Bounds checking */
if (new_forward_button < 6 || new_forward_button > UPPER_MOUSE_LIMIT)
{
return;
}
mouse_forward_button = new_forward_button;
}
static void
use_extra_mouse_buttons_changed (gpointer callback_data)
{
mouse_extra_buttons = g_settings_get_boolean (nautilus_preferences, NAUTILUS_PREFERENCES_MOUSE_USE_EXTRA_BUTTONS);
}
static void
nautilus_window_init (NautilusWindow *window)
{
GtkWindowGroup *window_group;
g_type_ensure (NAUTILUS_TYPE_TOOLBAR);
g_type_ensure (NAUTILUS_TYPE_NOTEBOOK);
gtk_widget_init_template (GTK_WIDGET (window));
g_signal_connect (window, "notify::is-maximized",
G_CALLBACK (on_is_maximized_changed), NULL);
g_signal_connect_object (window->in_app_notification_undo_close_button, "clicked",
G_CALLBACK (on_in_app_notification_undo_close_button_clicked), window, 0);
g_signal_connect_object (window->in_app_notification_undo_undo_button, "clicked",
G_CALLBACK (on_in_app_notification_undo_undo_button_clicked), window, 0);
window->slots = NULL;
window->active_slot = NULL;
gtk_style_context_add_class (gtk_widget_get_style_context (GTK_WIDGET (window)),
"nautilus-window");
window->toolbar = nautilus_toolbar_new ();
window_group = gtk_window_group_new ();
gtk_window_group_add_window (window_group, GTK_WINDOW (window));
g_object_unref (window_group);
window->tab_data_queue = g_queue_new ();
window->pad_controller = gtk_pad_controller_new (GTK_WINDOW (window),
G_ACTION_GROUP (window),
NULL);
gtk_pad_controller_set_action_entries (window->pad_controller,
pad_actions, G_N_ELEMENTS (pad_actions));
window->multi_press_gesture = gtk_gesture_multi_press_new (GTK_WIDGET (window));
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (window->multi_press_gesture),
GTK_PHASE_CAPTURE);
gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (window->multi_press_gesture), 0);
g_signal_connect (window->multi_press_gesture, "pressed",
G_CALLBACK (on_multi_press_gesture_pressed), NULL);
window->notebook_multi_press_gesture = gtk_gesture_multi_press_new (window->notebook);
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (window->notebook_multi_press_gesture),
GTK_PHASE_CAPTURE);
gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (window->notebook_multi_press_gesture),
GDK_BUTTON_SECONDARY);
}
static void
nautilus_window_class_init (NautilusWindowClass *class)
{
GObjectClass *oclass = G_OBJECT_CLASS (class);
GtkWidgetClass *wclass = GTK_WIDGET_CLASS (class);
oclass->dispose = nautilus_window_dispose;
oclass->finalize = nautilus_window_finalize;
oclass->constructed = nautilus_window_constructed;
wclass->destroy = nautilus_window_destroy;
wclass->show = nautilus_window_show;
wclass->realize = nautilus_window_realize;
wclass->key_press_event = nautilus_window_key_press_event;
wclass->delete_event = nautilus_window_delete_event;
wclass->grab_focus = nautilus_window_grab_focus;
gtk_widget_class_set_template_from_resource (wclass,
"/org/gnome/nautilus/ui/nautilus-window.ui");
gtk_widget_class_bind_template_child (wclass, NautilusWindow, content_paned);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, sidebar);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, places_sidebar);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, main_view);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, notebook);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, in_app_notification_undo);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, in_app_notification_undo_label);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, in_app_notification_undo_undo_button);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, in_app_notification_undo_close_button);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, notification_operation);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, notification_operation_label);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, notification_operation_open);
gtk_widget_class_bind_template_child (wclass, NautilusWindow, notification_operation_close);
gtk_widget_class_bind_template_callback (wclass, places_sidebar_show_other_locations_with_flags);
gtk_widget_class_bind_template_callback (wclass, places_sidebar_show_starred_location);
signals[SLOT_ADDED] =
g_signal_new ("slot-added",
G_TYPE_FROM_CLASS (class),
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
0,
NULL, NULL,
g_cclosure_marshal_VOID__OBJECT,
G_TYPE_NONE, 1, NAUTILUS_TYPE_WINDOW_SLOT);
signals[SLOT_REMOVED] =
g_signal_new ("slot-removed",
G_TYPE_FROM_CLASS (class),
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
0,
NULL, NULL,
g_cclosure_marshal_VOID__OBJECT,
G_TYPE_NONE, 1, NAUTILUS_TYPE_WINDOW_SLOT);
signals[ACTIVE_SELECTION_CHANGED] =
g_signal_new ("active-selection-changed",
G_TYPE_FROM_CLASS (class),
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
0,
NULL, NULL, NULL,
G_TYPE_NONE, 0);
g_signal_connect_swapped (nautilus_preferences,
"changed::" NAUTILUS_PREFERENCES_MOUSE_BACK_BUTTON,
G_CALLBACK (mouse_back_button_changed),
NULL);
g_signal_connect_swapped (nautilus_preferences,
"changed::" NAUTILUS_PREFERENCES_MOUSE_FORWARD_BUTTON,
G_CALLBACK (mouse_forward_button_changed),
NULL);
g_signal_connect_swapped (nautilus_preferences,
"changed::" NAUTILUS_PREFERENCES_MOUSE_USE_EXTRA_BUTTONS,
G_CALLBACK (use_extra_mouse_buttons_changed),
NULL);
gtk_widget_class_bind_template_callback (wclass, on_notification_operation_open_clicked);
gtk_widget_class_bind_template_callback (wclass, on_notification_operation_close_clicked);
}
NautilusWindow *
nautilus_window_new (GdkScreen *screen)
{
return g_object_new (NAUTILUS_TYPE_WINDOW,
"icon-name", APPLICATION_ID,
"screen", screen,
NULL);
}
NautilusWindowOpenFlags
nautilus_event_get_window_open_flags (void)
{
NautilusWindowOpenFlags flags = 0;
GdkEvent *event;
event = gtk_get_current_event ();
if (event == NULL)
{
return flags;
}
if ((event->type == GDK_BUTTON_PRESS || event->type == GDK_BUTTON_RELEASE) &&
(event->button.button == GDK_BUTTON_MIDDLE))
{
flags |= NAUTILUS_WINDOW_OPEN_FLAG_NEW_TAB;
}
gdk_event_free (event);
return flags;
}
void
nautilus_window_show_about_dialog (NautilusWindow *window)
{
const gchar *artists[] =
{
"The GNOME Project",
NULL
};
const gchar *authors[] =
{
"Alexander Larsson",
"Ali Abdin",
"Anders Carlsson",
"Andrew Walton",
"Andy Hertzfeld",
"Arlo Rose",
"Christian Neumair",
"Cosimo Cecchi",
"Darin Adler",
"David Camp",
"Eli Goldberg",
"Elliot Lee",
"Eskil Heyn Olsen",
"Ettore Perazzoli",
"Gene Z. Ragan",
"George Lebl",
"Ian McKellar",
"J Shane Culpepper",
"James Willcox",
"Jan Arne Petersen",
"John Harper",
"John Sullivan",
"Josh Barrow",
"Maciej Stachowiak",
"Mark McLoughlin",
"Mathieu Lacage",
"Mike Engber",
"Mike Fleming",
"Pavel Cisler",
"Ramiro Estrugo",
"Raph Levien",
"Rebecca Schulman",
"Robey Pointer",
"Robin * Slomkowski",
"Seth Nickell",
"Susan Kare",
"Tomas Bzatek",
"William Jon McCann",
NULL
};
const gchar *documenters[] =
{
"GNOME Documentation Team",
"Sun Microsystems",
NULL
};
g_autofree gchar *program_name = NULL;
/* “Files” is the generic application name and the suffix is
* an arbitrary and deliberately unlocalized string only shown
* in development builds.
*/
program_name = g_strconcat (_("Files"), NAME_SUFFIX, NULL);
gtk_show_about_dialog (window ? GTK_WINDOW (window) : NULL,
"program-name", program_name,
"version", VERSION,
"comments", _("Access and organize your files"),
"website", "https://wiki.gnome.org/action/show/Apps/Files",
"copyright", "© 1999–2018 The Files Authors",
"license-type", GTK_LICENSE_GPL_3_0,
"artists", artists,
"authors", authors,
"documenters", documenters,
/* Translators should localize the following string
* which will be displayed at the bottom of the about
* box to give credit to the translator(s).
*/
"translator-credits", _("translator-credits"),
"logo-icon-name", APPLICATION_ID,
NULL);
}
void
nautilus_window_search (NautilusWindow *window,
NautilusQuery *query)
{
NautilusWindowSlot *active_slot;
active_slot = nautilus_window_get_active_slot (window);
if (active_slot)
{
nautilus_window_slot_search (active_slot, query);
}
else
{
g_warning ("Trying search on a slot but no active slot present");
}
}
/* Ideally, this method should be a simple wrapper for the slot method. However,
* going back or forward can result in a new slot (or another subclass), so we
* workaround that by duplicating part of nautilus_window_slot_back_or_forward()
*/
void
nautilus_window_back_or_forward (NautilusWindow *window,
gboolean back,
guint distance,
NautilusWindowOpenFlags flags)
{
NautilusWindowSlot *slot;
GList *next_location_list, *back_list, *forward_list;
GFile *next_location;
guint len;
NautilusBookmark *next_location_bookmark;
gboolean active_slot_handles_location;
slot = nautilus_window_get_active_slot (window);
back_list = nautilus_window_slot_get_back_history (slot);
forward_list = nautilus_window_slot_get_forward_history (slot);
next_location_list = back ? back_list : forward_list;
len = (guint) g_list_length (next_location_list);
/* If we can't move in the direction at all, just return. */
if (len == 0)
{
return;
}
/* If the distance to move is off the end of the list, go to the end
* of the list. */
if (distance >= len)
{
distance = len - 1;
}
next_location_bookmark = g_list_nth_data (next_location_list, distance);
next_location = nautilus_bookmark_get_location (next_location_bookmark);
active_slot_handles_location = nautilus_window_slot_handles_location (slot, next_location);
if (!active_slot_handles_location)
{
NautilusNavigationState *navigation_state;
NautilusLocationChangeType location_change_type;
navigation_state = nautilus_window_slot_get_navigation_state (slot);
location_change_type = back ? NAUTILUS_LOCATION_CHANGE_BACK : NAUTILUS_LOCATION_CHANGE_FORWARD;
slot = replace_active_slot (window, next_location, flags);
nautilus_window_slot_open_location_set_navigation_state (slot,
next_location, flags, NULL,
location_change_type,
navigation_state, distance);
free_navigation_state (navigation_state);
}
else
{
nautilus_window_slot_back_or_forward (slot, back, distance, flags);
}
}