mirror of
https://github.com/LIV2/WinUAE.git
synced 2025-12-06 00:12:52 +00:00
375 lines
10 KiB
C++
375 lines
10 KiB
C++
#include "sysconfig.h"
|
|
#include "sysdeps.h"
|
|
|
|
#include <ctype.h>
|
|
#include <assert.h>
|
|
|
|
#include "options.h"
|
|
#include "uae.h"
|
|
#include "xwin.h"
|
|
#include "gui.h"
|
|
#include "custom.h"
|
|
#include "drawing.h"
|
|
#include "statusline.h"
|
|
|
|
#define STATUSLINE_MS 3000
|
|
|
|
/*
|
|
* Some code to put status information on the screen.
|
|
*/
|
|
|
|
void statusline_getpos (int *x, int *y, int width, int height)
|
|
{
|
|
if (currprefs.osd_pos.x >= 20000) {
|
|
if (currprefs.osd_pos.x >= 30000)
|
|
*y = width * (currprefs.osd_pos.x - 30000) / 1000;
|
|
else
|
|
*y = width - (width * (30000 - currprefs.osd_pos.y) / 1000);
|
|
} else {
|
|
if (currprefs.osd_pos.x >= 0)
|
|
*x = currprefs.osd_pos.x;
|
|
else
|
|
*x = -currprefs.osd_pos.x + 1;
|
|
}
|
|
if (currprefs.osd_pos.y >= 20000) {
|
|
if (currprefs.osd_pos.y >= 30000)
|
|
*y = (height - TD_TOTAL_HEIGHT) * (currprefs.osd_pos.y - 30000) / 1000;
|
|
else
|
|
*y = (height - TD_TOTAL_HEIGHT) - ((height - TD_TOTAL_HEIGHT) * (30000 - currprefs.osd_pos.y) / 1000);
|
|
} else {
|
|
if (currprefs.osd_pos.y >= 0)
|
|
*y = height - TD_TOTAL_HEIGHT - currprefs.osd_pos.y;
|
|
else
|
|
*y = -currprefs.osd_pos.y + 1;
|
|
}
|
|
}
|
|
|
|
static const char *numbers = { /* ugly 0123456789CHD%+- */
|
|
"+++++++--++++-+++++++++++++++++-++++++++++++++++++++++++++++++++++++++++++++-++++++-++++----++---+--------------"
|
|
"+xxxxx+--+xx+-+xxxxx++xxxxx++x+-+x++xxxxx++xxxxx++xxxxx++xxxxx++xxxxx++xxxx+-+x++x+-+xxx++-+xx+-+x---+----------"
|
|
"+x+++x+--++x+-+++++x++++++x++x+++x++x++++++x++++++++++x++x+++x++x+++x++x++++-+x++x+-+x++x+--+x++x+--+x+----+++--"
|
|
"+x+-+x+---+x+-+xxxxx++xxxxx++xxxxx++xxxxx++xxxxx+--++x+-+xxxxx++xxxxx++x+----+xxxx+-+x++x+----+x+--+xxx+--+xxx+-"
|
|
"+x+++x+---+x+-+x++++++++++x++++++x++++++x++x+++x+--+x+--+x+++x++++++x++x++++-+x++x+-+x++x+---+x+x+--+x+----+++--"
|
|
"+xxxxx+---+x+-+xxxxx++xxxxx+----+x++xxxxx++xxxxx+--+x+--+xxxxx++xxxxx++xxxx+-+x++x+-+xxx+---+x++xx--------------"
|
|
"+++++++---+++-++++++++++++++----+++++++++++++++++--+++--++++++++++++++++++++-++++++-++++------------------------"
|
|
};
|
|
|
|
STATIC_INLINE uae_u32 ledcolor (uae_u32 c, uae_u32 *rc, uae_u32 *gc, uae_u32 *bc, uae_u32 *a)
|
|
{
|
|
uae_u32 v = rc[(c >> 16) & 0xff] | gc[(c >> 8) & 0xff] | bc[(c >> 0) & 0xff];
|
|
if (a)
|
|
v |= a[255 - ((c >> 24) & 0xff)];
|
|
return v;
|
|
}
|
|
|
|
static void write_tdnumber (uae_u8 *buf, int bpp, int x, int y, int num, uae_u32 c1, uae_u32 c2)
|
|
{
|
|
int j;
|
|
const char *numptr;
|
|
|
|
numptr = numbers + num * TD_NUM_WIDTH + NUMBERS_NUM * TD_NUM_WIDTH * y;
|
|
for (j = 0; j < TD_NUM_WIDTH; j++) {
|
|
if (*numptr == 'x')
|
|
putpixel (buf, bpp, x + j, c1, 1);
|
|
else if (*numptr == '+')
|
|
putpixel (buf, bpp, x + j, c2, 0);
|
|
numptr++;
|
|
}
|
|
}
|
|
|
|
void draw_status_line_single (uae_u8 *buf, int bpp, int y, int totalwidth, uae_u32 *rc, uae_u32 *gc, uae_u32 *bc, uae_u32 *alpha)
|
|
{
|
|
int x_start, j, led, border;
|
|
uae_u32 c1, c2, cb;
|
|
|
|
c1 = ledcolor (0x00ffffff, rc, gc, bc, alpha);
|
|
c2 = ledcolor (0x00000000, rc, gc, bc, alpha);
|
|
cb = ledcolor (TD_BORDER, rc, gc, bc, alpha);
|
|
|
|
if (td_pos & TD_RIGHT)
|
|
x_start = totalwidth - TD_PADX - VISIBLE_LEDS * TD_WIDTH;
|
|
else
|
|
x_start = TD_PADX;
|
|
|
|
for (led = 0; led < LED_MAX; led++) {
|
|
int side, pos, num1 = -1, num2 = -1, num3 = -1, num4 = -1;
|
|
int x, c, on = 0, am = 2;
|
|
xcolnr on_rgb = 0, on_rgb2 = 0, off_rgb = 0, pen_rgb = 0;
|
|
int half = 0;
|
|
|
|
if (!(currprefs.leds_on_screen_mask[picasso_on ? 1 : 0] & (1 << led)))
|
|
continue;
|
|
|
|
pen_rgb = c1;
|
|
if (led >= LED_DF0 && led <= LED_DF3) {
|
|
int pled = led - LED_DF0;
|
|
int track = gui_data.drive_track[pled];
|
|
pos = 6 + pled;
|
|
on_rgb = 0x00cc00;
|
|
on_rgb2 = 0x006600;
|
|
off_rgb = 0x003300;
|
|
if (!gui_data.drive_disabled[pled]) {
|
|
num1 = -1;
|
|
num2 = track / 10;
|
|
num3 = track % 10;
|
|
on = gui_data.drive_motor[pled];
|
|
if (gui_data.drive_writing[pled]) {
|
|
on_rgb = 0xcc0000;
|
|
on_rgb2 = 0x880000;
|
|
}
|
|
half = gui_data.drive_side ? 1 : -1;
|
|
if (gui_data.df[pled][0] == 0)
|
|
pen_rgb = ledcolor (0x00aaaaaa, rc, gc, bc, alpha);
|
|
}
|
|
side = gui_data.drive_side;
|
|
} else if (led == LED_POWER) {
|
|
pos = 3;
|
|
on_rgb = ((gui_data.powerled_brightness * 10 / 16) + 0x33) << 16;
|
|
on = 1;
|
|
off_rgb = 0x330000;
|
|
} else if (led == LED_CD) {
|
|
pos = 5;
|
|
if (gui_data.cd >= 0) {
|
|
on = gui_data.cd & (LED_CD_AUDIO | LED_CD_ACTIVE);
|
|
on_rgb = (on & LED_CD_AUDIO) ? 0x00cc00 : 0x0000cc;
|
|
if ((gui_data.cd & LED_CD_ACTIVE2) && !(gui_data.cd & LED_CD_AUDIO)) {
|
|
on_rgb &= 0xfefefe;
|
|
on_rgb >>= 1;
|
|
}
|
|
off_rgb = 0x000033;
|
|
num1 = -1;
|
|
num2 = 10;
|
|
num3 = 12;
|
|
}
|
|
} else if (led == LED_HD) {
|
|
pos = 4;
|
|
if (gui_data.hd >= 0) {
|
|
on = gui_data.hd;
|
|
on_rgb = on == 2 ? 0xcc0000 : 0x0000cc;
|
|
off_rgb = 0x000033;
|
|
num1 = -1;
|
|
num2 = 11;
|
|
num3 = 12;
|
|
}
|
|
} else if (led == LED_FPS) {
|
|
int fps = (gui_data.fps + 5) / 10;
|
|
pos = 2;
|
|
on_rgb = 0x000000;
|
|
off_rgb = gui_data.fps_color ? 0xcccc00 : 0x000000;
|
|
if (fps > 999)
|
|
fps = 999;
|
|
num1 = fps / 100;
|
|
num2 = (fps - num1 * 100) / 10;
|
|
num3 = fps % 10;
|
|
am = 3;
|
|
if (num1 == 0)
|
|
am = 2;
|
|
} else if (led == LED_CPU) {
|
|
int idle = (gui_data.idle + 5) / 10;
|
|
pos = 1;
|
|
on_rgb = 0xcc0000;
|
|
off_rgb = 0x000000;
|
|
if (gui_data.cpu_halted) {
|
|
on_rgb = 0xcccc00;
|
|
idle = 0;
|
|
on = 1;
|
|
num1 = -1;
|
|
num2 = 11;
|
|
num3 = gui_data.cpu_halted;
|
|
am = 2;
|
|
} else {
|
|
num1 = idle / 100;
|
|
num2 = (idle - num1 * 100) / 10;
|
|
num3 = idle % 10;
|
|
num4 = num1 == 0 ? 13 : -1;
|
|
am = 3;
|
|
}
|
|
} else if (led == LED_SND) {
|
|
int snd = abs(gui_data.sndbuf + 5) / 10;
|
|
if (snd > 99)
|
|
snd = 99;
|
|
pos = 0;
|
|
on = gui_data.sndbuf_status;
|
|
if (on < 3) {
|
|
num1 = gui_data.sndbuf < 0 ? 15 : 14;
|
|
num2 = snd / 10;
|
|
num3 = snd % 10;
|
|
}
|
|
on_rgb = 0x000000;
|
|
if (on < 0)
|
|
on_rgb = 0xcccc00; // underflow
|
|
else if (on == 2)
|
|
on_rgb = 0xcc0000; // really big overflow
|
|
else if (on == 1)
|
|
on_rgb = 0x0000cc; // "normal" overflow
|
|
off_rgb = 0x000000;
|
|
am = 3;
|
|
} else if (led == LED_MD && gui_data.drive_disabled[3]) {
|
|
// DF3 reused as internal non-volatile ram led (cd32/cdtv)
|
|
pos = 6 + 3;
|
|
if (gui_data.md >= 0) {
|
|
on = gui_data.md;
|
|
on_rgb = on == 2 ? 0xcc0000 : 0x00cc00;
|
|
off_rgb = 0x003300;
|
|
}
|
|
num1 = -1;
|
|
num2 = -1;
|
|
num3 = -1;
|
|
} else
|
|
return;
|
|
on_rgb |= 0x33000000;
|
|
off_rgb |= 0x33000000;
|
|
if (half > 0) {
|
|
c = ledcolor (on ? (y >= TD_TOTAL_HEIGHT / 2 ? on_rgb2 : on_rgb) : off_rgb, rc, gc, bc, alpha);
|
|
} else if (half < 0) {
|
|
c = ledcolor (on ? (y < TD_TOTAL_HEIGHT / 2 ? on_rgb2 : on_rgb) : off_rgb, rc, gc, bc, alpha);
|
|
} else {
|
|
c = ledcolor (on ? on_rgb : off_rgb, rc, gc, bc, alpha);
|
|
}
|
|
border = 0;
|
|
if (y == 0 || y == TD_TOTAL_HEIGHT - 1) {
|
|
c = ledcolor (TD_BORDER, rc, gc, bc, alpha);
|
|
border = 1;
|
|
}
|
|
|
|
x = x_start + pos * TD_WIDTH;
|
|
if (!border)
|
|
putpixel (buf, bpp, x - 1, cb, 0);
|
|
for (j = 0; j < TD_LED_WIDTH; j++)
|
|
putpixel (buf, bpp, x + j, c, 0);
|
|
if (!border)
|
|
putpixel (buf, bpp, x + j, cb, 0);
|
|
|
|
if (y >= TD_PADY && y - TD_PADY < TD_NUM_HEIGHT) {
|
|
if (num3 >= 0) {
|
|
x += (TD_LED_WIDTH - am * TD_NUM_WIDTH) / 2;
|
|
if (num1 > 0) {
|
|
write_tdnumber (buf, bpp, x, y - TD_PADY, num1, pen_rgb, c2);
|
|
x += TD_NUM_WIDTH;
|
|
}
|
|
if (num2 >= 0) {
|
|
write_tdnumber (buf, bpp, x, y - TD_PADY, num2, pen_rgb, c2);
|
|
x += TD_NUM_WIDTH;
|
|
}
|
|
write_tdnumber (buf, bpp, x, y - TD_PADY, num3, pen_rgb, c2);
|
|
x += TD_NUM_WIDTH;
|
|
if (num4 > 0)
|
|
write_tdnumber (buf, bpp, x, y - TD_PADY, num4, pen_rgb, c2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#define MAX_STATUSLINE_QUEUE 8
|
|
static TCHAR *statusline_text[MAX_STATUSLINE_QUEUE];
|
|
static TCHAR *statusline_text_active;
|
|
static int statusline_delay;
|
|
static bool statusline_had_changed;
|
|
|
|
bool has_statusline_updated(void)
|
|
{
|
|
bool v = statusline_had_changed;
|
|
statusline_had_changed = false;
|
|
return v;
|
|
}
|
|
|
|
static void statusline_update_notification(void)
|
|
{
|
|
statusline_had_changed = true;
|
|
statusline_updated();
|
|
}
|
|
|
|
void statusline_clear(void)
|
|
{
|
|
statusline_text_active = NULL;
|
|
statusline_delay = 0;
|
|
for (int i = 0; i < MAX_STATUSLINE_QUEUE; i++) {
|
|
xfree(statusline_text[i]);
|
|
statusline_text[i] = NULL;
|
|
}
|
|
statusline_update_notification();
|
|
}
|
|
|
|
const TCHAR *statusline_fetch(void)
|
|
{
|
|
return statusline_text_active;
|
|
}
|
|
|
|
void statusline_add_message(const TCHAR *format, ...)
|
|
{
|
|
va_list parms;
|
|
TCHAR buffer[256];
|
|
|
|
if (isguiactive())
|
|
return;
|
|
|
|
va_start(parms, format);
|
|
buffer[0] = ' ';
|
|
_vsntprintf(buffer + 1, 256 - 2, format, parms);
|
|
_tcscat(buffer, _T(" "));
|
|
|
|
if (statusline_text[1]) {
|
|
for (int i = 0; i < MAX_STATUSLINE_QUEUE; i++) {
|
|
if (statusline_text[i] && !_tcscmp(statusline_text[i], buffer)) {
|
|
xfree(statusline_text[i]);
|
|
for (int j = i + 1; j < MAX_STATUSLINE_QUEUE; j++) {
|
|
statusline_text[j - 1] = statusline_text[j];
|
|
}
|
|
statusline_text[MAX_STATUSLINE_QUEUE - 1] = NULL;
|
|
i = 0;
|
|
}
|
|
}
|
|
} else if (statusline_text[0]) {
|
|
if (!_tcscmp(statusline_text[0], buffer))
|
|
return;
|
|
}
|
|
|
|
for (int i = 0; i < MAX_STATUSLINE_QUEUE; i++) {
|
|
if (statusline_text[i] == NULL) {
|
|
statusline_text[i] = my_strdup(buffer);
|
|
if (i == 0)
|
|
statusline_delay = STATUSLINE_MS * vblank_hz / (1000 * 1);
|
|
statusline_text_active = statusline_text[0];
|
|
statusline_update_notification();
|
|
return;
|
|
}
|
|
}
|
|
statusline_text_active = NULL;
|
|
xfree(statusline_text[0]);
|
|
for (int i = 1; i < MAX_STATUSLINE_QUEUE; i++) {
|
|
statusline_text[i - 1] = statusline_text[i];
|
|
}
|
|
statusline_text[MAX_STATUSLINE_QUEUE - 1] = my_strdup(buffer);
|
|
statusline_text_active = statusline_text[0];
|
|
statusline_update_notification();
|
|
va_end(parms);
|
|
}
|
|
|
|
void statusline_vsync(void)
|
|
{
|
|
if (!statusline_text[0])
|
|
return;
|
|
if (statusline_delay == 0)
|
|
statusline_delay = STATUSLINE_MS * vblank_hz / (1000 * 1);
|
|
if (statusline_delay > STATUSLINE_MS * vblank_hz / (1000 * 1))
|
|
statusline_delay = STATUSLINE_MS * vblank_hz / (1000 * 1);
|
|
if (statusline_delay > STATUSLINE_MS * vblank_hz / (1000 * 3) && statusline_text[1])
|
|
statusline_delay = STATUSLINE_MS * vblank_hz / (1000 * 3);
|
|
statusline_delay--;
|
|
if (statusline_delay)
|
|
return;
|
|
statusline_text_active = NULL;
|
|
xfree(statusline_text[0]);
|
|
for (int i = 1; i < MAX_STATUSLINE_QUEUE; i++) {
|
|
statusline_text[i - 1] = statusline_text[i];
|
|
}
|
|
statusline_text[MAX_STATUSLINE_QUEUE - 1] = NULL;
|
|
statusline_text_active = statusline_text[0];
|
|
statusline_update_notification();
|
|
}
|
|
|
|
void statusline_single_erase(uae_u8 *buf, int bpp, int y, int totalwidth)
|
|
{
|
|
memset(buf, 0, bpp * totalwidth);
|
|
}
|