mc/gnome/gmain.c
Pavel Roskin 74f034e8d4 * gmain.c (gmc_window_setup_from_panel): Fix crash when called
early, before the panel is created.
2001-07-15 06:25:16 +00:00

577 lines
12 KiB
C

/*
* Midnight Commander -- GNOME frontend
*
* Copyright (C) 1997, 1998 The Free Software Foundation
*
* Author: Miguel de Icaza (miguel@gnu.org)
*
*/
#include <config.h>
#include <stdarg.h>
#include <string.h>
#include <ctype.h>
#include <signal.h>
#define WANT_WIDGETS /* bleah */
#include "tty.h" /* for KEY_BACKSPACE */
#include "x.h"
#include "main.h"
#include "key.h"
#include "global.h"
#include "dir.h"
#include "panel.h"
#include "gscreen.h"
#include "gcmd.h"
#include "gcorba.h"
#include "gdesktop.h"
#include "gsession.h"
#include "command.h"
#include "cmd.h"
#include <gdk/gdkx.h>
#ifdef HAVE_GNOME_WINDOW_ICON
#include <libgnomeui/gnome-window-icon.h>
#endif
GdkColorContext *mc_cc;
#define MAX_COLOR_PAIRS 40
struct gmc_color_pairs_s gmc_color_pairs [MAX_COLOR_PAIRS];
char *default_edition_colors =
"normal=black:"
"selected=white,darkblue:"
"viewunderline=red,blue:"
"directory=blue:"
"markselect=yellow,darkblue:"
"marked=yellow,seagreen:"
"execute=slateblue:"
"link=green:"
"stalledlink=red:"
"device=magenta:"
"core=red:"
"menuhotsel=cyan,black:"
"errors=white,red:"
"reverse=black,lightcyan:"
"special=black";
int dialog_panel_callback (struct Dlg_head *h, int id, int msg);
/* The Dlg_head for the whole desktop */
Dlg_head *desktop_dlg;
/* Session client */
GnomeClient *session_client;
/* Used during argument processing */
extern int finish_program;
/* Whether the gmc server was present or not */
int corba_have_server = FALSE;
void
init_pair (int index, GdkColor *fore, GdkColor *back)
{
if (index < 0 || index > MAX_COLOR_PAIRS){
printf ("init_pair called with invalid index\n");
exit (1);
}
gmc_color_pairs [index].fore = fore;
gmc_color_pairs [index].back = back;
}
void
get_color (char *cpp, GdkColor **colp)
{
GdkColor *new_color;
int result;
gint status;
new_color = g_new (GdkColor, 1);
result = gdk_color_parse (cpp, new_color);
new_color->pixel = 0;
if (result) {
status = 0;
gdk_color_context_get_pixels (mc_cc,
&new_color->red, &new_color->green, &new_color->blue,
1, &new_color->pixel, &status);
} else
new_color->red = new_color->green = new_color->blue = 0;
*colp = new_color;
}
static void
gmc_color_init (void)
{
mc_cc = gdk_color_context_new (gtk_widget_get_default_visual (),
gtk_widget_get_default_colormap ());
}
int
xtoolkit_init (int *argc, char *argv [])
{
signal (SIGTTIN, SIG_IGN);
signal (SIGTTOU, SIG_IGN);
gmc_color_init ();
return 0;
}
void
interactive_display (char *filename, char *node)
{
/* FIXME: Implement gnome version */
}
void
xtoolkit_end (void)
{
/* Do nothing */
}
/*
* Keystroke event handler for all of the Gtk widgets created by
* the GNOME Midnight Commander. A special case is handled at
* the top
*/
int
dialog_key_pressed (GtkWidget *win, GdkEventKey *event, Dlg_head *h)
{
GtkWidget *w;
static int on_escape;
int key;
/*
* Find out if the focused widget is an IconList and
* if so, check if it has a currently focused item is
* on editing mode as we do not want to handle key
* events while the icon name is being edited.
*/
w = win;
while (w && (GTK_IS_CONTAINER (w) && !GNOME_IS_ICON_LIST (w)))
w = GTK_CONTAINER (w)->focus_child;
if (w && GNOME_IS_ICON_LIST (w)){
GnomeCanvas *c = GNOME_CANVAS (w);
if (c->focused_item && GNOME_IS_ICON_TEXT_ITEM (c->focused_item)){
GnomeIconTextItem *i = GNOME_ICON_TEXT_ITEM (c->focused_item);
if (i->editing)
return FALSE;
}
}
key = translate_gdk_keysym_to_curses (event);
if (key == -1)
return FALSE;
if (!on_escape){
if (key == 27){
on_escape = 1;
gtk_signal_emit_stop_by_name (GTK_OBJECT (win), "key_press_event");
return TRUE;
}
} else {
if (key != 27){
if (key >= '0' && key <= '9')
key = KEY_F(key - '0');
else
key = ALT (key);
if (key == ALT('<'))
key = KEY_HOME;
if (key == ALT('>'))
key = KEY_END;
}
on_escape = 0;
}
if (dlg_key_event (h, key)){
gtk_signal_emit_stop_by_name (GTK_OBJECT (win), "key_press_event");
return TRUE;
} else
return FALSE;
}
void
bind_gtk_keys (GtkWidget *w, Dlg_head *h)
{
gtk_signal_connect (GTK_OBJECT (w),
"key_press_event",
GTK_SIGNAL_FUNC (dialog_key_pressed),
h);
}
widget_data
xtoolkit_create_dialog (Dlg_head *h, int flags)
{
GtkWidget *win, *ted;
if (!(flags & DLG_NO_TOPLEVEL)){
if (flags & DLG_GNOME_APP)
win = gnome_app_new ("mc", h->name);
else {
win = gtk_window_new (GTK_WINDOW_TOPLEVEL);
gtk_window_set_position (GTK_WINDOW (win), GTK_WIN_POS_MOUSE);
#ifdef HAVE_GNOME_WINDOW_ICON
gnome_window_icon_set_from_default (GTK_WINDOW (win));
#endif
}
} else
win = 0;
h->grided = flags;
h->idle_fn_tag = -1;
if (!(flags & DLG_NO_TED)){
char *layout = concat_dir_and_file (mc_home, "layout");
g_warning ("Should never use GtkTed!!! Write a real dialog!!!");
ted = gtk_ted_new_layout (h->name, layout);
g_free (layout);
gtk_container_add (GTK_CONTAINER (win), ted);
gtk_widget_show (ted);
bind_gtk_keys (GTK_WIDGET (ted), h);
}
if (win){
bind_gtk_keys (GTK_WIDGET (win), h);
}
return (widget_data) win;
}
/* Used to bind a window for an already created Dlg_head. This is
* used together with the DLG_NO_TOPLEVEL: the dialog is created
* with the DLG_NO_TOPLEVEL and later, when the window is created
* it is assigned with this routine
*/
void
x_dlg_set_window (Dlg_head *h, GtkWidget *win)
{
h->wdata = (widget_data) win;
bind_gtk_keys (GTK_WIDGET (win), h);
}
void
x_set_dialog_title (Dlg_head *h, const char *title)
{
gtk_window_set_title (GTK_WINDOW (h->wdata), title);
}
widget_data
xtoolkit_get_main_dialog (Dlg_head *h)
{
GtkWidget *win;
win = gtk_window_new (GTK_WINDOW_TOPLEVEL);
#ifdef HAVE_GNOME_WINDOW_ICON
gnome_window_icon_set_from_default (GTK_WINDOW (win));
#endif
return (widget_data) win;
}
/* Creates the containers */
widget_data
x_create_panel_container (int which)
{
GtkWidget *win;
win = gtk_window_new (GTK_WINDOW_TOPLEVEL);
#ifdef HAVE_GNOME_WINDOW_ICON
gnome_window_icon_set_from_default (GTK_WINDOW (win));
#endif
return (widget_data) 0;
}
void
x_panel_container_show (widget_data wdata)
{
gtk_widget_show (GTK_WIDGET (wdata));
}
void
x_add_widget (Dlg_head *h, Widget_Item *w)
{
if (!(h->grided & DLG_NO_TED)){
GtkTed *ted = GTK_TED (GTK_BIN (h->wdata)->child);
g_warning ("Should never use GtkTed!!! Write a real dialog!!!");
gtk_ted_add (ted, GTK_WIDGET (w->widget->wdata), w->widget->tkname);
bind_gtk_keys (GTK_WIDGET (w->widget->wdata), h);
}
}
static int
gnome_dlg_send_destroy (GtkWidget *widget, GdkEvent *event, Dlg_head *h)
{
gtk_widget_hide (GTK_WIDGET (h->wdata));
h->ret_value = B_CANCEL;
dlg_stop (h);
return TRUE;
}
void
x_init_dlg (Dlg_head *h)
{
if (!(h->grided & DLG_NO_TED)){
GtkTed *ted = GTK_TED (GTK_BIN (h->wdata)->child);
Widget_Item *p, *first;
g_warning ("Should never use GtkTed!!! Write a real dialog!!!");
first = p = h->current;
do {
gtk_ted_add (ted, GTK_WIDGET (p->widget->wdata), p->widget->tkname);
bind_gtk_keys (GTK_WIDGET (p->widget->wdata), h);
p = p->next;
} while (p != first);
gtk_ted_prepare (ted);
if (!ted->need_gui){
gtk_grab_add (GTK_WIDGET (ted));
gtk_window_set_policy (GTK_WINDOW (h->wdata), 0, 0, 0);
}
gtk_widget_show (GTK_WIDGET (h->wdata));
}
gtk_signal_connect (GTK_OBJECT ((void *)h->wdata), "delete_event",
GTK_SIGNAL_FUNC (gnome_dlg_send_destroy), h);
if (h->current)
x_focus_widget (h->current);
}
/*
* This function is invoked when the dialog is started to be
* destroyed, before any widgets have been destroyed.
*
* We only hide the toplevel Gtk widget to avoid the flickering
* of the destruction process
*/
void
x_destroy_dlg_start (Dlg_head *h)
{
gtk_widget_hide (GTK_WIDGET (h->wdata));
}
/*
* Called when the Dlg_head has been destroyed. This only cleans
* up/releases the frontend resources
*/
void
x_destroy_dlg (Dlg_head *h)
{
if (!(h->grided & DLG_NO_TED))
gtk_grab_remove (GTK_WIDGET (GTK_BIN (h->wdata)->child));
if (h->wdata){
gtk_widget_destroy (GTK_WIDGET(h->wdata));
h->wdata = 0;
}
}
void
gtkrundlg_event (Dlg_head *h)
{
gtk_main ();
}
void
edition_pre_exec ()
{
}
void
edition_post_exec ()
{
}
void
done_screen ()
{
}
void
setup_sigwinch ()
{
}
void
x_flush_events (void)
{
while (gtk_events_pending ())
gtk_main_iteration ();
}
static int
gnome_idle_handler (gpointer data)
{
Dlg_head *h = data;
if (h->send_idle_msg){
(*h->callback)(h, 0, DLG_IDLE);
return TRUE;
} else
return FALSE;
}
/* Turn on and off the idle message sending */
void
x_set_idle (Dlg_head *h, int enable_idle)
{
if (enable_idle){
if (h->idle_fn_tag != -1)
return;
h->idle_fn_tag = gtk_idle_add (gnome_idle_handler, h);
} else {
if (h->idle_fn_tag == -1)
return;
gtk_idle_remove (h->idle_fn_tag);
h->idle_fn_tag = -1;
gnome_idle_handler (h);
}
}
int
dialog_panel_callback (struct Dlg_head *h, int id, int msg)
{
Widget_Item *dh;
void *current_widget; /* The current widget */
if (msg == DLG_KEY && id == '\n'){
current_widget = (void *) h->current->widget;
if (is_a_panel (current_widget))
return 0;
dh = h->current;
do {
if (is_a_panel (dh->widget)){
WPanel *p = (WPanel *) dh->widget;
if (current_widget == p->current_dir){
WInput *in = p->current_dir;
do_panel_cd (p, in->buffer, cd_parse_command);
assign_text (in, p->cwd);
update_input (in, 1);
return MSG_HANDLED;
}
}
dh = dh->next;
} while (dh != h->current);
}
if (msg == DLG_UNHANDLED_KEY || msg == DLG_HOTKEY_HANDLED)
return midnight_callback (h, id, msg);
return 0;
}
/* Initializes the desktop and creates the initial panels. This is to be used
* when we do not have a CORBA server.
*/
static void
non_corba_create_panels (char *startup_dir)
{
desktop_init ();
gnome_init_panels ();
the_hint = label_new (0, 0, 0, NULL);
desktop_dlg = create_dlg (0, 0, 24, 80, 0,
dialog_panel_callback, "[panel]", "midnight", DLG_NO_TED);
session_load ();
run_dlg (desktop_dlg);
desktop_destroy ();
}
/* The GNOME version of create_panels() */
void
create_panels (void)
{
corba_activate_server ();
if (!corba_have_server)
non_corba_create_panels (this_dir ? this_dir : ".");
else {
if (!nowindows)
corba_create_window (this_dir);
session_set_restart (FALSE);
}
}
void
gmc_do_quit (void)
{
/* FIXME: This wont get us out from a dialog box */
gtk_main_quit ();
quit = 1;
dlg_stop (desktop_dlg);
}
/*
* Configures the GtkWindow/GnomeDialog from a WPanel.
*
* This makes the window centered on the screen and binds it to
* its parent container for better window manager experience
*/
void
gmc_window_setup_from_panel (GnomeDialog *dialog, WPanel *panel)
{
gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_MOUSE);
if (panel && panel->xwindow) {
gnome_dialog_set_parent (dialog, GTK_WINDOW (panel->xwindow));
}
}
/**
* gnome_check_super_user:
* @void:
*
* Puts out a warning if the user is running gmc as root. If the user selects
* Cancel, then gmc will terminate.
**/
void
gnome_check_super_user (void)
{
GtkWidget *warning_dlg;
GtkWidget *cbox;
gboolean show_dialog;
if (geteuid () != 0)
return;
gnome_config_push_prefix( "/gmc/Root/");
show_dialog = gnome_config_get_bool ("show_dialog=true");
gnome_config_pop_prefix ();
if (!show_dialog)
return;
cbox = gtk_check_button_new_with_label (_("Don't show this window again"));
warning_dlg = gnome_message_box_new (
_("You are running the GNOME File Manager as root.\n\n"
"As root, you can damage your system, and the "
"File Manager will not stop you.\n"
"Your manual explains how to add a non-privileged user "
"account to the system.\n"),
GNOME_MESSAGE_BOX_WARNING,
GNOME_STOCK_BUTTON_OK, NULL);
gtk_widget_show (cbox);
gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (warning_dlg)->vbox), cbox, FALSE, FALSE, 0);
gnome_dialog_close_hides (GNOME_DIALOG (warning_dlg), TRUE);
gnome_dialog_run_and_close (GNOME_DIALOG (warning_dlg));
if (GTK_TOGGLE_BUTTON (cbox)->active) {
gnome_config_push_prefix( "/gmc/Root/");
gnome_config_set_bool ("show_dialog", FALSE);
gnome_config_pop_prefix ();
}
gnome_config_sync ();
gtk_widget_destroy (warning_dlg);
}