2016-05-02 18:54:01 +00:00
|
|
|
// Pager support.
|
2016-02-28 04:04:57 +00:00
|
|
|
#ifndef FISH_PAGER_H
|
|
|
|
#define FISH_PAGER_H
|
|
|
|
|
2015-07-25 15:14:25 +00:00
|
|
|
#include <stddef.h>
|
2017-02-14 04:37:27 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
#include <memory>
|
2015-07-25 15:14:25 +00:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2016-04-21 06:00:54 +00:00
|
|
|
|
2015-07-25 15:14:25 +00:00
|
|
|
#include "common.h"
|
2013-12-07 20:43:40 +00:00
|
|
|
#include "complete.h"
|
2014-01-27 08:56:13 +00:00
|
|
|
#include "reader.h"
|
2016-05-02 18:54:01 +00:00
|
|
|
#include "screen.h"
|
2020-06-08 01:59:15 +00:00
|
|
|
#include "termsize.h"
|
2013-12-07 20:43:40 +00:00
|
|
|
|
2020-04-08 23:56:59 +00:00
|
|
|
#define PAGER_SELECTION_NONE static_cast<size_t>(-1)
|
2014-01-27 10:17:31 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
/// Represents rendering from the pager.
|
|
|
|
class page_rendering_t {
|
|
|
|
public:
|
2020-08-22 19:04:47 +00:00
|
|
|
size_t term_width{size_t(-1)};
|
|
|
|
size_t term_height{size_t(-1)};
|
|
|
|
size_t rows{0};
|
|
|
|
size_t cols{0};
|
|
|
|
size_t row_start{0};
|
|
|
|
size_t row_end{0};
|
|
|
|
size_t selected_completion_idx{size_t(-1)};
|
|
|
|
screen_data_t screen_data{};
|
|
|
|
|
|
|
|
size_t remaining_to_disclose{0};
|
|
|
|
|
|
|
|
bool search_field_shown{false};
|
|
|
|
editable_line_t search_field_line{};
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Returns a rendering with invalid data, useful to indicate "no rendering".
|
2014-01-17 20:04:03 +00:00
|
|
|
page_rendering_t();
|
2013-12-07 20:43:40 +00:00
|
|
|
};
|
|
|
|
|
2019-04-28 21:06:03 +00:00
|
|
|
enum class selection_motion_t {
|
|
|
|
// Visual directions.
|
|
|
|
north,
|
|
|
|
east,
|
|
|
|
south,
|
|
|
|
west,
|
|
|
|
page_north,
|
|
|
|
page_south,
|
|
|
|
|
|
|
|
// Logical directions.
|
|
|
|
next,
|
|
|
|
prev,
|
|
|
|
|
|
|
|
// Special value that means deselect.
|
|
|
|
deselect
|
|
|
|
};
|
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// The space between adjacent completions.
|
2014-01-20 00:41:26 +00:00
|
|
|
#define PAGER_SPACER_STRING L" "
|
|
|
|
#define PAGER_SPACER_STRING_WIDTH 2
|
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// How many rows we will show in the "initial" pager.
|
2014-01-23 01:45:27 +00:00
|
|
|
#define PAGER_UNDISCLOSED_MAX_ROWS 4
|
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
class pager_t {
|
2020-08-23 11:07:49 +00:00
|
|
|
size_t available_term_width;
|
|
|
|
size_t available_term_height;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2020-08-23 11:07:49 +00:00
|
|
|
size_t selected_completion_idx;
|
|
|
|
size_t suggested_row_start;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Fully disclosed means that we show all completions.
|
2020-08-23 11:07:49 +00:00
|
|
|
bool fully_disclosed;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Whether we show the search field.
|
2020-08-23 11:07:49 +00:00
|
|
|
bool search_field_shown;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Returns the index of the completion that should draw selected, using the given number of
|
|
|
|
// columns.
|
2014-01-20 00:41:26 +00:00
|
|
|
size_t visual_selected_completion_index(size_t rows, size_t cols) const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
public:
|
|
|
|
/// Data structure describing one or a group of related completions.
|
|
|
|
struct comp_t {
|
2019-11-25 11:03:25 +00:00
|
|
|
/// The list of all completion strings this entry applies to.
|
2020-08-23 11:07:49 +00:00
|
|
|
wcstring_list_t comp;
|
2016-05-02 18:54:01 +00:00
|
|
|
/// The description.
|
2020-08-23 11:07:49 +00:00
|
|
|
wcstring desc;
|
2016-05-02 18:54:01 +00:00
|
|
|
/// The representative completion.
|
2020-08-23 11:07:49 +00:00
|
|
|
completion_t representative;
|
2016-05-02 18:54:01 +00:00
|
|
|
/// On-screen width of the completion string.
|
2020-08-23 11:07:49 +00:00
|
|
|
size_t comp_width;
|
2016-05-02 18:54:01 +00:00
|
|
|
/// On-screen width of the description information.
|
2020-08-23 11:07:49 +00:00
|
|
|
size_t desc_width;
|
|
|
|
/// Minimum acceptable width.
|
|
|
|
// size_t min_width;
|
|
|
|
|
|
|
|
comp_t() : comp(), desc(), representative(L""), comp_width(0), desc_width(0) {}
|
2016-12-03 03:24:54 +00:00
|
|
|
|
2016-12-03 10:40:07 +00:00
|
|
|
// Our text looks like this:
|
|
|
|
// completion (description)
|
|
|
|
// Two spaces separating, plus parens, yields 4 total extra space
|
|
|
|
// but only if we have a description of course
|
|
|
|
size_t description_punctuated_width() const {
|
|
|
|
return this->desc_width + (this->desc_width ? 4 : 0);
|
2016-12-03 03:24:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the preferred width, containing the sum of the
|
2016-12-03 10:40:07 +00:00
|
|
|
// width of the completion, separator, description
|
2016-12-03 03:24:54 +00:00
|
|
|
size_t preferred_width() const {
|
2016-12-03 10:40:07 +00:00
|
|
|
return this->comp_width + this->description_punctuated_width();
|
2016-12-03 03:24:54 +00:00
|
|
|
}
|
2014-01-14 00:41:22 +00:00
|
|
|
};
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
private:
|
2020-08-23 11:07:49 +00:00
|
|
|
typedef std::vector<comp_t> comp_info_list_t;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// The filtered list of completion infos.
|
2014-01-14 00:41:22 +00:00
|
|
|
comp_info_list_t completion_infos;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// The unfiltered list. Note there's a lot of duplication here.
|
2014-01-27 08:56:13 +00:00
|
|
|
comp_info_list_t unfiltered_completion_infos;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2014-01-16 02:21:38 +00:00
|
|
|
wcstring prefix;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
bool completion_try_print(size_t cols, const wcstring &prefix, const comp_info_list_t &lst,
|
|
|
|
page_rendering_t *rendering, size_t suggested_start_row) const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
void recalc_min_widths(comp_info_list_t *lst) const;
|
2014-01-14 00:41:22 +00:00
|
|
|
void measure_completion_infos(std::vector<comp_t> *infos, const wcstring &prefix) const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2014-01-27 08:56:13 +00:00
|
|
|
bool completion_info_passes_filter(const comp_t &info) const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2019-11-19 00:54:36 +00:00
|
|
|
void completion_print(size_t cols, const size_t *width_by_column, size_t row_start,
|
2016-12-04 04:12:53 +00:00
|
|
|
size_t row_stop, const wcstring &prefix, const comp_info_list_t &lst,
|
2016-05-02 18:54:01 +00:00
|
|
|
page_rendering_t *rendering) const;
|
|
|
|
line_t completion_print_item(const wcstring &prefix, const comp_t *c, size_t row, size_t column,
|
2016-12-03 10:40:07 +00:00
|
|
|
size_t width, bool secondary, bool selected,
|
2016-05-02 18:54:01 +00:00
|
|
|
page_rendering_t *rendering) const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
public:
|
|
|
|
// The text of the search field.
|
2014-01-27 08:56:13 +00:00
|
|
|
editable_line_t search_field_line;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Sets the set of completions.
|
2019-11-19 00:54:36 +00:00
|
|
|
void set_completions(const completion_list_t &raw_completions);
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Sets the prefix.
|
2014-01-16 02:21:38 +00:00
|
|
|
void set_prefix(const wcstring &pref);
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2020-06-08 01:59:15 +00:00
|
|
|
// Sets the terminal size.
|
|
|
|
void set_term_size(termsize_t ts);
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Changes the selected completion in the given direction according to the layout of the given
|
|
|
|
// rendering. Returns true if the selection changed.
|
2019-04-28 21:06:03 +00:00
|
|
|
bool select_next_completion_in_direction(selection_motion_t direction,
|
2016-05-02 18:54:01 +00:00
|
|
|
const page_rendering_t &rendering);
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Returns the currently selected completion for the given rendering.
|
2014-01-20 00:41:26 +00:00
|
|
|
const completion_t *selected_completion(const page_rendering_t &rendering) const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Indicates the row and column for the given rendering. Returns -1 if no selection.
|
2014-01-24 23:59:18 +00:00
|
|
|
size_t get_selected_row(const page_rendering_t &rendering) const;
|
|
|
|
size_t get_selected_column(const page_rendering_t &rendering) const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Produces a rendering of the completions, at the given term size.
|
2014-01-14 00:41:22 +00:00
|
|
|
page_rendering_t render() const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2020-08-23 11:07:49 +00:00
|
|
|
// Updates the rendering if it's stale.
|
2014-01-17 20:04:03 +00:00
|
|
|
void update_rendering(page_rendering_t *rendering) const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Indicates if there are no completions, and therefore nothing to render.
|
2014-01-16 02:21:38 +00:00
|
|
|
bool empty() const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Clears all completions and the prefix.
|
2014-01-16 02:21:38 +00:00
|
|
|
void clear();
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Updates the completions list per the filter.
|
2014-01-27 08:56:13 +00:00
|
|
|
void refilter_completions();
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Sets whether the search field is shown.
|
2014-01-27 08:56:13 +00:00
|
|
|
void set_search_field_shown(bool flag);
|
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Gets whether the search field shown.
|
2014-01-27 08:56:13 +00:00
|
|
|
bool is_search_field_shown() const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Indicates if we are navigating our contents.
|
2014-01-27 08:56:13 +00:00
|
|
|
bool is_navigating_contents() const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Become fully disclosed.
|
2014-02-20 21:30:26 +00:00
|
|
|
void set_fully_disclosed(bool flag);
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-02 18:54:01 +00:00
|
|
|
// Position of the cursor.
|
2014-01-27 08:56:13 +00:00
|
|
|
size_t cursor_position() const;
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2020-08-23 11:07:49 +00:00
|
|
|
// Constructor
|
2014-01-16 02:21:38 +00:00
|
|
|
pager_t();
|
2014-01-14 00:41:22 +00:00
|
|
|
};
|
2016-02-28 04:04:57 +00:00
|
|
|
|
|
|
|
#endif
|