mirror of
https://github.com/MidnightCommander/mc
synced 2025-01-11 05:49:18 +03:00
e0faaa671f
the ChangeLogs for full information. - Federico
796 lines
30 KiB
C
796 lines
30 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*- */
|
|
/* New dialogs... */
|
|
#include <config.h>
|
|
#include "util.h"
|
|
#include "panel.h"
|
|
#include <gnome.h>
|
|
#include "file.h"
|
|
#include "filegui.h"
|
|
#include "fileopctx.h"
|
|
#include "regex.h"
|
|
#include "../vfs/vfs.h"
|
|
|
|
enum {
|
|
REPLACE_PROMPT,
|
|
REPLACE_ALWAYS,
|
|
REPLACE_UPDATE,
|
|
REPLACE_NEVER,
|
|
REPLACE_ABORT,
|
|
REPLACE_SIZE,
|
|
REPLACE_OPTION_MENU
|
|
} FileReplaceCode;
|
|
|
|
/* This structure describes the UI and internal data required by a file
|
|
* operation context.
|
|
*/
|
|
typedef struct {
|
|
/* The progress window */
|
|
GtkWidget *op_win;
|
|
|
|
/* Set to FALSE in file_op_context_create_ui, set on the cancel_cb if
|
|
* user click on Cancel.
|
|
*/
|
|
gboolean aborting;
|
|
|
|
/* Source file label */
|
|
GtkWidget *op_source_label;
|
|
|
|
/* Target file label */
|
|
GtkWidget *op_target_label;
|
|
|
|
/* File number label */
|
|
GtkObject *count_label;
|
|
|
|
/* Current file label */
|
|
GtkWidget *file_label;
|
|
|
|
/* Bytes progress bar */
|
|
GtkObject *byte_prog;
|
|
|
|
/* Copy status in query replace dialog */
|
|
int copy_status;
|
|
int minor_copy_status;
|
|
|
|
/* Overwrite toggle */
|
|
GtkWidget *op_radio;
|
|
} FileOpContextUI;
|
|
|
|
|
|
#define GDIALOG_TO_STRING "To: "
|
|
#define GDIALOG_FROM_STRING "Copying from: "
|
|
#define GDIALOG_PROGRESS_WIDTH 350
|
|
|
|
|
|
|
|
/* Callbacks go here... */
|
|
static void
|
|
fmd_check_box_callback (GtkWidget *widget, gpointer data)
|
|
{
|
|
if (data)
|
|
*((gint*)data) = GTK_TOGGLE_BUTTON (widget)->active;
|
|
}
|
|
|
|
static gchar *
|
|
trim_file_name (FileOpContextUI *ui, gchar *path, gint length, gint cur_length)
|
|
{
|
|
static gint dotdotdot = 0;
|
|
gchar *path_copy = NULL;
|
|
gint len;
|
|
|
|
if (!dotdotdot)
|
|
dotdotdot = gdk_string_width (ui->op_source_label->style->font, "...");
|
|
/* Cut the font length of path to length. */
|
|
|
|
length -= dotdotdot;
|
|
len = (gint) ((1.0 - (gfloat) length / (gfloat) cur_length) * strlen (path));
|
|
|
|
/* we guess a starting point */
|
|
if (gdk_string_width (ui->op_source_label->style->font, path + len) < length) {
|
|
while (gdk_string_width (ui->op_source_label->style->font, path + len) < length)
|
|
len --;
|
|
len++;
|
|
} else {
|
|
while (gdk_string_width (ui->op_source_label->style->font, path + len) > length)
|
|
len ++;
|
|
}
|
|
path_copy = g_malloc (sizeof (gchar [1 + 3 + strlen (path) - len]));
|
|
sprintf (path_copy, "...%s", path + len);
|
|
return path_copy;
|
|
}
|
|
|
|
FileProgressStatus
|
|
file_progress_show_source (FileOpContext *ctx, char *path)
|
|
{
|
|
static gint from_width = 0;
|
|
FileOpContextUI *ui;
|
|
gint path_width;
|
|
gchar *path_copy = NULL;
|
|
|
|
g_return_val_if_fail (ctx != NULL, FILE_CONT);
|
|
g_return_val_if_fail (ctx->ui != NULL, FILE_CONT);
|
|
|
|
ui = ctx->ui;
|
|
|
|
g_return_val_if_fail (ui->op_source_label != NULL, FILE_CONT);
|
|
|
|
if (ui->aborting)
|
|
return FILE_ABORT;
|
|
|
|
if (path == NULL){
|
|
gtk_label_set (GTK_LABEL (ui->op_source_label), "");
|
|
return FILE_CONT;
|
|
}
|
|
|
|
if (!from_width){
|
|
from_width = gdk_string_width (ui->op_source_label->style->font,
|
|
_(GDIALOG_FROM_STRING));
|
|
}
|
|
path_width = gdk_string_width (ui->op_source_label->style->font, path);
|
|
if (from_width + path_width < GDIALOG_PROGRESS_WIDTH)
|
|
gtk_label_set (GTK_LABEL (ui->op_source_label), path);
|
|
else {
|
|
path_copy = trim_file_name (ui, path, GDIALOG_PROGRESS_WIDTH - from_width,
|
|
path_width);
|
|
|
|
gtk_label_set (GTK_LABEL (ui->op_source_label), path_copy);
|
|
g_free (path_copy);
|
|
}
|
|
|
|
return FILE_CONT;
|
|
}
|
|
|
|
FileProgressStatus
|
|
file_progress_show_target (FileOpContext *ctx, char *path)
|
|
{
|
|
static gint to_width = 0;
|
|
FileOpContextUI *ui;
|
|
gint path_width;
|
|
gchar *path_copy = NULL;
|
|
|
|
g_return_val_if_fail (ctx != NULL, FILE_CONT);
|
|
g_return_val_if_fail (ctx->ui != NULL, FILE_CONT);
|
|
|
|
ui = ctx->ui;
|
|
|
|
g_return_val_if_fail (ui->op_target_label != NULL, FILE_CONT);
|
|
|
|
if (ui->aborting)
|
|
return FILE_ABORT;
|
|
|
|
if (path == NULL){
|
|
gtk_label_set (GTK_LABEL (ui->op_target_label), "");
|
|
return FILE_CONT;
|
|
}
|
|
|
|
if (!to_width)
|
|
to_width = gdk_string_width (ui->op_target_label->style->font,
|
|
_(GDIALOG_TO_STRING));
|
|
path_width = gdk_string_width (ui->op_target_label->style->font, path);
|
|
if (to_width + path_width < GDIALOG_PROGRESS_WIDTH)
|
|
gtk_label_set (GTK_LABEL (ui->op_target_label), path);
|
|
else {
|
|
path_copy = trim_file_name (ui, path, GDIALOG_PROGRESS_WIDTH - to_width, path_width);
|
|
gtk_label_set (GTK_LABEL (ui->op_target_label), path_copy);
|
|
g_free (path_copy);
|
|
}
|
|
|
|
return FILE_CONT;
|
|
}
|
|
|
|
FileProgressStatus
|
|
file_progress_show_deleting (FileOpContext *ctx, char *path)
|
|
{
|
|
FileOpContextUI *ui;
|
|
|
|
g_return_val_if_fail (ctx != NULL, FILE_CONT);
|
|
g_return_val_if_fail (ctx->ui != NULL, FILE_CONT);
|
|
|
|
ui = ctx->ui;
|
|
|
|
if (ui->aborting)
|
|
return FILE_ABORT;
|
|
|
|
/* FIXME */
|
|
g_warning ("memo: file_progress_show_deleting!\npath\t%s\n", path);
|
|
return FILE_CONT;
|
|
}
|
|
|
|
FileProgressStatus
|
|
file_progress_show (FileOpContext *ctx, long done, long total)
|
|
{
|
|
static gchar count[10];
|
|
FileOpContextUI *ui;
|
|
|
|
g_return_val_if_fail (ctx != NULL, FILE_CONT);
|
|
g_return_val_if_fail (ctx->ui != NULL, FILE_CONT);
|
|
|
|
ui = ctx->ui;
|
|
|
|
if (ui->aborting)
|
|
return FILE_ABORT;
|
|
|
|
snprintf (count, 9, "%d%%", (gint)(100.0 *(gfloat)done/(gfloat)total));
|
|
gtk_label_set (GTK_LABEL (ui->file_label), count);
|
|
while (gtk_events_pending ())
|
|
gtk_main_iteration ();
|
|
return FILE_CONT;
|
|
}
|
|
|
|
FileProgressStatus
|
|
file_progress_show_count (FileOpContext *ctx, long done, long total)
|
|
{
|
|
static gchar count[14]; /* that's a lot of files... */
|
|
FileOpContextUI *ui;
|
|
|
|
g_return_val_if_fail (ctx != NULL, FILE_CONT);
|
|
g_return_val_if_fail (ctx->ui != NULL, FILE_CONT);
|
|
|
|
ui = ctx->ui;
|
|
|
|
if (ui->aborting)
|
|
return FILE_ABORT;
|
|
|
|
snprintf (count, 13, "%d/%d", done, total);
|
|
gtk_label_set (GTK_LABEL (ui->count_label), count);
|
|
while (gtk_events_pending ())
|
|
gtk_main_iteration ();
|
|
return FILE_CONT;
|
|
}
|
|
|
|
FileProgressStatus
|
|
file_progress_show_bytes (FileOpContext *ctx, double done, double total)
|
|
{
|
|
FileOpContextUI *ui;
|
|
|
|
g_return_val_if_fail (ctx != NULL, FILE_CONT);
|
|
g_return_val_if_fail (ctx->ui != NULL, FILE_CONT);
|
|
|
|
ui = ctx->ui;
|
|
|
|
if (ui->aborting)
|
|
return FILE_ABORT;
|
|
|
|
if (total == 0.0)
|
|
gtk_progress_bar_update (GTK_PROGRESS_BAR (ui->byte_prog), 0.0);
|
|
else
|
|
gtk_progress_bar_update (GTK_PROGRESS_BAR (ui->byte_prog),
|
|
(gfloat) done/(gfloat) total);
|
|
while (gtk_events_pending ())
|
|
gtk_main_iteration ();
|
|
return FILE_CONT;
|
|
}
|
|
|
|
static void
|
|
option_menu_policy_callback (GtkWidget *item, gpointer data)
|
|
{
|
|
FileOpContextUI *ui;
|
|
int status;
|
|
|
|
ui = data;
|
|
status = GPOINTER_TO_INT (gtk_object_get_user_data (GTK_OBJECT (item)));
|
|
|
|
ui->minor_copy_status = status;
|
|
ui->copy_status = status;
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (ui->op_radio), TRUE);
|
|
}
|
|
|
|
static void
|
|
policy_callback (GtkWidget *button, gpointer data)
|
|
{
|
|
FileOpContextUI *ui;
|
|
int status;
|
|
|
|
ui = data;
|
|
status = GPOINTER_TO_INT (gtk_object_get_user_data (GTK_OBJECT (button)));
|
|
|
|
if (GTK_TOGGLE_BUTTON (button)->active) {
|
|
g_print ("toggle\n");
|
|
if (status == REPLACE_OPTION_MENU) {
|
|
ui->copy_status = ui->minor_copy_status;
|
|
} else
|
|
ui->copy_status = status;
|
|
}
|
|
}
|
|
|
|
FileProgressStatus
|
|
file_progress_query_replace_policy (FileOpContext *ctx, gboolean dialog_needed)
|
|
{
|
|
FileOpContextUI *ui;
|
|
GtkWidget *qrp_dlg;
|
|
GtkWidget *radio;
|
|
GtkWidget *vbox;
|
|
GtkWidget *hbox;
|
|
GtkWidget *icon;
|
|
GtkWidget *label;
|
|
GtkWidget *hrbox;
|
|
GSList *group = NULL;
|
|
GtkWidget *omenu;
|
|
GtkWidget *menu;
|
|
GtkWidget *menu_item;
|
|
|
|
g_return_val_if_fail (ctx != NULL, FILE_CONT);
|
|
g_return_val_if_fail (ctx->ui != NULL, FILE_CONT);
|
|
|
|
ui = ctx->ui;
|
|
|
|
if (ui->aborting)
|
|
return FILE_ABORT;
|
|
|
|
ui->copy_status = REPLACE_PROMPT;
|
|
if (dialog_needed == FALSE)
|
|
return FILE_CONT;
|
|
ui->minor_copy_status = REPLACE_ALWAYS;
|
|
g_print ("in file_progress_query_replace_policy\n");
|
|
qrp_dlg = gnome_dialog_new (_("Files Exist"),
|
|
GNOME_STOCK_BUTTON_OK,
|
|
GNOME_STOCK_BUTTON_CANCEL,
|
|
NULL);
|
|
vbox = gtk_vbox_new (FALSE, GNOME_PAD_SMALL);
|
|
hbox = gtk_hbox_new (FALSE, GNOME_PAD_SMALL);
|
|
icon = gnome_stock_pixmap_widget (hbox, GNOME_STOCK_PIXMAP_HELP);
|
|
|
|
gtk_box_pack_start (GTK_BOX (hbox), icon, FALSE, FALSE, 0);
|
|
gtk_box_pack_start (GTK_BOX (hbox), vbox, FALSE, FALSE, GNOME_PAD_SMALL);
|
|
|
|
label = gtk_label_new (_("Some of the files you are trying to copy already "
|
|
"exist in the destination folder."));
|
|
gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
|
|
gtk_label_set_justify (GTK_LABEL (label), GTK_JUSTIFY_LEFT);
|
|
gtk_label_set_line_wrap (GTK_LABEL (label), TRUE);
|
|
|
|
gtk_box_pack_start (GTK_BOX (vbox), label, FALSE, FALSE, 0);
|
|
gtk_box_pack_start (GTK_BOX (vbox), gtk_hseparator_new (), FALSE, FALSE, 0);
|
|
|
|
radio = gtk_radio_button_new_with_label (group, _("Prompt me before overwriting any file."));
|
|
gtk_object_set_user_data (GTK_OBJECT (radio), GINT_TO_POINTER (REPLACE_PROMPT));
|
|
gtk_signal_connect (GTK_OBJECT (radio), "toggled",
|
|
GTK_SIGNAL_FUNC (policy_callback), ui);
|
|
gtk_box_pack_start (GTK_BOX (vbox), radio, FALSE, FALSE, 0);
|
|
group = gtk_radio_button_group (GTK_RADIO_BUTTON (radio));
|
|
|
|
radio = gtk_radio_button_new_with_label (group, _("Don't overwrite any files."));
|
|
gtk_object_set_user_data (GTK_OBJECT (radio), GINT_TO_POINTER (REPLACE_NEVER));
|
|
gtk_signal_connect (GTK_OBJECT (radio), "toggled",
|
|
GTK_SIGNAL_FUNC (policy_callback), ui);
|
|
gtk_box_pack_start (GTK_BOX (vbox), radio, FALSE, FALSE, 0);
|
|
group = gtk_radio_button_group (GTK_RADIO_BUTTON (radio));
|
|
|
|
ui->op_radio = gtk_radio_button_new (group);
|
|
gtk_object_set_user_data (GTK_OBJECT (ui->op_radio), GINT_TO_POINTER (REPLACE_OPTION_MENU));
|
|
gtk_signal_connect (GTK_OBJECT (ui->op_radio), "toggled",
|
|
GTK_SIGNAL_FUNC (policy_callback), ui);
|
|
hrbox = gtk_hbox_new (FALSE, GNOME_PAD_SMALL);
|
|
gtk_box_pack_start (GTK_BOX (hrbox), gtk_label_new (_("Overwrite:")), FALSE, FALSE, 0);
|
|
|
|
/* we set up the option menu. */
|
|
omenu = gtk_option_menu_new ();
|
|
gtk_box_pack_start (GTK_BOX (hrbox), omenu, FALSE, FALSE, 0);
|
|
menu = gtk_menu_new ();
|
|
|
|
menu_item = gtk_menu_item_new_with_label ( _("Older files."));
|
|
gtk_menu_append (GTK_MENU (menu), menu_item);
|
|
gtk_object_set_user_data (GTK_OBJECT (menu_item), GINT_TO_POINTER (REPLACE_UPDATE));
|
|
gtk_signal_connect (GTK_OBJECT (menu_item), "activate",
|
|
GTK_SIGNAL_FUNC (option_menu_policy_callback), ui);
|
|
|
|
menu_item = gtk_menu_item_new_with_label ( _("Files only if size differs."));
|
|
gtk_menu_append (GTK_MENU (menu), menu_item);
|
|
gtk_object_set_user_data (GTK_OBJECT (menu_item), GINT_TO_POINTER (REPLACE_SIZE));
|
|
gtk_signal_connect (GTK_OBJECT (menu_item), "activate",
|
|
GTK_SIGNAL_FUNC (option_menu_policy_callback), ui);
|
|
|
|
menu_item = gtk_menu_item_new_with_label ( _("All files."));
|
|
gtk_menu_append (GTK_MENU (menu), menu_item);
|
|
gtk_object_set_user_data (GTK_OBJECT (menu_item), GINT_TO_POINTER (REPLACE_ALWAYS));
|
|
gtk_signal_connect (GTK_OBJECT (menu_item), "activate",
|
|
GTK_SIGNAL_FUNC (option_menu_policy_callback), ui);
|
|
|
|
gtk_widget_show_all (menu);
|
|
gtk_option_menu_set_menu (GTK_OPTION_MENU (omenu), menu);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (ui->op_radio), "toggled",
|
|
GTK_SIGNAL_FUNC (policy_callback), (gpointer) REPLACE_ALWAYS);
|
|
gtk_box_pack_start (GTK_BOX (vbox), ui->op_radio, FALSE, FALSE, 0);
|
|
group = gtk_radio_button_group (GTK_RADIO_BUTTON (ui->op_radio));
|
|
gtk_container_add (GTK_CONTAINER (ui->op_radio), hrbox);
|
|
|
|
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (qrp_dlg)->vbox),
|
|
hbox, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_show_all (GTK_WIDGET (GNOME_DIALOG (qrp_dlg)->vbox));
|
|
switch (gnome_dialog_run_and_close (GNOME_DIALOG (qrp_dlg))) {
|
|
case 0:
|
|
break;
|
|
case -1:
|
|
default:
|
|
ui->copy_status = REPLACE_ABORT;
|
|
return FILE_ABORT;
|
|
}
|
|
return FILE_CONT;
|
|
}
|
|
|
|
FileProgressStatus
|
|
file_progress_real_query_replace (FileOpContext *ctx, enum OperationMode mode, char *destname,
|
|
struct stat *_s_stat, struct stat *_d_stat)
|
|
{
|
|
FileOpContextUI *ui;
|
|
GtkWidget *qr_dlg;
|
|
gchar msg[128];
|
|
GtkWidget *label;
|
|
|
|
g_return_val_if_fail (ctx != NULL, FILE_CONT);
|
|
g_return_val_if_fail (ctx->ui != NULL, FILE_CONT);
|
|
|
|
ui = ctx->ui;
|
|
|
|
if (ui->aborting)
|
|
return FILE_ABORT;
|
|
|
|
/* so what's the situation? Do we prompt or don't we prompt. */
|
|
if (ui->copy_status == REPLACE_PROMPT){
|
|
qr_dlg = gnome_dialog_new (_("File Exists"),
|
|
GNOME_STOCK_BUTTON_YES,
|
|
GNOME_STOCK_BUTTON_NO,
|
|
GNOME_STOCK_BUTTON_CANCEL, NULL);
|
|
|
|
snprintf (msg, sizeof (msg)-1, _("The target file already exists: %s"), destname);
|
|
label = gtk_label_new (msg);
|
|
gtk_label_set_justify (GTK_LABEL (label), GTK_JUSTIFY_LEFT);
|
|
gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
|
|
gtk_label_set_line_wrap (GTK_LABEL (label), TRUE);
|
|
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (qr_dlg)->vbox),
|
|
label, FALSE, FALSE, 0);
|
|
label = gtk_label_new (_("Replace it?"));
|
|
gtk_label_set_justify (GTK_LABEL (label), GTK_JUSTIFY_LEFT);
|
|
gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
|
|
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (qr_dlg)->vbox),
|
|
label, FALSE, FALSE, 0);
|
|
gtk_widget_show_all (GNOME_DIALOG (qr_dlg)->vbox);
|
|
switch (gnome_dialog_run_and_close (GNOME_DIALOG (qr_dlg))) {
|
|
case 0:
|
|
return FILE_CONT;
|
|
case 1:
|
|
return FILE_SKIP;
|
|
default:
|
|
return FILE_ABORT;
|
|
}
|
|
}
|
|
|
|
switch (ui->copy_status){
|
|
case REPLACE_UPDATE:
|
|
if (_s_stat->st_mtime > _d_stat->st_mtime)
|
|
return FILE_CONT;
|
|
else
|
|
return FILE_SKIP;
|
|
case REPLACE_SIZE:
|
|
if (_s_stat->st_size == _d_stat->st_size)
|
|
return FILE_SKIP;
|
|
else
|
|
return FILE_CONT;
|
|
case REPLACE_ALWAYS:
|
|
return FILE_CONT;
|
|
case REPLACE_NEVER:
|
|
return FILE_SKIP;
|
|
case REPLACE_ABORT:
|
|
default:
|
|
return FILE_ABORT;
|
|
}
|
|
}
|
|
|
|
void
|
|
file_progress_set_stalled_label (FileOpContext *ctx, char *stalled_msg)
|
|
{
|
|
g_return_if_fail (ctx != NULL);
|
|
g_return_if_fail (ctx->ui != NULL);
|
|
|
|
if (!stalled_msg || !*stalled_msg)
|
|
return;
|
|
/* FIXME */
|
|
g_warning ("FIXME: file_progress_set_stalled_label!\nmsg\t%s\n",stalled_msg);
|
|
}
|
|
char *
|
|
file_mask_dialog (FileOpContext *ctx, FileOperation operation, char *text, char *def_text,
|
|
int only_one, int *do_background)
|
|
{
|
|
GtkWidget *fmd_win;
|
|
GtkWidget *notebook;
|
|
GtkWidget *hbox;
|
|
GtkWidget *vbox, *label;
|
|
GtkWidget *alignment;
|
|
GtkWidget *fentry;
|
|
GtkWidget *cbox;
|
|
GtkWidget *icon;
|
|
int source_easy_patterns = easy_patterns;
|
|
char *source_mask, *orig_mask, *dest_dir;
|
|
const char *error;
|
|
struct stat buf;
|
|
|
|
g_return_val_if_fail (ctx != NULL, NULL);
|
|
|
|
ctx->stable_symlinks = 0;
|
|
|
|
/* Basic window */
|
|
if (operation == OP_COPY)
|
|
fmd_win = gnome_dialog_new (_("Copy"), GNOME_STOCK_BUTTON_OK,
|
|
GNOME_STOCK_BUTTON_CANCEL, NULL);
|
|
else if (operation == OP_MOVE)
|
|
fmd_win = gnome_dialog_new (_("Move"), GNOME_STOCK_BUTTON_OK,
|
|
GNOME_STOCK_BUTTON_CANCEL, NULL);
|
|
|
|
hbox = gtk_hbox_new (FALSE, GNOME_PAD);
|
|
notebook = gtk_notebook_new ();
|
|
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (fmd_win)->vbox),
|
|
notebook, FALSE, FALSE, 0);
|
|
/*FIXME: I wan't a bigger, badder, better Icon here... */
|
|
icon = gnome_stock_pixmap_widget (hbox, GNOME_STOCK_PIXMAP_HELP);
|
|
gtk_box_pack_start (GTK_BOX (hbox), icon, FALSE, FALSE, 0);
|
|
vbox = gtk_vbox_new (FALSE, GNOME_PAD);
|
|
gtk_box_pack_start (GTK_BOX (hbox), vbox, TRUE, TRUE, 0);
|
|
gtk_container_set_border_width (GTK_CONTAINER (vbox), GNOME_PAD);
|
|
gtk_notebook_append_page (GTK_NOTEBOOK (notebook),
|
|
hbox,
|
|
gtk_label_new (_("Destination")));
|
|
alignment = gtk_alignment_new (0.0, 0.5, 0, 0);
|
|
label = gtk_label_new (text);
|
|
gtk_container_add (GTK_CONTAINER (alignment), label);
|
|
fentry = gnome_file_entry_new ("gmc-copy-file", _("Find Destination Folder"));
|
|
gnome_file_entry_set_directory (GNOME_FILE_ENTRY (fentry), TRUE);
|
|
gtk_entry_set_text (GTK_ENTRY (gnome_file_entry_gtk_entry (GNOME_FILE_ENTRY (fentry))),
|
|
def_text);
|
|
gnome_file_entry_set_default_path (GNOME_FILE_ENTRY (fentry), def_text);
|
|
cbox = gtk_check_button_new_with_label (_("Copy as a background process"));
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (cbox), *do_background);
|
|
gtk_signal_connect (GTK_OBJECT (cbox), "toggled", (GtkSignalFunc) fmd_check_box_callback, do_background);
|
|
#if 0
|
|
gnome_widget_add_help (cbox, "Selecting this will run the copying in the background. "
|
|
"This is useful for transfers over networks that might take a long "
|
|
"time to complete.");
|
|
#endif
|
|
gtk_box_pack_end (GTK_BOX (vbox), cbox, FALSE, FALSE, 0);
|
|
gtk_box_pack_end (GTK_BOX (vbox), gtk_hseparator_new (), FALSE, FALSE, 0);
|
|
gtk_box_pack_end (GTK_BOX (vbox), fentry, FALSE, FALSE, 0);
|
|
gnome_file_entry_set_modal(GNOME_FILE_ENTRY (fentry),TRUE);
|
|
|
|
gtk_box_pack_end (GTK_BOX (vbox), alignment, FALSE, FALSE, 0);
|
|
|
|
/* Advanced Options */
|
|
hbox = gtk_hbox_new (FALSE, GNOME_PAD_SMALL);
|
|
vbox = gtk_vbox_new (FALSE, GNOME_PAD_SMALL);
|
|
gtk_notebook_append_page (GTK_NOTEBOOK (notebook),
|
|
hbox,
|
|
gtk_label_new (_("Advanced Options")));
|
|
gtk_container_set_border_width (GTK_CONTAINER (hbox), GNOME_PAD);
|
|
gtk_box_pack_start (GTK_BOX (hbox), vbox, TRUE, TRUE, 0);
|
|
|
|
cbox = gtk_check_button_new_with_label (_("Preserve symlinks"));
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (cbox), ctx->stable_symlinks);
|
|
gtk_signal_connect (GTK_OBJECT (cbox), "toggled",
|
|
(GtkSignalFunc) fmd_check_box_callback, &ctx->stable_symlinks);
|
|
#if 0
|
|
gnome_widget_add_help (cbox, "FIXME: Add something here Miguel");
|
|
#endif
|
|
gtk_box_pack_start (GTK_BOX (vbox), cbox, FALSE, FALSE, 0);
|
|
|
|
if (operation == OP_COPY) {
|
|
cbox = gtk_check_button_new_with_label (_("Follow links."));
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (cbox), ctx->follow_links);
|
|
gtk_signal_connect (GTK_OBJECT (cbox), "toggled",
|
|
(GtkSignalFunc) fmd_check_box_callback, &ctx->follow_links);
|
|
gnome_widget_add_help (cbox,
|
|
_("Selecting this will copy the files that symlinks point "
|
|
"to instead of just copying the link."));
|
|
gtk_box_pack_start (GTK_BOX (vbox), cbox, FALSE, FALSE, 0);
|
|
|
|
cbox = gtk_check_button_new_with_label (_("Preserve file attributes."));
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (cbox), ctx->op_preserve);
|
|
gtk_signal_connect (GTK_OBJECT (cbox), "toggled",
|
|
(GtkSignalFunc) fmd_check_box_callback, &ctx->op_preserve);
|
|
gnome_widget_add_help (cbox, _("Preserves the permissions and the UID/GID if possible"));
|
|
gtk_box_pack_start (GTK_BOX (vbox), cbox, FALSE, FALSE, 0);
|
|
|
|
vbox = gtk_vbox_new (FALSE, GNOME_PAD_SMALL);
|
|
gtk_box_pack_start (GTK_BOX (hbox), vbox, TRUE, TRUE, 0);
|
|
cbox = gtk_check_button_new_with_label (_("Recursively copy subdirectories."));
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (cbox), ctx->dive_into_subdirs);
|
|
gtk_signal_connect (GTK_OBJECT (cbox), "toggled",
|
|
(GtkSignalFunc) fmd_check_box_callback, &ctx->dive_into_subdirs);
|
|
gnome_widget_add_help (cbox, _("If set, this will copy the directories recursively"));
|
|
gtk_box_pack_start (GTK_BOX (vbox), cbox, FALSE, FALSE, 0);
|
|
}
|
|
|
|
gtk_widget_show_all (GNOME_DIALOG (fmd_win)->vbox);
|
|
gtk_window_set_modal (GTK_WINDOW (fmd_win), FALSE);
|
|
gnome_dialog_set_close (GNOME_DIALOG (fmd_win), TRUE);
|
|
gnome_dialog_close_hides (GNOME_DIALOG (fmd_win), TRUE);
|
|
|
|
/* Off to the races!!! */
|
|
if (gnome_dialog_run (GNOME_DIALOG (fmd_win)) == 1) {
|
|
gtk_widget_destroy (fmd_win);
|
|
return NULL;
|
|
}
|
|
|
|
dest_dir = gnome_file_entry_get_full_path(GNOME_FILE_ENTRY (fentry), TRUE);
|
|
gtk_widget_destroy (fmd_win);
|
|
easy_patterns = 1;
|
|
if (!dest_dir || !*dest_dir)
|
|
return NULL;
|
|
|
|
if (ctx->follow_links && operation != OP_MOVE)
|
|
ctx->stat_func = mc_stat;
|
|
else
|
|
ctx->stat_func = mc_lstat;
|
|
|
|
if (ctx->op_preserve || operation == OP_MOVE){
|
|
ctx->preserve = 1;
|
|
ctx->umask_kill = 0777777;
|
|
ctx->preserve_uidgid = (geteuid () == 0) ? 1 : 0;
|
|
} else {
|
|
int i;
|
|
ctx->preserve = ctx->preserve_uidgid = 0;
|
|
i = umask (0);
|
|
umask (i);
|
|
ctx->umask_kill = i ^ 0777777;
|
|
}
|
|
source_mask = strdup ("*");
|
|
orig_mask = source_mask;
|
|
if (!dest_dir || !*dest_dir){
|
|
if (source_mask)
|
|
free (source_mask);
|
|
return dest_dir;
|
|
}
|
|
|
|
if (!dest_dir)
|
|
return NULL;
|
|
|
|
if (!*dest_dir) {
|
|
g_free (dest_dir);
|
|
return NULL;
|
|
}
|
|
|
|
if (source_easy_patterns) {
|
|
source_easy_patterns = easy_patterns;
|
|
easy_patterns = 1;
|
|
source_mask = convert_pattern (source_mask, match_file, 1);
|
|
easy_patterns = source_easy_patterns;
|
|
error = re_compile_pattern (source_mask, strlen (source_mask), &ctx->rx);
|
|
free (source_mask);
|
|
} else
|
|
error = re_compile_pattern (source_mask, strlen (source_mask), &ctx->rx);
|
|
|
|
if (error)
|
|
g_warning ("%s\n",error);
|
|
|
|
if (orig_mask)
|
|
free (orig_mask);
|
|
ctx->dest_mask = strrchr (dest_dir, PATH_SEP);
|
|
if (ctx->dest_mask == NULL)
|
|
ctx->dest_mask = dest_dir;
|
|
else
|
|
ctx->dest_mask++;
|
|
orig_mask = ctx->dest_mask;
|
|
if (!*ctx->dest_mask || (!ctx->dive_into_subdirs && !is_wildcarded (ctx->dest_mask) &&
|
|
(!only_one || (!mc_stat (dest_dir, &buf)
|
|
&& S_ISDIR (buf.st_mode)))) ||
|
|
(ctx->dive_into_subdirs && ((!only_one && !is_wildcarded (ctx->dest_mask)) ||
|
|
(only_one && !mc_stat (dest_dir, &buf)
|
|
&& S_ISDIR (buf.st_mode)))))
|
|
ctx->dest_mask = strdup ("*");
|
|
else {
|
|
ctx->dest_mask = strdup (ctx->dest_mask);
|
|
*orig_mask = 0;
|
|
}
|
|
if (!*dest_dir){
|
|
free (dest_dir);
|
|
dest_dir = strdup ("./");
|
|
}
|
|
return dest_dir;
|
|
}
|
|
|
|
static void
|
|
cancel_cb (GtkWidget *widget, gpointer data)
|
|
{
|
|
FileOpContextUI *ui;
|
|
|
|
ui = data;
|
|
ui->aborting = TRUE;
|
|
}
|
|
|
|
void
|
|
file_op_context_create_ui (FileOpContext *ctx, FileOperation op, int with_eta)
|
|
{
|
|
FileOpContextUI *ui;
|
|
GtkWidget *alignment;
|
|
GtkWidget *hbox;
|
|
|
|
g_return_if_fail (ctx != NULL);
|
|
g_return_if_fail (ctx->ui == NULL);
|
|
|
|
ui = g_new0 (FileOpContextUI, 1);
|
|
ctx->ui = ui;
|
|
|
|
switch (op) {
|
|
case OP_MOVE:
|
|
ui->op_win = gnome_dialog_new ("Move Progress", GNOME_STOCK_BUTTON_CANCEL, NULL);
|
|
break;
|
|
case OP_COPY:
|
|
ui->op_win = gnome_dialog_new ("Copy Progress", GNOME_STOCK_BUTTON_CANCEL, NULL);
|
|
break;
|
|
case OP_DELETE:
|
|
ui->op_win = gnome_dialog_new ("Delete Progress", GNOME_STOCK_BUTTON_CANCEL, NULL);
|
|
gtk_widget_show_all (GNOME_DIALOG (ui->op_win)->vbox);
|
|
gtk_widget_show_now (ui->op_win);
|
|
return;
|
|
}
|
|
|
|
gnome_dialog_button_connect (GNOME_DIALOG (ui->op_win), 0,
|
|
GTK_SIGNAL_FUNC (cancel_cb), ui);
|
|
|
|
alignment = gtk_alignment_new (0.0, 0.5, 0, 0);
|
|
hbox = gtk_hbox_new (FALSE, 0);
|
|
gtk_container_add (GTK_CONTAINER (alignment), hbox);
|
|
gtk_box_pack_start (GTK_BOX (hbox), gtk_label_new (_(GDIALOG_FROM_STRING)), FALSE, FALSE, 0);
|
|
ui->op_source_label = gtk_label_new ("");
|
|
|
|
gtk_box_pack_start (GTK_BOX (hbox), ui->op_source_label, FALSE, FALSE, 0);
|
|
gtk_box_set_spacing (GTK_BOX (GNOME_DIALOG (ui->op_win)->vbox), GNOME_PAD_SMALL);
|
|
gtk_container_set_border_width (GTK_CONTAINER (GNOME_DIALOG (ui->op_win)->vbox), GNOME_PAD);
|
|
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (ui->op_win)->vbox),
|
|
alignment, FALSE, FALSE, 0);
|
|
|
|
alignment = gtk_alignment_new (0.0, 0.5, 0, 0);
|
|
hbox = gtk_hbox_new (FALSE, 0);
|
|
gtk_container_add (GTK_CONTAINER (alignment), hbox);
|
|
gtk_box_pack_start (GTK_BOX (hbox), gtk_label_new (_(GDIALOG_TO_STRING)), FALSE, FALSE, 0);
|
|
ui->op_target_label = gtk_label_new ("");
|
|
gtk_box_pack_start (GTK_BOX (hbox), ui->op_target_label, FALSE, FALSE, 0);
|
|
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (ui->op_win)->vbox),
|
|
alignment, FALSE, FALSE, 0);
|
|
|
|
alignment = gtk_alignment_new (0.0, 0.5, 0, 0);
|
|
hbox = gtk_hbox_new (FALSE, 0);
|
|
gtk_box_pack_start (GTK_BOX (hbox), gtk_label_new (_("File ")), FALSE, FALSE, 0);
|
|
ui->count_label = GTK_OBJECT (gtk_label_new (""));
|
|
gtk_box_pack_start (GTK_BOX (hbox), GTK_WIDGET (ui->count_label), FALSE, FALSE, 0);
|
|
|
|
gtk_box_pack_start (GTK_BOX (hbox), gtk_label_new (_(" is ")), FALSE, FALSE, 0);
|
|
ui->file_label = gtk_label_new ("");
|
|
gtk_box_pack_start (GTK_BOX (hbox), ui->file_label, FALSE, FALSE, 0);
|
|
gtk_box_pack_start (GTK_BOX (hbox), gtk_label_new (_(" Done.")), FALSE, FALSE, 0);
|
|
|
|
gtk_container_add (GTK_CONTAINER (alignment), hbox);
|
|
|
|
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (ui->op_win)->vbox),
|
|
alignment, FALSE, FALSE, 0);
|
|
|
|
ui->byte_prog = GTK_OBJECT (gtk_progress_bar_new ());
|
|
gtk_widget_set_usize (GTK_WIDGET (ui->byte_prog), GDIALOG_PROGRESS_WIDTH, -1);
|
|
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (ui->op_win)->vbox),
|
|
GTK_WIDGET (ui->byte_prog), FALSE, FALSE, 0);
|
|
|
|
/*done with things */
|
|
gtk_widget_show_all (GNOME_DIALOG (ui->op_win)->vbox);
|
|
gtk_widget_show_now (ui->op_win);
|
|
}
|
|
|
|
void
|
|
file_op_context_destroy_ui (FileOpContext *ctx)
|
|
{
|
|
FileOpContextUI *ui;
|
|
|
|
g_return_if_fail (ctx != NULL);
|
|
g_return_if_fail (ctx->ui != NULL);
|
|
|
|
ui = ctx->ui;
|
|
|
|
gtk_widget_destroy (ui->op_win);
|
|
|
|
g_free (ui);
|
|
ctx->ui = NULL;
|
|
}
|
|
|
|
void
|
|
fmd_init_i18n (int force)
|
|
{
|
|
/* unneccessary func */
|
|
}
|