WinUAE/statusline.cpp
2014-07-30 00:04:46 +02:00

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);
}