/* Tk panel stuff. Copyright (C) 1995, 1997 Miguel de Icaza 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. 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. 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 #include #include /* atoi */ #include "fs.h" #include "dir.h" #include "tkmain.h" #include "panel.h" #include "command.h" #include "panel.h" /* current_panel */ #include "command.h" /* cmdline */ #include "main.h" #include "mouse.h" #include "layout.h" /* get_panel_widget */ #include "ext.h" /* regex_command */ #include "cmd.h" /* copy_cmd, ren_cmd, delete_cmd, ... */ #include "tkscreen.h" void do_enter (WPanel *panel); /* These two variables keep track of the dimensions of a character in the * panel listing. */ static int panel_font_width; static int panel_font_height; static char *attrib; void repaint_file (WPanel *panel, int file_index, int move, int attr, int isstatus) { } void show_dir (WPanel *panel) { char *tkw = wtk_win (panel->widget); char *filter = panel->filter ? panel->filter : "*"; if (!panel->widget.wdata) return; tk_evalf ("%s.cwd configure -text {%s [%s]}", tkw, panel->cwd, filter); } void x_fill_panel (WPanel *panel) { const int top = panel->count; const int panel_width = panel->widget.cols - 2; char *panel_name, *tag; char buffer [255]; int i; int selected; if (!panel->widget.wcontainer) return; selected = panel->selected; panel_name = copy_strings (wtk_win (panel->widget), ".m.p.panel", 0); tk_evalf ("%s delete 0.0 end", panel_name); for (i = 0; i < top; i++){ file_entry *fe = &panel->dir.list [i]; int marked = fe->f.marked; format_file (buffer, panel, i, panel_width, NORMAL, 0); if (i) tk_evalf ("%s insert end \\n", panel_name); /* Set the mark tag name -- this probably should go into generic code */ if (marked) tag = "marked"; else { tag = "regular"; if (S_ISLNK (fe->buf.st_mode)){ if (fe->f.link_to_dir) tag = "directory"; } else if (S_ISDIR (fe->buf.st_mode)) tag = "directory"; else if (is_exe (fe->buf.st_mode)) tag = "executable"; } tk_evalf ("%s insert end {%s} %s", panel_name, buffer, tag); } if (panel->active) x_select_item (panel); #if 0 /* The first time, the panel does not yet have a command */ if (panel->widget.wdata) x_select_item (panel); free (panel_name); #endif } /* Adjusts a panel size */ static void tk_panel_set_size (int index, int boot) { Widget *w; WPanel *p; char *pn; int cols, lines; w = (Widget *) get_panel_widget (index); tk_evalf ("panel_info cols"); cols = atoi (interp->result); tk_evalf ("panel_info lines"); lines = atoi (interp->result); if ((lines < 10) || (cols < 3)) return; w = get_panel_widget (index); p = (WPanel *) w; w->cols = cols; w->lines = lines; set_panel_formats (p); paint_panel (p); /* FIXME: paint_panel() includes calls to paint_frame() & x_fill_panel(). */ /* Why does we call them 2 times ? Timur */ if (!boot) paint_frame (p); x_fill_panel (p); } /* Called by panel bootstrap routine */ void x_panel_set_size (int index) { tk_panel_set_size (index, 1); } /* Called at runtime when the size of the window changes */ static void x_change_size (char *panel, int boot) { if (strncmp (panel, ".left", 5) == 0) tk_panel_set_size (0, 0); else tk_panel_set_size (1, 0); } #if TCL_MAJOR_VERSION > 7 /* Tcl 8.xx support */ static void compute_font_size (char *font, char *win, char *dest) { Tk_Window window_id; Tk_Font tkfont; Tk_FontMetrics fmPtr; int width, height; window_id = Tk_NameToWindow (interp, win, Tk_MainWindow (interp)); if (window_id == NULL) { fprintf (stderr, "Error: %s\n\r", interp->result); exit (1); } tkfont = Tk_GetFont (interp, window_id, font); if (tkfont == NULL) { fprintf (stderr, "This should not happend: %s\n", interp->result); exit (1); } Tk_GetFontMetrics (tkfont, &fmPtr); width = Tk_TextWidth (tkfont, "O", 1); height = fmPtr.linespace; Tk_FreeFont (tkfont); sprintf (dest, "%d %d", height, width); } #else #ifdef HAVE_TKFONT /* PreTcl 8.xx support */ #define GetFontMetrics(tkfont) \ ((CONST TkFontMetrics *) &((TkFont *) (tkfont))->fm) static void compute_font_size (char *font, char *win, char *dest) { Tk_Window window_id; Tk_Font tkfont; const TkFontMetrics *fmPtr; int width, height; window_id = Tk_NameToWindow (interp, win, Tk_MainWindow (interp)); if (window_id == NULL) { fprintf (stderr, "Error: %s\n\r", interp->result); exit (1); } tkfont = Tk_GetFont (interp, window_id, font); if (tkfont == NULL) { fprintf (stderr, "This should not happend: %s\n", interp->result); exit (1); } fmPtr = GetFontMetrics (tkfont); width = fmPtr->maxWidth; height = fmPtr->ascent + fmPtr->descent; Tk_FreeFont (tkfont); sprintf (dest, "%d %d", height, width); } #else static char compute_font_size (char *font, char *win, char *dest) { Tk_Uid font_uid; Tk_Window window_id; XFontStruct *f; int width, height; font_uid = Tk_GetUid (font); window_id = Tk_NameToWindow (interp, win, Tk_MainWindow (interp)); if (window_id == NULL){ fprintf (stderr, "Error: %s\n\r", interp->result); exit (1); } f = Tk_GetFontStruct (interp, window_id, font_uid); if (f == NULL){ fprintf (stderr, "This should not happend: %s\n", interp->result); exit (1); } width = f->max_bounds.width; height = f->ascent + f->descent; /* Ok, we will use this dummy until monday, the correct thing * to do is extract the Tk font information from Tk's hash * table (tkFont.c). * * TkFont->widths ['0'] holds the width */ sprintf (dest, "%d %d", height, width); Tk_FreeFontStruct (f); } #endif #endif /* * This routine is called when the user has chosen an action from the popup * menu. Handles the Copy, Move and Delete commands internally, anything * else ends in the regex_command routine. */ static void tk_invoke (WPanel *panel, int idx, char *operation) { char *filename = panel->dir.list [idx].fname; int movedir; if (STREQ (operation, "Copy")) copy_cmd (); else if (STREQ (operation, "Move")) ren_cmd (); else if (STREQ (operation, "Delete")) delete_cmd (); else { regex_command (filename, operation, NULL, &movedir); } } static void tk_add_popup_entry (char *option, char *cmd, int idx) { tk_evalf (".m add command -label {%s} -command {%s invoke %d {%s}}", option, cmd, idx, option); } /* This routine assumes the Tcl code has already created the .m menu */ static void tk_load_popup (WPanel *panel, char *cmd, int idx) { char *filename = panel->dir.list [idx].fname; char *p, *q; int c; tk_evalf ("popup_add_action {%s} %s %d", filename, cmd, idx); p = regex_command (filename, NULL, NULL, NULL); if (!p) return; tk_evalf (".m add separator"); for (;;){ while (*p == ' ' || *p == '\t') p++; if (!*p) break; q = p; while (*q && *q != '=' && *q != '\t') q++; c = *q; *q = 0; tk_add_popup_entry (p, cmd, idx); if (!c) break; p = q + 1; } } void tk_set_sort (WPanel *panel, char *name) { sortfn *f; if (!name || (f = get_sort_fn (name)) == NULL) return; if (!name) return; set_sort_to (panel, f); } void tk_return_drag_text (WPanel *panel) { if (panel->marked){ sprintf (interp->result, "%d file%s", panel->marked, panel->marked== 1 ? "" : "s"); } else strcpy (interp->result, "1 file"); } int tk_panel_callback (ClientData cd, Tcl_Interp *interp, int ac, char *av[]) { Gpm_Event e; WPanel *panel = (WPanel *) cd; char b [20]; char *p; int mouse_etype = 0; p = av [1]; if (STREQ (p, "mdown")) mouse_etype = GPM_DOWN; else if (STREQ (p, "mup")) mouse_etype = GPM_UP; else if (STREQ (p, "double")) mouse_etype = GPM_DOUBLE|GPM_UP; else if (STREQ (p, "motion")) mouse_etype = GPM_MOVE|GPM_DRAG; else if (STREQ (p, "resize")){ x_change_size (av [2], 0); return TCL_OK; } else if (STREQ (p, "fontdim")){ compute_font_size (av [2], av [3], interp->result); return TCL_OK; } else if (STREQ (p, "load")){ tk_load_popup (panel, av [0], atoi (av [2])-1); tk_evalf ("tk_popup .m %s %s", av [3], av [4]); return TCL_OK; } else if (STREQ (p, "invoke")){ tk_invoke (panel, atoi (av [2]), av [3]); return TCL_OK; } else if (STREQ (p, "refresh")){ reread_cmd (); return TCL_OK; } else if (STREQ (p, "setmask")){ filter_cmd (); return TCL_OK; } else if (STREQ (p, "nomask")){ if (panel->filter){ free (panel->filter); panel->filter = 0; } reread_cmd (); return TCL_OK; } else if (STREQ (p, "sort")) { tk_set_sort (panel, av [2]); return TCL_OK; } else if (STREQ (p, "reverse")){ panel->reverse = !panel->reverse; do_re_sort (panel); return TCL_OK; } else if (STREQ (p, "dragtext")){ tk_return_drag_text (panel); return TCL_OK; } if (STREQ (p, "top")){ int old_sel = panel->selected; int new = old_sel; panel->top_file = atoi (av [2]) - 1; if (old_sel < panel->top_file) new = panel->top_file; if (old_sel > (panel->top_file + ITEMS (panel))) new = panel->top_file + ITEMS (panel) - 1; if (new != old_sel){ unselect_item (panel); panel->selected = new; select_item (panel); } return TCL_OK; } if (mouse_etype){ for (p = av [3]; *p && *p != '.'; p++) ; *p++ = 0; e.buttons = atoi (av [2]); e.y = atoi (av [3]) + 2 - panel->top_file; e.x = atoi (p); e.type = mouse_etype; panel_event (&e, panel); } else { /* We are currently dealing with those */ fprintf (stderr, "Unknown command: %s\n", p); return TCL_ERROR; } return TCL_OK; } void x_create_panel (Dlg_head *h, widget_data parent, WPanel *panel) { char *cmd; widget_data container = panel->widget.wcontainer; cmd = tk_new_command (container, panel, tk_panel_callback, 'p'); tk_evalf ("panel_setup %s %s", wtk_win (panel->widget), cmd); tk_evalf ("panel_bind %s %s", wtk_win (panel->widget), cmd); paint_frame (panel); x_fill_panel (panel); } /* Called when f.marked has changed on a file */ void x_panel_select_item (WPanel *panel, int index, int value) { char *s; s = value ? "add" : "remove"; tk_evalf ("panel_mark_entry %s %s %d", wtk_win(panel->widget), s, index+1); } void x_unselect_item (WPanel *panel) { const int selected = panel->selected + 1; if (panel->active) tk_evalf ("panel_unmark_entry %s %d\n", wtk_win (panel->widget), selected); } void x_select_item (WPanel *panel) { const int selected = panel->selected + 1; char *cmd = wtcl_cmd (panel->widget); if (!(cmd && *cmd)) return; if (panel->active) tk_evalf ("panel_select %s %d %s", wtk_win (panel->widget), selected, cmd); } void x_adjust_top_file (WPanel *panel) { tk_evalf ("%s.m.p.panel yview %d.0", wtk_win (panel->widget), panel->top_file+1); } void x_filter_changed (WPanel *panel) { show_dir (panel); } /* The following two routines may be called at the very beginning of the * program, so we have to check if the widget has already been constructed */ static sort_label_last_pos; void x_add_sort_label (WPanel *panel, int index, char *text, char *tag, void *sr) { int pos; if (!panel->format_modified) return; if (!panel->widget.wdata) return; /* We may still not have this information */ if (!panel_font_width){ tk_evalf ("panel_info heightc"); panel_font_height = atoi (interp->result); tk_evalf ("panel_info widthc"); panel_font_width = atoi (interp->result); } pos = sort_label_last_pos; sort_label_last_pos += (strlen (text) * panel_font_width) + (4); /* The addition down there is to account for the canvas border width */ tk_evalf ("panel_add_sort %s %d {%s} %d %d {%s}", wtk_win(panel->widget), strlen (text), text, pos , sort_label_last_pos, *tag ? tag : "Type"); } void x_sort_label_start (WPanel *panel) { if (!panel->widget.wdata) return; sort_label_last_pos = 0; tk_evalf ("panel_sort_label_start %s", wtk_win (panel->widget)); } void x_reset_sort_labels (WPanel *panel) { if (!panel->widget.wdata) return; sort_label_last_pos = 0; tk_evalf ("panel_reset_sort_labels %s", wtk_win (panel->widget)); } void panel_update_cols (Widget *widget, int frame_size) { /* nothing */ } void display_mini_info (WPanel *panel) { /* FIXME: implement */ }