1
/* gb-new-file-popover.c
3
* Copyright (C) 2015 Christian Hergert <christian@hergert.me>
5
* This program is free software: you can redistribute it and/or modify
6
* it under the terms of the GNU General Public License as published by
7
* the Free Software Foundation, either version 3 of the License, or
8
* (at your option) any later version.
10
* This program is distributed in the hope that it will be useful,
11
* but WITHOUT ANY WARRANTY; without even the implied warranty of
12
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13
* GNU General Public License for more details.
15
* You should have received a copy of the GNU General Public License
16
* along with this program. If not, see <http://www.gnu.org/licenses/>.
19
#include <glib/gi18n.h>
21
#include "gb-new-file-popover.h"
22
#include "gb-string.h"
23
#include "gb-widget.h"
25
struct _GbNewFilePopover
27
GtkPopover parent_instance;
31
GCancellable *cancellable;
39
G_DEFINE_TYPE (GbNewFilePopover, gb_new_file_popover, GTK_TYPE_POPOVER)
53
static GParamSpec *gParamSpecs [LAST_PROP];
54
static guint gSignals [LAST_SIGNAL];
57
gb_new_file_popover__button_clicked (GbNewFilePopover *self,
60
g_autoptr(GFile) file = NULL;
63
g_assert (GB_IS_NEW_FILE_POPOVER (self));
64
g_assert (GTK_IS_BUTTON (button));
66
if (self->directory == NULL)
69
path = gtk_entry_get_text (self->entry);
70
if (gb_str_empty0 (path))
73
file = g_file_get_child (self->directory, path);
75
g_signal_emit (self, gSignals [CREATE_FILE], 0, file, self->file_type);
79
gb_new_file_popover__entry_activate (GbNewFilePopover *self,
82
g_assert (GB_IS_NEW_FILE_POPOVER (self));
83
g_assert (GTK_IS_ENTRY (entry));
85
if (gtk_widget_get_sensitive (GTK_WIDGET (self->button)))
86
gtk_widget_activate (GTK_WIDGET (self->button));
90
gb_new_file_popover__query_info_cb (GObject *object,
94
GFile *file = (GFile *)object;
95
g_autoptr(GFileInfo) file_info = NULL;
96
g_autoptr(GbNewFilePopover) self = user_data;
97
g_autoptr(GError) error = NULL;
100
file_info = g_file_query_info_finish (file, result, &error);
102
if (file_info == NULL &&
103
g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
106
if ((file_info == NULL) &&
107
g_error_matches (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND))
109
gtk_label_set_label (self->message, NULL);
110
gtk_widget_set_sensitive (GTK_WIDGET (self->button), TRUE);
114
if (file_info == NULL)
116
gtk_label_set_label (self->message, error->message);
120
file_type = g_file_info_get_file_type (file_info);
122
if (file_type == G_FILE_TYPE_DIRECTORY)
123
gtk_label_set_label (self->message,
124
_("A folder with that name already exists."));
126
gtk_label_set_label (self->message,
127
_("A file with that name already exists."));
129
gtk_widget_set_sensitive (GTK_WIDGET (self->button), FALSE);
133
gb_new_file_popover_check_exists (GbNewFilePopover *self,
137
g_autoptr(GFile) child = NULL;
139
g_assert (GB_IS_NEW_FILE_POPOVER (self));
140
g_assert (!directory || G_IS_FILE (directory));
142
if (self->cancellable != NULL)
144
if (!g_cancellable_is_cancelled (self->cancellable))
145
g_cancellable_cancel (self->cancellable);
146
g_clear_object (&self->cancellable);
149
gtk_label_set_label (self->message, NULL);
150
gtk_widget_set_sensitive (GTK_WIDGET (self->button), FALSE);
152
if (directory == NULL)
155
if (gb_str_empty0 (path))
158
child = g_file_get_child (directory, path);
160
self->cancellable = g_cancellable_new ();
162
g_file_query_info_async (child,
163
G_FILE_ATTRIBUTE_STANDARD_TYPE,
164
G_FILE_QUERY_INFO_NONE,
167
gb_new_file_popover__query_info_cb,
168
g_object_ref (self));
173
gb_new_file_popover__entry_changed (GbNewFilePopover *self,
178
g_assert (GB_IS_NEW_FILE_POPOVER (self));
179
g_assert (GTK_IS_ENTRY (entry));
181
text = gtk_entry_get_text (entry);
183
gtk_widget_set_sensitive (GTK_WIDGET (self->button), !gb_str_empty0 (text));
185
gb_new_file_popover_check_exists (self, self->directory, text);
189
gb_new_file_popover_finalize (GObject *object)
191
GbNewFilePopover *self = (GbNewFilePopover *)object;
193
if (self->cancellable && !g_cancellable_is_cancelled (self->cancellable))
194
g_cancellable_cancel (self->cancellable);
196
g_clear_object (&self->cancellable);
197
g_clear_object (&self->directory);
199
G_OBJECT_CLASS (gb_new_file_popover_parent_class)->finalize (object);
203
gb_new_file_popover_get_property (GObject *object,
208
GbNewFilePopover *self = GB_NEW_FILE_POPOVER(object);
213
g_value_set_object (value, gb_new_file_popover_get_directory (self));
217
g_value_set_enum (value, gb_new_file_popover_get_file_type (self));
221
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
226
* gb_new_file_popover_set_property:
227
* @object: (in): A #GObject.
228
* @prop_id: (in): The property identifier.
229
* @value: (in): The given property.
230
* @pspec: (in): A #ParamSpec.
232
* Set a given #GObject property.
235
gb_new_file_popover_set_property (GObject *object,
240
GbNewFilePopover *self = GB_NEW_FILE_POPOVER(object);
245
gb_new_file_popover_set_directory (self, g_value_get_object (value));
249
gb_new_file_popover_set_file_type (self, g_value_get_enum (value));
253
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
258
gb_new_file_popover_class_init (GbNewFilePopoverClass *klass)
260
GObjectClass *object_class = G_OBJECT_CLASS (klass);
262
object_class->finalize = gb_new_file_popover_finalize;
263
object_class->get_property = gb_new_file_popover_get_property;
264
object_class->set_property = gb_new_file_popover_set_property;
266
gParamSpecs [PROP_DIRECTORY] =
267
g_param_spec_object ("directory",
271
(G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
273
gParamSpecs [PROP_FILE_TYPE] =
274
g_param_spec_enum ("file-type",
276
_("The file type to create."),
279
(G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
281
g_object_class_install_properties (object_class, LAST_PROP, gParamSpecs);
283
gSignals [CREATE_FILE] =
284
g_signal_new ("create-file",
285
G_TYPE_FROM_CLASS (klass),
294
GB_WIDGET_CLASS_TEMPLATE (klass, "gb-new-file-popover.ui");
296
GB_WIDGET_CLASS_BIND (klass, GbNewFilePopover, button);
297
GB_WIDGET_CLASS_BIND (klass, GbNewFilePopover, entry);
298
GB_WIDGET_CLASS_BIND (klass, GbNewFilePopover, message);
299
GB_WIDGET_CLASS_BIND (klass, GbNewFilePopover, title);
303
gb_new_file_popover_init (GbNewFilePopover *self)
305
self->file_type = G_FILE_TYPE_REGULAR;
307
gtk_widget_init_template (GTK_WIDGET (self));
309
g_signal_connect_object (self->entry,
311
G_CALLBACK (gb_new_file_popover__entry_activate),
315
g_signal_connect_object (self->entry,
317
G_CALLBACK (gb_new_file_popover__entry_changed),
321
g_signal_connect_object (self->button,
323
G_CALLBACK (gb_new_file_popover__button_clicked),
329
gb_new_file_popover_get_file_type (GbNewFilePopover *self)
331
g_return_val_if_fail (GB_IS_NEW_FILE_POPOVER (self), 0);
333
return self->file_type;
337
gb_new_file_popover_set_file_type (GbNewFilePopover *self,
340
g_return_if_fail (GB_IS_NEW_FILE_POPOVER (self));
341
g_return_if_fail ((file_type == G_FILE_TYPE_REGULAR) ||
342
(file_type == G_FILE_TYPE_DIRECTORY));
344
if (file_type != self->file_type)
346
self->file_type = file_type;
348
if (file_type == G_FILE_TYPE_REGULAR)
349
gtk_label_set_label (self->title, _("File Name"));
351
gtk_label_set_label (self->title, _("Folder Name"));
353
g_object_notify_by_pspec (G_OBJECT (self), gParamSpecs [PROP_FILE_TYPE]);
358
gb_new_file_popover_set_directory (GbNewFilePopover *self,
361
g_return_if_fail (GB_IS_NEW_FILE_POPOVER (self));
362
g_return_if_fail (G_IS_FILE (directory));
364
if (g_set_object (&self->directory, directory))
368
path = gtk_entry_get_text (self->entry);
369
gb_new_file_popover_check_exists (self, directory, path);
370
g_object_notify_by_pspec (G_OBJECT (self), gParamSpecs [PROP_DIRECTORY]);
375
* gb_new_file_popover_get_directory:
377
* Returns: (transfer none) (nullable): A #GFile or %NULL.
380
gb_new_file_popover_get_directory (GbNewFilePopover *self)
382
g_return_val_if_fail (GB_IS_NEW_FILE_POPOVER (self), NULL);
384
return self->directory;