2001-10-03 17:10:38 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2011-02-25 01:05:47 +03:00
|
|
|
// $Id$
|
2001-10-03 17:10:38 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2024-01-01 13:20:05 +03:00
|
|
|
// Copyright (C) 2001-2024 The Bochs Project
|
2001-04-10 05:04:59 +04:00
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
2009-02-08 12:05:52 +03:00
|
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-10-25 01:07:56 +04:00
|
|
|
// Define BX_PLUGGABLE in files that can be compiled into plugins. For
|
2008-02-06 01:57:43 +03:00
|
|
|
// platforms that require a special tag on exported symbols, BX_PLUGGABLE
|
2002-10-25 01:07:56 +04:00
|
|
|
// is used to know when we are exporting symbols and when we are importing.
|
|
|
|
#define BX_PLUGGABLE
|
|
|
|
|
2002-11-19 08:47:45 +03:00
|
|
|
#include "bochs.h"
|
2010-02-26 17:18:19 +03:00
|
|
|
#include "param_names.h"
|
2009-02-23 14:06:53 +03:00
|
|
|
#include "keymap.h"
|
2004-06-19 19:20:15 +04:00
|
|
|
#include "iodev.h"
|
2009-01-31 13:04:25 +03:00
|
|
|
#include "enh_dbg.h"
|
2002-11-19 08:47:45 +03:00
|
|
|
#if BX_WITH_X11
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
#define XK_PUBLISHING
|
|
|
|
#define XK_TECHNICAL
|
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
extern "C" {
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xutil.h>
|
|
|
|
#include <X11/Xos.h>
|
|
|
|
#include <X11/Xatom.h>
|
|
|
|
#include <X11/keysym.h>
|
2019-04-21 23:08:32 +03:00
|
|
|
#if BX_HAVE_XRANDR_H
|
2011-02-11 02:00:56 +03:00
|
|
|
#include <X11/extensions/Xrandr.h>
|
2019-04-21 23:08:32 +03:00
|
|
|
#endif
|
2003-08-11 23:27:57 +04:00
|
|
|
#if BX_HAVE_XPM_H
|
2002-12-30 16:17:39 +03:00
|
|
|
#include <X11/xpm.h>
|
2003-06-13 20:05:03 +04:00
|
|
|
#endif
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2003-08-11 23:27:57 +04:00
|
|
|
#if BX_HAVE_XPM_H
|
2002-12-30 16:17:39 +03:00
|
|
|
#include "icon_bochs.xpm"
|
2003-06-13 20:05:03 +04:00
|
|
|
#else
|
|
|
|
#include "icon_bochs.h"
|
|
|
|
#endif
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-01-17 00:14:11 +03:00
|
|
|
#include "font/vga.bitmap.h"
|
|
|
|
|
2020-07-15 19:26:46 +03:00
|
|
|
#define COMMAND_MODE_KEYSYM XK_F7
|
|
|
|
|
2002-10-25 01:07:56 +04:00
|
|
|
class bx_x_gui_c : public bx_gui_c {
|
|
|
|
public:
|
2017-01-15 14:44:43 +03:00
|
|
|
bx_x_gui_c(void);
|
2002-10-25 01:07:56 +04:00
|
|
|
DECLARE_GUI_VIRTUAL_METHODS()
|
2004-08-15 23:27:15 +04:00
|
|
|
DECLARE_GUI_NEW_VIRTUAL_METHODS()
|
2021-01-30 21:32:52 +03:00
|
|
|
virtual void set_font(bool lg);
|
2020-11-16 18:39:35 +03:00
|
|
|
virtual void draw_char(Bit8u ch, Bit8u fc, Bit8u bc, Bit16u xc, Bit16u yc,
|
|
|
|
Bit8u fw, Bit8u fh, Bit8u fx, Bit8u fy,
|
2023-12-29 19:47:44 +03:00
|
|
|
bool gfxcharw9, Bit8u cs, Bit8u ce, bool curs, bool font2);
|
2004-02-02 02:48:57 +03:00
|
|
|
virtual void beep_on(float frequency);
|
|
|
|
virtual void beep_off();
|
2019-04-21 23:08:32 +03:00
|
|
|
#if BX_HAVE_XRANDR_H
|
2004-02-22 16:03:02 +03:00
|
|
|
virtual void get_capabilities(Bit16u *xres, Bit16u *yres, Bit16u *bpp);
|
2019-04-21 23:08:32 +03:00
|
|
|
#endif
|
2017-01-15 15:57:06 +03:00
|
|
|
virtual void set_display_mode(disp_mode_t newmode);
|
2021-01-30 21:32:52 +03:00
|
|
|
virtual void set_mouse_mode_absxy(bool mode);
|
2006-01-22 21:15:48 +03:00
|
|
|
#if BX_SHOW_IPS
|
2006-01-22 15:31:16 +03:00
|
|
|
virtual void show_ips(Bit32u ips_count);
|
2006-01-22 21:15:48 +03:00
|
|
|
#endif
|
2017-01-15 15:57:06 +03:00
|
|
|
#if BX_USE_IDLE_HACK
|
|
|
|
virtual void sim_is_idle(void);
|
|
|
|
#endif
|
2021-01-30 21:32:52 +03:00
|
|
|
virtual void statusbar_setitem_specific(int element, bool active, bool w);
|
2017-01-01 20:45:06 +03:00
|
|
|
private:
|
|
|
|
void send_mouse_status(void);
|
|
|
|
void xkeypress(KeySym keysym, int press_release);
|
2002-10-25 01:07:56 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
// declare one instance of the gui object and call macro to insert the
|
|
|
|
// plugin code
|
|
|
|
static bx_x_gui_c *theGui = NULL;
|
|
|
|
IMPLEMENT_GUI_PLUGIN_CODE(x)
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-10-25 01:07:56 +04:00
|
|
|
#define LOG_THIS theGui->
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
#define MAX_MAPPED_STRING_LENGTH 10
|
|
|
|
|
|
|
|
/* These are used as arguments to nearly every Xlib routine, so it saves
|
|
|
|
* routine arguments to declare them global. If there were
|
|
|
|
* additional source files, they would be declared extern there. */
|
|
|
|
Display *bx_x_display;
|
|
|
|
int bx_x_screen_num;
|
2004-08-15 23:27:15 +04:00
|
|
|
static Visual *default_visual;
|
2001-04-10 05:04:59 +04:00
|
|
|
static Colormap default_cmap;
|
2021-01-30 21:32:52 +03:00
|
|
|
static bool x11_private_colormap;
|
2001-04-10 05:04:59 +04:00
|
|
|
static unsigned long white_pixel=0, black_pixel=0;
|
|
|
|
|
|
|
|
static char *progname; /* name this program was invoked by */
|
|
|
|
|
2011-03-26 11:27:13 +03:00
|
|
|
// text display
|
|
|
|
static unsigned font_width, font_height;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2011-03-26 11:27:13 +03:00
|
|
|
// graphics display
|
2001-04-10 05:04:59 +04:00
|
|
|
static Window win;
|
|
|
|
static GC gc, gc_inv, gc_headerbar, gc_headerbar_inv;
|
|
|
|
static unsigned dimension_x=0, dimension_y=0;
|
2021-06-06 12:38:47 +03:00
|
|
|
static Bit16u x11_max_xres=0, x11_max_yres=0;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
static XImage *ximage = NULL;
|
|
|
|
static unsigned imDepth, imWide, imBPP;
|
|
|
|
|
2011-03-26 11:27:13 +03:00
|
|
|
// mouse cursor
|
2021-01-30 21:32:52 +03:00
|
|
|
static bool mouse_captured = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
static int prev_x=-1, prev_y=-1;
|
2004-12-07 00:12:11 +03:00
|
|
|
static int current_x=-1, current_y=-1, current_z=0;
|
2001-04-10 05:04:59 +04:00
|
|
|
static unsigned mouse_button_state = 0;
|
2021-01-30 21:32:52 +03:00
|
|
|
static bool x11_mouse_mode_absxy = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
static int warp_home_x = 200;
|
|
|
|
static int warp_home_y = 200;
|
|
|
|
static int mouse_enable_x = 0;
|
|
|
|
static int mouse_enable_y = 0;
|
|
|
|
static int warp_dx = 0;
|
|
|
|
static int warp_dy = 0;
|
|
|
|
|
|
|
|
static void warp_cursor(int dx, int dy);
|
|
|
|
static void disable_cursor();
|
|
|
|
static void enable_cursor();
|
|
|
|
|
2011-07-31 19:46:37 +04:00
|
|
|
// keyboard
|
2021-01-30 21:32:52 +03:00
|
|
|
static bool x11_use_kbd_mapping = 0;
|
2017-01-15 14:44:43 +03:00
|
|
|
static Bit32u convertStringToXKeysym(const char *string);
|
2002-03-06 12:31:55 +03:00
|
|
|
|
2021-01-30 21:32:52 +03:00
|
|
|
static bool x_init_done = 0;
|
2003-01-30 21:41:00 +03:00
|
|
|
|
2023-12-29 19:47:44 +03:00
|
|
|
static Pixmap vgafont[2][256];
|
2003-01-17 00:14:11 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
// header bar
|
2010-01-24 15:46:42 +03:00
|
|
|
static unsigned bx_bitmap_entries = 0;
|
2006-01-17 20:15:29 +03:00
|
|
|
static struct {
|
2001-04-10 05:04:59 +04:00
|
|
|
Pixmap bmap;
|
|
|
|
unsigned xdim;
|
|
|
|
unsigned ydim;
|
2010-01-24 15:46:42 +03:00
|
|
|
} bx_bitmaps[BX_MAX_PIXMAPS];
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
static unsigned bx_headerbar_y = 0;
|
|
|
|
static unsigned bx_bitmap_left_xorigin = 0; // pixels from left
|
|
|
|
static unsigned bx_bitmap_right_xorigin = 0; // pixels from right
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
// status bar
|
2004-02-12 22:39:13 +03:00
|
|
|
static unsigned bx_statusbar_y = 18;
|
|
|
|
static unsigned bx_statusitem_pos[12] = {
|
2004-02-15 22:46:13 +03:00
|
|
|
0, 200, 240, 280, 320, 360, 400, 440, 480, 520, 560, 600
|
2010-01-24 15:46:42 +03:00
|
|
|
};
|
2021-01-30 21:32:52 +03:00
|
|
|
static bool bx_statusitem_active[12];
|
2020-12-16 15:24:35 +03:00
|
|
|
static long bx_status_leds[3];
|
2020-12-15 14:29:36 +03:00
|
|
|
static long bx_status_graytext;
|
2005-02-27 19:55:34 +03:00
|
|
|
static char bx_status_info_text[34];
|
2006-01-23 21:34:47 +03:00
|
|
|
#if BX_SHOW_IPS
|
2024-03-29 22:12:41 +03:00
|
|
|
static bool x11_ips_update = 0;
|
2006-01-23 21:34:47 +03:00
|
|
|
static char x11_ips_text[20];
|
2020-07-15 19:26:46 +03:00
|
|
|
static Bit8u x11_show_info_msg = 0;
|
|
|
|
static Bit8u x11_info_msg_counter = 0;
|
2006-01-23 21:34:47 +03:00
|
|
|
#endif
|
2004-02-12 22:39:13 +03:00
|
|
|
|
2001-06-26 02:57:27 +04:00
|
|
|
Bit32u ascii_to_key_event[0x5f] = {
|
|
|
|
// !"#$%&'
|
|
|
|
BX_KEY_SPACE,
|
|
|
|
BX_KEY_1,
|
|
|
|
BX_KEY_SINGLE_QUOTE,
|
|
|
|
BX_KEY_3,
|
|
|
|
BX_KEY_4,
|
|
|
|
BX_KEY_5,
|
|
|
|
BX_KEY_7,
|
|
|
|
BX_KEY_SINGLE_QUOTE,
|
|
|
|
|
|
|
|
// ()*+,-./
|
|
|
|
BX_KEY_9,
|
|
|
|
BX_KEY_0,
|
|
|
|
BX_KEY_8,
|
|
|
|
BX_KEY_EQUALS,
|
|
|
|
BX_KEY_COMMA,
|
|
|
|
BX_KEY_MINUS,
|
|
|
|
BX_KEY_PERIOD,
|
|
|
|
BX_KEY_SLASH,
|
|
|
|
|
|
|
|
// 01234567
|
|
|
|
BX_KEY_0,
|
|
|
|
BX_KEY_1,
|
|
|
|
BX_KEY_2,
|
|
|
|
BX_KEY_3,
|
|
|
|
BX_KEY_4,
|
|
|
|
BX_KEY_5,
|
|
|
|
BX_KEY_6,
|
|
|
|
BX_KEY_7,
|
|
|
|
|
|
|
|
// 89:;<=>?
|
|
|
|
BX_KEY_8,
|
|
|
|
BX_KEY_9,
|
|
|
|
BX_KEY_SEMICOLON,
|
|
|
|
BX_KEY_SEMICOLON,
|
|
|
|
BX_KEY_COMMA,
|
|
|
|
BX_KEY_EQUALS,
|
|
|
|
BX_KEY_PERIOD,
|
|
|
|
BX_KEY_SLASH,
|
|
|
|
|
|
|
|
// @ABCDEFG
|
|
|
|
BX_KEY_2,
|
|
|
|
BX_KEY_A,
|
|
|
|
BX_KEY_B,
|
|
|
|
BX_KEY_C,
|
|
|
|
BX_KEY_D,
|
|
|
|
BX_KEY_E,
|
|
|
|
BX_KEY_F,
|
|
|
|
BX_KEY_G,
|
|
|
|
|
|
|
|
|
|
|
|
// HIJKLMNO
|
|
|
|
BX_KEY_H,
|
|
|
|
BX_KEY_I,
|
|
|
|
BX_KEY_J,
|
|
|
|
BX_KEY_K,
|
|
|
|
BX_KEY_L,
|
|
|
|
BX_KEY_M,
|
|
|
|
BX_KEY_N,
|
|
|
|
BX_KEY_O,
|
|
|
|
|
|
|
|
|
|
|
|
// PQRSTUVW
|
|
|
|
BX_KEY_P,
|
|
|
|
BX_KEY_Q,
|
|
|
|
BX_KEY_R,
|
|
|
|
BX_KEY_S,
|
|
|
|
BX_KEY_T,
|
|
|
|
BX_KEY_U,
|
|
|
|
BX_KEY_V,
|
|
|
|
BX_KEY_W,
|
|
|
|
|
|
|
|
// XYZ[\]^_
|
|
|
|
BX_KEY_X,
|
|
|
|
BX_KEY_Y,
|
|
|
|
BX_KEY_Z,
|
|
|
|
BX_KEY_LEFT_BRACKET,
|
|
|
|
BX_KEY_BACKSLASH,
|
|
|
|
BX_KEY_RIGHT_BRACKET,
|
|
|
|
BX_KEY_6,
|
|
|
|
BX_KEY_MINUS,
|
|
|
|
|
|
|
|
// `abcdefg
|
|
|
|
BX_KEY_GRAVE,
|
|
|
|
BX_KEY_A,
|
|
|
|
BX_KEY_B,
|
|
|
|
BX_KEY_C,
|
|
|
|
BX_KEY_D,
|
|
|
|
BX_KEY_E,
|
|
|
|
BX_KEY_F,
|
|
|
|
BX_KEY_G,
|
|
|
|
|
|
|
|
// hijklmno
|
|
|
|
BX_KEY_H,
|
|
|
|
BX_KEY_I,
|
|
|
|
BX_KEY_J,
|
|
|
|
BX_KEY_K,
|
|
|
|
BX_KEY_L,
|
|
|
|
BX_KEY_M,
|
|
|
|
BX_KEY_N,
|
|
|
|
BX_KEY_O,
|
|
|
|
|
|
|
|
// pqrstuvw
|
|
|
|
BX_KEY_P,
|
|
|
|
BX_KEY_Q,
|
|
|
|
BX_KEY_R,
|
|
|
|
BX_KEY_S,
|
|
|
|
BX_KEY_T,
|
|
|
|
BX_KEY_U,
|
|
|
|
BX_KEY_V,
|
|
|
|
BX_KEY_W,
|
|
|
|
|
|
|
|
// xyz{|}~
|
|
|
|
BX_KEY_X,
|
|
|
|
BX_KEY_Y,
|
|
|
|
BX_KEY_Z,
|
|
|
|
BX_KEY_LEFT_BRACKET,
|
|
|
|
BX_KEY_BACKSLASH,
|
|
|
|
BX_KEY_RIGHT_BRACKET,
|
|
|
|
BX_KEY_GRAVE
|
2008-02-16 01:05:43 +03:00
|
|
|
};
|
2001-06-26 02:57:27 +04:00
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
extern Bit8u graphics_snapshot[32 * 1024];
|
|
|
|
|
2003-01-17 21:16:02 +03:00
|
|
|
static void create_internal_vga_font(void);
|
2001-04-10 05:04:59 +04:00
|
|
|
// extern "C" void select_visual(void);
|
|
|
|
|
|
|
|
#define ROUNDUP(nbytes, pad) ((((nbytes) + ((pad)-1)) / (pad)) * ((pad)>>3))
|
|
|
|
|
|
|
|
|
2002-03-20 02:38:08 +03:00
|
|
|
#define MAX_VGA_COLORS 256
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-03-20 02:38:08 +03:00
|
|
|
unsigned long col_vals[MAX_VGA_COLORS]; // 256 VGA colors
|
2001-04-10 05:04:59 +04:00
|
|
|
unsigned curr_foreground, curr_background;
|
|
|
|
|
2017-01-15 14:44:43 +03:00
|
|
|
BxEvent *x11_notify_callback(void *unused, BxEvent *event);
|
2009-10-24 17:22:36 +04:00
|
|
|
static bxevent_handler old_callback = NULL;
|
|
|
|
static void *old_callback_arg = NULL;
|
2005-10-16 17:11:38 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
// X11 specific funcions
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-03-20 02:38:08 +03:00
|
|
|
// Try to allocate NCOLORS at once in the colormap provided. If it can
|
|
|
|
// be done, return true. If not, return false. (In either case, free
|
|
|
|
// up the color cells so that we don't add to the problem!) This is used
|
|
|
|
// to determine whether Bochs should use a private colormap even when the
|
|
|
|
// user did not specify it.
|
2021-01-30 21:32:52 +03:00
|
|
|
static bool test_alloc_colors(Colormap cmap, Bit32u n_tries)
|
2017-01-15 14:44:43 +03:00
|
|
|
{
|
2002-03-20 02:38:08 +03:00
|
|
|
XColor color;
|
|
|
|
unsigned long pixel[MAX_VGA_COLORS];
|
2021-01-30 21:32:52 +03:00
|
|
|
bool pixel_valid[MAX_VGA_COLORS];
|
2002-03-20 02:38:08 +03:00
|
|
|
Bit32u n_allocated = 0;
|
|
|
|
Bit32u i;
|
|
|
|
color.flags = DoRed | DoGreen | DoBlue;
|
|
|
|
for (i=0; i<n_tries; i++) {
|
2008-02-06 01:57:43 +03:00
|
|
|
// choose wierd color values that are unlikely to already be in the
|
2002-03-20 02:38:08 +03:00
|
|
|
// colormap.
|
|
|
|
color.red = ((i+41)%MAX_VGA_COLORS) << 8;
|
|
|
|
color.green = ((i+42)%MAX_VGA_COLORS) << 8;
|
|
|
|
color.blue = ((i+43)%MAX_VGA_COLORS) << 8;
|
|
|
|
pixel_valid[i] = false;
|
|
|
|
if (XAllocColor (bx_x_display, cmap, &color)) {
|
|
|
|
pixel[i] = color.pixel;
|
|
|
|
pixel_valid[i] = true;
|
|
|
|
n_allocated++;
|
|
|
|
}
|
|
|
|
}
|
2017-01-15 14:44:43 +03:00
|
|
|
BX_INFO(("test_alloc_colors: %d colors available out of %d colors tried", n_allocated, n_tries));
|
2002-03-20 02:38:08 +03:00
|
|
|
// now free them all
|
|
|
|
for (i=0; i<n_tries; i++) {
|
2017-01-15 14:44:43 +03:00
|
|
|
if (pixel_valid[i]) XFreeColors(bx_x_display, cmap, &pixel[i], 1, 0);
|
2002-03-20 02:38:08 +03:00
|
|
|
}
|
|
|
|
return (n_allocated == n_tries);
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
void create_internal_vga_font(void)
|
|
|
|
{
|
|
|
|
// Default values
|
|
|
|
font_width=8;
|
|
|
|
font_height=16;
|
|
|
|
|
|
|
|
for(int i=0; i<256; i++) {
|
2023-12-29 19:47:44 +03:00
|
|
|
vgafont[0][i]=XCreateBitmapFromData(bx_x_display, win, (const char*)bx_vgafont[i].data,
|
2017-01-15 15:57:06 +03:00
|
|
|
font_width, font_height);
|
2023-12-29 19:47:44 +03:00
|
|
|
if(vgafont[0][i] == None)
|
|
|
|
BX_PANIC(("Can't create vga font [%d]", i));
|
|
|
|
vgafont[1][i]=XCreateBitmapFromData(bx_x_display, win, (const char*)bx_vgafont[i].data,
|
|
|
|
font_width, font_height);
|
|
|
|
if(vgafont[1][i] == None)
|
2017-01-15 15:57:06 +03:00
|
|
|
BX_PANIC(("Can't create vga font [%d]", i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check to see if this is a repeated key. (grabbed from SDL 1.2)
|
|
|
|
(idea shamelessly lifted from GII -- thanks guys! :)
|
|
|
|
*/
|
|
|
|
static int X11_KeyRepeat(Display *display, XEvent *event)
|
|
|
|
{
|
|
|
|
XEvent peekevent;
|
|
|
|
int repeated;
|
|
|
|
|
|
|
|
repeated = 0;
|
|
|
|
if (XPending(display)) {
|
|
|
|
XPeekEvent(display, &peekevent);
|
|
|
|
if ((peekevent.type == KeyPress) &&
|
|
|
|
(peekevent.xkey.keycode == event->xkey.keycode) &&
|
|
|
|
((peekevent.xkey.time-event->xkey.time) < 2)) {
|
|
|
|
repeated = 1;
|
|
|
|
XNextEvent(display, &peekevent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return repeated;
|
|
|
|
}
|
|
|
|
|
2021-01-30 21:32:52 +03:00
|
|
|
void x11_set_status_text(int element, const char *text, bool active,
|
2020-12-15 14:29:36 +03:00
|
|
|
Bit8u color=0)
|
2017-01-15 15:57:06 +03:00
|
|
|
{
|
|
|
|
int xleft, xsize, sb_ypos;
|
|
|
|
|
|
|
|
xleft = bx_statusitem_pos[element] + 2;
|
|
|
|
xsize = bx_statusitem_pos[element+1] - xleft;
|
|
|
|
sb_ypos = dimension_y + bx_headerbar_y;
|
|
|
|
if (element < 1) {
|
|
|
|
if (strcmp(bx_status_info_text, text)) {
|
|
|
|
strcpy(bx_status_info_text, text);
|
|
|
|
}
|
|
|
|
XFillRectangle(bx_x_display, win, gc_headerbar_inv, xleft, sb_ypos+2, xsize,
|
|
|
|
bx_statusbar_y-2);
|
2020-07-15 19:26:46 +03:00
|
|
|
if (strlen(text) > 0) {
|
|
|
|
XDrawString(bx_x_display, win, gc_headerbar, xleft, sb_ypos+bx_statusbar_y-2,
|
|
|
|
text, strlen(text));
|
|
|
|
}
|
|
|
|
#if BX_SHOW_IPS
|
|
|
|
if (!active) {
|
2020-12-15 14:29:36 +03:00
|
|
|
if (color == 0) { // volatile info text for 3 seconds
|
2020-07-15 19:26:46 +03:00
|
|
|
x11_info_msg_counter = 3;
|
|
|
|
} else {
|
|
|
|
x11_show_info_msg = (strlen(text) > 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2017-01-15 15:57:06 +03:00
|
|
|
} else if (element <= BX_MAX_STATUSITEMS) {
|
|
|
|
bx_statusitem_active[element] = active;
|
|
|
|
if (active) {
|
2020-12-16 15:24:35 +03:00
|
|
|
XSetForeground(bx_x_display, gc_headerbar, bx_status_leds[color]);
|
2017-01-15 15:57:06 +03:00
|
|
|
XFillRectangle(bx_x_display, win, gc_headerbar, xleft, sb_ypos+2, xsize-1, bx_statusbar_y-2);
|
|
|
|
XSetForeground(bx_x_display, gc_headerbar, black_pixel);
|
|
|
|
} else {
|
|
|
|
XFillRectangle(bx_x_display, win, gc_headerbar_inv, xleft, sb_ypos+2, xsize-1, bx_statusbar_y-2);
|
|
|
|
XSetForeground(bx_x_display, gc_headerbar, bx_status_graytext);
|
|
|
|
}
|
|
|
|
XDrawString(bx_x_display, win, gc_headerbar, xleft, sb_ypos+bx_statusbar_y-2,
|
|
|
|
text, strlen(text));
|
|
|
|
XSetForeground(bx_x_display, gc_headerbar, black_pixel);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void warp_cursor(int dx, int dy)
|
|
|
|
{
|
|
|
|
if (mouse_captured && (warp_dx || warp_dy || dx || dy)) {
|
|
|
|
warp_dx = dx;
|
|
|
|
warp_dy = dy;
|
|
|
|
XWarpPointer(bx_x_display, None, None, 0, 0, 0, 0, dx, dy);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void disable_cursor()
|
|
|
|
{
|
|
|
|
static Cursor cursor;
|
|
|
|
static unsigned cursor_created = 0;
|
|
|
|
|
|
|
|
static int shape_width = 16,
|
|
|
|
shape_height = 16,
|
|
|
|
mask_width = 16,
|
|
|
|
mask_height = 16;
|
|
|
|
|
|
|
|
static Bit32u shape_bits[(16*16)/32] = {
|
|
|
|
0x00000000, 0x00000000, 0x00000000, 0x00000000,
|
|
|
|
0x00000000, 0x00000000, 0x00000000, 0x00000000,
|
|
|
|
};
|
|
|
|
static Bit32u mask_bits[(16*16)/32] = {
|
|
|
|
0x00000000, 0x00000000, 0x00000000, 0x00000000,
|
|
|
|
0x00000000, 0x00000000, 0x00000000, 0x00000000,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!cursor_created) {
|
|
|
|
Pixmap shape, mask;
|
|
|
|
XColor white, black;
|
|
|
|
shape = XCreatePixmapFromBitmapData(bx_x_display,
|
|
|
|
RootWindow(bx_x_display,bx_x_screen_num),
|
|
|
|
(char*)shape_bits,
|
|
|
|
shape_width,
|
|
|
|
shape_height,
|
|
|
|
1, 0, 1);
|
|
|
|
mask = XCreatePixmapFromBitmapData(bx_x_display,
|
|
|
|
RootWindow(bx_x_display,bx_x_screen_num),
|
|
|
|
(char*)mask_bits,
|
|
|
|
mask_width,
|
|
|
|
mask_height,
|
|
|
|
1, 0, 1);
|
|
|
|
XParseColor(bx_x_display, default_cmap, "black", &black);
|
|
|
|
XParseColor(bx_x_display, default_cmap, "white", &white);
|
|
|
|
cursor = XCreatePixmapCursor(bx_x_display, shape, mask,
|
|
|
|
&white, &black, 1, 1);
|
|
|
|
cursor_created = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
XDefineCursor(bx_x_display, win, cursor);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void enable_cursor()
|
|
|
|
{
|
|
|
|
XUndefineCursor(bx_x_display, win);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* convertStringToXKeysym is a keymap callback
|
|
|
|
* used when reading the keymap file.
|
|
|
|
* It converts a Symblic String to a GUI Constant
|
|
|
|
*
|
|
|
|
* It returns a Bit32u constant or BX_KEYMAP_UNKNOWN if it fails
|
|
|
|
*/
|
|
|
|
static Bit32u convertStringToXKeysym(const char *string)
|
|
|
|
{
|
|
|
|
if (strncmp ("XK_", string, 3) != 0)
|
|
|
|
return BX_KEYMAP_UNKNOWN;
|
|
|
|
KeySym keysym=XStringToKeysym(string+3);
|
|
|
|
|
|
|
|
// failure, return unknown
|
|
|
|
if(keysym==NoSymbol) return BX_KEYMAP_UNKNOWN;
|
|
|
|
|
|
|
|
return((Bit32u)keysym);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if BX_USE_IDLE_HACK
|
|
|
|
|
|
|
|
/* BX_USE_IDLE_HACK: a small idle hack by
|
|
|
|
* Roland.Mainz@informatik.med.uni-giessen.de to prevent bochs
|
|
|
|
* from consuming 100% CPU time even when it is not required (for
|
|
|
|
* example, the OS in the emulator calls HLT to wait for an interupt)
|
|
|
|
* pro:
|
|
|
|
* - no more 100% CPU usage
|
|
|
|
* contra:
|
|
|
|
* - we're sleeping too long
|
|
|
|
* - bochs still consumes ~10%-20% CPU time while executing an idle
|
|
|
|
* linux kernel
|
|
|
|
* - this is an hack
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* XPeekEvent() with timeout
|
|
|
|
* (adopted from mozilla/gfx/src/xprint/xprintutil_printtofile.c#XNextEventTimeout())
|
|
|
|
*/
|
|
|
|
static Bool XPeekEventTimeout(Display *display, XEvent *event_return, struct timeval *timeout)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
fd_set readfds;
|
|
|
|
int display_fd = XConnectionNumber(display);
|
|
|
|
|
|
|
|
/* small shortcut... */
|
|
|
|
if(timeout == NULL)
|
|
|
|
{
|
|
|
|
XPeekEvent(display, event_return);
|
|
|
|
return(True);
|
|
|
|
}
|
|
|
|
|
|
|
|
FD_ZERO(&readfds);
|
|
|
|
FD_SET(display_fd, &readfds);
|
|
|
|
|
|
|
|
/* Note/bug: In the case of internal X events (like used to trigger callbacks
|
|
|
|
* registered by XpGetDocumentData()&co.) select() will return with "new info"
|
|
|
|
* - but XNextEvent() below processes these _internal_ events silently - and
|
|
|
|
* will block if there are no other non-internal events.
|
|
|
|
* The workaround here is to check with XEventsQueued() if there are non-internal
|
|
|
|
* events queued - if not select() will be called again - unfortunately we use
|
|
|
|
* the old timeout here instead of the "remaining" time... (this only would hurt
|
|
|
|
* if the timeout would be really long - but for current use with values below
|
|
|
|
* 1/2 secs it does not hurt... =:-)
|
|
|
|
*/
|
|
|
|
while(XEventsQueued(display, QueuedAfterFlush) == 0)
|
|
|
|
{
|
|
|
|
res = select(display_fd+1, &readfds, NULL, NULL, timeout);
|
|
|
|
|
|
|
|
switch(res)
|
|
|
|
{
|
|
|
|
case -1: /* select() error - should not happen */
|
|
|
|
if (errno == EINTR)
|
|
|
|
break; // caused e.g. by alarm(3)
|
|
|
|
perror("XPeekEventTimeout: select() failure");
|
|
|
|
return(False);
|
|
|
|
|
|
|
|
case 0: /* timeout */
|
|
|
|
return(False);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
XPeekEvent(display, event_return);
|
|
|
|
return(True);
|
|
|
|
}
|
|
|
|
#endif
|
2017-01-15 14:44:43 +03:00
|
|
|
|
|
|
|
// X11 implementation of the bx_gui_c methods (see nogui.cc for details)
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
bx_x_gui_c::bx_x_gui_c() {}
|
|
|
|
|
2012-03-03 16:41:24 +04:00
|
|
|
void bx_x_gui_c::specific_init(int argc, char **argv, unsigned headerbar_y)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2008-05-12 22:46:46 +04:00
|
|
|
int i;
|
2001-04-10 05:04:59 +04:00
|
|
|
int x, y; /* window position */
|
|
|
|
unsigned int border_width = 4; /* four pixels */
|
2007-10-25 03:09:59 +04:00
|
|
|
const char *window_name = BOCHS_WINDOW_NAME;
|
|
|
|
const char *icon_name = "Bochs";
|
2003-06-13 20:05:03 +04:00
|
|
|
Pixmap icon_pixmap;
|
2003-08-11 23:27:57 +04:00
|
|
|
#if BX_HAVE_XPM_H
|
2003-06-13 20:05:03 +04:00
|
|
|
Pixmap icon_mask;
|
|
|
|
#endif
|
2001-04-10 05:04:59 +04:00
|
|
|
XSizeHints size_hints;
|
|
|
|
char *display_name = NULL;
|
|
|
|
/* create GC for text and drawing */
|
|
|
|
unsigned long valuemask = 0; /* ignore XGCvalues and use defaults */
|
|
|
|
XGCValues values;
|
2021-06-06 12:38:47 +03:00
|
|
|
int default_depth;
|
2001-04-10 05:04:59 +04:00
|
|
|
XEvent report;
|
|
|
|
XSetWindowAttributes win_attr;
|
|
|
|
unsigned long plane_masks_return[1];
|
|
|
|
XColor color;
|
2023-12-23 17:44:00 +03:00
|
|
|
#if BX_HAVE_XRANDR_H
|
2021-06-06 12:38:47 +03:00
|
|
|
int event_base, error_base;
|
2023-12-23 17:44:00 +03:00
|
|
|
#endif
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-10-25 01:07:56 +04:00
|
|
|
put("XGUI");
|
2001-04-10 05:04:59 +04:00
|
|
|
bx_headerbar_y = headerbar_y;
|
|
|
|
progname = argv[0];
|
2021-06-11 17:46:38 +03:00
|
|
|
console.present = 1;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2024-03-30 13:11:07 +03:00
|
|
|
#if (BX_DEBUGGER && BX_DEBUGGER_GUI) || BX_SUPPORT_SOUNDLOW || BX_SUPPORT_VOODOO
|
|
|
|
// This is only necessary when GTK+ and Xlib are sharing the same
|
|
|
|
// connection. XInitThreads() must finish before any calls to GTK+
|
|
|
|
// or Xlib are made.
|
|
|
|
if (XInitThreads() == 0) {
|
|
|
|
BX_PANIC(("trying to run Bochs with the GTK+ "
|
|
|
|
"debugger on a non-threading X11."));
|
2014-12-16 22:48:50 +03:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
/* connect to X server */
|
2008-02-16 01:05:43 +03:00
|
|
|
if ((bx_x_display=XOpenDisplay(display_name)) == NULL)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("%s: cannot connect to X server %s",
|
merge in BRANCH-io-cleanup.
To see the commit logs for this use either cvsweb or
cvs update -r BRANCH-io-cleanup and then 'cvs log' the various files.
In general this provides a generic interface for logging.
logfunctions:: is a class that is inherited by some classes, and also
. allocated as a standalone global called 'genlog'. All logging uses
. one of the ::info(), ::error(), ::ldebug(), ::panic() methods of this
. class through 'BX_INFO(), BX_ERROR(), BX_DEBUG(), BX_PANIC()' macros
. respectively.
.
. An example usage:
. BX_INFO(("Hello, World!\n"));
iofunctions:: is a class that is allocated once by default, and assigned
as the iofunction of each logfunctions instance. It is this class that
maintains the file descriptor and other output related code, at this
point using vfprintf(). At some future point, someone may choose to
write a gui 'console' for bochs to which messages would be redirected
simply by assigning a different iofunction class to the various logfunctions
objects.
More cleanup is coming, but this works for now. If you want to see alot
of debugging output, in main.cc, change onoff[LOGLEV_DEBUG]=0 to =1.
Comments, bugs, flames, to me: todd@fries.net
2001-05-15 18:49:57 +04:00
|
|
|
progname, XDisplayName(display_name)));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* get screen size from display structure macro */
|
|
|
|
bx_x_screen_num = DefaultScreen(bx_x_display);
|
|
|
|
|
2024-03-30 13:11:07 +03:00
|
|
|
// redirect notify callback to X11 specific code
|
|
|
|
SIM->get_notify_callback(&old_callback, &old_callback_arg);
|
|
|
|
assert(old_callback != NULL);
|
|
|
|
SIM->set_notify_callback(x11_notify_callback, NULL);
|
|
|
|
|
|
|
|
// parse x11 specific options
|
|
|
|
Bit8u flags = BX_GUI_OPT_NOKEYREPEAT | BX_GUI_OPT_HIDE_IPS | BX_GUI_OPT_CMDMODE
|
|
|
|
| BX_GUI_OPT_NO_GUI_CONSOLE;
|
|
|
|
if (argc > 1) {
|
|
|
|
for (i = 1; i < argc; i++) {
|
|
|
|
if (!parse_common_gui_options(argv[i], flags)) {
|
|
|
|
BX_PANIC(("Unknown x11 option '%s'", argv[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
/* Note that in a real application, x and y would default to 0
|
|
|
|
* but would be settable from the command line or resource database.
|
|
|
|
*/
|
|
|
|
x = y = 0;
|
|
|
|
|
2003-01-17 00:14:11 +03:00
|
|
|
// Temporary values so we can create the window
|
|
|
|
font_width = 8;
|
|
|
|
font_height = 16;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2020-11-22 12:02:38 +03:00
|
|
|
dimension_x = guest_xres;
|
|
|
|
dimension_y = guest_yres;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* create opaque window */
|
|
|
|
win = XCreateSimpleWindow(bx_x_display, RootWindow(bx_x_display,bx_x_screen_num),
|
|
|
|
x, y,
|
|
|
|
dimension_x,
|
2004-04-11 14:58:09 +04:00
|
|
|
dimension_y + bx_headerbar_y + bx_statusbar_y,
|
2001-04-10 05:04:59 +04:00
|
|
|
border_width,
|
|
|
|
BlackPixel(bx_x_display, bx_x_screen_num),
|
|
|
|
BlackPixel(bx_x_display, bx_x_screen_num));
|
|
|
|
|
|
|
|
// (attempt to) enable backing store
|
|
|
|
win_attr.save_under=1;
|
|
|
|
win_attr.backing_store=Always;
|
|
|
|
XChangeWindowAttributes(bx_x_display,win,CWSaveUnder|CWBackingStore,&win_attr);
|
|
|
|
|
|
|
|
default_depth = DefaultDepth(bx_x_display, bx_x_screen_num);
|
|
|
|
default_visual = DefaultVisual(bx_x_display, bx_x_screen_num);
|
|
|
|
|
2012-01-20 19:56:17 +04:00
|
|
|
x11_private_colormap = SIM->get_param_bool(BXPN_PRIVATE_COLORMAP)->get();
|
|
|
|
if (!x11_private_colormap) {
|
2002-03-20 02:38:08 +03:00
|
|
|
default_cmap = DefaultColormap(bx_x_display, bx_x_screen_num);
|
|
|
|
// try to use default colormap. If not enough colors are available,
|
|
|
|
// then switch to private colormap despite the user setting. There
|
|
|
|
// are too many cases when no colors are available and Bochs simply
|
|
|
|
// draws everything in black on black.
|
2017-01-15 14:44:43 +03:00
|
|
|
if (!test_alloc_colors(default_cmap, 16)) {
|
|
|
|
BX_ERROR(("I can't even allocate 16 colors! Switching to a private colormap"));
|
2012-01-20 19:56:17 +04:00
|
|
|
x11_private_colormap = 1;
|
2002-03-20 02:38:08 +03:00
|
|
|
}
|
|
|
|
col_vals[0] = BlackPixel(bx_x_display, bx_x_screen_num);
|
|
|
|
col_vals[15] = WhitePixel(bx_x_display, bx_x_screen_num);
|
|
|
|
for (i = 1; i < MAX_VGA_COLORS; i++) {
|
|
|
|
if (i==15) continue;
|
|
|
|
col_vals[i] = col_vals[0];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-20 19:56:17 +04:00
|
|
|
if (x11_private_colormap) {
|
2001-04-10 05:04:59 +04:00
|
|
|
default_cmap = XCreateColormap(bx_x_display, DefaultRootWindow(bx_x_display),
|
|
|
|
default_visual, AllocNone);
|
|
|
|
if (XAllocColorCells(bx_x_display, default_cmap, False,
|
2002-03-20 02:38:08 +03:00
|
|
|
plane_masks_return, 0, col_vals, MAX_VGA_COLORS) == 0) {
|
2002-12-05 22:19:34 +03:00
|
|
|
BX_PANIC(("XAllocColorCells returns error. Maybe your screen does not support a private colormap?"));
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
win_attr.colormap = default_cmap;
|
|
|
|
XChangeWindowAttributes(bx_x_display, win, CWColormap, &win_attr);
|
|
|
|
|
|
|
|
color.flags = DoRed | DoGreen | DoBlue;
|
|
|
|
|
2002-03-20 02:38:08 +03:00
|
|
|
for (i=0; i < MAX_VGA_COLORS; i++) {
|
2001-04-10 05:04:59 +04:00
|
|
|
color.pixel = i;
|
|
|
|
if (i==15) {
|
|
|
|
color.red = 0xffff;
|
|
|
|
color.green = 0xffff;
|
|
|
|
color.blue = 0xffff;
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
|
|
|
color.red = 0;
|
|
|
|
color.green = 0;
|
|
|
|
color.blue = 0;
|
|
|
|
}
|
2008-02-16 01:05:43 +03:00
|
|
|
XStoreColor(bx_x_display, default_cmap, &color);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
// convenience variables which hold the black & white color indeces
|
|
|
|
black_pixel = col_vals[0];
|
|
|
|
white_pixel = col_vals[15];
|
|
|
|
|
2001-08-31 20:06:32 +04:00
|
|
|
BX_INFO(("font %u wide x %u high, display depth = %d",
|
2017-01-15 14:44:43 +03:00
|
|
|
(unsigned) font_width, (unsigned) font_height, default_depth));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
//select_visual();
|
|
|
|
|
|
|
|
/* Get available icon sizes from Window manager */
|
|
|
|
|
2003-08-11 23:27:57 +04:00
|
|
|
#if BX_HAVE_XPM_H
|
2002-12-30 16:17:39 +03:00
|
|
|
/* Create pixmap from XPM for icon */
|
2011-02-11 02:00:56 +03:00
|
|
|
XCreatePixmapFromData(bx_x_display, win, (char **)icon_bochs_xpm, &icon_pixmap, &icon_mask, NULL);
|
2003-06-13 20:05:03 +04:00
|
|
|
#else
|
|
|
|
/* Create pixmap of depth 1 (bitmap) for icon */
|
|
|
|
icon_pixmap = XCreateBitmapFromData(bx_x_display, win,
|
2011-02-11 02:00:56 +03:00
|
|
|
(char *)bochs_icon_bits, bochs_icon_width, bochs_icon_height);
|
2003-06-13 20:05:03 +04:00
|
|
|
#endif
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* Set size hints for window manager. The window manager may
|
|
|
|
* override these settings. Note that in a real
|
|
|
|
* application if size or position were set by the user
|
|
|
|
* the flags would be UPosition and USize, and these would
|
|
|
|
* override the window manager's preferences for this window. */
|
|
|
|
/* x, y, width, and height hints are now taken from
|
|
|
|
* the actual settings of the window when mapped. Note
|
|
|
|
* that PPosition and PSize must be specified anyway. */
|
|
|
|
|
|
|
|
size_hints.flags = PPosition | PSize | PMinSize | PMaxSize;
|
|
|
|
size_hints.max_width = size_hints.min_width = dimension_x;
|
2004-04-11 14:58:09 +04:00
|
|
|
size_hints.max_height = size_hints.min_height = dimension_y + bx_headerbar_y +
|
|
|
|
bx_statusbar_y;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
{
|
|
|
|
/* format of the window name and icon name
|
|
|
|
* arguments has changed in R4 */
|
|
|
|
XTextProperty windowName, iconName;
|
|
|
|
|
|
|
|
/* These calls store window_name and icon_name into
|
|
|
|
* XTextProperty structures and set their other
|
|
|
|
* fields properly. */
|
2007-10-25 03:09:59 +04:00
|
|
|
if (XStringListToTextProperty((char **)&window_name, 1, &windowName) == 0) {
|
2023-12-23 17:44:00 +03:00
|
|
|
BX_PANIC(("%s: structure allocation for windowName failed.", progname));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2007-10-25 03:09:59 +04:00
|
|
|
if (XStringListToTextProperty((char **)&icon_name, 1, &iconName) == 0) {
|
2023-12-23 17:44:00 +03:00
|
|
|
BX_PANIC(("%s: structure allocation for iconName failed.", progname));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2023-12-23 17:44:00 +03:00
|
|
|
XWMHints wm_hints;
|
2001-04-10 05:04:59 +04:00
|
|
|
wm_hints.initial_state = NormalState;
|
|
|
|
wm_hints.input = True;
|
|
|
|
wm_hints.icon_pixmap = icon_pixmap;
|
2003-08-11 23:27:57 +04:00
|
|
|
#if BX_HAVE_XPM_H
|
2002-12-30 16:17:39 +03:00
|
|
|
wm_hints.icon_mask = icon_mask;
|
|
|
|
wm_hints.flags = StateHint | IconPixmapHint | IconMaskHint | InputHint;
|
2003-06-13 20:05:03 +04:00
|
|
|
#else
|
|
|
|
wm_hints.flags = StateHint | IconPixmapHint | InputHint;
|
|
|
|
#endif
|
2023-12-23 17:44:00 +03:00
|
|
|
|
|
|
|
XClassHint class_hints;
|
2001-04-10 05:04:59 +04:00
|
|
|
class_hints.res_name = progname;
|
2007-10-25 03:09:59 +04:00
|
|
|
class_hints.res_class = (char *)"Bochs";
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
XSetWMProperties(bx_x_display, win, &windowName, &iconName,
|
|
|
|
argv, argc, &size_hints, &wm_hints,
|
|
|
|
&class_hints);
|
2006-04-14 17:27:17 +04:00
|
|
|
XFree(windowName.value);
|
|
|
|
XFree(iconName.value);
|
2006-10-07 14:01:20 +04:00
|
|
|
Atom wm_delete = XInternAtom(bx_x_display, "WM_DELETE_WINDOW", 1);
|
|
|
|
XSetWMProtocols(bx_x_display, win, &wm_delete, 1);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Select event types wanted */
|
|
|
|
XSelectInput(bx_x_display, win, ExposureMask | KeyPressMask | KeyReleaseMask |
|
|
|
|
ButtonPressMask | ButtonReleaseMask | StructureNotifyMask | PointerMotionMask |
|
2014-12-26 23:25:27 +03:00
|
|
|
EnterWindowMask | LeaveWindowMask | FocusChangeMask);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* Create default Graphics Context */
|
|
|
|
gc = XCreateGC(bx_x_display, win, valuemask, &values);
|
|
|
|
gc_inv = XCreateGC(bx_x_display, win, valuemask, &values);
|
|
|
|
gc_headerbar = XCreateGC(bx_x_display, win, valuemask, &values);
|
|
|
|
gc_headerbar_inv = XCreateGC(bx_x_display, win, valuemask, &values);
|
|
|
|
|
|
|
|
XSetState(bx_x_display, gc, white_pixel, black_pixel, GXcopy,AllPlanes);
|
|
|
|
|
2001-12-13 21:36:29 +03:00
|
|
|
XSetState(bx_x_display, gc_inv, black_pixel, white_pixel, GXinvert,AllPlanes);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
XSetState(bx_x_display, gc_headerbar, black_pixel, white_pixel, GXcopy,AllPlanes);
|
|
|
|
|
|
|
|
XSetState(bx_x_display, gc_headerbar_inv, white_pixel, black_pixel, GXcopy,AllPlanes);
|
|
|
|
|
|
|
|
|
|
|
|
/* Display window */
|
|
|
|
XMapWindow(bx_x_display, win);
|
|
|
|
XSync(bx_x_display, /* no discard */ 0);
|
|
|
|
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_DEBUG(("waiting for MapNotify"));
|
2001-04-10 05:04:59 +04:00
|
|
|
while (1) {
|
|
|
|
XNextEvent(bx_x_display, &report);
|
|
|
|
if (report.type == MapNotify) break;
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_DEBUG(("MapNotify found."));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-01-17 00:14:11 +03:00
|
|
|
// Create the VGA font
|
2003-01-17 21:16:02 +03:00
|
|
|
create_internal_vga_font();
|
2003-01-17 00:14:11 +03:00
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
ximage = XCreateImage(bx_x_display, default_visual,
|
|
|
|
default_depth, // depth of image (bitplanes)
|
|
|
|
ZPixmap,
|
|
|
|
0, // offset
|
|
|
|
NULL, // malloc() space after
|
|
|
|
x_tilesize, y_tilesize, // x & y size of image
|
|
|
|
32, // # bits of padding
|
2008-02-16 01:05:43 +03:00
|
|
|
0); // bytes_per_line, let X11 calculate
|
2001-04-10 05:04:59 +04:00
|
|
|
if (!ximage)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("vga: couldn't XCreateImage()"));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
imDepth = default_depth;
|
|
|
|
imWide = ximage->bytes_per_line;
|
|
|
|
imBPP = ximage->bits_per_pixel;
|
|
|
|
|
2023-12-23 17:44:00 +03:00
|
|
|
ximage->data = new char[ximage->bytes_per_line * y_tilesize];
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
if (imBPP < imDepth) {
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("vga_x: bits_per_pixel < depth ?"));
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2004-02-12 22:39:13 +03:00
|
|
|
for (i=0; i<12; i++) bx_statusitem_active[i] = 0;
|
|
|
|
switch (imBPP) {
|
2004-02-15 22:46:13 +03:00
|
|
|
case 16:
|
2020-12-16 15:24:35 +03:00
|
|
|
bx_status_leds[0] = 0x07e0;
|
|
|
|
bx_status_leds[1] = 0xf900;
|
|
|
|
bx_status_leds[2] = 0xffe0;
|
2004-02-15 22:46:13 +03:00
|
|
|
bx_status_graytext = 0x8410;
|
|
|
|
break;
|
2004-02-12 22:39:13 +03:00
|
|
|
case 24:
|
2004-02-15 22:46:13 +03:00
|
|
|
case 32:
|
2020-12-16 15:24:35 +03:00
|
|
|
bx_status_leds[0] = 0x00ff00;
|
|
|
|
bx_status_leds[1] = 0xff4000;
|
|
|
|
bx_status_leds[2] = 0xffff00;
|
2004-02-15 22:46:13 +03:00
|
|
|
bx_status_graytext = 0x808080;
|
|
|
|
break;
|
|
|
|
default:
|
2020-12-16 15:24:35 +03:00
|
|
|
bx_status_leds[0] = 0;
|
|
|
|
bx_status_leds[1] = 0;
|
|
|
|
bx_status_leds[2] = 0;
|
2004-02-15 22:46:13 +03:00
|
|
|
bx_status_graytext = 0;
|
2004-02-12 22:39:13 +03:00
|
|
|
}
|
2010-05-16 13:01:36 +04:00
|
|
|
sprintf(bx_status_info_text, "%s enables mouse", get_toggle_info());
|
2004-02-12 22:39:13 +03:00
|
|
|
|
2011-03-26 11:27:13 +03:00
|
|
|
x_init_done = 1;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
curr_background = 0;
|
|
|
|
XSetBackground(bx_x_display, gc, col_vals[curr_background]);
|
|
|
|
curr_foreground = 1;
|
|
|
|
XSetForeground(bx_x_display, gc, col_vals[curr_foreground]);
|
2008-02-16 01:05:43 +03:00
|
|
|
//XGrabPointer(bx_x_display, win, True, 0, GrabModeAsync, GrabModeAsync,
|
|
|
|
// win, None, CurrentTime);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
XFlush(bx_x_display);
|
2002-03-06 12:31:55 +03:00
|
|
|
|
|
|
|
// loads keymap for x11
|
2012-01-20 19:56:17 +04:00
|
|
|
x11_use_kbd_mapping = SIM->get_param_bool(BXPN_KBD_USEMAPPING)->get();
|
|
|
|
if (x11_use_kbd_mapping) {
|
2002-03-06 12:31:55 +03:00
|
|
|
bx_keymap.loadKeymap(convertStringToXKeysym);
|
2006-02-22 22:18:29 +03:00
|
|
|
}
|
2004-08-15 23:27:15 +04:00
|
|
|
|
2009-01-31 13:04:25 +03:00
|
|
|
#if BX_DEBUGGER && BX_DEBUGGER_GUI
|
2014-12-16 22:48:50 +03:00
|
|
|
// initialize debugger gui
|
2024-03-29 22:12:41 +03:00
|
|
|
if (enh_dbg_gui_enabled) {
|
2014-12-16 22:48:50 +03:00
|
|
|
SIM->set_debug_gui(1);
|
2024-03-29 12:40:59 +03:00
|
|
|
init_debug_dialog(enh_dbg_global_ini);
|
2008-05-12 22:46:46 +04:00
|
|
|
}
|
2014-12-16 22:48:50 +03:00
|
|
|
#endif
|
2008-05-12 22:46:46 +04:00
|
|
|
|
2021-06-06 12:38:47 +03:00
|
|
|
Display *dpy = XOpenDisplay(NULL);
|
|
|
|
if (dpy == NULL) {
|
|
|
|
BX_PANIC(("Cannot connect to X display"));
|
|
|
|
}
|
|
|
|
Window root = RootWindow(dpy, 0);
|
|
|
|
|
2022-07-27 01:26:55 +03:00
|
|
|
#if BX_HAVE_XRANDR_H
|
2021-06-06 12:38:47 +03:00
|
|
|
if (XRRQueryExtension(dpy, &event_base, &error_base)) {
|
|
|
|
int num_sizes;
|
|
|
|
Rotation original_rotation;
|
|
|
|
|
|
|
|
XRRScreenSize *xrrs = XRRSizes(dpy, 0, &num_sizes);
|
|
|
|
XRRScreenConfiguration *conf = XRRGetScreenInfo(dpy, root);
|
|
|
|
SizeID original_size_id = XRRConfigCurrentConfiguration(conf, &original_rotation);
|
|
|
|
x11_max_xres = xrrs[original_size_id].width;
|
|
|
|
x11_max_yres = xrrs[original_size_id].height - bx_headerbar_y - bx_statusbar_y;
|
|
|
|
free(conf);
|
2022-08-23 21:38:00 +03:00
|
|
|
}
|
2022-07-27 01:26:55 +03:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
2021-06-06 12:38:47 +03:00
|
|
|
int screen = DefaultScreen(dpy);
|
|
|
|
x11_max_xres = DisplayWidth(dpy, screen);
|
|
|
|
x11_max_yres = DisplayHeight(dpy, screen) - bx_headerbar_y - bx_statusbar_y;
|
|
|
|
}
|
|
|
|
XCloseDisplay(dpy);
|
|
|
|
BX_INFO(("maximum host resolution: x=%d y=%d", x11_max_xres, x11_max_yres));
|
|
|
|
|
2004-08-15 23:27:15 +04:00
|
|
|
new_gfx_api = 1;
|
2020-11-16 18:39:35 +03:00
|
|
|
new_text_api = 1;
|
2008-05-18 12:52:48 +04:00
|
|
|
dialog_caps |= (BX_GUI_DLG_USER | BX_GUI_DLG_SNAPSHOT | BX_GUI_DLG_CDROM);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
void bx_x_gui_c::handle_events(void)
|
2004-02-12 22:39:13 +03:00
|
|
|
{
|
2017-01-15 15:57:06 +03:00
|
|
|
XEvent report;
|
|
|
|
XKeyEvent *key_event;
|
|
|
|
KeySym keysym;
|
|
|
|
char buffer[MAX_MAPPED_STRING_LENGTH];
|
|
|
|
int bufsize = MAX_MAPPED_STRING_LENGTH;
|
2021-01-30 21:32:52 +03:00
|
|
|
bool mouse_update;
|
2017-01-15 15:57:06 +03:00
|
|
|
int y, height;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
XPointerMovedEvent *pointer_event;
|
|
|
|
XEnterWindowEvent *enter_event;
|
2012-11-02 12:23:09 +04:00
|
|
|
// XLeaveWindowEvent *leave_event;
|
2001-04-10 05:04:59 +04:00
|
|
|
XButtonEvent *button_event;
|
|
|
|
XExposeEvent *expose_event;
|
|
|
|
|
|
|
|
//current_x = -1;
|
|
|
|
//current_y = -1;
|
|
|
|
mouse_update = 0;
|
|
|
|
|
|
|
|
while (XPending(bx_x_display) > 0) {
|
|
|
|
XNextEvent(bx_x_display, &report);
|
2004-12-07 00:12:11 +03:00
|
|
|
current_z = 0;
|
2011-07-31 19:46:37 +04:00
|
|
|
|
2024-03-29 22:12:41 +03:00
|
|
|
if (gui_nokeyrepeat && (report.type == KeyRelease) && X11_KeyRepeat(bx_x_display, &report)) {
|
2011-07-31 19:46:37 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-15 14:44:43 +03:00
|
|
|
switch (report.type) {
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
case Expose:
|
2003-04-28 16:28:08 +04:00
|
|
|
expose_event = &report.xexpose;
|
|
|
|
/* Adjust y, and reduce height if it overlaps headerbar. */
|
|
|
|
y = expose_event->y - BX_HEADER_BAR_Y;
|
|
|
|
height = expose_event->height;
|
|
|
|
if (y < 0) {
|
2008-02-16 01:05:43 +03:00
|
|
|
height += y;
|
|
|
|
y = 0;
|
2003-04-28 16:28:08 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2017-01-02 22:11:23 +03:00
|
|
|
if (console_running()) {
|
2017-01-01 20:45:06 +03:00
|
|
|
console_refresh(1);
|
2017-01-02 22:11:23 +03:00
|
|
|
} else {
|
2017-01-01 20:45:06 +03:00
|
|
|
DEV_vga_redraw_area((unsigned) expose_event->x, y,
|
|
|
|
(unsigned) expose_event->width, height);
|
|
|
|
}
|
2003-04-28 16:28:08 +04:00
|
|
|
/* Always draw headerbar, even if not touched by expose event.
|
|
|
|
* As a small optimization, only do it on last contigous expose.
|
|
|
|
*/
|
|
|
|
if (expose_event->count == 0) {
|
2017-01-01 20:45:06 +03:00
|
|
|
show_headerbar();
|
2003-04-28 16:28:08 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ConfigureNotify:
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_DEBUG(("ConfigureNotify Xevent"));
|
2003-04-28 16:28:08 +04:00
|
|
|
/* FIXME: It's not clear why we have to show the headerbar here.
|
|
|
|
* This should be forced by the following expose events.
|
|
|
|
*/
|
2001-04-10 05:04:59 +04:00
|
|
|
show_headerbar();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ButtonPress:
|
|
|
|
button_event = (XButtonEvent *) &report;
|
2023-12-23 17:44:00 +03:00
|
|
|
// BX_DEBUG(("xxx: buttonpress"));
|
2001-04-10 05:04:59 +04:00
|
|
|
if (button_event->y < BX_HEADER_BAR_Y) {
|
2023-12-23 17:44:00 +03:00
|
|
|
// BX_DEBUG(("xxx: in headerbar"));
|
2001-04-10 05:04:59 +04:00
|
|
|
if (mouse_update) {
|
2023-12-23 17:44:00 +03:00
|
|
|
// BX_DEBUG(("xxx: mouse_update=1"));
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2001-04-10 05:04:59 +04:00
|
|
|
mouse_update = 0;
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
prev_x = current_x = -1;
|
|
|
|
prev_y = current_y = -1;
|
2017-01-01 20:45:06 +03:00
|
|
|
headerbar_click(button_event->x);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
current_x = button_event->x;
|
|
|
|
current_y = button_event->y;
|
|
|
|
mouse_update = 1;
|
2023-12-23 17:44:00 +03:00
|
|
|
// BX_DEBUG(("xxx: x,y=(%d,%d)", current_x, current_y));
|
2001-04-10 05:04:59 +04:00
|
|
|
switch (button_event->button) {
|
|
|
|
case Button1:
|
|
|
|
mouse_button_state |= 0x01;
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2001-04-10 05:04:59 +04:00
|
|
|
mouse_update = 0;
|
|
|
|
break;
|
|
|
|
case Button2:
|
2010-05-16 13:01:36 +04:00
|
|
|
if (mouse_toggle_check(BX_MT_MBUTTON, 1)) {
|
2004-02-15 22:46:13 +03:00
|
|
|
toggle_mouse_enable();
|
|
|
|
} else {
|
|
|
|
mouse_button_state |= 0x04;
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2004-02-15 22:46:13 +03:00
|
|
|
mouse_update = 0;
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
case Button3:
|
|
|
|
mouse_button_state |= 0x02;
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2001-04-10 05:04:59 +04:00
|
|
|
mouse_update = 0;
|
|
|
|
break;
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ButtonRelease:
|
|
|
|
button_event = (XButtonEvent *) &report;
|
|
|
|
if (button_event->y < BX_HEADER_BAR_Y) {
|
|
|
|
if (mouse_update) {
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2001-04-10 05:04:59 +04:00
|
|
|
mouse_update = 0;
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
prev_x = current_x = -1;
|
|
|
|
prev_y = current_y = -1;
|
|
|
|
// ignore, in headerbar area
|
|
|
|
break;
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
current_x = button_event->x;
|
|
|
|
current_y = button_event->y;
|
|
|
|
mouse_update = 1;
|
|
|
|
switch (button_event->button) {
|
|
|
|
case Button1:
|
|
|
|
mouse_button_state &= ~0x01;
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2001-04-10 05:04:59 +04:00
|
|
|
mouse_update = 0;
|
|
|
|
break;
|
|
|
|
case Button2:
|
2010-05-16 13:01:36 +04:00
|
|
|
mouse_toggle_check(BX_MT_MBUTTON, 0);
|
2004-02-15 22:46:13 +03:00
|
|
|
mouse_button_state &= ~0x04;
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2004-02-15 22:46:13 +03:00
|
|
|
mouse_update = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
case Button3:
|
|
|
|
mouse_button_state &= ~0x02;
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2001-04-10 05:04:59 +04:00
|
|
|
mouse_update = 0;
|
|
|
|
break;
|
2004-12-07 00:12:11 +03:00
|
|
|
case Button4:
|
|
|
|
current_z = 1;
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2004-12-07 00:12:11 +03:00
|
|
|
mouse_update = 0;
|
|
|
|
break;
|
|
|
|
case Button5:
|
|
|
|
current_z = -1;
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2004-12-07 00:12:11 +03:00
|
|
|
mouse_update = 0;
|
|
|
|
break;
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case KeyPress:
|
|
|
|
key_event = (XKeyEvent *) &report;
|
2012-11-02 12:23:09 +04:00
|
|
|
XLookupString(key_event, buffer, bufsize, &keysym, NULL);
|
2001-06-26 02:57:27 +04:00
|
|
|
xkeypress(keysym, 0);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case KeyRelease:
|
|
|
|
key_event = (XKeyEvent *) &report;
|
2012-11-02 12:23:09 +04:00
|
|
|
XLookupString(key_event, buffer, bufsize, &keysym, NULL);
|
2001-06-26 02:57:27 +04:00
|
|
|
xkeypress(keysym, 1);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MotionNotify:
|
|
|
|
pointer_event = (XPointerMovedEvent *) &report;
|
|
|
|
current_x = pointer_event->x;
|
|
|
|
current_y = pointer_event->y;
|
|
|
|
mouse_update = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EnterNotify:
|
|
|
|
enter_event = (XEnterWindowEvent *) &report;
|
|
|
|
prev_x = current_x = enter_event->x;
|
|
|
|
prev_y = current_y = enter_event->y;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LeaveNotify:
|
2023-12-23 17:44:00 +03:00
|
|
|
// leave_event = (XLeaveWindowEvent *) &report;
|
2001-04-10 05:04:59 +04:00
|
|
|
prev_x = current_x = -1;
|
|
|
|
prev_y = current_y = -1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MapNotify:
|
|
|
|
/* screen needs redraw, since X would have tossed previous
|
|
|
|
* requests before window mapped
|
|
|
|
*/
|
|
|
|
//retval = 1;
|
|
|
|
break;
|
|
|
|
|
2014-12-27 18:57:30 +03:00
|
|
|
case FocusOut:
|
2014-12-26 23:25:27 +03:00
|
|
|
DEV_kbd_release_keys();
|
|
|
|
break;
|
|
|
|
|
2006-10-07 14:01:20 +04:00
|
|
|
case ClientMessage:
|
|
|
|
if (!strcmp(XGetAtomName(bx_x_display, report.xclient.message_type), "WM_PROTOCOLS")) {
|
|
|
|
bx_stop_simulation();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
default:
|
2004-02-15 22:46:13 +03:00
|
|
|
// (mch) Ignore...
|
|
|
|
BX_DEBUG(("XXX: default Xevent type"));
|
2001-04-10 05:04:59 +04:00
|
|
|
/* all events selected by StructureNotifyMask are thrown away here,
|
|
|
|
* since nothing is done with them */
|
|
|
|
break;
|
|
|
|
} /* end switch */
|
|
|
|
} /* end while */
|
|
|
|
|
|
|
|
if (mouse_update) {
|
2004-02-15 22:46:13 +03:00
|
|
|
BX_DEBUG(("handle_events(): send mouse status"));
|
2017-01-01 20:45:06 +03:00
|
|
|
send_mouse_status();
|
2006-01-23 21:34:47 +03:00
|
|
|
}
|
|
|
|
#if BX_SHOW_IPS
|
2020-07-15 19:26:46 +03:00
|
|
|
if (x11_ips_update && !x11_show_info_msg) {
|
2006-01-23 21:34:47 +03:00
|
|
|
x11_ips_update = 0;
|
2017-01-15 15:57:06 +03:00
|
|
|
x11_set_status_text(0, x11_ips_text, 1);
|
2006-01-23 21:34:47 +03:00
|
|
|
}
|
|
|
|
#endif
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2021-01-30 21:32:52 +03:00
|
|
|
void bx_x_gui_c::set_font(bool lg)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2020-11-16 18:39:35 +03:00
|
|
|
unsigned i, j, k;
|
|
|
|
Bit8u fbits, fmask, frow;
|
2017-01-15 15:57:06 +03:00
|
|
|
unsigned char cell[96];
|
2021-01-30 21:32:52 +03:00
|
|
|
bool gfxchar, dwidth;
|
2020-11-16 18:39:35 +03:00
|
|
|
|
2024-01-01 13:20:05 +03:00
|
|
|
BX_DEBUG(("charmap update. Font is %d x %d", font_width, font_height));
|
2023-12-29 19:47:44 +03:00
|
|
|
for (unsigned m = 0; m < 2; m++) {
|
|
|
|
for (unsigned c = 0; c < 256; c++) {
|
|
|
|
if (char_changed[m][c]) {
|
|
|
|
XFreePixmap(bx_x_display, vgafont[m][c]);
|
|
|
|
gfxchar = lg && ((c & 0xE0) == 0xC0);
|
|
|
|
dwidth = font_width > 9;
|
|
|
|
i = 0;
|
|
|
|
j = 0;
|
|
|
|
memset(cell, 0, sizeof(cell));
|
|
|
|
if (dwidth) {
|
|
|
|
do {
|
|
|
|
frow = vga_charmap[m][(c<<5)+j];
|
|
|
|
fmask = 0x80;
|
|
|
|
fbits = 0x03;
|
|
|
|
for (k=0; k<8; k++) {
|
|
|
|
if (frow & fmask) cell[i] |= fbits;
|
|
|
|
fmask >>= 1;
|
|
|
|
fbits <<= 2;
|
|
|
|
if (k == 3) {
|
|
|
|
i++;
|
|
|
|
fbits = 0x03;
|
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
}
|
2023-12-29 19:47:44 +03:00
|
|
|
if (gfxchar) {
|
|
|
|
if (frow & 0x01) cell[i+1] = 0x03;
|
|
|
|
}
|
|
|
|
i += 2;
|
|
|
|
} while (++j < font_height);
|
|
|
|
vgafont[m][c] = XCreateBitmapFromData(bx_x_display, win,
|
|
|
|
(const char*)cell, 18, font_height);
|
|
|
|
} else {
|
|
|
|
do {
|
|
|
|
frow = vga_charmap[m][(c<<5)+j];
|
|
|
|
fmask = 0x80;
|
|
|
|
fbits = 0x01;
|
|
|
|
for (k=0; k<8; k++) {
|
|
|
|
if (frow & fmask) cell[i] |= fbits;
|
|
|
|
fmask >>= 1;
|
|
|
|
fbits <<= 1;
|
|
|
|
}
|
|
|
|
if (gfxchar) {
|
|
|
|
if (frow & 0x01) cell[i+1] = 0x01;
|
|
|
|
}
|
|
|
|
i += 2;
|
|
|
|
} while (++j < font_height);
|
|
|
|
vgafont[m][c] = XCreateBitmapFromData(bx_x_display, win,
|
|
|
|
(const char*)cell, 9, font_height);
|
|
|
|
}
|
|
|
|
if(vgafont[m][c] == None)
|
|
|
|
BX_PANIC(("Can't create vga font [%d]", c));
|
|
|
|
char_changed[m][c] = 0;
|
2017-01-15 15:57:06 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
}
|
2020-11-16 18:39:35 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2020-11-16 18:39:35 +03:00
|
|
|
void bx_x_gui_c::draw_char(Bit8u ch, Bit8u fc, Bit8u bc, Bit16u xc, Bit16u yc,
|
|
|
|
Bit8u fw, Bit8u fh, Bit8u fx, Bit8u fy,
|
2023-12-29 19:47:44 +03:00
|
|
|
bool gfxcharw9, Bit8u cs, Bit8u ce, bool curs, bool font2)
|
2020-11-16 18:39:35 +03:00
|
|
|
{
|
|
|
|
yc += bx_headerbar_y;
|
|
|
|
XSetForeground(bx_x_display, gc, col_vals[fc]);
|
|
|
|
XSetBackground(bx_x_display, gc, col_vals[bc]);
|
2023-12-29 19:47:44 +03:00
|
|
|
Bit8u m = (font2) ? 1 : 0;
|
|
|
|
XCopyPlane(bx_x_display, vgafont[m][ch], win, gc, fx, fy, fw, fh, xc, yc, 1);
|
2020-11-16 18:39:35 +03:00
|
|
|
if (curs && (ce >= fy) && (cs < (fh + fy))) {
|
|
|
|
if (cs > fy) {
|
|
|
|
yc += (cs - fy);
|
|
|
|
fh -= (cs - fy);
|
2003-06-11 22:44:45 +04:00
|
|
|
}
|
2020-11-16 18:39:35 +03:00
|
|
|
if ((ce - cs + 1) < fh) {
|
|
|
|
fh = ce - cs + 1;
|
2004-04-11 14:58:09 +04:00
|
|
|
}
|
2020-11-16 18:39:35 +03:00
|
|
|
XSetForeground(bx_x_display, gc, col_vals[bc]);
|
|
|
|
XSetBackground(bx_x_display, gc, col_vals[fc]);
|
2023-12-29 19:47:44 +03:00
|
|
|
XCopyPlane(bx_x_display, vgafont[m][ch], win, gc, fx, cs, fw, fh, xc, yc, 1);
|
2020-11-16 18:39:35 +03:00
|
|
|
}
|
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
|
2020-11-16 18:39:35 +03:00
|
|
|
void bx_x_gui_c::text_update(Bit8u *old_text, Bit8u *new_text,
|
|
|
|
unsigned long cursor_x, unsigned long cursor_y,
|
|
|
|
bx_vga_tminfo_t *tm_info)
|
|
|
|
{
|
|
|
|
// present for compatibility
|
2017-01-15 15:57:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void bx_x_gui_c::graphics_tile_update(Bit8u *tile, unsigned x0, unsigned y0)
|
|
|
|
{
|
|
|
|
unsigned x, y, y_size;
|
|
|
|
unsigned color, offset;
|
|
|
|
Bit8u b0, b1, b2, b3;
|
|
|
|
|
|
|
|
if ((y0 + y_tilesize) > dimension_y) {
|
|
|
|
y_size = dimension_y - y0;
|
|
|
|
} else {
|
|
|
|
y_size = y_tilesize;
|
|
|
|
}
|
|
|
|
switch (guest_bpp) {
|
|
|
|
case 8: // 8 bits per pixel
|
|
|
|
for (y=0; y<y_size; y++) {
|
|
|
|
for (x=0; x<x_tilesize; x++) {
|
|
|
|
color = col_vals[tile[y*x_tilesize + x]];
|
|
|
|
switch (imBPP) {
|
|
|
|
case 8: // 8 bits per pixel
|
|
|
|
ximage->data[imWide*y + x] = color;
|
|
|
|
break;
|
|
|
|
case 16: // 16 bits per pixel
|
|
|
|
offset = imWide*y + 2*x;
|
|
|
|
b0 = color >> 0;
|
|
|
|
b1 = color >> 8;
|
|
|
|
if (ximage->byte_order == LSBFirst) {
|
|
|
|
ximage->data[offset + 0] = b0;
|
|
|
|
ximage->data[offset + 1] = b1;
|
|
|
|
}
|
|
|
|
else { // MSBFirst
|
|
|
|
ximage->data[offset + 0] = b1;
|
|
|
|
ximage->data[offset + 1] = b0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 24: // 24 bits per pixel
|
|
|
|
offset = imWide*y + 3*x;
|
|
|
|
b0 = color >> 0;
|
|
|
|
b1 = color >> 8;
|
|
|
|
b2 = color >> 16;
|
|
|
|
if (ximage->byte_order == LSBFirst) {
|
|
|
|
ximage->data[offset + 0] = b0;
|
|
|
|
ximage->data[offset + 1] = b1;
|
|
|
|
ximage->data[offset + 2] = b2;
|
|
|
|
}
|
|
|
|
else { // MSBFirst
|
|
|
|
ximage->data[offset + 0] = b2;
|
|
|
|
ximage->data[offset + 1] = b1;
|
|
|
|
ximage->data[offset + 2] = b0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 32: // 32 bits per pixel
|
|
|
|
offset = imWide*y + 4*x;
|
|
|
|
b0 = color >> 0;
|
|
|
|
b1 = color >> 8;
|
|
|
|
b2 = color >> 16;
|
|
|
|
b3 = color >> 24;
|
|
|
|
if (ximage->byte_order == LSBFirst) {
|
|
|
|
ximage->data[offset + 0] = b0;
|
|
|
|
ximage->data[offset + 1] = b1;
|
|
|
|
ximage->data[offset + 2] = b2;
|
|
|
|
ximage->data[offset + 3] = b3;
|
|
|
|
}
|
|
|
|
else { // MSBFirst
|
|
|
|
ximage->data[offset + 0] = b3;
|
|
|
|
ximage->data[offset + 1] = b2;
|
|
|
|
ximage->data[offset + 2] = b1;
|
|
|
|
ximage->data[offset + 3] = b0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BX_PANIC(("X_graphics_tile_update: bits_per_pixel %u not implemented",
|
|
|
|
(unsigned) imBPP));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BX_PANIC(("X_graphics_tile_update: bits_per_pixel %u handled by new graphics API",
|
|
|
|
(unsigned) guest_bpp));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
XPutImage(bx_x_display, win, gc, ximage, 0, 0, x0, y0+bx_headerbar_y,
|
|
|
|
x_tilesize, y_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void bx_x_gui_c::flush(void)
|
|
|
|
{
|
|
|
|
if (bx_x_display)
|
|
|
|
XFlush(bx_x_display);
|
|
|
|
}
|
|
|
|
|
|
|
|
void bx_x_gui_c::clear_screen(void)
|
|
|
|
{
|
|
|
|
XClearArea(bx_x_display, win, 0, bx_headerbar_y, dimension_x, dimension_y, 0);
|
|
|
|
}
|
|
|
|
|
2021-01-30 21:32:52 +03:00
|
|
|
bool bx_x_gui_c::palette_change(Bit8u index, Bit8u red, Bit8u green, Bit8u blue)
|
2017-01-15 15:57:06 +03:00
|
|
|
{
|
|
|
|
// returns: 0=no screen update needed (color map change has direct effect)
|
|
|
|
// 1=screen updated needed (redraw using current colormap)
|
|
|
|
XColor color;
|
|
|
|
|
|
|
|
color.flags = DoRed | DoGreen | DoBlue;
|
|
|
|
color.red = red << 8;
|
|
|
|
color.green = green << 8;
|
|
|
|
color.blue = blue << 8;
|
|
|
|
|
|
|
|
if (x11_private_colormap) {
|
|
|
|
color.pixel = index;
|
|
|
|
XStoreColor(bx_x_display, default_cmap, &color);
|
|
|
|
return(0); // no screen update needed
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
XAllocColor(bx_x_display, DefaultColormap(bx_x_display, bx_x_screen_num),
|
|
|
|
&color);
|
|
|
|
col_vals[index] = color.pixel;
|
|
|
|
return(1); // screen update needed
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void bx_x_gui_c::dimension_update(unsigned x, unsigned y, unsigned fheight, unsigned fwidth, unsigned bpp)
|
|
|
|
{
|
|
|
|
if ((bpp == 8) || (bpp == 15) || (bpp == 16) || (bpp == 24) || (bpp == 32)) {
|
|
|
|
guest_bpp = bpp;
|
|
|
|
} else {
|
|
|
|
BX_PANIC(("%d bpp graphics mode not supported", bpp));
|
|
|
|
}
|
|
|
|
guest_textmode = (fheight > 0);
|
2020-11-22 12:02:38 +03:00
|
|
|
guest_fwidth = fwidth;
|
|
|
|
guest_fheight = fheight;
|
2017-01-15 15:57:06 +03:00
|
|
|
guest_xres = x;
|
|
|
|
guest_yres = y;
|
|
|
|
if (guest_textmode) {
|
|
|
|
font_height = fheight;
|
|
|
|
if (fwidth != font_width) {
|
|
|
|
font_width = fwidth;
|
|
|
|
charmap_updated = 1;
|
2023-12-29 19:47:44 +03:00
|
|
|
for (int i = 0; i < 256; i++) char_changed[0][i] = 1;
|
|
|
|
for (int i = 0; i < 256; i++) char_changed[1][i] = 1;
|
2017-01-15 15:57:06 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((x != dimension_x) || (y != dimension_y)) {
|
|
|
|
XSizeHints hints;
|
|
|
|
long supplied_return;
|
|
|
|
|
|
|
|
if (XGetWMNormalHints(bx_x_display, win, &hints, &supplied_return) &&
|
|
|
|
supplied_return & PMaxSize) {
|
|
|
|
hints.max_width = hints.min_width = x;
|
|
|
|
hints.max_height = hints.min_height = y+bx_headerbar_y+bx_statusbar_y;
|
|
|
|
XSetWMNormalHints(bx_x_display, win, &hints);
|
|
|
|
}
|
|
|
|
XResizeWindow(bx_x_display, win, x, y+bx_headerbar_y+bx_statusbar_y);
|
|
|
|
dimension_x = x;
|
|
|
|
dimension_y = y;
|
|
|
|
warp_home_x = x / 2;
|
|
|
|
warp_home_y = y / 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned bx_x_gui_c::create_bitmap(const unsigned char *bmap, unsigned xdim, unsigned ydim)
|
|
|
|
{
|
|
|
|
if (bx_bitmap_entries >= BX_MAX_PIXMAPS) {
|
|
|
|
BX_PANIC(("x: too many pixmaps, increase BX_MAX_PIXMAPS"));
|
|
|
|
}
|
|
|
|
|
|
|
|
bx_bitmaps[bx_bitmap_entries].bmap =
|
|
|
|
XCreateBitmapFromData(bx_x_display, win, (const char *) bmap, xdim, ydim);
|
|
|
|
bx_bitmaps[bx_bitmap_entries].xdim = xdim;
|
|
|
|
bx_bitmaps[bx_bitmap_entries].ydim = ydim;
|
|
|
|
if (!bx_bitmaps[bx_bitmap_entries].bmap) {
|
|
|
|
BX_PANIC(("x: could not create bitmap"));
|
|
|
|
}
|
|
|
|
bx_bitmap_entries++;
|
|
|
|
return (bx_bitmap_entries-1); // return index as handle
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned bx_x_gui_c::headerbar_bitmap(unsigned bmap_id, unsigned alignment, void (*f)(void))
|
|
|
|
{
|
|
|
|
if ((bx_headerbar_entries+1) > BX_MAX_HEADERBAR_ENTRIES)
|
|
|
|
BX_PANIC(("too many headerbar entries, increase BX_MAX_HEADERBAR_ENTRIES"));
|
|
|
|
|
2023-12-23 17:44:00 +03:00
|
|
|
unsigned hb_index = bx_headerbar_entries++;
|
2017-01-15 15:57:06 +03:00
|
|
|
|
|
|
|
bx_headerbar_entry[hb_index].bmap_id = bmap_id;
|
|
|
|
bx_headerbar_entry[hb_index].xdim = bx_bitmaps[bmap_id].xdim;
|
|
|
|
bx_headerbar_entry[hb_index].ydim = bx_bitmaps[bmap_id].ydim;
|
|
|
|
bx_headerbar_entry[hb_index].alignment = alignment;
|
|
|
|
bx_headerbar_entry[hb_index].f = f;
|
|
|
|
if (alignment == BX_GRAVITY_LEFT) {
|
|
|
|
bx_headerbar_entry[hb_index].xorigin = bx_bitmap_left_xorigin;
|
|
|
|
bx_bitmap_left_xorigin += bx_bitmaps[bmap_id].xdim;
|
|
|
|
} else { // BX_GRAVITY_RIGHT
|
|
|
|
bx_bitmap_right_xorigin += bx_bitmaps[bmap_id].xdim;
|
|
|
|
bx_headerbar_entry[hb_index].xorigin = bx_bitmap_right_xorigin;
|
|
|
|
}
|
|
|
|
return hb_index;
|
|
|
|
}
|
|
|
|
|
|
|
|
void bx_x_gui_c::replace_bitmap(unsigned hbar_id, unsigned bmap_id)
|
|
|
|
{
|
|
|
|
unsigned xorigin;
|
|
|
|
|
|
|
|
bx_headerbar_entry[hbar_id].bmap_id = bmap_id;
|
|
|
|
|
|
|
|
if (bx_headerbar_entry[hbar_id].alignment == BX_GRAVITY_LEFT)
|
|
|
|
xorigin = bx_headerbar_entry[hbar_id].xorigin;
|
|
|
|
else
|
|
|
|
xorigin = dimension_x - bx_headerbar_entry[hbar_id].xorigin;
|
|
|
|
XCopyPlane(bx_x_display, bx_bitmaps[bmap_id].bmap, win, gc_headerbar, 0, 0,
|
|
|
|
bx_headerbar_entry[hbar_id].xdim, bx_headerbar_entry[hbar_id].ydim,
|
|
|
|
xorigin, 0, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void bx_x_gui_c::show_headerbar(void)
|
|
|
|
{
|
|
|
|
unsigned xorigin;
|
|
|
|
int xleft, xright, sb_ypos;
|
|
|
|
|
|
|
|
sb_ypos = dimension_y + bx_headerbar_y;
|
|
|
|
// clear header bar and status bar area to white
|
|
|
|
XFillRectangle(bx_x_display, win, gc_headerbar_inv, 0,0, dimension_x, bx_headerbar_y);
|
|
|
|
XFillRectangle(bx_x_display, win, gc_headerbar_inv, 0,sb_ypos, dimension_x, bx_statusbar_y);
|
|
|
|
|
|
|
|
xleft = 0;
|
|
|
|
xright = dimension_x;
|
|
|
|
for (unsigned i=0; i<bx_headerbar_entries; i++) {
|
|
|
|
if (bx_headerbar_entry[i].alignment == BX_GRAVITY_LEFT) {
|
|
|
|
xorigin = bx_headerbar_entry[i].xorigin;
|
|
|
|
xleft += bx_headerbar_entry[i].xdim;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
xorigin = dimension_x - bx_headerbar_entry[i].xorigin;
|
|
|
|
xright = xorigin;
|
|
|
|
}
|
|
|
|
if (xright < xleft) break;
|
|
|
|
Pixmap bitmap = bx_bitmaps[bx_headerbar_entry[i].bmap_id].bmap;
|
|
|
|
XCopyPlane(bx_x_display, bitmap, win, gc_headerbar, 0, 0,
|
|
|
|
bx_headerbar_entry[i].xdim, bx_headerbar_entry[i].ydim,
|
|
|
|
xorigin, 0, 1);
|
|
|
|
}
|
|
|
|
for (unsigned i=0; i<12; i++) {
|
|
|
|
xleft = bx_statusitem_pos[i];
|
|
|
|
if (i > 0) {
|
|
|
|
XDrawLine(bx_x_display, win, gc_inv, xleft, sb_ypos+1, xleft,
|
|
|
|
sb_ypos+bx_statusbar_y);
|
|
|
|
if (i <= statusitem_count) {
|
|
|
|
x11_set_status_text(i, statusitem[i-1].text, bx_statusitem_active[i]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
x11_set_status_text(0, bx_status_info_text, 0);
|
|
|
|
}
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2006-10-13 21:55:53 +04:00
|
|
|
int bx_x_gui_c::get_clipboard_text(Bit8u **bytes, Bit32s *nbytes)
|
2002-03-16 14:30:06 +03:00
|
|
|
{
|
2002-09-05 19:57:37 +04:00
|
|
|
int len;
|
2017-01-15 14:44:43 +03:00
|
|
|
Bit8u *tmp = (Bit8u *)XFetchBytes(bx_x_display, &len);
|
2002-09-05 19:57:37 +04:00
|
|
|
// according to man XFetchBytes, tmp must be freed by XFree(). So allocate
|
|
|
|
// a new buffer with "new". The keyboard code will free it with delete []
|
|
|
|
// when the paste is done.
|
|
|
|
Bit8u *buf = new Bit8u[len];
|
2017-01-15 14:44:43 +03:00
|
|
|
memcpy(buf, tmp, len);
|
2002-09-05 19:57:37 +04:00
|
|
|
*bytes = buf;
|
|
|
|
*nbytes = len;
|
2017-01-15 14:44:43 +03:00
|
|
|
XFree(tmp);
|
2002-03-16 14:30:06 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2006-10-13 21:55:53 +04:00
|
|
|
int bx_x_gui_c::set_clipboard_text(char *text_snapshot, Bit32u len)
|
2002-03-15 19:45:10 +03:00
|
|
|
{
|
|
|
|
// this writes data to the clipboard.
|
2017-01-15 14:44:43 +03:00
|
|
|
BX_INFO(("storing %d bytes to X windows clipboard", len));
|
2002-03-19 21:43:22 +03:00
|
|
|
XSetSelectionOwner(bx_x_display, XA_PRIMARY, None, CurrentTime);
|
2017-01-15 14:44:43 +03:00
|
|
|
XStoreBytes(bx_x_display, (char *)text_snapshot, len);
|
2002-03-16 14:30:06 +03:00
|
|
|
return 1;
|
2002-03-15 19:45:10 +03:00
|
|
|
}
|
|
|
|
|
2021-01-30 21:32:52 +03:00
|
|
|
void bx_x_gui_c::mouse_enabled_changed_specific(bool val)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2017-01-15 15:57:06 +03:00
|
|
|
if (val != mouse_captured) {
|
|
|
|
BX_INFO(("Mouse capture %s", val ? "on":"off"));
|
|
|
|
sprintf(bx_status_info_text, "%s %sables mouse", get_toggle_info(), val ? "dis":"en");
|
|
|
|
x11_set_status_text(0, bx_status_info_text, 0);
|
|
|
|
}
|
|
|
|
mouse_captured = val;
|
|
|
|
if (val) {
|
|
|
|
mouse_enable_x = current_x;
|
|
|
|
mouse_enable_y = current_y;
|
|
|
|
disable_cursor();
|
|
|
|
// Move the cursor to a 'safe' place
|
|
|
|
warp_cursor(warp_home_x-current_x, warp_home_y-current_y);
|
2004-02-27 21:42:53 +03:00
|
|
|
} else {
|
2017-01-15 15:57:06 +03:00
|
|
|
enable_cursor();
|
|
|
|
warp_cursor(mouse_enable_x-current_x, mouse_enable_y-current_y);
|
2004-02-27 21:42:53 +03:00
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void bx_x_gui_c::exit(void)
|
|
|
|
{
|
|
|
|
if (!x_init_done) return;
|
|
|
|
|
|
|
|
// Delete the font bitmaps
|
|
|
|
for (int i=0; i<256; i++) {
|
|
|
|
//if (vgafont[i] != NULL)
|
2023-12-29 19:47:44 +03:00
|
|
|
XFreePixmap(bx_x_display,vgafont[0][i]);
|
|
|
|
XFreePixmap(bx_x_display,vgafont[1][i]);
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
|
|
|
|
if (mouse_captured)
|
|
|
|
enable_cursor();
|
|
|
|
|
|
|
|
#if BX_DEBUGGER && BX_DEBUGGER_GUI
|
|
|
|
if (SIM->has_debug_gui()) {
|
|
|
|
close_debug_dialog();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (bx_x_display)
|
|
|
|
XCloseDisplay(bx_x_display);
|
|
|
|
|
|
|
|
BX_INFO(("Exit"));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
// X11 implementation of the new graphics API (see gui.cc / gui.h)
|
|
|
|
|
2008-02-16 01:05:43 +03:00
|
|
|
bx_svga_tileinfo_t *bx_x_gui_c::graphics_tile_info(bx_svga_tileinfo_t *info)
|
2004-08-15 23:27:15 +04:00
|
|
|
{
|
|
|
|
info->bpp = ximage->bits_per_pixel;
|
|
|
|
info->pitch = ximage->bytes_per_line;
|
|
|
|
info->red_shift = 0;
|
|
|
|
info->green_shift = 0;
|
|
|
|
info->blue_shift = 0;
|
|
|
|
info->red_mask = ximage->red_mask;
|
|
|
|
info->green_mask = ximage->green_mask;
|
|
|
|
info->blue_mask = ximage->blue_mask;
|
|
|
|
|
|
|
|
int i, rf, gf, bf;
|
|
|
|
unsigned long red, green, blue;
|
|
|
|
|
|
|
|
i = rf = gf = bf = 0;
|
|
|
|
red = ximage->red_mask;
|
|
|
|
green = ximage->green_mask;
|
|
|
|
blue = ximage->blue_mask;
|
|
|
|
|
|
|
|
while (red || rf || green || gf || blue || bf) {
|
|
|
|
if (rf) {
|
|
|
|
if (!(red & 1)) {
|
|
|
|
info->red_shift = i;
|
|
|
|
rf = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (red & 1) {
|
|
|
|
rf = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gf) {
|
|
|
|
if (!(green & 1)) {
|
|
|
|
info->green_shift = i;
|
|
|
|
gf = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (green & 1) {
|
|
|
|
gf = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bf) {
|
|
|
|
if (!(blue & 1)) {
|
|
|
|
info->blue_shift = i;
|
|
|
|
bf = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (blue & 1) {
|
|
|
|
bf = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
red >>= 1;
|
|
|
|
green >>= 1;
|
|
|
|
blue >>= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
info->is_indexed = (default_visual->c_class != TrueColor) &&
|
|
|
|
(default_visual->c_class != DirectColor);
|
|
|
|
info->is_little_endian = (ximage->byte_order == LSBFirst);
|
|
|
|
|
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
2008-02-16 01:05:43 +03:00
|
|
|
Bit8u *bx_x_gui_c::graphics_tile_get(unsigned x0, unsigned y0,
|
2004-08-15 23:27:15 +04:00
|
|
|
unsigned *w, unsigned *h)
|
|
|
|
{
|
|
|
|
if (x0+x_tilesize > dimension_x) {
|
|
|
|
*w = dimension_x - x0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*w = x_tilesize;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (y0+y_tilesize > dimension_y) {
|
|
|
|
*h = dimension_y - y0;
|
|
|
|
}
|
|
|
|
else {
|
2017-01-15 15:57:06 +03:00
|
|
|
*h = y_tilesize;
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
|
|
|
|
return (Bit8u *)ximage->data + ximage->xoffset*ximage->bits_per_pixel/8;
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
void bx_x_gui_c::graphics_tile_update_in_place(unsigned x0, unsigned y0,
|
|
|
|
unsigned w, unsigned h)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2017-01-15 15:57:06 +03:00
|
|
|
XPutImage(bx_x_display, win, gc, ximage, 0, 0,
|
|
|
|
x0, y0+bx_headerbar_y, w, h);
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
// X11 implementation of optional bx_gui_c methods (see gui.h)
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
void bx_x_gui_c::beep_on(float frequency)
|
|
|
|
{
|
|
|
|
BX_INFO(("X11 Beep ON (frequency=%.2f)",frequency));
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
void bx_x_gui_c::beep_off()
|
|
|
|
{
|
|
|
|
BX_INFO(("X11 Beep OFF"));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2019-04-21 23:08:32 +03:00
|
|
|
#if BX_HAVE_XRANDR_H
|
2017-01-15 15:57:06 +03:00
|
|
|
void bx_x_gui_c::get_capabilities(Bit16u *xres, Bit16u *yres, Bit16u *bpp)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2021-06-06 12:38:47 +03:00
|
|
|
*xres = x11_max_xres;
|
|
|
|
*yres = x11_max_yres;
|
2017-01-15 15:57:06 +03:00
|
|
|
// always return 32 bit depth
|
|
|
|
*bpp = 32;
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2019-04-21 23:08:32 +03:00
|
|
|
#endif
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
void bx_x_gui_c::set_display_mode(disp_mode_t newmode)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2017-01-15 15:57:06 +03:00
|
|
|
// if no mode change, do nothing.
|
|
|
|
if (disp_mode == newmode) return;
|
|
|
|
// remember the display mode for next time
|
|
|
|
disp_mode = newmode;
|
|
|
|
if ((newmode == DISP_MODE_SIM) && console_running()) {
|
|
|
|
console_cleanup();
|
2003-01-17 21:16:02 +03:00
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
}
|
2003-01-17 21:16:02 +03:00
|
|
|
|
2021-01-30 21:32:52 +03:00
|
|
|
void bx_x_gui_c::set_mouse_mode_absxy(bool mode)
|
2017-01-15 15:57:06 +03:00
|
|
|
{
|
|
|
|
x11_mouse_mode_absxy = mode;
|
|
|
|
}
|
2012-07-15 19:17:10 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
#if BX_SHOW_IPS
|
|
|
|
void bx_x_gui_c::show_ips(Bit32u ips_count)
|
|
|
|
{
|
2020-07-15 19:26:46 +03:00
|
|
|
if (x11_info_msg_counter == 0) {
|
2024-03-29 22:12:41 +03:00
|
|
|
if (!x11_ips_update && !gui_hide_ips) {
|
2017-01-15 15:57:06 +03:00
|
|
|
ips_count /= 1000;
|
|
|
|
sprintf(x11_ips_text, "IPS: %u.%3.3uM", ips_count / 1000, ips_count % 1000);
|
|
|
|
x11_ips_update = 1;
|
|
|
|
}
|
|
|
|
} else {
|
2020-07-15 19:26:46 +03:00
|
|
|
x11_info_msg_counter--;
|
2014-08-17 16:48:05 +04:00
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
}
|
2014-08-17 16:48:05 +04:00
|
|
|
#endif
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
#if BX_USE_IDLE_HACK
|
|
|
|
void bx_x_gui_c::sim_is_idle()
|
|
|
|
{
|
|
|
|
XEvent dummy;
|
|
|
|
struct timeval timeout;
|
|
|
|
timeout.tv_sec = 0;
|
|
|
|
timeout.tv_usec = 1000; /* 1/1000 s */
|
|
|
|
XPeekEventTimeout(bx_x_display, &dummy, &timeout);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
#endif /* BX_USE_IDLE_HACK */
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2021-01-30 21:32:52 +03:00
|
|
|
void bx_x_gui_c::statusbar_setitem_specific(int element, bool active, bool w)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2020-12-15 14:29:36 +03:00
|
|
|
Bit8u color = 0;
|
|
|
|
if (w) {
|
|
|
|
color = (statusitem[element].auto_off) ? 1 : 2;
|
|
|
|
}
|
|
|
|
x11_set_status_text(element+1, statusitem[element].text, active, color);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
// X11 gui: private methods
|
2008-02-16 01:05:43 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
void bx_x_gui_c::send_mouse_status(void)
|
|
|
|
{
|
|
|
|
int dx, dy, dz;
|
|
|
|
BX_DEBUG(("XXX: prev=(%d,%d) curr=(%d,%d)",
|
|
|
|
prev_x, prev_y, current_x, current_y));
|
2008-02-16 01:05:43 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
if (console_running()) return;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
if (x11_mouse_mode_absxy) {
|
|
|
|
if ((current_y >= (int)bx_headerbar_y) && (current_y < (int)(dimension_y + bx_headerbar_y))) {
|
|
|
|
dx = current_x * 0x7fff / dimension_x;
|
|
|
|
dy = (current_y - bx_headerbar_y) * 0x7fff / dimension_y;
|
|
|
|
dz = current_z;
|
|
|
|
DEV_mouse_motion(dx, dy, dz, mouse_button_state, 1);
|
|
|
|
}
|
|
|
|
prev_x = current_x;
|
|
|
|
prev_y = current_y;
|
|
|
|
return;
|
2008-02-16 01:05:43 +03:00
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
if (((prev_x!=-1) && (current_x!=-1) && (prev_y!=-1) && (current_y!=-1)) ||
|
|
|
|
(current_z != 0)) {
|
|
|
|
// (mch) consider warping here
|
|
|
|
dx = current_x - prev_x - warp_dx;
|
|
|
|
dy = -(current_y - prev_y - warp_dy);
|
|
|
|
dz = current_z;
|
|
|
|
warp_cursor(warp_home_x-current_x, warp_home_y-current_y);
|
2008-02-16 01:05:43 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
DEV_mouse_motion(dx, dy, dz, mouse_button_state, 0);
|
|
|
|
prev_x = current_x;
|
|
|
|
prev_y = current_y;
|
|
|
|
} else {
|
|
|
|
if ((current_x!=-1) && (current_y!=-1)) {
|
|
|
|
prev_x = current_x;
|
|
|
|
prev_y = current_y;
|
|
|
|
} else {
|
|
|
|
prev_x = current_x = -1;
|
|
|
|
prev_y = current_y = -1;
|
|
|
|
}
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
void bx_x_gui_c::xkeypress(KeySym keysym, int press_release)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2017-01-15 15:57:06 +03:00
|
|
|
Bit32u key_event;
|
2020-07-30 23:04:31 +03:00
|
|
|
Bit8u kmodchange = 0;
|
2021-01-30 21:32:52 +03:00
|
|
|
bool mouse_toggle = 0;
|
2001-11-12 03:45:09 +03:00
|
|
|
|
2020-07-30 23:04:31 +03:00
|
|
|
if ((keysym == XK_Shift_L) || (keysym == XK_Shift_R)) {
|
|
|
|
kmodchange = set_modifier_keys(BX_MOD_KEY_SHIFT, !press_release);
|
|
|
|
} else if ((keysym == XK_Control_L) || (keysym == XK_Control_R)) {
|
|
|
|
kmodchange = set_modifier_keys(BX_MOD_KEY_CTRL, !press_release);
|
|
|
|
} else if (keysym == XK_Alt_L) {
|
|
|
|
kmodchange = set_modifier_keys(BX_MOD_KEY_ALT, !press_release);
|
|
|
|
} else if (keysym == XK_Caps_Lock) {
|
|
|
|
kmodchange = set_modifier_keys(BX_MOD_KEY_CAPS, !press_release);
|
|
|
|
}
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
if (console_running() && !press_release) {
|
|
|
|
if (((keysym >= XK_space) && (keysym <= XK_asciitilde)) ||
|
|
|
|
(keysym == XK_Return) || (keysym == XK_BackSpace)) {
|
|
|
|
console_key_enq((Bit8u)(keysym & 0xff));
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2020-07-30 23:04:31 +03:00
|
|
|
|
|
|
|
if ((kmodchange & BX_MOD_KEY_CTRL) > 0) {
|
|
|
|
mouse_toggle = mouse_toggle_check(BX_MT_KEY_CTRL, !press_release);
|
|
|
|
} else if ((kmodchange & BX_MOD_KEY_ALT) > 0) {
|
|
|
|
mouse_toggle = mouse_toggle_check(BX_MT_KEY_ALT, !press_release);
|
2017-01-15 15:57:06 +03:00
|
|
|
} else if (keysym == XK_F10) {
|
2020-07-30 23:04:31 +03:00
|
|
|
mouse_toggle = mouse_toggle_check(BX_MT_KEY_F10, !press_release);
|
2017-01-15 15:57:06 +03:00
|
|
|
} else if (keysym == XK_F12) {
|
2020-07-30 23:04:31 +03:00
|
|
|
mouse_toggle = bx_gui->mouse_toggle_check(BX_MT_KEY_F12, !press_release);
|
2017-01-15 15:57:06 +03:00
|
|
|
}
|
|
|
|
if (mouse_toggle) {
|
|
|
|
toggle_mouse_enable();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-15 19:26:46 +03:00
|
|
|
if (!press_release) {
|
|
|
|
if (bx_gui->command_mode_active()) {
|
2020-07-18 16:36:21 +03:00
|
|
|
if (keysym == XK_a) {
|
|
|
|
bx_gui->floppyA_handler();
|
|
|
|
} else if (keysym == XK_b) {
|
|
|
|
bx_gui->floppyB_handler();
|
|
|
|
} else if (keysym == XK_c) {
|
2020-07-15 19:26:46 +03:00
|
|
|
bx_gui->copy_handler();
|
2020-07-30 23:04:31 +03:00
|
|
|
} else if (keysym == XK_C) {
|
2020-07-15 19:26:46 +03:00
|
|
|
bx_gui->config_handler();
|
2020-08-21 22:12:37 +03:00
|
|
|
} else if (keysym == XK_m) {
|
|
|
|
bx_gui->marklog_handler();
|
2020-07-15 19:26:46 +03:00
|
|
|
} else if (keysym == XK_p) {
|
|
|
|
bx_gui->paste_handler();
|
2020-07-30 23:04:31 +03:00
|
|
|
} else if (keysym == XK_P) {
|
|
|
|
bx_gui->power_handler();
|
2020-07-15 19:26:46 +03:00
|
|
|
} else if (keysym == XK_r) {
|
|
|
|
bx_gui->reset_handler();
|
|
|
|
} else if (keysym == XK_s) {
|
2020-07-30 23:04:31 +03:00
|
|
|
bx_gui->snapshot_handler();
|
|
|
|
} else if (keysym == XK_S) {
|
2020-07-15 19:26:46 +03:00
|
|
|
bx_gui->save_restore_handler();
|
|
|
|
} else if (keysym == XK_u) {
|
|
|
|
bx_gui->userbutton_handler();
|
|
|
|
}
|
2020-07-30 23:04:31 +03:00
|
|
|
if (kmodchange == 0) {
|
|
|
|
bx_gui->set_command_mode(0);
|
|
|
|
x11_set_status_text(0, "", 0, 1);
|
|
|
|
}
|
2020-07-15 19:26:46 +03:00
|
|
|
if (keysym != COMMAND_MODE_KEYSYM) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
2020-07-30 23:04:31 +03:00
|
|
|
if ((bx_gui->has_command_mode()) && (bx_gui->get_modifier_keys() == 0) &&
|
|
|
|
(keysym == COMMAND_MODE_KEYSYM)) {
|
2020-07-15 19:26:46 +03:00
|
|
|
bx_gui->set_command_mode(1);
|
|
|
|
x11_set_status_text(0, "Command mode", 0, 1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
/* Old (no mapping) behavior */
|
|
|
|
if (!x11_use_kbd_mapping) {
|
|
|
|
|
|
|
|
// this depends on the fact that the X11 keysyms which
|
|
|
|
// correspond to the ascii characters space .. tilde
|
|
|
|
// are in consequtive order.
|
|
|
|
if ((keysym >= XK_space) && (keysym <= XK_asciitilde)) {
|
|
|
|
key_event = ascii_to_key_event[keysym - XK_space];
|
|
|
|
}
|
|
|
|
else switch (keysym) {
|
|
|
|
case XK_KP_1:
|
|
|
|
#ifdef XK_KP_End
|
|
|
|
case XK_KP_End:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_END; break;
|
|
|
|
|
|
|
|
case XK_KP_2:
|
|
|
|
#ifdef XK_KP_Down
|
|
|
|
case XK_KP_Down:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_DOWN; break;
|
2002-03-06 12:31:55 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_3:
|
|
|
|
#ifdef XK_KP_Page_Down
|
|
|
|
case XK_KP_Page_Down:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_PAGE_DOWN; break;
|
2002-03-06 12:31:55 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_4:
|
|
|
|
#ifdef XK_KP_Left
|
|
|
|
case XK_KP_Left:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_LEFT; break;
|
2002-03-06 12:31:55 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_5:
|
|
|
|
#ifdef XK_KP_Begin
|
|
|
|
case XK_KP_Begin:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_5; break;
|
2001-11-12 03:45:09 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_6:
|
|
|
|
#ifdef XK_KP_Right
|
|
|
|
case XK_KP_Right:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_RIGHT; break;
|
2001-11-12 03:45:09 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_7:
|
|
|
|
#ifdef XK_KP_Home
|
|
|
|
case XK_KP_Home:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_HOME; break;
|
2001-11-12 03:45:09 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_8:
|
|
|
|
#ifdef XK_KP_Up
|
|
|
|
case XK_KP_Up:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_UP; break;
|
2008-02-06 01:57:43 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_9:
|
|
|
|
#ifdef XK_KP_Page_Up
|
|
|
|
case XK_KP_Page_Up:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_PAGE_UP; break;
|
2008-02-16 01:05:43 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_0:
|
|
|
|
#ifdef XK_KP_Insert
|
|
|
|
case XK_KP_Insert:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_INSERT; break;
|
2008-02-06 01:57:43 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_Decimal:
|
|
|
|
#ifdef XK_KP_Delete
|
|
|
|
case XK_KP_Delete:
|
|
|
|
#endif
|
|
|
|
key_event = BX_KEY_KP_DELETE; break;
|
2005-07-04 22:08:36 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
#ifdef XK_KP_Enter
|
|
|
|
case XK_KP_Enter: key_event = BX_KEY_KP_ENTER; break;
|
|
|
|
#endif
|
2008-02-06 01:57:43 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_Subtract: key_event = BX_KEY_KP_SUBTRACT; break;
|
|
|
|
case XK_KP_Add: key_event = BX_KEY_KP_ADD; break;
|
2001-11-12 03:45:09 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_KP_Multiply: key_event = BX_KEY_KP_MULTIPLY; break;
|
|
|
|
case XK_KP_Divide: key_event = BX_KEY_KP_DIVIDE; break;
|
2002-11-19 08:47:45 +03:00
|
|
|
|
2004-02-02 02:48:57 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_Up: key_event = BX_KEY_UP; break;
|
|
|
|
case XK_Down: key_event = BX_KEY_DOWN; break;
|
|
|
|
case XK_Left: key_event = BX_KEY_LEFT; break;
|
|
|
|
case XK_Right: key_event = BX_KEY_RIGHT; break;
|
2004-02-02 02:48:57 +03:00
|
|
|
|
2011-02-11 02:00:56 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
case XK_Delete: key_event = BX_KEY_DELETE; break;
|
|
|
|
case XK_BackSpace: key_event = BX_KEY_BACKSPACE; break;
|
|
|
|
case XK_Tab: key_event = BX_KEY_TAB; break;
|
|
|
|
#ifdef XK_ISO_Left_Tab
|
|
|
|
case XK_ISO_Left_Tab: key_event = BX_KEY_TAB; break;
|
|
|
|
#endif
|
|
|
|
case XK_Return: key_event = BX_KEY_ENTER; break;
|
|
|
|
case XK_Escape: key_event = BX_KEY_ESC; break;
|
|
|
|
case XK_F1: key_event = BX_KEY_F1; break;
|
|
|
|
case XK_F2: key_event = BX_KEY_F2; break;
|
|
|
|
case XK_F3: key_event = BX_KEY_F3; break;
|
|
|
|
case XK_F4: key_event = BX_KEY_F4; break;
|
|
|
|
case XK_F5: key_event = BX_KEY_F5; break;
|
|
|
|
case XK_F6: key_event = BX_KEY_F6; break;
|
|
|
|
case XK_F7: key_event = BX_KEY_F7; break;
|
|
|
|
case XK_F8: key_event = BX_KEY_F8; break;
|
|
|
|
case XK_F9: key_event = BX_KEY_F9; break;
|
|
|
|
case XK_F10: key_event = BX_KEY_F10; break;
|
|
|
|
case XK_F11: key_event = BX_KEY_F11; break;
|
|
|
|
case XK_F12: key_event = BX_KEY_F12; break;
|
|
|
|
case XK_Control_L: key_event = BX_KEY_CTRL_L; break;
|
|
|
|
#ifdef XK_Control_R
|
|
|
|
case XK_Control_R: key_event = BX_KEY_CTRL_R; break;
|
|
|
|
#endif
|
|
|
|
case XK_Shift_L: key_event = BX_KEY_SHIFT_L; break;
|
|
|
|
#ifdef XK_ISO_Prev_Group
|
|
|
|
case XK_ISO_Prev_Group: key_event = BX_KEY_SHIFT_L; break;
|
|
|
|
#endif
|
|
|
|
case XK_Shift_R: key_event = BX_KEY_SHIFT_R; break;
|
|
|
|
#ifdef XK_ISO_Next_Group
|
|
|
|
case XK_ISO_Next_Group: key_event = BX_KEY_SHIFT_R; break;
|
|
|
|
#endif
|
|
|
|
case XK_Alt_L: key_event = BX_KEY_ALT_L; break;
|
|
|
|
#ifdef XK_Alt_R
|
|
|
|
case XK_Alt_R: key_event = BX_KEY_ALT_R; break;
|
|
|
|
#endif
|
|
|
|
case XK_Caps_Lock: key_event = BX_KEY_CAPS_LOCK; break;
|
|
|
|
case XK_Num_Lock: key_event = BX_KEY_NUM_LOCK; break;
|
|
|
|
#ifdef XK_Scroll_Lock
|
|
|
|
case XK_Scroll_Lock: key_event = BX_KEY_SCRL_LOCK; break;
|
|
|
|
#endif
|
|
|
|
#ifdef XK_Print
|
|
|
|
case XK_Print: key_event = BX_KEY_PRINT; break;
|
|
|
|
#endif
|
|
|
|
#ifdef XK_Pause
|
|
|
|
case XK_Pause: key_event = BX_KEY_PAUSE; break;
|
|
|
|
#endif
|
|
|
|
#ifdef XK_Break
|
|
|
|
case XK_Break: key_event = BX_KEY_CTRL_BREAK; break;
|
|
|
|
#endif
|
|
|
|
case XK_Insert: key_event = BX_KEY_INSERT; break;
|
|
|
|
case XK_Home: key_event = BX_KEY_HOME; break;
|
|
|
|
case XK_End: key_event = BX_KEY_END; break;
|
|
|
|
case XK_Page_Up: key_event = BX_KEY_PAGE_UP; break;
|
|
|
|
case XK_Page_Down: key_event = BX_KEY_PAGE_DOWN; break;
|
2011-04-01 19:52:13 +04:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
default:
|
|
|
|
BX_ERROR(("xkeypress(): keysym %x unhandled!", (unsigned) keysym));
|
|
|
|
return;
|
|
|
|
break;
|
|
|
|
}
|
2011-04-01 19:52:13 +04:00
|
|
|
}
|
|
|
|
else {
|
2017-01-15 15:57:06 +03:00
|
|
|
/* use mapping */
|
|
|
|
BXKeyEntry *entry = bx_keymap.findHostKey(keysym);
|
|
|
|
if (!entry) {
|
|
|
|
BX_ERROR(("xkeypress(): keysym %x unhandled!", (unsigned) keysym));
|
|
|
|
return;
|
2006-01-23 21:34:47 +03:00
|
|
|
}
|
2017-01-15 15:57:06 +03:00
|
|
|
key_event = entry->baseKey;
|
2006-01-23 21:34:47 +03:00
|
|
|
}
|
2006-01-22 15:31:16 +03:00
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
if (press_release)
|
|
|
|
key_event |= BX_KEY_RELEASED;
|
|
|
|
|
|
|
|
DEV_kbd_gen_scancode(key_event);
|
2017-01-01 20:45:06 +03:00
|
|
|
}
|
|
|
|
|
2017-01-15 15:57:06 +03:00
|
|
|
// X11 dialog box implementation
|
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
// X11 control class
|
2006-01-23 21:34:47 +03:00
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
enum {
|
|
|
|
XDC_BUTTON = 0,
|
|
|
|
XDC_EDIT,
|
|
|
|
XDC_CHECKBOX
|
|
|
|
};
|
|
|
|
|
|
|
|
class x11_control_c {
|
|
|
|
public:
|
2016-12-27 19:26:57 +03:00
|
|
|
x11_control_c(int type, int x, int y, unsigned int w, unsigned int h,
|
|
|
|
const char *text);
|
2011-08-12 19:02:39 +04:00
|
|
|
virtual ~x11_control_c();
|
|
|
|
|
|
|
|
void draw(Display *display, Window win, GC gc);
|
2011-08-13 01:50:32 +04:00
|
|
|
void draw_rect(Display *display, Window win, GC gc);
|
|
|
|
void draw_text(Display *display, Window win, GC gc, const char *text);
|
2016-12-27 19:26:57 +03:00
|
|
|
void set_pos(int x, int y);
|
2011-08-12 19:02:39 +04:00
|
|
|
int test(XButtonEvent *bev);
|
|
|
|
int get_type() {return type;}
|
2016-12-27 19:26:57 +03:00
|
|
|
void set_param(int value) {param = value;}
|
|
|
|
int get_param() {return param;}
|
2011-08-12 19:02:39 +04:00
|
|
|
const char* get_text() {return text;}
|
|
|
|
// checkbox
|
|
|
|
int get_status() {return status;}
|
2021-01-30 21:32:52 +03:00
|
|
|
void set_status(bool new_stat) {status = new_stat;}
|
2011-08-12 19:02:39 +04:00
|
|
|
// edit
|
|
|
|
void set_maxlen(unsigned int max);
|
|
|
|
int process_input(KeySym key, const char *str);
|
|
|
|
const char* get_value() {return value;}
|
|
|
|
private:
|
|
|
|
unsigned int width, height;
|
2016-12-27 19:26:57 +03:00
|
|
|
int type, param, xmin, xmax, ymin, ymax;
|
2011-08-12 19:02:39 +04:00
|
|
|
const char *text;
|
|
|
|
// checkbox
|
2021-01-30 21:32:52 +03:00
|
|
|
bool status;
|
2011-08-12 19:02:39 +04:00
|
|
|
// edit
|
|
|
|
char *value;
|
|
|
|
char editstr[27];
|
|
|
|
unsigned int len, pos, max;
|
|
|
|
};
|
|
|
|
|
2016-12-27 19:26:57 +03:00
|
|
|
x11_control_c::x11_control_c(int _type, int x, int y, unsigned int w, unsigned int h,
|
|
|
|
const char *_text)
|
2011-08-12 19:02:39 +04:00
|
|
|
{
|
|
|
|
type = _type;
|
|
|
|
width = w;
|
|
|
|
height = h;
|
2016-12-27 19:26:57 +03:00
|
|
|
set_pos(x, y);
|
|
|
|
param = 0;
|
2011-08-12 19:02:39 +04:00
|
|
|
if (type == XDC_EDIT) {
|
|
|
|
len = strlen(_text);
|
|
|
|
max = len;
|
2016-12-27 19:26:57 +03:00
|
|
|
value = new char[max+1];
|
2011-08-12 19:02:39 +04:00
|
|
|
strcpy(value, _text);
|
|
|
|
pos = (len < 25) ? 0 : (len - 24);
|
|
|
|
strncpy(editstr, value+pos, 24);
|
|
|
|
editstr[len-pos] = 0;
|
|
|
|
text = editstr;
|
|
|
|
} else {
|
|
|
|
text = _text;
|
|
|
|
value = NULL;
|
|
|
|
if (type == XDC_CHECKBOX) {
|
|
|
|
status = !strcmp(text, "X");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
x11_control_c::~x11_control_c()
|
|
|
|
{
|
|
|
|
if (value != NULL) free(value);
|
|
|
|
}
|
|
|
|
|
2016-12-27 19:26:57 +03:00
|
|
|
void x11_control_c::set_pos(int x, int y)
|
|
|
|
{
|
|
|
|
xmin = x;
|
|
|
|
ymin = y;
|
|
|
|
xmax = x + width;
|
|
|
|
ymax = y + height;
|
|
|
|
}
|
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
void x11_control_c::draw(Display *display, Window win, GC gc)
|
|
|
|
{
|
2011-08-13 01:50:32 +04:00
|
|
|
int ty;
|
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
XDrawRectangle(display, win, gc, xmin, ymin, width, height);
|
2011-08-13 01:50:32 +04:00
|
|
|
ty = ymin + height - (height - 12) / 2;
|
|
|
|
XDrawImageString(display, win, gc, xmin+4, ty, (char *)text, strlen(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
void x11_control_c::draw_rect(Display *display, Window win, GC gc)
|
|
|
|
{
|
|
|
|
XDrawRectangle(display, win, gc, xmin - 2, ymin - 2, width + 4, height + 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
void x11_control_c::draw_text(Display *display, Window win, GC gc, const char *_text)
|
|
|
|
{
|
|
|
|
int ty;
|
|
|
|
|
|
|
|
ty = ymin + height - (height - 12) / 2;
|
|
|
|
XDrawImageString(display, win, gc, xmin+4, ty, (char *)_text, strlen(_text));
|
2011-08-12 19:02:39 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int x11_control_c::test(XButtonEvent *bev)
|
|
|
|
{
|
|
|
|
if ((bev->y > ymin) && (bev->y < ymax)) {
|
|
|
|
if ((bev->x > xmin) && (bev->x < xmax)) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void x11_control_c::set_maxlen(unsigned int _max)
|
2005-10-16 17:11:38 +04:00
|
|
|
{
|
2011-08-12 19:02:39 +04:00
|
|
|
max = _max;
|
2016-12-27 19:26:57 +03:00
|
|
|
char *newval = new char[max+1];
|
|
|
|
strcpy(newval, value);
|
|
|
|
delete [] value;
|
|
|
|
value = newval;
|
2011-08-12 19:02:39 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int x11_control_c::process_input(KeySym key, const char *str)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (key == XK_BackSpace) {
|
|
|
|
if (len > 0) {
|
|
|
|
value[--len] = 0;
|
|
|
|
if (pos > 0) pos--;
|
|
|
|
ret = 1;
|
|
|
|
}
|
|
|
|
} else if ((key == 0) && (len < max)) {
|
|
|
|
strcat(value, str);
|
|
|
|
len = strlen(value);
|
|
|
|
if (len > 24) pos++;
|
|
|
|
ret = 1;
|
|
|
|
}
|
|
|
|
strncpy(editstr, value+pos, 24);
|
|
|
|
editstr[len-pos] = 0;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// X11 static text structure
|
|
|
|
|
|
|
|
typedef struct _x11_static_t {
|
|
|
|
char *text;
|
|
|
|
int x, y;
|
|
|
|
struct _x11_static_t *next;
|
|
|
|
} x11_static_t;
|
|
|
|
|
|
|
|
// X11 dialog class
|
|
|
|
|
|
|
|
class x11_dialog_c {
|
|
|
|
public:
|
2016-12-27 19:26:57 +03:00
|
|
|
x11_dialog_c(const char *name, int width, int _height, int num_ctrls);
|
2011-08-12 19:02:39 +04:00
|
|
|
virtual ~x11_dialog_c();
|
|
|
|
|
2016-12-27 19:26:57 +03:00
|
|
|
int add_control(int type, int x, int y, unsigned int width, unsigned int height,
|
|
|
|
const char *text);
|
|
|
|
int add_button(const char *text);
|
2011-08-13 01:50:32 +04:00
|
|
|
void add_static_text(int x, int y, const char *text, int length);
|
2011-08-12 19:02:39 +04:00
|
|
|
void draw_text(Display *display, int x, int y, const char *text, int length);
|
2023-10-30 23:27:37 +03:00
|
|
|
void redraw(Display *display);
|
2016-12-27 19:26:57 +03:00
|
|
|
void set_control_param(int id, int value);
|
2011-08-12 19:02:39 +04:00
|
|
|
int run(int start_ctrl, int ok, int cancel);
|
2016-12-27 19:26:57 +03:00
|
|
|
x11_control_c* get_control(int id);
|
2011-08-12 19:02:39 +04:00
|
|
|
private:
|
|
|
|
Window dlgwin;
|
2005-10-16 17:11:38 +04:00
|
|
|
GC gc, gc_inv;
|
2016-12-27 19:26:57 +03:00
|
|
|
int btn_base, ctrl_cnt, cur_ctrl, old_ctrl, height, width;
|
2011-08-12 19:02:39 +04:00
|
|
|
x11_control_c **controls;
|
|
|
|
x11_static_t *static_items;
|
|
|
|
};
|
|
|
|
|
2016-12-27 19:26:57 +03:00
|
|
|
x11_dialog_c::x11_dialog_c(const char *name, int _width, int _height, int num_ctrls)
|
2011-08-12 19:02:39 +04:00
|
|
|
{
|
|
|
|
Window dialogw;
|
2008-05-12 22:46:46 +04:00
|
|
|
XSizeHints hint;
|
2005-10-16 17:11:38 +04:00
|
|
|
unsigned long black_pixel, white_pixel;
|
|
|
|
|
2016-12-27 19:26:57 +03:00
|
|
|
width = _width;
|
|
|
|
height = _height;
|
2005-10-16 17:11:38 +04:00
|
|
|
hint.flags = PPosition | PSize | PMinSize | PMaxSize;
|
|
|
|
hint.x = 100;
|
|
|
|
hint.y = 100;
|
2008-05-12 22:46:46 +04:00
|
|
|
hint.width = hint.min_width = hint.max_width = width;
|
|
|
|
hint.height = hint.min_height = hint.max_height = height;
|
2005-10-16 17:11:38 +04:00
|
|
|
black_pixel = BlackPixel(bx_x_display, bx_x_screen_num);
|
|
|
|
white_pixel = WhitePixel(bx_x_display, bx_x_screen_num);
|
2011-08-12 19:02:39 +04:00
|
|
|
dialogw = XCreateSimpleWindow(bx_x_display, RootWindow(bx_x_display,bx_x_screen_num),
|
2005-10-16 17:11:38 +04:00
|
|
|
hint.x, hint.y, hint.width, hint.height, 4, black_pixel, white_pixel);
|
2011-08-12 19:02:39 +04:00
|
|
|
XSetStandardProperties(bx_x_display, dialogw, name, name, None, NULL, 0, &hint);
|
2006-10-07 14:01:20 +04:00
|
|
|
Atom wm_delete = XInternAtom(bx_x_display, "WM_DELETE_WINDOW", 1);
|
2011-08-12 19:02:39 +04:00
|
|
|
XSetWMProtocols(bx_x_display, dialogw, &wm_delete, 1);
|
2005-10-16 17:11:38 +04:00
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
gc = XCreateGC(bx_x_display, dialogw, 0, 0);
|
|
|
|
gc_inv = XCreateGC(bx_x_display, dialogw, 0, 0);
|
2005-10-16 17:11:38 +04:00
|
|
|
XSetState(bx_x_display, gc_inv, white_pixel, black_pixel, GXcopy, AllPlanes);
|
2011-08-12 19:02:39 +04:00
|
|
|
XSetBackground(bx_x_display, gc, WhitePixel(bx_x_display, bx_x_screen_num));
|
|
|
|
XSetForeground(bx_x_display, gc, BlackPixel(bx_x_display, bx_x_screen_num));
|
2005-10-16 17:11:38 +04:00
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
XSelectInput(bx_x_display, dialogw, ButtonPressMask
|
2005-10-16 17:11:38 +04:00
|
|
|
| ButtonReleaseMask
|
|
|
|
| KeyPressMask
|
|
|
|
| KeyReleaseMask
|
|
|
|
| ExposureMask
|
|
|
|
| PointerMotionMask
|
|
|
|
| EnterWindowMask
|
|
|
|
| LeaveWindowMask);
|
2011-08-12 19:02:39 +04:00
|
|
|
XMapWindow(bx_x_display, dialogw);
|
2005-10-16 17:11:38 +04:00
|
|
|
XFlush(bx_x_display);
|
2011-08-12 19:02:39 +04:00
|
|
|
dlgwin = dialogw;
|
2016-12-27 19:26:57 +03:00
|
|
|
btn_base = 0;
|
2011-08-12 19:02:39 +04:00
|
|
|
ctrl_cnt = num_ctrls;
|
2023-10-30 23:27:37 +03:00
|
|
|
if (ctrl_cnt > 0) {
|
|
|
|
controls = new x11_control_c* [num_ctrls];
|
|
|
|
} else {
|
|
|
|
controls = NULL;
|
|
|
|
}
|
2011-08-12 19:02:39 +04:00
|
|
|
static_items = NULL;
|
|
|
|
cur_ctrl = 0;
|
|
|
|
old_ctrl = -1;
|
2008-05-12 22:46:46 +04:00
|
|
|
}
|
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
x11_dialog_c::~x11_dialog_c()
|
2011-08-09 18:28:25 +04:00
|
|
|
{
|
2011-08-12 19:02:39 +04:00
|
|
|
x11_static_t *temp;
|
|
|
|
|
|
|
|
for (int i = 0; i < ctrl_cnt; i++) {
|
|
|
|
delete controls[i];
|
|
|
|
}
|
|
|
|
delete [] controls;
|
|
|
|
while (static_items != NULL) {
|
|
|
|
temp = static_items;
|
|
|
|
static_items = temp->next;
|
|
|
|
delete [] temp->text;
|
|
|
|
delete temp;
|
|
|
|
}
|
|
|
|
XFreeGC(bx_x_display, gc);
|
|
|
|
XFreeGC(bx_x_display, gc_inv);
|
|
|
|
XDestroyWindow(bx_x_display, dlgwin);
|
2011-08-09 18:28:25 +04:00
|
|
|
}
|
|
|
|
|
2016-12-27 19:26:57 +03:00
|
|
|
int x11_dialog_c::add_control(int type, int x, int y, unsigned int w,
|
|
|
|
unsigned int h, const char *text)
|
2008-05-12 22:46:46 +04:00
|
|
|
{
|
2016-12-27 19:26:57 +03:00
|
|
|
x11_control_c *xctrl = new x11_control_c(type, x, y, w, h, text);
|
2011-08-12 19:02:39 +04:00
|
|
|
if (cur_ctrl < ctrl_cnt) {
|
2016-12-27 19:26:57 +03:00
|
|
|
controls[cur_ctrl] = xctrl;
|
|
|
|
}
|
|
|
|
if (type != XDC_BUTTON) {
|
|
|
|
btn_base = cur_ctrl + 1;
|
2011-08-12 19:02:39 +04:00
|
|
|
}
|
2016-12-27 19:26:57 +03:00
|
|
|
return cur_ctrl++;
|
|
|
|
}
|
|
|
|
|
|
|
|
int x11_dialog_c::add_button(const char *text)
|
|
|
|
{
|
|
|
|
int btn_id = add_control(XDC_BUTTON, 0, height - 30, 65, 20, text);
|
|
|
|
int n = cur_ctrl - btn_base;
|
|
|
|
int x0 = (width - ((n * 65) + ((n - 1) * 20))) / 2;
|
|
|
|
for (int i = btn_base; i < cur_ctrl; i++) {
|
|
|
|
controls[i]->set_pos(x0, height - 30);
|
|
|
|
x0 += 85;
|
|
|
|
}
|
|
|
|
return btn_id;
|
2008-05-12 22:46:46 +04:00
|
|
|
}
|
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
void x11_dialog_c::add_static_text(int x, int y, const char *text, int length)
|
2008-05-12 22:46:46 +04:00
|
|
|
{
|
2011-08-12 19:02:39 +04:00
|
|
|
x11_static_t *static_item, *temp;
|
|
|
|
|
2016-12-27 19:26:57 +03:00
|
|
|
static_item = new x11_static_t;
|
2011-08-12 19:02:39 +04:00
|
|
|
static_item->x = x;
|
|
|
|
static_item->y = y;
|
|
|
|
static_item->text = new char[length+1];
|
|
|
|
strncpy(static_item->text, text, length);
|
|
|
|
static_item->text[length] = 0;
|
|
|
|
static_item->next = NULL;
|
|
|
|
if (static_items == NULL) {
|
|
|
|
static_items = static_item;
|
|
|
|
} else {
|
|
|
|
temp = static_items;
|
|
|
|
while (temp->next) {
|
|
|
|
temp = temp->next;
|
2008-05-12 22:46:46 +04:00
|
|
|
}
|
2011-08-12 19:02:39 +04:00
|
|
|
temp->next = static_item;
|
2008-05-12 22:46:46 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
void x11_dialog_c::draw_text(Display *display, int x, int y, const char *text, int length)
|
|
|
|
{
|
|
|
|
XDrawImageString(display, dlgwin, gc, x, y, text, length);
|
|
|
|
}
|
|
|
|
|
2023-10-30 23:27:37 +03:00
|
|
|
void x11_dialog_c::redraw(Display *display)
|
|
|
|
{
|
|
|
|
x11_static_t *temp = static_items;
|
|
|
|
while (temp != NULL) {
|
|
|
|
draw_text(display, temp->x, temp->y, temp->text, strlen(temp->text));
|
|
|
|
temp = temp->next;
|
|
|
|
}
|
|
|
|
for (int i = 0; i < ctrl_cnt; i++) {
|
|
|
|
controls[i]->draw(display, dlgwin, gc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-27 19:26:57 +03:00
|
|
|
void x11_dialog_c::set_control_param(int id, int value)
|
|
|
|
{
|
|
|
|
if (id < ctrl_cnt) {
|
|
|
|
controls[id]->set_param(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
int x11_dialog_c::run(int start_ctrl, int ok, int cancel)
|
2008-05-12 22:46:46 +04:00
|
|
|
{
|
|
|
|
XEvent xevent;
|
|
|
|
KeySym key;
|
2021-01-30 21:32:52 +03:00
|
|
|
bool done = 0, valid = 0, status;
|
2016-12-27 19:26:57 +03:00
|
|
|
int i, init = 0;
|
2011-08-12 19:02:39 +04:00
|
|
|
char text[10], editstr[27];
|
2008-05-12 22:46:46 +04:00
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
if (start_ctrl < 0) {
|
|
|
|
cur_ctrl = ctrl_cnt - 1;
|
|
|
|
} else {
|
|
|
|
cur_ctrl = start_ctrl;
|
|
|
|
}
|
2005-10-16 17:11:38 +04:00
|
|
|
while (!done) {
|
|
|
|
XNextEvent(bx_x_display, &xevent);
|
|
|
|
switch (xevent.type) {
|
|
|
|
case Expose:
|
|
|
|
if (xevent.xexpose.count == 0) {
|
2023-10-30 23:27:37 +03:00
|
|
|
redraw(xevent.xexpose.display);
|
2011-08-12 19:02:39 +04:00
|
|
|
old_ctrl = cur_ctrl - 1;
|
2016-12-27 19:26:57 +03:00
|
|
|
if (old_ctrl < 0) old_ctrl = ctrl_cnt - 1;
|
2011-08-12 19:02:39 +04:00
|
|
|
init = 1;
|
2005-10-16 17:11:38 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ButtonPress:
|
|
|
|
if (xevent.xbutton.button == Button1) {
|
2011-08-12 19:02:39 +04:00
|
|
|
for (i = 0; i < ctrl_cnt; i++) {
|
|
|
|
if (controls[i]->test(&xevent.xbutton)) {
|
|
|
|
cur_ctrl = i;
|
|
|
|
if (controls[cur_ctrl]->get_type() == XDC_BUTTON) {
|
|
|
|
valid = 1;
|
|
|
|
} else if (controls[cur_ctrl]->get_type() == XDC_CHECKBOX) {
|
|
|
|
status = !controls[cur_ctrl]->get_status();
|
|
|
|
controls[cur_ctrl]->set_status(status);
|
2011-08-13 01:50:32 +04:00
|
|
|
controls[cur_ctrl]->draw_text(bx_x_display, dlgwin, gc, status ? "X":" ");
|
2011-08-12 19:02:39 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2005-10-16 17:11:38 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ButtonRelease:
|
2005-10-21 22:00:17 +04:00
|
|
|
if ((xevent.xbutton.button == Button1) && (valid == 1)) {
|
2005-10-16 17:11:38 +04:00
|
|
|
done = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case KeyPress:
|
|
|
|
i = XLookupString((XKeyEvent *)&xevent, text, 10, &key, 0);
|
|
|
|
if (key == XK_Tab) {
|
2011-08-12 19:02:39 +04:00
|
|
|
cur_ctrl++;
|
|
|
|
if (cur_ctrl >= ctrl_cnt) cur_ctrl = 0;
|
2005-10-21 22:00:17 +04:00
|
|
|
} else if (key == XK_Escape) {
|
2011-08-12 19:02:39 +04:00
|
|
|
cur_ctrl = cancel;
|
2005-10-21 22:00:17 +04:00
|
|
|
done = 1;
|
2011-08-12 19:02:39 +04:00
|
|
|
} else if (controls[cur_ctrl]->get_type() == XDC_EDIT) {
|
|
|
|
if (key == XK_Return) {
|
|
|
|
cur_ctrl = ok;
|
|
|
|
done = 1;
|
|
|
|
} else if (key == XK_BackSpace) {
|
|
|
|
if (controls[cur_ctrl]->process_input(key, "")) {
|
|
|
|
old_ctrl = cur_ctrl ^ 1;
|
|
|
|
}
|
|
|
|
} else if (i == 1) {
|
|
|
|
if (controls[cur_ctrl]->process_input(0, text)) {
|
|
|
|
old_ctrl = cur_ctrl ^ 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (controls[cur_ctrl]->get_type() == XDC_CHECKBOX) {
|
|
|
|
if (key == XK_space) {
|
|
|
|
status = !controls[cur_ctrl]->get_status();
|
|
|
|
controls[cur_ctrl]->set_status(status);
|
2011-08-13 01:50:32 +04:00
|
|
|
controls[cur_ctrl]->draw_text(bx_x_display, dlgwin, gc, status ? "X":" ");
|
2011-08-12 19:02:39 +04:00
|
|
|
} else if (key == XK_Return) {
|
|
|
|
cur_ctrl = ok;
|
|
|
|
done = 1;
|
|
|
|
}
|
2005-10-16 17:11:38 +04:00
|
|
|
} else if ((key == XK_space) || (key == XK_Return)) {
|
|
|
|
done = 1;
|
|
|
|
}
|
|
|
|
break;
|
2006-10-07 14:01:20 +04:00
|
|
|
case ClientMessage:
|
|
|
|
if (!strcmp(XGetAtomName(bx_x_display, xevent.xclient.message_type), "WM_PROTOCOLS")) {
|
2011-08-12 19:02:39 +04:00
|
|
|
cur_ctrl = cancel;
|
2006-10-07 14:01:20 +04:00
|
|
|
done = 1;
|
|
|
|
}
|
|
|
|
break;
|
2008-05-12 22:46:46 +04:00
|
|
|
case LeaveNotify:
|
|
|
|
valid = 0;
|
|
|
|
break;
|
2005-10-16 17:11:38 +04:00
|
|
|
}
|
2016-12-27 19:26:57 +03:00
|
|
|
if (init > 0) {
|
|
|
|
if ((init == 1) || (cur_ctrl != old_ctrl)) {
|
|
|
|
if (controls[old_ctrl]->get_type() == XDC_EDIT) {
|
|
|
|
sprintf(editstr, "%s ", controls[old_ctrl]->get_text());
|
|
|
|
controls[old_ctrl]->draw_text(bx_x_display, dlgwin, gc, editstr);
|
|
|
|
} else {
|
|
|
|
controls[old_ctrl]->draw_rect(bx_x_display, dlgwin, gc_inv);
|
|
|
|
}
|
|
|
|
if (controls[cur_ctrl]->get_type() == XDC_EDIT) {
|
|
|
|
sprintf(editstr, "%s_ ", controls[cur_ctrl]->get_text());
|
|
|
|
controls[cur_ctrl]->draw_text(bx_x_display, dlgwin, gc, editstr);
|
|
|
|
} else {
|
|
|
|
controls[cur_ctrl]->draw_rect(bx_x_display, dlgwin, gc);
|
|
|
|
}
|
|
|
|
old_ctrl = cur_ctrl;
|
|
|
|
init = 2;
|
2011-08-12 19:02:39 +04:00
|
|
|
}
|
2005-10-16 17:11:38 +04:00
|
|
|
}
|
|
|
|
}
|
2011-08-12 19:02:39 +04:00
|
|
|
return cur_ctrl;
|
|
|
|
}
|
|
|
|
|
2016-12-27 19:26:57 +03:00
|
|
|
x11_control_c* x11_dialog_c::get_control(int id)
|
|
|
|
{
|
2016-12-30 02:36:15 +03:00
|
|
|
if ((id >= 0) && (id < ctrl_cnt)) {
|
2016-12-27 19:26:57 +03:00
|
|
|
return controls[id];
|
|
|
|
} else {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-30 02:36:15 +03:00
|
|
|
// generic X11 message box
|
|
|
|
|
|
|
|
#define MAX_BUTTONS 5
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
unsigned int count;
|
|
|
|
int def_id;
|
|
|
|
int ok_id;
|
|
|
|
int cancel_id;
|
|
|
|
struct {
|
|
|
|
const char *label;
|
|
|
|
int code;
|
|
|
|
} btn[MAX_BUTTONS];
|
|
|
|
} x11_button_t;
|
|
|
|
|
|
|
|
int x11_message_box(const char *title, const char *message, x11_button_t *buttons)
|
|
|
|
{
|
|
|
|
unsigned int size_x, size_y;
|
|
|
|
unsigned int loffs[10], llen[10];
|
|
|
|
unsigned int i, cpos, ctrl_id, len, lines, maxlen, retcode, ypos;
|
|
|
|
|
|
|
|
cpos = 0;
|
|
|
|
lines = 0;
|
|
|
|
maxlen = 0;
|
|
|
|
while (cpos < strlen(message) && (lines < 10)) {
|
|
|
|
loffs[lines] = cpos;
|
|
|
|
while ((cpos < strlen(message)) && (message[cpos] != 0x0a)) cpos++;
|
|
|
|
len = cpos - loffs[lines];
|
|
|
|
llen[lines] = len;
|
|
|
|
if (len > maxlen) maxlen = len;
|
|
|
|
lines++;
|
|
|
|
cpos++;
|
|
|
|
}
|
|
|
|
size_x = buttons->count * 85 + 20;
|
|
|
|
if (maxlen > ((size_x - 30) / 6)) {
|
|
|
|
size_x = 30 + maxlen * 6;
|
|
|
|
}
|
|
|
|
size_y = 75 + lines * 15;
|
|
|
|
x11_dialog_c *xdlg = new x11_dialog_c(title, size_x, size_y, buttons->count);
|
|
|
|
ypos = 34;
|
|
|
|
for (i = 0; i < lines; i++) {
|
|
|
|
xdlg->add_static_text(20, ypos, message+loffs[i], llen[i]);
|
|
|
|
ypos += 15;
|
|
|
|
}
|
|
|
|
for (i = 0; i < buttons->count; i++) {
|
|
|
|
ctrl_id = xdlg->add_button(buttons->btn[i].label);
|
|
|
|
xdlg->set_control_param(ctrl_id, buttons->btn[i].code);
|
|
|
|
}
|
|
|
|
ctrl_id = xdlg->run(buttons->def_id, buttons->ok_id, buttons->cancel_id);
|
|
|
|
retcode = xdlg->get_control(ctrl_id)->get_param();
|
|
|
|
delete xdlg;
|
|
|
|
return retcode;
|
|
|
|
}
|
|
|
|
|
2023-10-30 23:27:37 +03:00
|
|
|
x11_dialog_c* x11_message_box_ml(const char *title, const char *message)
|
|
|
|
{
|
|
|
|
unsigned int size_x, size_y;
|
|
|
|
unsigned int loffs[10], llen[10];
|
|
|
|
unsigned int i, cpos, len, lines, maxlen, ypos, count;
|
|
|
|
XEvent xevent;
|
|
|
|
|
|
|
|
cpos = 0;
|
|
|
|
lines = 0;
|
|
|
|
maxlen = 0;
|
|
|
|
while (cpos < strlen(message) && (lines < 10)) {
|
|
|
|
loffs[lines] = cpos;
|
|
|
|
while ((cpos < strlen(message)) && (message[cpos] != 0x0a)) cpos++;
|
|
|
|
len = cpos - loffs[lines];
|
|
|
|
llen[lines] = len;
|
|
|
|
if (len > maxlen) maxlen = len;
|
|
|
|
lines++;
|
|
|
|
cpos++;
|
|
|
|
}
|
|
|
|
size_x = 105;
|
|
|
|
if (maxlen > ((size_x - 30) / 6)) {
|
|
|
|
size_x = 30 + maxlen * 6;
|
|
|
|
}
|
|
|
|
size_y = 75 + lines * 15;
|
|
|
|
x11_dialog_c *xdlg = new x11_dialog_c(title, size_x, size_y, 0);
|
|
|
|
ypos = 34;
|
|
|
|
for (i = 0; i < lines; i++) {
|
|
|
|
xdlg->add_static_text(20, ypos, message+loffs[i], llen[i]);
|
|
|
|
ypos += 15;
|
|
|
|
}
|
|
|
|
count = 2;
|
|
|
|
while (count > 0) {
|
|
|
|
XNextEvent(bx_x_display, &xevent);
|
|
|
|
if (xevent.type == Expose) {
|
|
|
|
if (xevent.xexpose.count == 0) {
|
|
|
|
xdlg->redraw(xevent.xexpose.display);
|
2023-12-03 17:28:23 +03:00
|
|
|
#if BX_HAVE_USLEEP
|
|
|
|
usleep(10000);
|
|
|
|
#endif
|
2023-10-30 23:27:37 +03:00
|
|
|
count--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
XFlush(bx_x_display);
|
|
|
|
return xdlg;
|
|
|
|
}
|
|
|
|
|
2011-08-12 19:02:39 +04:00
|
|
|
// X11 dialog box functions
|
|
|
|
|
|
|
|
int x11_ask_dialog(BxEvent *event)
|
|
|
|
{
|
2016-12-30 02:36:15 +03:00
|
|
|
const char *title = SIM->get_log_level_name(event->u.logmsg.level);
|
|
|
|
char message[256];
|
|
|
|
x11_button_t buttons;
|
|
|
|
int i = 0, mode;
|
|
|
|
|
|
|
|
sprintf(message, "Device: %s\n\nMessage: %s", event->u.logmsg.prefix,
|
|
|
|
event->u.logmsg.msg);
|
|
|
|
mode = event->u.logmsg.mode;
|
|
|
|
buttons.def_id = 0;
|
|
|
|
buttons.ok_id = 0;
|
|
|
|
buttons.cancel_id = 0;
|
|
|
|
if ((mode == BX_LOG_DLG_ASK) || (mode == BX_LOG_DLG_WARN)) {
|
|
|
|
buttons.btn[0].label = "Continue";
|
|
|
|
buttons.btn[0].code = BX_LOG_ASK_CHOICE_CONTINUE;
|
|
|
|
buttons.btn[1].label = "Alwayscont";
|
|
|
|
buttons.btn[1].code = BX_LOG_ASK_CHOICE_CONTINUE_ALWAYS;
|
|
|
|
i = 2;
|
|
|
|
}
|
|
|
|
if (mode == BX_LOG_DLG_ASK) {
|
2011-08-12 19:02:39 +04:00
|
|
|
#if BX_DEBUGGER || BX_GDBSTUB
|
2016-12-30 02:36:15 +03:00
|
|
|
buttons.btn[i].label = "Debugger";
|
|
|
|
buttons.btn[i].code = BX_LOG_ASK_CHOICE_ENTER_DEBUG;
|
|
|
|
i++;
|
2011-08-12 19:02:39 +04:00
|
|
|
#endif
|
2016-12-30 02:36:15 +03:00
|
|
|
#if BX_HAVE_ABORT
|
|
|
|
buttons.btn[i].label = "Dump Core";
|
|
|
|
buttons.btn[i].code = BX_LOG_ASK_CHOICE_DUMP_CORE;
|
|
|
|
i++;
|
2011-08-12 19:02:39 +04:00
|
|
|
#endif
|
2016-12-30 02:36:15 +03:00
|
|
|
}
|
|
|
|
if ((mode == BX_LOG_DLG_ASK) || (mode == BX_LOG_DLG_QUIT)) {
|
|
|
|
buttons.btn[i].label = "Quit";
|
|
|
|
buttons.btn[i].code = BX_LOG_ASK_CHOICE_DIE;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
buttons.count = i;
|
|
|
|
if (mode == BX_LOG_DLG_ASK) {
|
|
|
|
buttons.def_id = i - 1;
|
|
|
|
buttons.cancel_id = i - 1;
|
|
|
|
}
|
|
|
|
return x11_message_box(title, message, &buttons);
|
2005-10-16 17:11:38 +04:00
|
|
|
}
|
|
|
|
|
2013-01-22 22:26:49 +04:00
|
|
|
int x11_string_dialog(bx_param_string_c *param, bx_param_enum_c *param2)
|
2005-10-21 22:00:17 +04:00
|
|
|
{
|
2012-11-02 12:23:09 +04:00
|
|
|
x11_control_c *xctl_edit, *xbtn_status = NULL;
|
2016-12-27 19:26:57 +03:00
|
|
|
int h, num_ctrls, ctrl_id, edit_id, status_id, ok_id, cancel_id, retcode = -1;
|
2021-01-30 21:32:52 +03:00
|
|
|
bool status = 0;
|
2011-08-12 19:02:39 +04:00
|
|
|
char name[80], text[10];
|
|
|
|
const char *value;
|
2005-10-21 22:00:17 +04:00
|
|
|
|
2008-05-18 12:52:48 +04:00
|
|
|
if (param2 != NULL) {
|
|
|
|
strcpy(name, "First CD-ROM image/device");
|
2013-01-22 22:26:49 +04:00
|
|
|
status = (param2->get() == BX_INSERTED);
|
2008-05-18 12:52:48 +04:00
|
|
|
h = 110;
|
|
|
|
num_ctrls = 4;
|
2008-05-11 13:50:02 +04:00
|
|
|
} else {
|
2008-05-18 12:52:48 +04:00
|
|
|
if (param->get_label() != NULL) {
|
|
|
|
strcpy(name, param->get_label());
|
|
|
|
} else {
|
|
|
|
strcpy(name, param->get_name());
|
|
|
|
}
|
|
|
|
h = 90;
|
|
|
|
num_ctrls = 3;
|
2008-05-11 13:50:02 +04:00
|
|
|
}
|
2011-08-12 19:02:39 +04:00
|
|
|
x11_dialog_c *xdlg = new x11_dialog_c(name, 250, h, num_ctrls);
|
2016-12-27 19:26:57 +03:00
|
|
|
edit_id = xdlg->add_control(XDC_EDIT, 45, 20, 160, 20, param->getptr());
|
|
|
|
xctl_edit = xdlg->get_control(edit_id);
|
2011-08-12 19:02:39 +04:00
|
|
|
xctl_edit->set_maxlen(param->get_maxsize());
|
|
|
|
if (param2 != NULL) {
|
|
|
|
strcpy(text, status ? "X":" ");
|
2016-12-27 19:26:57 +03:00
|
|
|
status_id = xdlg->add_control(XDC_CHECKBOX, 45, 50, 15, 16, text);
|
|
|
|
xbtn_status = xdlg->get_control(status_id);
|
2011-08-12 19:02:39 +04:00
|
|
|
xdlg->add_static_text(70, 62, "Inserted", 8);
|
2005-10-21 22:00:17 +04:00
|
|
|
}
|
2016-12-27 19:26:57 +03:00
|
|
|
ok_id = xdlg->add_button("OK");
|
|
|
|
cancel_id = xdlg->add_button("Cancel");
|
|
|
|
ctrl_id = xdlg->run(edit_id, ok_id, cancel_id);
|
|
|
|
if (ctrl_id == ok_id) {
|
2011-08-12 19:02:39 +04:00
|
|
|
value = xctl_edit->get_value();
|
2008-05-18 12:52:48 +04:00
|
|
|
if (param2 != NULL) {
|
2011-08-12 19:02:39 +04:00
|
|
|
status = xbtn_status->get_status();
|
2008-05-18 12:52:48 +04:00
|
|
|
if (status == 1) {
|
2011-08-12 19:02:39 +04:00
|
|
|
if (strlen(value) > 0) {
|
2008-05-18 12:52:48 +04:00
|
|
|
param->set(value);
|
2009-03-22 23:18:17 +03:00
|
|
|
param2->set(1);
|
2008-05-18 12:52:48 +04:00
|
|
|
} else {
|
2009-03-22 23:18:17 +03:00
|
|
|
param2->set(0);
|
2008-05-18 12:52:48 +04:00
|
|
|
}
|
|
|
|
} else {
|
2009-03-22 23:18:17 +03:00
|
|
|
param2->set(0);
|
2008-05-18 12:52:48 +04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
param->set(value);
|
|
|
|
}
|
2016-12-27 19:26:57 +03:00
|
|
|
retcode = 1;
|
2008-05-18 12:52:48 +04:00
|
|
|
}
|
2011-08-12 19:02:39 +04:00
|
|
|
delete xdlg;
|
2016-12-27 19:26:57 +03:00
|
|
|
return retcode;
|
2005-10-21 22:00:17 +04:00
|
|
|
}
|
|
|
|
|
2016-12-30 02:36:15 +03:00
|
|
|
int x11_yesno_dialog(bx_param_bool_c *param)
|
2008-05-25 16:28:27 +04:00
|
|
|
{
|
2016-12-30 02:36:15 +03:00
|
|
|
const char *title, *message;
|
|
|
|
x11_button_t buttons;
|
|
|
|
int retcode;
|
2008-05-25 16:28:27 +04:00
|
|
|
|
|
|
|
if (param->get_label() != NULL) {
|
2016-12-30 02:36:15 +03:00
|
|
|
title = param->get_label();
|
2008-05-25 16:28:27 +04:00
|
|
|
} else {
|
2016-12-30 02:36:15 +03:00
|
|
|
title = param->get_name();
|
2008-05-25 16:28:27 +04:00
|
|
|
}
|
2016-12-27 19:26:57 +03:00
|
|
|
message = param->get_description();
|
2016-12-30 02:36:15 +03:00
|
|
|
buttons.btn[0].label = "Yes";
|
|
|
|
buttons.btn[0].code = 1;
|
|
|
|
buttons.btn[1].label = "No";
|
|
|
|
buttons.btn[1].code = 0;
|
|
|
|
buttons.count = 2;
|
|
|
|
buttons.def_id = 1 - param->get();
|
|
|
|
buttons.ok_id = 0;
|
|
|
|
buttons.cancel_id = 1;
|
|
|
|
retcode = x11_message_box(title, message, &buttons);
|
|
|
|
param->set(retcode);
|
|
|
|
return retcode;
|
2008-05-25 16:28:27 +04:00
|
|
|
}
|
|
|
|
|
2016-12-27 19:26:57 +03:00
|
|
|
BxEvent *x11_notify_callback(void *unused, BxEvent *event)
|
2005-10-16 17:11:38 +04:00
|
|
|
{
|
2005-10-21 22:00:17 +04:00
|
|
|
int opts;
|
|
|
|
bx_param_c *param;
|
|
|
|
bx_param_string_c *sparam;
|
2013-01-22 22:26:49 +04:00
|
|
|
bx_param_enum_c *eparam;
|
2008-05-18 12:52:48 +04:00
|
|
|
bx_list_c *list;
|
2020-10-18 22:57:05 +03:00
|
|
|
x11_button_t buttons;
|
2005-10-21 22:00:17 +04:00
|
|
|
|
2005-10-16 17:11:38 +04:00
|
|
|
switch (event->type)
|
|
|
|
{
|
2016-12-29 12:47:07 +03:00
|
|
|
case BX_SYNC_EVT_LOG_DLG:
|
2016-12-30 02:36:15 +03:00
|
|
|
event->retcode = x11_ask_dialog(event);
|
2005-10-16 17:11:38 +04:00
|
|
|
return event;
|
2020-10-18 22:57:05 +03:00
|
|
|
case BX_SYNC_EVT_MSG_BOX:
|
|
|
|
buttons.count = 1;
|
|
|
|
buttons.def_id = 0;
|
|
|
|
buttons.ok_id = 0;
|
|
|
|
buttons.btn[0].label = "OK";
|
|
|
|
buttons.btn[0].code = 0;
|
|
|
|
x11_message_box(event->u.logmsg.prefix, event->u.logmsg.msg, &buttons);
|
|
|
|
return event;
|
2023-10-30 23:27:37 +03:00
|
|
|
case BX_SYNC_EVT_ML_MSG_BOX:
|
|
|
|
event->param0 = (void*)x11_message_box_ml(event->u.logmsg.prefix, event->u.logmsg.msg);
|
|
|
|
return event;
|
|
|
|
case BX_SYNC_EVT_ML_MSG_BOX_KILL:
|
2023-12-03 17:28:23 +03:00
|
|
|
if (event->param0) {
|
|
|
|
delete (x11_dialog_c*)event->param0;
|
|
|
|
event->param0 = NULL;
|
|
|
|
}
|
2023-10-30 23:27:37 +03:00
|
|
|
return event;
|
2005-10-21 22:00:17 +04:00
|
|
|
case BX_SYNC_EVT_ASK_PARAM:
|
|
|
|
param = event->u.param.param;
|
|
|
|
if (param->get_type() == BXT_PARAM_STRING) {
|
|
|
|
sparam = (bx_param_string_c *)param;
|
2009-03-16 00:16:17 +03:00
|
|
|
opts = sparam->get_options();
|
2005-10-21 22:00:17 +04:00
|
|
|
if ((opts & sparam->IS_FILENAME) == 0) {
|
2008-05-18 12:52:48 +04:00
|
|
|
event->retcode = x11_string_dialog(sparam, NULL);
|
2005-10-21 22:00:17 +04:00
|
|
|
return event;
|
2008-05-25 16:28:27 +04:00
|
|
|
} else if ((opts & sparam->SAVE_FILE_DIALOG) ||
|
|
|
|
(opts & sparam->SELECT_FOLDER_DLG)) {
|
2008-05-18 12:52:48 +04:00
|
|
|
event->retcode = x11_string_dialog(sparam, NULL);
|
2008-05-11 13:50:02 +04:00
|
|
|
return event;
|
2005-10-21 22:00:17 +04:00
|
|
|
}
|
2008-05-18 12:52:48 +04:00
|
|
|
} else if (param->get_type() == BXT_LIST) {
|
|
|
|
list = (bx_list_c *)param;
|
|
|
|
sparam = (bx_param_string_c *)list->get_by_name("path");
|
2013-01-22 22:26:49 +04:00
|
|
|
eparam = (bx_param_enum_c *)list->get_by_name("status");
|
|
|
|
event->retcode = x11_string_dialog(sparam, eparam);
|
2008-05-18 12:52:48 +04:00
|
|
|
return event;
|
2008-05-25 16:28:27 +04:00
|
|
|
} else if (param->get_type() == BXT_PARAM_BOOL) {
|
2016-12-30 02:36:15 +03:00
|
|
|
event->retcode = x11_yesno_dialog((bx_param_bool_c*)param);
|
2008-05-25 16:28:27 +04:00
|
|
|
return event;
|
2005-10-21 22:00:17 +04:00
|
|
|
}
|
2005-10-16 17:11:38 +04:00
|
|
|
case BX_SYNC_EVT_TICK: // called periodically by siminterface.
|
|
|
|
case BX_ASYNC_EVT_REFRESH: // called when some bx_param_c parameters have changed.
|
|
|
|
// fall into default case
|
|
|
|
default:
|
|
|
|
return (*old_callback)(old_callback_arg, event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-11-19 08:47:45 +03:00
|
|
|
#endif /* if BX_WITH_X11 */
|