mc/src/screen.c

2698 lines
63 KiB
C
Raw Normal View History

1998-02-27 07:54:42 +03:00
/* Panel managing.
Copyright (C) 1994, 1995 Miguel de Icaza.
1998-02-27 07:54:42 +03:00
This program 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.
1998-02-27 07:54:42 +03:00
This program 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.
Written by: 1995 Miguel de Icaza
1997, 1999 Timur Bakeyev
1998-02-27 07:54:42 +03:00
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
#include <config.h>
#include "tty.h"
#include <sys/param.h>
#include <string.h>
#include <stdio.h>
#include <errno.h>
#ifdef HAVE_UNISTD_H
# include <unistd.h> /* For chdir(), readlink() and getwd()/getcwd() */
#endif
#include "global.h"
#include "dir.h"
#include "panel.h"
#include "color.h"
#include "tree.h"
#include "win.h"
#include "main.h"
#include "ext.h" /* regexp_command */
#include "mouse.h" /* For Gpm_Event */
#include "cons.saver.h" /* For console_flag */
#include "layout.h" /* Most layout variables are here */
#include "dialog.h" /* for message (...) */
#include "cmd.h"
#include "key.h" /* XCTRL and ALT macros */
#include "setup.h" /* For loading/saving panel options */
#include "user.h"
#include "profile.h"
#include "widget.h"
#include "../vfs/vfs.h"
#include "../vfs/extfs.h"
#ifdef NEEDS_DRIVE_H
1998-02-27 07:54:42 +03:00
# include "drive.h"
#endif
#include "x.h"
#define ELEMENTS(arr) ( sizeof(arr) / sizeof((arr)[0]) )
/* If true, show the mini-info on the panel */
int show_mini_info = 1;
/* If true, then use stat() on the cwd to determine directory changes */
int fast_reload = 0;
/* If true, use some usability hacks by Torben */
int torben_fj_mode = 0;
/* If true, up/down keys scroll the pane listing by pages */
int panel_scroll_pages = 1;
/* If 1, we use permission hilighting */
int permission_mode = 0;
/* If 1 - then add per file type hilighting */
int filetype_mode = 1;
/* This gives abilitiy to determine colored user priveleges */
extern user_in_groups *current_user_gid;
extern uid_t current_user_uid;
/* If we have an info panel, this points to it */
WPanel *the_info_panel = 0;
/* The hook list for the select file function */
Hook *select_file_hook = 0;
static int panel_callback (Dlg_head *h, WPanel *p, int Msg, int Par);
1998-02-27 07:54:42 +03:00
int panel_event (Gpm_Event *event, WPanel *panel);
#ifndef PORT_HAS_PANEL_ADJUST_TOP_FILE
1998-02-27 07:54:42 +03:00
# define x_adjust_top_file(p)
#endif
#ifndef PORT_HAS_PANEL_RESET_SORT_LABELS
# define x_reset_sort_labels(x)
1998-02-27 07:54:42 +03:00
#endif
/* This macro extracts the number of available lines in a panel */
#ifndef PORT_HAS_LLINES
#define llines(p) (p->widget.lines-3 - (show_mini_info ? 2 : 0))
#else
#define llines(p) (p->widget.lines)
#endif
#ifdef PORT_NOT_FOCUS_SELECT_ITEM
# define focus_select_item(x)
#else
# define focus_select_item(x) select_item(x)
#endif
#ifdef PORT_NOT_UNFOCUS_UNSELECT_ITEM
# define unfocus_unselect_item(x)
#else
# define unfocus_unselect_item(x) unselect_item(x)
#endif
1998-02-27 07:54:42 +03:00
#ifdef HAVE_X
# define set_colors(x)
#else
# define x_create_panel(x,y,z) 1;
# define x_panel_load_index(p,x)
# define x_panel_select_item(a,b,c)
Ok, most of the "Elliot Lee confidential bug report" has been dealt with with this commit. This also addresses a bunch of the comments from the status.shtml from DrMike. Miguel. 1998-04-15 Miguel de Icaza <miguel@nuclecu.unam.mx> * screen.c (GT): Assign two spaces for the minimum size of the "type" field for the GNOME edition. This gives some extra space for the icon that gets displayed. * dlg.c (remove_widget): New function: used to remove a widget from an existing Dlg_head; (destroy_widget): Destroy a specific Widget. (add_widgetl): Extended to deal with the fact that a running Dlg_head can become empty. * panelize.c (l_call): Update the input line every time the user selects the entry with the mouse (pretty common in the gnome edition). * hotlist.c (add_new_group_input): Removed an extra field that was causing problems. * find.c (find_parameters): Tree button is gone for gnome until we get the tree function working on gnome. * cmd.c (save_setup_cmd): Per Elliot's suggestion, do not pop up a dialog box to inform the user about the saved setup. 1998-04-15 Miguel de Icaza <miguel@nuclecu.unam.mx> * gcmd.c (gnome_close_panel): Implement the close-the-panel functionality. * gscreen.c (x_panel_destroy): Implement the gnome mode way of destroying the panel. * gview.c (view_status): Add cacheing of the information status, to avoid excessive flicker. It would be better if GtkLabel did not flicker though. (scrollbar_moved): Scroll correctly. We now use view_move_backward and view_move_forward to adjust the scrollbar contents. This displays the scrollbar correctly. * gwidget.c (x_listbox_select_nth): This may be invoked before the widget has been created, work around this problem. * gscreen.c (show_dir): Set the title bar for the window to the current directoy. Reported by both Mike and Elliot. * layout: Updated to the new hotlist dialog box.
1998-04-16 06:45:53 +04:00
# define x_panel_destroy(p)
1998-02-27 07:54:42 +03:00
void
set_colors (WPanel *panel)
{
standend ();
if (hascolors)
attrset (NORMAL_COLOR);
}
#endif
#ifndef ICONS_PER_ROW
# define ICONS_PER_ROW(x) 1
#endif
1998-02-27 07:54:42 +03:00
/* Delete format string, it is a linked list */
1998-12-03 00:27:27 +03:00
static void
1998-02-27 07:54:42 +03:00
delete_format (format_e *format)
{
format_e *next;
1998-02-27 07:54:42 +03:00
while (format){
next = format->next;
g_free (format);
1998-02-27 07:54:42 +03:00
format = next;
}
}
#ifndef HAVE_X
1998-02-27 07:54:42 +03:00
/* This code relies on the default justification!!! */
1998-12-03 00:27:27 +03:00
static void
1998-02-27 07:54:42 +03:00
add_permission_string (char *dest, int width, file_entry *fe, int attr, int color, int is_octal)
{
int i, r, l;
1998-02-27 07:54:42 +03:00
l = get_user_rights (&fe->buf);
1998-02-27 07:54:42 +03:00
if (is_octal){
/* Place of the access bit in octal mode */
l = width + l - 3;
r = l + 1;
} else {
/* The same to the triplet in string mode */
l = l * 3 + 1;
r = l + 3;
}
1998-02-27 07:54:42 +03:00
for(i = 0; i < width; i++){
if (i >= l && i < r){
if (attr == SELECTED || attr == MARKED_SELECTED)
attrset (MARKED_SELECTED_COLOR);
else
attrset (MARKED_COLOR);
} else
attrset (color);
1998-02-27 07:54:42 +03:00
addch (dest[i]);
}
}
#endif /* HAVE_X */
1998-02-27 07:54:42 +03:00
/* String representations of various file attributes */
/* name */
char *
string_file_name (file_entry *fe, int len)
{
return fe->fname;
}
/* size */
char *
string_file_size (file_entry *fe, int len)
{
static char buffer [BUF_TINY];
int i;
#ifdef HAVE_ST_RDEV
if (S_ISBLK (fe->buf.st_mode) || S_ISCHR (fe->buf.st_mode))
g_snprintf (buffer, sizeof (buffer), "%3d,%3d",
(int) ((fe->buf.st_rdev >> 8) & 0xff),
(int) (fe->buf.st_rdev & 0xff));
else
#endif
{
g_snprintf (buffer, sizeof (buffer), "%lu", (unsigned long) fe->buf.st_size);
if (len && (i = strlen (buffer)) > len) {
if (i - 2 > len) {
if (i - 5 > len)
g_snprintf (buffer, sizeof (buffer), "%luG", (unsigned long) ((fe->buf.st_size) >> 30));
else
g_snprintf (buffer, sizeof (buffer), "%luM", (unsigned long) ((fe->buf.st_size) >> 20));
} else
g_snprintf (buffer, sizeof (buffer), "%luK", (unsigned long) ((fe->buf.st_size) >> 10));
}
}
return buffer;
}
/* bsize */
char *
string_file_size_brief (file_entry *fe, int len)
{
static char buffer [BUF_TINY];
if (S_ISDIR (fe->buf.st_mode)){
2000-04-18 12:58:42 +04:00
strcpy (buffer, _(strcmp (fe->fname, "..") ? N_("SUB-DIR") : N_("UP--DIR")));
return buffer;
}
return string_file_size (fe, len);
}
1998-02-27 07:54:42 +03:00
/* This functions return a string representation of a file entry */
/* type */
1998-02-27 07:54:42 +03:00
char *
string_file_type (file_entry *fe, int len)
{
static char buffer [2];
if (S_ISDIR (fe->buf.st_mode))
buffer [0] = PATH_SEP;
else if (S_ISLNK (fe->buf.st_mode)) {
if (fe->f.link_to_dir)
buffer [0] = '~';
else if (fe->f.stalled_link)
buffer [0] = '!';
else
buffer [0] = '@';
} else if (S_ISSOCK (fe->buf.st_mode))
buffer [0] = '=';
else if (S_ISCHR (fe->buf.st_mode))
buffer [0] = '-';
else if (S_ISBLK (fe->buf.st_mode))
buffer [0] = '+';
else if (S_ISFIFO (fe->buf.st_mode))
buffer [0] = '|';
else if (is_exe (fe->buf.st_mode))
buffer [0] = '*';
else
buffer [0] = ' ';
buffer [1] = '\0';
1998-02-27 07:54:42 +03:00
return buffer;
}
/* mtime */
1998-02-27 07:54:42 +03:00
char *
string_file_mtime (file_entry *fe, int len)
1998-02-27 07:54:42 +03:00
{
#ifdef PORT_STATIC_IN_STRING_FILE_XTIME
static char timebuf [MAX_I18NTIMELENGTH + 1];
return strcpy (timebuf, file_date (fe->buf.st_mtime));
#else
return file_date (fe->buf.st_mtime);
#endif
}
1998-02-27 07:54:42 +03:00
/* atime */
char *
string_file_atime (file_entry *fe, int len)
{
#ifdef PORT_STATIC_IN_STRING_FILE_XTIME
static char timebuf [MAX_I18NTIMELENGTH + 1];
return strcpy (timebuf, file_date (fe->buf.st_atime));
#else
return file_date (fe->buf.st_atime);
#endif
}
1998-02-27 07:54:42 +03:00
/* ctime */
char *
string_file_ctime (file_entry *fe, int len)
{
#ifdef PORT_STATIC_IN_STRING_FILE_XTIME
static char timebuf [MAX_I18NTIMELENGTH + 1];
return strcpy (timebuf, file_date (fe->buf.st_ctime));
#else
return file_date (fe->buf.st_ctime);
#endif
1998-02-27 07:54:42 +03:00
}
/* perm */
1998-02-27 07:54:42 +03:00
char *
string_file_permission (file_entry *fe, int len)
{
return string_perm (fe->buf.st_mode);
}
/* mode */
1998-02-27 07:54:42 +03:00
char *
string_file_perm_octal (file_entry *fe, int len)
1998-02-27 07:54:42 +03:00
{
static char buffer [10];
1998-02-27 07:54:42 +03:00
g_snprintf (buffer, sizeof (buffer), "0%06o", fe->buf.st_mode);
1998-02-27 07:54:42 +03:00
return buffer;
}
/* nlink */
1998-02-27 07:54:42 +03:00
char *
string_file_nlinks (file_entry *fe, int len)
1998-02-27 07:54:42 +03:00
{
static char buffer [BUF_TINY];
1998-02-27 07:54:42 +03:00
g_snprintf (buffer, sizeof (buffer), "%16d", fe->buf.st_nlink);
1998-02-27 07:54:42 +03:00
return buffer;
}
/* inode */
1998-02-27 07:54:42 +03:00
char *
string_inode (file_entry *fe, int len)
1998-02-27 07:54:42 +03:00
{
static char buffer [10];
1998-02-27 07:54:42 +03:00
g_snprintf (buffer, sizeof (buffer), "%lu", (unsigned long) fe->buf.st_ino);
return buffer;
1998-02-27 07:54:42 +03:00
}
/* nuid */
1998-02-27 07:54:42 +03:00
char *
string_file_nuid (file_entry *fe, int len)
1998-02-27 07:54:42 +03:00
{
static char buffer [10];
1998-02-27 07:54:42 +03:00
g_snprintf (buffer, sizeof (buffer), "%d", fe->buf.st_uid);
return buffer;
}
/* ngid */
char *
string_file_ngid (file_entry *fe, int len)
{
static char buffer [10];
g_snprintf (buffer, sizeof (buffer), "%d", fe->buf.st_gid);
return buffer;
1998-03-07 03:05:06 +03:00
}
/* owner */
char *
string_file_owner (file_entry *fe, int len)
{
return get_owner (fe->buf.st_uid);
}
/* group */
char *
string_file_group (file_entry *fe, int len)
{
return get_group (fe->buf.st_gid);
}
/* mark */
1998-02-27 07:54:42 +03:00
char *
string_marked (file_entry *fe, int len)
1998-02-27 07:54:42 +03:00
{
return fe->f.marked ? "*" : " ";
1998-02-27 07:54:42 +03:00
}
/* space */
1998-02-27 07:54:42 +03:00
char *
string_space (file_entry *fe, int len)
{
return " ";
}
/* dot */
1998-02-27 07:54:42 +03:00
char *
string_dot (file_entry *fe, int len)
{
return ".";
}
Ok, most of the "Elliot Lee confidential bug report" has been dealt with with this commit. This also addresses a bunch of the comments from the status.shtml from DrMike. Miguel. 1998-04-15 Miguel de Icaza <miguel@nuclecu.unam.mx> * screen.c (GT): Assign two spaces for the minimum size of the "type" field for the GNOME edition. This gives some extra space for the icon that gets displayed. * dlg.c (remove_widget): New function: used to remove a widget from an existing Dlg_head; (destroy_widget): Destroy a specific Widget. (add_widgetl): Extended to deal with the fact that a running Dlg_head can become empty. * panelize.c (l_call): Update the input line every time the user selects the entry with the mouse (pretty common in the gnome edition). * hotlist.c (add_new_group_input): Removed an extra field that was causing problems. * find.c (find_parameters): Tree button is gone for gnome until we get the tree function working on gnome. * cmd.c (save_setup_cmd): Per Elliot's suggestion, do not pop up a dialog box to inform the user about the saved setup. 1998-04-15 Miguel de Icaza <miguel@nuclecu.unam.mx> * gcmd.c (gnome_close_panel): Implement the close-the-panel functionality. * gscreen.c (x_panel_destroy): Implement the gnome mode way of destroying the panel. * gview.c (view_status): Add cacheing of the information status, to avoid excessive flicker. It would be better if GtkLabel did not flicker though. (scrollbar_moved): Scroll correctly. We now use view_move_backward and view_move_forward to adjust the scrollbar contents. This displays the scrollbar correctly. * gwidget.c (x_listbox_select_nth): This may be invoked before the widget has been created, work around this problem. * gscreen.c (show_dir): Set the title bar for the window to the current directoy. Reported by both Mike and Elliot. * layout: Updated to the new hotlist dialog box.
1998-04-16 06:45:53 +04:00
#ifdef HAVE_GNOME
# define GT 2
#else
# define GT 1
#endif
1998-02-27 07:54:42 +03:00
static struct {
char *id;
int min_size;
int expands;
int default_just;
char *title;
int use_in_gui;
char *(*string_fn)(file_entry *, int);
sortfn *sort_routine;
} formats [] = {
{ "name", 12, 1, J_LEFT_FIT, N_("Name"), 1, string_file_name, (sortfn *) sort_name },
{ "size", 7, 0, J_RIGHT, N_("Size"), 1, string_file_size, (sortfn *) sort_size },
{ "bsize", 7, 0, J_RIGHT, N_("Size"), 1, string_file_size_brief, (sortfn *) sort_size },
{ "type", GT, 0, J_LEFT, "", 2, string_file_type, (sortfn *) sort_type },
{ "mtime", 12, 0, J_RIGHT, N_("MTime"), 1, string_file_mtime, (sortfn *) sort_time },
{ "atime", 12, 0, J_RIGHT, N_("ATime"), 1, string_file_atime, (sortfn *) sort_atime },
{ "ctime", 12, 0, J_RIGHT, N_("CTime"), 1, string_file_ctime, (sortfn *) sort_ctime },
{ "perm", 10, 0, J_LEFT, N_("Permission"),1,string_file_permission, NULL },
{ "mode", 6, 0, J_RIGHT, N_("Perm"), 1, string_file_perm_octal, NULL },
{ "nlink", 2, 0, J_RIGHT, N_("Nl"), 1, string_file_nlinks, (sortfn *) sort_links },
{ "inode", 5, 0, J_RIGHT, N_("Inode"), 1, string_inode, (sortfn *) sort_inode },
{ "nuid", 5, 0, J_RIGHT, N_("UID"), 1, string_file_nuid, (sortfn *) sort_nuid },
{ "ngid", 5, 0, J_RIGHT, N_("GID"), 1, string_file_ngid, (sortfn *) sort_ngid },
{ "owner", 8, 0, J_LEFT_FIT, N_("Owner"), 1, string_file_owner, (sortfn *) sort_owner },
{ "group", 8, 0, J_LEFT_FIT, N_("Group"), 1, string_file_group, (sortfn *) sort_group },
{ "mark", 1, 0, J_RIGHT, " ", 1, string_marked, NULL },
{ "|", 1, 0, J_RIGHT, " ", 0, NULL, NULL },
{ "space", 1, 0, J_RIGHT, " ", 0, string_space, NULL },
{ "dot", 1, 0, J_RIGHT, " ", 0, string_dot, NULL },
1998-02-27 07:54:42 +03:00
};
static char *
to_buffer (char *dest, int just_mode, int len, char *txt)
{
int txtlen = strlen (txt);
int still, over;
1998-02-27 07:54:42 +03:00
/* Fill buffer with spaces */
memset (dest, ' ', len);
still = (over=(txtlen > len)) ? (txtlen - len) : (len - txtlen);
switch (HIDE_FIT(just_mode)){
case J_LEFT:
still = 0;
break;
case J_CENTER:
still /= 2;
break;
case J_RIGHT:
default:
break;
}
if (over){
if (IS_FIT(just_mode))
strcpy (dest, name_trunc(txt, len));
else
strncpy (dest, txt+still, len);
} else
strncpy (dest+still, txt, txtlen);
1999-02-01 03:08:16 +03:00
dest[len] = '\0';
return (dest + len);
1998-02-27 07:54:42 +03:00
}
int
file_compute_color (int attr, file_entry *fe)
{
switch (attr){
case SELECTED:
return (SELECTED_COLOR);
1998-02-27 07:54:42 +03:00
case MARKED:
return (MARKED_COLOR);
1998-02-27 07:54:42 +03:00
case MARKED_SELECTED:
return (MARKED_SELECTED_COLOR);
1998-02-27 07:54:42 +03:00
case STATUS:
return (NORMAL_COLOR);
1998-02-27 07:54:42 +03:00
case NORMAL:
default:
if (!filetype_mode)
return (NORMAL_COLOR);
1998-02-27 07:54:42 +03:00
}
/* if filetype_mode == true */
if (S_ISDIR (fe->buf.st_mode))
return (DIRECTORY_COLOR);
else if (S_ISLNK (fe->buf.st_mode)){
if (fe->f.link_to_dir)
return (DIRECTORY_COLOR);
else if (fe->f.stalled_link)
return (STALLED_LINK_COLOR);
else
return (LINK_COLOR);
} else if (S_ISSOCK (fe->buf.st_mode))
return (SPECIAL_COLOR);
else if (S_ISCHR (fe->buf.st_mode))
return (DEVICE_COLOR);
else if (S_ISBLK (fe->buf.st_mode))
return (DEVICE_COLOR);
else if (S_ISFIFO (fe->buf.st_mode))
return (SPECIAL_COLOR);
else if (is_exe (fe->buf.st_mode))
return (EXECUTABLE_COLOR);
else if (fe->fname && (!strcmp (fe->fname, "core") || !strcmp (extension(fe->fname), "core")))
return (CORE_COLOR);
return (NORMAL_COLOR); /* just for safeness */
1998-02-27 07:54:42 +03:00
}
/* Formats the file number file_index of panel in the buffer dest */
void
format_file (char *dest, WPanel *panel, int file_index, int width, int attr, int isstatus)
{
int color, length, empty_line;
char *txt;
char *old_pos;
char *cdest = dest;
format_e *format, *home;
file_entry *fe;
1998-02-27 07:54:42 +03:00
length = 0;
empty_line = (file_index >= panel->count);
home = (isstatus) ? panel->status_format : panel->format;
1998-02-27 07:54:42 +03:00
fe = &panel->dir.list [file_index];
if (!empty_line)
color = file_compute_color (attr, fe);
else
color = NORMAL_COLOR;
1998-02-27 07:54:42 +03:00
for (format = home; format; format = format->next){
if (length == width)
break;
1998-02-27 07:54:42 +03:00
if (format->string_fn){
int len;
1998-02-27 07:54:42 +03:00
if (empty_line)
txt = " ";
else
txt = (*format->string_fn)(fe, format->field_len);
old_pos = cdest;
len = format->field_len;
if (len + length > width)
len = width - length;
cdest = to_buffer (cdest, format->just_mode, len, txt);
length += len;
#ifndef HAVE_X
1998-02-27 07:54:42 +03:00
attrset (color);
1998-02-27 07:54:42 +03:00
if (permission_mode && !strcmp(format->id, "perm"))
add_permission_string (old_pos, format->field_len, fe, attr, color, 0);
else if (permission_mode && !strcmp(format->id, "mode"))
add_permission_string (old_pos, format->field_len, fe, attr, color, 1);
else
addstr (old_pos);
1998-02-27 07:54:42 +03:00
#endif
} else {
#ifndef HAVE_X
if (attr == SELECTED || attr == MARKED_SELECTED)
attrset (SELECTED_COLOR);
else
attrset (NORMAL_COLOR);
one_vline ();
#else
*cdest++ = ' ';
#endif
length++;
}
}
1998-02-27 07:54:42 +03:00
if (length < width){
int still = width - length;
while (still--)
#ifdef HAVE_X
*cdest++ = ' ';
*cdest = '\0';
1998-02-27 07:54:42 +03:00
#else
addch (' ');
#endif
}
}
#ifndef HAVE_X
void
repaint_file (WPanel *panel, int file_index, int mv, int attr, int isstatus)
{
int second_column = 0;
int width, offset;
char buffer [BUF_MEDIUM];
1998-02-27 07:54:42 +03:00
offset = 0;
if (!isstatus && panel->split){
1998-02-27 07:54:42 +03:00
second_column = (file_index - panel->top_file) / llines (panel);
width = (panel->widget.cols - 2)/2 - 1;
1998-02-27 07:54:42 +03:00
if (second_column){
offset = 1 + width;
1998-02-27 07:54:42 +03:00
width = (panel->widget.cols-2) - (panel->widget.cols-2)/2 - 1;
}
1998-02-27 07:54:42 +03:00
} else
width = (panel->widget.cols - 2);
if (mv){
if (!isstatus && panel->split){
widget_move (&panel->widget,
(file_index - panel->top_file) %
llines (panel) + 2,
(offset + 1));
} else
widget_move (&panel->widget, file_index - panel->top_file + 2, 1);
}
1998-02-27 07:54:42 +03:00
format_file (buffer, panel, file_index, width, attr, isstatus);
1998-02-27 07:54:42 +03:00
if (!isstatus && panel->split){
if (second_column)
addch (' ');
else {
attrset (NORMAL_COLOR);
one_vline ();
}
}
}
#endif
#ifndef PORT_HAS_DISPLAY_MINI_INFO
1998-02-27 07:54:42 +03:00
void
display_mini_info (WPanel *panel)
{
if (!show_mini_info)
return;
1998-02-27 07:54:42 +03:00
widget_move (&panel->widget, llines (panel)+3, 1);
1998-02-27 07:54:42 +03:00
if (panel->searching){
attrset (INPUT_COLOR);
printw ("/%-*s", panel->widget.cols-3, panel->search_buffer);
attrset (NORMAL_COLOR);
return;
}
/* Status displays total marked size */
if (panel->marked){
char buffer [BUF_SMALL];
char *p = " %-*s";
int cols = panel->widget.cols-2;
1998-02-27 07:54:42 +03:00
attrset (MARKED_COLOR);
printw ("%*s", cols, " ");
1998-02-27 07:54:42 +03:00
widget_move (&panel->widget, llines (panel)+3, 1);
g_snprintf (buffer, sizeof (buffer), _((panel->marked == 1) ?
N_("%s bytes in %d file") : N_("%s bytes in %d files")),
size_trunc_sep (panel->total), panel->marked);
if (strlen (buffer) > cols-2){
buffer [cols] = 0;
1998-02-27 07:54:42 +03:00
p += 2;
} else
cols -= 2;
printw (p, cols, buffer);
1998-02-27 07:54:42 +03:00
return;
}
/* Status resolves links and show them */
set_colors (panel);
#ifndef OS2_NT
1998-02-27 07:54:42 +03:00
if (S_ISLNK (panel->dir.list [panel->selected].buf.st_mode)){
char *link, link_target [MC_MAXPATHLEN];
int len;
link = concat_dir_and_file (panel->cwd, panel->dir.list [panel->selected].fname);
len = mc_readlink (link, link_target, MC_MAXPATHLEN);
g_free (link);
1998-02-27 07:54:42 +03:00
if (len > 0){
link_target[len] = 0;
printw ("-> %-*s", panel->widget.cols - 5,
name_trunc (link_target, panel->widget.cols - 5));
} else
addstr (_("<readlink failed>"));
1998-02-27 07:54:42 +03:00
return;
}
#endif
/* Default behaviour */
repaint_file (panel, panel->selected, 0, STATUS, 1);
return;
}
#endif
1998-02-27 07:54:42 +03:00
#ifndef HAVE_X
void
paint_dir (WPanel *panel)
{
int i;
int color; /* Color value of the line */
int items; /* Number of items */
items = llines (panel) * (panel->split ? 2 : 1);
1998-02-27 07:54:42 +03:00
for (i = 0; i < items; i++){
if (i+panel->top_file >= panel->count)
color = 0;
else {
color = 2 * (panel->dir.list [i+panel->top_file].f.marked);
color += (panel->selected==i+panel->top_file && panel->active);
}
repaint_file (panel, i+panel->top_file, 1, color, 0);
1998-02-27 07:54:42 +03:00
}
standend ();
panel->dirty = 0;
}
#endif
#ifdef HAVE_X
#define mini_info_separator(x)
#else
static void
mini_info_separator (WPanel *panel)
{
if (!show_mini_info)
return;
1998-02-27 07:54:42 +03:00
standend ();
widget_move (&panel->widget, llines (panel)+2, 1);
#ifdef HAVE_SLANG
attrset (NORMAL_COLOR);
hline (ACS_HLINE, panel->widget.cols-2);
#else
hline ((slow_terminal ? '-' : ACS_HLINE) | NORMAL_COLOR,
panel->widget.cols-2);
#endif
1998-02-27 07:54:42 +03:00
}
void
show_dir (WPanel *panel)
{
char tmp [200];
set_colors (panel);
draw_double_box (panel->widget.parent,
panel->widget.y, panel->widget.x,
panel->widget.lines, panel->widget.cols);
#ifdef HAVE_SLANG
1998-02-27 07:54:42 +03:00
if (show_mini_info) {
#ifdef linux_unicode
if (SLtt_Unicode) {
SLsmg_draw_unicode (panel->widget.y + llines (panel) + 2,
1998-02-27 07:54:42 +03:00
panel->widget.x, SLUNI_DSLTEE_CHAR);
SLsmg_draw_unicode (panel->widget.y + llines (panel) + 2,
1998-02-27 07:54:42 +03:00
panel->widget.x + panel->widget.cols - 1, SLUNI_DSRTEE_CHAR);
} else
#endif /* linux_unicode */
{
SLsmg_draw_object (panel->widget.y + llines (panel) + 2,
panel->widget.x, SLSMG_LTEE_CHAR);
SLsmg_draw_object (panel->widget.y + llines (panel) + 2,
panel->widget.x + panel->widget.cols - 1, SLSMG_RTEE_CHAR);
}
}
#endif /* have_slang */
1998-02-27 07:54:42 +03:00
if (panel->active)
attrset (REVERSE_COLOR);
widget_move (&panel->widget, 0, 3);
1998-02-27 07:54:42 +03:00
trim (strip_home_and_password (panel->cwd), tmp, panel->widget.cols-7);
1998-02-27 07:54:42 +03:00
addstr (tmp);
widget_move (&panel->widget, 0, 1);
addstr ("<");
widget_move (&panel->widget, 0, panel->widget.cols-2);
addstr (">");
widget_move (&panel->widget, 0, panel->widget.cols-3);
addstr ("v");
1998-02-27 07:54:42 +03:00
if (panel->active)
standend ();
}
#endif
#ifndef HAVE_TK
/* To be used only by long_frame and full_frame to adjust top_file */
static void
adjust_top_file (WPanel *panel)
{
int old_top = panel->top_file;
1998-02-27 07:54:42 +03:00
if (panel->selected - old_top > llines (panel))
panel->top_file = panel->selected;
if (old_top - panel->count > llines (panel))
panel->top_file = panel->count - llines (panel);
#ifdef HAVE_TK
if (old_top != panel->top_file)
x_adjust_top_file (panel);
#endif
}
#else
#define adjust_top_file(p)
#endif
/* Repaints the information that changes after a command */
void
panel_update_contents (WPanel *panel)
{
show_dir (panel);
#ifdef HAVE_X
x_fill_panel (panel);
#else
paint_dir (panel);
#endif
display_mini_info (panel);
}
void
paint_panel (WPanel *panel)
{
paint_frame (panel);
panel_update_contents (panel);
mini_info_separator (panel);
}
static void
do_select (WPanel *panel, int i)
{
if (i != panel->selected){
panel->selected = i;
panel->top_file = panel->selected - (panel->widget.lines-2)/2;
if (panel->top_file < 0)
panel->top_file = 0;
x_adjust_top_file (panel);
}
}
1998-02-27 07:54:42 +03:00
void
Xtry_to_select (WPanel *panel, char *name)
{
int i;
1998-02-27 07:54:42 +03:00
char *subdir;
if (!name) {
do_select(panel, 0);
1998-02-27 07:54:42 +03:00
return;
}
/* We only want the last component of the directory */
subdir = x_basename (name);
subdir = vfs_strip_suffix_from_filename (subdir);
1998-02-27 07:54:42 +03:00
/* Search that subdirectory, if found select it */
for (i = 0; i < panel->count; i++){
if (strcmp (subdir, panel->dir.list [i].fname) == 0) {
do_select (panel, i);
g_free (subdir);
return;
}
1998-02-27 07:54:42 +03:00
}
/* Try to select a file near the file that is missing */
if (panel->selected >= panel->count)
do_select (panel, panel->count-1);
g_free (subdir);
1998-02-27 07:54:42 +03:00
}
#ifndef PORT_HAS_PANEL_UPDATE_COLS
void
panel_update_cols (Widget *widget, int frame_size)
{
int cols, origin;
1998-02-27 07:54:42 +03:00
if (horizontal_split){
widget->cols = COLS;
return;
}
if (frame_size == frame_full){
cols = COLS;
origin = 0;
} else {
if (widget == get_panel_widget (0)){
cols = first_panel_size;
origin = 0;
} else {
cols = COLS-first_panel_size;
origin = first_panel_size;
}
}
1998-02-27 07:54:42 +03:00
widget->cols = cols;
widget->x = origin;
}
#endif
static char *
panel_save_name (WPanel *panel)
{
extern int saving_setup;
1998-02-27 07:54:42 +03:00
/* If the program is shuting down */
if ((midnight_shutdown && auto_save_setup) || saving_setup)
return g_strdup (panel->panel_name);
1998-02-27 07:54:42 +03:00
else
return g_strconcat ("Temporal:", panel->panel_name, NULL);
1998-02-27 07:54:42 +03:00
}
static void
panel_destroy (WPanel *p)
{
int i;
1998-02-27 07:54:42 +03:00
char *name = panel_save_name (p);
panel_save_setup (p, name);
Ok, most of the "Elliot Lee confidential bug report" has been dealt with with this commit. This also addresses a bunch of the comments from the status.shtml from DrMike. Miguel. 1998-04-15 Miguel de Icaza <miguel@nuclecu.unam.mx> * screen.c (GT): Assign two spaces for the minimum size of the "type" field for the GNOME edition. This gives some extra space for the icon that gets displayed. * dlg.c (remove_widget): New function: used to remove a widget from an existing Dlg_head; (destroy_widget): Destroy a specific Widget. (add_widgetl): Extended to deal with the fact that a running Dlg_head can become empty. * panelize.c (l_call): Update the input line every time the user selects the entry with the mouse (pretty common in the gnome edition). * hotlist.c (add_new_group_input): Removed an extra field that was causing problems. * find.c (find_parameters): Tree button is gone for gnome until we get the tree function working on gnome. * cmd.c (save_setup_cmd): Per Elliot's suggestion, do not pop up a dialog box to inform the user about the saved setup. 1998-04-15 Miguel de Icaza <miguel@nuclecu.unam.mx> * gcmd.c (gnome_close_panel): Implement the close-the-panel functionality. * gscreen.c (x_panel_destroy): Implement the gnome mode way of destroying the panel. * gview.c (view_status): Add cacheing of the information status, to avoid excessive flicker. It would be better if GtkLabel did not flicker though. (scrollbar_moved): Scroll correctly. We now use view_move_backward and view_move_forward to adjust the scrollbar contents. This displays the scrollbar correctly. * gwidget.c (x_listbox_select_nth): This may be invoked before the widget has been created, work around this problem. * gscreen.c (show_dir): Set the title bar for the window to the current directoy. Reported by both Mike and Elliot. * layout: Updated to the new hotlist dialog box.
1998-04-16 06:45:53 +04:00
x_panel_destroy (p);
panel_clean_dir (p);
/* save and clean history */
if (p->dir_history){
Hist *current, *old;
history_put (p->hist_name, p->dir_history);
current = p->dir_history;
while (current->next)
current = current->next;
while (current){
old = current;
current = current->prev;
g_free (old->text);
g_free (old);
}
}
g_free (p->hist_name);
1998-02-27 07:54:42 +03:00
delete_format (p->format);
delete_format (p->status_format);
g_free (p->user_format);
1998-02-27 07:54:42 +03:00
for (i = 0; i < LIST_TYPES; i++)
g_free (p->user_status_format [i]);
g_free (p->dir.list);
g_free (p->panel_name);
g_free (name);
1998-02-27 07:54:42 +03:00
}
static void
panel_format_modified (WPanel *panel)
{
panel->format_modified = 1;
x_reset_sort_labels (panel);
}
int
is_a_panel (Widget *w)
{
return (w->callback == (void *) panel_callback);
}
1998-02-27 07:54:42 +03:00
/* Panel creation */
/* The parameter specifies the name of the panel for setup retieving */
WPanel *
panel_new (char *panel_name)
{
These are a bunch of changes to fix CORBA and session management. They are almost complete (i.e. to handle all nitty gritty cases), but they seem to be working OK right now. SM should be much more stable now. Please tell me if you find any weird behavior - Federico 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * gdesktop-icon.c (desktop_icon_realize): Remove the WM_CLIENT_LEADER property from icon windows so that window managers will not store SM information for them. * gnome-open-dialog.c: Added missing #includes. * gdesktop-init.c (desktop_init_at): Removed an unused variable. * gdesktop.h: Added some missing prototypes. * gmain.h: Added some missing prototypes. * Makefile.in: Added gsession.[ch] to the list of sources. * gmain.c (create_panels): Consider whether we have a CORBA server and session management. * gdesktop.c: #include "gdesktop-init.h" * gdesktop.c: Added a missing cast to GNOME_DIALOG. * gmain.c (create_panels): Removed the run_desktop global variable. * glayout.c (create_container): Set the wmclass of the panel to include its unique ID. * gsession.[ch]: New file with the functions that deal with session management. * glayout.c (gnome_exit): Use session_set_restart(). * gcorba.c (corba_init): Now returns an int with an error value. (corba_init_server): Initialize the server properly. Fixed all the object implementation code. (corba_create_window): New function used to create a window with the CORBA server. * gmain.c (gnome_check_super_user): Now the check for running as root is done here. There should be no GUI code in src/. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * dlg.c (dlg_run_done): Do not call the callback of a NULL current widget. * setup.h: Added missing prototype for setup_init(). * filegui.c (check_progress_buttons): Added a missing return value. * dlg.c (remove_widget): Added a missing return value. * main.c: Removed the global directory_list variable. Removed the main_corba_register_server() function. * main.h: Removed the global run_desktop variable. * panel.h: Now the panel structure has a unique numerical ID used for session management. * screen.c (panel_new): Maintain a unique ID for each panel. * main.c (maybe_display_linksdir): Handle display of the desktop init dir here. (main): Call gnome_check_super_user(). (init_corba_with_args): Call corba_init_server(). * main.c (init_corba_with_args): Do CORBA initialization here. Also removed the global force_activation option. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * vfs.c (vfs_add_current_stamps): Only do stamping of the panels if they exist. * mcserv.c: #include <sys/wait.h> (get_client): Put `#ifdef __EMX__' around an otherwise-unused variable. * utilvfs.c (vfs_split_url): Fix NULL <-> 0 confusion when comparing characters. * ftpfs.c (retrieve_dir): Removed unused variable dot_dot_found. * extfs.c (extfs_init): Assign `key' to c, not `&key'.
1999-03-30 10:09:56 +04:00
#ifdef HAVE_GNOME
static int panel_id = 0;
#endif
1998-02-27 07:54:42 +03:00
WPanel *panel;
char *section;
int i, err;
panel = g_new0 (WPanel, 1);
These are a bunch of changes to fix CORBA and session management. They are almost complete (i.e. to handle all nitty gritty cases), but they seem to be working OK right now. SM should be much more stable now. Please tell me if you find any weird behavior - Federico 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * gdesktop-icon.c (desktop_icon_realize): Remove the WM_CLIENT_LEADER property from icon windows so that window managers will not store SM information for them. * gnome-open-dialog.c: Added missing #includes. * gdesktop-init.c (desktop_init_at): Removed an unused variable. * gdesktop.h: Added some missing prototypes. * gmain.h: Added some missing prototypes. * Makefile.in: Added gsession.[ch] to the list of sources. * gmain.c (create_panels): Consider whether we have a CORBA server and session management. * gdesktop.c: #include "gdesktop-init.h" * gdesktop.c: Added a missing cast to GNOME_DIALOG. * gmain.c (create_panels): Removed the run_desktop global variable. * glayout.c (create_container): Set the wmclass of the panel to include its unique ID. * gsession.[ch]: New file with the functions that deal with session management. * glayout.c (gnome_exit): Use session_set_restart(). * gcorba.c (corba_init): Now returns an int with an error value. (corba_init_server): Initialize the server properly. Fixed all the object implementation code. (corba_create_window): New function used to create a window with the CORBA server. * gmain.c (gnome_check_super_user): Now the check for running as root is done here. There should be no GUI code in src/. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * dlg.c (dlg_run_done): Do not call the callback of a NULL current widget. * setup.h: Added missing prototype for setup_init(). * filegui.c (check_progress_buttons): Added a missing return value. * dlg.c (remove_widget): Added a missing return value. * main.c: Removed the global directory_list variable. Removed the main_corba_register_server() function. * main.h: Removed the global run_desktop variable. * panel.h: Now the panel structure has a unique numerical ID used for session management. * screen.c (panel_new): Maintain a unique ID for each panel. * main.c (maybe_display_linksdir): Handle display of the desktop init dir here. (main): Call gnome_check_super_user(). (init_corba_with_args): Call corba_init_server(). * main.c (init_corba_with_args): Do CORBA initialization here. Also removed the global force_activation option. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * vfs.c (vfs_add_current_stamps): Only do stamping of the panels if they exist. * mcserv.c: #include <sys/wait.h> (get_client): Put `#ifdef __EMX__' around an otherwise-unused variable. * utilvfs.c (vfs_split_url): Fix NULL <-> 0 confusion when comparing characters. * ftpfs.c (retrieve_dir): Removed unused variable dot_dot_found. * extfs.c (extfs_init): Assign `key' to c, not `&key'.
1999-03-30 10:09:56 +04:00
1998-02-27 07:54:42 +03:00
/* No know sizes of the panel at startup */
init_widget (&panel->widget, 0, 0, 0, 0, (callback_fn)
panel_callback, (destroy_fn) panel_destroy,
(mouse_h) panel_event, NULL);
/* We do not want the cursor */
widget_want_cursor (panel->widget, 0);
These are a bunch of changes to fix CORBA and session management. They are almost complete (i.e. to handle all nitty gritty cases), but they seem to be working OK right now. SM should be much more stable now. Please tell me if you find any weird behavior - Federico 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * gdesktop-icon.c (desktop_icon_realize): Remove the WM_CLIENT_LEADER property from icon windows so that window managers will not store SM information for them. * gnome-open-dialog.c: Added missing #includes. * gdesktop-init.c (desktop_init_at): Removed an unused variable. * gdesktop.h: Added some missing prototypes. * gmain.h: Added some missing prototypes. * Makefile.in: Added gsession.[ch] to the list of sources. * gmain.c (create_panels): Consider whether we have a CORBA server and session management. * gdesktop.c: #include "gdesktop-init.h" * gdesktop.c: Added a missing cast to GNOME_DIALOG. * gmain.c (create_panels): Removed the run_desktop global variable. * glayout.c (create_container): Set the wmclass of the panel to include its unique ID. * gsession.[ch]: New file with the functions that deal with session management. * glayout.c (gnome_exit): Use session_set_restart(). * gcorba.c (corba_init): Now returns an int with an error value. (corba_init_server): Initialize the server properly. Fixed all the object implementation code. (corba_create_window): New function used to create a window with the CORBA server. * gmain.c (gnome_check_super_user): Now the check for running as root is done here. There should be no GUI code in src/. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * dlg.c (dlg_run_done): Do not call the callback of a NULL current widget. * setup.h: Added missing prototype for setup_init(). * filegui.c (check_progress_buttons): Added a missing return value. * dlg.c (remove_widget): Added a missing return value. * main.c: Removed the global directory_list variable. Removed the main_corba_register_server() function. * main.h: Removed the global run_desktop variable. * panel.h: Now the panel structure has a unique numerical ID used for session management. * screen.c (panel_new): Maintain a unique ID for each panel. * main.c (maybe_display_linksdir): Handle display of the desktop init dir here. (main): Call gnome_check_super_user(). (init_corba_with_args): Call corba_init_server(). * main.c (init_corba_with_args): Do CORBA initialization here. Also removed the global force_activation option. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * vfs.c (vfs_add_current_stamps): Only do stamping of the panels if they exist. * mcserv.c: #include <sys/wait.h> (get_client): Put `#ifdef __EMX__' around an otherwise-unused variable. * utilvfs.c (vfs_split_url): Fix NULL <-> 0 confusion when comparing characters. * ftpfs.c (retrieve_dir): Removed unused variable dot_dot_found. * extfs.c (extfs_init): Assign `key' to c, not `&key'.
1999-03-30 10:09:56 +04:00
1998-02-27 07:54:42 +03:00
mc_get_current_wd (panel->cwd, sizeof (panel->cwd)-2);
strcpy (panel->lwd, ".");
panel->hist_name = g_strconcat ("Dir Hist ", panel_name, NULL);
panel->dir_history = history_get (panel->hist_name);
directory_history_add (panel, panel->cwd);
panel->dir.list = g_new (file_entry, MIN_FILES);
panel->dir.size = MIN_FILES;
panel->active = 0;
panel->filter = 0;
panel->split = 0;
panel->top_file = 0;
panel->selected = 0;
panel->marked = 0;
panel->total = 0;
panel->reverse = 0;
panel->dirty = 1;
panel->searching = 0;
panel->dirs_marked = 0;
panel->is_panelized = 0;
panel->format = 0;
panel->status_format = 0;
panel->format_modified = 1;
1999-03-10 Federico Mena Quintero <federico@nuclecu.unam.mx> This is a semi-big slew of changes to integrate the DESKTOP_BRANCH into the main trunk. Now the gdesktop code creates a temprary WPanel structure and passes it on to the core file management functions. Also, the menu code is improved (we now use gpopup2; gpopup should go away shortly and gpopup2 should replace it). This makes the desktop act consistently with the file panels. Thanks to Jonathan for his help with this branch. * gdnd.c (perform_action): Use mc_lstat(), not mc_stat(). * glayout.c (update_panels): Do not update the panels that are desktop panels. * gdesktop.c (icon_is_in_area): An icon is not considered to be in a 0x0 area. * gpopup2.c (handle_open): Fetch the desktop icon from the filename and call desktop_icon_info_open(). * gdesktop.c (desktop_icon_info_get_by_filename): New function to look up a desktop icon by its filename. * glayout.c (create_new_menu_from): Test for the ".desktop" suffix correctly. * gpopup2.c (handle_properties): If the file comes from a desktop panel, always allow edition of the icon image. 1999-03-10 Federico Mena Quintero <federico@nuclecu.unam.mx> * file.c (erase_dir): Erase metadata for directories as well. (erase_dir_iff_empty): Likewise. (copy_file_file): Delete/copy the metadata even for char/block/sock/fifo files. Same thing for when copying symlinks. (copy_dir_dir): Delete/copy the metadata. (move_dir_dir): Delete/move the metadata. (recursive_erase): Delete the metadata.
1999-03-11 05:40:53 +03:00
#ifdef HAVE_GNOME
panel->drag_tree_row = -1;
1999-03-10 Federico Mena Quintero <federico@nuclecu.unam.mx> This is a semi-big slew of changes to integrate the DESKTOP_BRANCH into the main trunk. Now the gdesktop code creates a temprary WPanel structure and passes it on to the core file management functions. Also, the menu code is improved (we now use gpopup2; gpopup should go away shortly and gpopup2 should replace it). This makes the desktop act consistently with the file panels. Thanks to Jonathan for his help with this branch. * gdnd.c (perform_action): Use mc_lstat(), not mc_stat(). * glayout.c (update_panels): Do not update the panels that are desktop panels. * gdesktop.c (icon_is_in_area): An icon is not considered to be in a 0x0 area. * gpopup2.c (handle_open): Fetch the desktop icon from the filename and call desktop_icon_info_open(). * gdesktop.c (desktop_icon_info_get_by_filename): New function to look up a desktop icon by its filename. * glayout.c (create_new_menu_from): Test for the ".desktop" suffix correctly. * gpopup2.c (handle_properties): If the file comes from a desktop panel, always allow edition of the icon image. 1999-03-10 Federico Mena Quintero <federico@nuclecu.unam.mx> * file.c (erase_dir): Erase metadata for directories as well. (erase_dir_iff_empty): Likewise. (copy_file_file): Delete/copy the metadata even for char/block/sock/fifo files. Same thing for when copying symlinks. (copy_dir_dir): Delete/copy the metadata. (move_dir_dir): Delete/move the metadata. (recursive_erase): Delete the metadata.
1999-03-11 05:40:53 +03:00
panel->is_a_desktop_panel = FALSE;
These are a bunch of changes to fix CORBA and session management. They are almost complete (i.e. to handle all nitty gritty cases), but they seem to be working OK right now. SM should be much more stable now. Please tell me if you find any weird behavior - Federico 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * gdesktop-icon.c (desktop_icon_realize): Remove the WM_CLIENT_LEADER property from icon windows so that window managers will not store SM information for them. * gnome-open-dialog.c: Added missing #includes. * gdesktop-init.c (desktop_init_at): Removed an unused variable. * gdesktop.h: Added some missing prototypes. * gmain.h: Added some missing prototypes. * Makefile.in: Added gsession.[ch] to the list of sources. * gmain.c (create_panels): Consider whether we have a CORBA server and session management. * gdesktop.c: #include "gdesktop-init.h" * gdesktop.c: Added a missing cast to GNOME_DIALOG. * gmain.c (create_panels): Removed the run_desktop global variable. * glayout.c (create_container): Set the wmclass of the panel to include its unique ID. * gsession.[ch]: New file with the functions that deal with session management. * glayout.c (gnome_exit): Use session_set_restart(). * gcorba.c (corba_init): Now returns an int with an error value. (corba_init_server): Initialize the server properly. Fixed all the object implementation code. (corba_create_window): New function used to create a window with the CORBA server. * gmain.c (gnome_check_super_user): Now the check for running as root is done here. There should be no GUI code in src/. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * dlg.c (dlg_run_done): Do not call the callback of a NULL current widget. * setup.h: Added missing prototype for setup_init(). * filegui.c (check_progress_buttons): Added a missing return value. * dlg.c (remove_widget): Added a missing return value. * main.c: Removed the global directory_list variable. Removed the main_corba_register_server() function. * main.h: Removed the global run_desktop variable. * panel.h: Now the panel structure has a unique numerical ID used for session management. * screen.c (panel_new): Maintain a unique ID for each panel. * main.c (maybe_display_linksdir): Handle display of the desktop init dir here. (main): Call gnome_check_super_user(). (init_corba_with_args): Call corba_init_server(). * main.c (init_corba_with_args): Do CORBA initialization here. Also removed the global force_activation option. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * vfs.c (vfs_add_current_stamps): Only do stamping of the panels if they exist. * mcserv.c: #include <sys/wait.h> (get_client): Put `#ifdef __EMX__' around an otherwise-unused variable. * utilvfs.c (vfs_split_url): Fix NULL <-> 0 confusion when comparing characters. * ftpfs.c (retrieve_dir): Removed unused variable dot_dot_found. * extfs.c (extfs_init): Assign `key' to c, not `&key'.
1999-03-30 10:09:56 +04:00
panel->id = panel_id++;
panel->servant = NULL;
1999-03-10 Federico Mena Quintero <federico@nuclecu.unam.mx> This is a semi-big slew of changes to integrate the DESKTOP_BRANCH into the main trunk. Now the gdesktop code creates a temprary WPanel structure and passes it on to the core file management functions. Also, the menu code is improved (we now use gpopup2; gpopup should go away shortly and gpopup2 should replace it). This makes the desktop act consistently with the file panels. Thanks to Jonathan for his help with this branch. * gdnd.c (perform_action): Use mc_lstat(), not mc_stat(). * glayout.c (update_panels): Do not update the panels that are desktop panels. * gdesktop.c (icon_is_in_area): An icon is not considered to be in a 0x0 area. * gpopup2.c (handle_open): Fetch the desktop icon from the filename and call desktop_icon_info_open(). * gdesktop.c (desktop_icon_info_get_by_filename): New function to look up a desktop icon by its filename. * glayout.c (create_new_menu_from): Test for the ".desktop" suffix correctly. * gpopup2.c (handle_properties): If the file comes from a desktop panel, always allow edition of the icon image. 1999-03-10 Federico Mena Quintero <federico@nuclecu.unam.mx> * file.c (erase_dir): Erase metadata for directories as well. (erase_dir_iff_empty): Likewise. (copy_file_file): Delete/copy the metadata even for char/block/sock/fifo files. Same thing for when copying symlinks. (copy_dir_dir): Delete/copy the metadata. (move_dir_dir): Delete/move the metadata. (recursive_erase): Delete the metadata.
1999-03-11 05:40:53 +03:00
#endif
These are a bunch of changes to fix CORBA and session management. They are almost complete (i.e. to handle all nitty gritty cases), but they seem to be working OK right now. SM should be much more stable now. Please tell me if you find any weird behavior - Federico 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * gdesktop-icon.c (desktop_icon_realize): Remove the WM_CLIENT_LEADER property from icon windows so that window managers will not store SM information for them. * gnome-open-dialog.c: Added missing #includes. * gdesktop-init.c (desktop_init_at): Removed an unused variable. * gdesktop.h: Added some missing prototypes. * gmain.h: Added some missing prototypes. * Makefile.in: Added gsession.[ch] to the list of sources. * gmain.c (create_panels): Consider whether we have a CORBA server and session management. * gdesktop.c: #include "gdesktop-init.h" * gdesktop.c: Added a missing cast to GNOME_DIALOG. * gmain.c (create_panels): Removed the run_desktop global variable. * glayout.c (create_container): Set the wmclass of the panel to include its unique ID. * gsession.[ch]: New file with the functions that deal with session management. * glayout.c (gnome_exit): Use session_set_restart(). * gcorba.c (corba_init): Now returns an int with an error value. (corba_init_server): Initialize the server properly. Fixed all the object implementation code. (corba_create_window): New function used to create a window with the CORBA server. * gmain.c (gnome_check_super_user): Now the check for running as root is done here. There should be no GUI code in src/. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * dlg.c (dlg_run_done): Do not call the callback of a NULL current widget. * setup.h: Added missing prototype for setup_init(). * filegui.c (check_progress_buttons): Added a missing return value. * dlg.c (remove_widget): Added a missing return value. * main.c: Removed the global directory_list variable. Removed the main_corba_register_server() function. * main.h: Removed the global run_desktop variable. * panel.h: Now the panel structure has a unique numerical ID used for session management. * screen.c (panel_new): Maintain a unique ID for each panel. * main.c (maybe_display_linksdir): Handle display of the desktop init dir here. (main): Call gnome_check_super_user(). (init_corba_with_args): Call corba_init_server(). * main.c (init_corba_with_args): Do CORBA initialization here. Also removed the global force_activation option. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * vfs.c (vfs_add_current_stamps): Only do stamping of the panels if they exist. * mcserv.c: #include <sys/wait.h> (get_client): Put `#ifdef __EMX__' around an otherwise-unused variable. * utilvfs.c (vfs_split_url): Fix NULL <-> 0 confusion when comparing characters. * ftpfs.c (retrieve_dir): Removed unused variable dot_dot_found. * extfs.c (extfs_init): Assign `key' to c, not `&key'.
1999-03-30 10:09:56 +04:00
panel->panel_name = g_strdup (panel_name);
panel->user_format = g_strdup (DEFAULT_USER_FORMAT);
These are a bunch of changes to fix CORBA and session management. They are almost complete (i.e. to handle all nitty gritty cases), but they seem to be working OK right now. SM should be much more stable now. Please tell me if you find any weird behavior - Federico 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * gdesktop-icon.c (desktop_icon_realize): Remove the WM_CLIENT_LEADER property from icon windows so that window managers will not store SM information for them. * gnome-open-dialog.c: Added missing #includes. * gdesktop-init.c (desktop_init_at): Removed an unused variable. * gdesktop.h: Added some missing prototypes. * gmain.h: Added some missing prototypes. * Makefile.in: Added gsession.[ch] to the list of sources. * gmain.c (create_panels): Consider whether we have a CORBA server and session management. * gdesktop.c: #include "gdesktop-init.h" * gdesktop.c: Added a missing cast to GNOME_DIALOG. * gmain.c (create_panels): Removed the run_desktop global variable. * glayout.c (create_container): Set the wmclass of the panel to include its unique ID. * gsession.[ch]: New file with the functions that deal with session management. * glayout.c (gnome_exit): Use session_set_restart(). * gcorba.c (corba_init): Now returns an int with an error value. (corba_init_server): Initialize the server properly. Fixed all the object implementation code. (corba_create_window): New function used to create a window with the CORBA server. * gmain.c (gnome_check_super_user): Now the check for running as root is done here. There should be no GUI code in src/. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * dlg.c (dlg_run_done): Do not call the callback of a NULL current widget. * setup.h: Added missing prototype for setup_init(). * filegui.c (check_progress_buttons): Added a missing return value. * dlg.c (remove_widget): Added a missing return value. * main.c: Removed the global directory_list variable. Removed the main_corba_register_server() function. * main.h: Removed the global run_desktop variable. * panel.h: Now the panel structure has a unique numerical ID used for session management. * screen.c (panel_new): Maintain a unique ID for each panel. * main.c (maybe_display_linksdir): Handle display of the desktop init dir here. (main): Call gnome_check_super_user(). (init_corba_with_args): Call corba_init_server(). * main.c (init_corba_with_args): Do CORBA initialization here. Also removed the global force_activation option. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * vfs.c (vfs_add_current_stamps): Only do stamping of the panels if they exist. * mcserv.c: #include <sys/wait.h> (get_client): Put `#ifdef __EMX__' around an otherwise-unused variable. * utilvfs.c (vfs_split_url): Fix NULL <-> 0 confusion when comparing characters. * ftpfs.c (retrieve_dir): Removed unused variable dot_dot_found. * extfs.c (extfs_init): Assign `key' to c, not `&key'.
1999-03-30 10:09:56 +04:00
1998-02-27 07:54:42 +03:00
for(i = 0; i < LIST_TYPES; i++)
panel->user_status_format [i] = g_strdup (DEFAULT_USER_FORMAT);
These are a bunch of changes to fix CORBA and session management. They are almost complete (i.e. to handle all nitty gritty cases), but they seem to be working OK right now. SM should be much more stable now. Please tell me if you find any weird behavior - Federico 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * gdesktop-icon.c (desktop_icon_realize): Remove the WM_CLIENT_LEADER property from icon windows so that window managers will not store SM information for them. * gnome-open-dialog.c: Added missing #includes. * gdesktop-init.c (desktop_init_at): Removed an unused variable. * gdesktop.h: Added some missing prototypes. * gmain.h: Added some missing prototypes. * Makefile.in: Added gsession.[ch] to the list of sources. * gmain.c (create_panels): Consider whether we have a CORBA server and session management. * gdesktop.c: #include "gdesktop-init.h" * gdesktop.c: Added a missing cast to GNOME_DIALOG. * gmain.c (create_panels): Removed the run_desktop global variable. * glayout.c (create_container): Set the wmclass of the panel to include its unique ID. * gsession.[ch]: New file with the functions that deal with session management. * glayout.c (gnome_exit): Use session_set_restart(). * gcorba.c (corba_init): Now returns an int with an error value. (corba_init_server): Initialize the server properly. Fixed all the object implementation code. (corba_create_window): New function used to create a window with the CORBA server. * gmain.c (gnome_check_super_user): Now the check for running as root is done here. There should be no GUI code in src/. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * dlg.c (dlg_run_done): Do not call the callback of a NULL current widget. * setup.h: Added missing prototype for setup_init(). * filegui.c (check_progress_buttons): Added a missing return value. * dlg.c (remove_widget): Added a missing return value. * main.c: Removed the global directory_list variable. Removed the main_corba_register_server() function. * main.h: Removed the global run_desktop variable. * panel.h: Now the panel structure has a unique numerical ID used for session management. * screen.c (panel_new): Maintain a unique ID for each panel. * main.c (maybe_display_linksdir): Handle display of the desktop init dir here. (main): Call gnome_check_super_user(). (init_corba_with_args): Call corba_init_server(). * main.c (init_corba_with_args): Do CORBA initialization here. Also removed the global force_activation option. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * vfs.c (vfs_add_current_stamps): Only do stamping of the panels if they exist. * mcserv.c: #include <sys/wait.h> (get_client): Put `#ifdef __EMX__' around an otherwise-unused variable. * utilvfs.c (vfs_split_url): Fix NULL <-> 0 confusion when comparing characters. * ftpfs.c (retrieve_dir): Removed unused variable dot_dot_found. * extfs.c (extfs_init): Assign `key' to c, not `&key'.
1999-03-30 10:09:56 +04:00
1998-02-27 07:54:42 +03:00
panel->search_buffer [0] = 0;
panel->frame_size = frame_half;
section = g_strconcat ("Temporal:", panel->panel_name, NULL);
1998-02-27 07:54:42 +03:00
if (!profile_has_section (section, profile_name)){
g_free (section);
section = g_strdup (panel->panel_name);
1998-02-27 07:54:42 +03:00
}
panel_load_setup (panel, section);
g_free (section);
1998-02-27 07:54:42 +03:00
/* Load format strings */
err = set_panel_formats (panel);
if (err){
set_panel_formats (panel);
}
These are a bunch of changes to fix CORBA and session management. They are almost complete (i.e. to handle all nitty gritty cases), but they seem to be working OK right now. SM should be much more stable now. Please tell me if you find any weird behavior - Federico 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * gdesktop-icon.c (desktop_icon_realize): Remove the WM_CLIENT_LEADER property from icon windows so that window managers will not store SM information for them. * gnome-open-dialog.c: Added missing #includes. * gdesktop-init.c (desktop_init_at): Removed an unused variable. * gdesktop.h: Added some missing prototypes. * gmain.h: Added some missing prototypes. * Makefile.in: Added gsession.[ch] to the list of sources. * gmain.c (create_panels): Consider whether we have a CORBA server and session management. * gdesktop.c: #include "gdesktop-init.h" * gdesktop.c: Added a missing cast to GNOME_DIALOG. * gmain.c (create_panels): Removed the run_desktop global variable. * glayout.c (create_container): Set the wmclass of the panel to include its unique ID. * gsession.[ch]: New file with the functions that deal with session management. * glayout.c (gnome_exit): Use session_set_restart(). * gcorba.c (corba_init): Now returns an int with an error value. (corba_init_server): Initialize the server properly. Fixed all the object implementation code. (corba_create_window): New function used to create a window with the CORBA server. * gmain.c (gnome_check_super_user): Now the check for running as root is done here. There should be no GUI code in src/. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * dlg.c (dlg_run_done): Do not call the callback of a NULL current widget. * setup.h: Added missing prototype for setup_init(). * filegui.c (check_progress_buttons): Added a missing return value. * dlg.c (remove_widget): Added a missing return value. * main.c: Removed the global directory_list variable. Removed the main_corba_register_server() function. * main.h: Removed the global run_desktop variable. * panel.h: Now the panel structure has a unique numerical ID used for session management. * screen.c (panel_new): Maintain a unique ID for each panel. * main.c (maybe_display_linksdir): Handle display of the desktop init dir here. (main): Call gnome_check_super_user(). (init_corba_with_args): Call corba_init_server(). * main.c (init_corba_with_args): Do CORBA initialization here. Also removed the global force_activation option. 1999-03-30 Federico Mena Quintero <federico@nuclecu.unam.mx> * vfs.c (vfs_add_current_stamps): Only do stamping of the panels if they exist. * mcserv.c: #include <sys/wait.h> (get_client): Put `#ifdef __EMX__' around an otherwise-unused variable. * utilvfs.c (vfs_split_url): Fix NULL <-> 0 confusion when comparing characters. * ftpfs.c (retrieve_dir): Removed unused variable dot_dot_found. * extfs.c (extfs_init): Assign `key' to c, not `&key'.
1999-03-30 10:09:56 +04:00
1998-02-27 07:54:42 +03:00
/* Load the default format */
panel->count = do_load_dir (&panel->dir, panel->sort_type,
panel->reverse, panel->case_sensitive, panel->filter);
return panel;
}
void
panel_reload (WPanel *panel)
{
struct stat current_stat;
if (fast_reload
&& !stat (panel->cwd, &current_stat)
&& current_stat.st_ctime == panel->dir_stat.st_ctime
&& current_stat.st_mtime == panel->dir_stat.st_mtime)
return;
while (mc_chdir (panel->cwd) == -1){
char *last_slash;
if (panel->cwd [0] == PATH_SEP && panel->cwd [1] == 0){
panel_clean_dir (panel);
1998-02-27 07:54:42 +03:00
panel->count = set_zero_dir (&panel->dir);
return;
}
last_slash = strrchr (panel->cwd, PATH_SEP);
if (!last_slash || last_slash == panel->cwd)
strcpy (panel->cwd, PATH_SEP_STR);
else
*last_slash = 0;
memset (&(panel->dir_stat), 0, sizeof (panel->dir_stat));
1998-02-27 07:54:42 +03:00
show_dir (panel);
}
1998-02-27 07:54:42 +03:00
panel->count = do_reload_dir (&panel->dir, panel->sort_type, panel->count,
panel->reverse, panel->case_sensitive, panel->filter);
panel->dirty = 1;
if (panel->selected >= panel->count)
do_select (panel, panel->count-1);
recalculate_panel_summary (panel);
1998-02-27 07:54:42 +03:00
}
#ifndef PORT_HAS_PAINT_FRAME
void
paint_frame (WPanel *panel)
{
int header_len;
int spaces, extra;
int side, width;
1998-02-27 07:54:42 +03:00
char *txt, buffer[30]; /*Hope that this is enough ;-) */
if (!panel->split)
adjust_top_file (panel);
1998-02-27 07:54:42 +03:00
widget_erase (&panel->widget);
show_dir (panel);
widget_move (&panel->widget, 1, 1);
for (side = 0; side <= panel->split; side++){
format_e *format;
1998-02-27 07:54:42 +03:00
if (side){
attrset (NORMAL_COLOR);
one_vline ();
width = panel->widget.cols - panel->widget.cols/2 - 1;
} else if (panel->split)
width = panel->widget.cols/2 - 3;
else
width = panel->widget.cols - 2;
1998-02-27 07:54:42 +03:00
for (format = panel->format; format; format = format->next){
if (format->string_fn){
txt = format->title;
1998-02-27 07:54:42 +03:00
header_len = strlen (txt);
if (header_len > format->field_len){
strcpy (buffer, txt);
txt = buffer;
txt [format->field_len] = 0;
header_len = strlen (txt);
}
1998-02-27 07:54:42 +03:00
attrset (MARKED_COLOR);
spaces = (format->field_len - header_len) / 2;
extra = (format->field_len - header_len) % 2;
printw ("%*s%-s%*s", spaces, "",
txt, spaces+extra, "");
width -= 2 * spaces + extra + header_len;
} else {
attrset (NORMAL_COLOR);
one_vline ();
width --;
continue;
}
}
1998-02-27 07:54:42 +03:00
if (width > 0)
printw ("%*s", width, "");
}
}
#endif
static char *
parse_panel_size (WPanel *panel, char *format, int isstatus)
{
int frame = frame_half;
format = skip_separators (format);
if (!strncmp (format, "full", 4)){
frame = frame_full;
format += 4;
} else if (!strncmp (format, "half", 4)){
frame = frame_half;
format += 4;
}
if (!isstatus){
panel->frame_size = frame;
panel->split = 0;
}
1998-02-27 07:54:42 +03:00
/* Now, the optional column specifier */
format = skip_separators (format);
1998-02-27 07:54:42 +03:00
if (*format == '1' || *format == '2'){
if (!isstatus)
panel->split = *format == '2';
format++;
}
if (!isstatus)
panel_update_cols (&(panel->widget), panel->frame_size);
1998-02-27 07:54:42 +03:00
return skip_separators (format);
}
/* Format is:
all := panel_format? format
panel_format := [full|half] [1|2]
format := one_format_e
| format , one_format_e
one_format_e := just format.id [opt_size]
just := [<=>]
1998-02-27 07:54:42 +03:00
opt_size := : size [opt_expand]
size := [0-9]+
opt_expand := +
1998-02-27 07:54:42 +03:00
*/
1998-12-03 00:27:27 +03:00
static format_e *
1998-02-27 07:54:42 +03:00
parse_display_format (WPanel *panel, char *format, char **error, int isstatus, int *res_total_cols)
{
format_e *darr, *old, *home = 0; /* The formats we return */
int total_cols = 0; /* Used columns by the format */
int set_justify; /* flag: set justification mode? */
int justify = 0; /* Which mode. */
int items = 0; /* Number of items in the format */
int i;
1999-04-30 16:41:41 +04:00
static size_t i18n_timelength = 0; /* flag: check ?Time length at startup */
1998-02-27 07:54:42 +03:00
*error = 0;
1999-04-30 16:41:41 +04:00
if (i18n_timelength == 0) {
i18n_timelength = i18n_checktimelength (); /* Musn't be 0 */
1999-04-30 16:41:41 +04:00
for (i = 0; i < ELEMENTS(formats); i++)
if (strcmp ("time", formats [i].id+1) == 0)
formats [i].min_size = i18n_timelength;
}
1998-02-27 07:54:42 +03:00
/*
* This makes sure that the panel and mini status full/half mode
* setting is equal
*/
format = parse_panel_size (panel, format, isstatus);
while (*format){ /* format can be an empty string */
1998-02-27 07:54:42 +03:00
int found = 0;
darr = g_new (format_e, 1);
1998-02-27 07:54:42 +03:00
/* I'm so ugly, don't look at me :-) */
if (!home)
home = old = darr;
1998-02-27 07:54:42 +03:00
old->next = darr;
darr->next = 0;
old = darr;
1998-02-27 07:54:42 +03:00
format = skip_separators (format);
if (strchr ("<=>", *format)){
1998-02-27 07:54:42 +03:00
set_justify = 1;
switch (*format)
{
case '<':
justify = J_LEFT;
break;
case '=':
justify = J_CENTER;
break;
case '>':
default:
justify = J_RIGHT;
break;
}
1998-02-27 07:54:42 +03:00
format = skip_separators (format+1);
} else
set_justify = 0;
1998-02-27 07:54:42 +03:00
for (i = 0; i < ELEMENTS(formats); i++){
int klen = strlen (formats [i].id);
if (strncmp (format, formats [i].id, klen) != 0)
continue;
format += klen;
if (formats [i].use_in_gui)
items++;
darr->use_in_gui = formats [i].use_in_gui;
darr->requested_field_len = formats [i].min_size;
darr->string_fn = formats [i].string_fn;
if (formats [i].title [0])
darr->title = _(formats [i].title);
else
darr->title = "";
1998-02-27 07:54:42 +03:00
darr->id = formats [i].id;
darr->expand = formats [i].expands;
darr->just_mode = formats [i].default_just;
if (set_justify) {
if (IS_FIT(darr->just_mode))
darr->just_mode = MAKE_FIT(justify);
else
darr->just_mode = justify;
}
1998-02-27 07:54:42 +03:00
found = 1;
format = skip_separators (format);
/* If we have a size specifier */
if (*format == ':'){
int req_length;
/* If the size was specified, we don't want
* auto-expansion by default
*/
darr->expand = 0;
format++;
req_length = atoi (format);
darr->requested_field_len = req_length;
format = skip_numbers (format);
/* Now, if they insist on expansion */
if (*format == '+'){
darr->expand = 1;
format++;
}
}
break;
}
if (!found){
char old_char;
1998-02-27 07:54:42 +03:00
int pos = min (8, strlen (format));
delete_format (home);
old_char = format [pos];
format [pos] = 0;
*error = g_strconcat (_("Unknow tag on display format: "), format, NULL);
1998-02-27 07:54:42 +03:00
format [pos] = old_char;
return 0;
}
total_cols += darr->requested_field_len;
}
1998-02-27 07:54:42 +03:00
*res_total_cols = total_cols;
if (home)
home->items = items;
1998-02-27 07:54:42 +03:00
return home;
}
format_e *
use_display_format (WPanel *panel, char *format, char **error, int isstatus)
{
#define MAX_EXPAND 4
int expand_top = 0; /* Max used element in expand */
int usable_columns; /* Usable columns in the panel */
int total_cols;
char *expand_list [MAX_EXPAND]; /* Expand at most 4 fields. */
int i;
format_e *darr, *home;
1998-02-27 07:54:42 +03:00
if (!format)
format = DEFAULT_USER_FORMAT;
1998-02-27 07:54:42 +03:00
home = parse_display_format (panel, format, error, isstatus, &total_cols);
if (*error)
return 0;
1998-02-27 07:54:42 +03:00
/* Status needn't to be split */
usable_columns = ((panel->widget.cols-2)/((isstatus)
? 1
: (panel->split+1))) - (!isstatus && panel->split);
1998-02-27 07:54:42 +03:00
/* Clean expand list */
for (i = 0; i < MAX_EXPAND; i++)
expand_list [i] = '\0';
1998-02-27 07:54:42 +03:00
/* Look for the expandable fields and set field_len based on the requested field len */
for (darr = home; darr && expand_top < MAX_EXPAND; darr = darr->next){
darr->field_len = darr->requested_field_len;
if (darr->expand)
expand_list [expand_top++] = darr->id;
}
1998-02-27 07:54:42 +03:00
/* If we used more columns than the available columns, adjust that */
if (total_cols > usable_columns){
int pdif, dif = total_cols - usable_columns;
1998-02-27 07:54:42 +03:00
while (dif){
pdif = dif;
for (darr = home; darr; darr = darr->next){
if (dif && darr->field_len - 1){
darr->field_len--;
dif--;
}
}
/* avoid endless loop if num fields > 40 */
if (pdif == dif)
break;
1998-02-27 07:54:42 +03:00
}
total_cols = usable_columns; /* give up, the rest should be truncated */
}
/* Expand the available space */
if ((usable_columns > total_cols) && expand_top){
int spaces = (usable_columns - total_cols) / expand_top;
int extra = (usable_columns - total_cols) % expand_top;
for (i = 0, darr = home; darr && (i < expand_top); darr = darr->next)
if (darr->expand){
darr->field_len += (spaces + ((i == 0) ? extra : 0));
i++;
}
}
return home;
}
/* Switches the panel to the mode specified in the format */
/* Seting up both format and status string. Return: 0 - on success; */
/* 1 - format error; 2 - status error; 3 - errors in both formats. */
int
set_panel_formats (WPanel *p)
{
format_e *form;
char *err;
1998-02-27 07:54:42 +03:00
int retcode = 0;
1998-02-27 07:54:42 +03:00
form = use_display_format (p, panel_format (p), &err, 0);
1998-02-27 07:54:42 +03:00
if (err){
g_free (err);
1998-02-27 07:54:42 +03:00
retcode = 1;
}
else {
if (p->format)
delete_format (p->format);
1998-02-27 07:54:42 +03:00
p->format = form;
}
1998-02-27 07:54:42 +03:00
if (show_mini_info){
1998-02-27 07:54:42 +03:00
form = use_display_format (p, mini_status_format (p), &err, 1);
1998-02-27 07:54:42 +03:00
if (err){
g_free (err);
1998-02-27 07:54:42 +03:00
retcode += 2;
}
else {
if (p->status_format)
delete_format (p->status_format);
1998-02-27 07:54:42 +03:00
p->status_format = form;
}
}
1998-02-27 07:54:42 +03:00
panel_format_modified (p);
panel_update_cols (&(p->widget), p->frame_size);
if (retcode)
message( 1, _(" Warning " ), _( "User suplied format looks invalid, reverting to default." ) );
if (retcode & 0x01){
g_free (p->user_format);
p->user_format = g_strdup (DEFAULT_USER_FORMAT);
}
if (retcode & 0x02){
g_free (p->user_status_format [p->list_type]);
p->user_status_format [p->list_type] = g_strdup (DEFAULT_USER_FORMAT);
}
1998-02-27 07:54:42 +03:00
return retcode;
}
/* Given the panel->view_type returns the format string to be parsed */
char *
panel_format (WPanel *panel)
{
switch (panel->list_type){
case list_long:
return "full perm,space,nlink,space,owner,space,group,space,size,space,mtime,space,name";
case list_brief:
return "half 2,type,name";
case list_user:
return panel->user_format;
default:
case list_full:
return "half type,name,|,size,|,mtime";
}
}
char *
mini_status_format (WPanel *panel)
{
if (panel->user_mini_status)
return panel->user_status_format [panel->list_type];
switch (panel->list_type){
case list_long:
return "full perm,space,nlink,space,owner,space,group,space,size,space,mtime,space,name";
case list_brief:
return "half type,name,space,bsize,space,perm,space";
case list_full:
return "half type,name";
default:
case list_user:
return panel->user_format;
}
}
/* */
/* Panel operation commands */
/* */
/* Returns the number of items in the given panel */
int
ITEMS (WPanel *p)
{
#ifdef HAVE_TK
return p->widget.lines;
#else
if (p->split)
return llines (p) * 2;
else
return llines (p);
#endif
}
/* This function sets redisplays the selection */
void
select_item (WPanel *panel)
{
int repaint = 0;
int items = ITEMS (panel);
1998-02-27 07:54:42 +03:00
/* Although currently all over the code we set the selection and
top file to decent values before calling select_item, I could
forget it someday, so it's better to do the actual fitting here */
#ifdef HAVE_X
int old_top;
old_top = panel->top_file;
#endif
if (is_a_desktop_panel (panel))
return;
1998-02-27 07:54:42 +03:00
if (panel->top_file < 0){
repaint = 1;
panel->top_file = 0;
}
if (panel->selected < 0)
panel->selected = 0;
if (panel->selected > panel->count-1)
panel->selected = panel->count - 1;
if (panel->top_file > panel->count-1){
repaint = 1;
panel->top_file = panel->count-1;
}
1998-02-27 07:54:42 +03:00
if ((panel->count - panel->top_file) < items){
repaint = 1;
panel->top_file = panel->count - items;
if (panel->top_file < 0)
panel->top_file = 0;
}
1998-02-27 07:54:42 +03:00
if (panel->selected < panel->top_file){
repaint = 1;
panel->top_file = panel->selected;
}
if ((panel->selected - panel->top_file) >= items){
repaint = 1;
panel->top_file = panel->selected - items + 1;
}
#ifndef HAVE_X
1998-02-27 07:54:42 +03:00
if (repaint)
paint_panel (panel);
else
repaint_file (panel, panel->selected, 1, 2*selection (panel)->f.marked+1, 0);
#else
if (old_top != panel->top_file)
x_adjust_top_file (panel);
x_select_item (panel);
#endif
display_mini_info (panel);
execute_hooks (select_file_hook);
}
/* Clears all files in the panel, used only when one file was marked */
void
unmark_files (WPanel *panel)
{
int i;
if (!panel->marked)
return;
for (i = 0; i < panel->count; i++)
file_mark (panel, i, 0);
panel->dirs_marked = 0;
panel->marked = 0;
panel->total = 0;
}
#ifdef HAVE_X
void
unselect_item (WPanel *panel)
{
x_unselect_item (panel);
}
#else
void
unselect_item (WPanel *panel)
{
repaint_file (panel, panel->selected, 1, 2*selection (panel)->f.marked, 0);
}
#endif
static void
do_move_down (WPanel *panel)
1998-02-27 07:54:42 +03:00
{
if (panel->selected+1 == panel->count)
return;
1998-02-27 07:54:42 +03:00
unselect_item (panel);
panel->selected++;
#ifndef HAVE_X
if (panel->selected - panel->top_file == ITEMS (panel) &&
panel_scroll_pages){
/* Scroll window half screen */
panel->top_file += ITEMS (panel)/2;
if (panel->top_file > panel->count - ITEMS (panel))
panel->top_file = panel->count - ITEMS (panel);
paint_dir (panel);
select_item (panel);
}
#endif
select_item (panel);
}
static void
do_move_up (WPanel *panel)
1998-02-27 07:54:42 +03:00
{
if (panel->selected == 0)
return;
unselect_item (panel);
panel->selected--;
#ifndef HAVE_X
if (panel->selected < panel->top_file && panel_scroll_pages){
/* Scroll window half screen */
panel->top_file -= ITEMS (panel)/2;
if (panel->top_file < 0) panel->top_file = 0;
paint_dir (panel);
}
#endif
select_item (panel);
}
static void
move_rel (WPanel *panel, int rel)
{
unselect_item (panel);
if (rel < 0){
if (panel->selected + rel < 0)
panel->selected = 0;
else
panel->selected = panel->selected + rel;
} else {
if (panel->selected + rel >= panel->count)
panel->selected = panel->count - 1;
else
panel->selected = panel->selected + rel;
}
select_item (panel);
}
/* */
/* Panel key binded commands */
/* */
static void
move_up (WPanel *panel)
{
if (panel->list_type == list_icons){
move_rel (panel, -ICONS_PER_ROW (panel));
} else
do_move_up (panel);
}
static void
move_down (WPanel *panel)
{
if (panel->list_type == list_icons){
move_rel (panel, ICONS_PER_ROW (panel));
} else
do_move_down (panel);
}
1998-02-27 07:54:42 +03:00
/* Changes the selection by lines (may be negative) */
static void
move_selection (WPanel *panel, int lines)
{
int new_pos;
#ifndef HAVE_X
1998-02-27 07:54:42 +03:00
int adjust = 0;
#endif
1998-02-27 07:54:42 +03:00
new_pos = panel->selected + lines;
if (new_pos >= panel->count)
new_pos = panel->count-1;
if (new_pos < 0)
new_pos = 0;
unselect_item (panel);
panel->selected = new_pos;
#ifndef HAVE_X
if (panel->selected - panel->top_file >= ITEMS (panel)){
panel->top_file += lines;
adjust = 1;
}
1998-02-27 07:54:42 +03:00
if (panel->selected - panel->top_file < 0){
panel->top_file += lines;
adjust = 1;
}
1998-02-27 07:54:42 +03:00
if (adjust){
if (panel->top_file > panel->selected)
panel->top_file = panel->selected;
if (panel->top_file < 0)
panel->top_file = 0;
paint_dir (panel);
}
#endif
select_item (panel);
}
static int
move_left (WPanel *panel, int c_code)
{
if (panel->list_type == list_icons){
do_move_up (panel);
1998-02-27 07:54:42 +03:00
return 1;
} else {
if (panel->split){
move_selection (panel, -llines (panel));
return 1;
} else
return maybe_cd (c_code, 0);
}
1998-02-27 07:54:42 +03:00
}
static int
move_right (WPanel *panel, int c_code)
{
if (panel->list_type == list_icons){
do_move_down (panel);
1998-02-27 07:54:42 +03:00
return 1;
} else {
if (panel->split){
move_selection (panel, llines (panel));
return 1;
} else
return maybe_cd (c_code, 1);
}
1998-02-27 07:54:42 +03:00
}
static void
prev_page (WPanel *panel)
{
int items;
if (!panel->selected && !panel->top_file)
return;
unselect_item (panel);
items = ITEMS (panel);
if (panel->top_file < items)
items = panel->top_file;
if (!items)
panel->selected = 0;
else
panel->selected -= items;
panel->top_file -= items;
/* This keeps the selection in a reasonable place */
if (panel->selected < 0)
panel->selected = 0;
if (panel->top_file < 0)
panel->top_file = 0;
x_adjust_top_file (panel);
select_item (panel);
#ifndef HAVE_X
paint_dir (panel);
#endif
}
static void
prev_page_key (WPanel *panel)
{
if (console_flag && ctrl_pressed ()){
do_cd ("..", cd_exact);
} else
prev_page (panel);
}
static void
next_page (WPanel *panel)
{
int items;
if (panel->selected == panel->count - 1)
return;
unselect_item (panel);
items = ITEMS (panel);
if (panel->top_file > panel->count - 2 * items)
items = panel->count - items - panel->top_file;
if (panel->top_file + items < 0)
items = - panel->top_file;
if (!items)
panel->selected = panel->count - 1;
else
panel->selected += items;
panel->top_file += items;
/* This keeps the selection in it's relative position */
if (panel->selected >= panel->count)
panel->selected = panel->count - 1;
if (panel->top_file >= panel->count)
panel->top_file = panel->count - 1;
x_adjust_top_file (panel);
select_item (panel);
#ifndef HAVE_X
paint_dir (panel);
#endif
}
static void next_page_key (WPanel *panel)
{
if (console_flag&&ctrl_pressed()&&
(S_ISDIR(selection (panel)->buf.st_mode) ||
link_isdir (selection (panel))))
do_cd (selection (panel)->fname, cd_exact);
else
next_page (panel);
}
static void
goto_top_file (WPanel *panel)
{
unselect_item (panel);
panel->selected = panel->top_file;
select_item (panel);
}
static void
goto_middle_file (WPanel *panel)
{
unselect_item (panel);
panel->selected = panel->top_file + (ITEMS (panel)/2);
if (panel->selected >= panel->count)
panel->selected = panel->count - 1;
select_item (panel);
}
static void
goto_bottom_file (WPanel *panel)
{
unselect_item (panel);
panel->selected = panel->top_file + ITEMS (panel)-1;
if (panel->selected >= panel->count)
panel->selected = panel->count - 1;
select_item (panel);
}
static void
move_home (WPanel *panel)
{
if (panel->selected == 0)
return;
unselect_item (panel);
1998-02-27 07:54:42 +03:00
if (torben_fj_mode){
int middle_pos = panel->top_file + (ITEMS (panel)/2);
if (panel->selected > middle_pos){
goto_middle_file (panel);
return;
}
if (panel->selected != panel->top_file){
goto_top_file (panel);
return;
}
}
1998-02-27 07:54:42 +03:00
panel->top_file = 0;
panel->selected = 0;
#ifndef HAVE_X
paint_dir (panel);
#endif
select_item (panel);
}
static void
move_end (WPanel *panel)
{
if (panel->selected == panel->count-1)
return;
unselect_item (panel);
if (torben_fj_mode){
int middle_pos = panel->top_file + (ITEMS (panel)/2);
if (panel->selected < middle_pos){
goto_middle_file (panel);
return;
}
if (panel->selected != (panel->top_file + ITEMS(panel)-1)){
goto_bottom_file (panel);
return;
}
}
1998-02-27 07:54:42 +03:00
panel->selected = panel->count-1;
#ifndef HAVE_X
paint_dir (panel);
#endif
select_item (panel);
}
/* Recalculate the panels summary information, used e.g. when marked
files might have been removed by an external command */
void
recalculate_panel_summary (WPanel *panel)
{
int i;
panel->marked = 0;
panel->dirs_marked = 0;
panel->total = 0;
for (i = 0; i < panel->count; i++)
if (panel->dir.list [i].f.marked){
/* do_file_mark will return immediately if newmark == oldmark.
So we have to first unmark it to get panel's summary information
updated. (Norbert) */
panel->dir.list [i].f.marked = 0;
do_file_mark (panel, i, 1);
}
}
1998-02-27 07:54:42 +03:00
/* This routine marks a file or a directory */
void
do_file_mark (WPanel *panel, int idx, int mark)
{
if (panel->dir.list [idx].f.marked == mark)
return;
/*
* Only '..' can't be marked, '.' isn't visible.
*/
if (strcmp (panel->dir.list [idx].fname, "..")){
file_mark (panel, idx, mark);
if (panel->dir.list [idx].f.marked){
panel->marked++;
if (S_ISDIR (panel->dir.list [idx].buf.st_mode)) {
if (panel->dir.list [idx].f.dir_size_computed)
1998-02-27 07:54:42 +03:00
panel->total += panel->dir.list [idx].buf.st_size;
panel->dirs_marked++;
} else
1998-02-27 07:54:42 +03:00
panel->total += panel->dir.list [idx].buf.st_size;
set_colors (panel);
} else {
if (S_ISDIR(panel->dir.list [idx].buf.st_mode)) {
if (panel->dir.list [idx].f.dir_size_computed)
1998-02-27 07:54:42 +03:00
panel->total -= panel->dir.list [idx].buf.st_size;
panel->dirs_marked--;
} else
panel->total -= panel->dir.list [idx].buf.st_size;
panel->marked--;
}
}
}
void
do_file_mark_range (WPanel *panel, int r1, int r2)
{
const int start = min (r1, r2);
const int end = max (r1, r2);
int i, mark;
mark = !panel->dir.list [start].f.marked;
for (i = start; i < end; i++)
do_file_mark (panel, i, mark);
}
1998-02-27 07:54:42 +03:00
static void
do_mark_file (WPanel *panel, int do_move)
{
int idx = panel->selected;
1998-02-27 07:54:42 +03:00
do_file_mark (panel, idx, selection (panel)->f.marked ? 0 : 1);
repaint_file (panel, idx, 1, 2*panel->dir.list [idx].f.marked+1, 0);
if (mark_moves_down && do_move)
move_down (panel);
display_mini_info (panel);
}
static void
mark_file (WPanel *panel)
{
do_mark_file (panel, 1);
}
/* Incremental search of a file name in the panel */
static void
do_search (WPanel *panel, int c_code)
{
int l, i;
int wrapped = 0;
int found;
l = strlen (panel->search_buffer);
if (l && (c_code == 8 || c_code == 0177 || c_code == KEY_BACKSPACE))
panel->search_buffer [--l] = 0;
else {
if (c_code && l < sizeof (panel->search_buffer)){
panel->search_buffer [l] = c_code;
panel->search_buffer [l+1] = 0;
l++;
}
}
found = 0;
for (i = panel->selected; !wrapped || i != panel->selected; i++){
if (i >= panel->count){
i = 0;
if (wrapped)
break;
wrapped = 1;
}
if (STRNCOMP (panel->dir.list [i].fname, panel->search_buffer, l) == 0){
1998-02-27 07:54:42 +03:00
unselect_item (panel);
panel->selected = i;
select_item (panel);
found = 1;
break;
}
}
if (!found)
panel->search_buffer [--l] = 0;
#ifndef HAVE_X
1998-02-27 07:54:42 +03:00
paint_panel (panel);
#endif
}
void
1998-02-27 07:54:42 +03:00
start_search (WPanel *panel)
{
if (panel->searching){
if (panel->selected+1 == panel->count)
panel->selected = 0;
else
do_move_down (panel);
1998-02-27 07:54:42 +03:00
do_search (panel, 0);
} else {
panel->searching = 1;
panel->search_buffer [0] = 0;
display_mini_info (panel);
mc_refresh ();
}
}
/* This procedure was getting really messy with all the rewriting and ifdef's
* so I just splet them out. -jrb */
#ifdef HAVE_GNOME
extern void set_cursor_busy (WPanel *panel);
extern void set_cursor_normal (WPanel *panel);
int
do_enter_on_file_entry (file_entry *fe)
{
gint retval;
char *full_name;
set_cursor_busy (cpanel);
/* Can we change dirs? */
if (S_ISDIR (fe->buf.st_mode) || link_isdir (fe)) {
do_cd (fe->fname, cd_exact);
set_cursor_normal (cpanel);
return 1;
}
/* do metadata/mime stuff tell us anything? */
if (gmc_open (fe) != 0) {
set_cursor_normal (cpanel);
return 1;
}
/* can we change dirs? */
full_name = concat_dir_and_file (cpanel->cwd, fe->fname);
if (is_exe (fe->buf.st_mode) && if_link_is_exe (full_name, fe)) {
g_free (full_name);
#ifdef USE_VFS
if (vfs_current_is_local ())
#endif
{
char *tmp = name_quote (fe->fname, 0);
char *cmd = g_strconcat (".", PATH_SEP_STR, tmp, NULL);
2000-04-18 12:58:42 +04:00
g_free (tmp);
if (!confirm_execute || (query_dialog (_(" The Midnight Commander "),
_(" Do you really want to execute? "),
0, 2, _("Yes"), _("No")) == 0))
execute (cmd);
g_free (cmd);
}
#ifdef USE_VFS
else {
char *tmp;
tmp = concat_dir_and_file (vfs_get_current_dir(), fe->fname);
if (!mc_setctl (tmp, MCCTL_EXTFS_RUN, NULL))
message (1, _(" Warning "), _(" No action taken "));
g_free (tmp);
}
#endif /* USE_VFS */
set_cursor_normal (cpanel);
return 1;
}
g_free (full_name);
/* looks like we couldn't open it. Let's ask the user */
retval = gmc_open_with (fe->fname);
set_cursor_normal (cpanel);
return retval;
}
#else
int
do_enter_on_file_entry (file_entry *fe)
1998-02-27 07:54:42 +03:00
{
char *full_name;
if (S_ISDIR (fe->buf.st_mode) || link_isdir (fe)) {
do_cd (fe->fname, cd_exact);
return 1;
} else {
full_name = concat_dir_and_file (cpanel->cwd, fe->fname);
if (is_exe (fe->buf.st_mode) && if_link_is_exe (full_name, fe)) {
g_free (full_name);
#ifdef USE_VFS
if (vfs_current_is_local ())
#endif
1998-02-27 07:54:42 +03:00
{
char *tmp = name_quote (fe->fname, 0);
char *cmd = g_strconcat (".", PATH_SEP_STR, tmp, NULL);
2000-04-18 12:58:42 +04:00
g_free (tmp);
if (!confirm_execute || (query_dialog (_(" The Midnight Commander "),
_(" Do you really want to execute? "),
0, 2, _("&Yes"), _("&No")) == 0))
1998-02-27 07:54:42 +03:00
execute (cmd);
g_free (cmd);
}
#ifdef USE_VFS
1998-10-13 02:07:53 +04:00
else {
/* if (vfs_current_is_extfs ()) - I see no reason why
filesystems other than extfs could not implement same
call... -- pavel@ucw.cz*/
char *tmp;
1998-02-27 07:54:42 +03:00
tmp = concat_dir_and_file (vfs_get_current_dir(), fe->fname);
1998-10-13 02:07:53 +04:00
if (!mc_setctl (tmp, MCCTL_EXTFS_RUN, NULL))
message (1, _(" Warning "), _(" No action taken "));
g_free (tmp);
1998-02-27 07:54:42 +03:00
}
#endif /* USE_VFS */
return 1;
} else {
char *p;
g_free (full_name);
p = regex_command (fe->fname, "Open", NULL, 0);
if (p && (strcmp (p, "Success") == 0))
return 1;
else
return 0;
}
1998-02-27 07:54:42 +03:00
}
}
#endif /* else not HAVE_GNOME */
int
do_enter (WPanel *panel)
{
return do_enter_on_file_entry (selection (panel));
1998-02-27 07:54:42 +03:00
}
static void
chdir_other_panel (WPanel *panel)
{
char *new_dir;
if (get_other_type () != view_listing)
return;
if (!S_ISDIR (panel->dir.list [panel->selected].buf.st_mode))
new_dir = concat_dir_and_file (panel->cwd, "..");
1998-02-27 07:54:42 +03:00
else
new_dir = concat_dir_and_file (panel->cwd, panel->dir.list [panel->selected].fname);
1998-02-27 07:54:42 +03:00
change_panel ();
do_cd (new_dir, cd_exact);
change_panel ();
move_down (panel);
g_free (new_dir);
1998-02-27 07:54:42 +03:00
}
static void
chdir_to_readlink (WPanel *panel)
{
char *new_dir;
if (get_other_type () != view_listing)
return;
if (S_ISLNK (panel->dir.list [panel->selected].buf.st_mode)) {
1998-02-27 07:54:42 +03:00
char buffer [MC_MAXPATHLEN], *p;
int i;
struct stat mybuf;
i = readlink (selection (panel)->fname, buffer, MC_MAXPATHLEN);
1998-02-27 07:54:42 +03:00
if (i < 0)
return;
if (mc_stat (selection (panel)->fname, &mybuf) < 0)
1998-02-27 07:54:42 +03:00
return;
buffer [i] = 0;
if (!S_ISDIR (mybuf.st_mode)) {
p = strrchr (buffer, PATH_SEP);
if (p && !p[1]) {
*p = 0;
p = strrchr (buffer, PATH_SEP);
}
if (!p)
return;
p[1] = 0;
}
if (*buffer == PATH_SEP)
new_dir = g_strdup (buffer);
1998-02-27 07:54:42 +03:00
else
new_dir = concat_dir_and_file (panel->cwd, buffer);
1998-02-27 07:54:42 +03:00
change_panel ();
do_cd (new_dir, cd_exact);
change_panel ();
1998-02-27 07:54:42 +03:00
move_down (panel);
g_free (new_dir);
1998-02-27 07:54:42 +03:00
}
}
2000-04-18 12:58:42 +04:00
static const key_map panel_keymap [] = {
1998-02-27 07:54:42 +03:00
{ KEY_DOWN, move_down },
{ KEY_UP, move_up },
/* The action button :-) */
1998-09-12 03:27:49 +04:00
{ '\n', (key_callback) do_enter },
{ KEY_ENTER, (key_callback) do_enter },
1998-02-27 07:54:42 +03:00
{ KEY_IC, mark_file },
{ KEY_HOME, move_home },
{ KEY_C1, move_end },
{ KEY_END, move_end },
{ KEY_A1, move_home },
{ KEY_NPAGE, next_page_key },
{ KEY_PPAGE, prev_page_key },
/* To quickly move in the panel */
{ ALT('g'), goto_top_file },
{ ALT('r'), goto_middle_file }, /* M-r like emacs */
{ ALT('j'), goto_bottom_file },
#ifdef OS2_NT
{ ALT(KEY_F(11)), drive_cmd_a },
{ ALT(KEY_F(12)), drive_cmd_b },
{ ALT('d'), drive_chg },
#endif
1998-02-27 07:54:42 +03:00
/* Emacs-like bindings */
{ XCTRL('v'), next_page }, /* C-v like emacs */
{ ALT('v'), prev_page }, /* M-v like emacs */
{ XCTRL('p'), move_up }, /* C-p like emacs */
{ XCTRL('n'), move_down }, /* C-n like emacs */
{ XCTRL('s'), start_search }, /* C-s like emacs */
{ ALT('s'), start_search }, /* M-s not like emacs */
{ XCTRL('t'), mark_file },
{ ALT('o'), chdir_other_panel },
{ ALT('l'), chdir_to_readlink },
{ KEY_F(13), view_simple_cmd },
{ KEY_F(14), edit_cmd_new },
{ ALT('y'), directory_history_prev },
{ ALT('u'), directory_history_next },
{ ALT('H'), directory_history_list },
Wed Apr 1 00:15:30 1998 Norbert Warmuth <k3190@fh-sw.de> * key.c, key.h (numeric_keypad_mode, application_keypad_mode): New functions which encapsulate two hardcoded escape sequences from main.c. * main.c (main): Use the two new functions from key.c * main.c, screen.c: Moved all file selection keys from the default keymap to the keymap for panels in listing mode. Changed *_selection_cmd to *_selection_cmd_panel in panel_keymap (functions in panel_keymap get a WPanel * as first parameter, i.e. the indirection with cpanel isn't necessary). * main.c (midnight_callback): Keys '*' and '-' were not treated when only_leading_plus_minus==0; Optimized the if-clauses a little bit (i.e. removed duplicate checks). More optimation is possible but it would make the whole stuff completly unreadable. * key.c (correct_key_code): KP_ADD, KP_SUBTRACT and KP_MULTIPLY will be translated to +, - and * only if the option alternate_plus_minus is turned off. * learn.c (learn_keys): Turn alternate_plus_minus temporarily on to avoid translation of KP_ADD, KP_SUBTRACT and KP_MULTIPLY in correct_key_code/make sure keypad is in application mode (makes it possible to learn this keys). * cmd.c (reverse_selection_cmd_panel): New function (renamed from reverse_selection_cmd, takes a WPanel * as parameter, references to cpanel changed to panel/the passed parameter). reverse_selection_cmd now simply calls this function with cpanel. This pair was missing among the *_selection_cmd* functions. * cmd.h: Added function prototypes.
1998-04-01 02:36:24 +04:00
{ ALT('+'), select_cmd_panel },
{ KEY_KP_ADD, select_cmd_panel },
{ ALT('\\'), unselect_cmd_panel },
{ ALT('-'), unselect_cmd_panel },
{ KEY_KP_SUBTRACT, unselect_cmd_panel },
{ ALT('*'), reverse_selection_cmd_panel },
{ KEY_KP_MULTIPLY, reverse_selection_cmd_panel },
Wed Apr 1 00:15:30 1998 Norbert Warmuth <k3190@fh-sw.de> * key.c, key.h (numeric_keypad_mode, application_keypad_mode): New functions which encapsulate two hardcoded escape sequences from main.c. * main.c (main): Use the two new functions from key.c * main.c, screen.c: Moved all file selection keys from the default keymap to the keymap for panels in listing mode. Changed *_selection_cmd to *_selection_cmd_panel in panel_keymap (functions in panel_keymap get a WPanel * as first parameter, i.e. the indirection with cpanel isn't necessary). * main.c (midnight_callback): Keys '*' and '-' were not treated when only_leading_plus_minus==0; Optimized the if-clauses a little bit (i.e. removed duplicate checks). More optimation is possible but it would make the whole stuff completly unreadable. * key.c (correct_key_code): KP_ADD, KP_SUBTRACT and KP_MULTIPLY will be translated to +, - and * only if the option alternate_plus_minus is turned off. * learn.c (learn_keys): Turn alternate_plus_minus temporarily on to avoid translation of KP_ADD, KP_SUBTRACT and KP_MULTIPLY in correct_key_code/make sure keypad is in application mode (makes it possible to learn this keys). * cmd.c (reverse_selection_cmd_panel): New function (renamed from reverse_selection_cmd, takes a WPanel * as parameter, references to cpanel changed to panel/the passed parameter). reverse_selection_cmd now simply calls this function with cpanel. This pair was missing among the *_selection_cmd* functions. * cmd.h: Added function prototypes.
1998-04-01 02:36:24 +04:00
#ifdef HAVE_GNOME
Wed Apr 1 00:15:30 1998 Norbert Warmuth <k3190@fh-sw.de> * key.c, key.h (numeric_keypad_mode, application_keypad_mode): New functions which encapsulate two hardcoded escape sequences from main.c. * main.c (main): Use the two new functions from key.c * main.c, screen.c: Moved all file selection keys from the default keymap to the keymap for panels in listing mode. Changed *_selection_cmd to *_selection_cmd_panel in panel_keymap (functions in panel_keymap get a WPanel * as first parameter, i.e. the indirection with cpanel isn't necessary). * main.c (midnight_callback): Keys '*' and '-' were not treated when only_leading_plus_minus==0; Optimized the if-clauses a little bit (i.e. removed duplicate checks). More optimation is possible but it would make the whole stuff completly unreadable. * key.c (correct_key_code): KP_ADD, KP_SUBTRACT and KP_MULTIPLY will be translated to +, - and * only if the option alternate_plus_minus is turned off. * learn.c (learn_keys): Turn alternate_plus_minus temporarily on to avoid translation of KP_ADD, KP_SUBTRACT and KP_MULTIPLY in correct_key_code/make sure keypad is in application mode (makes it possible to learn this keys). * cmd.c (reverse_selection_cmd_panel): New function (renamed from reverse_selection_cmd, takes a WPanel * as parameter, references to cpanel changed to panel/the passed parameter). reverse_selection_cmd now simply calls this function with cpanel. This pair was missing among the *_selection_cmd* functions. * cmd.h: Added function prototypes.
1998-04-01 02:36:24 +04:00
{ '+', select_cmd_panel },
{ '\\', unselect_cmd_panel },
{ '-', unselect_cmd_panel },
{ '*', reverse_selection_cmd_panel },
{ XCTRL('r'), reread_cmd },
{ KEY_F(3), view_panel_cmd },
{ KEY_F(4), edit_panel_cmd },
{ KEY_F(5), copy_cmd },
{ KEY_F(6), ren_cmd },
{ KEY_F(7), mkdir_panel_cmd },
{ KEY_F(8), delete_cmd },
{ KEY_DC, delete_cmd },
#endif
1998-02-27 07:54:42 +03:00
{ 0, 0 }
};
1998-02-27 07:54:42 +03:00
static inline int
panel_key (WPanel *panel, int key)
{
int i;
for (i = 0; panel_keymap [i].key_code; i++){
if (key == panel_keymap [i].key_code){
if (panel_keymap [i].fn != start_search)
panel->searching = 0;
(*panel_keymap [i].fn)(panel);
return 1;
}
}
if (torben_fj_mode && key == ALT('h')) {
1998-02-27 07:54:42 +03:00
goto_middle_file (panel);
return 1;
}
1998-02-27 07:54:42 +03:00
/* We do not want to take a key press if nothing can be done with it */
/* The command line widget may do something more usefull */
if (key == KEY_LEFT)
return move_left (panel, key);
if (key == KEY_RIGHT)
return move_right (panel, key);
1998-02-27 07:54:42 +03:00
if (is_abort_char (key)) {
panel->searching = 0;
display_mini_info (panel);
return 1;
}
/* Do not eat characters not meant for the panel below ' ' (e.g. C-l). */
if ((key >= ' '&& key <= 255) || key == 8 || key == KEY_BACKSPACE) {
1998-02-27 07:54:42 +03:00
if (panel->searching){
do_search (panel, key);
return 1;
}
if (!command_prompt) {
start_search (panel);
do_search (panel, key);
return 1;
1998-02-27 07:54:42 +03:00
}
}
return 0;
1998-02-27 07:54:42 +03:00
}
2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * src/user.c: Add macro %k it is block file name Add macro %e it is error file name Add macro %i it is cursor column indent of spaces, only for edit Add macro %y, it is syntax of current file in editor, only for edit Add condition y , it is syntax pattern of current file in edit Add macro %x it is extension of current file Add macro %m it is current menu filename 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * gtkedit/editcmd.c: Modify External Formatter (was C indent formatter) . Autocreate a scripts in home, from templates /usr/lib/mc/edit.indent.rc, edit.ispell.rc, etc. Remove leading and trailing spaces into _(""), (the message of David H. Martin <dmartina@usa.net>) 2000-05-05 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * src/user.c: Add condition (x filename) into mc.menu . for "Open next a free console" and like. 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> src/user.c: fix segfault in chunk_alloc of glibc, when into condition of .mnu we have quoted space. (~.mc/menu: + f \.\ test$). 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * gtkedit/edit.c: Add ability user edit menus: system: /usr/lib/mc/cedit.menu, * gtkedit/editmenu.c: home: ~/.cedit.menu, local: .cedit.menu Marked block is access now from an user edit menu Access ~/.cedit/cooledit.block for insert to cursor place from user edit menu. Created system cedit.menu 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * gtkedit/editdraw.c: Improved a status string of cool editor for best understand, and to add char,hex view. 2000-05-04 Richard Hestilow <hestgray@ionet.net> * gnome/gdesktop.c: Fixes snap to grid.
2000-05-09 05:18:27 +04:00
void user_file_menu_cmd (void) { user_menu_cmd (NULL); }
static int
1998-02-27 07:54:42 +03:00
panel_callback (Dlg_head *h, WPanel *panel, int msg, int par)
{
switch (msg){
case WIDGET_INIT:
#ifdef HAVE_X
define_label (h, (Widget *)panel, 1, _("Help"), help_cmd);
2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * src/user.c: Add macro %k it is block file name Add macro %e it is error file name Add macro %i it is cursor column indent of spaces, only for edit Add macro %y, it is syntax of current file in editor, only for edit Add condition y , it is syntax pattern of current file in edit Add macro %x it is extension of current file Add macro %m it is current menu filename 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * gtkedit/editcmd.c: Modify External Formatter (was C indent formatter) . Autocreate a scripts in home, from templates /usr/lib/mc/edit.indent.rc, edit.ispell.rc, etc. Remove leading and trailing spaces into _(""), (the message of David H. Martin <dmartina@usa.net>) 2000-05-05 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * src/user.c: Add condition (x filename) into mc.menu . for "Open next a free console" and like. 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> src/user.c: fix segfault in chunk_alloc of glibc, when into condition of .mnu we have quoted space. (~.mc/menu: + f \.\ test$). 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * gtkedit/edit.c: Add ability user edit menus: system: /usr/lib/mc/cedit.menu, * gtkedit/editmenu.c: home: ~/.cedit.menu, local: .cedit.menu Marked block is access now from an user edit menu Access ~/.cedit/cooledit.block for insert to cursor place from user edit menu. Created system cedit.menu 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * gtkedit/editdraw.c: Improved a status string of cool editor for best understand, and to add char,hex view. 2000-05-04 Richard Hestilow <hestgray@ionet.net> * gnome/gdesktop.c: Fixes snap to grid.
2000-05-09 05:18:27 +04:00
define_label (h, (Widget *)panel, 2, _("Menu"), user_file_menu_cmd);
define_label (h, (Widget *)panel, 3, _("View"), view_panel_cmd);
define_label (h, (Widget *)panel, 4, _("Edit"), edit_panel_cmd);
define_label (h, (Widget *)panel, 5, _("Copy"), copy_cmd);
define_label (h, (Widget *)panel, 6, _("RenMov"), ren_cmd);
define_label (h, (Widget *)panel, 7, _("Mkdir"), mkdir_panel_cmd);
define_label (h, (Widget *)panel, 8, _("Delete"), delete_cmd);
1998-02-27 07:54:42 +03:00
x_create_panel (h, h->wdata, panel);
#endif
1998-02-27 07:54:42 +03:00
return 1;
1998-02-27 07:54:42 +03:00
case WIDGET_DRAW:
paint_panel (panel);
break;
case WIDGET_FOCUS:
#ifndef HAVE_GNOME
current_panel = panel;
#endif
panel->active = 1;
if (mc_chdir (panel->cwd) != 0){
2000-04-18 12:58:42 +04:00
message (1, MSG_ERROR, _(" Can't chdir to \"%s\" \n %s "),
1998-02-27 07:54:42 +03:00
panel->cwd, unix_error_string (errno));
} else
subshell_chdir (panel->cwd);
1998-02-27 07:54:42 +03:00
show_dir (panel);
focus_select_item (panel);
#ifndef HAVE_X
define_label (h, (Widget *)panel, 1, _("Help"), help_cmd);
2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * src/user.c: Add macro %k it is block file name Add macro %e it is error file name Add macro %i it is cursor column indent of spaces, only for edit Add macro %y, it is syntax of current file in editor, only for edit Add condition y , it is syntax pattern of current file in edit Add macro %x it is extension of current file Add macro %m it is current menu filename 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * gtkedit/editcmd.c: Modify External Formatter (was C indent formatter) . Autocreate a scripts in home, from templates /usr/lib/mc/edit.indent.rc, edit.ispell.rc, etc. Remove leading and trailing spaces into _(""), (the message of David H. Martin <dmartina@usa.net>) 2000-05-05 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * src/user.c: Add condition (x filename) into mc.menu . for "Open next a free console" and like. 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> src/user.c: fix segfault in chunk_alloc of glibc, when into condition of .mnu we have quoted space. (~.mc/menu: + f \.\ test$). 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * gtkedit/edit.c: Add ability user edit menus: system: /usr/lib/mc/cedit.menu, * gtkedit/editmenu.c: home: ~/.cedit.menu, local: .cedit.menu Marked block is access now from an user edit menu Access ~/.cedit/cooledit.block for insert to cursor place from user edit menu. Created system cedit.menu 2000-05-04 Valery Kornienkov <vlk@dimavb.st.simbirsk.su> * gtkedit/editdraw.c: Improved a status string of cool editor for best understand, and to add char,hex view. 2000-05-04 Richard Hestilow <hestgray@ionet.net> * gnome/gdesktop.c: Fixes snap to grid.
2000-05-09 05:18:27 +04:00
define_label (h, (Widget *)panel, 2, _("Menu"), user_file_menu_cmd);
define_label (h, (Widget *)panel, 3, _("View"), view_panel_cmd);
define_label (h, (Widget *)panel, 4, _("Edit"), edit_panel_cmd);
define_label (h, (Widget *)panel, 5, _("Copy"), copy_cmd);
define_label (h, (Widget *)panel, 6, _("RenMov"), ren_cmd);
define_label (h, (Widget *)panel, 7, _("Mkdir"), mkdir_panel_cmd);
define_label (h, (Widget *)panel, 8, _("Delete"), delete_cmd);
1998-02-27 07:54:42 +03:00
redraw_labels (h, (Widget *)panel);
#endif
/* Chain behaviour */
default_proc (h, WIDGET_FOCUS, par);
1998-02-27 07:54:42 +03:00
return 1;
1998-02-27 07:54:42 +03:00
case WIDGET_UNFOCUS:
/* Janne: look at this for the multiple panel options */
if (panel->searching){
panel->searching = 0;
display_mini_info (panel);
}
#ifdef HAVE_X
1998-02-27 07:54:42 +03:00
show_dir (panel);
unfocus_unselect_item (panel);
1998-02-27 07:54:42 +03:00
panel->active = 0;
#else
panel->active = 0;
show_dir (panel);
unselect_item (panel);
#endif
return 1;
1998-02-27 07:54:42 +03:00
case WIDGET_KEY:
return panel_key (panel, par);
break;
}
return default_proc (h, msg, par);
}
/* */
/* Panel mouse events support routines */
/* */
static int mouse_marking = 0;
static void
mouse_toggle_mark (WPanel *panel)
{
do_mark_file (panel, 0);
mouse_marking = selection (panel)->f.marked;
}
static void
mouse_set_mark (WPanel *panel)
{
if (mouse_marking && !(selection (panel)->f.marked))
do_mark_file (panel, 0);
else if (!mouse_marking && (selection (panel)->f.marked))
do_mark_file (panel, 0);
}
static inline int
mark_if_marking (WPanel *panel, Gpm_Event *event)
{
if (event->buttons & GPM_B_RIGHT){
if (event->type & GPM_DOWN)
mouse_toggle_mark (panel);
else
mouse_set_mark (panel);
return 1;
}
return 0;
}
void
file_mark (WPanel *panel, int index, int val)
{
panel->dir.list [index].f.marked = val;
x_panel_select_item (panel, index, val);
}
#ifdef PORT_WANTS_GET_SORT_FN
sortfn *
get_sort_fn (char *name)
{
int i;
/* First, try the long name options, from dir.c */
for (i = 0; i < SORT_TYPES_TOTAL; i++)
if (strcmp (name, sort_orders [i].sort_name) == 0)
return (sortfn *)sort_orders [i].sort_fn;
/* Then try the short name options, from our local table */
for (i = 0; i < ELEMENTS (formats); i++){
if (strcmp (name, formats [i].id) == 0 && formats [i].sort_routine)
return formats [i].sort_routine;
}
return NULL;
}
/* not static because it's called from Tk's version */
int
panel_event (Gpm_Event *event, WPanel *panel)
{
const int lines = panel->count;
1998-02-27 07:54:42 +03:00
int my_index;
event->y -= 2;
if ((event->type & (GPM_DOWN|GPM_DRAG))){
if (panel != (WPanel *) current_dlg->current->widget)
change_panel ();
if (event->y <= 0){
mark_if_marking (panel, event);
return MOU_REPEAT;
}
if (!((panel->top_file + event->y <= panel->count) &&
event->y <= lines)){
mark_if_marking (panel, event);
return MOU_REPEAT;
}
my_index = panel->top_file + event->y - 1;
if (panel->split){
if (event->x > ((panel->widget.cols-2)/2))
my_index += llines (panel);
}
if (my_index >= panel->count)
my_index = panel->count - 1;
1998-02-27 07:54:42 +03:00
if (my_index != panel->selected){
unselect_item (panel);
panel->selected = my_index;
select_item (panel);
}
/* This one is new */
mark_if_marking (panel, event);
1998-02-27 07:54:42 +03:00
} else if ((event->type & (GPM_UP|GPM_DOUBLE)) == (GPM_UP|GPM_DOUBLE)){
if (event->y > 0 && event->y <= lines)
do_enter (panel);
}
return MOU_NORMAL;
}
#else
int
panel_event (Gpm_Event *event, WPanel *panel)
{
const int lines = llines (panel);
1998-02-27 07:54:42 +03:00
int my_index;
if (event->type & GPM_DOWN && event->x == 1 + 1 && event->y == 0 + 1) {
directory_history_prev (panel);
return MOU_NORMAL;
}
if (event->type & GPM_DOWN && event->x == panel->widget.cols - 2 + 1 && event->y == 0 + 1) {
directory_history_next (panel);
return MOU_NORMAL;
}
if (event->type & GPM_DOWN && event->x == panel->widget.cols - 3 + 1 && event->y == 0 + 1) {
directory_history_list (panel);
return MOU_NORMAL;
}
1998-02-27 07:54:42 +03:00
event->y -= 2;
if ((event->type & (GPM_DOWN|GPM_DRAG))){
if (panel != (WPanel *) current_dlg->current->widget)
change_panel ();
if (event->y <= 0){
mark_if_marking (panel, event);
if (mouse_move_pages)
prev_page (panel);
else
move_up (panel);
return MOU_REPEAT;
}
if (!((panel->top_file + event->y <= panel->count) &&
event->y <= lines)){
mark_if_marking (panel, event);
if (mouse_move_pages)
next_page (panel);
else
move_down (panel);
return MOU_REPEAT;
}
my_index = panel->top_file + event->y - 1;
if (panel->split){
if (event->x > ((panel->widget.cols-2)/2))
my_index += llines (panel);
}
if (my_index >= panel->count)
my_index = panel->count - 1;
1998-02-27 07:54:42 +03:00
if (my_index != panel->selected){
unselect_item (panel);
panel->selected = my_index;
select_item (panel);
}
/* This one is new */
mark_if_marking (panel, event);
1998-02-27 07:54:42 +03:00
} else if ((event->type & (GPM_UP|GPM_DOUBLE)) == (GPM_UP|GPM_DOUBLE)){
if (event->y > 0 && event->y <= lines)
do_enter (panel);
}
return MOU_NORMAL;
}
#endif
void
panel_update_marks (WPanel *panel)
{
#ifdef PORT_HAS_UPDATE_MARKS
x_panel_update_marks (panel);
#endif
}
void
panel_re_sort (WPanel *panel)
{
char *filename;
int i;
if (panel == NULL)
return;
filename = g_strdup (selection (panel)->fname);
unselect_item (panel);
do_sort (&panel->dir, panel->sort_type, panel->count-1, panel->reverse, panel->case_sensitive);
panel->selected = -1;
for (i = panel->count; i; i--){
if (!strcmp (panel->dir.list [i-1].fname, filename)){
panel->selected = i-1;
break;
}
}
g_free (filename);
panel->top_file = panel->selected - ITEMS (panel)/2;
if (panel->top_file < 0)
panel->top_file = 0;
select_item (panel);
panel_update_contents (panel);
}
void
panel_set_sort_order (WPanel *panel, sortfn *sort_order)
{
if (sort_order == 0)
return;
panel->sort_type = sort_order;
/* The directory is already sorted, we have to load the unsorted stuff */
if (sort_order == (sortfn *) unsorted){
char *current_file;
current_file = g_strdup (panel->dir.list [panel->selected].fname);
panel_reload (panel);
try_to_select (panel, current_file);
g_free (current_file);
}
panel_re_sort (panel);
}