2016-05-03 17:36:22 +00:00
|
|
|
// Prototypes for functions for reading data from stdin and passing to the parser. If stdin is a
|
|
|
|
// keyboard, it supplies a killring, history, syntax highlighting, tab-completion and various other
|
|
|
|
// features.
|
2005-10-04 15:11:39 +00:00
|
|
|
#ifndef FISH_READER_H
|
|
|
|
#define FISH_READER_H
|
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
#include <stddef.h>
|
2017-02-14 04:37:27 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2005-10-04 15:11:39 +00:00
|
|
|
|
2012-02-11 01:54:21 +00:00
|
|
|
#include "common.h"
|
2012-02-24 20:13:35 +00:00
|
|
|
#include "complete.h"
|
2014-01-15 09:01:25 +00:00
|
|
|
#include "highlight.h"
|
2015-05-03 00:49:38 +00:00
|
|
|
#include "parse_constants.h"
|
2005-10-04 15:11:39 +00:00
|
|
|
|
2018-09-09 09:25:51 +00:00
|
|
|
class environment_t;
|
2012-02-06 00:42:24 +00:00
|
|
|
class history_t;
|
2016-04-21 06:00:54 +00:00
|
|
|
class io_chain_t;
|
2020-01-16 01:14:47 +00:00
|
|
|
class operation_context_t;
|
|
|
|
class parser_t;
|
2012-01-23 05:40:08 +00:00
|
|
|
|
2020-02-04 11:47:44 +00:00
|
|
|
/// An edit action that can be undone.
|
|
|
|
struct edit_t {
|
|
|
|
/// When undoing the edit we use this to restore the previous cursor position.
|
|
|
|
size_t cursor_position_before_edit = 0;
|
|
|
|
|
|
|
|
/// The span of text that is replaced by this edit.
|
|
|
|
size_t offset, length;
|
|
|
|
|
|
|
|
/// The strings that are removed and added by this edit, respectively.
|
|
|
|
wcstring old, replacement;
|
|
|
|
|
|
|
|
explicit edit_t(size_t offset, size_t length, wcstring replacement)
|
|
|
|
: offset(offset), length(length), replacement(std::move(replacement)) {}
|
|
|
|
|
|
|
|
/// Used for testing.
|
|
|
|
bool operator==(const edit_t &other) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Modify a string according to the given edit.
|
|
|
|
/// Currently exposed for testing only.
|
|
|
|
void apply_edit(wcstring *target, const edit_t &edit);
|
|
|
|
|
|
|
|
/// The history of all edits to some command line.
|
|
|
|
struct undo_history_t {
|
|
|
|
/// The stack of edits that can be undone or redone atomically.
|
|
|
|
std::vector<edit_t> edits;
|
|
|
|
|
|
|
|
/// The position in the undo stack that corresponds to the current
|
|
|
|
/// state of the input line.
|
|
|
|
/// Invariants:
|
|
|
|
/// edits_applied - 1 is the index of the next edit to undo.
|
|
|
|
/// edits_applied is the index of the next edit to redo.
|
|
|
|
///
|
|
|
|
/// For example, if nothing was undone, edits_applied is edits.size().
|
|
|
|
/// If every single edit was undone, edits_applied is 0.
|
|
|
|
size_t edits_applied = 0;
|
|
|
|
|
|
|
|
/// Whether we allow the next edit to be grouped together with the
|
|
|
|
/// last one.
|
|
|
|
bool may_coalesce = false;
|
|
|
|
|
|
|
|
/// Empty the history.
|
|
|
|
void clear();
|
|
|
|
};
|
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Helper class for storing a command line.
|
|
|
|
class editable_line_t {
|
|
|
|
/// The command line.
|
2020-02-04 11:47:44 +00:00
|
|
|
wcstring text_;
|
2016-05-03 17:36:22 +00:00
|
|
|
/// The current position of the cursor in the command line.
|
2020-02-04 11:47:44 +00:00
|
|
|
size_t position_ = 0;
|
2020-02-04 11:47:44 +00:00
|
|
|
|
|
|
|
public:
|
2020-02-04 11:47:44 +00:00
|
|
|
undo_history_t undo_history;
|
|
|
|
|
2020-02-04 11:47:44 +00:00
|
|
|
const wcstring &text() const { return text_; }
|
2020-02-04 11:47:44 +00:00
|
|
|
/// Set the text directly without maintaining undo invariants. Use with caution.
|
|
|
|
void set_text_bypassing_undo_history(wcstring &&text) { text_ = text; }
|
2020-02-04 11:47:44 +00:00
|
|
|
|
|
|
|
size_t position() const { return position_; }
|
2020-02-04 11:47:44 +00:00
|
|
|
void set_position(size_t position) { position_ = position; }
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
// Gets the length of the text.
|
2020-02-04 11:47:44 +00:00
|
|
|
size_t size() const { return text().size(); }
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2020-02-04 11:47:44 +00:00
|
|
|
bool empty() const { return text().empty(); }
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2020-03-13 20:59:10 +00:00
|
|
|
wchar_t at(size_t idx) const { return text().at(idx); }
|
2020-02-04 11:47:44 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
void clear() {
|
2020-02-04 11:47:44 +00:00
|
|
|
undo_history.clear();
|
|
|
|
if (empty()) return;
|
|
|
|
set_text_bypassing_undo_history(L"");
|
2020-02-04 11:47:44 +00:00
|
|
|
set_position(0);
|
2014-01-27 08:56:13 +00:00
|
|
|
}
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2020-02-04 11:47:44 +00:00
|
|
|
/// Modify the commandline according to @edit. Most modifications to the
|
|
|
|
/// text should pass through this function. You can use one of the wrappers below.
|
|
|
|
void push_edit(edit_t &&edit);
|
2014-03-31 17:01:39 +00:00
|
|
|
|
2020-02-04 11:47:44 +00:00
|
|
|
/// Erase @length characters starting at @offset.
|
|
|
|
void erase_substring(size_t offset, size_t length);
|
|
|
|
/// Replace the text of length @length at @offset by @replacement.
|
|
|
|
void replace_substring(size_t offset, size_t length, wcstring &&replacement);
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Inserts a substring of str given by start, len at the cursor position.
|
2014-07-25 19:23:42 +00:00
|
|
|
void insert_string(const wcstring &str, size_t start = 0, size_t len = wcstring::npos);
|
2020-02-04 11:47:44 +00:00
|
|
|
|
|
|
|
/// Undo the most recent edit that was not yet undone. Returns true on success.
|
|
|
|
bool undo();
|
|
|
|
|
|
|
|
/// Redo the most recent undo. Returns true on success.
|
|
|
|
bool redo();
|
2014-01-26 08:41:30 +00:00
|
|
|
};
|
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Read commands from \c fd until encountering EOF.
|
2020-01-28 19:39:26 +00:00
|
|
|
/// The fd is not closed.
|
2019-05-27 01:51:26 +00:00
|
|
|
int reader_read(parser_t &parser, int fd, const io_chain_t &io);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2020-07-06 03:09:53 +00:00
|
|
|
/// Mark that we encountered SIGHUP and must (soon) exit. This is invoked from a signal handler.
|
|
|
|
void reader_sighup();
|
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Initialize the reader.
|
2005-09-20 13:26:39 +00:00
|
|
|
void reader_init();
|
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Restore the term mode at startup.
|
2013-10-26 22:22:20 +00:00
|
|
|
void restore_term_mode();
|
|
|
|
|
2017-07-28 04:32:49 +00:00
|
|
|
/// Change the history file for the current command reading context.
|
2018-09-09 05:13:54 +00:00
|
|
|
void reader_change_history(const wcstring &name);
|
2017-07-28 04:32:49 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Write the title to the titlebar. This function is called just before a new application starts
|
|
|
|
/// executing and just after it finishes.
|
|
|
|
///
|
|
|
|
/// \param cmd Command line string passed to \c fish_title if is defined.
|
2019-05-05 03:20:52 +00:00
|
|
|
/// \param parser The parser to use for autoloading fish_title.
|
2016-05-03 17:36:22 +00:00
|
|
|
/// \param reset_cursor_position If set, issue a \r so the line driver knows where we are
|
2019-05-05 03:20:52 +00:00
|
|
|
void reader_write_title(const wcstring &cmd, parser_t &parser, bool reset_cursor_position = true);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2020-08-12 18:05:04 +00:00
|
|
|
/// Tell the reader that it needs to re-exec the prompt and repaint.
|
|
|
|
/// This may be called in response to e.g. a color variable change.
|
|
|
|
void reader_schedule_prompt_repaint();
|
2012-03-25 10:00:38 +00:00
|
|
|
|
2019-06-02 22:41:23 +00:00
|
|
|
/// Enqueue an event to the back of the reader's input queue.
|
|
|
|
class char_event_t;
|
|
|
|
void reader_queue_ch(const char_event_t &ch);
|
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Run the specified command with the correct terminal modes, and while taking care to perform job
|
|
|
|
/// notification, set the title, etc.
|
2013-04-07 19:40:08 +00:00
|
|
|
void reader_run_command(const wcstring &buff);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Get the string of character currently entered into the command buffer, or 0 if interactive mode
|
|
|
|
/// is uninitialized.
|
2012-02-06 09:45:16 +00:00
|
|
|
const wchar_t *reader_get_buffer();
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Returns the current reader's history.
|
2018-07-14 18:47:34 +00:00
|
|
|
history_t *reader_get_history();
|
2012-02-06 00:42:24 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Set the string of characters in the command buffer, as well as the cursor position.
|
|
|
|
///
|
|
|
|
/// \param b the new buffer value
|
|
|
|
/// \param p the cursor position. If \c p is larger than the length of the command line, the cursor
|
|
|
|
/// is placed on the last character.
|
2018-08-11 21:30:10 +00:00
|
|
|
void reader_set_buffer(const wcstring &b, size_t p = -1);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Get the current cursor position in the command line. If interactive mode is uninitialized,
|
2016-05-04 22:19:47 +00:00
|
|
|
/// return (size_t)-1.
|
2012-08-04 20:47:56 +00:00
|
|
|
size_t reader_get_cursor_pos();
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Get the current selection range in the command line. Returns false if there is no active
|
|
|
|
/// selection, true otherwise.
|
2014-03-29 21:19:45 +00:00
|
|
|
bool reader_get_selection(size_t *start, size_t *len);
|
2014-01-15 14:07:22 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Return the value of the interrupted flag, which is set by the sigint handler, and clear it if it
|
2020-01-14 23:20:04 +00:00
|
|
|
/// was set. In practice this will return 0 or SIGINT.
|
|
|
|
int reader_test_and_clear_interrupted();
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Clear the interrupted flag unconditionally without handling anything. The flag could have been
|
|
|
|
/// set e.g. when an interrupt arrived just as we were ending an earlier \c reader_readline
|
|
|
|
/// invocation but before the \c is_interactive_read flag was cleared.
|
2013-01-20 21:23:27 +00:00
|
|
|
void reader_reset_interrupted();
|
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Return the value of the interrupted flag, which is set by the sigint handler, and clear it if it
|
|
|
|
/// was set. If the current reader is interruptible, call \c reader_exit().
|
2013-01-22 10:00:02 +00:00
|
|
|
int reader_reading_interrupted();
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Read one line of input. Before calling this function, reader_push() must have been called in
|
|
|
|
/// order to set up a valid reader environment. If nchars > 0, return after reading that many
|
|
|
|
/// characters even if a full line has not yet been read. Note: the returned value may be longer
|
|
|
|
/// than nchars if a single keypress resulted in multiple characters being inserted into the
|
|
|
|
/// commandline.
|
2019-02-24 21:59:49 +00:00
|
|
|
maybe_t<wcstring> reader_readline(int nchars);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// Configuration that we provide to a reader.
|
|
|
|
struct reader_config_t {
|
|
|
|
/// Left prompt command, typically fish_prompt.
|
|
|
|
wcstring left_prompt_cmd{};
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// Right prompt command, typically fish_right_prompt.
|
|
|
|
wcstring right_prompt_cmd{};
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// Whether tab completion is OK.
|
|
|
|
bool complete_ok{false};
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// Whether to perform syntax highlighting.
|
|
|
|
bool highlight_ok{false};
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// Whether to perform syntax checking before returning.
|
|
|
|
bool syntax_check_ok{false};
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// Whether to allow autosuggestions.
|
|
|
|
bool autosuggest_ok{false};
|
2012-11-08 03:59:20 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// Whether to expand abbreviations.
|
|
|
|
bool expand_abbrev_ok{false};
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// Whether to exit on interrupt (^C).
|
|
|
|
bool exit_on_interrupt{false};
|
2012-11-18 10:43:35 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// If set, do not show what is typed.
|
|
|
|
bool in_silent_mode{false};
|
|
|
|
};
|
2013-07-17 07:38:04 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// Push a new reader environment controlled by \p conf.
|
|
|
|
/// If \p history_name is not empty, then use the history with that name.
|
|
|
|
void reader_push(parser_t &parser, const wcstring &history_name, reader_config_t &&conf);
|
2013-01-20 21:23:27 +00:00
|
|
|
|
2020-08-03 21:53:09 +00:00
|
|
|
/// Return to previous reader environment.
|
|
|
|
void reader_pop();
|
2017-04-09 18:48:21 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// The readers interrupt signal handler. Cancels all currently running blocks.
|
2016-10-09 21:38:26 +00:00
|
|
|
void reader_handle_sigint();
|
2005-10-05 22:37:08 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// This function returns true if fish is exiting by force, i.e. because stdin died.
|
2017-01-24 23:15:38 +00:00
|
|
|
bool reader_exit_forced();
|
2006-05-14 10:16:23 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Test whether the interactive reader is in search mode.
|
2018-08-11 21:30:10 +00:00
|
|
|
bool reader_is_in_search_mode();
|
2007-09-21 14:05:49 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Test whether the interactive reader has visible pager contents.
|
2018-08-11 21:30:10 +00:00
|
|
|
bool reader_has_pager_contents();
|
2014-01-17 20:53:01 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Given a command line and an autosuggestion, return the string that gets shown to the user.
|
|
|
|
/// Exposed for testing purposes only.
|
|
|
|
wcstring combine_command_and_autosuggestion(const wcstring &cmdline,
|
|
|
|
const wcstring &autosuggestion);
|
2014-01-17 20:53:01 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Expand abbreviations at the given cursor position. Exposed for testing purposes only.
|
2019-08-06 23:09:23 +00:00
|
|
|
/// \return none if no abbreviations were expanded, otherwise the new command line.
|
2020-02-04 11:47:44 +00:00
|
|
|
maybe_t<edit_t> reader_expand_abbreviation_in_command(const wcstring &cmdline, size_t cursor_pos,
|
|
|
|
const environment_t &vars);
|
2013-07-17 07:38:04 +00:00
|
|
|
|
2016-05-03 17:36:22 +00:00
|
|
|
/// Apply a completion string. Exposed for testing only.
|
|
|
|
wcstring completion_apply_to_command_line(const wcstring &val_str, complete_flags_t flags,
|
|
|
|
const wcstring &command_line, size_t *inout_cursor_pos,
|
|
|
|
bool append_only);
|
2013-02-02 22:50:22 +00:00
|
|
|
|
2018-08-21 03:50:27 +00:00
|
|
|
/// Return the current interactive reads loop count. Useful for determining how many commands have
|
|
|
|
/// been executed between invocations of code.
|
2019-04-29 01:13:55 +00:00
|
|
|
uint64_t reader_run_count();
|
2018-08-21 03:50:27 +00:00
|
|
|
|
2020-08-02 20:37:19 +00:00
|
|
|
/// Returns the current "generation" of interactive status. Useful for determining whether the
|
|
|
|
/// previous command produced a status.
|
|
|
|
uint64_t reader_status_count();
|
|
|
|
|
2005-10-04 15:11:39 +00:00
|
|
|
#endif
|