mirror of
https://github.com/RPCS3/rpcs3.git
synced 2026-04-20 22:05:06 +00:00
Some checks are pending
Generate Translation Template / Generate Translation Template (push) Waiting to run
Build RPCS3 / RPCS3 Linux ${{ matrix.os }} ${{ matrix.compiler }} (/rpcs3/.ci/build-linux-aarch64.sh, gcc, rpcs3/rpcs3-ci-jammy-aarch64:1.9, ubuntu-24.04-arm) (push) Waiting to run
Build RPCS3 / RPCS3 Linux ${{ matrix.os }} ${{ matrix.compiler }} (/rpcs3/.ci/build-linux.sh, gcc, rpcs3/rpcs3-ci-jammy:1.9, ubuntu-24.04) (push) Waiting to run
Build RPCS3 / RPCS3 Linux ${{ matrix.os }} ${{ matrix.compiler }} (a1d35836e8d45bfc6f63c26f0a3e5d46ef622fe1, rpcs3/rpcs3-binaries-linux-arm64, /rpcs3/.ci/build-linux-aarch64.sh, clang, rpcs3/rpcs3-ci-jammy-aarch64:1.9, ubuntu-24.04-arm) (push) Waiting to run
Build RPCS3 / RPCS3 Linux ${{ matrix.os }} ${{ matrix.compiler }} (d812f1254a1157c80fd402f94446310560f54e5f, rpcs3/rpcs3-binaries-linux, /rpcs3/.ci/build-linux.sh, clang, rpcs3/rpcs3-ci-jammy:1.9, ubuntu-24.04) (push) Waiting to run
Build RPCS3 / RPCS3 Mac ${{ matrix.name }} (0, 51ae32f468089a8169aaf1567de355ff4a3e0842, rpcs3/rpcs3-binaries-mac, Intel) (push) Waiting to run
Build RPCS3 / RPCS3 Mac ${{ matrix.name }} (1, 8e21bdbc40711a3fccd18fbf17b742348b0f4281, rpcs3/rpcs3-binaries-mac-arm64, Apple Silicon) (push) Waiting to run
Build RPCS3 / RPCS3 Windows (push) Waiting to run
Build RPCS3 / RPCS3 Windows Clang ${{ matrix.arch }} (aarch64, clang, clangarm64, ARM64, windows-11-arm) (push) Waiting to run
Build RPCS3 / RPCS3 Windows Clang ${{ matrix.arch }} (x86_64, clang, clang64, X64, windows-2025) (push) Waiting to run
Build RPCS3 / RPCS3 FreeBSD (push) Waiting to run
362 lines
8.7 KiB
C++
362 lines
8.7 KiB
C++
#include "stdafx.h"
|
|
#include "overlay_list_view.hpp"
|
|
#include "Emu/system_config.h"
|
|
|
|
namespace rsx
|
|
{
|
|
namespace overlays
|
|
{
|
|
list_view::list_view(u16 width, u16 height, bool use_separators, bool can_deny)
|
|
: m_use_separators(use_separators)
|
|
{
|
|
w = width;
|
|
h = height;
|
|
|
|
auto scroll_indicator_track = std::make_unique<overlay_element>(5, height);
|
|
auto scroll_indicator_grip = std::make_unique<rounded_rect>(5, height);
|
|
|
|
scroll_indicator_grip->set_pos(1, 0);
|
|
scroll_indicator_grip->set_size(5, 5);
|
|
scroll_indicator_grip->radius = 2;
|
|
scroll_indicator_track->set_size(7, height);
|
|
|
|
m_scroll_indicator = std::make_unique<box_layout>();
|
|
m_scroll_indicator_track = m_scroll_indicator->add_element(scroll_indicator_track);
|
|
m_scroll_indicator_grip = m_scroll_indicator->add_element(scroll_indicator_grip);
|
|
|
|
m_accept_btn = std::make_unique<image_button>(120, 20);
|
|
m_cancel_btn = std::make_unique<image_button>(120, 20);
|
|
m_highlight_box = std::make_unique<overlay_element>(width, 0);
|
|
|
|
m_accept_btn->set_size(120, 30);
|
|
m_cancel_btn->set_size(120, 30);
|
|
|
|
if (g_cfg.sys.enter_button_assignment == enter_button_assign::circle)
|
|
{
|
|
m_accept_btn->set_image_resource(resource_config::standard_image_resource::circle);
|
|
m_cancel_btn->set_image_resource(resource_config::standard_image_resource::cross);
|
|
}
|
|
else
|
|
{
|
|
m_accept_btn->set_image_resource(resource_config::standard_image_resource::cross);
|
|
m_cancel_btn->set_image_resource(resource_config::standard_image_resource::circle);
|
|
}
|
|
|
|
m_scroll_indicator->set_pos(width - 7, 0);
|
|
m_accept_btn->set_pos(30, height + 20);
|
|
|
|
if (can_deny)
|
|
{
|
|
m_deny_btn = std::make_unique<image_button>(120, 20);
|
|
m_deny_btn->set_size(120, 30);
|
|
m_deny_btn->set_image_resource(resource_config::standard_image_resource::triangle);
|
|
m_deny_btn->set_pos(180, height + 20);
|
|
m_deny_btn->set_text(localized_string_id::RSX_OVERLAYS_LIST_DENY);
|
|
m_deny_btn->set_font("Arial", 16);
|
|
|
|
m_cancel_btn->set_pos(330, height + 20);
|
|
}
|
|
else
|
|
{
|
|
m_cancel_btn->set_pos(180, height + 20);
|
|
}
|
|
|
|
m_accept_btn->set_text(localized_string_id::RSX_OVERLAYS_LIST_SELECT);
|
|
m_cancel_btn->set_text(localized_string_id::RSX_OVERLAYS_LIST_CANCEL);
|
|
|
|
m_accept_btn->set_font("Arial", 16);
|
|
m_cancel_btn->set_font("Arial", 16);
|
|
|
|
auto_resize = false;
|
|
back_color = {0.15f, 0.15f, 0.15f, 0.8f};
|
|
|
|
m_highlight_box->back_color = {.5f, .5f, .8f, 0.2f};
|
|
m_highlight_box->pulse_effect_enabled = true;
|
|
}
|
|
|
|
const overlay_element* list_view::get_selected_entry() const
|
|
{
|
|
if (m_selected_entry < 0)
|
|
{
|
|
return nullptr; // Ideally unreachable but it should still be possible to recover by user interaction.
|
|
}
|
|
|
|
const usz current_index = static_cast<usz>(m_selected_entry) * (m_use_separators ? 2 : 1);
|
|
|
|
if (current_index >= m_items.size())
|
|
{
|
|
return nullptr; // Ideally unreachable but it should still be possible to recover by user interaction.
|
|
}
|
|
|
|
return m_items[current_index].get();
|
|
}
|
|
|
|
void list_view::update_selection()
|
|
{
|
|
const overlay_element* current_element = get_selected_entry();
|
|
|
|
for (auto& item : m_items)
|
|
{
|
|
if (item)
|
|
{
|
|
item->set_selected(item.get() == current_element);
|
|
}
|
|
}
|
|
|
|
if (!current_element)
|
|
{
|
|
return; // Ideally unreachable but it should still be possible to recover by user interaction.
|
|
}
|
|
|
|
// Calculate bounds
|
|
const auto min_y = current_element->y - y;
|
|
const auto max_y = current_element->y + current_element->h + pack_padding + 2 - y;
|
|
|
|
if (min_y < scroll_offset_value)
|
|
{
|
|
scroll_offset_value = min_y;
|
|
}
|
|
else if (max_y > (h + scroll_offset_value))
|
|
{
|
|
scroll_offset_value = max_y - h - 2;
|
|
}
|
|
|
|
if (m_elements_height > h)
|
|
{
|
|
update_scroll_indicator();
|
|
}
|
|
|
|
m_highlight_box->set_pos(current_element->x, current_element->y);
|
|
m_highlight_box->h = current_element->h + pack_padding;
|
|
m_highlight_box->y -= scroll_offset_value;
|
|
|
|
m_highlight_box->refresh();
|
|
refresh();
|
|
}
|
|
|
|
void list_view::select_entry(s32 entry)
|
|
{
|
|
const s32 max_entry = m_elements_count - 1;
|
|
|
|
// Reset the pulse slightly below 1 rising on each user interaction
|
|
m_highlight_box->set_sinus_offset(1.6f);
|
|
|
|
if (m_selected_entry != entry)
|
|
{
|
|
m_selected_entry = std::max(0, std::min(entry, max_entry));
|
|
update_selection();
|
|
}
|
|
else
|
|
{
|
|
refresh();
|
|
}
|
|
}
|
|
|
|
void list_view::select_next(u16 count)
|
|
{
|
|
select_entry(m_selected_entry + count);
|
|
}
|
|
|
|
void list_view::select_previous(u16 count)
|
|
{
|
|
select_entry(m_selected_entry - count);
|
|
}
|
|
|
|
void list_view::add_entry(std::unique_ptr<overlay_element>& entry)
|
|
{
|
|
// Add entry view
|
|
add_element(entry);
|
|
m_elements_count++;
|
|
|
|
// Add separator
|
|
if (m_use_separators)
|
|
{
|
|
auto separator = std::make_unique<overlay_element>();
|
|
separator->back_color = fore_color;
|
|
separator->w = w;
|
|
separator->h = 2;
|
|
add_element(separator);
|
|
}
|
|
|
|
if (m_selected_entry < 0)
|
|
m_selected_entry = 0;
|
|
|
|
m_elements_height = advance_pos;
|
|
update_selection();
|
|
}
|
|
|
|
void list_view::clear_items()
|
|
{
|
|
layout_container::clear_items();
|
|
|
|
m_elements_height = 0;
|
|
m_selected_entry = -1;
|
|
m_elements_count = 0;
|
|
}
|
|
|
|
u16 list_view::get_elements_count() const
|
|
{
|
|
return m_elements_count;
|
|
}
|
|
|
|
s32 list_view::get_selected_index() const
|
|
{
|
|
return m_selected_entry;
|
|
}
|
|
|
|
void list_view::hide_prompt_buttons(bool hidden)
|
|
{
|
|
m_accept_btn->set_visible(!hidden);
|
|
m_cancel_btn->set_visible(!hidden);
|
|
|
|
if (m_deny_btn)
|
|
{
|
|
m_deny_btn->set_visible(!hidden);
|
|
}
|
|
|
|
refresh();
|
|
}
|
|
|
|
void list_view::hide_scroll_indicator(bool hidden)
|
|
{
|
|
m_scroll_indicator_track->set_visible(!hidden);
|
|
m_scroll_indicator_grip->set_visible(!hidden);
|
|
|
|
refresh();
|
|
}
|
|
|
|
void list_view::hide_row_highliter(bool hidden)
|
|
{
|
|
m_highlight_box->set_visible(!hidden);
|
|
refresh();
|
|
}
|
|
|
|
void list_view::disable_selection_pulse(bool disabled)
|
|
{
|
|
m_highlight_box->pulse_effect_enabled = !disabled;
|
|
m_highlight_box->set_sinus_offset(1.5f);
|
|
m_highlight_box->refresh();
|
|
|
|
refresh();
|
|
}
|
|
|
|
void list_view::set_cancel_only(bool cancel_only)
|
|
{
|
|
if (cancel_only)
|
|
m_cancel_btn->set_pos(x + 30, y + h + 20);
|
|
else if (m_deny_btn)
|
|
m_cancel_btn->set_pos(x + 330, y + h + 20);
|
|
else
|
|
m_cancel_btn->set_pos(x + 180, y + h + 20);
|
|
|
|
m_cancel_only = cancel_only;
|
|
refresh();
|
|
}
|
|
|
|
bool list_view::get_cancel_only() const
|
|
{
|
|
return m_cancel_only;
|
|
}
|
|
|
|
void list_view::translate(s16 _x, s16 _y)
|
|
{
|
|
layout_container::translate(_x, _y);
|
|
m_scroll_indicator->translate(_x, _y);
|
|
m_accept_btn->translate(_x, _y);
|
|
m_cancel_btn->translate(_x, _y);
|
|
|
|
if (m_deny_btn)
|
|
{
|
|
m_deny_btn->translate(_x, _y);
|
|
}
|
|
}
|
|
|
|
void list_view::set_size(u16 w, u16 h)
|
|
{
|
|
vertical_layout::set_size(w, h);
|
|
|
|
m_highlight_box->w = w;
|
|
|
|
update_scroll_indicator();
|
|
|
|
m_scroll_indicator->refresh();
|
|
m_highlight_box->refresh();
|
|
}
|
|
|
|
void list_view::set_pos(s16 x, s16 y)
|
|
{
|
|
vertical_layout::set_pos(x, y);
|
|
|
|
update_selection();
|
|
}
|
|
|
|
void list_view::update_scroll_indicator()
|
|
{
|
|
// Always reposition
|
|
m_scroll_indicator->set_pos(x + w - 7, y);
|
|
m_scroll_indicator->set_size(7, h);
|
|
|
|
// Render grip and track
|
|
m_scroll_indicator_track->back_color = this->fore_color * 0.25f;
|
|
m_scroll_indicator_track->back_color.a = 1.f;
|
|
m_scroll_indicator_track->h = m_scroll_indicator->h;
|
|
|
|
m_scroll_indicator_grip->back_color = this->fore_color;
|
|
m_scroll_indicator_grip->back_color.a = 1.f;
|
|
m_scroll_indicator_grip->h = 0;
|
|
|
|
if (m_elements_height < h)
|
|
{
|
|
return;
|
|
}
|
|
|
|
const f32 viewable_ratio = static_cast<f32>(m_scroll_indicator->h) / m_elements_height;
|
|
const f32 scroll_ratio = static_cast<f32>(scroll_offset_value) / m_elements_height;
|
|
|
|
m_scroll_indicator_grip->h = static_cast<u16>(viewable_ratio * m_scroll_indicator->h);
|
|
m_scroll_indicator_grip->y = m_scroll_indicator->y + static_cast<s16>(scroll_ratio * m_scroll_indicator->h);
|
|
}
|
|
|
|
compiled_resource& list_view::get_compiled()
|
|
{
|
|
if (is_compiled())
|
|
{
|
|
return compiled_resources;
|
|
}
|
|
|
|
compiled_resources.clear();
|
|
|
|
if (!is_visible())
|
|
{
|
|
m_is_compiled = true;
|
|
return compiled_resources;
|
|
}
|
|
|
|
auto& compiled = vertical_layout::get_compiled();
|
|
compiled.add(m_highlight_box->get_compiled());
|
|
compiled.add(m_cancel_btn->get_compiled());
|
|
|
|
if (m_elements_height > h)
|
|
{
|
|
// Auto overflow
|
|
compiled.add(m_scroll_indicator->get_compiled());
|
|
}
|
|
|
|
if (m_cancel_only)
|
|
{
|
|
m_is_compiled = true;
|
|
return compiled_resources;
|
|
}
|
|
|
|
compiled.add(m_accept_btn->get_compiled());
|
|
|
|
if (m_deny_btn)
|
|
{
|
|
compiled.add(m_deny_btn->get_compiled());
|
|
}
|
|
|
|
m_is_compiled = true;
|
|
return compiled_resources;
|
|
}
|
|
} // namespace overlays
|
|
} // namespace rsx
|