mirror of
https://github.com/libretro/RetroArch.git
synced 2024-12-04 22:46:46 +00:00
2243 lines
74 KiB
C
2243 lines
74 KiB
C
/* RetroArch - A frontend for libretro.
|
|
* Copyright (C) 2014-2017 - Jean-André Santoni
|
|
* Copyright (C) 2015-2018 - Andre Leiradella
|
|
* Copyright (C) 2018-2020 - natinusala
|
|
*
|
|
* 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/>.
|
|
*/
|
|
|
|
#include <retro_miscellaneous.h>
|
|
#include <retro_inline.h>
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "../config.h"
|
|
#endif
|
|
|
|
#include <queues/fifo_queue.h>
|
|
#include <file/file_path.h>
|
|
#include <streams/file_stream.h>
|
|
#include <string/stdstring.h>
|
|
#include <retro_math.h>
|
|
|
|
#include "gfx_display.h"
|
|
#include "gfx_widgets.h"
|
|
#include "font_driver.h"
|
|
|
|
#ifdef HAVE_MENU
|
|
#include "../menu/menu_defines.h"
|
|
#endif
|
|
|
|
#include "../configuration.h"
|
|
#include "../file_path_special.h"
|
|
#include "../msg_hash.h"
|
|
|
|
#include "../tasks/task_content.h"
|
|
#include "../tasks/tasks_internal.h"
|
|
|
|
#define BASE_FONT_SIZE 32.0f
|
|
|
|
#define MSG_QUEUE_FONT_SIZE 20.0f
|
|
|
|
/* Icons */
|
|
static const char
|
|
*gfx_widgets_icons_names[MENU_WIDGETS_ICON_LAST] = {
|
|
"menu_pause.png",
|
|
"menu_frameskip.png",
|
|
"menu_rewind.png",
|
|
"resume.png",
|
|
|
|
"menu_hourglass.png",
|
|
"menu_check.png",
|
|
"menu_add.png",
|
|
"menu_exit.png",
|
|
|
|
"menu_info.png",
|
|
|
|
"menu_achievements.png"
|
|
};
|
|
|
|
static dispgfx_widget_t dispwidget_st = {0}; /* uint64_t alignment */
|
|
|
|
static void INLINE gfx_widgets_font_free(gfx_widget_font_data_t *font_data)
|
|
{
|
|
if (font_data->font)
|
|
font_driver_free(font_data->font);
|
|
|
|
font_data->font = NULL;
|
|
font_data->usage_count = 0;
|
|
}
|
|
|
|
/* Widgets list */
|
|
const static gfx_widget_t* const widgets[] = {
|
|
#ifdef HAVE_NETWORKING
|
|
&gfx_widget_netplay_chat,
|
|
&gfx_widget_netplay_ping,
|
|
#endif
|
|
#ifdef HAVE_SCREENSHOTS
|
|
&gfx_widget_screenshot,
|
|
#endif
|
|
&gfx_widget_volume,
|
|
#ifdef HAVE_CHEEVOS
|
|
&gfx_widget_achievement_popup,
|
|
&gfx_widget_leaderboard_display,
|
|
#endif
|
|
&gfx_widget_generic_message,
|
|
&gfx_widget_libretro_message,
|
|
&gfx_widget_progress_message,
|
|
&gfx_widget_load_content_animation
|
|
};
|
|
|
|
#if defined(HAVE_MENU) && defined(HAVE_XMB)
|
|
static float gfx_display_get_widget_pixel_scale(
|
|
gfx_display_t *p_disp,
|
|
settings_t *settings,
|
|
unsigned width, unsigned height, bool fullscreen)
|
|
{
|
|
static unsigned last_width = 0;
|
|
static unsigned last_height = 0;
|
|
static float scale = 0.0f;
|
|
static bool scale_cached = false;
|
|
bool scale_updated = false;
|
|
static float last_menu_scale_factor = 0.0f;
|
|
static enum menu_driver_id_type last_menu_driver_id = MENU_DRIVER_ID_UNKNOWN;
|
|
static float adjusted_scale = 1.0f;
|
|
bool gfx_widget_scale_auto = settings->bools.menu_widget_scale_auto;
|
|
#if (defined(RARCH_CONSOLE) || defined(RARCH_MOBILE))
|
|
float menu_widget_scale_factor = settings->floats.menu_widget_scale_factor;
|
|
#else
|
|
float menu_widget_scale_factor_fullscreen = settings->floats.menu_widget_scale_factor;
|
|
float menu_widget_scale_factor_windowed = settings->floats.menu_widget_scale_factor_windowed;
|
|
float menu_widget_scale_factor = fullscreen
|
|
? menu_widget_scale_factor_fullscreen
|
|
: menu_widget_scale_factor_windowed;
|
|
#endif
|
|
float menu_scale_factor = menu_widget_scale_factor;
|
|
|
|
if (gfx_widget_scale_auto)
|
|
menu_scale_factor = settings->floats.menu_scale_factor;
|
|
|
|
/* We need to perform a square root here, which
|
|
* can be slow on some platforms (not *slow*, but
|
|
* it involves enough work that it's worth trying
|
|
* to optimise). We therefore cache the pixel scale,
|
|
* and only update on first run or when the video
|
|
* size changes */
|
|
if ( !scale_cached
|
|
|| (width != last_width)
|
|
|| (height != last_height))
|
|
{
|
|
/* Baseline reference is a 1080p display */
|
|
scale = (float)(
|
|
sqrt((double)((width * width) + (height * height))) /
|
|
DIAGONAL_PIXELS_1080P);
|
|
|
|
scale_cached = true;
|
|
scale_updated = true;
|
|
last_width = width;
|
|
last_height = height;
|
|
}
|
|
|
|
/* Adjusted scale calculation may also be slow, so
|
|
* only update if something changes */
|
|
if ( scale_updated
|
|
|| (menu_scale_factor != last_menu_scale_factor)
|
|
|| (p_disp->menu_driver_id != last_menu_driver_id))
|
|
{
|
|
adjusted_scale = scale * menu_scale_factor;
|
|
adjusted_scale = (adjusted_scale > 0.0001f) ? adjusted_scale : 1.0f;
|
|
last_menu_scale_factor = menu_scale_factor;
|
|
last_menu_driver_id = p_disp->menu_driver_id;
|
|
}
|
|
|
|
return adjusted_scale;
|
|
}
|
|
#endif
|
|
|
|
static void msg_widget_msg_transition_animation_done(void *userdata)
|
|
{
|
|
disp_widget_msg_t *msg = (disp_widget_msg_t*)userdata;
|
|
|
|
if (msg->msg)
|
|
free(msg->msg);
|
|
msg->msg = NULL;
|
|
|
|
if (msg->msg_new)
|
|
msg->msg = strdup(msg->msg_new);
|
|
|
|
msg->msg_transition_animation = 0.0f;
|
|
}
|
|
|
|
void gfx_widgets_msg_queue_push(
|
|
retro_task_t *task,
|
|
const char *msg,
|
|
unsigned duration,
|
|
char *title,
|
|
enum message_queue_icon icon,
|
|
enum message_queue_category category,
|
|
unsigned prio, bool flush,
|
|
bool menu_is_alive)
|
|
{
|
|
disp_widget_msg_t *msg_widget = NULL;
|
|
dispgfx_widget_t *p_dispwidget = &dispwidget_st;
|
|
|
|
if (FIFO_WRITE_AVAIL_NONPTR(p_dispwidget->msg_queue) > 0)
|
|
{
|
|
/* Get current msg if it exists */
|
|
if (task && task->frontend_userdata)
|
|
{
|
|
msg_widget = (disp_widget_msg_t*)task->frontend_userdata;
|
|
/* msg_widgets can be passed between tasks */
|
|
msg_widget->task_ptr = task;
|
|
}
|
|
|
|
/* Spawn a new notification */
|
|
if (!msg_widget)
|
|
{
|
|
const char *title = msg;
|
|
size_t title_length = strlen(title);
|
|
|
|
msg_widget = (disp_widget_msg_t*)malloc(sizeof(*msg_widget));
|
|
|
|
msg_widget->msg = NULL;
|
|
msg_widget->msg_new = NULL;
|
|
msg_widget->msg_transition_animation = 0.0f;
|
|
msg_widget->msg_len = 0;
|
|
msg_widget->duration = duration;
|
|
|
|
msg_widget->text_height = 0;
|
|
|
|
msg_widget->offset_y = 0;
|
|
msg_widget->alpha = 1.0f;
|
|
|
|
msg_widget->width = 0;
|
|
|
|
msg_widget->expiration_timer = 0;
|
|
|
|
msg_widget->task_ptr = task;
|
|
msg_widget->task_count = 0;
|
|
|
|
msg_widget->task_progress = 0;
|
|
msg_widget->task_ident = 0;
|
|
|
|
msg_widget->unfold = 0.0f;
|
|
|
|
msg_widget->hourglass_rotation = 0.0f;
|
|
msg_widget->hourglass_timer = 0.0f;
|
|
msg_widget->flags = 0;
|
|
|
|
if (!(p_dispwidget->flags & DISPGFX_WIDGET_FLAG_MSG_QUEUE_HAS_ICONS))
|
|
{
|
|
msg_widget->flags |= DISPWIDG_FLAG_UNFOLDED;
|
|
msg_widget->flags &= ~DISPWIDG_FLAG_UNFOLDING;
|
|
msg_widget->unfold = 1.0f;
|
|
}
|
|
|
|
if (task)
|
|
{
|
|
title = msg_widget->msg = strdup(task->title);
|
|
msg_widget->msg_new = strdup(title);
|
|
title_length = strlen(title);
|
|
msg_widget->msg_len = title_length;
|
|
|
|
if (!string_is_empty(task->error))
|
|
msg_widget->flags |= DISPWIDG_FLAG_TASK_ERROR;
|
|
if (task->cancelled)
|
|
msg_widget->flags |= DISPWIDG_FLAG_TASK_CANCELLED;
|
|
if (task->finished)
|
|
msg_widget->flags |= DISPWIDG_FLAG_TASK_FINISHED;
|
|
msg_widget->task_progress = task->progress;
|
|
msg_widget->task_ident = task->ident;
|
|
msg_widget->task_count = 1;
|
|
|
|
msg_widget->flags |= DISPWIDG_FLAG_UNFOLDED;
|
|
|
|
if (task->style == TASK_STYLE_POSITIVE)
|
|
msg_widget->flags |= DISPWIDG_FLAG_POSITIVE;
|
|
else if (task->style == TASK_STYLE_NEGATIVE)
|
|
msg_widget->flags |= DISPWIDG_FLAG_NEGATIVE;
|
|
|
|
msg_widget->width = font_driver_get_message_width(
|
|
p_dispwidget->gfx_widget_fonts.msg_queue.font,
|
|
title,
|
|
msg_widget->msg_len, 1.0f) +
|
|
p_dispwidget->simple_widget_padding / 2;
|
|
|
|
task->frontend_userdata = msg_widget;
|
|
|
|
msg_widget->hourglass_rotation = 0;
|
|
}
|
|
else
|
|
{
|
|
/* Compute rect width, wrap if necessary */
|
|
/* Single line text > two lines text > two lines
|
|
* text with expanded width */
|
|
char *msg = NULL;
|
|
size_t msg_len = 0;
|
|
unsigned width = menu_is_alive
|
|
? p_dispwidget->msg_queue_default_rect_width_menu_alive
|
|
: p_dispwidget->msg_queue_default_rect_width;
|
|
unsigned text_width = font_driver_get_message_width(
|
|
p_dispwidget->gfx_widget_fonts.msg_queue.font,
|
|
title,
|
|
title_length,
|
|
1.0f);
|
|
msg_widget->text_height = p_dispwidget->gfx_widget_fonts.msg_queue.line_height;
|
|
/* 1 byte uses for inserting '\n' */
|
|
msg_len = title_length + 1 + 1;
|
|
if (!(msg = (char *)malloc(msg_len)))
|
|
return;
|
|
msg[0] = '\0';
|
|
|
|
/* Text is too wide, split it into two lines */
|
|
if (text_width > width)
|
|
{
|
|
size_t wrap_length = 0;
|
|
|
|
/* If the second line is too short, the widget may
|
|
* look unappealing - ensure that second line is at
|
|
* least 25% of the total width */
|
|
if ((text_width - (text_width >> 2)) < width)
|
|
width = text_width - (text_width >> 2);
|
|
|
|
word_wrap(msg, msg_len, title, title_length,
|
|
(int)((title_length * width) / text_width),
|
|
100, 2);
|
|
|
|
/* Recalculate widget width with longest wrapped line */
|
|
wrap_length = string_index_last_occurance(msg, '\n');
|
|
if (wrap_length)
|
|
{
|
|
title_length -= wrap_length;
|
|
|
|
if (title_length < wrap_length)
|
|
title_length = wrap_length;
|
|
|
|
text_width = font_driver_get_message_width(
|
|
p_dispwidget->gfx_widget_fonts.msg_queue.font,
|
|
title,
|
|
title_length,
|
|
1.0f);
|
|
|
|
width = text_width;
|
|
}
|
|
|
|
msg_widget->text_height *= 2;
|
|
msg_widget->msg_len = strlen(msg);
|
|
}
|
|
else
|
|
{
|
|
width = text_width;
|
|
msg_widget->msg_len = strlcpy(msg, title, msg_len);
|
|
}
|
|
|
|
msg_widget->msg = msg;
|
|
msg_widget->width = width + (p_dispwidget->simple_widget_padding / 2);
|
|
}
|
|
|
|
fifo_write(&p_dispwidget->msg_queue,
|
|
&msg_widget, sizeof(msg_widget));
|
|
}
|
|
/* Update task info */
|
|
else
|
|
{
|
|
if (msg_widget->flags & DISPWIDG_FLAG_EXPIRATION_TIMER_STARTED)
|
|
{
|
|
uintptr_t _tag = (uintptr_t)&msg_widget->expiration_timer;
|
|
gfx_animation_kill_by_tag(&_tag);
|
|
msg_widget->flags &= ~DISPWIDG_FLAG_EXPIRATION_TIMER_STARTED;
|
|
}
|
|
|
|
if (!string_is_equal(task->title, msg_widget->msg_new))
|
|
{
|
|
size_t len;
|
|
unsigned new_width;
|
|
|
|
if (msg_widget->msg_new)
|
|
{
|
|
free(msg_widget->msg_new);
|
|
msg_widget->msg_new = NULL;
|
|
}
|
|
|
|
title = msg_widget->msg_new = strdup(task->title);
|
|
|
|
len = strlen(title);
|
|
new_width = font_driver_get_message_width(
|
|
p_dispwidget->gfx_widget_fonts.msg_queue.font,
|
|
title,
|
|
len,
|
|
1.0f);
|
|
|
|
msg_widget->msg_len = len;
|
|
msg_widget->msg_transition_animation = 0;
|
|
|
|
if (!task->alternative_look)
|
|
{
|
|
gfx_animation_ctx_entry_t entry;
|
|
|
|
entry.easing_enum = EASING_OUT_QUAD;
|
|
entry.tag = (uintptr_t)msg_widget;
|
|
entry.duration = MSG_QUEUE_ANIMATION_DURATION*2;
|
|
entry.target_value = p_dispwidget->msg_queue_height / 2.0f;
|
|
entry.subject = &msg_widget->msg_transition_animation;
|
|
entry.cb = msg_widget_msg_transition_animation_done;
|
|
entry.userdata = msg_widget;
|
|
|
|
gfx_animation_push(&entry);
|
|
}
|
|
else
|
|
msg_widget_msg_transition_animation_done(msg_widget);
|
|
|
|
msg_widget->task_count++;
|
|
|
|
msg_widget->width = new_width;
|
|
}
|
|
|
|
if (!string_is_empty(task->error))
|
|
msg_widget->flags |= DISPWIDG_FLAG_TASK_ERROR;
|
|
if (task->cancelled)
|
|
msg_widget->flags |= DISPWIDG_FLAG_TASK_CANCELLED;
|
|
if (task->finished)
|
|
msg_widget->flags |= DISPWIDG_FLAG_TASK_FINISHED;
|
|
msg_widget->task_progress = task->progress;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void gfx_widgets_unfold_end(void *userdata)
|
|
{
|
|
disp_widget_msg_t *unfold = (disp_widget_msg_t*)userdata;
|
|
dispgfx_widget_t *p_dispwidget = &dispwidget_st;
|
|
|
|
unfold->flags &= ~DISPWIDG_FLAG_UNFOLDING;
|
|
p_dispwidget->flags &= ~DISPGFX_WIDGET_FLAG_MOVING;
|
|
}
|
|
|
|
static void gfx_widgets_move_end(void *userdata)
|
|
{
|
|
dispgfx_widget_t *p_dispwidget = &dispwidget_st;
|
|
|
|
if (userdata)
|
|
{
|
|
gfx_animation_ctx_entry_t entry;
|
|
disp_widget_msg_t *unfold = (disp_widget_msg_t*)userdata;
|
|
|
|
entry.cb = gfx_widgets_unfold_end;
|
|
entry.duration = MSG_QUEUE_ANIMATION_DURATION;
|
|
entry.easing_enum = EASING_OUT_QUAD;
|
|
entry.subject = &unfold->unfold;
|
|
entry.tag = (uintptr_t)unfold;
|
|
entry.target_value = 1.0f;
|
|
entry.userdata = unfold;
|
|
|
|
gfx_animation_push(&entry);
|
|
|
|
unfold->flags |= DISPWIDG_FLAG_UNFOLDED
|
|
| DISPWIDG_FLAG_UNFOLDING;
|
|
}
|
|
else
|
|
p_dispwidget->flags &= ~DISPGFX_WIDGET_FLAG_MOVING;
|
|
}
|
|
|
|
static void gfx_widgets_msg_queue_expired(void *userdata)
|
|
{
|
|
disp_widget_msg_t *msg = (disp_widget_msg_t *)userdata;
|
|
|
|
if (msg && !(msg->flags & DISPWIDG_FLAG_EXPIRED))
|
|
msg->flags |= DISPWIDG_FLAG_EXPIRED;
|
|
}
|
|
|
|
static void gfx_widgets_msg_queue_move(dispgfx_widget_t *p_dispwidget)
|
|
{
|
|
int i;
|
|
float y = 0;
|
|
/* there should always be one and only one unfolded message */
|
|
disp_widget_msg_t *unfold = NULL;
|
|
|
|
#ifdef HAVE_THREADS
|
|
slock_lock(p_dispwidget->current_msgs_lock);
|
|
#endif
|
|
|
|
for (i = (int)(p_dispwidget->current_msgs_size - 1); i >= 0; i--)
|
|
{
|
|
disp_widget_msg_t* msg = p_dispwidget->current_msgs[i];
|
|
|
|
if (!msg || (msg->flags & DISPWIDG_FLAG_DYING))
|
|
continue;
|
|
|
|
if (y == 0)
|
|
y += (p_dispwidget->msg_queue_padding * 4.0f);
|
|
|
|
y += (p_dispwidget->msg_queue_height / 2.0f / (msg->task_ptr ? 2.0f : 1.0f))
|
|
+ (p_dispwidget->msg_queue_spacing * (msg->task_ptr ? 1.0f : 2.0f))
|
|
+ floor(p_dispwidget->divider_width_1px);
|
|
|
|
if (!(msg->flags & DISPWIDG_FLAG_UNFOLDED))
|
|
unfold = msg;
|
|
|
|
if (msg->offset_y != y)
|
|
{
|
|
gfx_animation_ctx_entry_t entry;
|
|
|
|
entry.cb = (i == 0) ? gfx_widgets_move_end : NULL;
|
|
entry.duration = MSG_QUEUE_ANIMATION_DURATION;
|
|
entry.easing_enum = EASING_OUT_QUAD;
|
|
entry.subject = &msg->offset_y;
|
|
entry.tag = (uintptr_t)msg;
|
|
entry.target_value = ceilf(y);
|
|
entry.userdata = unfold;
|
|
|
|
gfx_animation_push(&entry);
|
|
|
|
p_dispwidget->flags |= DISPGFX_WIDGET_FLAG_MOVING;
|
|
}
|
|
}
|
|
|
|
#ifdef HAVE_THREADS
|
|
slock_unlock(p_dispwidget->current_msgs_lock);
|
|
#endif
|
|
}
|
|
|
|
static void gfx_widgets_msg_queue_free(
|
|
dispgfx_widget_t *p_dispwidget,
|
|
disp_widget_msg_t *msg)
|
|
{
|
|
uintptr_t tag = (uintptr_t)msg;
|
|
uintptr_t hourglass_timer_tag = (uintptr_t)&msg->hourglass_timer;
|
|
|
|
if (msg->task_ptr)
|
|
{
|
|
/* remove the reference the task has of ourself
|
|
only if the task is not finished already
|
|
(finished tasks are freed before the widget) */
|
|
if ( !(msg->flags & DISPWIDG_FLAG_TASK_FINISHED)
|
|
&& !(msg->flags & DISPWIDG_FLAG_TASK_ERROR)
|
|
&& !(msg->flags & DISPWIDG_FLAG_TASK_CANCELLED))
|
|
msg->task_ptr->frontend_userdata = NULL;
|
|
|
|
/* update tasks count */
|
|
p_dispwidget->msg_queue_tasks_count--;
|
|
}
|
|
|
|
/* Kill all animations */
|
|
gfx_animation_kill_by_tag(&hourglass_timer_tag);
|
|
gfx_animation_kill_by_tag(&tag);
|
|
|
|
/* Kill all timers */
|
|
if (msg->flags & DISPWIDG_FLAG_EXPIRATION_TIMER_STARTED)
|
|
{
|
|
uintptr_t _tag = (uintptr_t)&msg->expiration_timer;
|
|
gfx_animation_kill_by_tag(&_tag);
|
|
}
|
|
|
|
/* Free it */
|
|
if (msg->msg)
|
|
free(msg->msg);
|
|
|
|
if (msg->msg_new)
|
|
free(msg->msg_new);
|
|
|
|
p_dispwidget->flags &= ~DISPGFX_WIDGET_FLAG_MOVING;
|
|
}
|
|
|
|
static void gfx_widgets_msg_queue_kill_end(void *userdata)
|
|
{
|
|
disp_widget_msg_t* msg;
|
|
dispgfx_widget_t *p_dispwidget = &dispwidget_st;
|
|
|
|
#ifdef HAVE_THREADS
|
|
slock_lock(p_dispwidget->current_msgs_lock);
|
|
#endif
|
|
|
|
if ((msg = p_dispwidget->current_msgs[p_dispwidget->msg_queue_kill]))
|
|
{
|
|
int i;
|
|
/* Remove it from the list */
|
|
for (i = p_dispwidget->msg_queue_kill; i < (int)(p_dispwidget->current_msgs_size - 1); i++)
|
|
p_dispwidget->current_msgs[i] = p_dispwidget->current_msgs[i + 1];
|
|
|
|
p_dispwidget->current_msgs_size--;
|
|
p_dispwidget->current_msgs[p_dispwidget->current_msgs_size] = NULL;
|
|
|
|
/* clean up the item */
|
|
gfx_widgets_msg_queue_free(p_dispwidget, msg);
|
|
|
|
/* free the associated memory */
|
|
free(msg);
|
|
}
|
|
|
|
#ifdef HAVE_THREADS
|
|
slock_unlock(p_dispwidget->current_msgs_lock);
|
|
#endif
|
|
}
|
|
|
|
static void gfx_widgets_msg_queue_kill(
|
|
dispgfx_widget_t *p_dispwidget,
|
|
unsigned idx)
|
|
{
|
|
gfx_animation_ctx_entry_t entry;
|
|
disp_widget_msg_t *msg = p_dispwidget->current_msgs[idx];
|
|
|
|
if (!msg)
|
|
return;
|
|
|
|
p_dispwidget->flags |= DISPGFX_WIDGET_FLAG_MOVING;
|
|
msg->flags |= DISPWIDG_FLAG_DYING;
|
|
|
|
p_dispwidget->msg_queue_kill = idx;
|
|
|
|
/* Drop down */
|
|
entry.cb = NULL;
|
|
entry.duration = MSG_QUEUE_ANIMATION_DURATION;
|
|
entry.easing_enum = EASING_OUT_QUAD;
|
|
entry.tag = (uintptr_t)msg;
|
|
entry.userdata = NULL;
|
|
entry.subject = &msg->offset_y;
|
|
entry.target_value = msg->offset_y -
|
|
p_dispwidget->msg_queue_height / 4;
|
|
|
|
gfx_animation_push(&entry);
|
|
|
|
/* Fade out */
|
|
entry.cb = gfx_widgets_msg_queue_kill_end;
|
|
entry.subject = &msg->alpha;
|
|
entry.target_value = 0.0f;
|
|
|
|
gfx_animation_push(&entry);
|
|
|
|
/* Move all messages back to their correct position */
|
|
if (p_dispwidget->current_msgs_size != 0)
|
|
gfx_widgets_msg_queue_move(p_dispwidget);
|
|
}
|
|
|
|
void gfx_widgets_draw_icon(
|
|
void *userdata,
|
|
void *data_disp,
|
|
unsigned video_width,
|
|
unsigned video_height,
|
|
unsigned icon_width,
|
|
unsigned icon_height,
|
|
uintptr_t texture,
|
|
float x, float y,
|
|
float radians,
|
|
float cosine,
|
|
float sine,
|
|
float *color)
|
|
{
|
|
gfx_display_ctx_draw_t draw;
|
|
struct video_coords coords;
|
|
math_matrix_4x4 mymat;
|
|
gfx_display_t *p_disp = (gfx_display_t*)data_disp;
|
|
gfx_display_ctx_driver_t *dispctx = p_disp->dispctx;
|
|
|
|
if (!texture)
|
|
return;
|
|
|
|
if (!p_disp->dispctx->handles_transform)
|
|
gfx_display_rotate_z(p_disp, &mymat, cosine, sine, userdata);
|
|
|
|
coords.vertices = 4;
|
|
coords.vertex = NULL;
|
|
coords.tex_coord = NULL;
|
|
coords.lut_tex_coord = NULL;
|
|
coords.color = color;
|
|
|
|
draw.x = x;
|
|
draw.y = video_height - y - icon_height;
|
|
draw.width = icon_width;
|
|
draw.height = icon_height;
|
|
draw.scale_factor = 1.0f;
|
|
draw.rotation = radians;
|
|
draw.coords = &coords;
|
|
draw.matrix_data = &mymat;
|
|
draw.texture = texture;
|
|
draw.prim_type = GFX_DISPLAY_PRIM_TRIANGLESTRIP;
|
|
draw.pipeline_id = 0;
|
|
|
|
if (draw.height > 0 && draw.width > 0)
|
|
if (dispctx->draw)
|
|
dispctx->draw(&draw, userdata, video_width, video_height);
|
|
}
|
|
|
|
void gfx_widgets_draw_text(
|
|
gfx_widget_font_data_t* font_data,
|
|
const char *text,
|
|
float x, float y,
|
|
int width, int height,
|
|
uint32_t color,
|
|
enum text_alignment text_align,
|
|
bool draw_outside)
|
|
{
|
|
if (!font_data || string_is_empty(text))
|
|
return;
|
|
|
|
gfx_display_draw_text(
|
|
font_data->font,
|
|
text,
|
|
x, y,
|
|
width, height,
|
|
color,
|
|
text_align,
|
|
1.0f,
|
|
false,
|
|
0.0f,
|
|
draw_outside);
|
|
|
|
font_data->usage_count++;
|
|
}
|
|
|
|
void gfx_widgets_flush_text(
|
|
unsigned video_width, unsigned video_height,
|
|
gfx_widget_font_data_t* font_data)
|
|
{
|
|
/* Flushing is slow - only do it if font
|
|
* has actually been used */
|
|
if (!font_data || (font_data->usage_count == 0))
|
|
return;
|
|
|
|
if (font_data->font && font_data->font->renderer && font_data->font->renderer->flush)
|
|
font_data->font->renderer->flush(video_width, video_height, font_data->font->renderer_data);
|
|
font_data->raster_block.carr.coords.vertices = 0;
|
|
font_data->usage_count = 0;
|
|
}
|
|
|
|
float gfx_widgets_get_thumbnail_scale_factor(
|
|
const float dst_width, const float dst_height,
|
|
const float image_width, const float image_height)
|
|
{
|
|
float dst_ratio = dst_width / dst_height;
|
|
float image_ratio = image_width / image_height;
|
|
|
|
if (dst_ratio > image_ratio)
|
|
return (dst_height / image_height);
|
|
return (dst_width / image_width);
|
|
}
|
|
|
|
static void gfx_widgets_start_msg_expiration_timer(
|
|
disp_widget_msg_t *msg_widget, unsigned duration)
|
|
{
|
|
gfx_timer_ctx_entry_t timer;
|
|
|
|
timer.cb = gfx_widgets_msg_queue_expired;
|
|
timer.duration = duration;
|
|
timer.userdata = msg_widget;
|
|
|
|
gfx_animation_timer_start(&msg_widget->expiration_timer, &timer);
|
|
|
|
msg_widget->flags |=
|
|
DISPWIDG_FLAG_EXPIRATION_TIMER_STARTED;
|
|
}
|
|
|
|
static void gfx_widgets_hourglass_tick(void *userdata);
|
|
|
|
static void gfx_widgets_hourglass_end(void *userdata)
|
|
{
|
|
gfx_timer_ctx_entry_t timer;
|
|
disp_widget_msg_t *msg = (disp_widget_msg_t*)userdata;
|
|
|
|
msg->hourglass_rotation = 0.0f;
|
|
|
|
timer.cb = gfx_widgets_hourglass_tick;
|
|
timer.duration = HOURGLASS_INTERVAL;
|
|
timer.userdata = msg;
|
|
|
|
gfx_animation_timer_start(&msg->hourglass_timer, &timer);
|
|
}
|
|
|
|
static void gfx_widgets_hourglass_tick(void *userdata)
|
|
{
|
|
gfx_animation_ctx_entry_t entry;
|
|
disp_widget_msg_t *msg = (disp_widget_msg_t*)userdata;
|
|
uintptr_t tag = (uintptr_t)msg;
|
|
|
|
entry.easing_enum = EASING_OUT_QUAD;
|
|
entry.tag = tag;
|
|
entry.duration = HOURGLASS_DURATION;
|
|
entry.target_value = -(2 * M_PI);
|
|
entry.subject = &msg->hourglass_rotation;
|
|
entry.cb = gfx_widgets_hourglass_end;
|
|
entry.userdata = msg;
|
|
|
|
gfx_animation_push(&entry);
|
|
}
|
|
|
|
static void gfx_widgets_font_init(
|
|
gfx_display_t *p_disp,
|
|
dispgfx_widget_t *p_dispwidget,
|
|
gfx_widget_font_data_t *font_data,
|
|
bool is_threaded, char *font_path, float font_size)
|
|
{
|
|
int glyph_width = 0;
|
|
float scaled_size = font_size * p_dispwidget->last_scale_factor;
|
|
|
|
/* Limit minimum font size to keep it readable */
|
|
if (scaled_size < 9)
|
|
scaled_size = 9;
|
|
|
|
/* Free existing font */
|
|
if (font_data->font)
|
|
{
|
|
font_driver_free(font_data->font);
|
|
font_data->font = NULL;
|
|
}
|
|
|
|
/* Get approximate glyph width */
|
|
font_data->glyph_width = scaled_size * (3.0f / 4.0f);
|
|
|
|
/* Create font */
|
|
font_data->font = gfx_display_font_file(p_disp,
|
|
font_path, scaled_size, is_threaded);
|
|
|
|
/* Get font metadata */
|
|
glyph_width = font_driver_get_message_width(font_data->font, "a", 1, 1.0f);
|
|
if (glyph_width > 0)
|
|
font_data->glyph_width = (float)glyph_width;
|
|
font_data->line_height = (float)font_driver_get_line_height(font_data->font, 1.0f);
|
|
font_data->line_ascender = (float)font_driver_get_line_ascender(font_data->font, 1.0f);
|
|
font_data->line_descender = (float)font_driver_get_line_descender(font_data->font, 1.0f);
|
|
font_data->line_centre_offset = (float)font_driver_get_line_centre_offset(font_data->font, 1.0f);
|
|
|
|
font_data->usage_count = 0;
|
|
}
|
|
|
|
static void gfx_widgets_layout(
|
|
gfx_display_t *p_disp,
|
|
dispgfx_widget_t *p_dispwidget,
|
|
bool is_threaded, const char *dir_assets, char *font_path)
|
|
{
|
|
size_t i;
|
|
|
|
/* Initialise fonts */
|
|
if (string_is_empty(font_path))
|
|
{
|
|
char font_file[PATH_MAX_LENGTH];
|
|
/* Create regular font */
|
|
gfx_widgets_font_init(p_disp, p_dispwidget,
|
|
&p_dispwidget->gfx_widget_fonts.regular,
|
|
is_threaded, p_dispwidget->ozone_regular_font_path, BASE_FONT_SIZE);
|
|
/* Create bold font */
|
|
gfx_widgets_font_init(p_disp, p_dispwidget,
|
|
&p_dispwidget->gfx_widget_fonts.bold,
|
|
is_threaded, p_dispwidget->ozone_bold_font_path, BASE_FONT_SIZE);
|
|
|
|
/* Create msg_queue font */
|
|
switch (*msg_hash_get_uint(MSG_HASH_USER_LANGUAGE))
|
|
{
|
|
case RETRO_LANGUAGE_ARABIC:
|
|
case RETRO_LANGUAGE_PERSIAN:
|
|
fill_pathname_join_special(font_file, p_dispwidget->assets_pkg_dir, "fallback-font.ttf", sizeof(font_file));
|
|
break;
|
|
case RETRO_LANGUAGE_CHINESE_SIMPLIFIED:
|
|
case RETRO_LANGUAGE_CHINESE_TRADITIONAL:
|
|
fill_pathname_join_special(font_file, p_dispwidget->assets_pkg_dir, "chinese-fallback-font.ttf", sizeof(font_file));
|
|
break;
|
|
case RETRO_LANGUAGE_KOREAN:
|
|
fill_pathname_join_special(font_file, p_dispwidget->assets_pkg_dir, "korean-fallback-font.ttf", sizeof(font_file));
|
|
break;
|
|
default:
|
|
strlcpy(font_file, p_dispwidget->ozone_regular_font_path, sizeof(font_file));
|
|
break;
|
|
}
|
|
gfx_widgets_font_init(p_disp, p_dispwidget,
|
|
&p_dispwidget->gfx_widget_fonts.msg_queue,
|
|
is_threaded, font_file, MSG_QUEUE_FONT_SIZE);
|
|
}
|
|
else
|
|
{
|
|
/* Load fonts from user-supplied path */
|
|
gfx_widgets_font_init(p_disp, p_dispwidget,
|
|
&p_dispwidget->gfx_widget_fonts.regular,
|
|
is_threaded, font_path, BASE_FONT_SIZE);
|
|
gfx_widgets_font_init(p_disp, p_dispwidget,
|
|
&p_dispwidget->gfx_widget_fonts.bold,
|
|
is_threaded, font_path, BASE_FONT_SIZE);
|
|
gfx_widgets_font_init(p_disp, p_dispwidget,
|
|
&p_dispwidget->gfx_widget_fonts.msg_queue,
|
|
is_threaded, font_path, MSG_QUEUE_FONT_SIZE);
|
|
}
|
|
|
|
/* Calculate dimensions */
|
|
p_dispwidget->simple_widget_padding = p_dispwidget->gfx_widget_fonts.regular.line_height * (2.0f / 3.0f) + 0.5f;
|
|
p_dispwidget->simple_widget_height = p_dispwidget->gfx_widget_fonts.regular.line_height + p_dispwidget->simple_widget_padding;
|
|
|
|
p_dispwidget->msg_queue_height = p_dispwidget->gfx_widget_fonts.msg_queue.line_height * 2.5f * (BASE_FONT_SIZE / MSG_QUEUE_FONT_SIZE);
|
|
p_dispwidget->msg_queue_padding = (unsigned)(((float)p_dispwidget->gfx_widget_fonts.msg_queue.line_height * (2.0f / 3.0f)) + 0.5f);
|
|
|
|
if (p_dispwidget->flags & DISPGFX_WIDGET_FLAG_MSG_QUEUE_HAS_ICONS)
|
|
{
|
|
#if 0
|
|
p_dispwidget->msg_queue_icon_size_y = p_dispwidget->msg_queue_height
|
|
* 1.2347826087f; /* original image is 280x284 */
|
|
p_dispwidget->msg_queue_icon_size_x = 0.98591549295f * p_dispwidget->msg_queue_icon_size_y;
|
|
#else
|
|
p_dispwidget->msg_queue_icon_size_y = p_dispwidget->msg_queue_height;
|
|
p_dispwidget->msg_queue_icon_size_x = p_dispwidget->msg_queue_icon_size_y;
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
p_dispwidget->msg_queue_icon_size_x = p_dispwidget->simple_widget_padding * 1.4f;
|
|
p_dispwidget->msg_queue_icon_size_y = 0;
|
|
}
|
|
|
|
p_dispwidget->msg_queue_spacing = p_dispwidget->msg_queue_height / 4.0f;
|
|
p_dispwidget->msg_queue_rect_start_x = ceil((p_dispwidget->msg_queue_padding * 2.0f) - (p_dispwidget->simple_widget_padding * 0.15f));
|
|
p_dispwidget->msg_queue_internal_icon_size = p_dispwidget->msg_queue_icon_size_y;
|
|
p_dispwidget->msg_queue_internal_icon_offset = (p_dispwidget->msg_queue_icon_size_y - p_dispwidget->msg_queue_internal_icon_size) / 2;
|
|
p_dispwidget->msg_queue_icon_offset_y = (p_dispwidget->msg_queue_icon_size_y - p_dispwidget->msg_queue_height) / 2;
|
|
p_dispwidget->msg_queue_scissor_start_x = p_dispwidget->msg_queue_spacing + p_dispwidget->msg_queue_icon_size_x - (p_dispwidget->msg_queue_icon_size_x * 0.28928571428f);
|
|
|
|
if (p_dispwidget->flags & DISPGFX_WIDGET_FLAG_MSG_QUEUE_HAS_ICONS)
|
|
p_dispwidget->msg_queue_regular_padding_x = p_dispwidget->simple_widget_padding / 2;
|
|
else
|
|
p_dispwidget->msg_queue_regular_padding_x = p_dispwidget->simple_widget_padding;
|
|
|
|
p_dispwidget->msg_queue_task_rect_start_x = p_dispwidget->msg_queue_rect_start_x;
|
|
|
|
p_dispwidget->msg_queue_task_text_start_x = p_dispwidget->msg_queue_task_rect_start_x + (p_dispwidget->msg_queue_height / 2.0f) + (p_dispwidget->simple_widget_padding / 2.0f);
|
|
|
|
if (!p_dispwidget->gfx_widgets_icons_textures[MENU_WIDGETS_ICON_HOURGLASS])
|
|
p_dispwidget->msg_queue_task_text_start_x -= p_dispwidget->gfx_widget_fonts.msg_queue.glyph_width * 2.0f;
|
|
|
|
p_dispwidget->msg_queue_regular_text_start = p_dispwidget->msg_queue_rect_start_x + p_dispwidget->msg_queue_icon_size_x + (p_dispwidget->simple_widget_padding / 2.5f);
|
|
|
|
p_dispwidget->msg_queue_task_hourglass_x = p_dispwidget->msg_queue_task_rect_start_x + (p_dispwidget->simple_widget_padding / 2.5f);
|
|
|
|
p_dispwidget->generic_message_height = p_dispwidget->gfx_widget_fonts.regular.line_height * 2.0f;
|
|
|
|
p_dispwidget->msg_queue_default_rect_width_menu_alive
|
|
= p_dispwidget->gfx_widget_fonts.msg_queue.glyph_width * 40.0f;
|
|
p_dispwidget->msg_queue_default_rect_width = p_dispwidget->last_video_width
|
|
- p_dispwidget->msg_queue_regular_text_start - (2 * p_dispwidget->simple_widget_padding);
|
|
|
|
p_dispwidget->divider_width_1px = 1;
|
|
if (p_dispwidget->last_scale_factor > 1.0f)
|
|
p_dispwidget->divider_width_1px = (unsigned)(p_dispwidget->last_scale_factor + 0.5f);
|
|
|
|
for (i = 0; i < ARRAY_SIZE(widgets); i++)
|
|
{
|
|
const gfx_widget_t* widget = widgets[i];
|
|
|
|
if (widget->layout)
|
|
widget->layout(p_dispwidget,
|
|
is_threaded, dir_assets, font_path);
|
|
}
|
|
}
|
|
|
|
void gfx_widgets_iterate(
|
|
void *data_disp,
|
|
void *settings_data,
|
|
unsigned width, unsigned height, bool fullscreen,
|
|
const char *dir_assets, char *font_path,
|
|
bool is_threaded)
|
|
{
|
|
size_t i;
|
|
dispgfx_widget_t *p_dispwidget = &dispwidget_st;
|
|
/* c.f. https://gcc.gnu.org/bugzilla/show_bug.cgi?id=323
|
|
* On some platforms (e.g. 32-bit x86 without SSE),
|
|
* gcc can produce inconsistent floating point results
|
|
* depending upon optimisation level. This can break
|
|
* floating point variable comparisons. A workaround is
|
|
* to declare the affected variable as 'volatile', which
|
|
* disables optimisations and removes excess precision
|
|
* (https://gcc.gnu.org/bugzilla/show_bug.cgi?id=323#c87) */
|
|
volatile float scale_factor = 0.0f;
|
|
gfx_display_t *p_disp = (gfx_display_t*)data_disp;
|
|
settings_t *settings = (settings_t*)settings_data;
|
|
#ifdef HAVE_XMB
|
|
enum menu_driver_id_type type = p_disp->menu_driver_id;
|
|
if (type == MENU_DRIVER_ID_XMB)
|
|
scale_factor = gfx_display_get_widget_pixel_scale(p_disp, settings, width, height, fullscreen);
|
|
else
|
|
#endif
|
|
scale_factor = gfx_display_get_dpi_scale(
|
|
p_disp,
|
|
settings, width, height, fullscreen, true);
|
|
|
|
/* Check whether screen dimensions or menu scale
|
|
* factor have changed */
|
|
if ((scale_factor != p_dispwidget->last_scale_factor) ||
|
|
(width != p_dispwidget->last_video_width) ||
|
|
(height != p_dispwidget->last_video_height))
|
|
{
|
|
p_dispwidget->last_scale_factor = scale_factor;
|
|
p_dispwidget->last_video_width = width;
|
|
p_dispwidget->last_video_height = height;
|
|
|
|
/* Note: We don't need a full context reset here
|
|
* > Just rescale layout, and reset frame time counter */
|
|
gfx_widgets_layout(p_disp, p_dispwidget,
|
|
is_threaded, dir_assets, font_path);
|
|
video_driver_monitor_reset();
|
|
}
|
|
|
|
for (i = 0; i < ARRAY_SIZE(widgets); i++)
|
|
{
|
|
const gfx_widget_t* widget = widgets[i];
|
|
|
|
if (widget->iterate)
|
|
widget->iterate(p_dispwidget,
|
|
width, height, fullscreen,
|
|
dir_assets, font_path, is_threaded);
|
|
}
|
|
|
|
/* Messages queue */
|
|
|
|
/* Consume one message if available */
|
|
if ((FIFO_READ_AVAIL_NONPTR(p_dispwidget->msg_queue) > 0)
|
|
&& !(p_dispwidget->flags & DISPGFX_WIDGET_FLAG_MOVING)
|
|
&& (p_dispwidget->current_msgs_size < ARRAY_SIZE(p_dispwidget->current_msgs)))
|
|
{
|
|
disp_widget_msg_t *msg_widget = NULL;
|
|
|
|
#ifdef HAVE_THREADS
|
|
slock_lock(p_dispwidget->current_msgs_lock);
|
|
#endif
|
|
|
|
if (p_dispwidget->current_msgs_size < ARRAY_SIZE(p_dispwidget->current_msgs))
|
|
{
|
|
if (FIFO_READ_AVAIL_NONPTR(p_dispwidget->msg_queue) > 0)
|
|
fifo_read(&p_dispwidget->msg_queue,
|
|
&msg_widget, sizeof(msg_widget));
|
|
|
|
if (msg_widget)
|
|
{
|
|
/* Task messages always appear from the bottom of the screen, append it */
|
|
if (p_dispwidget->msg_queue_tasks_count == 0 || msg_widget->task_ptr)
|
|
p_dispwidget->current_msgs[p_dispwidget->current_msgs_size] = msg_widget;
|
|
/* Regular messages are always above tasks, make room and insert it */
|
|
else
|
|
{
|
|
unsigned idx = (unsigned)(p_dispwidget->current_msgs_size -
|
|
p_dispwidget->msg_queue_tasks_count);
|
|
for (i = p_dispwidget->current_msgs_size; i > idx; i--)
|
|
p_dispwidget->current_msgs[i] = p_dispwidget->current_msgs[i - 1];
|
|
p_dispwidget->current_msgs[idx] = msg_widget;
|
|
}
|
|
|
|
p_dispwidget->current_msgs_size++;
|
|
}
|
|
}
|
|
|
|
#ifdef HAVE_THREADS
|
|
slock_unlock(p_dispwidget->current_msgs_lock);
|
|
#endif
|
|
|
|
if (msg_widget)
|
|
{
|
|
/* Start expiration timer if not associated to a task */
|
|
if (!msg_widget->task_ptr)
|
|
{
|
|
if (!(msg_widget->flags & DISPWIDG_FLAG_EXPIRATION_TIMER_STARTED))
|
|
gfx_widgets_start_msg_expiration_timer(
|
|
msg_widget, MSG_QUEUE_ANIMATION_DURATION * 2
|
|
+ msg_widget->duration);
|
|
}
|
|
/* Else, start hourglass animation timer */
|
|
else
|
|
{
|
|
p_dispwidget->msg_queue_tasks_count++;
|
|
gfx_widgets_hourglass_end(msg_widget);
|
|
}
|
|
|
|
if (p_dispwidget->current_msgs_size != 0)
|
|
gfx_widgets_msg_queue_move(p_dispwidget);
|
|
}
|
|
}
|
|
|
|
/* Kill first expired message */
|
|
/* Start expiration timer of dead tasks */
|
|
for (i = 0; i < p_dispwidget->current_msgs_size; i++)
|
|
{
|
|
disp_widget_msg_t *msg_widget = p_dispwidget->current_msgs[i];
|
|
|
|
if (!msg_widget)
|
|
continue;
|
|
|
|
if (msg_widget->task_ptr
|
|
&& ((msg_widget->flags & DISPWIDG_FLAG_TASK_FINISHED)
|
|
|| (msg_widget->flags & DISPWIDG_FLAG_TASK_CANCELLED)))
|
|
if (!(msg_widget->flags & DISPWIDG_FLAG_EXPIRATION_TIMER_STARTED))
|
|
gfx_widgets_start_msg_expiration_timer(msg_widget, TASK_FINISHED_DURATION);
|
|
|
|
if ( (msg_widget->flags & DISPWIDG_FLAG_EXPIRED)
|
|
&& !(p_dispwidget->flags & DISPGFX_WIDGET_FLAG_MOVING))
|
|
{
|
|
gfx_widgets_msg_queue_kill(p_dispwidget,
|
|
(unsigned)i);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static int gfx_widgets_draw_indicator(
|
|
dispgfx_widget_t *p_dispwidget,
|
|
gfx_display_t *p_disp,
|
|
gfx_display_ctx_driver_t *dispctx,
|
|
void *userdata,
|
|
unsigned video_width,
|
|
unsigned video_height,
|
|
uintptr_t icon, int y, int top_right_x_advance,
|
|
enum msg_hash_enums msg)
|
|
{
|
|
unsigned width;
|
|
|
|
gfx_display_set_alpha(p_dispwidget->backdrop_orig, DEFAULT_BACKDROP);
|
|
|
|
if (icon)
|
|
{
|
|
unsigned height = p_dispwidget->simple_widget_height * 2;
|
|
width = height;
|
|
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width, video_height,
|
|
top_right_x_advance - width, y,
|
|
width, height,
|
|
video_width, video_height,
|
|
p_dispwidget->backdrop_orig,
|
|
NULL
|
|
);
|
|
|
|
gfx_display_set_alpha(p_dispwidget->pure_white, 1.0f);
|
|
|
|
if (dispctx && dispctx->blend_begin)
|
|
dispctx->blend_begin(userdata);
|
|
gfx_widgets_draw_icon(
|
|
userdata,
|
|
p_disp,
|
|
video_width,
|
|
video_height,
|
|
width,
|
|
height,
|
|
icon,
|
|
top_right_x_advance - width, y,
|
|
0.0f, /* rad */
|
|
1.0f, /* cos(rad) = cos(0) = 1.0f */
|
|
0.0f, /* sine(rad) = sine(0) = 0.0f */
|
|
p_dispwidget->pure_white
|
|
);
|
|
if (dispctx && dispctx->blend_end)
|
|
dispctx->blend_end(userdata);
|
|
}
|
|
else
|
|
{
|
|
unsigned height = p_dispwidget->simple_widget_height;
|
|
const char *txt = msg_hash_to_str(msg);
|
|
|
|
width = font_driver_get_message_width(
|
|
p_dispwidget->gfx_widget_fonts.regular.font,
|
|
txt, strlen(txt), 1.0f)
|
|
+ p_dispwidget->simple_widget_padding * 2;
|
|
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width, video_height,
|
|
top_right_x_advance - width, y,
|
|
width, height,
|
|
video_width, video_height,
|
|
p_dispwidget->backdrop_orig,
|
|
NULL
|
|
);
|
|
|
|
gfx_widgets_draw_text(&p_dispwidget->gfx_widget_fonts.regular,
|
|
txt,
|
|
top_right_x_advance - width
|
|
+ p_dispwidget->simple_widget_padding,
|
|
y + (height / 2.0f) +
|
|
p_dispwidget->gfx_widget_fonts.regular.line_centre_offset,
|
|
video_width, video_height,
|
|
0xFFFFFFFF, TEXT_ALIGN_LEFT,
|
|
false);
|
|
}
|
|
|
|
return width;
|
|
}
|
|
|
|
static void gfx_widgets_draw_task_msg(
|
|
dispgfx_widget_t *p_dispwidget,
|
|
gfx_display_t *p_disp,
|
|
gfx_display_ctx_driver_t *dispctx,
|
|
disp_widget_msg_t *msg,
|
|
void *userdata,
|
|
unsigned video_width,
|
|
unsigned video_height)
|
|
{
|
|
/* Color of first progress bar in a task message */
|
|
static float msg_queue_task_progress_1[16] = COLOR_HEX_TO_FLOAT(0x1A1A1A, 1.0f);
|
|
/* Color of second progress bar in a task message
|
|
* (for multiple tasks with same message) */
|
|
static float msg_queue_task_progress_2[16] = COLOR_HEX_TO_FLOAT(0x1D5B1D, 1.0f);
|
|
/* Margin bar */
|
|
static float msg_queue_bar[16] = COLOR_HEX_TO_FLOAT(0xCCCCCC, 1.0f);
|
|
/* Green icon */
|
|
static float msg_queue_task_positive[16] = COLOR_HEX_TO_FLOAT(0x00C80A, 1.0f);
|
|
/* Red icon */
|
|
static float msg_queue_task_negative[16] = COLOR_HEX_TO_FLOAT(0xC60000, 1.0f);
|
|
|
|
unsigned text_color;
|
|
unsigned bar_width;
|
|
|
|
unsigned rect_margin;
|
|
unsigned rect_x;
|
|
unsigned rect_y;
|
|
unsigned rect_width;
|
|
unsigned rect_height;
|
|
float text_y_base;
|
|
|
|
float *msg_queue_current_background;
|
|
float *msg_queue_current_bar;
|
|
|
|
size_t _len = 0;
|
|
char task_percentage[256] = "";
|
|
bool draw_msg_new = false;
|
|
size_t task_percentage_offset = 0;
|
|
|
|
if (msg->msg_new)
|
|
draw_msg_new = !string_is_equal(msg->msg_new, msg->msg);
|
|
|
|
if (msg->flags & DISPWIDG_FLAG_TASK_FINISHED)
|
|
{
|
|
if (msg->flags & DISPWIDG_FLAG_TASK_ERROR)
|
|
_len = strlcpy(task_percentage, msg_hash_to_str(MSG_ERROR), sizeof(task_percentage));
|
|
}
|
|
else if (msg->task_progress >= 0 && msg->task_progress <= 100)
|
|
_len = snprintf(task_percentage, sizeof(task_percentage),
|
|
"%i%%", msg->task_progress);
|
|
|
|
task_percentage_offset = p_dispwidget->gfx_widget_fonts.msg_queue.glyph_width * _len;
|
|
rect_width = p_dispwidget->msg_queue_padding * 5.0f + msg->width + task_percentage_offset;
|
|
bar_width = rect_width * msg->task_progress/100.0f;
|
|
text_color = COLOR_TEXT_ALPHA(0xFFFFFF00, (unsigned)(msg->alpha*255.0f));
|
|
|
|
/* Rect */
|
|
msg_queue_current_background = msg_queue_task_progress_1;
|
|
|
|
if (msg->flags & DISPWIDG_FLAG_TASK_FINISHED)
|
|
{
|
|
if (msg->task_count == 1)
|
|
msg_queue_current_background = msg_queue_task_progress_1;
|
|
else
|
|
msg_queue_current_background = msg_queue_task_progress_2;
|
|
}
|
|
|
|
rect_x = p_dispwidget->msg_queue_rect_start_x;
|
|
rect_y = video_height - msg->offset_y;
|
|
rect_height = p_dispwidget->msg_queue_height / 2;
|
|
rect_margin = p_dispwidget->simple_widget_padding * 0.15f;
|
|
|
|
gfx_display_set_alpha(msg_queue_bar, msg->alpha);
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width, video_height,
|
|
rect_x, rect_y,
|
|
rect_margin, rect_height,
|
|
video_width, video_height,
|
|
msg_queue_bar,
|
|
NULL
|
|
);
|
|
|
|
gfx_display_set_alpha(msg_queue_current_background, msg->alpha);
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width, video_height,
|
|
rect_x + rect_margin, rect_y,
|
|
rect_width, rect_height,
|
|
video_width, video_height,
|
|
msg_queue_current_background,
|
|
NULL
|
|
);
|
|
|
|
/* Progress bar */
|
|
if ( !(msg->flags & DISPWIDG_FLAG_TASK_FINISHED)
|
|
&& (msg->task_progress >= 0)
|
|
&& (msg->task_progress <= 100))
|
|
{
|
|
if (msg->task_count == 1)
|
|
msg_queue_current_bar = msg_queue_task_progress_1;
|
|
else
|
|
msg_queue_current_bar = msg_queue_task_progress_2;
|
|
|
|
gfx_display_set_alpha(msg_queue_current_bar, 1.0f);
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width, video_height,
|
|
p_dispwidget->msg_queue_task_rect_start_x + rect_margin,
|
|
video_height - msg->offset_y,
|
|
bar_width, rect_height,
|
|
video_width, video_height,
|
|
msg_queue_current_bar,
|
|
NULL
|
|
);
|
|
}
|
|
|
|
/* Icon */
|
|
|
|
if (dispctx && dispctx->blend_begin)
|
|
dispctx->blend_begin(userdata);
|
|
{
|
|
float radians = 0.0f; /* rad */
|
|
float cosine = 1.0f; /* cos(rad) = cos(0) = 1.0f */
|
|
float sine = 0.0f; /* sine(rad) = sine(0) = 0.0f */
|
|
int texture = MENU_WIDGETS_ICON_CHECK;
|
|
float *color = msg_queue_task_positive;
|
|
|
|
if (!(msg->flags & DISPWIDG_FLAG_TASK_FINISHED))
|
|
{
|
|
texture = MENU_WIDGETS_ICON_HOURGLASS;
|
|
color = msg_queue_bar;
|
|
radians = msg->hourglass_rotation;
|
|
}
|
|
else if (msg->flags & DISPWIDG_FLAG_POSITIVE)
|
|
{
|
|
texture = MENU_WIDGETS_ICON_ADD;
|
|
color = msg_queue_task_positive;
|
|
}
|
|
else if (msg->flags & DISPWIDG_FLAG_NEGATIVE)
|
|
{
|
|
texture = MENU_WIDGETS_ICON_EXIT;
|
|
color = msg_queue_task_negative;
|
|
}
|
|
|
|
gfx_display_set_alpha(color, msg->alpha);
|
|
gfx_widgets_draw_icon(
|
|
userdata,
|
|
p_disp,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->msg_queue_height / 2.5f,
|
|
p_dispwidget->msg_queue_height / 2.5f,
|
|
p_dispwidget->gfx_widgets_icons_textures[texture],
|
|
p_dispwidget->msg_queue_task_hourglass_x + (p_dispwidget->msg_queue_height / MSG_QUEUE_FONT_SIZE),
|
|
video_height - msg->offset_y + (p_dispwidget->msg_queue_height / MSG_QUEUE_FONT_SIZE),
|
|
radians,
|
|
cosine,
|
|
sine,
|
|
color);
|
|
}
|
|
if (dispctx && dispctx->blend_end)
|
|
dispctx->blend_end(userdata);
|
|
|
|
/* Text */
|
|
text_y_base = video_height
|
|
- msg->offset_y
|
|
+ p_dispwidget->msg_queue_height / 4.0f
|
|
+ p_dispwidget->gfx_widget_fonts.msg_queue.line_centre_offset;
|
|
|
|
if (draw_msg_new)
|
|
{
|
|
gfx_widgets_flush_text(video_width, video_height,
|
|
&p_dispwidget->gfx_widget_fonts.msg_queue);
|
|
|
|
gfx_display_scissor_begin(p_disp,
|
|
userdata,
|
|
video_width, video_height,
|
|
rect_x, rect_y, rect_width, rect_height);
|
|
|
|
gfx_widgets_draw_text(&p_dispwidget->gfx_widget_fonts.msg_queue,
|
|
msg->msg_new,
|
|
p_dispwidget->msg_queue_task_text_start_x,
|
|
text_y_base - p_dispwidget->msg_queue_height / 2.0f + msg->msg_transition_animation,
|
|
video_width, video_height,
|
|
text_color,
|
|
TEXT_ALIGN_LEFT,
|
|
true);
|
|
}
|
|
|
|
gfx_widgets_draw_text(&p_dispwidget->gfx_widget_fonts.msg_queue,
|
|
msg->msg,
|
|
p_dispwidget->msg_queue_task_text_start_x,
|
|
text_y_base + msg->msg_transition_animation,
|
|
video_width, video_height,
|
|
text_color,
|
|
TEXT_ALIGN_LEFT,
|
|
true);
|
|
|
|
if (draw_msg_new)
|
|
{
|
|
gfx_widgets_flush_text(video_width, video_height,
|
|
&p_dispwidget->gfx_widget_fonts.msg_queue);
|
|
if (dispctx && dispctx->scissor_end)
|
|
dispctx->scissor_end(userdata,
|
|
video_width, video_height);
|
|
}
|
|
|
|
/* Progress text */
|
|
text_color = COLOR_TEXT_ALPHA(0xFFFFFF00, (unsigned)(msg->alpha/2*255.0f));
|
|
gfx_widgets_draw_text(&p_dispwidget->gfx_widget_fonts.msg_queue,
|
|
task_percentage,
|
|
p_dispwidget->msg_queue_rect_start_x
|
|
+ rect_width - p_dispwidget->gfx_widget_fonts.msg_queue.glyph_width,
|
|
text_y_base,
|
|
video_width, video_height,
|
|
text_color,
|
|
TEXT_ALIGN_RIGHT,
|
|
true);
|
|
}
|
|
|
|
static void gfx_widgets_draw_regular_msg(
|
|
dispgfx_widget_t *p_dispwidget,
|
|
gfx_display_t *p_disp,
|
|
gfx_display_ctx_driver_t *dispctx,
|
|
disp_widget_msg_t *msg,
|
|
void *userdata,
|
|
unsigned video_width,
|
|
unsigned video_height)
|
|
{
|
|
static float msg_queue_info[16] = COLOR_HEX_TO_FLOAT(0x0C99D6, 1.0f);
|
|
static float msg_queue_bar[16] = COLOR_HEX_TO_FLOAT(0xCCCCCC, 1.0f);
|
|
unsigned rect_width;
|
|
unsigned rect_margin;
|
|
unsigned text_color;
|
|
static float last_alpha = 0.0f;
|
|
|
|
msg->flags &= ~DISPWIDG_FLAG_UNFOLDING;
|
|
msg->flags |= DISPWIDG_FLAG_UNFOLDED;
|
|
|
|
if (last_alpha != msg->alpha)
|
|
{
|
|
/* Icon */
|
|
gfx_display_set_alpha(msg_queue_info, msg->alpha);
|
|
gfx_display_set_alpha(p_dispwidget->pure_white, msg->alpha);
|
|
gfx_display_set_alpha(p_dispwidget->msg_queue_bg, msg->alpha);
|
|
last_alpha = msg->alpha;
|
|
}
|
|
|
|
if ( !(msg->flags & DISPWIDG_FLAG_UNFOLDED)
|
|
|| (msg->flags & DISPWIDG_FLAG_UNFOLDING))
|
|
{
|
|
gfx_widgets_flush_text(video_width, video_height,
|
|
&p_dispwidget->gfx_widget_fonts.regular);
|
|
gfx_widgets_flush_text(video_width, video_height,
|
|
&p_dispwidget->gfx_widget_fonts.bold);
|
|
gfx_widgets_flush_text(video_width, video_height,
|
|
&p_dispwidget->gfx_widget_fonts.msg_queue);
|
|
|
|
gfx_display_scissor_begin(p_disp,
|
|
userdata,
|
|
video_width, video_height,
|
|
p_dispwidget->msg_queue_scissor_start_x, 0,
|
|
(p_dispwidget->msg_queue_scissor_start_x + msg->width -
|
|
p_dispwidget->simple_widget_padding * 2)
|
|
* msg->unfold, video_height);
|
|
}
|
|
|
|
/* Background */
|
|
rect_width = p_dispwidget->simple_widget_padding + msg->width + p_dispwidget->msg_queue_icon_size_x;
|
|
rect_margin = p_dispwidget->simple_widget_padding * 0.15f;
|
|
gfx_display_set_alpha(msg_queue_bar, msg->alpha);
|
|
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->msg_queue_rect_start_x + rect_margin,
|
|
video_height - msg->offset_y,
|
|
rect_width - rect_margin,
|
|
p_dispwidget->msg_queue_height,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->msg_queue_bg,
|
|
NULL
|
|
);
|
|
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->msg_queue_rect_start_x,
|
|
video_height - msg->offset_y,
|
|
rect_margin,
|
|
p_dispwidget->msg_queue_height,
|
|
video_width,
|
|
video_height,
|
|
msg_queue_bar,
|
|
NULL
|
|
);
|
|
|
|
/* Text */
|
|
text_color = COLOR_TEXT_ALPHA(0xFFFFFF00, (unsigned)(msg->alpha*255.0f));
|
|
|
|
gfx_widgets_draw_text(&p_dispwidget->gfx_widget_fonts.msg_queue,
|
|
msg->msg,
|
|
p_dispwidget->msg_queue_regular_text_start,
|
|
video_height - msg->offset_y
|
|
+ (p_dispwidget->msg_queue_height - msg->text_height) / 2.0f
|
|
+ p_dispwidget->gfx_widget_fonts.msg_queue.line_ascender,
|
|
video_width, video_height,
|
|
text_color,
|
|
TEXT_ALIGN_LEFT,
|
|
true);
|
|
|
|
if ( !(msg->flags & DISPWIDG_FLAG_UNFOLDED)
|
|
|| (msg->flags & DISPWIDG_FLAG_UNFOLDING))
|
|
{
|
|
gfx_widgets_flush_text(video_width, video_height, &p_dispwidget->gfx_widget_fonts.regular);
|
|
gfx_widgets_flush_text(video_width, video_height, &p_dispwidget->gfx_widget_fonts.bold);
|
|
gfx_widgets_flush_text(video_width, video_height, &p_dispwidget->gfx_widget_fonts.msg_queue);
|
|
|
|
if (dispctx && dispctx->scissor_end)
|
|
dispctx->scissor_end(userdata, video_width, video_height);
|
|
}
|
|
|
|
if (p_dispwidget->flags & DISPGFX_WIDGET_FLAG_MSG_QUEUE_HAS_ICONS)
|
|
{
|
|
if (dispctx && dispctx->blend_begin)
|
|
dispctx->blend_begin(userdata);
|
|
|
|
gfx_widgets_draw_icon(
|
|
userdata,
|
|
p_disp,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->msg_queue_icon_size_x,
|
|
p_dispwidget->msg_queue_icon_size_y,
|
|
p_dispwidget->gfx_widgets_icons_textures[MENU_WIDGETS_ICON_INFO],
|
|
p_dispwidget->msg_queue_rect_start_x
|
|
+ (p_dispwidget->simple_widget_padding / 4.0f),
|
|
video_height - msg->offset_y - p_dispwidget->msg_queue_icon_offset_y,
|
|
0.0f, /* rad */
|
|
1.0f, /* cos(rad) = cos(0) = 1.0f */
|
|
0.0f, /* sine(rad) = sine(0) = 0.0f */
|
|
msg_queue_info);
|
|
|
|
if (dispctx && dispctx->blend_end)
|
|
dispctx->blend_end(userdata);
|
|
}
|
|
}
|
|
|
|
static void INLINE gfx_widgets_font_bind(gfx_widget_font_data_t *font_data)
|
|
{
|
|
font_driver_bind_block(font_data->font, &font_data->raster_block);
|
|
font_data->raster_block.carr.coords.vertices = 0;
|
|
font_data->usage_count = 0;
|
|
}
|
|
|
|
static void INLINE gfx_widgets_font_unbind(gfx_widget_font_data_t *font_data)
|
|
{
|
|
font_driver_bind_block(font_data->font, NULL);
|
|
}
|
|
|
|
|
|
void gfx_widgets_frame(void *data)
|
|
{
|
|
size_t i;
|
|
video_frame_info_t *video_info = (video_frame_info_t*)data;
|
|
gfx_display_t *p_disp = (gfx_display_t*)video_info->disp_userdata;
|
|
gfx_display_ctx_driver_t *dispctx= p_disp->dispctx;
|
|
video_driver_state_t *video_st = video_state_get_ptr();
|
|
dispgfx_widget_t *p_dispwidget = (dispgfx_widget_t*)video_info->widgets_userdata;
|
|
bool fps_show = video_info->fps_show;
|
|
bool framecount_show = video_info->framecount_show;
|
|
bool memory_show = video_info->memory_show;
|
|
bool core_status_msg_show = video_info->core_status_msg_show;
|
|
void *userdata = video_info->userdata;
|
|
unsigned video_width = video_info->width;
|
|
unsigned video_height = video_info->height;
|
|
bool widgets_is_paused = (video_info->video_st_flags & VIDEO_FLAG_WIDGETS_PAUSED) ? true : false;
|
|
bool widgets_is_fastforwarding = (video_info->video_st_flags & VIDEO_FLAG_WIDGETS_FAST_FORWARD) ? true : false;
|
|
bool widgets_is_rewinding = (video_info->video_st_flags & VIDEO_FLAG_WIDGETS_REWINDING) ? true : false;
|
|
bool runloop_is_slowmotion = video_info->runloop_is_slowmotion;
|
|
#ifdef HAVE_MENU
|
|
bool menu_screensaver_active = (video_info->menu_st_flags & MENU_ST_FLAG_SCREENSAVER_ACTIVE) ? true : false;
|
|
#endif
|
|
bool notifications_hidden = video_info->notifications_hidden || video_info->msg_queue_delay;
|
|
int top_right_x_advance = video_width;
|
|
|
|
p_dispwidget->gfx_widgets_frame_count++;
|
|
|
|
#ifdef HAVE_MENU
|
|
/* If menu screensaver is active, draw nothing */
|
|
if (menu_screensaver_active)
|
|
return;
|
|
#endif
|
|
/* If notifications are hidden, draw nothing */
|
|
if (notifications_hidden)
|
|
return;
|
|
|
|
if (video_st->current_video && video_st->current_video->set_viewport)
|
|
video_st->current_video->set_viewport(
|
|
video_st->data, video_width, video_height, true, false);
|
|
|
|
/* Font setup */
|
|
gfx_widgets_font_bind(&p_dispwidget->gfx_widget_fonts.regular);
|
|
gfx_widgets_font_bind(&p_dispwidget->gfx_widget_fonts.bold);
|
|
gfx_widgets_font_bind(&p_dispwidget->gfx_widget_fonts.msg_queue);
|
|
|
|
#ifdef HAVE_TRANSLATE
|
|
/* AI Service overlay */
|
|
if (p_dispwidget->ai_service_overlay_state > 0)
|
|
{
|
|
float outline_color[16] = {
|
|
0.00, 1.00, 0.00, 1.00,
|
|
0.00, 1.00, 0.00, 1.00,
|
|
0.00, 1.00, 0.00, 1.00,
|
|
0.00, 1.00, 0.00, 1.00,
|
|
};
|
|
|
|
gfx_display_set_alpha(p_dispwidget->pure_white, 1.0f);
|
|
|
|
if (p_dispwidget->ai_service_overlay_texture)
|
|
{
|
|
if (dispctx->blend_begin)
|
|
dispctx->blend_begin(userdata);
|
|
gfx_widgets_draw_icon(
|
|
userdata,
|
|
p_disp,
|
|
video_width,
|
|
video_height,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->ai_service_overlay_texture,
|
|
0,
|
|
0,
|
|
0.0f, /* rad */
|
|
1.0f, /* cos(rad) = cos(0) = 1.0f */
|
|
0.0f, /* sine(rad) = sine(0) = 0.0f */
|
|
p_dispwidget->pure_white
|
|
);
|
|
if (dispctx->blend_end)
|
|
dispctx->blend_end(userdata);
|
|
}
|
|
|
|
/* top line */
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width, video_height,
|
|
0, 0,
|
|
video_width,
|
|
p_dispwidget->divider_width_1px,
|
|
video_width,
|
|
video_height,
|
|
outline_color,
|
|
NULL
|
|
);
|
|
/* bottom line */
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width, video_height,
|
|
0,
|
|
video_height - p_dispwidget->divider_width_1px,
|
|
video_width,
|
|
p_dispwidget->divider_width_1px,
|
|
video_width,
|
|
video_height,
|
|
outline_color,
|
|
NULL
|
|
);
|
|
/* left line */
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width,
|
|
video_height,
|
|
0,
|
|
0,
|
|
p_dispwidget->divider_width_1px,
|
|
video_height,
|
|
video_width,
|
|
video_height,
|
|
outline_color,
|
|
NULL
|
|
);
|
|
/* right line */
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width, video_height,
|
|
video_width - p_dispwidget->divider_width_1px,
|
|
0,
|
|
p_dispwidget->divider_width_1px,
|
|
video_height,
|
|
video_width,
|
|
video_height,
|
|
outline_color,
|
|
NULL
|
|
);
|
|
if (p_dispwidget->ai_service_overlay_state == 2)
|
|
p_dispwidget->ai_service_overlay_state = 3;
|
|
}
|
|
#endif
|
|
|
|
/* Status Text (fps, framecount, memory, core status message) */
|
|
if ( fps_show
|
|
|| framecount_show
|
|
|| memory_show
|
|
|| core_status_msg_show
|
|
)
|
|
{
|
|
const char *txt = *p_dispwidget->gfx_widgets_status_text == '\0'
|
|
? msg_hash_to_str(MENU_ENUM_LABEL_VALUE_NOT_AVAILABLE)
|
|
: p_dispwidget->gfx_widgets_status_text;
|
|
|
|
int txt_width = font_driver_get_message_width(
|
|
p_dispwidget->gfx_widget_fonts.regular.font,
|
|
txt, strlen(txt), 1.0f);
|
|
int total_width = txt_width
|
|
+ p_dispwidget->simple_widget_padding * 2;
|
|
|
|
int status_txt_x = top_right_x_advance
|
|
- p_dispwidget->simple_widget_padding - txt_width;
|
|
/* Ensure that left hand side of text does
|
|
* not bleed off the edge of the screen */
|
|
if (status_txt_x < 0)
|
|
status_txt_x = 0;
|
|
|
|
gfx_display_set_alpha(p_dispwidget->backdrop_orig, DEFAULT_BACKDROP);
|
|
|
|
gfx_display_draw_quad(
|
|
p_disp,
|
|
userdata,
|
|
video_width,
|
|
video_height,
|
|
top_right_x_advance - total_width,
|
|
0,
|
|
total_width,
|
|
p_dispwidget->simple_widget_height,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->backdrop_orig,
|
|
NULL
|
|
);
|
|
|
|
gfx_widgets_draw_text(&p_dispwidget->gfx_widget_fonts.regular,
|
|
txt,
|
|
status_txt_x,
|
|
p_dispwidget->simple_widget_height / 2.0f
|
|
+ p_dispwidget->gfx_widget_fonts.regular.line_centre_offset,
|
|
video_width, video_height,
|
|
0xFFFFFFFF,
|
|
TEXT_ALIGN_LEFT,
|
|
true);
|
|
}
|
|
|
|
/* Indicators */
|
|
if (widgets_is_paused)
|
|
top_right_x_advance -= gfx_widgets_draw_indicator(
|
|
p_dispwidget,
|
|
p_disp,
|
|
dispctx,
|
|
userdata,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->gfx_widgets_icons_textures[
|
|
MENU_WIDGETS_ICON_PAUSED],
|
|
(fps_show
|
|
? p_dispwidget->simple_widget_height
|
|
: 0),
|
|
top_right_x_advance,
|
|
MSG_PAUSED);
|
|
|
|
if (widgets_is_fastforwarding)
|
|
top_right_x_advance -= gfx_widgets_draw_indicator(
|
|
p_dispwidget,
|
|
p_disp,
|
|
dispctx,
|
|
userdata,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->gfx_widgets_icons_textures[
|
|
MENU_WIDGETS_ICON_FAST_FORWARD],
|
|
(fps_show ? p_dispwidget->simple_widget_height : 0),
|
|
top_right_x_advance,
|
|
MSG_FAST_FORWARD);
|
|
|
|
if (widgets_is_rewinding)
|
|
top_right_x_advance -= gfx_widgets_draw_indicator(
|
|
p_dispwidget,
|
|
p_disp,
|
|
dispctx,
|
|
userdata,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->gfx_widgets_icons_textures[
|
|
MENU_WIDGETS_ICON_REWIND],
|
|
(fps_show ? p_dispwidget->simple_widget_height : 0),
|
|
top_right_x_advance,
|
|
MSG_REWINDING);
|
|
|
|
if (runloop_is_slowmotion)
|
|
{
|
|
top_right_x_advance -= gfx_widgets_draw_indicator(
|
|
p_dispwidget,
|
|
p_disp,
|
|
dispctx,
|
|
userdata,
|
|
video_width,
|
|
video_height,
|
|
p_dispwidget->gfx_widgets_icons_textures[
|
|
MENU_WIDGETS_ICON_SLOW_MOTION],
|
|
(fps_show ? p_dispwidget->simple_widget_height : 0),
|
|
top_right_x_advance,
|
|
MSG_SLOW_MOTION);
|
|
(void)top_right_x_advance;
|
|
}
|
|
|
|
for (i = 0; i < ARRAY_SIZE(widgets); i++)
|
|
{
|
|
const gfx_widget_t* widget = widgets[i];
|
|
|
|
if (widget->frame)
|
|
widget->frame(data, p_dispwidget);
|
|
}
|
|
|
|
/* Draw all messages */
|
|
if (p_dispwidget->current_msgs_size)
|
|
{
|
|
#ifdef HAVE_THREADS
|
|
slock_lock(p_dispwidget->current_msgs_lock);
|
|
#endif
|
|
|
|
for (i = 0; i < p_dispwidget->current_msgs_size; i++)
|
|
{
|
|
disp_widget_msg_t* msg = p_dispwidget->current_msgs[i];
|
|
|
|
if (!msg)
|
|
continue;
|
|
|
|
if (msg->task_ptr)
|
|
gfx_widgets_draw_task_msg(
|
|
p_dispwidget,
|
|
p_disp,
|
|
dispctx,
|
|
msg, userdata,
|
|
video_width, video_height);
|
|
else
|
|
gfx_widgets_draw_regular_msg(
|
|
p_dispwidget,
|
|
p_disp,
|
|
dispctx,
|
|
msg, userdata,
|
|
video_width, video_height);
|
|
}
|
|
|
|
#ifdef HAVE_THREADS
|
|
slock_unlock(p_dispwidget->current_msgs_lock);
|
|
#endif
|
|
}
|
|
|
|
/* Ensure all text is flushed */
|
|
gfx_widgets_flush_text(video_width, video_height,
|
|
&p_dispwidget->gfx_widget_fonts.regular);
|
|
gfx_widgets_flush_text(video_width, video_height,
|
|
&p_dispwidget->gfx_widget_fonts.bold);
|
|
gfx_widgets_flush_text(video_width, video_height,
|
|
&p_dispwidget->gfx_widget_fonts.msg_queue);
|
|
|
|
/* Unbind fonts */
|
|
gfx_widgets_font_unbind(&p_dispwidget->gfx_widget_fonts.regular);
|
|
gfx_widgets_font_unbind(&p_dispwidget->gfx_widget_fonts.bold);
|
|
gfx_widgets_font_unbind(&p_dispwidget->gfx_widget_fonts.msg_queue);
|
|
|
|
if (video_st->current_video && video_st->current_video->set_viewport)
|
|
video_st->current_video->set_viewport(
|
|
video_st->data, video_width, video_height, false, true);
|
|
}
|
|
|
|
static void gfx_widgets_free(dispgfx_widget_t *p_dispwidget)
|
|
{
|
|
size_t i;
|
|
|
|
p_dispwidget->flags &= ~DISPGFX_WIDGET_FLAG_INITED;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(widgets); i++)
|
|
{
|
|
const gfx_widget_t* widget = widgets[i];
|
|
|
|
if (widget->free)
|
|
widget->free();
|
|
}
|
|
|
|
/* Kill all running animations */
|
|
gfx_animation_kill_by_tag(
|
|
&p_dispwidget->gfx_widgets_generic_tag);
|
|
|
|
/* Purge everything from the fifo */
|
|
while (FIFO_READ_AVAIL_NONPTR(p_dispwidget->msg_queue) > 0)
|
|
{
|
|
disp_widget_msg_t *msg_widget;
|
|
|
|
fifo_read(&p_dispwidget->msg_queue,
|
|
&msg_widget, sizeof(msg_widget));
|
|
|
|
/* Note: gfx_widgets_free() is only called when
|
|
* main_exit() is invoked. At this stage, we cannot
|
|
* guarantee that any task pointers are valid (the
|
|
* task may have been free()'d, but we can't know
|
|
* that here) - so all we can do is unset the task
|
|
* pointer associated with each message
|
|
* > If we don't do this, gfx_widgets_msg_queue_free()
|
|
* will generate heap-use-after-free errors */
|
|
msg_widget->task_ptr = NULL;
|
|
|
|
gfx_widgets_msg_queue_free(p_dispwidget, msg_widget);
|
|
free(msg_widget);
|
|
}
|
|
|
|
fifo_deinitialize(&p_dispwidget->msg_queue);
|
|
|
|
/* Purge everything from the list */
|
|
#ifdef HAVE_THREADS
|
|
slock_lock(p_dispwidget->current_msgs_lock);
|
|
#endif
|
|
|
|
p_dispwidget->current_msgs_size = 0;
|
|
for (i = 0; i < ARRAY_SIZE(p_dispwidget->current_msgs); i++)
|
|
{
|
|
disp_widget_msg_t *msg = p_dispwidget->current_msgs[i];
|
|
if (!msg)
|
|
continue;
|
|
|
|
/* Note: gfx_widgets_free() is only called when
|
|
* main_exit() is invoked. At this stage, we cannot
|
|
* guarantee that any task pointers are valid (the
|
|
* task may have been free()'d, but we can't know
|
|
* that here) - so all we can do is unset the task
|
|
* pointer associated with each message
|
|
* > If we don't do this, gfx_widgets_msg_queue_free()
|
|
* will generate heap-use-after-free errors */
|
|
msg->task_ptr = NULL;
|
|
|
|
gfx_widgets_msg_queue_free(p_dispwidget, msg);
|
|
}
|
|
#ifdef HAVE_THREADS
|
|
slock_unlock(p_dispwidget->current_msgs_lock);
|
|
|
|
slock_free(p_dispwidget->current_msgs_lock);
|
|
p_dispwidget->current_msgs_lock = NULL;
|
|
#endif
|
|
|
|
p_dispwidget->msg_queue_tasks_count = 0;
|
|
|
|
/* Font */
|
|
video_coord_array_free(
|
|
&p_dispwidget->gfx_widget_fonts.regular.raster_block.carr);
|
|
video_coord_array_free(
|
|
&p_dispwidget->gfx_widget_fonts.bold.raster_block.carr);
|
|
video_coord_array_free(
|
|
&p_dispwidget->gfx_widget_fonts.msg_queue.raster_block.carr);
|
|
|
|
font_driver_bind_block(NULL, NULL);
|
|
}
|
|
|
|
static void gfx_widgets_context_reset(
|
|
dispgfx_widget_t *p_dispwidget,
|
|
gfx_display_t *p_disp,
|
|
settings_t *settings,
|
|
bool is_threaded,
|
|
unsigned width, unsigned height, bool fullscreen,
|
|
const char *dir_assets, char *font_path)
|
|
{
|
|
size_t i;
|
|
|
|
/* Load textures */
|
|
/* Icons */
|
|
for (i = 0; i < MENU_WIDGETS_ICON_LAST; i++)
|
|
gfx_display_reset_textures_list(
|
|
gfx_widgets_icons_names[i],
|
|
p_dispwidget->monochrome_png_path,
|
|
&p_dispwidget->gfx_widgets_icons_textures[i],
|
|
TEXTURE_FILTER_MIPMAP_LINEAR,
|
|
NULL,
|
|
NULL);
|
|
|
|
if (p_dispwidget->gfx_widgets_icons_textures[MENU_WIDGETS_ICON_INFO])
|
|
p_dispwidget->flags |= DISPGFX_WIDGET_FLAG_MSG_QUEUE_HAS_ICONS;
|
|
else
|
|
p_dispwidget->flags &= ~DISPGFX_WIDGET_FLAG_MSG_QUEUE_HAS_ICONS;
|
|
|
|
#if 0
|
|
/* Message queue */
|
|
gfx_display_reset_textures_list(
|
|
"msg_queue_icon.png",
|
|
p_dispwidget->gfx_widgets_path,
|
|
&p_dispwidget->msg_queue_icon,
|
|
TEXTURE_FILTER_LINEAR,
|
|
NULL,
|
|
NULL);
|
|
gfx_display_reset_textures_list(
|
|
"msg_queue_icon_outline.png",
|
|
p_dispwidget->gfx_widgets_path,
|
|
&p_dispwidget->msg_queue_icon_outline,
|
|
TEXTURE_FILTER_LINEAR,
|
|
NULL,
|
|
NULL);
|
|
gfx_display_reset_textures_list(
|
|
"msg_queue_icon_rect.png",
|
|
p_dispwidget->gfx_widgets_path,
|
|
&p_dispwidget->msg_queue_icon_rect,
|
|
TEXTURE_FILTER_NEAREST,
|
|
NULL,
|
|
NULL);
|
|
|
|
if ( p_dispwidget->msg_queue_icon
|
|
&& p_dispwidget->msg_queue_icon_outline
|
|
&& p_dispwidget->msg_queue_icon_rect)
|
|
p_dispwidget->flags |= DISPGFX_WIDGET_FLAG_MSG_QUEUE_HAS_ICONS;
|
|
else
|
|
p_dispwidget->flags &= ~DISPGFX_WIDGET_FLAG_MSG_QUEUE_HAS_ICONS;
|
|
#endif
|
|
|
|
for (i = 0; i < ARRAY_SIZE(widgets); i++)
|
|
{
|
|
const gfx_widget_t* widget = widgets[i];
|
|
|
|
if (widget->context_reset)
|
|
widget->context_reset(is_threaded, width, height,
|
|
fullscreen, dir_assets, font_path,
|
|
p_dispwidget->monochrome_png_path,
|
|
p_dispwidget->gfx_widgets_path);
|
|
}
|
|
|
|
/* Update scaling/dimensions */
|
|
p_dispwidget->last_video_width = width;
|
|
p_dispwidget->last_video_height = height;
|
|
#ifdef HAVE_XMB
|
|
if (p_disp->menu_driver_id == MENU_DRIVER_ID_XMB)
|
|
p_dispwidget->last_scale_factor = gfx_display_get_widget_pixel_scale(
|
|
p_disp, settings,
|
|
p_dispwidget->last_video_width,
|
|
p_dispwidget->last_video_height, fullscreen);
|
|
else
|
|
#endif
|
|
p_dispwidget->last_scale_factor = gfx_display_get_dpi_scale(
|
|
p_disp, settings,
|
|
p_dispwidget->last_video_width,
|
|
p_dispwidget->last_video_height,
|
|
fullscreen, true);
|
|
|
|
gfx_widgets_layout(p_disp, p_dispwidget,
|
|
is_threaded, dir_assets, font_path);
|
|
video_driver_monitor_reset();
|
|
}
|
|
|
|
bool gfx_widgets_init(
|
|
void *data_disp,
|
|
void *data_anim,
|
|
void *settings_data,
|
|
uintptr_t widgets_active_ptr,
|
|
bool video_is_threaded,
|
|
unsigned width, unsigned height, bool fullscreen,
|
|
const char *dir_assets, char *font_path)
|
|
{
|
|
size_t i;
|
|
unsigned color = 0x1A1A1A;
|
|
dispgfx_widget_t *p_dispwidget = &dispwidget_st;
|
|
gfx_display_t *p_disp = (gfx_display_t*)data_disp;
|
|
gfx_animation_t *p_anim = (gfx_animation_t*)data_anim;
|
|
settings_t *settings = (settings_t*)settings_data;
|
|
p_dispwidget->divider_width_1px = 1;
|
|
p_dispwidget->gfx_widgets_generic_tag = (uintptr_t)widgets_active_ptr;
|
|
|
|
if (!gfx_display_init_first_driver(p_disp, video_is_threaded))
|
|
goto error;
|
|
gfx_display_set_alpha(p_dispwidget->backdrop_orig, 0.75f);
|
|
for (i = 0; i < 16; i++)
|
|
p_dispwidget->pure_white[i] = 1.00f;
|
|
|
|
p_dispwidget->msg_queue_bg[0] = HEX_R(color);
|
|
p_dispwidget->msg_queue_bg[1] = HEX_G(color);
|
|
p_dispwidget->msg_queue_bg[2] = HEX_B(color);
|
|
p_dispwidget->msg_queue_bg[3] = 1.0f;
|
|
p_dispwidget->msg_queue_bg[4] = HEX_R(color);
|
|
p_dispwidget->msg_queue_bg[5] = HEX_G(color);
|
|
p_dispwidget->msg_queue_bg[6] = HEX_B(color);
|
|
p_dispwidget->msg_queue_bg[7] = 1.0f;
|
|
p_dispwidget->msg_queue_bg[8] = HEX_R(color);
|
|
p_dispwidget->msg_queue_bg[9] = HEX_G(color);
|
|
p_dispwidget->msg_queue_bg[10] = HEX_B(color);
|
|
p_dispwidget->msg_queue_bg[11] = 1.0f;
|
|
p_dispwidget->msg_queue_bg[12] = HEX_R(color);
|
|
p_dispwidget->msg_queue_bg[13] = HEX_G(color);
|
|
p_dispwidget->msg_queue_bg[14] = HEX_B(color);
|
|
p_dispwidget->msg_queue_bg[15] = 1.0f;
|
|
|
|
if (!(p_dispwidget->flags & DISPGFX_WIDGET_FLAG_INITED))
|
|
{
|
|
char theme_path[PATH_MAX_LENGTH];
|
|
p_dispwidget->gfx_widgets_frame_count = 0;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(widgets); i++)
|
|
{
|
|
const gfx_widget_t* widget = widgets[i];
|
|
|
|
if (widget->init)
|
|
widget->init(p_disp, p_anim, video_is_threaded, fullscreen);
|
|
}
|
|
|
|
if (!fifo_initialize(&p_dispwidget->msg_queue,
|
|
MSG_QUEUE_PENDING_MAX * sizeof(disp_widget_msg_t*)))
|
|
goto error;
|
|
|
|
memset(&p_dispwidget->current_msgs[0], 0, sizeof(p_dispwidget->current_msgs));
|
|
p_dispwidget->current_msgs_size = 0;
|
|
|
|
#ifdef HAVE_THREADS
|
|
p_dispwidget->current_msgs_lock = slock_new();
|
|
#endif
|
|
|
|
fill_pathname_join_special(
|
|
p_dispwidget->gfx_widgets_path,
|
|
dir_assets,
|
|
"menu_widgets",
|
|
sizeof(p_dispwidget->gfx_widgets_path)
|
|
);
|
|
fill_pathname_join_special(
|
|
p_dispwidget->xmb_path,
|
|
dir_assets,
|
|
"xmb",
|
|
sizeof(p_dispwidget->xmb_path)
|
|
);
|
|
/* Base path */
|
|
fill_pathname_join_special(p_dispwidget->ozone_path,
|
|
dir_assets,
|
|
"ozone",
|
|
sizeof(p_dispwidget->ozone_path));
|
|
fill_pathname_join_special(p_dispwidget->ozone_regular_font_path,
|
|
p_dispwidget->ozone_path, "regular.ttf",
|
|
sizeof(p_dispwidget->ozone_regular_font_path));
|
|
fill_pathname_join_special(p_dispwidget->ozone_bold_font_path,
|
|
p_dispwidget->ozone_path, "bold.ttf",
|
|
sizeof(p_dispwidget->ozone_bold_font_path));
|
|
fill_pathname_join_special(
|
|
theme_path,
|
|
p_dispwidget->xmb_path,
|
|
"monochrome",
|
|
sizeof(theme_path)
|
|
);
|
|
fill_pathname_join_special(
|
|
p_dispwidget->monochrome_png_path,
|
|
theme_path,
|
|
"png",
|
|
sizeof(p_dispwidget->monochrome_png_path)
|
|
);
|
|
fill_pathname_join_special(p_dispwidget->assets_pkg_dir,
|
|
settings->paths.directory_assets, "pkg",
|
|
sizeof(p_dispwidget->assets_pkg_dir));
|
|
|
|
p_dispwidget->flags |= DISPGFX_WIDGET_FLAG_INITED;
|
|
}
|
|
|
|
gfx_widgets_context_reset(
|
|
p_dispwidget,
|
|
p_disp,
|
|
settings,
|
|
video_is_threaded,
|
|
width, height, fullscreen,
|
|
dir_assets, font_path);
|
|
|
|
return true;
|
|
|
|
error:
|
|
gfx_widgets_free(p_dispwidget);
|
|
return false;
|
|
}
|
|
|
|
static void gfx_widgets_context_destroy(dispgfx_widget_t *p_dispwidget)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(widgets); i++)
|
|
{
|
|
const gfx_widget_t* widget = widgets[i];
|
|
|
|
if (widget->context_destroy)
|
|
widget->context_destroy();
|
|
}
|
|
|
|
/* TODO: Dismiss onscreen notifications that have been freed */
|
|
|
|
/* Textures */
|
|
for (i = 0; i < MENU_WIDGETS_ICON_LAST; i++)
|
|
video_driver_texture_unload(&p_dispwidget->gfx_widgets_icons_textures[i]);
|
|
|
|
#if 0
|
|
video_driver_texture_unload(&p_dispwidget->msg_queue_icon);
|
|
video_driver_texture_unload(&p_dispwidget->msg_queue_icon_outline);
|
|
video_driver_texture_unload(&p_dispwidget->msg_queue_icon_rect);
|
|
|
|
p_dispwidget->msg_queue_icon = 0;
|
|
p_dispwidget->msg_queue_icon_outline = 0;
|
|
p_dispwidget->msg_queue_icon_rect = 0;
|
|
#endif
|
|
|
|
/* Fonts */
|
|
gfx_widgets_font_free(&p_dispwidget->gfx_widget_fonts.regular);
|
|
gfx_widgets_font_free(&p_dispwidget->gfx_widget_fonts.bold);
|
|
gfx_widgets_font_free(&p_dispwidget->gfx_widget_fonts.msg_queue);
|
|
}
|
|
|
|
|
|
void gfx_widgets_deinit(bool widgets_persisting)
|
|
{
|
|
dispgfx_widget_t *p_dispwidget = &dispwidget_st;
|
|
|
|
gfx_widgets_context_destroy(p_dispwidget);
|
|
|
|
if (!widgets_persisting)
|
|
gfx_widgets_free(p_dispwidget);
|
|
}
|
|
|
|
#ifdef HAVE_TRANSLATE
|
|
bool gfx_widgets_ai_service_overlay_load(
|
|
char* buffer, unsigned buffer_len,
|
|
enum image_type_enum image_type)
|
|
{
|
|
dispgfx_widget_t *p_dispwidget = &dispwidget_st;
|
|
if (p_dispwidget->ai_service_overlay_state == 0)
|
|
{
|
|
if (!gfx_display_reset_textures_list_buffer(
|
|
&p_dispwidget->ai_service_overlay_texture,
|
|
TEXTURE_FILTER_MIPMAP_LINEAR,
|
|
(void *) buffer, buffer_len, image_type,
|
|
&p_dispwidget->ai_service_overlay_width,
|
|
&p_dispwidget->ai_service_overlay_height))
|
|
return false;
|
|
p_dispwidget->ai_service_overlay_state = 1;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void gfx_widgets_ai_service_overlay_unload(void)
|
|
{
|
|
dispgfx_widget_t *p_dispwidget = &dispwidget_st;
|
|
if (p_dispwidget->ai_service_overlay_state == 1)
|
|
{
|
|
video_driver_texture_unload(&p_dispwidget->ai_service_overlay_texture);
|
|
p_dispwidget->ai_service_overlay_texture = 0;
|
|
p_dispwidget->ai_service_overlay_state = 0;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
dispgfx_widget_t *dispwidget_get_ptr(void)
|
|
{
|
|
return &dispwidget_st;
|
|
}
|
|
|
|
bool gfx_widgets_ready(void)
|
|
{
|
|
#ifdef HAVE_GFX_WIDGETS
|
|
return dispwidget_st.active;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|