dxx-rebirth/similar/main/newmenu.cpp

2088 lines
56 KiB
C++
Raw Normal View History

2006-03-20 17:12:09 +00:00
/*
2014-06-01 17:55:23 +00:00
* Portions of this file are copyright Rebirth contributors and licensed as
* described in COPYING.txt.
* Portions of this file are copyright Parallax Software and licensed
* according to the Parallax license below.
* See COPYING.txt for license details.
2006-03-20 17:12:09 +00:00
THE COMPUTER CODE CONTAINED HEREIN IS THE SOLE PROPERTY OF PARALLAX
SOFTWARE CORPORATION ("PARALLAX"). PARALLAX, IN DISTRIBUTING THE CODE TO
END-USERS, AND SUBJECT TO ALL OF THE TERMS AND CONDITIONS HEREIN, GRANTS A
ROYALTY-FREE, PERPETUAL LICENSE TO SUCH END-USERS FOR USE BY SUCH END-USERS
IN USING, DISPLAYING, AND CREATING DERIVATIVE WORKS THEREOF, SO LONG AS
SUCH USE, DISPLAY OR CREATION IS FOR NON-COMMERCIAL, ROYALTY OR REVENUE
FREE PURPOSES. IN NO EVENT SHALL THE END-USER USE THE COMPUTER CODE
CONTAINED HEREIN FOR REVENUE-BEARING PURPOSES. THE END-USER UNDERSTANDS
AND AGREES TO THE TERMS HEREIN AND ACCEPTS THE SAME BY USE OF THIS FILE.
COPYRIGHT 1993-1999 PARALLAX SOFTWARE CORPORATION. ALL RIGHTS RESERVED.
*/
/*
*
* Routines for menus.
*
*/
#include <stdio.h>
#include <cstdlib>
2006-03-20 17:12:09 +00:00
#include <string.h>
#include <stdarg.h>
#include <ctype.h>
#include "pstypes.h"
#include "dxxerror.h"
2006-03-20 17:12:09 +00:00
#include "gr.h"
2006-12-24 09:15:37 +00:00
#include "grdef.h"
#include "window.h"
2006-03-20 17:12:09 +00:00
#include "songs.h"
#include "key.h"
#include "mouse.h"
2006-03-20 17:12:09 +00:00
#include "palette.h"
#include "game.h"
#include "text.h"
#include "menu.h"
#include "newmenu.h"
#include "gamefont.h"
#include "iff.h"
#include "pcx.h"
#include "u_mem.h"
#include "mouse.h"
#include "joy.h"
#include "digi.h"
#include "multi.h"
#include "endlevel.h"
#include "screens.h"
#include "config.h"
#include "player.h"
#include "state.h"
2006-03-20 17:12:09 +00:00
#include "newdemo.h"
#include "kconfig.h"
#include "strutil.h"
#include "vers_id.h"
#include "timer.h"
#include "playsave.h"
#include "automap.h"
#include "rbaudio.h"
#if defined(DXX_BUILD_DESCENT_II)
#include "args.h"
#include "gamepal.h"
#endif
2006-03-20 17:12:09 +00:00
#ifdef OGL
#include "ogl_init.h"
#endif
2015-03-12 02:21:21 +00:00
#include "compiler-range_for.h"
#include "partial_range.h"
2006-03-20 17:12:09 +00:00
#define MAXDISPLAYABLEITEMS 14
#define MAXDISPLAYABLEITEMSTINY 21
#define MESSAGEBOX_TEXT_SIZE 2176 // How many characters in messagebox
#define MAX_TEXT_WIDTH FSPACX(120) // How many pixels wide a input box can be
2006-03-20 17:12:09 +00:00
2014-09-07 18:06:59 +00:00
struct newmenu : embed_window_pointer_t
{
int x,y,w,h;
short swidth, sheight; float fntscalex, fntscaley; // with these we check if resolution or fonts have changed so menu structure can be recreated
const char *title;
const char *subtitle;
2015-03-12 02:21:21 +00:00
unsigned nitems;
2015-01-24 19:16:34 +00:00
int citem;
newmenu_item *items;
2014-10-04 21:47:13 +00:00
int (*subfunction)(newmenu *menu,const d_event &event, void *userdata);
const char *filename;
int tiny_mode;
int tabs_flag;
int scroll_offset, last_scroll_check, max_displayable;
int all_text; //set true if all text items
int is_scroll_box; // Is this a scrolling box? Set to false at init
int max_on_menu;
int mouse_state, dblclick_flag;
int *rval; // Pointer to return value (for polling newmenus)
void *userdata; // For whatever - like with window system
2015-03-12 02:21:21 +00:00
partial_range_t<newmenu_item *> item_range()
{
return unchecked_partial_range(items, nitems);
}
};
grs_bitmap nm_background, nm_background1;
2014-07-22 02:50:01 +00:00
static grs_subbitmap_ptr nm_background_sub;
void newmenu_free_background() {
if (nm_background.bm_data)
{
2014-07-22 02:50:01 +00:00
nm_background_sub.reset();
2014-11-30 22:09:20 +00:00
gr_free_bitmap_data(nm_background);
}
if (nm_background1.bm_data)
2014-11-30 22:09:20 +00:00
gr_free_bitmap_data(nm_background1);
}
2006-03-20 17:12:09 +00:00
// Draws the custom menu background pcx, if available
static void nm_draw_background1(const char * filename)
2006-03-20 17:12:09 +00:00
{
int pcx_error;
if (filename != NULL)
{
if (nm_background1.bm_data == NULL)
{
2014-11-30 22:09:20 +00:00
gr_init_bitmap_data(nm_background1);
2014-12-02 03:35:01 +00:00
pcx_error = pcx_read_bitmap( filename, nm_background1, BM_LINEAR, gr_palette );
Assert(pcx_error == PCX_ERROR_NONE);
2011-09-26 23:31:19 +00:00
(void)pcx_error;
}
gr_palette_load( gr_palette );
2014-12-02 03:35:01 +00:00
show_fullscr(nm_background1);
2006-03-20 17:12:09 +00:00
}
#if defined(DXX_BUILD_DESCENT_II)
strcpy(last_palette_loaded,""); //force palette load next time
#endif
2006-03-20 17:12:09 +00:00
}
#define MENU_BACKGROUND_BITMAP_HIRES (PHYSFSX_exists("scoresb.pcx",1)?"scoresb.pcx":"scores.pcx")
#define MENU_BACKGROUND_BITMAP_LORES (PHYSFSX_exists("scores.pcx",1)?"scores.pcx":"scoresb.pcx")
#define MENU_BACKGROUND_BITMAP (HIRESMODE?MENU_BACKGROUND_BITMAP_HIRES:MENU_BACKGROUND_BITMAP_LORES)
2006-03-20 17:12:09 +00:00
// Draws the frame background for menus
2006-03-20 17:12:09 +00:00
void nm_draw_background(int x1, int y1, int x2, int y2 )
{
int w,h,init_sub=0;
static float BGScaleX=1,BGScaleY=1;
2014-09-06 23:55:43 +00:00
grs_canvas *old;
2006-03-20 17:12:09 +00:00
if (nm_background.bm_data == NULL)
{
2006-03-20 17:12:09 +00:00
int pcx_error;
palette_array_t background_palette;
2014-11-30 22:09:20 +00:00
gr_init_bitmap_data(nm_background);
2014-12-02 03:35:01 +00:00
pcx_error = pcx_read_bitmap(MENU_BACKGROUND_BITMAP, nm_background,BM_LINEAR,background_palette);
2006-03-20 17:12:09 +00:00
Assert(pcx_error == PCX_ERROR_NONE);
2011-09-26 23:31:19 +00:00
(void)pcx_error;
2015-01-25 05:32:45 +00:00
gr_remap_bitmap_good(nm_background, background_palette, -1, -1);
BGScaleX=((float)SWIDTH/nm_background.bm_w);
BGScaleY=((float)SHEIGHT/nm_background.bm_h);
init_sub=1;
2006-03-20 17:12:09 +00:00
}
if ( x1 < 0 ) x1 = 0;
if ( y1 < 0 ) y1 = 0;
if ( x2 > SWIDTH - 1) x2 = SWIDTH - 1;
if ( y2 > SHEIGHT - 1) y2 = SHEIGHT - 1;
2006-03-20 17:12:09 +00:00
w = x2-x1;
h = y2-y1;
if (w > SWIDTH) w = SWIDTH;
if (h > SHEIGHT) h = SHEIGHT;
2006-03-20 17:12:09 +00:00
old=grd_curcanv;
2015-01-17 18:31:40 +00:00
auto tmp = gr_create_sub_canvas(*old, x1, y1, w, h);
gr_set_current_canvas(tmp);
gr_palette_load( gr_palette );
2014-12-02 03:35:01 +00:00
show_fullscr(nm_background); // show so we load all necessary data for the sub-bitmap
if (!init_sub && ((nm_background_sub->bm_w != w*(((float) nm_background.bm_w)/SWIDTH)) || (nm_background_sub->bm_h != h*(((float) nm_background.bm_h)/SHEIGHT))))
{
init_sub=1;
}
if (init_sub)
2014-11-30 22:09:21 +00:00
nm_background_sub = gr_create_sub_bitmap(nm_background,0,0,w*(((float) nm_background.bm_w)/SWIDTH),h*(((float) nm_background.bm_h)/SHEIGHT));
2014-12-02 03:35:01 +00:00
show_fullscr(*nm_background_sub.get());
gr_set_current_canvas(old);
2006-03-20 17:12:09 +00:00
gr_settransblend(14, GR_BLEND_NORMAL);
gr_setcolor( BM_XRGB(1,1,1) );
for (w=5*BGScaleX;w>0;w--)
gr_urect( x2-w, y1+w*(BGScaleY/BGScaleX), x2-w, y2-w*(BGScaleY/BGScaleX) );//right edge
gr_setcolor( BM_XRGB(0,0,0) );
for (h=5*BGScaleY;h>0;h--)
gr_urect( x1+h*(BGScaleX/BGScaleY), y2-h, x2-h*(BGScaleX/BGScaleY), y2-h );//bottom edge
gr_settransblend(GR_FADE_OFF, GR_BLEND_NORMAL);
2006-03-20 17:12:09 +00:00
}
// Draw a left justfied string
2012-07-01 03:44:16 +00:00
static void nm_string( int w1,int x, int y, const char * s, int tabs_flag)
2006-03-20 17:12:09 +00:00
{
2015-09-29 02:41:22 +00:00
int w,t=0,i;
char *p,*s1,measure[2];
int XTabs[]={18,90,127,165,231,256};
2006-03-20 17:12:09 +00:00
p=s1=NULL;
RAIIdmem<char[]> s2(d_strdup(s));
2006-03-20 17:12:09 +00:00
2015-06-13 22:42:20 +00:00
const auto &&fspacx = FSPACX();
2015-03-12 02:21:21 +00:00
range_for (auto &i, XTabs)
{
2015-06-13 22:42:20 +00:00
i = fspacx(i) + x;
2006-03-20 17:12:09 +00:00
}
2006-03-20 17:12:09 +00:00
measure[1]=0;
if (!tabs_flag) {
p = strchr(s2.get(), '\t');
2006-03-20 17:12:09 +00:00
if (p && (w1>0) ) {
*p = '\0';
s1 = p+1;
}
}
if (w1 > 0)
w = w1;
2015-09-29 02:41:22 +00:00
else
gr_get_string_size(s2.get(), &w, nullptr, nullptr);
2006-03-20 17:12:09 +00:00
if (tabs_flag) {
for (i=0;s2[i];i++) {
if (s2[i]=='\t' && tabs_flag) {
2006-03-20 17:12:09 +00:00
x=XTabs[t];
t++;
continue;
}
measure[0]=s2[i];
2015-09-29 02:41:22 +00:00
int tx, th;
gr_get_string_size(measure, &tx, &th, nullptr);
2015-09-29 02:41:22 +00:00
gr_string(x, y, measure, tx, th);
2006-03-20 17:12:09 +00:00
x+=tx;
}
}
else
gr_string(x, y, s2.get());
if (!tabs_flag && p && (w1>0) ) {
2015-09-29 02:41:22 +00:00
int w, h;
gr_get_string_size(s1, &w, &h, nullptr);
2006-03-20 17:12:09 +00:00
2015-09-29 02:41:22 +00:00
gr_string(x + w1 - w, y, s1, w, h);
2006-03-20 17:12:09 +00:00
*p = '\t';
}
}
// Draw a slider and it's string
2013-10-27 22:00:14 +00:00
static void nm_string_slider( int w1,int x, int y, char * s )
2006-03-20 17:12:09 +00:00
{
char *p,*s1;
s1=NULL;
p = strchr( s, '\t' );
if (p) {
*p = '\0';
s1 = p+1;
}
gr_string( x, y, s );
2006-03-20 17:12:09 +00:00
if (p) {
2015-09-29 02:41:22 +00:00
int w, h;
gr_get_string_size(s1, &w, &h, nullptr);
2015-09-29 02:41:22 +00:00
gr_string(x + w1 - w, y, s1, w, h);
2006-03-20 17:12:09 +00:00
*p = '\t';
}
2006-03-20 17:12:09 +00:00
}
// Draw a left justfied string with black background.
2012-07-01 03:44:16 +00:00
static void nm_string_black( int w1,int x, int y, const char * s )
2006-03-20 17:12:09 +00:00
{
2015-09-29 02:41:22 +00:00
int w,h;
gr_get_string_size(s, &w, &h, nullptr);
2006-03-20 17:12:09 +00:00
if (w1 == 0) w1 = w;
gr_setcolor( BM_XRGB(5,5,5));
2015-06-13 22:42:20 +00:00
const auto &&fspacx = FSPACX();
2015-06-13 22:42:20 +00:00
const auto &&fspacy = FSPACY();
gr_rect(x - fspacx(2), y - fspacy(1), x + w1, y + h);
gr_setcolor( BM_XRGB(2,2,2));
2015-06-13 22:42:20 +00:00
gr_rect(x - fspacx(2), y - fspacy(1), x, y + h);
gr_setcolor( BM_XRGB(0,0,0));
2015-06-13 22:42:20 +00:00
gr_rect(x - fspacx(1), y - fspacy(1), x + w1 - fspacx(1), y + h);
2015-09-29 02:41:22 +00:00
gr_string(x, y, s, w, h);
2006-03-20 17:12:09 +00:00
}
// Draw a right justfied string
static void nm_rstring( int w1,int x, int y, const char * s )
2006-03-20 17:12:09 +00:00
{
2015-09-29 02:41:22 +00:00
int w, h;
gr_get_string_size(s, &w, &h, nullptr);
x -= FSPACX(3);
2006-03-20 17:12:09 +00:00
if (w1 == 0) w1 = w;
2015-09-29 02:41:22 +00:00
gr_string(x - w, y, s, w, h);
2006-03-20 17:12:09 +00:00
}
2012-07-01 03:44:16 +00:00
static void nm_string_inputbox( int w, int x, int y, const char * text, int current )
2006-03-20 17:12:09 +00:00
{
2015-09-29 02:41:22 +00:00
int w1;
2006-03-20 17:12:09 +00:00
// even with variable char widths and a box that goes over the whole screen, we maybe never get more than 75 chars on the line
if (strlen(text)>75)
text+=strlen(text)-75;
2006-03-20 17:12:09 +00:00
while( *text ) {
2015-09-29 02:41:22 +00:00
gr_get_string_size(text, &w1, nullptr, nullptr);
if ( w1 > w-FSPACX(10) )
2006-03-20 17:12:09 +00:00
text++;
else
break;
}
if ( *text == 0 )
w1 = 0;
nm_string_black( w, x, y, text );
if ( current && timer_query() & 0x8000 )
gr_string( x+w1, y, CURSOR_STRING );
2006-03-20 17:12:09 +00:00
}
2013-10-27 22:00:14 +00:00
static void draw_item( newmenu_item *item, int is_current, int tiny, int tabs_flag, int scroll_offset )
2006-03-20 17:12:09 +00:00
{
if (tiny)
{
if (is_current)
gr_set_fontcolor(gr_find_closest_color_current(57,49,20),-1);
else
gr_set_fontcolor(gr_find_closest_color_current(29,29,47),-1);
if (item->text[0]=='\t')
gr_set_fontcolor (gr_find_closest_color_current(63,63,63),-1);
}
else
{
gr_set_curfont(is_current?MEDIUM2_FONT:MEDIUM1_FONT);
2006-03-20 17:12:09 +00:00
}
2015-06-13 22:42:22 +00:00
const int line_spacing = static_cast<int>(LINE_SPACING);
2006-03-20 17:12:09 +00:00
switch( item->type ) {
case NM_TYPE_TEXT:
case NM_TYPE_MENU:
2015-06-13 22:42:22 +00:00
nm_string(item->w, item->x, item->y - (line_spacing * scroll_offset), item->text, tabs_flag);
break;
case NM_TYPE_SLIDER:
{
int i;
if (item->value < item->min_value) item->value=item->min_value;
if (item->value > item->max_value) item->value=item->max_value;
i = snprintf(item->saved_text.data(), item->saved_text.size(), "%s\t%s", item->text, SLIDER_LEFT);
for (uint_fast32_t j = (item->max_value-item->min_value+1); j--;)
{
i += snprintf(item->saved_text.data() + i, item->saved_text.size() - i, "%s", SLIDER_MIDDLE);
}
i += snprintf(item->saved_text.data() + i, item->saved_text.size() - i, "%s", SLIDER_RIGHT);
item->saved_text[item->value+1+strlen(item->text)+1] = SLIDER_MARKER[0];
2015-06-13 22:42:22 +00:00
nm_string_slider(item->w, item->x, item->y - (line_spacing * scroll_offset), item->saved_text.data());
2006-03-20 17:12:09 +00:00
}
break;
case NM_TYPE_INPUT_MENU:
if ( item->group==0 )
{
2015-06-13 22:42:22 +00:00
nm_string(item->w, item->x, item->y - (line_spacing * scroll_offset), item->text, tabs_flag);
} else {
2015-06-13 22:42:22 +00:00
nm_string_inputbox(item->w, item->x, item->y - (line_spacing * scroll_offset), item->text, is_current);
}
break;
case NM_TYPE_INPUT:
2015-06-13 22:42:22 +00:00
nm_string_inputbox(item->w, item->x, item->y - (line_spacing * scroll_offset), item->text, is_current);
break;
case NM_TYPE_CHECK:
2015-06-13 22:42:22 +00:00
nm_string(item->w, item->x, item->y - (line_spacing * scroll_offset), item->text, tabs_flag);
if (item->value)
2015-06-13 22:42:22 +00:00
nm_rstring(item->right_offset, item->x, item->y - (line_spacing * scroll_offset), CHECKED_CHECK_BOX);
else
2015-06-13 22:42:22 +00:00
nm_rstring(item->right_offset, item->x, item->y - (line_spacing * scroll_offset), NORMAL_CHECK_BOX);
break;
case NM_TYPE_RADIO:
2015-06-13 22:42:22 +00:00
nm_string(item->w, item->x, item->y - (line_spacing * scroll_offset), item->text, tabs_flag);
if (item->value)
2015-06-13 22:42:22 +00:00
nm_rstring(item->right_offset, item->x, item->y - (line_spacing * scroll_offset), CHECKED_RADIO_BOX);
else
2015-06-13 22:42:22 +00:00
nm_rstring(item->right_offset, item->x, item->y - (line_spacing * scroll_offset), NORMAL_RADIO_BOX);
break;
case NM_TYPE_NUMBER:
{
2015-03-12 02:21:20 +00:00
char text[sizeof("-2147483647")];
if (item->value < item->min_value) item->value=item->min_value;
if (item->value > item->max_value) item->value=item->max_value;
2015-06-13 22:42:22 +00:00
nm_string(item->w, item->x, item->y - (line_spacing * scroll_offset), item->text, tabs_flag);
2015-03-12 02:21:20 +00:00
snprintf(text, sizeof(text), "%d", item->value );
2015-06-13 22:42:22 +00:00
nm_rstring(item->right_offset,item->x, item->y - (line_spacing * scroll_offset), text);
2006-03-20 17:12:09 +00:00
}
break;
2006-03-20 17:12:09 +00:00
}
}
const char *Newmenu_allowed_chars=NULL;
2006-03-20 17:12:09 +00:00
//returns true if char is allowed
2015-03-12 02:21:20 +00:00
static bool char_disallowed(char c)
2006-03-20 17:12:09 +00:00
{
const char *p = Newmenu_allowed_chars;
2006-03-20 17:12:09 +00:00
if (!p)
2015-03-12 02:21:20 +00:00
return false;
for (uint8_t a, b; (a = p[0]) && (b = p[1]); p += 2)
{
if (likely(c >= a && c <= b))
return false;
2006-03-20 17:12:09 +00:00
}
2015-03-12 02:21:20 +00:00
return true;
}
2006-03-20 17:12:09 +00:00
2015-03-12 02:21:20 +00:00
static bool char_allowed(char c)
{
return !char_disallowed(c);
2006-03-20 17:12:09 +00:00
}
2013-10-27 22:00:14 +00:00
static void strip_end_whitespace( char * text )
2006-03-20 17:12:09 +00:00
{
2015-03-12 02:21:20 +00:00
char *ns = text;
for (char c; (c = *text);)
{
++ text;
if (!isspace(static_cast<unsigned>(c)))
ns = text;
2006-03-20 17:12:09 +00:00
}
2015-03-12 02:21:20 +00:00
*ns = 0;
2006-03-20 17:12:09 +00:00
}
2015-03-12 02:21:21 +00:00
int newmenu_do2( const char * title, const char * subtitle, uint_fast32_t nitems, newmenu_item * item, newmenu_subfunction_t<void>::type subfunction, void *userdata, int citem, const char * filename )
2006-03-20 17:12:09 +00:00
{
newmenu *menu;
window *wind;
int rval = -1;
menu = newmenu_do3( title, subtitle, nitems, item, subfunction, userdata, citem, filename );
if (!menu)
return -1;
menu->rval = &rval;
wind = menu->wind; // avoid dereferencing a freed 'menu'
// newmenu_do2 and simpler get their own event loop
// This is so the caller doesn't have to provide a callback that responds to EVENT_NEWMENU_SELECTED
while (window_exists(wind))
event_process();
return rval;
2006-03-20 17:12:09 +00:00
}
2015-01-17 18:31:41 +00:00
static void swap_menu_item_entries(newmenu_item &a, newmenu_item &b)
{
using std::swap;
swap(a.text, b.text);
swap(a.value, b.value);
}
static int newmenu_save_selection_key(newmenu *menu, const d_event &event)
{
auto k = event_key_get(event);
switch(k)
{
case KEY_SHIFTED+KEY_UP:
if (menu->citem > 0)
{
auto &a = menu->items[menu->citem];
auto &b = menu->items[-- menu->citem];
swap_menu_item_entries(a, b);
}
break;
case KEY_SHIFTED+KEY_DOWN:
if (menu->citem < (menu->nitems - 1))
{
auto &a = menu->items[menu->citem];
auto &b = menu->items[++ menu->citem];
swap_menu_item_entries(a, b);
}
break;
}
return 0;
}
static int newmenu_save_selection_handler(newmenu *menu, const d_event &event, const unused_newmenu_userdata_t *)
{
switch(event.type)
{
case EVENT_KEY_COMMAND:
return newmenu_save_selection_key(menu, event);
default:
break;
}
return 0;
}
// Basically the same as do2 but sets reorderitems flag for weapon priority menu a bit redundant to get lose of a global variable but oh well...
2015-03-12 02:21:21 +00:00
void newmenu_doreorder( const char * title, const char * subtitle, uint_fast32_t nitems, newmenu_item * item)
{
2015-03-22 18:49:20 +00:00
newmenu_do2(title, subtitle, nitems, item, newmenu_save_selection_handler, unused_newmenu_userdata, 0, nullptr);
}
2006-03-20 17:12:09 +00:00
#ifdef NEWMENU_MOUSE
2013-12-04 02:47:30 +00:00
static const ubyte Hack_DblClick_MenuMode=0;
2006-03-20 17:12:09 +00:00
#endif
newmenu_item *newmenu_get_items(newmenu *menu)
{
return menu->items;
}
int newmenu_get_nitems(newmenu *menu)
{
return menu->nitems;
}
int newmenu_get_citem(newmenu *menu)
{
return menu->citem;
}
window *newmenu_get_window(newmenu *menu)
{
return menu->wind;
}
2015-03-12 02:21:21 +00:00
namespace {
struct step_down
{
template <typename T>
T operator()(T &t, int i) const
{
return i > 0 ? ++t : --t;
}
};
struct step_up
{
template <typename T>
T operator()(T &t, int i) const
{
return i > 0 ? --t : ++t;
}
};
}
template <typename S, typename O>
2015-03-12 02:21:21 +00:00
static void update_menu_position(newmenu &menu, newmenu_item *const stop, int_fast32_t amount, S step, O overflow)
2015-03-12 02:21:21 +00:00
{
auto icitem = menu.citem;
auto pcitem = &menu.items[icitem];
do // count until we reached a non NM_TYPE_TEXT item and reached our amount
{
2015-03-12 02:21:21 +00:00
if (pcitem == stop)
2015-03-12 02:21:21 +00:00
break;
step(icitem, 1);
step(pcitem, 1);
if (menu.is_scroll_box) // update scroll_offset as we go
{
menu.last_scroll_check = -1;
if (overflow(icitem))
step(menu.scroll_offset, 1);
}
} while (-- amount > 0 || pcitem->type == NM_TYPE_TEXT);
menu.citem = icitem;
}
2013-10-27 22:00:14 +00:00
static void newmenu_scroll(newmenu *menu, int amount)
{
if (amount == 0) // nothing to do for us
return;
if (menu->all_text)
{
menu->scroll_offset += amount;
if (menu->scroll_offset < 0)
menu->scroll_offset = 0;
if (menu->max_on_menu+menu->scroll_offset > menu->nitems)
menu->scroll_offset = menu->nitems-menu->max_on_menu;
return;
}
2015-03-12 02:21:21 +00:00
const auto &range = menu->item_range();
const auto predicate = [](const newmenu_item &n) {
return n.type != NM_TYPE_TEXT;
};
const auto first = std::find_if(range.begin(), range.end(), predicate);
if (first == range.end())
return;
2015-03-12 02:21:21 +00:00
const auto rlast = std::find_if(range.rbegin(), std::reverse_iterator<newmenu_item *>(first), predicate).base();
if (first == rlast) // nothing to do for us
return;
const auto last = std::prev(rlast);
auto citem = &menu->items[menu->citem];
if (citem == last && amount == 1) // if citem == last item and we want to go down one step, go to first item
{
newmenu_scroll(menu, -menu->nitems);
return;
}
2015-03-12 02:21:21 +00:00
if (citem == first && amount == -1) // if citem == first item and we want to go up one step, go to last item
{
newmenu_scroll(menu, menu->nitems);
return;
}
if (amount > 0) // down the list
{
2015-03-12 02:21:21 +00:00
const auto overflow = [menu](int icitem) {
return icitem + 4 >= menu->max_on_menu + menu->scroll_offset && menu->scroll_offset < menu->nitems - menu->max_on_menu;
};
update_menu_position(*menu, last, amount, step_down(), overflow);
}
else if (amount < 0) // up the list
{
2015-03-12 02:21:21 +00:00
const auto overflow = [menu](int icitem) {
return icitem - 4 < menu->scroll_offset && menu->scroll_offset > 0;
};
update_menu_position(*menu, first, std::abs(amount), step_up(), overflow);
}
}
2014-10-04 21:47:13 +00:00
static window_event_result newmenu_mouse(window *wind,const d_event &event, newmenu *menu, int button)
{
int old_choice, i, mx=0, my=0, mz=0, x1 = 0, x2, y1, y2, changed = 0;
2015-01-17 18:31:40 +00:00
grs_canvas *menu_canvas = &window_get_canvas(*wind), *save_canvas = grd_curcanv;
switch (button)
{
case MBTN_LEFT:
{
gr_set_current_canvas(menu_canvas);
old_choice = menu->citem;
2015-06-13 22:42:20 +00:00
const auto &&fspacx = FSPACX();
2014-10-04 21:47:13 +00:00
if ((event.type == EVENT_MOUSE_BUTTON_DOWN) && !menu->all_text)
{
mouse_get_pos(&mx, &my, &mz);
2015-06-13 22:42:22 +00:00
const int line_spacing = static_cast<int>(LINE_SPACING);
for (i=menu->scroll_offset; i<menu->max_on_menu+menu->scroll_offset; i++ ) {
2015-06-13 22:42:20 +00:00
x1 = grd_curcanv->cv_bitmap.bm_x + menu->items[i].x - fspacx(13) /*- menu->items[i].right_offset - 6*/;
x2 = x1 + menu->items[i].w + fspacx(13);
2015-06-13 22:42:22 +00:00
y1 = grd_curcanv->cv_bitmap.bm_y + menu->items[i].y - (line_spacing * menu->scroll_offset);
y2 = y1 + menu->items[i].h;
if (((mx > x1) && (mx < x2)) && ((my > y1) && (my < y2))) {
if (i != menu->citem) {
if(Hack_DblClick_MenuMode) menu->dblclick_flag = 0;
}
menu->citem = i;
2015-03-12 02:21:21 +00:00
auto &citem = menu->items[menu->citem];
switch (citem.type)
{
case NM_TYPE_CHECK:
2015-03-12 02:21:21 +00:00
if (citem.value)
citem.value = 0;
else
2015-03-12 02:21:21 +00:00
citem.value = 1;
if (menu->is_scroll_box)
menu->last_scroll_check=-1;
changed = 1;
break;
case NM_TYPE_RADIO:
2015-03-12 02:21:21 +00:00
range_for (auto &i, menu->item_range())
{
if (&i != &citem && i.type == NM_TYPE_RADIO && i.group == citem.group && i.value)
2015-03-12 02:21:21 +00:00
{
2015-03-12 02:21:21 +00:00
i.value = 0;
changed = 1;
}
}
2015-03-12 02:21:21 +00:00
citem.value = 1;
break;
case NM_TYPE_TEXT:
menu->citem=old_choice;
menu->mouse_state=0;
break;
}
break;
}
}
}
if ( menu->mouse_state ) {
mouse_get_pos(&mx, &my, &mz);
// check possible scrollbar stuff first
if (menu->is_scroll_box) {
2015-09-29 02:41:22 +00:00
int ScrollAllow=0;
static fix64 ScrollTime=0;
if (ScrollTime + F1_0/5 < timer_query())
{
ScrollTime = timer_query();
ScrollAllow = 1;
}
if (menu->scroll_offset != 0) {
2015-09-29 02:41:22 +00:00
int arrow_width, arrow_height;
gr_get_string_size(UP_ARROW_MARKER, &arrow_width, &arrow_height, nullptr);
2015-06-13 22:42:20 +00:00
x1 = grd_curcanv->cv_bitmap.bm_x + BORDERX - fspacx(12);
y1 = grd_curcanv->cv_bitmap.bm_y + menu->items[menu->scroll_offset].y-(((int)LINE_SPACING)*menu->scroll_offset);
x2 = x1 + arrow_width;
y2 = y1 + arrow_height;
if (((mx > x1) && (mx < x2)) && ((my > y1) && (my < y2)) && ScrollAllow) {
newmenu_scroll(menu, -1);
}
}
if (menu->scroll_offset+menu->max_displayable<menu->nitems) {
2015-09-29 02:41:22 +00:00
int arrow_width, arrow_height;
gr_get_string_size(DOWN_ARROW_MARKER, &arrow_width, &arrow_height, nullptr);
2015-06-13 22:42:20 +00:00
x1 = grd_curcanv->cv_bitmap.bm_x + BORDERX - fspacx(12);
y1 = grd_curcanv->cv_bitmap.bm_y + menu->items[menu->scroll_offset+menu->max_displayable-1].y-(((int)LINE_SPACING)*menu->scroll_offset);
x2 = x1 + arrow_width;
y2 = y1 + arrow_height;
if (((mx > x1) && (mx < x2)) && ((my > y1) && (my < y2)) && ScrollAllow) {
newmenu_scroll(menu, 1);
}
}
}
2015-06-13 22:42:22 +00:00
const int line_spacing = static_cast<int>(LINE_SPACING);
for (i=menu->scroll_offset; i<menu->max_on_menu+menu->scroll_offset; i++ ) {
2015-06-13 22:42:20 +00:00
x1 = grd_curcanv->cv_bitmap.bm_x + menu->items[i].x - fspacx(13);
x2 = x1 + menu->items[i].w + fspacx(13);
2015-06-13 22:42:22 +00:00
y1 = grd_curcanv->cv_bitmap.bm_y + menu->items[i].y - (line_spacing * menu->scroll_offset);
y2 = y1 + menu->items[i].h;
if (((mx > x1) && (mx < x2)) && ((my > y1) && (my < y2)) && (menu->items[i].type != NM_TYPE_TEXT) ) {
if (i != menu->citem) {
if(Hack_DblClick_MenuMode) menu->dblclick_flag = 0;
}
menu->citem = i;
2015-03-12 02:21:21 +00:00
auto &citem = menu->items[menu->citem];
if (citem.type == NM_TYPE_SLIDER)
{
char slider_text[NM_MAX_TEXT_LEN+1], *p, *s1;
2015-03-12 02:21:21 +00:00
strcpy(slider_text, citem.saved_text);
p = strchr(slider_text, '\t');
if (p) {
*p = '\0';
s1 = p+1;
}
if (p) {
2015-09-29 02:41:22 +00:00
int slider_width, sleft_width, sright_width, smiddle_width;
gr_get_string_size(s1, &slider_width, nullptr, nullptr);
gr_get_string_size(SLIDER_LEFT, &sleft_width, nullptr, nullptr);
gr_get_string_size(SLIDER_RIGHT, &sright_width, nullptr, nullptr);
gr_get_string_size(SLIDER_MIDDLE, &smiddle_width, nullptr, nullptr);
2015-03-12 02:21:21 +00:00
x1 = grd_curcanv->cv_bitmap.bm_x + citem.x + citem.w - slider_width;
x2 = x1 + slider_width + sright_width;
2015-03-12 02:21:21 +00:00
if (mx > x1 && mx < (x1 + sleft_width) && citem.value != citem.min_value)
{
citem.value = citem.min_value;
changed = 1;
2015-03-12 02:21:21 +00:00
}
else if (mx < x2 && mx > (x2 - sright_width) && citem.value != citem.max_value)
{
citem.value = citem.max_value;
changed = 1;
} else if ( (mx > (x1 + sleft_width)) && (mx < (x2 - sright_width)) ) {
int num_values, value_width, new_value;
2015-03-12 02:21:21 +00:00
num_values = citem.max_value - citem.min_value + 1;
value_width = (slider_width - sleft_width - sright_width) / num_values;
new_value = (mx - x1 - sleft_width) / value_width;
2015-03-12 02:21:21 +00:00
if (citem.value != new_value)
{
citem.value = new_value;
changed = 1;
}
}
*p = '\t';
}
}
if (menu->citem == old_choice)
break;
2015-03-12 02:21:21 +00:00
if (citem.type == NM_TYPE_INPUT && menu->citem != old_choice)
citem.value = -1;
if ((old_choice>-1) && (menu->items[old_choice].type==NM_TYPE_INPUT_MENU) && (old_choice!=menu->citem)) {
menu->items[old_choice].group=0;
strcpy(menu->items[old_choice].text, menu->items[old_choice].saved_text );
menu->items[old_choice].value = -1;
}
break;
}
}
}
2014-10-04 21:47:13 +00:00
if ((event.type == EVENT_MOUSE_BUTTON_UP) && !menu->all_text && (menu->citem != -1) && (menu->items[menu->citem].type == NM_TYPE_MENU) )
{
mouse_get_pos(&mx, &my, &mz);
2015-06-13 22:42:22 +00:00
const int line_spacing = static_cast<int>(LINE_SPACING);
for (i=menu->scroll_offset; i<menu->max_on_menu+menu->scroll_offset; i++ ) {
2015-06-13 22:42:20 +00:00
x1 = grd_curcanv->cv_bitmap.bm_x + menu->items[i].x - fspacx(13);
x2 = x1 + menu->items[i].w + fspacx(13);
2015-06-13 22:42:22 +00:00
y1 = grd_curcanv->cv_bitmap.bm_y + menu->items[i].y - (line_spacing * menu->scroll_offset);
y2 = y1 + menu->items[i].h;
if (((mx > x1) && (mx < x2)) && ((my > y1) && (my < y2))) {
if (Hack_DblClick_MenuMode) {
if (menu->dblclick_flag)
{
// Tell callback, allow staying in menu
const d_select_event selected{menu->citem};
2014-10-04 21:47:13 +00:00
if (menu->subfunction && (*menu->subfunction)(menu, selected, menu->userdata))
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
if (menu->rval)
*menu->rval = menu->citem;
gr_set_current_canvas(save_canvas);
2014-08-06 02:10:49 +00:00
return window_event_result::close;
}
else menu->dblclick_flag = 1;
}
else
{
// Tell callback, allow staying in menu
const d_select_event selected{menu->citem};
2014-10-04 21:47:13 +00:00
if (menu->subfunction && (*menu->subfunction)(menu, selected, menu->userdata))
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
if (menu->rval)
*menu->rval = menu->citem;
gr_set_current_canvas(save_canvas);
2014-08-06 02:10:49 +00:00
return window_event_result::close;
}
}
}
}
2015-03-12 02:21:21 +00:00
if (event.type == EVENT_MOUSE_BUTTON_UP && menu->citem > -1)
{
2015-03-12 02:21:21 +00:00
auto &citem = menu->items[menu->citem];
if (citem.type == NM_TYPE_INPUT_MENU && citem.group == 0)
{
citem.group = 1;
if (!d_stricmp(citem.saved_text, TXT_EMPTY))
{
citem.text[0] = 0;
citem.value = -1;
} else {
strip_end_whitespace(citem.text);
}
}
}
gr_set_current_canvas(save_canvas);
if (changed && menu->subfunction)
{
const d_change_event changed{menu->citem};
2014-10-04 21:47:13 +00:00
(*menu->subfunction)(menu, changed, menu->userdata);
}
break;
}
case MBTN_RIGHT:
if (menu->mouse_state)
{
2015-03-12 02:21:21 +00:00
if (!(menu->citem > -1))
return window_event_result::close;
auto &citem = menu->items[menu->citem];
if (citem.type == NM_TYPE_INPUT_MENU && citem.group == 1)
{
citem.group = 0;
strcpy(citem.text, citem.saved_text);
citem.value = -1;
} else {
2014-08-06 02:10:49 +00:00
return window_event_result::close;
}
}
break;
case MBTN_Z_UP:
if (menu->mouse_state)
newmenu_scroll(menu, -1);
break;
case MBTN_Z_DOWN:
if (menu->mouse_state)
newmenu_scroll(menu, 1);
break;
}
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
}
2015-01-18 01:58:33 +00:00
static window_event_result newmenu_key_command(window *, const d_event &event, newmenu *menu)
2006-03-20 17:12:09 +00:00
{
int k = event_key_get(event);
2015-03-12 02:21:21 +00:00
int old_choice;
int changed = 0;
2014-08-06 02:10:49 +00:00
window_event_result rval = window_event_result::handled;
if (keyd_pressed[KEY_NUMLOCK])
{
switch( k )
{
case KEY_PAD0: k = KEY_0; break;
case KEY_PAD1: k = KEY_1; break;
case KEY_PAD2: k = KEY_2; break;
case KEY_PAD3: k = KEY_3; break;
case KEY_PAD4: k = KEY_4; break;
case KEY_PAD5: k = KEY_5; break;
case KEY_PAD6: k = KEY_6; break;
case KEY_PAD7: k = KEY_7; break;
case KEY_PAD8: k = KEY_8; break;
case KEY_PAD9: k = KEY_9; break;
case KEY_PADPERIOD: k = KEY_PERIOD; break;
}
}
old_choice = menu->citem;
2015-03-12 02:21:21 +00:00
auto &citem = menu->items[menu->citem];
switch( k ) {
case KEY_HOME:
case KEY_PAD7:
newmenu_scroll(menu, -menu->nitems);
break;
case KEY_END:
case KEY_PAD1:
newmenu_scroll(menu, menu->nitems);
break;
2006-03-20 17:12:09 +00:00
case KEY_TAB + KEY_SHIFTED:
case KEY_UP:
case KEY_PAGEUP:
2006-03-20 17:12:09 +00:00
case KEY_PAD8:
if (k == KEY_PAGEUP)
newmenu_scroll(menu, -10);
else
newmenu_scroll(menu, -1);
2015-03-12 02:21:21 +00:00
if (citem.type == NM_TYPE_INPUT && menu->citem != old_choice)
citem.value = -1;
if ((old_choice>-1) && (menu->items[old_choice].type==NM_TYPE_INPUT_MENU) && (old_choice!=menu->citem)) {
menu->items[old_choice].group=0;
strcpy(menu->items[old_choice].text, menu->items[old_choice].saved_text );
menu->items[old_choice].value = -1;
2006-03-20 17:12:09 +00:00
}
break;
case KEY_TAB:
case KEY_DOWN:
case KEY_PAGEDOWN:
case KEY_PAD2:
if (k == KEY_PAGEDOWN)
newmenu_scroll(menu, 10);
else
newmenu_scroll(menu, 1);
2015-03-12 02:21:21 +00:00
if (citem.type == NM_TYPE_INPUT && menu->citem != old_choice)
citem.value = -1;
if ( (old_choice>-1) && (menu->items[old_choice].type==NM_TYPE_INPUT_MENU) && (old_choice!=menu->citem)) {
menu->items[old_choice].group=0;
strcpy(menu->items[old_choice].text, menu->items[old_choice].saved_text );
menu->items[old_choice].value = -1;
2006-03-20 17:12:09 +00:00
}
break;
case KEY_SPACEBAR:
if ( menu->citem > -1 ) {
2015-03-12 02:21:21 +00:00
switch (citem.type)
{
case NM_TYPE_MENU:
case NM_TYPE_INPUT:
case NM_TYPE_INPUT_MENU:
break;
case NM_TYPE_CHECK:
2015-03-12 02:21:21 +00:00
if (citem.value)
citem.value = 0;
else
2015-03-12 02:21:21 +00:00
citem.value = 1;
if (menu->is_scroll_box)
{
if (menu->citem==(menu->max_on_menu+menu->scroll_offset-1) || menu->citem==menu->scroll_offset)
{
menu->last_scroll_check=-1;
}
}
changed = 1;
break;
case NM_TYPE_RADIO:
2015-03-12 02:21:21 +00:00
range_for (auto &i, menu->item_range())
{
if (&i != &citem && i.type == NM_TYPE_RADIO && i.group == citem.group && i.value)
{
i.value = 0;
}
2006-03-20 17:12:09 +00:00
}
2015-03-12 02:21:21 +00:00
citem.value = 1;
changed = 1;
break;
}
2006-03-20 17:12:09 +00:00
}
break;
case KEY_ENTER:
case KEY_PADENTER:
2015-03-12 02:21:21 +00:00
if (menu->citem > -1 && citem.type == NM_TYPE_INPUT_MENU && citem.group == 0) {
citem.group = 1;
if (!d_stricmp(citem.saved_text, TXT_EMPTY))
{
2015-03-12 02:21:21 +00:00
citem.text[0] = 0;
citem.value = -1;
} else {
2015-03-12 02:21:21 +00:00
strip_end_whitespace(citem.text);
2006-03-20 17:12:09 +00:00
}
} else
{
2015-03-12 02:21:21 +00:00
if (citem.type == NM_TYPE_INPUT_MENU)
citem.group = 0; // go out of editing mode
// Tell callback, allow staying in menu
const d_select_event selected{menu->citem};
2014-10-04 21:47:13 +00:00
if (menu->subfunction && (*menu->subfunction)(menu, selected, menu->userdata))
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
if (menu->rval)
*menu->rval = menu->citem;
2014-08-06 02:10:49 +00:00
return window_event_result::close;
}
2006-03-20 17:12:09 +00:00
break;
case KEY_ESC:
2015-03-12 02:21:21 +00:00
if (menu->citem > -1 && citem.type == NM_TYPE_INPUT_MENU && citem.group == 1)
{
citem.group = 0;
strcpy(citem.text, citem.saved_text);
citem.value = -1;
2006-03-20 17:12:09 +00:00
} else {
2014-08-06 02:10:49 +00:00
return window_event_result::close;
2006-03-20 17:12:09 +00:00
}
break;
default:
2014-08-06 02:10:49 +00:00
rval = window_event_result::ignored;
break;
}
if ( menu->citem > -1 ) {
// Alerting callback of every keypress for NM_TYPE_INPUT. Alternatively, just respond to EVENT_NEWMENU_SELECTED
2015-03-12 02:21:21 +00:00
if ( ((citem.type == NM_TYPE_INPUT)||((citem.type == NM_TYPE_INPUT_MENU)&&(citem.group == 1)) )&& (old_choice == menu->citem) ) {
if ( k==KEY_LEFT || k==KEY_BACKSP || k==KEY_PAD4 ) {
2015-03-12 02:21:21 +00:00
if (citem.value == -1) citem.value = strlen(citem.text);
if (citem.value > 0)
citem.value--;
citem.text[citem.value] = 0;
2015-03-12 02:21:21 +00:00
if (citem.type == NM_TYPE_INPUT)
changed = 1;
2014-08-06 02:10:49 +00:00
rval = window_event_result::handled;
} else {
2015-03-12 02:21:20 +00:00
auto ascii = key_ascii();
2015-03-12 02:21:21 +00:00
if ((ascii < 255 ) && (citem.value < citem.text_len ))
{
2015-03-12 02:21:21 +00:00
if (citem.value == -1) {
citem.value = 0;
}
2015-03-12 02:21:20 +00:00
if (char_allowed(ascii) || (ascii == ' ' && char_allowed(ascii = '_')))
{
2015-03-12 02:21:21 +00:00
citem.text[citem.value++] = ascii;
citem.text[citem.value] = 0;
2015-03-12 02:21:21 +00:00
if (citem.type == NM_TYPE_INPUT)
changed = 1;
2006-03-20 17:12:09 +00:00
}
}
2006-03-20 17:12:09 +00:00
}
}
2015-03-12 02:21:21 +00:00
else if ((citem.type != NM_TYPE_INPUT) && (citem.type != NM_TYPE_INPUT_MENU) )
{
2015-03-12 02:21:20 +00:00
auto ascii = key_ascii();
if (ascii < 255 ) {
int choice1 = menu->citem;
ascii = toupper(ascii);
do {
int i,ch;
choice1++;
if (choice1 >= menu->nitems )
choice1=0;
for (i=0;(ch=menu->items[choice1].text[i])!=0 && ch==' ';i++);
if ( ( (menu->items[choice1].type==NM_TYPE_MENU) ||
(menu->items[choice1].type==NM_TYPE_CHECK) ||
(menu->items[choice1].type==NM_TYPE_RADIO) ||
(menu->items[choice1].type==NM_TYPE_NUMBER) ||
(menu->items[choice1].type==NM_TYPE_SLIDER) )
&& (ascii==toupper(ch)) )
{
k = 0;
menu->citem = choice1;
}
while (menu->citem+4>=menu->max_on_menu+menu->scroll_offset && menu->scroll_offset < menu->nitems-menu->max_on_menu)
menu->scroll_offset++;
while (menu->citem-4<menu->scroll_offset && menu->scroll_offset > 0)
menu->scroll_offset--;
} while (choice1 != menu->citem );
2006-03-20 17:12:09 +00:00
}
}
2015-03-12 02:21:21 +00:00
if ( (citem.type == NM_TYPE_NUMBER) || (citem.type == NM_TYPE_SLIDER))
{
switch( k ) {
case KEY_LEFT:
case KEY_PAD4:
2015-03-12 02:21:21 +00:00
citem.value -= 1;
changed = 1;
2014-08-06 02:10:49 +00:00
rval = window_event_result::handled;
break;
case KEY_RIGHT:
case KEY_PAD6:
2015-03-12 02:21:21 +00:00
citem.value++;
changed = 1;
2014-08-06 02:10:49 +00:00
rval = window_event_result::handled;
break;
case KEY_SPACEBAR:
2015-03-12 02:21:21 +00:00
citem.value += 10;
changed = 1;
2014-08-06 02:10:49 +00:00
rval = window_event_result::handled;
break;
case KEY_BACKSP:
2015-03-12 02:21:21 +00:00
citem.value -= 10;
changed = 1;
2014-08-06 02:10:49 +00:00
rval = window_event_result::handled;
break;
}
2015-03-12 02:21:21 +00:00
if (citem.value < citem.min_value) citem.value = citem.min_value;
if (citem.value > citem.max_value) citem.value = citem.max_value;
2006-03-20 17:12:09 +00:00
}
}
if (changed && menu->subfunction)
{
const d_change_event changed{menu->citem};
2014-10-04 21:47:13 +00:00
(*menu->subfunction)(menu, changed, menu->userdata);
}
return rval;
}
2013-10-27 22:00:14 +00:00
static void newmenu_create_structure( newmenu *menu )
{
2015-03-12 02:21:21 +00:00
int j,aw, tw, th, twidth,fm,right_offset;
int nmenus, nothers;
grs_canvas *save_canvas;
save_canvas = grd_curcanv;
gr_set_current_canvas(NULL);
2014-07-20 03:48:27 +00:00
const grs_font *save_font = grd_curcanv->cv_font;
tw = th = 0;
if ( menu->title ) {
gr_set_curfont(HUGE_FONT);
2015-09-29 02:41:22 +00:00
int string_width, string_height;
gr_get_string_size(menu->title, &string_width, &string_height, nullptr);
tw = string_width;
th = string_height;
}
if ( menu->subtitle ) {
gr_set_curfont(MEDIUM3_FONT);
2015-09-29 02:41:22 +00:00
int string_width, string_height;
gr_get_string_size(menu->subtitle, &string_width, &string_height, nullptr);
if (string_width > tw )
tw = string_width;
th += string_height;
}
th += FSPACY(5); //put some space between titles & body
gr_set_curfont(menu->tiny_mode?GAME_FONT:MEDIUM1_FONT);
menu->w = aw = 0;
menu->h = th;
nmenus = nothers = 0;
2015-06-13 22:42:20 +00:00
const auto &&fspacx = FSPACX();
2015-06-13 22:42:20 +00:00
const auto &&fspacy = FSPACY();
// Find menu height & width (store in w,h)
2015-03-12 02:21:21 +00:00
range_for (auto &i, menu->item_range())
{
i.y = menu->h;
2015-09-29 02:41:22 +00:00
int string_width, string_height, average_width;
2015-03-12 02:21:21 +00:00
gr_get_string_size(i.text,&string_width,&string_height,&average_width);
i.right_offset = 0;
2015-03-12 02:21:21 +00:00
i.saved_text[0] = '\0';
2015-03-12 02:21:21 +00:00
if (i.type == NM_TYPE_SLIDER)
{
2015-09-29 02:41:22 +00:00
int index,w1;
nothers++;
index = snprintf (i.saved_text.data(), i.saved_text.size(), "%s", SLIDER_LEFT);
for (uint_fast32_t j = (i.max_value - i.min_value + 1); j--;)
{
index += snprintf(i.saved_text.data() + index, i.saved_text.size() - index, "%s", SLIDER_MIDDLE);
}
index += snprintf(i.saved_text.data() + index, i.saved_text.size() - index, "%s", SLIDER_RIGHT);
2015-09-29 02:41:22 +00:00
gr_get_string_size(i.saved_text.data(), &w1, nullptr, nullptr);
string_width += w1 + aw;
2006-03-20 17:12:09 +00:00
}
2015-03-12 02:21:21 +00:00
if (i.type == NM_TYPE_MENU)
{
nmenus++;
}
2006-03-20 17:12:09 +00:00
2015-03-12 02:21:21 +00:00
if (i.type == NM_TYPE_CHECK)
{
2015-09-29 02:41:22 +00:00
int w1;
nothers++;
2015-09-29 02:41:22 +00:00
gr_get_string_size(NORMAL_CHECK_BOX, &w1, nullptr, nullptr);
2015-03-12 02:21:21 +00:00
i.right_offset = w1;
2015-09-29 02:41:22 +00:00
gr_get_string_size(CHECKED_CHECK_BOX, &w1, nullptr, nullptr);
2015-03-12 02:21:21 +00:00
if (w1 > i.right_offset)
i.right_offset = w1;
}
2015-03-12 02:21:21 +00:00
if (i.type == NM_TYPE_RADIO)
{
2015-09-29 02:41:22 +00:00
int w1;
nothers++;
2015-09-29 02:41:22 +00:00
gr_get_string_size(NORMAL_RADIO_BOX, &w1, nullptr, nullptr);
2015-03-12 02:21:21 +00:00
i.right_offset = w1;
2015-09-29 02:41:22 +00:00
gr_get_string_size(CHECKED_RADIO_BOX, &w1, nullptr, nullptr);
2015-03-12 02:21:21 +00:00
if (w1 > i.right_offset)
i.right_offset = w1;
}
2015-03-12 02:21:21 +00:00
if (i.type == NM_TYPE_NUMBER)
{
2015-09-29 02:41:22 +00:00
int w1;
char test_text[20];
nothers++;
2015-03-12 02:21:21 +00:00
sprintf (test_text, "%d", i.max_value);
2015-09-29 02:41:22 +00:00
gr_get_string_size(test_text, &w1, nullptr, nullptr);
2015-03-12 02:21:21 +00:00
i.right_offset = w1;
sprintf (test_text, "%d", i.min_value);
2015-09-29 02:41:22 +00:00
gr_get_string_size(test_text, &w1, nullptr, nullptr);
2015-03-12 02:21:21 +00:00
if (w1 > i.right_offset)
i.right_offset = w1;
2006-03-20 17:12:09 +00:00
}
2015-03-12 02:21:21 +00:00
if (i.type == NM_TYPE_INPUT || i.type == NM_TYPE_INPUT_MENU)
{
i.saved_text.copy_if(i.text);
2015-06-13 22:42:20 +00:00
string_width = i.text_len * fspacx(8) + i.text_len;
2015-03-12 02:21:21 +00:00
if (i.type == NM_TYPE_INPUT && string_width > MAX_TEXT_WIDTH)
string_width = MAX_TEXT_WIDTH;
2015-03-12 02:21:21 +00:00
i.value = -1;
i.group = 0;
if (i.type == NM_TYPE_INPUT_MENU)
nmenus++;
else
nothers++;
}
2015-03-12 02:21:21 +00:00
i.w = string_width;
i.h = string_height;
if ( string_width > menu->w )
menu->w = string_width; // Save maximum width
if ( average_width > aw )
aw = average_width;
2015-06-13 22:42:20 +00:00
menu->h += string_height + fspacy(1); // Find the height of all strings
}
2015-03-12 02:21:21 +00:00
if (menu->nitems > menu->max_on_menu)
{
menu->is_scroll_box=1;
menu->h = th+(LINE_SPACING*menu->max_on_menu);
menu->max_displayable=menu->max_on_menu;
// if our last citem was > menu->max_on_menu, make sure we re-scroll when we call this menu again
if (menu->citem > menu->max_on_menu-4)
{
menu->scroll_offset = menu->citem - (menu->max_on_menu-4);
if (menu->scroll_offset + menu->max_on_menu > menu->nitems)
menu->scroll_offset = menu->nitems - menu->max_on_menu;
}
}
else
{
menu->is_scroll_box=0;
2015-03-12 02:21:21 +00:00
menu->max_on_menu = menu->nitems;
}
right_offset=0;
2015-03-12 02:21:21 +00:00
range_for (auto &i, menu->item_range())
{
i.w = menu->w;
if (right_offset < i.right_offset)
right_offset = i.right_offset;
}
menu->w += right_offset;
twidth = 0;
if ( tw > menu->w ) {
twidth = ( tw - menu->w )/2;
menu->w = tw;
}
// Find min point of menu border
menu->w += BORDERX*2;
menu->h += BORDERY*2;
menu->x = (GWIDTH-menu->w)/2;
menu->y = (GHEIGHT-menu->h)/2;
if ( menu->x < 0 ) menu->x = 0;
if ( menu->y < 0 ) menu->y = 0;
nm_draw_background1( menu->filename );
// Update all item's x & y values.
2015-03-12 02:21:21 +00:00
range_for (auto &i, menu->item_range())
{
i.x = BORDERX + twidth + right_offset;
i.y += BORDERY;
if (i.type == NM_TYPE_RADIO) {
fm = -1; // find first marked one
for ( j=0; j<menu->nitems; j++ ) {
2015-03-12 02:21:21 +00:00
if (menu->items[j].type == NM_TYPE_RADIO && menu->items[j].group == i.group) {
if (fm==-1 && menu->items[j].value)
fm = j;
menu->items[j].value = 0;
}
}
if ( fm>=0 )
menu->items[fm].value=1;
else
2015-03-12 02:21:21 +00:00
i.value=1;
}
2006-03-20 17:12:09 +00:00
}
if (menu->citem != -1)
{
if (menu->citem < 0 ) menu->citem = 0;
if (menu->citem > menu->nitems-1 ) menu->citem = menu->nitems-1;
2006-03-20 17:12:09 +00:00
#ifdef NEWMENU_MOUSE
menu->dblclick_flag = 1;
#endif
2015-03-12 02:21:21 +00:00
uint_fast32_t i = 0;
while ( menu->items[menu->citem].type==NM_TYPE_TEXT ) {
menu->citem++;
i++;
if (menu->citem >= menu->nitems ) {
menu->citem=0;
}
if (i > menu->nitems ) {
menu->citem=0;
menu->all_text=1;
break;
}
}
}
2006-03-20 17:12:09 +00:00
menu->mouse_state = 0;
menu->swidth = SWIDTH;
menu->sheight = SHEIGHT;
menu->fntscalex = FNTScaleX;
menu->fntscaley = FNTScaleY;
gr_set_curfont(save_font);
gr_set_current_canvas(save_canvas);
}
2014-08-06 02:10:49 +00:00
static window_event_result newmenu_draw(window *wind, newmenu *menu)
{
2015-01-17 18:31:40 +00:00
grs_canvas *menu_canvas = &window_get_canvas(*wind), *save_canvas = grd_curcanv;
2011-09-26 23:31:19 +00:00
int th = 0, ty, sx, sy;
int i;
if (menu->swidth != SWIDTH || menu->sheight != SHEIGHT || menu->fntscalex != FNTScaleX || menu->fntscalex != FNTScaleY)
{
newmenu_create_structure ( menu );
if (menu_canvas)
{
2015-01-17 18:31:40 +00:00
gr_init_sub_canvas(*menu_canvas, grd_curscreen->sc_canvas, menu->x, menu->y, menu->w, menu->h);
}
}
gr_set_current_canvas( NULL );
nm_draw_background1(menu->filename);
if (menu->filename == NULL)
nm_draw_background(menu->x-(menu->is_scroll_box?FSPACX(5):0),menu->y,menu->x+menu->w,menu->y+menu->h);
gr_set_current_canvas( menu_canvas );
ty = BORDERY;
if ( menu->title ) {
gr_set_curfont(HUGE_FONT);
gr_set_fontcolor( BM_XRGB(31,31,31), -1 );
2015-09-29 02:41:22 +00:00
int string_width, string_height;
gr_get_string_size(menu->title, &string_width, &string_height, nullptr);
th = string_height;
2015-09-29 02:41:22 +00:00
gr_string(0x8000, ty, menu->title, string_width, string_height);
}
if ( menu->subtitle ) {
gr_set_curfont(MEDIUM3_FONT);
gr_set_fontcolor( BM_XRGB(21,21,21), -1 );
gr_string( 0x8000, ty+th, menu->subtitle );
}
gr_set_curfont(menu->tiny_mode?GAME_FONT:MEDIUM1_FONT);
// Redraw everything...
for (i=menu->scroll_offset; i<menu->max_displayable+menu->scroll_offset; i++ )
{
draw_item( &menu->items[i], (i==menu->citem && !menu->all_text),menu->tiny_mode, menu->tabs_flag, menu->scroll_offset );
}
if (menu->is_scroll_box)
{
menu->last_scroll_check=menu->scroll_offset;
gr_set_curfont(menu->tiny_mode?GAME_FONT:MEDIUM2_FONT);
2015-06-13 22:42:22 +00:00
const int line_spacing = static_cast<int>(LINE_SPACING);
sy = menu->items[menu->scroll_offset].y - (line_spacing * menu->scroll_offset);
2015-06-13 22:42:20 +00:00
const auto &&fspacx = FSPACX();
sx = BORDERX - fspacx(12);
if (menu->scroll_offset!=0)
gr_string( sx, sy, UP_ARROW_MARKER );
else
gr_string( sx, sy, " " );
2015-06-13 22:42:22 +00:00
sy = menu->items[menu->scroll_offset + menu->max_displayable - 1].y - (line_spacing * menu->scroll_offset);
2015-06-13 22:42:20 +00:00
sx = BORDERX - fspacx(12);
if (menu->scroll_offset+menu->max_displayable<menu->nitems)
gr_string( sx, sy, DOWN_ARROW_MARKER );
else
gr_string( sx, sy, " " );
}
{
d_event event;
event.type = EVENT_NEWMENU_DRAW;
if (menu->subfunction)
2014-10-04 21:47:13 +00:00
(*menu->subfunction)(menu, event, menu->userdata);
}
gr_set_current_canvas(save_canvas);
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
}
2014-10-04 21:47:13 +00:00
static window_event_result newmenu_handler(window *wind,const d_event &event, newmenu *menu)
{
2014-10-04 21:47:13 +00:00
if (event.type == EVENT_WINDOW_CLOSED)
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
if (menu->subfunction)
{
int rval = (*menu->subfunction)(menu, event, menu->userdata);
if (!window_exists(wind))
2014-08-06 02:10:49 +00:00
return window_event_result::handled; // some subfunction closed the window: bail!
if (rval)
{
if (rval < -1)
{
if (menu->rval)
*menu->rval = rval;
2014-08-06 02:10:49 +00:00
return window_event_result::close;
}
2014-08-06 02:10:49 +00:00
return window_event_result::handled; // event handled
}
}
2014-10-04 21:47:13 +00:00
switch (event.type)
{
case EVENT_WINDOW_ACTIVATED:
game_flush_inputs();
event_toggle_focus(0);
key_toggle_repeat(1);
break;
case EVENT_WINDOW_DEACTIVATED:
//event_toggle_focus(1); // No cursor recentering
#if defined(DXX_BUILD_DESCENT_I)
#define DXX_KEY_TOGGLE_REPEAT_VALUE 0
#elif defined(DXX_BUILD_DESCENT_II)
#define DXX_KEY_TOGGLE_REPEAT_VALUE 1
#endif
key_toggle_repeat(DXX_KEY_TOGGLE_REPEAT_VALUE);
menu->mouse_state = 0;
break;
case EVENT_MOUSE_BUTTON_DOWN:
case EVENT_MOUSE_BUTTON_UP:
{
int button = event_mouse_get_button(event);
2014-10-04 21:47:13 +00:00
menu->mouse_state = event.type == EVENT_MOUSE_BUTTON_DOWN;
return newmenu_mouse(wind, event, menu, button);
}
case EVENT_KEY_COMMAND:
return newmenu_key_command(wind, event, menu);
case EVENT_IDLE:
2015-07-04 21:01:17 +00:00
break;
case EVENT_WINDOW_DRAW:
return newmenu_draw(wind, menu);
case EVENT_WINDOW_CLOSE:
delete menu;
break;
default:
break;
}
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
}
2015-03-12 02:21:21 +00:00
newmenu *newmenu_do4( const char * title, const char * subtitle, uint_fast32_t nitems, newmenu_item * item, newmenu_subfunction subfunction, void *userdata, int citem, const char * filename, int TinyMode, int TabsFlag )
{
window *wind = NULL;
newmenu *menu = new newmenu{};
menu->citem = citem;
menu->scroll_offset = 0;
menu->last_scroll_check = -1;
menu->all_text = 0;
menu->is_scroll_box = 0;
menu->max_on_menu = TinyMode?MAXDISPLAYABLEITEMSTINY:MAXDISPLAYABLEITEMS;
menu->dblclick_flag = 0;
menu->title = title;
menu->subtitle = subtitle;
menu->nitems = nitems;
menu->subfunction = subfunction;
menu->items = item;
menu->filename = filename;
menu->tiny_mode = TinyMode;
menu->tabs_flag = TabsFlag;
menu->rval = NULL; // Default to not returning a value - respond to EVENT_NEWMENU_SELECTED instead
menu->userdata = userdata;
newmenu_free_background();
if (nitems < 1 )
{
delete menu;
return NULL;
}
menu->max_displayable=nitems;
//set_screen_mode(SCREEN_MENU); //hafta set the screen mode here or fonts might get changed/freed up if screen res changes
2010-07-29 08:30:46 +00:00
newmenu_create_structure(menu);
// Create the basic window
2014-01-07 20:59:22 +00:00
wind = window_create(&grd_curscreen->sc_canvas, menu->x, menu->y, menu->w, menu->h, newmenu_handler, menu);
if (!wind)
{
delete menu;
return NULL;
}
return menu;
2006-03-20 17:12:09 +00:00
}
int (vnm_messagebox_aN)(const char *title, const nm_messagebox_tie &tie, const char *format, ...)
2006-03-20 17:12:09 +00:00
{
va_list args;
char nm_text[MESSAGEBOX_TEXT_SIZE];
va_start(args, format);
2012-06-30 05:05:33 +00:00
vsnprintf(nm_text,sizeof(nm_text),format,args);
2006-03-20 17:12:09 +00:00
va_end(args);
return nm_messagebox_str(title, tie, nm_text);
}
2006-03-20 17:12:09 +00:00
int nm_messagebox_str(const char *title, const nm_messagebox_tie &tie, const char *str)
{
newmenu_item items[nm_messagebox_tie::maximum_arity];
for (unsigned i=0; i < tie.count(); ++i) {
const char *s = tie.string(i);
2015-01-18 01:58:31 +00:00
nm_set_item_menu(items[i], s);
}
return newmenu_do( title, str, tie.count(), items, unused_newmenu_subfunction, unused_newmenu_userdata );
2006-03-20 17:12:09 +00:00
}
// Example listbox callback function...
// int lb_callback( int * citem, int *nitems, char * items[], int *keypress )
// {
// int i;
//
2006-03-20 17:12:09 +00:00
// if ( *keypress = KEY_CTRLED+KEY_D ) {
// if ( *nitems > 1 ) {
// PHYSFS_delete(items[*citem]); // Delete the file
// for (i=*citem; i<*nitems-1; i++ ) {
// items[i] = items[i+1];
// }
// *nitems = *nitems - 1;
// d_free( items[*nitems] );
// items[*nitems] = NULL;
// return 1; // redraw;
// }
// *keypress = 0;
// }
2006-03-20 17:12:09 +00:00
// return 0;
// }
#define LB_ITEMS_ON_SCREEN 8
2014-09-07 18:06:59 +00:00
struct listbox : embed_window_pointer_t
2006-03-20 17:12:09 +00:00
{
const char *title;
2013-09-02 23:21:13 +00:00
const char **item;
2009-12-27 05:00:45 +00:00
int allow_abort_flag;
2014-12-05 02:45:22 +00:00
unsigned marquee_maxchars;
2014-10-04 21:47:13 +00:00
int (*listbox_callback)(listbox *lb,const d_event &event, void *userdata);
2015-03-12 02:21:21 +00:00
unsigned nitems;
int citem, first_item;
2014-12-05 02:45:22 +00:00
int marquee_charpos, marquee_scrollback;
2009-12-27 05:00:45 +00:00
int box_w, height, box_x, box_y, title_height;
short swidth, sheight; float fntscalex, fntscaley; // with these we check if resolution or fonts have changed so listbox structure can be recreated
int mouse_state;
2015-01-24 19:16:34 +00:00
fix64 marquee_lasttime; // to scroll text if string does not fit in box
void *userdata;
};
2013-09-02 23:21:13 +00:00
const char **listbox_get_items(listbox *lb)
{
return lb->item;
}
int listbox_get_nitems(listbox *lb)
{
return lb->nitems;
}
int listbox_get_citem(listbox *lb)
{
return lb->citem;
}
window *listbox_get_window(listbox *lb)
{
return lb->wind;
}
void listbox_delete_item(listbox *lb, int item)
{
int i;
Assert(item >= 0);
if (lb->nitems)
{
for (i=item; i<lb->nitems-1; i++ )
lb->item[i] = lb->item[i+1];
lb->nitems--;
lb->item[lb->nitems] = NULL;
if (lb->citem >= lb->nitems)
lb->citem = lb->nitems ? lb->nitems - 1 : 0;
}
}
2006-03-20 17:12:09 +00:00
2013-10-27 22:00:14 +00:00
static void update_scroll_position(listbox *lb)
{
if (lb->citem<0)
lb->citem = 0;
if (lb->citem>=lb->nitems)
lb->citem = lb->nitems-1;
if (lb->citem< lb->first_item)
lb->first_item = lb->citem;
if (lb->citem>=( lb->first_item+LB_ITEMS_ON_SCREEN))
lb->first_item = lb->citem-LB_ITEMS_ON_SCREEN+1;
if (lb->nitems <= LB_ITEMS_ON_SCREEN )
lb->first_item = 0;
if (lb->first_item>lb->nitems-LB_ITEMS_ON_SCREEN)
lb->first_item = lb->nitems-LB_ITEMS_ON_SCREEN;
if (lb->first_item < 0 ) lb->first_item = 0;
}
2015-04-26 20:15:56 +00:00
static window_event_result listbox_mouse(window *, const d_event &event, listbox *lb, int button)
{
int i, mx, my, mz, x1, x2, y1, y2;
switch (button)
{
case MBTN_LEFT:
{
if (lb->mouse_state)
{
mouse_get_pos(&mx, &my, &mz);
2015-06-13 22:42:22 +00:00
const auto &&line_spacing = LINE_SPACING;
for (i=lb->first_item; i<lb->first_item+LB_ITEMS_ON_SCREEN; i++ ) {
if (i >= lb->nitems)
break;
2015-09-29 02:41:22 +00:00
int h;
gr_get_string_size(lb->item[i], nullptr, &h, nullptr);
x1 = lb->box_x;
x2 = lb->box_x + lb->box_w;
2015-06-13 22:42:22 +00:00
y1 = (i - lb->first_item) * line_spacing + lb->box_y;
y2 = y1+h;
if ( ((mx > x1) && (mx < x2)) && ((my > y1) && (my < y2)) ) {
lb->citem = i;
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
}
}
}
2014-10-04 21:47:13 +00:00
else if (event.type == EVENT_MOUSE_BUTTON_UP)
{
2015-09-29 02:41:22 +00:00
int h;
if (lb->citem < 0)
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
mouse_get_pos(&mx, &my, &mz);
2015-09-29 02:41:22 +00:00
gr_get_string_size(lb->item[lb->citem], nullptr, &h, nullptr);
x1 = lb->box_x;
x2 = lb->box_x + lb->box_w;
y1 = (lb->citem-lb->first_item)*LINE_SPACING+lb->box_y;
y2 = y1+h;
if ( ((mx > x1) && (mx < x2)) && ((my > y1) && (my < y2)) )
{
// Tell callback, allow staying in menu
const d_select_event selected{lb->citem};
2014-10-04 21:47:13 +00:00
if (lb->listbox_callback && (*lb->listbox_callback)(lb, selected, lb->userdata))
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
return window_event_result::close;
}
}
break;
}
case MBTN_RIGHT:
{
if (lb->allow_abort_flag && lb->mouse_state) {
lb->citem = -1;
2014-08-06 02:10:49 +00:00
return window_event_result::close;
}
break;
}
case MBTN_Z_UP:
{
if (lb->mouse_state)
{
lb->citem--;
update_scroll_position(lb);
}
break;
}
case MBTN_Z_DOWN:
{
if (lb->mouse_state)
{
lb->citem++;
update_scroll_position(lb);
}
break;
}
default:
break;
}
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
}
2015-04-26 20:15:56 +00:00
static window_event_result listbox_key_command(window *, const d_event &event, listbox *lb)
2006-03-20 17:12:09 +00:00
{
int key = event_key_get(event);
2014-08-06 02:10:49 +00:00
window_event_result rval = window_event_result::handled;
2009-12-27 05:00:45 +00:00
switch(key) {
2006-03-20 17:12:09 +00:00
case KEY_HOME:
case KEY_PAD7:
2009-12-27 05:00:45 +00:00
lb->citem = 0;
2006-03-20 17:12:09 +00:00
break;
case KEY_END:
case KEY_PAD1:
2009-12-27 05:00:45 +00:00
lb->citem = lb->nitems-1;
2006-03-20 17:12:09 +00:00
break;
case KEY_UP:
case KEY_PAD8:
2009-12-27 05:00:45 +00:00
lb->citem--;
2006-03-20 17:12:09 +00:00
break;
case KEY_DOWN:
case KEY_PAD2:
2009-12-27 05:00:45 +00:00
lb->citem++;
2006-03-20 17:12:09 +00:00
break;
case KEY_PAGEDOWN:
2006-03-20 17:12:09 +00:00
case KEY_PAD3:
2009-12-27 05:00:45 +00:00
lb->citem += LB_ITEMS_ON_SCREEN;
2006-03-20 17:12:09 +00:00
break;
case KEY_PAGEUP:
case KEY_PAD9:
2009-12-27 05:00:45 +00:00
lb->citem -= LB_ITEMS_ON_SCREEN;
2006-03-20 17:12:09 +00:00
break;
case KEY_ESC:
2009-12-27 05:00:45 +00:00
if (lb->allow_abort_flag) {
lb->citem = -1;
2014-08-06 02:10:49 +00:00
return window_event_result::close;
2006-03-20 17:12:09 +00:00
}
break;
case KEY_ENTER:
case KEY_PADENTER:
// Tell callback, allow staying in menu
2014-10-04 21:47:13 +00:00
{
const d_select_event selected{lb->citem};
2014-10-04 21:47:13 +00:00
if (lb->listbox_callback && (*lb->listbox_callback)(lb, selected, lb->userdata))
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
2014-10-04 21:47:13 +00:00
}
2014-08-06 02:10:49 +00:00
return window_event_result::close;
default:
2009-12-27 05:00:45 +00:00
{
int ascii = key_ascii();
if ( ascii < 255 ) {
int cc,cc1;
cc=cc1=lb->citem+1;
if (cc1 < 0 ) cc1 = 0;
if (cc1 >= lb->nitems ) cc1 = 0;
while(1) {
if ( cc < 0 ) cc = 0;
if ( cc >= lb->nitems ) cc = 0;
if ( lb->citem == cc ) break;
2009-12-27 05:00:45 +00:00
if ( toupper( lb->item[cc][0] ) == toupper(ascii) ) {
lb->citem = cc;
break;
2006-03-20 17:12:09 +00:00
}
2009-12-27 05:00:45 +00:00
cc++;
2006-03-20 17:12:09 +00:00
}
}
2014-08-06 02:10:49 +00:00
rval = window_event_result::ignored;
2006-03-20 17:12:09 +00:00
}
2009-12-27 05:00:45 +00:00
}
update_scroll_position(lb);
return rval;
}
2013-10-27 22:00:14 +00:00
static void listbox_create_structure( listbox *lb)
{
gr_set_current_canvas(NULL);
gr_set_curfont(MEDIUM3_FONT);
lb->box_w = 0;
2015-06-13 22:42:20 +00:00
const auto &&fspacx = FSPACX();
2015-03-12 02:21:21 +00:00
range_for (auto &i, unchecked_partial_range(lb->item, lb->nitems))
{
2015-09-29 02:41:22 +00:00
int w;
gr_get_string_size(i, &w, nullptr, nullptr);
if ( w > lb->box_w )
2015-06-13 22:42:20 +00:00
lb->box_w = w + fspacx(10);
}
lb->height = LINE_SPACING * LB_ITEMS_ON_SCREEN;
{
2015-09-29 02:41:22 +00:00
int w, h;
gr_get_string_size(lb->title, &w, &h, nullptr);
if ( w > lb->box_w )
lb->box_w = w;
lb->title_height = h+FSPACY(5);
}
lb->marquee_maxchars = lb->marquee_charpos = lb->marquee_scrollback = lb->marquee_lasttime = 0;
// The box is bigger than we can fit on the screen since at least one string is too long. Check how many chars we can fit on the screen (at least only - MEDIUM*_FONT is variable font!) so we can make a marquee-like effect.
if (lb->box_w + (BORDERX*2) > SWIDTH)
{
2015-09-29 02:41:22 +00:00
int w;
lb->box_w = SWIDTH - (BORDERX*2);
2015-09-29 02:41:22 +00:00
gr_get_string_size("O", &w, nullptr, nullptr);
lb->marquee_maxchars = lb->box_w/w;
lb->marquee_lasttime = timer_query();
}
lb->box_x = (grd_curcanv->cv_bitmap.bm_w-lb->box_w)/2;
lb->box_y = (grd_curcanv->cv_bitmap.bm_h-(lb->height+lb->title_height))/2 + lb->title_height;
if ( lb->box_y < lb->title_height )
lb->box_y = lb->title_height;
if ( lb->citem < 0 ) lb->citem = 0;
if ( lb->citem >= lb->nitems ) lb->citem = 0;
lb->first_item = 0;
update_scroll_position(lb);
lb->mouse_state = 0; //dblclick_flag = 0;
lb->swidth = SWIDTH;
lb->sheight = SHEIGHT;
lb->fntscalex = FNTScaleX;
lb->fntscaley = FNTScaleY;
}
2015-01-18 01:58:33 +00:00
static window_event_result listbox_draw(window *, listbox *lb)
{
int i;
if (lb->swidth != SWIDTH || lb->sheight != SHEIGHT || lb->fntscalex != FNTScaleX || lb->fntscalex != FNTScaleY)
listbox_create_structure ( lb );
gr_set_current_canvas(NULL);
nm_draw_background( lb->box_x-BORDERX,lb->box_y-lb->title_height-BORDERY,lb->box_x+lb->box_w+BORDERX,lb->box_y+lb->height+BORDERY );
gr_set_curfont(MEDIUM3_FONT);
gr_string( 0x8000, lb->box_y - lb->title_height, lb->title );
gr_setcolor( BM_XRGB( 0,0,0) );
2015-06-13 22:42:22 +00:00
const auto &&line_spacing = LINE_SPACING;
for (i=lb->first_item; i<lb->first_item+LB_ITEMS_ON_SCREEN; i++ ) {
2015-06-13 22:42:22 +00:00
int y = (i - lb->first_item) * line_spacing + lb->box_y;
2015-06-13 22:42:20 +00:00
const auto &&fspacx = FSPACX();
2015-06-13 22:42:20 +00:00
const auto &&fspacy = FSPACY();
if ( i >= lb->nitems ) {
gr_setcolor( BM_XRGB(5,5,5));
2015-06-13 22:42:22 +00:00
gr_rect(lb->box_x + lb->box_w - fspacx(1), y - fspacy(1), lb->box_x + lb->box_w, y + line_spacing);
gr_setcolor( BM_XRGB(2,2,2));
2015-06-13 22:42:22 +00:00
gr_rect(lb->box_x - fspacx(1), y - fspacy(1), lb->box_x, y + line_spacing);
gr_setcolor( BM_XRGB(0,0,0));
2015-06-13 22:42:22 +00:00
gr_rect(lb->box_x, y - fspacy(1), lb->box_x + lb->box_w - fspacx(1), y + line_spacing);
} else {
gr_set_curfont(( i == lb->citem )?MEDIUM2_FONT:MEDIUM1_FONT);
gr_setcolor( BM_XRGB(5,5,5));
2015-06-13 22:42:22 +00:00
gr_rect(lb->box_x + lb->box_w - fspacx(1), y - fspacy(1), lb->box_x + lb->box_w, y + line_spacing);
gr_setcolor( BM_XRGB(2,2,2));
2015-06-13 22:42:22 +00:00
gr_rect(lb->box_x - fspacx(1), y - fspacy(1), lb->box_x, y + line_spacing);
gr_setcolor( BM_XRGB(0,0,0));
2015-06-13 22:42:22 +00:00
gr_rect(lb->box_x, y - fspacy(1), lb->box_x + lb->box_w - fspacx(1), y + line_spacing);
if (lb->marquee_maxchars && strlen(lb->item[i]) > lb->marquee_maxchars)
{
RAIIdmem<char[]> shrtstr;
CALLOC(shrtstr, char[], lb->marquee_maxchars+1);
static int prev_citem = -1;
if (prev_citem != lb->citem)
{
lb->marquee_charpos = lb->marquee_scrollback = 0;
lb->marquee_lasttime = timer_query();
prev_citem = lb->citem;
}
if (i == lb->citem)
{
if (lb->marquee_lasttime + (F1_0/3) < timer_query())
{
lb->marquee_charpos = lb->marquee_charpos+(lb->marquee_scrollback?-1:+1);
lb->marquee_lasttime = timer_query();
}
if (lb->marquee_charpos < 0) // reached beginning of string -> scroll forward
{
lb->marquee_charpos = 0;
lb->marquee_scrollback = 0;
}
if (lb->marquee_charpos + lb->marquee_maxchars - 1 > strlen(lb->item[i])) // reached end of string -> scroll backward
{
lb->marquee_charpos = strlen(lb->item[i]) - lb->marquee_maxchars + 1;
lb->marquee_scrollback = 1;
}
snprintf(shrtstr.get(), lb->marquee_maxchars, "%s", lb->item[i]+lb->marquee_charpos);
}
else
{
snprintf(shrtstr.get(), lb->marquee_maxchars, "%s", lb->item[i]);
}
2015-06-13 22:42:20 +00:00
gr_string(lb->box_x + fspacx(5), y, shrtstr.get());
}
else
{
2015-06-13 22:42:20 +00:00
gr_string(lb->box_x + fspacx(5), y, lb->item[i]);
}
}
}
{
d_event event;
event.type = EVENT_NEWMENU_DRAW;
if ( lb->listbox_callback )
2014-10-04 21:47:13 +00:00
(*lb->listbox_callback)(lb, event, lb->userdata);
}
2014-08-06 02:10:49 +00:00
return window_event_result::handled;
}
2014-10-04 21:47:13 +00:00
static window_event_result listbox_handler(window *wind,const d_event &event, listbox *lb)
{
2014-10-04 21:47:13 +00:00
if (event.type == EVENT_WINDOW_CLOSED)
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
if (lb->listbox_callback)
{
int rval = (*lb->listbox_callback)(lb, event, lb->userdata);
if (rval)
2014-08-06 02:10:49 +00:00
return window_event_result::handled; // event handled
}
2014-10-04 21:47:13 +00:00
switch (event.type)
{
2010-01-28 04:24:25 +00:00
case EVENT_WINDOW_ACTIVATED:
game_flush_inputs();
event_toggle_focus(0);
key_toggle_repeat(1);
2010-01-28 04:24:25 +00:00
break;
case EVENT_WINDOW_DEACTIVATED:
//event_toggle_focus(1); // No cursor recentering
key_toggle_repeat(0);
break;
case EVENT_MOUSE_BUTTON_DOWN:
case EVENT_MOUSE_BUTTON_UP:
2014-10-04 21:47:13 +00:00
lb->mouse_state = event.type == EVENT_MOUSE_BUTTON_DOWN;
return listbox_mouse(wind, event, lb, event_mouse_get_button(event));
case EVENT_KEY_COMMAND:
return listbox_key_command(wind, event, lb);
case EVENT_IDLE:
timer_delay2(50);
return listbox_mouse(wind, event, lb, -1);
case EVENT_WINDOW_DRAW:
return listbox_draw(wind, lb);
case EVENT_WINDOW_CLOSE:
2015-03-12 02:21:21 +00:00
std::default_delete<listbox>()(lb);
break;
default:
break;
}
2014-08-06 02:10:49 +00:00
return window_event_result::ignored;
2009-12-27 05:00:45 +00:00
}
2006-03-20 17:12:09 +00:00
2015-03-12 02:21:21 +00:00
listbox *newmenu_listbox1( const char * title, uint_fast32_t nitems, const char *items[], int allow_abort_flag, int default_item, listbox_subfunction_t<void>::type listbox_callback, void *userdata )
2009-12-27 05:00:45 +00:00
{
2015-03-12 02:21:21 +00:00
std::unique_ptr<listbox> lb{new listbox{}};
2009-12-27 05:00:45 +00:00
window *wind;
newmenu_free_background();
lb->title = title;
lb->nitems = nitems;
lb->item = items;
lb->citem = default_item;
lb->allow_abort_flag = allow_abort_flag;
lb->listbox_callback = listbox_callback;
lb->userdata = userdata;
2006-03-20 17:12:09 +00:00
2010-07-29 08:30:46 +00:00
set_screen_mode(SCREEN_MENU); //hafta set the screen mode here or fonts might get changed/freed up if screen res changes
2015-03-12 02:21:21 +00:00
listbox_create_structure(lb.get());
2006-03-20 17:12:09 +00:00
2015-03-12 02:21:21 +00:00
wind = window_create(&grd_curscreen->sc_canvas, lb->box_x-BORDERX, lb->box_y-lb->title_height-BORDERY, lb->box_w+2*BORDERX, lb->height+2*BORDERY, listbox_handler, lb.get());
2009-12-27 05:00:45 +00:00
if (!wind)
{
2015-03-12 02:21:21 +00:00
lb.reset();
2006-03-20 17:12:09 +00:00
}
2015-03-12 02:21:21 +00:00
return lb.release();
2006-03-20 17:12:09 +00:00
}