1998-10-20 00:46:58 +04:00
|
|
|
//
|
2000-06-06 01:21:24 +04:00
|
|
|
// "$Id: Fl.cxx,v 1.24.2.24 2000/06/05 21:20:46 mike Exp $"
|
1998-10-20 00:46:58 +04:00
|
|
|
//
|
|
|
|
// Main event handling code for the Fast Light Tool Kit (FLTK).
|
|
|
|
//
|
2000-04-26 02:17:00 +04:00
|
|
|
// Copyright 1998-2000 by Bill Spitzak and others.
|
1998-10-20 00:46:58 +04:00
|
|
|
//
|
1998-10-06 23:14:55 +04:00
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Library General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
1998-10-20 00:46:58 +04:00
|
|
|
//
|
1998-10-06 23:14:55 +04:00
|
|
|
// This library 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
|
|
|
|
// Library General Public License for more details.
|
1998-10-20 00:46:58 +04:00
|
|
|
//
|
1998-10-06 23:14:55 +04:00
|
|
|
// You should have received a copy of the GNU Library General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
|
|
// USA.
|
1998-10-20 00:46:58 +04:00
|
|
|
//
|
2000-06-06 01:21:24 +04:00
|
|
|
// Please report all bugs and problems to "fltk-bugs@fltk.org".
|
1998-10-20 00:46:58 +04:00
|
|
|
//
|
1998-10-06 23:14:55 +04:00
|
|
|
|
|
|
|
#include <FL/Fl.H>
|
|
|
|
#include <FL/Fl_Window.H>
|
|
|
|
#include <FL/x.H>
|
|
|
|
#include <ctype.h>
|
1999-06-07 11:03:34 +04:00
|
|
|
#include <stdlib.h>
|
1999-04-17 05:02:30 +04:00
|
|
|
#include <string.h>
|
1998-10-06 23:14:55 +04:00
|
|
|
|
1998-10-20 00:46:58 +04:00
|
|
|
//
|
|
|
|
// Globals...
|
|
|
|
//
|
|
|
|
|
|
|
|
Fl_Widget *Fl::belowmouse_,
|
|
|
|
*Fl::pushed_,
|
|
|
|
*Fl::focus_,
|
|
|
|
*Fl::selection_owner_;
|
|
|
|
int Fl::damage_,
|
|
|
|
Fl::e_x,
|
|
|
|
Fl::e_y,
|
|
|
|
Fl::e_x_root,
|
|
|
|
Fl::e_y_root,
|
|
|
|
Fl::e_state,
|
|
|
|
Fl::e_clicks,
|
|
|
|
Fl::e_is_click,
|
|
|
|
Fl::e_keysym;
|
1999-01-27 00:37:14 +03:00
|
|
|
char *Fl::e_text = "";
|
1998-10-20 00:46:58 +04:00
|
|
|
int Fl::e_length;
|
|
|
|
|
1999-02-19 17:26:30 +03:00
|
|
|
static double fl_elapsed();
|
1998-10-20 00:46:58 +04:00
|
|
|
|
|
|
|
//
|
|
|
|
// 'Fl:event_inside()' - Return whether or not the mouse event is inside
|
|
|
|
// the given rectangle.
|
|
|
|
//
|
1998-10-06 23:14:55 +04:00
|
|
|
|
|
|
|
int Fl::event_inside(int x,int y,int w,int h) /*const*/ {
|
1998-10-20 00:46:58 +04:00
|
|
|
int mx = event_x() - x;
|
|
|
|
int my = event_y() - y;
|
|
|
|
return (mx >= 0 && mx < w && my >= 0 && my < h);
|
1998-10-06 23:14:55 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int Fl::event_inside(const Fl_Widget *o) /*const*/ {
|
|
|
|
return event_inside(o->x(),o->y(),o->w(),o->h());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Timeouts are insert-sorted into order. This works good if there
|
|
|
|
// are only a small number:
|
|
|
|
|
1999-04-17 05:02:30 +04:00
|
|
|
static struct Timeout {
|
1998-10-06 23:14:55 +04:00
|
|
|
double time;
|
|
|
|
void (*cb)(void*);
|
|
|
|
void* arg;
|
1999-04-17 05:02:30 +04:00
|
|
|
} * timeout;
|
1998-10-06 23:14:55 +04:00
|
|
|
static int numtimeouts;
|
1999-04-17 05:02:30 +04:00
|
|
|
static int timeout_array_size;
|
1998-10-06 23:14:55 +04:00
|
|
|
|
2000-03-08 19:02:25 +03:00
|
|
|
void Fl::add_timeout(double t, Fl_Timeout_Handler cb, void *v) {
|
1999-03-04 21:24:44 +03:00
|
|
|
|
|
|
|
fl_elapsed();
|
|
|
|
|
1999-04-17 05:02:30 +04:00
|
|
|
if (numtimeouts >= timeout_array_size) {
|
|
|
|
timeout_array_size = 2*timeout_array_size+1;
|
|
|
|
timeout = (Timeout*)realloc(timeout, timeout_array_size*sizeof(Timeout));
|
|
|
|
}
|
|
|
|
|
|
|
|
// insert-sort the new timeout:
|
|
|
|
int i;
|
|
|
|
for (i=0; i<numtimeouts; i++) {
|
1998-10-06 23:14:55 +04:00
|
|
|
if (timeout[i].time > t) {
|
1999-04-17 05:02:30 +04:00
|
|
|
for (int j=numtimeouts; j>i; j--) timeout[j] = timeout[j-1];
|
1998-10-06 23:14:55 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
timeout[i].time = t;
|
|
|
|
timeout[i].cb = cb;
|
|
|
|
timeout[i].arg = v;
|
1999-04-17 05:02:30 +04:00
|
|
|
|
|
|
|
numtimeouts++;
|
1998-10-06 23:14:55 +04:00
|
|
|
}
|
|
|
|
|
2000-03-08 19:02:25 +03:00
|
|
|
void Fl::remove_timeout(Fl_Timeout_Handler cb, void *v) {
|
1998-10-06 23:14:55 +04:00
|
|
|
int i,j;
|
|
|
|
for (i=j=0; i<numtimeouts; i++) {
|
|
|
|
if (timeout[i].cb == cb && timeout[i].arg==v) ;
|
|
|
|
else {if (j<i) timeout[j]=timeout[i]; j++;}
|
|
|
|
}
|
|
|
|
numtimeouts = j;
|
|
|
|
}
|
|
|
|
|
1999-08-23 03:31:21 +04:00
|
|
|
static int call_timeouts() {
|
|
|
|
int expired = 0;
|
1999-04-17 05:02:30 +04:00
|
|
|
while (numtimeouts) {
|
|
|
|
if (timeout[0].time > 0) break;
|
|
|
|
// we must remove timeout from array before doing the callback:
|
|
|
|
void (*cb)(void*) = timeout[0].cb;
|
|
|
|
void *arg = timeout[0].arg;
|
1999-08-23 03:31:21 +04:00
|
|
|
numtimeouts--; expired++;
|
1999-04-17 05:02:30 +04:00
|
|
|
if (numtimeouts) memmove(timeout, timeout+1, numtimeouts*sizeof(Timeout));
|
|
|
|
// now it is safe for the callback to do add_timeout:
|
|
|
|
cb(arg);
|
1998-10-06 23:14:55 +04:00
|
|
|
}
|
1999-08-23 03:31:21 +04:00
|
|
|
return expired;
|
1998-10-06 23:14:55 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Fl::flush() {
|
|
|
|
if (damage()) {
|
|
|
|
damage_ = 0;
|
|
|
|
for (Fl_X* x = Fl_X::first; x; x = x->next) {
|
1999-10-23 10:19:59 +04:00
|
|
|
if (x->w->damage() && x->w->visible_r()) {
|
1999-04-10 12:09:39 +04:00
|
|
|
if (x->wait_for_expose) {
|
|
|
|
// leave Fl::damage() set so programs can tell damage still exists
|
|
|
|
damage_ = 1;
|
|
|
|
} else {
|
|
|
|
x->flush();
|
|
|
|
x->w->clear_damage();
|
|
|
|
}
|
1998-10-06 23:14:55 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#ifndef WIN32
|
|
|
|
if (fl_display) XFlush(fl_display);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
extern double fl_wait(int timeout_flag, double timeout);
|
|
|
|
extern int fl_ready();
|
|
|
|
|
|
|
|
static int initclock; // if false we didn't call fl_elapsed() last time
|
|
|
|
|
|
|
|
#ifndef WIN32
|
|
|
|
#include <sys/time.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// fl_elapsed must return the amount of time since the last time it was
|
1999-08-23 03:31:21 +04:00
|
|
|
// called. To reduce the number of system calls to get the
|
1998-10-20 00:46:58 +04:00
|
|
|
// current time, the "initclock" symbol is turned on by an indefinite
|
1998-10-06 23:14:55 +04:00
|
|
|
// wait. This should then reset the measured-from time and return zero
|
|
|
|
static double fl_elapsed() {
|
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
|
1999-05-24 18:03:47 +04:00
|
|
|
unsigned long newclock = GetTickCount();
|
1998-10-06 23:14:55 +04:00
|
|
|
const int TICKS_PER_SECOND = 1000; // divisor of the value to get seconds
|
|
|
|
static unsigned long prevclock;
|
1999-05-24 18:03:47 +04:00
|
|
|
if (!initclock) {prevclock = newclock; initclock = 1; return 0.0;}
|
1999-05-07 20:17:49 +04:00
|
|
|
else if (newclock < prevclock) return 0.0;
|
|
|
|
|
1998-10-06 23:14:55 +04:00
|
|
|
double t = double(newclock-prevclock)/TICKS_PER_SECOND;
|
|
|
|
prevclock = newclock;
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
static struct timeval prevclock;
|
|
|
|
struct timeval newclock;
|
1998-10-20 00:46:58 +04:00
|
|
|
gettimeofday(&newclock, NULL);
|
1998-10-06 23:14:55 +04:00
|
|
|
if (!initclock) {
|
|
|
|
prevclock.tv_sec = newclock.tv_sec;
|
|
|
|
prevclock.tv_usec = newclock.tv_usec;
|
|
|
|
initclock = 1;
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
double t = newclock.tv_sec - prevclock.tv_sec +
|
|
|
|
(newclock.tv_usec - prevclock.tv_usec)/1000000.0;
|
|
|
|
prevclock.tv_sec = newclock.tv_sec;
|
|
|
|
prevclock.tv_usec = newclock.tv_usec;
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// expire any timeouts:
|
|
|
|
if (t > 0.0) for (int i=0; i<numtimeouts; i++) timeout[i].time -= t;
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
void (*Fl::idle)();
|
|
|
|
static char in_idle;
|
|
|
|
static void callidle() {
|
|
|
|
if (!Fl::idle || in_idle) return;
|
|
|
|
in_idle = 1;
|
|
|
|
Fl::idle();
|
|
|
|
in_idle = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Fl::wait() {
|
|
|
|
callidle();
|
1999-08-23 03:31:21 +04:00
|
|
|
int expired = 0;
|
|
|
|
if (numtimeouts) {fl_elapsed(); expired = call_timeouts();}
|
1998-10-06 23:14:55 +04:00
|
|
|
flush();
|
1999-08-23 03:31:21 +04:00
|
|
|
if ((idle && !in_idle) || expired) {
|
1998-10-06 23:14:55 +04:00
|
|
|
fl_wait(1,0.0);
|
1999-08-23 03:31:21 +04:00
|
|
|
} else if (numtimeouts) {
|
1998-10-06 23:14:55 +04:00
|
|
|
fl_wait(1, timeout[0].time);
|
1999-08-23 03:31:21 +04:00
|
|
|
} else {
|
1998-10-06 23:14:55 +04:00
|
|
|
initclock = 0;
|
|
|
|
fl_wait(0,0);
|
|
|
|
}
|
2000-05-12 01:58:09 +04:00
|
|
|
return Fl_X::first != 0; // return true if there is a window
|
1998-10-06 23:14:55 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
double Fl::wait(double time) {
|
|
|
|
callidle();
|
1999-08-23 03:31:21 +04:00
|
|
|
int expired = 0;
|
|
|
|
if (numtimeouts) {time -= fl_elapsed(); expired = call_timeouts();}
|
1998-10-06 23:14:55 +04:00
|
|
|
flush();
|
1999-08-23 03:31:21 +04:00
|
|
|
double wait_time = (idle && !in_idle) || expired ? 0.0 : time;
|
1998-10-06 23:14:55 +04:00
|
|
|
if (numtimeouts && timeout[0].time < wait_time) wait_time = timeout[0].time;
|
|
|
|
fl_wait(1, wait_time);
|
|
|
|
return time - fl_elapsed();
|
|
|
|
}
|
|
|
|
|
|
|
|
int Fl::check() {
|
|
|
|
callidle();
|
|
|
|
if (numtimeouts) {fl_elapsed(); call_timeouts();}
|
|
|
|
fl_wait(1, 0.0);
|
1999-03-05 08:52:42 +03:00
|
|
|
flush();
|
1999-03-13 23:07:21 +03:00
|
|
|
return Fl_X::first != 0; // return true if there is a window
|
1998-10-06 23:14:55 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int Fl::ready() {
|
|
|
|
// if (idle && !in_idle) return 1; // should it do this?
|
|
|
|
if (numtimeouts) {fl_elapsed(); if (timeout[0].time <= 0) return 1;}
|
|
|
|
return fl_ready();
|
|
|
|
}
|
|
|
|
|
|
|
|
int Fl::run() {
|
|
|
|
while (wait());
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
// Window list management:
|
|
|
|
|
|
|
|
Fl_X* Fl_X::first;
|
|
|
|
|
|
|
|
Fl_Window* fl_find(Window xid) {
|
|
|
|
Fl_X *window;
|
|
|
|
for (Fl_X **pp = &Fl_X::first; (window = *pp); pp = &window->next)
|
|
|
|
if (window->xid == xid) {
|
|
|
|
if (window != Fl_X::first && !Fl::modal()) {
|
|
|
|
// make this window be first to speed up searches
|
|
|
|
// this is not done if modal is true to avoid messing up modal stack
|
|
|
|
*pp = window->next;
|
|
|
|
window->next = Fl_X::first;
|
|
|
|
Fl_X::first = window;
|
|
|
|
}
|
|
|
|
return window->w;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2000-05-14 00:03:20 +04:00
|
|
|
Fl_Window* Fl::first_window() {
|
|
|
|
Fl_X* x = Fl_X::first;
|
|
|
|
return x ? x->w : 0;
|
1998-10-06 23:14:55 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
Fl_Window* Fl::next_window(const Fl_Window* w) {
|
2000-05-14 00:03:20 +04:00
|
|
|
Fl_X* x = Fl_X::i(w)->next;
|
|
|
|
return x ? x->w : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Fl::first_window(Fl_Window* window) {
|
|
|
|
if (!window || !window->shown()) return;
|
|
|
|
fl_find(fl_xid(window));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Fl::redraw() {
|
|
|
|
for (Fl_X* x = Fl_X::first; x; x = x->next) x->w->redraw();
|
|
|
|
}
|
1998-10-06 23:14:55 +04:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
// Event handlers:
|
|
|
|
|
|
|
|
struct handler_link {
|
|
|
|
int (*handle)(int);
|
|
|
|
const handler_link *next;
|
|
|
|
};
|
|
|
|
|
|
|
|
static const handler_link *handlers = 0;
|
|
|
|
|
|
|
|
void Fl::add_handler(int (*h)(int)) {
|
|
|
|
handler_link *l = new handler_link;
|
|
|
|
l->handle = h;
|
|
|
|
l->next = handlers;
|
|
|
|
handlers = l;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int send_handlers(int event) {
|
|
|
|
for (const handler_link *h = handlers; h; h = h->next)
|
|
|
|
if (h->handle(event)) return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
Fl_Widget* fl_oldfocus; // kludge for Fl_Group...
|
|
|
|
|
|
|
|
void Fl::focus(Fl_Widget *o) {
|
1998-12-15 18:38:16 +03:00
|
|
|
if (grab()) return; // don't do anything while grab is on
|
1998-10-06 23:14:55 +04:00
|
|
|
Fl_Widget *p = focus_;
|
|
|
|
if (o != p) {
|
|
|
|
focus_ = o;
|
|
|
|
fl_oldfocus = 0;
|
|
|
|
for (; p && !p->contains(o); p = p->parent()) {
|
|
|
|
p->handle(FL_UNFOCUS);
|
|
|
|
fl_oldfocus = p;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Fl::belowmouse(Fl_Widget *o) {
|
1998-12-15 18:38:16 +03:00
|
|
|
if (grab()) return; // don't do anything while grab is on
|
1998-10-06 23:14:55 +04:00
|
|
|
Fl_Widget *p = belowmouse_;
|
|
|
|
if (o != p) {
|
|
|
|
event_is_click(0);
|
|
|
|
belowmouse_ = o;
|
|
|
|
for (; p && !p->contains(o); p = p->parent()) p->handle(FL_LEAVE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Fl::pushed(Fl_Widget *o) {
|
|
|
|
pushed_ = o;
|
|
|
|
}
|
|
|
|
|
|
|
|
Fl_Window *fl_xfocus; // which window X thinks has focus
|
1999-04-10 12:09:39 +04:00
|
|
|
Fl_Window *fl_xmousewin;// which window X thinks has FL_ENTER
|
1998-10-06 23:14:55 +04:00
|
|
|
Fl_Window *Fl::grab_; // most recent Fl::grab()
|
1999-04-10 12:09:39 +04:00
|
|
|
Fl_Window *Fl::modal_; // topmost modal() window
|
1998-10-06 23:14:55 +04:00
|
|
|
|
1999-04-10 12:09:39 +04:00
|
|
|
// Update modal(), focus() and other state according to system state,
|
|
|
|
// and send FL_ENTER, FL_LEAVE, FL_FOCUS, and/or FL_UNFOCUS events.
|
|
|
|
// This is the only function that produces these events in response
|
|
|
|
// to system activity.
|
1998-10-06 23:14:55 +04:00
|
|
|
// This is called whenever a window is added or hidden, and whenever
|
1999-04-10 12:09:39 +04:00
|
|
|
// X says the focus or mouse window have changed.
|
|
|
|
|
1998-10-06 23:14:55 +04:00
|
|
|
void fl_fix_focus() {
|
|
|
|
|
1998-12-15 18:38:16 +03:00
|
|
|
if (Fl::grab()) return; // don't do anything while grab is on.
|
|
|
|
|
1998-10-06 23:14:55 +04:00
|
|
|
// set focus based on Fl::modal() and fl_xfocus
|
1999-04-10 12:09:39 +04:00
|
|
|
Fl_Widget* w = fl_xfocus;
|
1998-10-06 23:14:55 +04:00
|
|
|
if (w) {
|
1999-04-10 12:09:39 +04:00
|
|
|
while (w->parent()) w = w->parent();
|
1998-10-06 23:14:55 +04:00
|
|
|
if (Fl::modal()) w = Fl::modal();
|
|
|
|
if (!w->contains(Fl::focus()))
|
|
|
|
if (!w->take_focus()) Fl::focus(w);
|
|
|
|
} else
|
|
|
|
Fl::focus(0);
|
|
|
|
|
1998-12-15 18:38:16 +03:00
|
|
|
if (!Fl::pushed()) {
|
1998-10-06 23:14:55 +04:00
|
|
|
|
|
|
|
// set belowmouse based on Fl::modal() and fl_xmousewin:
|
|
|
|
w = fl_xmousewin;
|
|
|
|
if (w) {
|
|
|
|
if (Fl::modal()) w = Fl::modal();
|
|
|
|
if (!w->contains(Fl::belowmouse())) {
|
1999-04-10 12:09:39 +04:00
|
|
|
Fl::belowmouse(w);
|
|
|
|
w->handle(FL_ENTER);
|
|
|
|
} else {
|
|
|
|
// send a FL_MOVE event so the enter/leave state is up to date
|
|
|
|
Fl::e_x = Fl::e_x_root-fl_xmousewin->x();
|
|
|
|
Fl::e_y = Fl::e_y_root-fl_xmousewin->y();
|
|
|
|
w->handle(FL_MOVE);
|
|
|
|
}
|
|
|
|
} else {
|
1998-10-06 23:14:55 +04:00
|
|
|
Fl::belowmouse(0);
|
1999-04-10 12:09:39 +04:00
|
|
|
}
|
1998-10-06 23:14:55 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-04-10 12:09:39 +04:00
|
|
|
#ifndef WIN32
|
|
|
|
Fl_Widget *fl_selection_requestor; // from Fl_cutpaste.C
|
|
|
|
#endif
|
1998-10-06 23:14:55 +04:00
|
|
|
|
1999-04-10 12:09:39 +04:00
|
|
|
// This function is called by ~Fl_Widget() and by Fl_Widget::deactivate
|
|
|
|
// and by Fl_Widget::hide(). It indicates that the widget does not want
|
|
|
|
// to receive any more events, and also removes all global variables that
|
|
|
|
// point at the widget.
|
|
|
|
// I changed this from the 1.0.1 behavior, the older version could send
|
|
|
|
// FL_LEAVE or FL_UNFOCUS events to the widget. This appears to not be
|
|
|
|
// desirable behavior and caused flwm to crash.
|
|
|
|
|
|
|
|
void fl_throw_focus(Fl_Widget *o) {
|
|
|
|
if (o->contains(Fl::pushed())) Fl::pushed_ = 0;
|
|
|
|
if (o->contains(Fl::selection_owner())) Fl::selection_owner_ = 0;
|
|
|
|
#ifndef WIN32
|
|
|
|
if (o->contains(fl_selection_requestor)) fl_selection_requestor = 0;
|
|
|
|
#endif
|
|
|
|
if (o->contains(Fl::belowmouse())) Fl::belowmouse_ = 0;
|
|
|
|
if (o->contains(Fl::focus())) Fl::focus_ = 0;
|
1999-04-23 10:55:53 +04:00
|
|
|
if (o == fl_xfocus) fl_xfocus = 0;
|
|
|
|
if (o == fl_xmousewin) fl_xmousewin = 0;
|
1999-04-10 12:09:39 +04:00
|
|
|
fl_fix_focus();
|
1998-12-15 18:38:16 +03:00
|
|
|
}
|
|
|
|
|
1999-04-10 12:09:39 +04:00
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
|
1999-07-23 01:37:04 +04:00
|
|
|
// Call to->handle but first replace the mouse x/y with the correct
|
|
|
|
// values to account for nested X windows. 'window' is the outermost
|
|
|
|
// window the event was posted to by X:
|
|
|
|
static int send(int event, Fl_Widget* to, Fl_Window* window) {
|
|
|
|
int dx = window->x();
|
|
|
|
int dy = window->y();
|
|
|
|
for (const Fl_Widget* w = to; w; w = w->parent())
|
|
|
|
if (w->type()>=FL_WINDOW) {dx -= w->x(); dy -= w->y();}
|
|
|
|
int save_x = Fl::e_x; Fl::e_x += dx;
|
|
|
|
int save_y = Fl::e_y; Fl::e_y += dy;
|
|
|
|
int ret = to->handle(event);
|
|
|
|
Fl::e_y = save_y;
|
|
|
|
Fl::e_x = save_x;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
1998-10-06 23:14:55 +04:00
|
|
|
int Fl::handle(int event, Fl_Window* window)
|
|
|
|
{
|
|
|
|
Fl_Widget* w = window;
|
|
|
|
|
|
|
|
switch (event) {
|
|
|
|
|
|
|
|
case FL_CLOSE:
|
1998-12-15 18:38:16 +03:00
|
|
|
if (grab() || modal() && window != modal()) return 0;
|
1998-10-06 23:14:55 +04:00
|
|
|
w->do_callback();
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case FL_SHOW:
|
|
|
|
((Fl_Widget*)w)->show();
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case FL_HIDE:
|
|
|
|
((Fl_Widget*)w)->hide();
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case FL_PUSH:
|
1998-12-15 18:38:16 +03:00
|
|
|
if (grab()) w = grab();
|
|
|
|
else if (modal() && w != modal()) return 0;
|
1999-07-23 01:37:04 +04:00
|
|
|
pushed_ = w;
|
|
|
|
if (send(event, w, window)) return 1;
|
1998-10-06 23:14:55 +04:00
|
|
|
// raise windows that are clicked on:
|
|
|
|
window->show();
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case FL_MOVE:
|
|
|
|
case FL_DRAG:
|
1999-07-23 01:37:04 +04:00
|
|
|
fl_xmousewin = window; // this should already be set, but just in case.
|
1998-12-15 18:38:16 +03:00
|
|
|
if (pushed()) {
|
|
|
|
w = pushed();
|
1998-10-20 01:00:26 +04:00
|
|
|
event = FL_DRAG;
|
1998-12-15 18:38:16 +03:00
|
|
|
} else if (modal() && w != modal()) {
|
1998-10-06 23:14:55 +04:00
|
|
|
w = 0;
|
1998-12-15 18:38:16 +03:00
|
|
|
}
|
1999-03-03 10:47:22 +03:00
|
|
|
if (grab()) w = grab();
|
1998-10-06 23:14:55 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case FL_RELEASE: {
|
1998-12-15 18:38:16 +03:00
|
|
|
if (pushed()) {
|
|
|
|
w = pushed();
|
|
|
|
pushed_ = 0; // must be zero before callback is done!
|
1998-10-06 23:14:55 +04:00
|
|
|
}
|
1998-12-15 18:38:16 +03:00
|
|
|
if (grab()) w = grab();
|
1999-07-23 01:37:04 +04:00
|
|
|
int r = send(event, w, window);
|
1998-10-06 23:14:55 +04:00
|
|
|
fl_fix_focus();
|
|
|
|
return r;}
|
|
|
|
|
|
|
|
case FL_UNFOCUS:
|
|
|
|
window = 0;
|
|
|
|
case FL_FOCUS:
|
|
|
|
fl_xfocus = window;
|
|
|
|
fl_fix_focus();
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case FL_KEYBOARD:
|
2000-02-18 10:11:09 +03:00
|
|
|
fl_xfocus = window; // this should not happen! But maybe it does:
|
1998-12-15 18:38:16 +03:00
|
|
|
|
1998-10-06 23:14:55 +04:00
|
|
|
// Try it as keystroke, sending it to focus and all parents:
|
1998-12-15 18:38:16 +03:00
|
|
|
for (w = grab() ? grab() : focus(); w; w = w->parent())
|
1999-07-23 01:37:04 +04:00
|
|
|
if (send(FL_KEYBOARD, w, window)) return 1;
|
1998-10-06 23:14:55 +04:00
|
|
|
|
1998-12-15 18:38:16 +03:00
|
|
|
// recursive call to try shortcut:
|
|
|
|
if (handle(FL_SHORTCUT, window)) return 1;
|
|
|
|
|
1999-07-23 01:37:04 +04:00
|
|
|
// and then try a shortcut with the case of the text swapped, by
|
|
|
|
// changing the text and falling through to FL_SHORTCUT case:
|
2000-02-15 11:31:46 +03:00
|
|
|
{char* c = (char*)event_text(); // cast away const
|
|
|
|
if (!isalpha(*c)) return 0;
|
|
|
|
*c = isupper(*c) ? tolower(*c) : toupper(*c);}
|
1998-12-15 18:38:16 +03:00
|
|
|
event = FL_SHORTCUT;
|
|
|
|
|
|
|
|
case FL_SHORTCUT:
|
|
|
|
|
1999-01-04 22:27:00 +03:00
|
|
|
if (grab()) {w = grab(); break;} // send it to grab window
|
1998-12-15 18:38:16 +03:00
|
|
|
|
1998-10-06 23:14:55 +04:00
|
|
|
// Try it as shortcut, sending to mouse widget and all parents:
|
1998-12-15 18:38:16 +03:00
|
|
|
w = belowmouse(); if (!w) {w = modal(); if (!w) w = window;}
|
1999-07-23 01:37:04 +04:00
|
|
|
for (; w; w = w->parent()) if (send(FL_SHORTCUT, w, window)) return 1;
|
1998-10-06 23:14:55 +04:00
|
|
|
|
|
|
|
// try using add_handle() functions:
|
|
|
|
if (send_handlers(FL_SHORTCUT)) return 1;
|
|
|
|
|
|
|
|
// make Escape key close windows:
|
1998-12-15 18:38:16 +03:00
|
|
|
if (event_key()==FL_Escape) {
|
1999-07-27 21:24:14 +04:00
|
|
|
w = modal(); if (!w) w = window;
|
|
|
|
w->do_callback();
|
1998-10-06 23:14:55 +04:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case FL_ENTER:
|
1999-04-10 12:09:39 +04:00
|
|
|
fl_xmousewin = window;
|
|
|
|
fl_fix_focus();
|
1998-10-06 23:14:55 +04:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
case FL_LEAVE:
|
1999-03-03 10:47:22 +03:00
|
|
|
if (window == fl_xmousewin) {fl_xmousewin = 0; fl_fix_focus();}
|
1998-10-06 23:14:55 +04:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
1999-07-23 01:37:04 +04:00
|
|
|
if (w && send(event, w, window)) return 1;
|
1998-10-06 23:14:55 +04:00
|
|
|
return send_handlers(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
// hide() destroys the X window, it does not do unmap!
|
|
|
|
|
|
|
|
void Fl_Window::hide() {
|
1998-12-15 18:34:36 +03:00
|
|
|
clear_visible();
|
1998-10-06 23:14:55 +04:00
|
|
|
if (!shown()) return;
|
|
|
|
|
|
|
|
// remove from the list of windows:
|
|
|
|
Fl_X* x = i;
|
|
|
|
Fl_X** pp = &Fl_X::first;
|
|
|
|
for (; *pp != x; pp = &(*pp)->next) if (!*pp) return;
|
|
|
|
*pp = x->next;
|
|
|
|
i = 0;
|
|
|
|
|
|
|
|
// recursively remove any subwindows:
|
|
|
|
for (Fl_X *w = Fl_X::first; w;) {
|
|
|
|
Fl_Window* W = w->w;
|
|
|
|
if (W->window() == this) {
|
|
|
|
W->hide();
|
|
|
|
W->set_visible();
|
|
|
|
w = Fl_X::first;
|
|
|
|
} else w = w->next;
|
|
|
|
}
|
|
|
|
|
1999-04-10 12:09:39 +04:00
|
|
|
if (this == Fl::modal_) { // we are closing the modal window, find next one:
|
|
|
|
Fl_Window* w;
|
|
|
|
for (w = Fl::first_window(); w; w = Fl::next_window(w))
|
|
|
|
if (w->modal()) break;
|
|
|
|
Fl::modal_ = w;
|
|
|
|
}
|
|
|
|
|
1998-10-06 23:14:55 +04:00
|
|
|
// Make sure no events are sent to this window:
|
|
|
|
fl_throw_focus(this);
|
|
|
|
handle(FL_HIDE);
|
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
if (x->private_dc) ReleaseDC(x->xid,x->private_dc);
|
2000-01-16 07:30:39 +03:00
|
|
|
if (x->xid == fl_window && fl_gc) {
|
|
|
|
ReleaseDC(fl_window, fl_gc);
|
|
|
|
fl_window = (HWND)-1;
|
|
|
|
fl_gc = 0;
|
|
|
|
}
|
1998-10-20 01:00:26 +04:00
|
|
|
#else
|
1998-10-06 23:14:55 +04:00
|
|
|
if (x->region) XDestroyRegion(x->region);
|
1998-10-20 01:00:26 +04:00
|
|
|
#endif
|
1998-10-06 23:14:55 +04:00
|
|
|
XDestroyWindow(fl_display, x->xid);
|
|
|
|
|
|
|
|
delete x;
|
|
|
|
}
|
|
|
|
|
|
|
|
Fl_Window::~Fl_Window() {
|
|
|
|
hide();
|
|
|
|
}
|
|
|
|
|
2000-06-03 12:37:09 +04:00
|
|
|
// FL_SHOW and FL_HIDE are called whenever the visibility of this widget
|
|
|
|
// or any parent changes. We must correctly map/unmap the system's window.
|
|
|
|
|
|
|
|
// For top-level windows it is assummed the window has already been
|
|
|
|
// mapped or unmapped!!! This is because this should only happen when
|
|
|
|
// Fl_Window::show() or Fl_Window::hide() is called, or in response to
|
|
|
|
// iconize/deiconize events from the system.
|
1998-10-06 23:14:55 +04:00
|
|
|
|
|
|
|
int Fl_Window::handle(int event) {
|
|
|
|
if (parent()) switch (event) {
|
|
|
|
case FL_SHOW:
|
|
|
|
if (!shown()) show();
|
2000-06-03 12:37:09 +04:00
|
|
|
else XMapWindow(fl_display, fl_xid(this)); // extra map calls are harmless
|
1998-10-06 23:14:55 +04:00
|
|
|
break;
|
|
|
|
case FL_HIDE:
|
2000-06-03 12:37:09 +04:00
|
|
|
if (shown()) {
|
|
|
|
// Find what really turned invisible, if is was a parent window
|
|
|
|
// we do nothing. We need to avoid unnecessary unmap calls
|
|
|
|
// because they cause the display to blink when the parent is
|
|
|
|
// remapped. However if this or any intermediate non-window
|
|
|
|
// widget has really had hide() called directly on it, we must
|
|
|
|
// unmap because when the parent window is remapped we don't
|
|
|
|
// want to reappear.
|
|
|
|
if (visible()) {
|
|
|
|
Fl_Widget* p = parent(); for (;p->visible();p = p->parent()) {}
|
|
|
|
if (p->type() >= FL_WINDOW) break; // don't do the unmap
|
|
|
|
}
|
|
|
|
XUnmapWindow(fl_display, fl_xid(this));
|
|
|
|
}
|
1998-10-06 23:14:55 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
return Fl_Group::handle(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
// ~Fl_Widget() calls this: this function must get rid of any
|
|
|
|
// global pointers to the widget. This is also called by hide()
|
|
|
|
// and deactivate().
|
|
|
|
|
|
|
|
// call this to free a selection (or change the owner):
|
|
|
|
void Fl::selection_owner(Fl_Widget *owner) {
|
|
|
|
if (selection_owner_ && owner != selection_owner_)
|
|
|
|
selection_owner_->handle(FL_SELECTIONCLEAR);
|
1999-02-26 10:07:56 +03:00
|
|
|
if (focus_ && owner != focus_ && focus_ != selection_owner_)
|
|
|
|
focus_->handle(FL_SELECTIONCLEAR); // clear non-X-selection highlight
|
1998-10-06 23:14:55 +04:00
|
|
|
selection_owner_ = owner;
|
|
|
|
}
|
|
|
|
|
1998-12-09 00:08:51 +03:00
|
|
|
#include <FL/fl_draw.H>
|
|
|
|
|
1999-01-07 19:47:41 +03:00
|
|
|
void Fl_Widget::redraw() {damage(FL_DAMAGE_ALL);}
|
|
|
|
|
1998-12-09 00:08:51 +03:00
|
|
|
void Fl_Widget::damage(uchar flags) {
|
1999-01-07 19:47:41 +03:00
|
|
|
if (type() < FL_WINDOW) {
|
|
|
|
// damage only the rectangle covered by a child widget:
|
|
|
|
damage(flags, x(), y(), w(), h());
|
|
|
|
} else {
|
|
|
|
// damage entire window by deleting the region:
|
|
|
|
Fl_X* i = Fl_X::i((Fl_Window*)this);
|
|
|
|
if (!i) return; // window not mapped, so ignore it
|
|
|
|
if (i->region) {XDestroyRegion(i->region); i->region = 0;}
|
|
|
|
damage_ |= flags;
|
1998-12-09 00:08:51 +03:00
|
|
|
Fl::damage(FL_DAMAGE_CHILD);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Fl_Widget::damage(uchar flags, int X, int Y, int W, int H) {
|
1999-01-07 19:47:41 +03:00
|
|
|
Fl_Widget* window = this;
|
|
|
|
// mark all parent widgets between this and window with FL_DAMAGE_CHILD:
|
|
|
|
while (window->type() < FL_WINDOW) {
|
|
|
|
window->damage_ |= flags;
|
|
|
|
window = window->parent();
|
|
|
|
if (!window) return;
|
1998-12-09 00:08:51 +03:00
|
|
|
flags = FL_DAMAGE_CHILD;
|
|
|
|
}
|
1999-01-07 19:47:41 +03:00
|
|
|
Fl_X* i = Fl_X::i((Fl_Window*)window);
|
|
|
|
if (!i) return; // window not mapped, so ignore it
|
|
|
|
|
|
|
|
if (X<=0 && Y<=0 && W>=window->w() && H>=window->h()) {
|
|
|
|
// if damage covers entire window delete region:
|
|
|
|
window->damage(flags);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// clip the damage to the window and quit if none:
|
|
|
|
if (X < 0) {W += X; X = 0;}
|
|
|
|
if (Y < 0) {H += Y; Y = 0;}
|
|
|
|
if (W > window->w()-X) W = window->w()-X;
|
|
|
|
if (H > window->h()-Y) H = window->h()-Y;
|
|
|
|
if (W <= 0 || H <= 0) return;
|
|
|
|
|
|
|
|
if (window->damage()) {
|
|
|
|
// if we already have damage we must merge with existing region:
|
|
|
|
if (i->region) {
|
1998-12-09 00:08:51 +03:00
|
|
|
#ifndef WIN32
|
1999-01-07 19:47:41 +03:00
|
|
|
XRectangle R;
|
|
|
|
R.x = X; R.y = Y; R.width = W; R.height = H;
|
|
|
|
XUnionRectWithRegion(&R, i->region, i->region);
|
1998-12-09 00:08:51 +03:00
|
|
|
#else
|
1999-01-07 19:47:41 +03:00
|
|
|
Region R = XRectangleRegion(X, Y, W, H);
|
|
|
|
CombineRgn(i->region, i->region, R, RGN_OR);
|
|
|
|
XDestroyRegion(R);
|
1998-12-09 00:08:51 +03:00
|
|
|
#endif
|
|
|
|
}
|
1999-01-07 19:47:41 +03:00
|
|
|
window->damage_ |= flags;
|
|
|
|
} else {
|
|
|
|
// create a new region:
|
|
|
|
if (i->region) XDestroyRegion(i->region);
|
|
|
|
i->region = XRectangleRegion(X,Y,W,H);
|
|
|
|
window->damage_ = flags;
|
1998-12-09 00:08:51 +03:00
|
|
|
}
|
1999-01-07 19:47:41 +03:00
|
|
|
Fl::damage(FL_DAMAGE_CHILD);
|
1998-12-09 00:08:51 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void Fl_Window::flush() {
|
|
|
|
make_current();
|
|
|
|
//if (damage() == FL_DAMAGE_EXPOSE && can_boxcheat(box())) fl_boxcheat = this;
|
|
|
|
fl_clip_region(i->region); i->region = 0;
|
|
|
|
draw();
|
|
|
|
}
|
|
|
|
|
1998-10-20 00:46:58 +04:00
|
|
|
//
|
2000-06-06 01:21:24 +04:00
|
|
|
// End of "$Id: Fl.cxx,v 1.24.2.24 2000/06/05 21:20:46 mike Exp $".
|
1998-10-20 00:46:58 +04:00
|
|
|
//
|