454 строки
10 KiB
C
454 строки
10 KiB
C
|
/* Widgets for the Midnight Commander, Tk interface code
|
|||
|
|
|||
|
Copyright (C) 1994, 1995, 1996 Miguel de Icaza
|
|||
|
Copyright (C) 1995 Jakub Jelinek
|
|||
|
Copyright (C) 1994 Radek Doulik
|
|||
|
|
|||
|
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.
|
|||
|
|
|||
|
TODO: Destroy all the command names that are created for each
|
|||
|
widget during runtime.
|
|||
|
|
|||
|
*/
|
|||
|
#include <config.h>
|
|||
|
#include <stdio.h>
|
|||
|
#include <string.h>
|
|||
|
#include <stdlib.h>
|
|||
|
#include <ctype.h>
|
|||
|
#include "tkmain.h"
|
|||
|
#include "tkwidget.h"
|
|||
|
#include "dlg.h"
|
|||
|
|
|||
|
static char
|
|||
|
widget_number (Widget *w)
|
|||
|
{
|
|||
|
Dlg_head *h = w->parent;
|
|||
|
const int count = h->count;
|
|||
|
Widget_Item *item;
|
|||
|
int i;
|
|||
|
|
|||
|
item = h->first;
|
|||
|
for (i = 0; i < count; i++){
|
|||
|
if (item->widget == w)
|
|||
|
return i;
|
|||
|
item = item->next;
|
|||
|
}
|
|||
|
/* Actually, if we get this far, this message should not appear */
|
|||
|
fprintf (stderr, "Inconsistent widget\n");
|
|||
|
exit (1);
|
|||
|
}
|
|||
|
|
|||
|
/* Returns a new command prefixed with an x:
|
|||
|
x.query.$the_frame.b0
|
|||
|
|
|||
|
for the command of the query dialog box button 0
|
|||
|
|
|||
|
This allow us to control the widget with the command:
|
|||
|
.query.$the_frame.b0
|
|||
|
|
|||
|
and to access the C code callback with the:
|
|||
|
x.query.$the_frame.b0
|
|||
|
command.
|
|||
|
*/
|
|||
|
char *
|
|||
|
tk_new_command (widget_data parent, void *widget,
|
|||
|
tcl_callback callback, char id)
|
|||
|
{
|
|||
|
char buffer [60];
|
|||
|
Widget *w = (Widget *) widget;
|
|||
|
Dlg_head *h = w->parent;
|
|||
|
char *cmd;
|
|||
|
|
|||
|
/* wdata holds the frame information, it will be replaced with the
|
|||
|
* the widget command name
|
|||
|
*/
|
|||
|
if (h->grided && w->tkname)
|
|||
|
sprintf (buffer, ".widgets.%s", w->tkname);
|
|||
|
else {
|
|||
|
char *format;
|
|||
|
|
|||
|
if (((char *) parent) [1] == 0)
|
|||
|
format = "%s%c%d";
|
|||
|
else
|
|||
|
format = ".%s%c%d";
|
|||
|
|
|||
|
sprintf (buffer, format, w->frame, id, widget_number (w));
|
|||
|
}
|
|||
|
cmd = copy_strings ("x", (char *)parent, buffer, 0);
|
|||
|
w->wdata = (widget_data) cmd;
|
|||
|
if (callback)
|
|||
|
Tcl_CreateCommand (interp, cmd, callback, w, NULL);
|
|||
|
return cmd;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Button routines */
|
|||
|
|
|||
|
static int
|
|||
|
is_default_button (WButton *b)
|
|||
|
{
|
|||
|
return b->action == B_ENTER;
|
|||
|
}
|
|||
|
|
|||
|
/* Called from the Tcl/Tk program when a button has been pressed */
|
|||
|
static int
|
|||
|
tk_button_callback (ClientData cd, Tcl_Interp *interp, int argc, char *argv [])
|
|||
|
{
|
|||
|
WButton *b = (WButton *) cd;
|
|||
|
Dlg_head *h = (Dlg_head *) b->widget.parent;
|
|||
|
int stop = 0;
|
|||
|
char *cmd = wtcl_cmd (b->widget);
|
|||
|
|
|||
|
if (b->callback)
|
|||
|
stop = (*b->callback)(b->action, b->callback_data);
|
|||
|
if (!b->callback || stop){
|
|||
|
h->running = 0;
|
|||
|
h->ret_value = b->action;
|
|||
|
}
|
|||
|
return TCL_OK;
|
|||
|
}
|
|||
|
|
|||
|
void
|
|||
|
x_button_set (WButton *b, char *text)
|
|||
|
{
|
|||
|
char *cmd = wtcl_cmd (b->widget);
|
|||
|
char *postfix = is_default_button (b) ? ".button" : "";
|
|||
|
|
|||
|
tk_evalf ("%s%s configure -text {%s}", cmd+1, postfix, text);
|
|||
|
}
|
|||
|
|
|||
|
/* Creates the button $parent.$newname and executes $newname command */
|
|||
|
int
|
|||
|
x_create_button (Dlg_head *h, widget_data parent, WButton *b)
|
|||
|
{
|
|||
|
char *cmd;
|
|||
|
|
|||
|
cmd = tk_new_command (parent, b, tk_button_callback, 'b');
|
|||
|
tk_evalf ("newbutton %s %s {%s} %d", cmd+1, cmd, b->text, is_default_button (b));
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/* Radio button routines */
|
|||
|
static int
|
|||
|
tk_radio_callback (ClientData cd, Tcl_Interp *interp, int argc, char *argv [])
|
|||
|
{
|
|||
|
WRadio *r = (WRadio *) cd;
|
|||
|
|
|||
|
if (STREQ (argv [1], "select")){
|
|||
|
r->pos = r->sel = atoi (argv [2]);
|
|||
|
}
|
|||
|
(r->widget.callback)(r->widget.parent, r, WIDGET_KEY, ' ');
|
|||
|
return TCL_OK;
|
|||
|
}
|
|||
|
|
|||
|
int
|
|||
|
x_create_radio (Dlg_head *h, widget_data parent, WRadio *r)
|
|||
|
{
|
|||
|
int i;
|
|||
|
char *cmd;
|
|||
|
|
|||
|
cmd = tk_new_command (parent, r, tk_radio_callback, 'r');
|
|||
|
tk_evalf ("newradio %s", cmd+1);
|
|||
|
|
|||
|
for (i = 0; i < r->count; i++){
|
|||
|
tk_evalf ("radio_item %d {%s} %s %d", i, r->texts [i], cmd,
|
|||
|
r->sel == i);
|
|||
|
}
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
/* Checkbuttons routines */
|
|||
|
|
|||
|
static int
|
|||
|
tk_check_callback (ClientData cd, Tcl_Interp *interp, int argc, char *argv [])
|
|||
|
{
|
|||
|
WCheck *c = (WCheck *) cd;
|
|||
|
Dlg_head *h = c->widget.parent;
|
|||
|
|
|||
|
(*c->widget.callback)(h, c, WIDGET_KEY, ' ');
|
|||
|
return TCL_OK;
|
|||
|
}
|
|||
|
|
|||
|
int
|
|||
|
x_create_check (Dlg_head *h, widget_data parent, WCheck *c)
|
|||
|
{
|
|||
|
char *cmd;
|
|||
|
|
|||
|
cmd = tk_new_command (parent, c, tk_check_callback, 'c');
|
|||
|
tk_evalf ("newcheck %s %s {%s} %d", cmd+1, cmd, c->text, c->state);
|
|||
|
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Input line */
|
|||
|
|
|||
|
int input_event (Gpm_Event *event, WInput *b);
|
|||
|
|
|||
|
/* This callback command accepts the following commands:
|
|||
|
**
|
|||
|
** $win mouse x-position
|
|||
|
** $win setmark x-position
|
|||
|
*/
|
|||
|
static int
|
|||
|
tk_input_callback (ClientData cd, Tcl_Interp *interp, int argc, char *av [])
|
|||
|
{
|
|||
|
Gpm_Event e;
|
|||
|
WInput *in = (WInput *) cd;
|
|||
|
|
|||
|
if (strcmp (av [1], "mouse") == 0){
|
|||
|
e.x = atoi (av [2]) + 1;
|
|||
|
e.y = 0;
|
|||
|
e.type = GPM_DOWN;
|
|||
|
input_event (&e, (WInput *) cd);
|
|||
|
return TCL_OK;
|
|||
|
} else if (strcmp (av [1], "setmark") == 0){
|
|||
|
in->mark = in->point;
|
|||
|
return TCL_OK;
|
|||
|
}
|
|||
|
|
|||
|
return TCL_OK;
|
|||
|
}
|
|||
|
|
|||
|
void
|
|||
|
x_update_input (WInput *in)
|
|||
|
{
|
|||
|
char *name = wtcl_cmd (in->widget);
|
|||
|
int p;
|
|||
|
|
|||
|
if (!name)
|
|||
|
return;
|
|||
|
|
|||
|
name++;
|
|||
|
if (in->is_password){
|
|||
|
tk_evalf ("%s delete 0 end", name);
|
|||
|
tk_evalf ("%s insert end {*secret*}", name);
|
|||
|
p = 9;
|
|||
|
} else {
|
|||
|
tk_evalf ("entry_save_sel %s", name);
|
|||
|
if (in->inserted_one && isascii (in->inserted_one) && in->inserted_one > ' '){
|
|||
|
tk_evalf ("%s insert insert {%c}", name, in->inserted_one);
|
|||
|
in->inserted_one = 0;
|
|||
|
} else {
|
|||
|
tk_evalf ("%s delete 0 end", name);
|
|||
|
tk_evalf ("%s insert end {%s}", name, in->buffer);
|
|||
|
}
|
|||
|
p = in->point;
|
|||
|
}
|
|||
|
tk_evalf ("%s icursor %d; entry_restore_sel %s", name, p, name);
|
|||
|
}
|
|||
|
|
|||
|
int
|
|||
|
x_create_input (Dlg_head *h, widget_data parent, WInput *in)
|
|||
|
{
|
|||
|
char *cmd;
|
|||
|
|
|||
|
cmd = tk_new_command (parent, in, tk_input_callback, 'i');
|
|||
|
tk_evalf ("newinput %s {%s}", cmd+1, in->buffer ? in->buffer: "");
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
void
|
|||
|
x_listbox_select_nth (WListbox *l, int nth)
|
|||
|
{
|
|||
|
if (!wtcl_cmd (l->widget))
|
|||
|
return;
|
|||
|
|
|||
|
tk_evalf ("listbox_sel %s %d", wtk_win (l->widget), nth);
|
|||
|
}
|
|||
|
|
|||
|
void
|
|||
|
x_listbox_delete_nth (WListbox *l, int nth)
|
|||
|
{
|
|||
|
tk_evalf ("%s delete %d", wtk_win (l->widget), nth);
|
|||
|
}
|
|||
|
|
|||
|
int
|
|||
|
x_create_listbox (Dlg_head *h, widget_data parent, WListbox *l)
|
|||
|
{
|
|||
|
char *cmd;
|
|||
|
int i;
|
|||
|
WLEntry *e;
|
|||
|
|
|||
|
cmd = tk_new_command (parent, l, 0, 'x');
|
|||
|
tk_evalf ("listbox %s -width %d -selectmode single", cmd+1, l->widget.cols);
|
|||
|
|
|||
|
/* As Jakub said on his very fine xvwidget.c: the user could add some
|
|||
|
* entries to the listbox before the tk-listbox was created, so we have
|
|||
|
* to add it manually :-)
|
|||
|
*/
|
|||
|
i = 0;
|
|||
|
if ((e = l->list) == NULL)
|
|||
|
return 1;
|
|||
|
|
|||
|
do {
|
|||
|
tk_evalf ("%s insert %d {%s}", cmd+1, i, e->text);
|
|||
|
if (e == l->current)
|
|||
|
x_listbox_select_nth (l, i);
|
|||
|
e = e->next;
|
|||
|
i++;
|
|||
|
} while (e != l->list);
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
/*
|
|||
|
* 30.10.96 bor: x_list_insert added in reverse order as x_create_listbox
|
|||
|
*/
|
|||
|
void
|
|||
|
x_list_insert (WListbox *l, WLEntry *p, WLEntry *e)
|
|||
|
{
|
|||
|
int i;
|
|||
|
char *t = e->text;
|
|||
|
|
|||
|
if (wtcl_cmd (l->widget) == NULL)
|
|||
|
return;
|
|||
|
|
|||
|
for (i = 0; p != NULL && p != e; p = p->next, i++)
|
|||
|
;
|
|||
|
tk_evalf ("%s insert %d {%s}", wtk_win (l->widget), i, t);
|
|||
|
}
|
|||
|
|
|||
|
int
|
|||
|
x_create_label (Dlg_head *g, widget_data parent, WLabel *l)
|
|||
|
{
|
|||
|
char *cmd;
|
|||
|
|
|||
|
cmd = tk_new_command (parent, l, 0, 'l');
|
|||
|
tk_evalf ("label %s -text {%s}", cmd+1, l->text);
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void
|
|||
|
x_label_set_text (WLabel *label, char *text)
|
|||
|
{
|
|||
|
char *lname;
|
|||
|
|
|||
|
if (!wtcl_cmd (label->widget))
|
|||
|
return;
|
|||
|
|
|||
|
tk_evalf ("%s configure -text {%s}", wtk_win (label->widget),
|
|||
|
text ? text : "");
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
static int
|
|||
|
tk_buttonbar_callback (ClientData cd, Tcl_Interp *in, int ac, char *av [])
|
|||
|
{
|
|||
|
WButtonBar *bb = (WButtonBar *) cd;
|
|||
|
Dlg_head *h = (Dlg_head *) bb->widget.parent;
|
|||
|
int which = atoi (av [1]);
|
|||
|
|
|||
|
(*bb->labels [which].function)(bb->labels [which].data);
|
|||
|
return TCL_OK;
|
|||
|
}
|
|||
|
|
|||
|
int
|
|||
|
x_create_buttonbar (Dlg_head *h, widget_data parent, WButtonBar *bb)
|
|||
|
{
|
|||
|
char *cmd;
|
|||
|
int i;
|
|||
|
|
|||
|
cmd = tk_new_command (parent, bb, tk_buttonbar_callback, 'n');
|
|||
|
tk_evalf ("frame %s", cmd+1);
|
|||
|
for (i = 0; i < 10; i++){
|
|||
|
tk_evalf ("newbutton %s.%d {%s %d} {%d %s} 0", cmd+1, i, cmd, i, i+1,
|
|||
|
bb->labels [i].text ? bb->labels [i].text : "");
|
|||
|
tk_evalf ("%s.%d configure -padx 1", cmd+1, i);
|
|||
|
tk_evalf ("pack %s.%d -side left -fill x -expand 1 -ipady 0",
|
|||
|
cmd+1, i);
|
|||
|
}
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
void
|
|||
|
x_redefine_label (WButtonBar *bb, int idx)
|
|||
|
{
|
|||
|
if (!bb->widget.wdata)
|
|||
|
return;
|
|||
|
|
|||
|
if (!*(char *) bb->widget.wdata)
|
|||
|
return;
|
|||
|
|
|||
|
if (!bb->labels [idx-1].text)
|
|||
|
return;
|
|||
|
tk_evalf ("%s.%d configure -text {%d %s}",
|
|||
|
wtk_win (bb->widget), idx-1, idx, bb->labels [idx-1].text);
|
|||
|
}
|
|||
|
|
|||
|
/* destructor for the Tk widgets */
|
|||
|
void
|
|||
|
x_destroy_cmd (void *w)
|
|||
|
{
|
|||
|
Widget *widget = (Widget *)w;
|
|||
|
|
|||
|
if (widget->wdata){
|
|||
|
Tcl_DeleteCommand (interp, wtcl_cmd (*widget));
|
|||
|
tk_evalf ("destroy %s", wtk_win (*widget));
|
|||
|
free (wtcl_cmd (*widget));
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
int
|
|||
|
x_find_buttonbar_check (WButtonBar *bb, Widget *paneletc)
|
|||
|
{
|
|||
|
return (strcmp ((char *) bb->widget.wcontainer,
|
|||
|
(char *)paneletc->wcontainer));
|
|||
|
}
|
|||
|
|
|||
|
/* The only parameter accepted is the size of the widget width */
|
|||
|
static int
|
|||
|
tk_gauge_callback (ClientData cd, Tcl_Interp *in, int ac, char *av [])
|
|||
|
{
|
|||
|
WGauge *g = (WGauge *) cd;
|
|||
|
|
|||
|
g->pixels = atoi (av [1]);
|
|||
|
return TCL_OK;
|
|||
|
}
|
|||
|
|
|||
|
int
|
|||
|
x_create_gauge (Dlg_head *h, widget_data parent, WGauge *g)
|
|||
|
{
|
|||
|
char *cmd;
|
|||
|
|
|||
|
cmd = tk_new_command (parent, g, tk_gauge_callback, 'g');
|
|||
|
tk_evalf ("newgauge %s", cmd+1);
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
static void
|
|||
|
x_gauge_display (WGauge *g)
|
|||
|
{
|
|||
|
tk_evalf ("%s coords gauge 0 0 %d 90", wtk_win (g->widget),
|
|||
|
(int) (g->current*g->pixels)/g->max);
|
|||
|
}
|
|||
|
|
|||
|
void
|
|||
|
x_gauge_show (WGauge *g)
|
|||
|
{
|
|||
|
tk_evalf ("gauge_%s %s", g->shown ? "shown":"hidden", wtk_win (g->widget));
|
|||
|
if (g->shown)
|
|||
|
x_gauge_display (g);
|
|||
|
}
|
|||
|
|
|||
|
void
|
|||
|
x_gauge_set_value (WGauge *g, int max, int current)
|
|||
|
{
|
|||
|
if (g->shown)
|
|||
|
x_gauge_display (g);
|
|||
|
}
|