RetroArch/menu/drivers/materialui.c

1433 lines
40 KiB
C
Raw Normal View History

/* RetroArch - A frontend for libretro.
2015-01-07 17:46:50 +01:00
* Copyright (C) 2011-2015 - Daniel De Matteis
* Copyright (C) 2014-2015 - Jean-André Santoni
*
* RetroArch is free software: you can redistribute it and/or modify it under the terms
* of the GNU General Public License as published by the Free Software Found-
* ation, either version 3 of the License, or (at your option) any later version.
*
* RetroArch 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with RetroArch.
* If not, see <http://www.gnu.org/licenses/>.
*/
2015-01-26 10:54:13 +01:00
#include <stdint.h>
#include <stdlib.h>
#include <stddef.h>
#include <stdio.h>
#include <string.h>
2015-10-29 02:54:26 +07:00
#include <string/stdstring.h>
#include <limits.h>
#include <retro_log.h>
2015-06-05 18:22:15 +02:00
#include <compat/posix_string.h>
#include <file/file_path.h>
2015-10-03 03:15:47 +02:00
#include "menu_generic.h"
2015-01-10 04:53:37 +01:00
#include "../menu.h"
#include "../menu_driver.h"
2015-09-27 01:10:15 +02:00
#include "../menu_animation.h"
2015-07-09 18:39:40 +02:00
#include "../menu_hash.h"
#include "../menu_display.h"
2015-07-09 18:39:40 +02:00
#include "../../runloop_data.h"
enum
{
2015-11-05 16:53:14 +07:00
MUI_TEXTURE_POINTER = 0,
MUI_TEXTURE_BACK,
MUI_TEXTURE_SWITCH_ON,
MUI_TEXTURE_SWITCH_OFF,
MUI_TEXTURE_TAB_MAIN_ACTIVE,
MUI_TEXTURE_TAB_PLAYLISTS_ACTIVE,
MUI_TEXTURE_TAB_SETTINGS_ACTIVE,
MUI_TEXTURE_TAB_MAIN_PASSIVE,
MUI_TEXTURE_TAB_PLAYLISTS_PASSIVE,
MUI_TEXTURE_TAB_SETTINGS_PASSIVE,
MUI_TEXTURE_LAST
};
enum
{
2015-11-05 16:53:14 +07:00
MUI_SYSTEM_TAB_MAIN = 0,
MUI_SYSTEM_TAB_PLAYLISTS,
MUI_SYSTEM_TAB_SETTINGS
};
2015-11-05 16:53:14 +07:00
#define MUI_SYSTEM_TAB_END MUI_SYSTEM_TAB_SETTINGS
struct mui_texture_item
{
GRuint id;
};
typedef struct mui_handle
{
2015-10-29 02:05:38 +07:00
unsigned tabs_height;
unsigned line_height;
2015-10-27 22:49:48 +07:00
unsigned icon_size;
unsigned margin;
2015-07-16 03:38:07 +02:00
unsigned glyph_width;
char box_message[PATH_MAX_LENGTH];
struct
{
struct
{
float alpha;
} arrow;
struct mui_texture_item bg;
struct mui_texture_item list[MUI_TEXTURE_LAST];
GRuint white;
} textures;
struct
{
struct
{
unsigned idx;
unsigned idx_old;
} active;
float x_pos;
size_t selection_ptr_old;
size_t selection_ptr;
} categories;
gfx_font_raster_block_t list_block;
} mui_handle_t;
static void mui_context_reset_textures(mui_handle_t *mui, const char *iconpath)
{
unsigned i;
2015-11-05 16:53:14 +07:00
for (i = 0; i < MUI_TEXTURE_LAST; i++)
{
struct texture_image ti = {0};
char path[PATH_MAX_LENGTH] = {0};
switch(i)
{
2015-11-05 16:53:14 +07:00
case MUI_TEXTURE_POINTER:
fill_pathname_join(path, iconpath, "pointer.png", sizeof(path));
break;
2015-11-05 16:53:14 +07:00
case MUI_TEXTURE_BACK:
2015-10-27 22:49:48 +07:00
fill_pathname_join(path, iconpath, "back.png", sizeof(path));
break;
2015-11-05 16:53:14 +07:00
case MUI_TEXTURE_SWITCH_ON:
2015-10-28 00:51:22 +07:00
fill_pathname_join(path, iconpath, "on.png", sizeof(path));
break;
2015-11-05 16:53:14 +07:00
case MUI_TEXTURE_SWITCH_OFF:
2015-10-28 00:51:22 +07:00
fill_pathname_join(path, iconpath, "off.png", sizeof(path));
break;
2015-11-05 16:53:14 +07:00
case MUI_TEXTURE_TAB_MAIN_ACTIVE:
fill_pathname_join(path, iconpath, "main_tab_active.png", sizeof(path));
break;
2015-11-05 16:53:14 +07:00
case MUI_TEXTURE_TAB_PLAYLISTS_ACTIVE:
fill_pathname_join(path, iconpath, "playlists_tab_active.png", sizeof(path));
break;
2015-11-05 16:53:14 +07:00
case MUI_TEXTURE_TAB_SETTINGS_ACTIVE:
fill_pathname_join(path, iconpath, "settings_tab_active.png", sizeof(path));
break;
2015-11-05 16:53:14 +07:00
case MUI_TEXTURE_TAB_MAIN_PASSIVE:
fill_pathname_join(path, iconpath, "main_tab_passive.png", sizeof(path));
break;
2015-11-05 16:53:14 +07:00
case MUI_TEXTURE_TAB_PLAYLISTS_PASSIVE:
fill_pathname_join(path, iconpath, "playlists_tab_passive.png", sizeof(path));
break;
2015-11-05 16:53:14 +07:00
case MUI_TEXTURE_TAB_SETTINGS_PASSIVE:
fill_pathname_join(path, iconpath, "settings_tab_passive.png", sizeof(path));
break;
}
if (path[0] == '\0' || !path_file_exists(path))
continue;
texture_image_load(&ti, path);
mui->textures.list[i].id = menu_display_texture_load(&ti,
TEXTURE_FILTER_MIPMAP_LINEAR);
texture_image_free(&ti);
}
}
static void mui_draw_icon(mui_handle_t *mui,
2015-10-27 22:49:48 +07:00
GRuint texture,
float x, float y,
unsigned width, unsigned height,
float rotation, float scale_factor,
GRfloat *color)
{
struct gfx_coords coords;
math_matrix_4x4 mymat;
2015-10-27 22:49:48 +07:00
menu_display_blend_begin();
menu_display_matrix_4x4_rotate_z(&mymat, rotation, scale_factor, scale_factor, 1, true);
2015-10-27 22:49:48 +07:00
coords.vertices = 4;
coords.vertex = NULL;
2015-11-02 22:46:23 +01:00
coords.tex_coord = NULL;
coords.lut_tex_coord = NULL;
2015-10-27 22:49:48 +07:00
coords.color = (const float*)color;
menu_display_draw_frame(
x,
height - y - mui->icon_size,
mui->icon_size,
mui->icon_size,
2015-11-02 18:55:05 +01:00
&coords, &mymat, texture,
2015-10-27 22:49:48 +07:00
MENU_DISPLAY_PRIM_TRIANGLESTRIP);
menu_display_blend_end();
2015-10-27 22:49:48 +07:00
}
static void mui_blit_line(float x, float y, unsigned width, unsigned height,
2015-04-21 16:59:02 +02:00
const char *message, uint32_t color, enum text_alignment text_align)
{
2015-09-26 23:37:25 +02:00
int font_size;
void *fb_buf = NULL;
2015-06-05 18:12:57 +02:00
struct font_params params = {0};
2015-06-12 16:20:45 +02:00
menu_handle_t *menu = menu_driver_get_ptr();
if (!menu)
return;
2015-09-26 23:37:25 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_FONT_SIZE, &font_size);
params.x = x / width;
params.y = 1.0f - (y + font_size / 3) / height;
2015-01-19 06:44:46 +01:00
params.scale = 1.0;
2015-03-14 19:44:27 -03:00
params.color = color;
params.full_screen = true;
2015-04-04 21:26:11 +02:00
params.text_align = text_align;
2015-01-19 06:44:46 +01:00
menu_display_ctl(MENU_DISPLAY_CTL_FONT_BUF, &fb_buf);
video_driver_set_osd_msg(message, &params, fb_buf);
}
static void mui_render_quad(int x, int y, unsigned w, unsigned h,
2015-09-06 20:38:45 +02:00
unsigned width, unsigned height,
2015-09-06 20:36:17 +02:00
GRfloat *coord_color)
2014-10-27 00:55:14 +01:00
{
2015-07-12 03:57:06 +02:00
struct gfx_coords coords;
2015-11-02 18:47:19 +01:00
menu_handle_t *menu = menu_driver_get_ptr();
mui_handle_t *mui = (mui_handle_t*)menu->userdata;
2015-06-30 18:53:57 +02:00
coords.vertices = 4;
coords.vertex = NULL;
2015-11-02 22:46:23 +01:00
coords.tex_coord = NULL;
coords.lut_tex_coord = NULL;
2015-09-06 20:36:17 +02:00
coords.color = coord_color;
2014-10-27 00:55:14 +01:00
menu_display_blend_begin();
menu_display_draw_frame(
2015-09-06 18:16:24 +02:00
x,
height - y - h,
w,
h,
&coords, NULL, mui->textures.white,
MENU_DISPLAY_PRIM_TRIANGLESTRIP);
menu_display_blend_end();
2014-10-27 00:55:14 +01:00
}
static void mui_draw_scrollbar(unsigned width, unsigned height, GRfloat *coord_color)
2015-04-21 09:47:02 +07:00
{
2015-09-25 21:52:04 +02:00
unsigned header_height;
float content_height, total_height, scrollbar_width, scrollbar_height, y;
mui_handle_t *mui = NULL;
2015-06-14 15:34:05 +02:00
menu_handle_t *menu = menu_driver_get_ptr();
2015-04-21 09:47:02 +07:00
if (!menu)
return;
2015-09-25 21:52:04 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_HEADER_HEIGHT, &header_height);
mui = (mui_handle_t*)menu->userdata;
content_height = menu_entries_get_end() * mui->line_height;
total_height = height - header_height - mui->tabs_height;
scrollbar_height = total_height / (content_height / total_height) - (header_height / 6);
2015-04-21 16:59:02 +02:00
y = total_height * menu->scroll_y / content_height;
2015-04-21 09:47:02 +07:00
if (content_height >= total_height)
2015-09-28 23:00:22 +02:00
{
scrollbar_width = (header_height / 12);
if (scrollbar_height <= header_height / 12)
scrollbar_height = header_height / 12;
2015-09-28 23:00:22 +02:00
mui_render_quad(
width - scrollbar_width - (header_height / 12),
2015-10-29 17:31:13 +07:00
header_height + y + (header_height / 12),
scrollbar_width,
scrollbar_height,
2015-09-06 20:38:45 +02:00
width, height,
2015-09-06 20:36:17 +02:00
coord_color);
2015-09-28 23:00:22 +02:00
}
2015-04-21 09:47:02 +07:00
}
static void mui_get_message(const char *message)
2014-09-15 12:36:52 +02:00
{
mui_handle_t *mui = NULL;
menu_handle_t *menu = menu_driver_get_ptr();
2014-09-15 12:36:52 +02:00
if (!menu)
return;
if (!message || !*message)
2014-09-15 12:36:52 +02:00
return;
mui = (mui_handle_t*)menu->userdata;
if (mui)
strlcpy(mui->box_message, message, sizeof(mui->box_message));
2014-09-15 12:36:52 +02:00
}
static void mui_render_messagebox(const char *message)
{
unsigned i;
unsigned width, height;
2015-03-16 16:57:27 +01:00
uint32_t normal_color;
2015-09-26 23:37:25 +02:00
int x, y, font_size;
struct string_list *list = NULL;
2015-06-05 18:12:57 +02:00
menu_handle_t *menu = menu_driver_get_ptr();
settings_t *settings = config_get_ptr();
2014-09-15 12:36:52 +02:00
if (!menu || !menu->userdata)
return;
list = (struct string_list*)string_split(message, "\n");
2014-09-15 12:36:52 +02:00
if (!list)
return;
2014-09-15 12:36:52 +02:00
if (list->elems == 0)
2015-02-02 18:51:48 +01:00
goto end;
2014-09-15 12:36:52 +02:00
video_driver_get_size(&width, &height);
2015-09-26 23:37:25 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_FONT_SIZE, &font_size);
x = width / 2;
2015-09-26 23:37:25 +02:00
y = height / 2 - list->size * font_size / 2;
2015-03-20 22:22:06 +01:00
normal_color = FONT_COLOR_ARGB_TO_RGBA(settings->menu.entry_normal_color);
2014-09-15 12:36:52 +02:00
for (i = 0; i < list->size; i++)
{
const char *msg = list->elems[i].data;
2015-02-02 18:51:48 +01:00
if (msg)
mui_blit_line(x, y + i * font_size,
width, height,
2015-06-25 08:26:59 +02:00
msg, normal_color, TEXT_ALIGN_CENTER);
2014-09-15 12:36:52 +02:00
}
2015-02-02 18:51:48 +01:00
end:
2014-09-15 12:36:52 +02:00
string_list_free(list);
}
static void mui_render(void)
{
2015-09-25 23:47:01 +02:00
float delta_time, dt;
unsigned bottom;
2015-09-25 21:52:04 +02:00
unsigned width, height, header_height;
2015-11-05 17:05:26 +07:00
mui_handle_t *mui = NULL;
menu_handle_t *menu = menu_driver_get_ptr();
settings_t *settings = config_get_ptr();
2015-05-01 09:35:27 -03:00
if (!menu || !menu->userdata)
return;
video_driver_get_size(&width, &height);
mui = (mui_handle_t*)menu->userdata;
2015-09-25 23:37:02 +02:00
menu_animation_ctl(MENU_ANIMATION_CTL_DELTA_TIME, &delta_time);
2015-09-25 23:47:01 +02:00
dt = delta_time / IDEAL_DT;
menu_animation_ctl(MENU_ANIMATION_CTL_UPDATE, &dt);
2015-04-20 21:22:55 +07:00
menu_display_ctl(MENU_DISPLAY_CTL_SET_WIDTH, &width);
menu_display_ctl(MENU_DISPLAY_CTL_SET_HEIGHT, &height);
2015-09-25 21:52:04 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_HEADER_HEIGHT, &header_height);
2015-04-04 21:26:11 +02:00
if (settings->menu.pointer.enable)
{
int16_t pointer_y = menu_input_pointer_state(MENU_POINTER_Y_AXIS);
float old_accel_val, new_accel_val;
2015-09-24 19:56:53 +02:00
unsigned new_pointer_val =
(pointer_y - mui->line_height + menu->scroll_y - 16)
/ mui->line_height;
2015-04-04 21:26:11 +02:00
2015-09-26 02:52:05 +02:00
menu_input_ctl(MENU_INPUT_CTL_POINTER_ACCEL_READ, &old_accel_val);
menu_input_ctl(MENU_INPUT_CTL_POINTER_PTR, &new_pointer_val);
2015-09-24 19:56:53 +02:00
menu->scroll_y -= old_accel_val / 60.0;
new_accel_val = old_accel_val * 0.96;
2015-09-26 02:52:05 +02:00
menu_input_ctl(MENU_INPUT_CTL_POINTER_ACCEL_WRITE, &new_accel_val);
2015-04-04 21:26:11 +02:00
}
if (settings->menu.mouse.enable)
{
2015-09-24 23:33:41 +02:00
int16_t mouse_y = menu_input_mouse_state(MENU_MOUSE_Y_AXIS);
2015-09-24 17:34:44 +02:00
unsigned new_pointer_val =
(mouse_y - mui->line_height + menu->scroll_y - 16)
/ mui->line_height;
2015-11-05 17:05:26 +07:00
menu_input_ctl(MENU_INPUT_CTL_MOUSE_PTR, &new_pointer_val);
2015-04-04 21:26:11 +02:00
}
2015-04-04 21:26:11 +02:00
if (menu->scroll_y < 0)
menu->scroll_y = 0;
bottom = menu_entries_get_end() * mui->line_height
- height + header_height + mui->tabs_height;
2015-04-04 21:26:11 +02:00
if (menu->scroll_y > bottom)
menu->scroll_y = bottom;
if (menu_entries_get_end() * mui->line_height
< height - header_height - mui->tabs_height)
2015-04-04 21:26:11 +02:00
menu->scroll_y = 0;
2015-07-14 00:38:16 +02:00
if (menu_entries_get_end() < height / mui->line_height)
2015-07-14 00:38:16 +02:00
menu_entries_set_start(0);
else
menu_entries_set_start(menu->scroll_y / mui->line_height);
}
static void mui_render_label_value(mui_handle_t *mui,
int y, unsigned width, unsigned height,
2015-10-28 00:51:22 +07:00
uint64_t index, uint32_t color, bool selected, const char *label,
const char *value, GRfloat *pure_white)
{
char label_str[PATH_MAX_LENGTH];
char value_str[PATH_MAX_LENGTH];
2015-11-05 17:05:26 +07:00
int value_len = strlen(value);
int ticker_limit = 0;
size_t usable_width = 0;
2015-10-28 00:51:22 +07:00
GRuint texture_switch = 0;
2015-11-05 17:05:26 +07:00
bool do_draw_text = false;
uint32_t hash_value = 0;
label_str[0] = '\0';
value_str[0] = '\0';
usable_width = width - (mui->margin * 2);
if (value_len * mui->glyph_width > usable_width / 2)
value_len = (usable_width/2) / mui->glyph_width;
2015-07-16 03:38:07 +02:00
ticker_limit = (usable_width / mui->glyph_width) - (value_len + 2);
2015-07-16 03:38:07 +02:00
menu_animation_ticker_str(label_str, ticker_limit, index, label, selected);
menu_animation_ticker_str(value_str, value_len, index, value, selected);
mui_blit_line(mui->margin, y + mui->line_height / 2,
width, height, label_str, color, TEXT_ALIGN_LEFT);
2015-10-28 00:51:22 +07:00
hash_value = menu_hash_calculate(value);
2015-11-05 17:05:26 +07:00
if (!strcmp(value, "disabled") || !strcmp(value, "off"))
2015-10-28 00:51:22 +07:00
{
if (mui->textures.list[MUI_TEXTURE_SWITCH_OFF].id)
texture_switch = mui->textures.list[MUI_TEXTURE_SWITCH_OFF].id;
else
do_draw_text = true;
}
2015-11-05 17:05:26 +07:00
else if (!strcmp(value, "enabled") || !strcmp(value, "on"))
{
if (mui->textures.list[MUI_TEXTURE_SWITCH_ON].id)
texture_switch = mui->textures.list[MUI_TEXTURE_SWITCH_ON].id;
else
2015-10-28 00:51:22 +07:00
do_draw_text = true;
}
else
{
switch (hash_value)
{
case MENU_VALUE_COMP:
break;
case MENU_VALUE_MORE:
break;
case MENU_VALUE_CORE:
break;
case MENU_VALUE_RDB:
break;
case MENU_VALUE_CURSOR:
break;
case MENU_VALUE_FILE:
break;
case MENU_VALUE_DIR:
break;
case MENU_VALUE_MUSIC:
break;
case MENU_VALUE_IMAGE:
break;
case MENU_VALUE_MOVIE:
break;
case MENU_VALUE_ON:
if (mui->textures.list[MUI_TEXTURE_SWITCH_ON].id)
texture_switch = mui->textures.list[MUI_TEXTURE_SWITCH_ON].id;
else
do_draw_text = true;
break;
case MENU_VALUE_OFF:
if (mui->textures.list[MUI_TEXTURE_SWITCH_OFF].id)
texture_switch = mui->textures.list[MUI_TEXTURE_SWITCH_OFF].id;
else
do_draw_text = true;
break;
default:
do_draw_text = true;
break;
}
2015-10-28 00:51:22 +07:00
}
if (do_draw_text)
mui_blit_line(width - mui->margin,
y + mui->line_height / 2,
width, height, value_str, color, TEXT_ALIGN_RIGHT);
2015-10-28 00:51:22 +07:00
if (texture_switch)
mui_draw_icon(mui, texture_switch,
width - mui->margin - mui->icon_size, y,
width, height, 0, 1, &pure_white[0]);
}
static void mui_render_menu_list(mui_handle_t *mui,
unsigned width, unsigned height,
2015-05-20 03:09:28 +02:00
menu_handle_t *menu,
2015-05-07 10:15:33 +02:00
uint32_t normal_color,
2015-10-28 00:51:22 +07:00
uint32_t hover_color,
GRfloat *pure_white)
{
2015-09-25 21:52:04 +02:00
unsigned header_height;
2015-07-14 00:38:16 +02:00
size_t i = 0;
2015-08-03 23:01:07 +02:00
uint64_t *frame_count = video_driver_get_frame_count();
2015-07-14 00:38:16 +02:00
size_t end = menu_entries_get_end();
2015-09-25 14:07:01 +02:00
if (!menu_display_ctl(MENU_DISPLAY_CTL_UPDATE_PENDING, NULL))
2015-04-24 22:46:19 +02:00
return;
2015-09-25 21:52:04 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_HEADER_HEIGHT, &header_height);
mui->list_block.carr.coords.vertices = 0;
for (i = menu_entries_get_start(); i < end; i++)
{
2015-09-25 14:57:37 +02:00
int y;
size_t selection;
bool entry_selected;
2015-07-14 00:38:16 +02:00
menu_entry_t entry;
2015-09-25 14:57:37 +02:00
if (!menu_navigation_ctl(MENU_NAVIGATION_CTL_GET_SELECTION, &selection))
continue;
y = header_height - menu->scroll_y + (mui->line_height * i);
if ((y - (int)mui->line_height) > (int)height
|| ((y + (int)mui->line_height) < 0))
continue;
2015-10-27 23:52:20 +01:00
menu_entry_get(&entry, 0, i, NULL, true);
2015-07-14 00:38:16 +02:00
2015-09-25 14:57:37 +02:00
entry_selected = selection == i;
mui_render_label_value(mui, y, width, height, *frame_count / 40,
entry_selected ? hover_color : normal_color, entry_selected,
2015-10-28 00:51:22 +07:00
entry.path, entry.value, pure_white);
}
}
static void mui_draw_cursor(mui_handle_t *mui,
GRfloat *color,
float x, float y, unsigned width, unsigned height)
{
struct gfx_coords coords;
coords.vertices = 4;
coords.vertex = NULL;
2015-11-02 22:46:23 +01:00
coords.tex_coord = NULL;
coords.lut_tex_coord = NULL;
coords.color = (const float*)color;
menu_display_blend_begin();
2015-11-01 19:26:24 +01:00
menu_display_draw_frame(
x - 32,
height - y - 32,
64,
64,
2015-11-02 19:34:01 +01:00
&coords, NULL,
mui->textures.list[MUI_TEXTURE_POINTER].id,
MENU_DISPLAY_PRIM_TRIANGLESTRIP);
2015-11-01 19:26:24 +01:00
menu_display_blend_end();
}
static size_t mui_list_get_size(void *data, menu_list_type_t type)
{
2015-11-05 17:05:26 +07:00
size_t list_size = 0;
switch (type)
{
case MENU_LIST_PLAIN:
list_size = menu_entries_get_stack_size(0);
break;
case MENU_LIST_TABS:
2015-11-05 16:53:14 +07:00
list_size = MUI_SYSTEM_TAB_END;
break;
default:
break;
}
return list_size;
}
2015-11-05 17:19:42 +07:00
static void bgcolor_setalpha(GRfloat *bg, float alpha)
{
bg[3] = alpha;
bg[7] = alpha;
bg[11] = alpha;
bg[15] = alpha;
}
static void mui_frame(void)
{
2015-10-27 13:00:18 +01:00
unsigned header_height;
bool display_kb;
2015-10-27 16:44:56 +07:00
GRfloat black_bg[16] = {
0, 0, 0, 0.75,
0, 0, 0, 0.75,
0, 0, 0, 0.75,
0, 0, 0, 0.75,
};
GRfloat blue_bg[16] = {
0.13, 0.59, 0.95, 1,
0.13, 0.59, 0.95, 1,
0.13, 0.59, 0.95, 1,
0.13, 0.59, 0.95, 1,
};
GRfloat lightblue_bg[16] = {
0.89, 0.95, 0.99, 1.00,
0.89, 0.95, 0.99, 1.00,
0.89, 0.95, 0.99, 1.00,
0.89, 0.95, 0.99, 1.00,
2015-10-27 16:44:56 +07:00
};
2015-10-28 00:51:22 +07:00
GRfloat pure_white[16]= {
1, 1, 1, 1,
1, 1, 1, 1,
1, 1, 1, 1,
1, 1, 1, 1,
};
2015-10-27 16:44:56 +07:00
GRfloat white_bg[16]= {
0.98, 0.98, 0.98, 1,
0.98, 0.98, 0.98, 1,
0.98, 0.98, 0.98, 1,
0.98, 0.98, 0.98, 1,
2015-10-27 16:44:56 +07:00
};
GRfloat white_transp_bg[16]= {
0.98, 0.98, 0.98, 0.90,
0.98, 0.98, 0.98, 0.90,
0.98, 0.98, 0.98, 0.90,
0.98, 0.98, 0.98, 0.90,
2015-10-27 16:44:56 +07:00
};
GRfloat grey_bg[16]= {
0.78, 0.78, 0.78, 1,
0.78, 0.78, 0.78, 1,
0.78, 0.78, 0.78, 1,
0.78, 0.78, 0.78, 1,
2015-10-27 16:44:56 +07:00
};
GRfloat shadow_bg[16]= {
0, 0, 0, 0,
0, 0, 0, 0,
0, 0, 0, 0.2,
0, 0, 0, 0.2,
};
2015-10-29 02:05:38 +07:00
unsigned width, height, ticker_limit, i, tab_width;
char msg[PATH_MAX_LENGTH];
char title[PATH_MAX_LENGTH];
char title_buf[PATH_MAX_LENGTH];
2015-09-25 14:57:37 +02:00
size_t selection;
2015-10-28 09:54:08 +07:00
size_t title_margin;
2015-11-05 17:05:26 +07:00
mui_handle_t *mui = NULL;
driver_t *driver = driver_get_ptr();
menu_handle_t *menu = menu_driver_get_ptr();
settings_t *settings = config_get_ptr();
uint64_t *frame_count = video_driver_get_frame_count();
const uint32_t normal_color = 0x212121ff;
const uint32_t hover_color = 0x212121ff;
const uint32_t title_color = 0xffffffff;
const uint32_t activetab_color = 0x0096f2ff;
const uint32_t passivetab_color = 0x9e9e9eff;
bool background_rendered = false;
bool libretro_running = menu_display_ctl(MENU_DISPLAY_CTL_LIBRETRO_RUNNING, NULL);
2015-10-29 13:32:58 +01:00
(void)passivetab_color;
(void)activetab_color;
2015-05-01 09:35:27 -03:00
if (!menu || !menu->userdata)
return;
mui = (mui_handle_t*)menu->userdata;
msg[0] = '\0';
title[0] = '\0';
title_buf[0] = '\0';
video_driver_get_size(&width, &height);
2015-09-25 13:27:15 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_SET_VIEWPORT, NULL);
2015-09-25 21:52:04 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_HEADER_HEIGHT, &header_height);
2015-11-01 19:49:19 +01:00
if (libretro_running)
{
menu_display_frame_background(
width, height,
mui->textures.white, 0.75f, false,
2015-11-01 19:49:19 +01:00
&white_transp_bg[0], &white_bg[0],
2015-11-02 22:46:23 +01:00
NULL, NULL, 4,
2015-11-01 19:49:19 +01:00
MENU_DISPLAY_PRIM_TRIANGLESTRIP);
}
else
{
menu_display_clear_color(1.0f, 1.0f, 1.0f, 0.75f);
2015-11-01 19:49:19 +01:00
if (mui->textures.bg.id)
{
background_rendered = true;
2015-11-01 21:02:45 +01:00
/* Set new opacity for transposed white background */
2015-11-05 17:19:42 +07:00
bgcolor_setalpha(white_transp_bg, 0.30);
menu_display_frame_background(
width, height,
mui->textures.bg.id, 0.75f, true,
&white_transp_bg[0], &white_bg[0],
2015-11-02 22:46:23 +01:00
NULL, NULL, 4,
MENU_DISPLAY_PRIM_TRIANGLESTRIP);
2015-11-01 21:02:45 +01:00
/* Restore opacity of transposed white background */
2015-11-05 17:19:42 +07:00
bgcolor_setalpha(white_transp_bg, 0.90);
}
}
2015-05-15 00:07:07 +02:00
menu_entries_get_title(title, sizeof(title));
2015-09-25 14:57:37 +02:00
if (!menu_navigation_ctl(MENU_NAVIGATION_CTL_GET_SELECTION, &selection))
return;
if (background_rendered || libretro_running)
2015-11-05 17:19:42 +07:00
bgcolor_setalpha(lightblue_bg, 0.75);
else
2015-11-05 17:19:42 +07:00
bgcolor_setalpha(lightblue_bg, 1.0);
2015-10-28 15:56:45 +01:00
/* highlighted entry */
mui_render_quad(0,
header_height - menu->scroll_y + mui->line_height *
selection, width, mui->line_height,
2015-09-06 20:38:45 +02:00
width, height,
2015-10-27 16:44:56 +07:00
&lightblue_bg[0]);
menu_display_font_bind_block(menu, driver->font_osd_driver, &mui->list_block);
2015-10-27 16:44:56 +07:00
mui_render_menu_list(mui, width, height, menu, normal_color, hover_color, &pure_white[0]);
2015-10-27 16:44:56 +07:00
2015-11-02 20:06:05 +01:00
menu_display_font_flush_block(menu, driver->font_osd_driver);
2015-09-25 23:42:00 +02:00
menu_animation_ctl(MENU_ANIMATION_CTL_SET_ACTIVE, NULL);
2015-04-04 21:26:11 +02:00
2015-10-28 15:56:45 +01:00
/* header */
2015-11-05 17:22:14 +07:00
mui_render_quad( 0, 0, width, header_height, width, height, &blue_bg[0]);
2015-04-04 21:26:11 +02:00
/* display tabs if depth equal one, if not hide them */
if (mui_list_get_size(menu, MENU_LIST_PLAIN) == 1)
{
float scale_factor;
menu_display_ctl(MENU_DISPLAY_CTL_GET_DPI, &scale_factor);
2015-11-05 17:22:14 +07:00
mui->tabs_height = scale_factor / 3;
/* tabs background */
mui_render_quad(0, height - mui->tabs_height, width,
mui->tabs_height,
width, height,
&white_bg[0]);
2015-10-29 17:40:06 +07:00
/* tabs separator */
mui_render_quad(0, height - mui->tabs_height, width,
1,
width, height,
&grey_bg[0]);
2015-10-29 02:05:38 +07:00
2015-11-05 16:53:14 +07:00
for (i = 0; i <= MUI_SYSTEM_TAB_END; i++)
2015-10-29 02:34:52 +07:00
{
2015-11-05 16:53:14 +07:00
unsigned tab_icon = MUI_TEXTURE_TAB_MAIN_PASSIVE;
switch (i)
{
2015-11-05 16:53:14 +07:00
case MUI_SYSTEM_TAB_MAIN:
tab_icon = (i == mui->categories.selection_ptr)
2015-11-05 16:53:14 +07:00
? MUI_TEXTURE_TAB_MAIN_ACTIVE
: MUI_TEXTURE_TAB_MAIN_PASSIVE;
break;
2015-11-05 16:53:14 +07:00
case MUI_SYSTEM_TAB_PLAYLISTS:
tab_icon = (i == mui->categories.selection_ptr)
2015-11-05 16:53:14 +07:00
? MUI_TEXTURE_TAB_PLAYLISTS_ACTIVE
: MUI_TEXTURE_TAB_PLAYLISTS_PASSIVE;
break;
2015-11-05 16:53:14 +07:00
case MUI_SYSTEM_TAB_SETTINGS:
tab_icon = (i == mui->categories.selection_ptr)
2015-11-05 16:53:14 +07:00
? MUI_TEXTURE_TAB_SETTINGS_ACTIVE
: MUI_TEXTURE_TAB_SETTINGS_PASSIVE;
break;
}
mui_draw_icon(mui, mui->textures.list[tab_icon].id,
width / (MUI_SYSTEM_TAB_END+1) * (i+0.5) - mui->icon_size/2,
height - mui->tabs_height,
width, height, 0, 1, &pure_white[0]);
2015-10-29 02:34:52 +07:00
}
/* active tab marker */
2015-11-05 16:53:14 +07:00
tab_width = width / (MUI_SYSTEM_TAB_END+1);
mui_render_quad(mui->categories.selection_ptr * tab_width,
height - (header_height/16),
tab_width,
header_height/16,
width, height,
&blue_bg[0]);
}
else
{
mui->tabs_height = 0;
2015-10-29 02:05:38 +07:00
}
mui_render_quad(0, header_height, width,
header_height/12,
width, height,
&shadow_bg[0]);
title_margin = mui->margin;
2015-04-04 21:26:11 +02:00
2015-05-15 00:16:39 +02:00
if (menu_entries_show_back())
2015-10-28 09:54:08 +07:00
{
title_margin = mui->icon_size;
mui_draw_icon(mui, mui->textures.list[MUI_TEXTURE_BACK].id,
0, 0, width, height, 0, 1, &pure_white[0]);
2015-10-28 09:54:08 +07:00
}
ticker_limit = (width - mui->margin*2) / mui->glyph_width;
2015-10-28 09:54:08 +07:00
menu_animation_ticker_str(title_buf, ticker_limit,
*frame_count / 100, title, true);
2015-10-28 15:56:45 +01:00
/* Title */
mui_blit_line(title_margin, header_height / 2, width, height,
title_buf, title_color, TEXT_ALIGN_LEFT);
2015-04-04 21:26:11 +02:00
mui_draw_scrollbar(width, height, &grey_bg[0]);
2015-04-21 09:47:02 +07:00
2015-09-26 02:52:05 +02:00
menu_input_ctl(MENU_INPUT_CTL_KEYBOARD_DISPLAY, &display_kb);
if (display_kb)
{
const char *str = NULL, *label = NULL;
2015-09-26 02:52:05 +02:00
menu_input_ctl(MENU_INPUT_CTL_KEYBOARD_BUFF_PTR, &str);
menu_input_ctl(MENU_INPUT_CTL_KEYBOARD_LABEL, &label);
2015-06-13 00:09:09 +02:00
if (!str)
str = "";
mui_render_quad(0, 0, width, height, width, height, &black_bg[0]);
snprintf(msg, sizeof(msg), "%s\n%s", label, str);
mui_render_messagebox(msg);
}
if (mui->box_message[0] != '\0')
2014-09-15 12:36:52 +02:00
{
mui_render_quad(0, 0, width, height, width, height, &black_bg[0]);
mui_render_messagebox(mui->box_message);
mui->box_message[0] = '\0';
2014-09-15 12:36:52 +02:00
}
if (settings->menu.mouse.enable && (settings->video.fullscreen || !video_driver_has_windowed()))
2015-09-24 17:29:46 +02:00
{
int16_t mouse_x = menu_input_mouse_state(MENU_MOUSE_X_AXIS);
int16_t mouse_y = menu_input_mouse_state(MENU_MOUSE_Y_AXIS);
mui_draw_cursor(mui, &white_bg[0], mouse_x, mouse_y, width, height);
2015-09-24 17:29:46 +02:00
}
2015-03-27 16:17:11 -03:00
menu_display_restore_clear_color();
2015-09-25 13:27:15 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_UNSET_VIEWPORT, NULL);
}
static void mui_allocate_white_texture(mui_handle_t *mui)
{
struct texture_image ti;
2015-06-12 16:20:45 +02:00
static const uint8_t white_data[] = { 0xff, 0xff, 0xff, 0xff };
ti.width = 1;
ti.height = 1;
ti.pixels = (uint32_t*)&white_data;
mui->textures.white = menu_display_texture_load(&ti,
TEXTURE_FILTER_NEAREST);
}
static void mui_font(menu_handle_t *menu)
{
2015-09-26 23:37:25 +02:00
int font_size;
char mediapath[PATH_MAX_LENGTH], fontpath[PATH_MAX_LENGTH];
settings_t *settings = config_get_ptr();
2015-09-26 23:37:25 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_FONT_SIZE, &font_size);
fill_pathname_join(mediapath, settings->assets_directory, "glui", sizeof(mediapath));
fill_pathname_join(fontpath, mediapath, "Roboto-Regular.ttf", sizeof(fontpath));
if (!menu_display_init_main_font(menu, fontpath, font_size))
RARCH_WARN("Failed to load font.");
}
static void mui_layout(menu_handle_t *menu, mui_handle_t *mui)
{
void *fb_buf;
2015-07-16 10:11:28 +02:00
float scale_factor;
2015-09-26 23:37:25 +02:00
int new_font_size;
unsigned width, height, new_header_height;
video_driver_get_size(&width, &height);
2015-07-08 17:26:51 +02:00
/* Mobiles platforms may have very small display metrics coupled to a high
2015-11-05 16:51:14 +07:00
resolution, so we should be dpi aware to ensure the entries hitboxes are
big enough. On desktops, we just care about readability, with every widget
size proportional to the display width. */
2015-09-25 14:03:20 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_GET_DPI, &scale_factor);
2015-11-05 16:51:14 +07:00
new_header_height = scale_factor / 3;
new_font_size = scale_factor / 9;
2015-09-26 23:37:25 +02:00
mui->tabs_height = scale_factor / 3;
mui->line_height = scale_factor / 3;
mui->margin = scale_factor / 9;
mui->icon_size = scale_factor / 3;
2015-09-26 23:37:25 +02:00
menu_display_ctl(MENU_DISPLAY_CTL_SET_HEADER_HEIGHT, &new_header_height);
menu_display_ctl(MENU_DISPLAY_CTL_SET_FONT_SIZE, &new_font_size);
/* we assume the average glyph aspect ratio is close to 3:4 */
mui->glyph_width = new_font_size * 3/4;
mui_font(menu);
menu_display_ctl(MENU_DISPLAY_CTL_FONT_BUF, &fb_buf);
if (fb_buf) /* calculate a more realistic ticker_limit */
{
2015-11-05 16:51:14 +07:00
driver_t *driver = driver_get_ptr();
unsigned m_width = driver->font_osd_driver->get_message_width(fb_buf, "a", 1, 1);
if (m_width)
mui->glyph_width = m_width;
}
}
static void *mui_init(void)
{
2015-11-05 17:05:26 +07:00
mui_handle_t *mui = NULL;
2015-11-05 17:14:51 +07:00
menu_handle_t *menu = (menu_handle_t*)calloc(1, sizeof(*menu));
if (!menu)
2015-09-25 21:52:04 +02:00
goto error;
if (!menu_display_check_compatibility((enum menu_display_driver_type)menu_ctx_mui.type))
2015-02-02 18:51:48 +01:00
goto error;
menu->userdata = (mui_handle_t*)calloc(1, sizeof(mui_handle_t));
if (!menu->userdata)
2015-02-02 18:51:48 +01:00
goto error;
mui = (mui_handle_t*)menu->userdata;
mui_layout(menu, mui);
mui_allocate_white_texture(mui);
return menu;
2015-02-02 18:51:48 +01:00
error:
if (menu)
free(menu);
return NULL;
}
static void mui_free(void *data)
{
2015-11-05 17:05:26 +07:00
menu_handle_t *menu = (menu_handle_t*)data;
driver_t *driver = driver_get_ptr();
2015-11-05 17:14:51 +07:00
mui_handle_t *mui = menu ? (mui_handle_t*)menu->userdata : NULL;
const struct font_renderer *font_driver = driver ?
(const struct font_renderer*)driver->font_osd_driver : NULL;
2015-11-05 17:14:51 +07:00
if (!mui)
return;
gfx_coord_array_free(&mui->list_block.carr);
2015-04-24 22:17:08 +02:00
if (font_driver && font_driver->bind_block)
font_driver->bind_block(driver->font_osd_data, NULL);
if (menu->userdata)
free(menu->userdata);
2015-06-11 21:12:08 +02:00
menu->userdata = NULL;
}
static void mui_context_bg_destroy(mui_handle_t *mui)
{
if (!mui)
2015-10-25 02:33:23 +01:00
return;
2015-05-11 13:49:46 -03:00
menu_display_texture_unload((uintptr_t*)&mui->textures.bg.id);
menu_display_texture_unload((uintptr_t*)&mui->textures.white);
}
static void mui_context_destroy(void)
{
unsigned i;
2015-11-05 17:05:26 +07:00
menu_handle_t *menu = menu_driver_get_ptr();
2015-11-05 17:14:51 +07:00
mui_handle_t *mui = menu ? (mui_handle_t*)menu->userdata : NULL;
if (!mui)
return;
2015-11-05 16:53:14 +07:00
for (i = 0; i < MUI_TEXTURE_LAST; i++)
menu_display_texture_unload((uintptr_t*)&mui->textures.list[i].id);
menu_display_free_main_font();
2015-05-01 09:02:03 -03:00
mui_context_bg_destroy(mui);
}
static bool mui_load_image(void *data, menu_image_type_t type)
{
2015-11-05 17:05:26 +07:00
mui_handle_t *mui = NULL;
menu_handle_t *menu = menu_driver_get_ptr();
2015-05-01 09:35:27 -03:00
if (!menu || !menu->userdata)
return false;
2015-07-08 17:26:51 +02:00
mui = (mui_handle_t*)menu->userdata;
switch (type)
{
case MENU_IMAGE_NONE:
break;
case MENU_IMAGE_WALLPAPER:
mui_context_bg_destroy(mui);
mui->textures.bg.id = menu_display_texture_load(data,
TEXTURE_FILTER_MIPMAP_LINEAR);
mui_allocate_white_texture(mui);
break;
case MENU_IMAGE_BOXART:
break;
}
return true;
}
static float mui_get_scroll(void)
2014-10-27 00:55:14 +01:00
{
2015-09-25 14:57:37 +02:00
size_t selection;
unsigned width, height, half = 0;
mui_handle_t *mui = NULL;
menu_handle_t *menu = menu_driver_get_ptr();
2015-09-25 14:57:37 +02:00
if (!menu_navigation_ctl(MENU_NAVIGATION_CTL_GET_SELECTION, &selection))
return 0;
2014-10-27 00:55:14 +01:00
2015-05-01 09:35:27 -03:00
if (!menu || !menu->userdata)
return 0;
2015-04-21 16:59:02 +02:00
2015-07-13 22:05:14 +02:00
video_driver_get_size(&width, &height);
mui = (mui_handle_t*)menu->userdata;
2015-11-05 17:05:26 +07:00
if (mui->line_height)
half = (height / mui->line_height) / 2;
2015-04-04 21:26:11 +02:00
if (selection < half)
return 0;
2015-11-05 17:05:26 +07:00
return ((selection + 2 - half) * mui->line_height);
}
static void mui_navigation_set(bool scroll)
{
menu_handle_t *menu = menu_driver_get_ptr();
float scroll_pos = mui_get_scroll();
2015-04-20 21:22:55 +07:00
if (!menu || !scroll)
return;
2015-09-25 22:24:07 +02:00
menu_animation_push(10, scroll_pos,
2015-07-13 22:05:14 +02:00
&menu->scroll_y, EASING_IN_OUT_QUAD, -1, NULL);
2014-10-27 00:55:14 +01:00
}
static void mui_list_set_selection(file_list_t *list)
{
mui_navigation_set(true);
}
static void mui_navigation_clear(bool pending_push)
{
menu_handle_t *menu = menu_driver_get_ptr();
if (!menu)
return;
menu_entries_set_start(0);
2015-11-05 16:51:14 +07:00
menu->scroll_y = 0;
}
static void mui_navigation_set_last(void)
{
mui_navigation_set(true);
}
static void mui_navigation_alphabet(size_t *unused)
{
mui_navigation_set(true);
}
static void mui_populate_entries(const char *path,
const char *label, unsigned i)
{
menu_handle_t *menu = menu_driver_get_ptr();
if (!menu)
return;
menu->scroll_y = mui_get_scroll();
}
static void mui_context_reset(void)
{
2015-11-05 17:05:26 +07:00
char iconpath[PATH_MAX_LENGTH] = {0};
menu_handle_t *menu = menu_driver_get_ptr();
2015-11-05 17:14:51 +07:00
mui_handle_t *mui = menu ? (mui_handle_t*)menu->userdata : NULL;
2015-11-05 17:05:26 +07:00
settings_t *settings = config_get_ptr();
2015-05-01 09:02:03 -03:00
2015-11-05 17:14:51 +07:00
if (!mui || !settings)
return;
fill_pathname_join(iconpath, settings->assets_directory,
"glui", sizeof(iconpath));
fill_pathname_slash(iconpath, sizeof(iconpath));
mui_layout(menu, mui);
mui_context_bg_destroy(mui);
mui_allocate_white_texture(mui);
mui_context_reset_textures(mui, iconpath);
rarch_main_data_msg_queue_push(DATA_TYPE_IMAGE,
settings->menu.wallpaper, "cb_menu_wallpaper", 0, 1, true);
}
static int mui_environ(menu_environ_cb_t type, void *data)
2015-07-08 00:37:44 +02:00
{
switch (type)
{
2015-07-09 00:24:10 +02:00
case 0:
2015-07-08 00:37:44 +02:00
default:
break;
2015-07-08 00:37:44 +02:00
}
return -1;
2015-07-08 00:37:44 +02:00
}
static void mui_preswitch_tabs(unsigned action)
2015-11-02 01:44:45 +07:00
{
2015-11-05 17:05:26 +07:00
size_t stack_size = 0;
file_list_t *menu_stack = NULL;
menu_handle_t *menu = menu_driver_get_ptr();
2015-11-05 17:14:51 +07:00
mui_handle_t *mui = menu ? (mui_handle_t*)menu->userdata : NULL;
2015-11-02 01:44:45 +07:00
if (!mui)
2015-11-02 01:44:45 +07:00
return;
2015-11-05 16:51:14 +07:00
size_t idx = 0;
2015-11-03 17:06:53 +07:00
menu_navigation_ctl(MENU_NAVIGATION_CTL_SET_SELECTION, &idx);
menu_stack = menu_entries_get_menu_stack_ptr(0);
stack_size = menu_stack->size;
2015-11-02 01:44:45 +07:00
if (menu_stack->list[stack_size - 1].label)
free(menu_stack->list[stack_size - 1].label);
menu_stack->list[stack_size - 1].label = NULL;
switch (mui->categories.selection_ptr)
2015-11-02 01:44:45 +07:00
{
2015-11-05 16:53:14 +07:00
case MUI_SYSTEM_TAB_MAIN:
2015-11-02 01:44:45 +07:00
menu_stack->list[stack_size - 1].label =
strdup(menu_hash_to_str(MENU_VALUE_MAIN_MENU));
menu_stack->list[stack_size - 1].type =
MENU_SETTINGS;
break;
2015-11-05 16:53:14 +07:00
case MUI_SYSTEM_TAB_PLAYLISTS:
2015-11-02 01:44:45 +07:00
menu_stack->list[stack_size - 1].label =
strdup(menu_hash_to_str(MENU_VALUE_PLAYLISTS_TAB));
menu_stack->list[stack_size - 1].label =
strdup(menu_hash_to_str(MENU_VALUE_PLAYLISTS_TAB));
menu_stack->list[stack_size - 1].type =
MENU_PLAYLISTS_TAB;
break;
2015-11-05 16:53:14 +07:00
case MUI_SYSTEM_TAB_SETTINGS:
2015-11-02 01:44:45 +07:00
menu_stack->list[stack_size - 1].label =
strdup(menu_hash_to_str(MENU_VALUE_SETTINGS_TAB));
menu_stack->list[stack_size - 1].type =
MENU_SETTINGS;
break;
}
}
static void mui_list_cache(menu_list_type_t type, unsigned action)
{
2015-11-01 21:47:36 +01:00
size_t list_size;
2015-11-05 17:05:26 +07:00
menu_handle_t *menu = menu_driver_get_ptr();
2015-11-05 17:14:51 +07:00
mui_handle_t *mui = menu ? (mui_handle_t*)menu->userdata : NULL;
if (!mui)
return;
2015-11-05 16:53:14 +07:00
list_size = MUI_SYSTEM_TAB_END;
switch (type)
{
case MENU_LIST_PLAIN:
break;
case MENU_LIST_HORIZONTAL:
mui->categories.selection_ptr_old = mui->categories.selection_ptr;
switch (action)
{
case MENU_ACTION_LEFT:
if (mui->categories.selection_ptr == 0)
{
mui->categories.selection_ptr = list_size;
mui->categories.active.idx = list_size - 1;
}
else
mui->categories.selection_ptr--;
break;
default:
if (mui->categories.selection_ptr == list_size)
{
mui->categories.selection_ptr = 0;
mui->categories.active.idx = 1;
}
else
mui->categories.selection_ptr++;
break;
}
mui_preswitch_tabs(action);
break;
default:
break;
}
}
static int mui_list_push(menu_displaylist_info_t *info, unsigned type)
2015-10-29 13:35:15 +01:00
{
int ret = -1;
2015-11-05 16:51:14 +07:00
menu_handle_t *menu = menu_driver_get_ptr();
global_t *global = global_get_ptr();
2015-10-29 13:35:15 +01:00
switch (type)
{
case DISPLAYLIST_LOAD_CONTENT_LIST:
menu_entries_clear(info->list);
menu_entries_push(info->list,
menu_hash_to_str(MENU_LABEL_VALUE_LOAD_CONTENT),
menu_hash_to_str(MENU_LABEL_LOAD_CONTENT),
MENU_SETTING_ACTION, 0, 0);
if (core_info_list_num_info_files(global->core_info.list))
{
menu_entries_push(info->list,
menu_hash_to_str(MENU_LABEL_VALUE_DETECT_CORE_LIST),
menu_hash_to_str(MENU_LABEL_DETECT_CORE_LIST),
MENU_SETTING_ACTION, 0, 0);
menu_entries_push(info->list,
menu_hash_to_str(MENU_LABEL_VALUE_DOWNLOADED_FILE_DETECT_CORE_LIST),
menu_hash_to_str(MENU_LABEL_DOWNLOADED_FILE_DETECT_CORE_LIST),
MENU_SETTING_ACTION, 0, 0);
}
info->need_push = true;
info->need_refresh = true;
ret = 0;
break;
2015-10-29 13:35:15 +01:00
case DISPLAYLIST_MAIN_MENU:
menu_entries_clear(info->list);
if (global->inited.main && (global->inited.core.type != CORE_TYPE_DUMMY))
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_CONTENT_SETTINGS), PARSE_ACTION, false);
#if defined(HAVE_DYNAMIC) || defined(HAVE_LIBRETRO_MANAGEMENT)
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_CORE_LIST), PARSE_ACTION, false);
#endif
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_LOAD_CONTENT_LIST), PARSE_ACTION, false);
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_LOAD_CONTENT_HISTORY), PARSE_ACTION, false);
#if defined(HAVE_NETWORKING)
#if defined(HAVE_LIBRETRODB)
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_ADD_CONTENT_LIST), PARSE_ACTION, false);
#endif
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_ONLINE_UPDATER), PARSE_ACTION, false);
#endif
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_INFORMATION_LIST), PARSE_ACTION, false);
#ifndef HAVE_DYNAMIC
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_RESTART_RETROARCH), PARSE_ACTION, false);
#endif
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_CONFIGURATIONS), PARSE_ACTION, false);
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_SAVE_NEW_CONFIG), PARSE_ACTION, false);
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_HELP_LIST), PARSE_ACTION, false);
#if !defined(IOS)
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_QUIT_RETROARCH), PARSE_ACTION, false);
#endif
menu_displaylist_parse_settings(menu, info,
menu_hash_to_str(MENU_LABEL_SHUTDOWN), PARSE_ACTION, false);
info->need_push = true;
ret = 0;
break;
}
return ret;
}
static size_t mui_list_get_selection(void *data)
{
2015-11-05 17:05:26 +07:00
menu_handle_t *menu = (menu_handle_t*)data;
mui_handle_t *mui = menu ? (mui_handle_t*)menu->userdata : NULL;
if (!mui)
return 0;
return mui->categories.selection_ptr;
}
static int mui_pointer_tap(unsigned x, unsigned y,
2015-11-01 20:44:04 +01:00
unsigned ptr, menu_file_list_cbs_t *cbs,
menu_entry_t *entry, unsigned action)
{
size_t selection, idx;
2015-11-02 01:44:45 +07:00
unsigned header_height, width, height, i;
bool scroll = false;
menu_handle_t *menu = menu_driver_get_ptr();
2015-11-05 17:05:26 +07:00
mui_handle_t *mui = menu ? (mui_handle_t*)menu->userdata : NULL;
2015-11-02 01:44:45 +07:00
file_list_t *menu_stack = menu_entries_get_menu_stack_ptr(0);
file_list_t *selection_buf = menu_entries_get_selection_buf_ptr(0);
if (!mui)
2015-11-02 01:44:45 +07:00
return 0;
video_driver_get_size(&width, &height);
menu_navigation_ctl(MENU_NAVIGATION_CTL_GET_SELECTION, &selection);
menu_display_ctl(MENU_DISPLAY_CTL_HEADER_HEIGHT, &header_height);
2015-11-01 20:44:04 +01:00
if (y < header_height)
{
menu_entries_pop_stack(&selection, 0);
menu_navigation_ctl(MENU_NAVIGATION_CTL_SET_SELECTION, &selection);
}
else if (y > height - mui->tabs_height)
2015-11-02 01:44:45 +07:00
{
2015-11-05 16:53:14 +07:00
for (i = 0; i <= MUI_SYSTEM_TAB_END; i++)
2015-11-02 01:44:45 +07:00
{
2015-11-05 16:53:14 +07:00
unsigned tab_width = width / (MUI_SYSTEM_TAB_END + 1);
2015-11-02 01:44:45 +07:00
unsigned start = tab_width * i;
2015-11-01 20:44:04 +01:00
if ((x >= start) && (x < (start + tab_width)))
2015-11-02 01:44:45 +07:00
{
mui->categories.selection_ptr = i;
2015-11-02 01:44:45 +07:00
mui_preswitch_tabs(action);
2015-11-02 01:44:45 +07:00
if (cbs && cbs->action_content_list_switch)
return cbs->action_content_list_switch(selection_buf, menu_stack,
"", "", 0);
}
}
}
2015-11-01 20:44:04 +01:00
else if (ptr <= (menu_entries_get_size() - 1))
{
2015-11-01 20:44:04 +01:00
if (ptr == selection && cbs && cbs->action_select)
return menu_entry_action(entry, selection, MENU_ACTION_SELECT);
2015-11-01 20:44:04 +01:00
idx = ptr;
menu_navigation_ctl(MENU_NAVIGATION_CTL_SET_SELECTION, &idx);
menu_navigation_ctl(MENU_NAVIGATION_CTL_SET, &scroll);
}
return 0;
}
menu_ctx_driver_t menu_ctx_mui = {
NULL,
mui_get_message,
generic_menu_iterate,
mui_render,
mui_frame,
mui_init,
mui_free,
mui_context_reset,
mui_context_destroy,
mui_populate_entries,
NULL,
mui_navigation_clear,
NULL,
NULL,
mui_navigation_set,
mui_navigation_set_last,
mui_navigation_alphabet,
mui_navigation_alphabet,
2015-10-03 04:32:38 +02:00
generic_menu_init_list,
NULL,
NULL,
NULL,
mui_list_cache,
mui_list_push,
mui_list_get_selection,
mui_list_get_size,
NULL,
mui_list_set_selection,
2015-06-15 19:00:52 +02:00
NULL,
mui_load_image,
"glui",
2015-08-17 05:05:29 +02:00
MENU_VIDEO_DRIVER_OPENGL,
mui_environ,
mui_pointer_tap,
};