Improve NCurses UI menu scroll bar, use a variable height
This commit is contained in:
parent
03c74b7a88
commit
c3bafea2cd
|
@ -359,6 +359,12 @@ Key NCursesUI::get_key()
|
||||||
return Key::Invalid;
|
return Key::Invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
T div_round_up(T a, T b)
|
||||||
|
{
|
||||||
|
return (a - T(1)) / b + T(1);
|
||||||
|
}
|
||||||
|
|
||||||
void NCursesUI::draw_menu()
|
void NCursesUI::draw_menu()
|
||||||
{
|
{
|
||||||
// menu show may have not created the window if it did not fit.
|
// menu show may have not created the window if it did not fit.
|
||||||
|
@ -366,24 +372,31 @@ void NCursesUI::draw_menu()
|
||||||
if (not m_menu_win)
|
if (not m_menu_win)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
auto menu_fg = get_color_pair(m_menu_fg);
|
const auto menu_fg = get_color_pair(m_menu_fg);
|
||||||
auto menu_bg = get_color_pair(m_menu_bg);
|
const auto menu_bg = get_color_pair(m_menu_bg);
|
||||||
|
|
||||||
auto scroll_fg = get_color_pair({ Colors::White, Colors::White });
|
const auto scroll_fg = get_color_pair({ Colors::White, Colors::White });
|
||||||
auto scroll_bg = get_color_pair(m_menu_bg);
|
const auto scroll_bg = get_color_pair(m_menu_bg);
|
||||||
|
|
||||||
wattron(m_menu_win, COLOR_PAIR(menu_bg));
|
wattron(m_menu_win, COLOR_PAIR(menu_bg));
|
||||||
wbkgdset(m_menu_win, COLOR_PAIR(menu_bg));
|
wbkgdset(m_menu_win, COLOR_PAIR(menu_bg));
|
||||||
DisplayCoord menu_size = window_size(m_menu_win);
|
|
||||||
CharCount column_width = (menu_size.column - 1) / m_menu_columns;
|
const int choice_count = (int)m_choices.size();
|
||||||
LineCount mark_line = (menu_size.line * m_selected_choice) / (int)m_choices.size();
|
const DisplayCoord win_size = window_size(m_menu_win);
|
||||||
for (auto line = 0_line; line < menu_size.line; ++line)
|
const LineCount win_lines = win_size.line;
|
||||||
|
|
||||||
|
const CharCount column_width = (win_size.column - 1) / m_menu_columns;
|
||||||
|
|
||||||
|
const LineCount menu_lines = div_round_up(choice_count, m_menu_columns);
|
||||||
|
const LineCount mark_height = clamp(div_round_up(win_lines * win_lines, menu_lines), 1_line, win_lines);
|
||||||
|
const LineCount mark_line = (win_lines - mark_height) * m_menu_top_line / std::max(1_line, menu_lines - win_lines);
|
||||||
|
for (auto line = 0_line; line < win_lines; ++line)
|
||||||
{
|
{
|
||||||
wmove(m_menu_win, (int)line, 0);
|
wmove(m_menu_win, (int)line, 0);
|
||||||
for (int col = 0; col < m_menu_columns; ++col)
|
for (int col = 0; col < m_menu_columns; ++col)
|
||||||
{
|
{
|
||||||
int choice_idx = (int)(m_menu_top_line + line) * m_menu_columns + col;
|
int choice_idx = (int)(m_menu_top_line + line) * m_menu_columns + col;
|
||||||
if (choice_idx >= m_choices.size())
|
if (choice_idx >= choice_count)
|
||||||
break;
|
break;
|
||||||
if (choice_idx == m_selected_choice)
|
if (choice_idx == m_selected_choice)
|
||||||
wattron(m_menu_win, COLOR_PAIR(menu_fg));
|
wattron(m_menu_win, COLOR_PAIR(menu_fg));
|
||||||
|
@ -396,9 +409,10 @@ void NCursesUI::draw_menu()
|
||||||
waddch(m_menu_win, ' ');
|
waddch(m_menu_win, ' ');
|
||||||
wattron(m_menu_win, COLOR_PAIR(menu_bg));
|
wattron(m_menu_win, COLOR_PAIR(menu_bg));
|
||||||
}
|
}
|
||||||
|
const bool is_mark = line >= mark_line and line < mark_line + mark_height;
|
||||||
wclrtoeol(m_menu_win);
|
wclrtoeol(m_menu_win);
|
||||||
wmove(m_menu_win, (int)line, (int)menu_size.column - 1);
|
wmove(m_menu_win, (int)line, (int)win_size.column - 1);
|
||||||
wattron(m_menu_win, COLOR_PAIR(line == mark_line ? scroll_fg : scroll_bg));
|
wattron(m_menu_win, COLOR_PAIR(is_mark ? scroll_fg : scroll_bg));
|
||||||
waddch(m_menu_win, ' ');
|
waddch(m_menu_win, ' ');
|
||||||
wattron(m_menu_win, COLOR_PAIR(menu_bg));
|
wattron(m_menu_win, COLOR_PAIR(menu_bg));
|
||||||
}
|
}
|
||||||
|
@ -431,7 +445,7 @@ void NCursesUI::menu_show(memoryview<String> choices,
|
||||||
longest += 1;
|
longest += 1;
|
||||||
|
|
||||||
m_menu_columns = (style == MenuStyle::Prompt) ? (int)((maxsize.column - 1) / longest) : 1;
|
m_menu_columns = (style == MenuStyle::Prompt) ? (int)((maxsize.column - 1) / longest) : 1;
|
||||||
int lines = std::min(10, (int)ceilf((float)m_choices.size()/m_menu_columns));
|
int lines = std::min(10, div_round_up((int)m_choices.size(), m_menu_columns));
|
||||||
|
|
||||||
DisplayCoord pos = { anchor.line+1, anchor.column };
|
DisplayCoord pos = { anchor.line+1, anchor.column };
|
||||||
if (pos.line + lines >= maxsize.line)
|
if (pos.line + lines >= maxsize.line)
|
||||||
|
@ -447,7 +461,9 @@ void NCursesUI::menu_show(memoryview<String> choices,
|
||||||
|
|
||||||
void NCursesUI::menu_select(int selected)
|
void NCursesUI::menu_select(int selected)
|
||||||
{
|
{
|
||||||
if (selected < 0 or selected >= m_choices.size())
|
const int choice_count = m_choices.size();
|
||||||
|
const LineCount menu_lines = (choice_count + m_menu_columns - 1) / m_menu_columns;
|
||||||
|
if (selected < 0 or selected >= choice_count)
|
||||||
{
|
{
|
||||||
m_selected_choice = -1;
|
m_selected_choice = -1;
|
||||||
m_menu_top_line = 0;
|
m_menu_top_line = 0;
|
||||||
|
@ -455,12 +471,12 @@ void NCursesUI::menu_select(int selected)
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
m_selected_choice = selected;
|
m_selected_choice = selected;
|
||||||
LineCount selected_line = m_selected_choice / m_menu_columns;
|
const LineCount selected_line = m_selected_choice / m_menu_columns;
|
||||||
DisplayCoord menu_size = window_size(m_menu_win);
|
const LineCount win_lines = window_size(m_menu_win).line;
|
||||||
if (selected_line < m_menu_top_line)
|
if (selected_line < m_menu_top_line)
|
||||||
m_menu_top_line = selected_line;
|
m_menu_top_line = selected_line;
|
||||||
if (selected_line >= m_menu_top_line + menu_size.line)
|
if (selected_line >= m_menu_top_line + win_lines)
|
||||||
m_menu_top_line = selected_line;
|
m_menu_top_line = std::min(selected_line, std::max(0_line, menu_lines - win_lines));
|
||||||
}
|
}
|
||||||
|
|
||||||
draw_menu();
|
draw_menu();
|
||||||
|
|
Loading…
Reference in New Issue
Block a user