mirror of
https://github.com/MidnightCommander/mc
synced 2025-01-20 18:29:19 +03:00
c9ac26efb4
* cmd.c, tree.c: Updated to the new file.c api. * file.c: (copy_file_file, copy_dir_dir, move_file_file, move_dir_dir): Updated to work with the new updating stuff. * setup.c: New configuration option to compute the totals to be transfered. * file.c (compute_dir_size, panel_compute_totals): New routines to compute the total ammount of information that is going to be transfered.
251 lines
6.0 KiB
C
251 lines
6.0 KiB
C
/* GNU Midnight Commander -- GNOME edition
|
|
*
|
|
* Properties dialog for files and desktop icons.
|
|
*
|
|
* Copyright (C) 1997 The Free Software Foundation
|
|
*
|
|
* Authors: Miguel de Icaza
|
|
* Federico Mena
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include <string.h>
|
|
#include <stdlib.h> /* atoi */
|
|
#include <pwd.h>
|
|
#include <grp.h>
|
|
#include <sys/types.h>
|
|
#include "fs.h"
|
|
#include "x.h"
|
|
#include "gprop.h"
|
|
#include "util.h"
|
|
#include "dialog.h"
|
|
#include "file.h"
|
|
#include "../vfs/vfs.h"
|
|
#include "gpageprop.h"
|
|
|
|
static int prop_dialog_result;
|
|
|
|
static GtkWidget *
|
|
c_spacing (GtkWidget *widget)
|
|
{
|
|
gtk_container_set_border_width (GTK_CONTAINER (widget), 6);
|
|
return widget;
|
|
}
|
|
|
|
static void
|
|
properties_button_click (GtkWidget *widget, gpointer value)
|
|
{
|
|
prop_dialog_result = (int) value;
|
|
gtk_main_quit ();
|
|
}
|
|
|
|
static void
|
|
kill_toplevel ()
|
|
{
|
|
gtk_main_quit ();
|
|
}
|
|
|
|
int
|
|
item_properties (GtkWidget *parent, char *fname, desktop_icon_t *di)
|
|
{
|
|
GtkWidget *parent_window;
|
|
GdkCursor *cursor;
|
|
GtkWidget *notebook, *ok, *cancel;
|
|
GtkWidget *vbox;
|
|
GpropFilename *name;
|
|
GpropPerm *perm;
|
|
GpropGeneral *gene;
|
|
GpropExec *exec;
|
|
GtkDialog *toplevel;
|
|
|
|
umode_t new_mode;
|
|
char *new_group;
|
|
char *new_owner;
|
|
char *new_title;
|
|
char *new_icon;
|
|
char *new_name;
|
|
char *base;
|
|
|
|
struct stat s;
|
|
int retval = 0;
|
|
|
|
/* Set a clock cursor while we create stuff and read users/groups */
|
|
|
|
parent_window = gtk_widget_get_toplevel (parent);
|
|
cursor = gdk_cursor_new (GDK_WATCH);
|
|
gdk_window_set_cursor (parent_window->window, cursor);
|
|
gdk_cursor_destroy (cursor);
|
|
|
|
toplevel = GTK_DIALOG (gtk_dialog_new ());
|
|
notebook = gtk_notebook_new ();
|
|
gtk_box_pack_start_defaults (GTK_BOX (toplevel->vbox), notebook);
|
|
gtk_window_set_title (GTK_WINDOW (toplevel), fname);
|
|
|
|
/* Create the property widgets */
|
|
mc_stat (fname, &s);
|
|
base = x_basename (fname);
|
|
|
|
vbox = gtk_vbox_new (FALSE, 6);
|
|
|
|
if (di && di->dentry) {
|
|
gene = gprop_general_new (di->dentry->name, di->dentry->icon);
|
|
gtk_box_pack_start (GTK_BOX (vbox), gene->top, FALSE, FALSE, 0);
|
|
|
|
exec = gprop_exec_new (di->dentry);
|
|
gtk_box_pack_start (GTK_BOX (vbox), exec->top, FALSE, FALSE, 0);
|
|
} else {
|
|
name = gprop_filename_new (fname, base);
|
|
gtk_box_pack_start (GTK_BOX (vbox), name->top, FALSE, FALSE, 0);
|
|
}
|
|
|
|
perm = gprop_perm_new (s.st_mode, get_owner (s.st_uid), get_group (s.st_gid));
|
|
|
|
/* Pack them into nice notebook */
|
|
gtk_notebook_append_page (GTK_NOTEBOOK (notebook), c_spacing (vbox), gtk_label_new ("General"));
|
|
gtk_notebook_append_page (GTK_NOTEBOOK (notebook), c_spacing (perm->top), gtk_label_new ("Permissions"));
|
|
|
|
/* Ok, Cancel */
|
|
ok = gnome_stock_button (GNOME_STOCK_BUTTON_OK);
|
|
GTK_WIDGET_SET_FLAGS (ok, GTK_CAN_DEFAULT);
|
|
|
|
cancel = gnome_stock_button (GNOME_STOCK_BUTTON_CANCEL);
|
|
GTK_WIDGET_SET_FLAGS (cancel, GTK_CAN_DEFAULT);
|
|
|
|
gtk_box_pack_start (GTK_BOX (toplevel->action_area), ok, 0, 0, 0);
|
|
gtk_box_pack_start (GTK_BOX (toplevel->action_area), cancel, 0, 0, 0);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (ok), "clicked", GTK_SIGNAL_FUNC (properties_button_click), (gpointer) 0);
|
|
gtk_signal_connect (GTK_OBJECT (cancel), "clicked", GTK_SIGNAL_FUNC (properties_button_click), (gpointer) 1);
|
|
gtk_signal_connect (GTK_OBJECT (toplevel), "delete_event", GTK_SIGNAL_FUNC (kill_toplevel), toplevel);
|
|
|
|
/* Start the dialog box */
|
|
prop_dialog_result = -1;
|
|
|
|
gtk_widget_grab_default (ok);
|
|
gtk_widget_show_all (GTK_WIDGET (toplevel));
|
|
|
|
/* Restore the arrow cursor and run the dialog */
|
|
|
|
cursor = gdk_cursor_new (GDK_TOP_LEFT_ARROW);
|
|
gdk_window_set_cursor (parent_window->window, cursor);
|
|
gdk_cursor_destroy (cursor);
|
|
|
|
gtk_grab_add (GTK_WIDGET (toplevel));
|
|
gtk_main ();
|
|
gtk_grab_remove (GTK_WIDGET (toplevel));
|
|
|
|
if (prop_dialog_result != 0) {
|
|
gtk_widget_destroy (GTK_WIDGET (toplevel));
|
|
return 0; /* nothing changed */
|
|
}
|
|
|
|
/* Check and change permissions */
|
|
|
|
gprop_perm_get_data (perm, &new_mode, &new_owner, &new_group);
|
|
|
|
if (new_mode != s.st_mode) {
|
|
mc_chmod (fname, new_mode);
|
|
retval |= GPROP_MODE;
|
|
}
|
|
|
|
if ((strcmp (new_owner, get_owner (s.st_uid)) != 0) ||
|
|
(strcmp (new_group, get_group (s.st_gid)) != 0)) {
|
|
struct passwd *p;
|
|
struct group *g;
|
|
uid_t uid;
|
|
gid_t gid;
|
|
|
|
/* Get uid */
|
|
p = getpwnam (new_owner);
|
|
if (!p) {
|
|
uid = atoi (new_owner);
|
|
if (uid == 0) {
|
|
int v;
|
|
v = query_dialog ("Warning",
|
|
"You entered an invalid user name, should I use `root'?",
|
|
0, 2, "Yes", "No");
|
|
if (v != 0)
|
|
goto ciao;
|
|
}
|
|
} else
|
|
uid = p->pw_uid;
|
|
|
|
/* get gid */
|
|
g = getgrnam (new_group);
|
|
if (!g) {
|
|
gid = atoi (new_group);
|
|
if (gid == 0) {
|
|
message (1, "Error", "You entered an invalid group name");
|
|
goto ciao2;
|
|
}
|
|
} else
|
|
gid = g->gr_gid;
|
|
|
|
mc_chown (fname, uid, gid);
|
|
retval |= GPROP_UID | GPROP_GID;
|
|
|
|
ciao2:
|
|
endgrent ();
|
|
ciao:
|
|
endpwent ();
|
|
}
|
|
|
|
/* Check and change filename */
|
|
|
|
if (di && di->dentry){
|
|
gprop_exec_get_data (exec, di->dentry);
|
|
} else {
|
|
gprop_filename_get_data (name, &new_name);
|
|
|
|
if (strchr (new_name, '/'))
|
|
message (1, "Error", "The new name includes the `/' character");
|
|
else if (strcmp (new_name, base) != 0) {
|
|
char *base = x_basename (fname);
|
|
char save = *base;
|
|
char *full_target;
|
|
long count = 0;
|
|
double bytes = 0;
|
|
|
|
*base = 0;
|
|
full_target = concat_dir_and_file (fname, new_name);
|
|
*base = save;
|
|
|
|
create_op_win (OP_MOVE, 0);
|
|
file_mask_defaults ();
|
|
move_file_file (fname, full_target, &count, &bytes);
|
|
destroy_op_win ();
|
|
|
|
if (di) {
|
|
free (di->pathname);
|
|
di->pathname = full_target;
|
|
} else
|
|
free (full_target);
|
|
|
|
retval |= GPROP_FILENAME;
|
|
}
|
|
}
|
|
|
|
/* Check and change title and icon -- change is handled by caller */
|
|
|
|
if (di && di->dentry) {
|
|
gprop_general_get_data (gene, &new_title, &new_icon);
|
|
|
|
if (strcmp (new_title, di->dentry->name) != 0) {
|
|
g_free (di->dentry->name);
|
|
|
|
di->dentry->name = new_title;
|
|
retval |= GPROP_TITLE;
|
|
}
|
|
|
|
if (strcmp (new_icon, di->dentry->icon) != 0) {
|
|
g_free (di->dentry->icon);
|
|
|
|
di->dentry->icon = new_icon;
|
|
retval |= GPROP_ICON;
|
|
}
|
|
}
|
|
|
|
gtk_widget_destroy (GTK_WIDGET (toplevel));
|
|
return retval;
|
|
}
|