2016-04-20 10:49:15 +08:00
|
|
|
// Functions used for implementing the commandline builtin.
|
2016-05-19 06:30:21 +08:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
2016-04-21 14:00:54 +08:00
|
|
|
#include <errno.h>
|
2017-02-13 12:24:22 +08:00
|
|
|
#include <stddef.h>
|
2016-04-20 10:49:15 +08:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <wchar.h>
|
2005-09-20 21:26:39 +08:00
|
|
|
|
|
|
|
#include "builtin.h"
|
|
|
|
#include "common.h"
|
2016-04-20 10:49:15 +08:00
|
|
|
#include "fallback.h" // IWYU pragma: keep
|
2005-09-20 21:26:39 +08:00
|
|
|
#include "input.h"
|
2016-04-21 14:00:54 +08:00
|
|
|
#include "io.h"
|
2016-04-20 10:49:15 +08:00
|
|
|
#include "parse_util.h"
|
|
|
|
#include "proc.h"
|
|
|
|
#include "reader.h"
|
|
|
|
#include "tokenizer.h"
|
|
|
|
#include "util.h"
|
|
|
|
#include "wgetopt.h"
|
2016-04-21 14:00:54 +08:00
|
|
|
#include "wutil.h" // IWYU pragma: keep
|
|
|
|
|
|
|
|
class parser_t;
|
2005-09-20 21:26:39 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
/// Which part of the comandbuffer are we operating on.
|
|
|
|
enum {
|
|
|
|
STRING_MODE = 1, // operate on entire buffer
|
|
|
|
JOB_MODE, // operate on job under cursor
|
|
|
|
PROCESS_MODE, // operate on process under cursor
|
|
|
|
TOKEN_MODE // operate on token under cursor
|
|
|
|
};
|
|
|
|
|
|
|
|
/// For text insertion, how should it be done.
|
|
|
|
enum {
|
|
|
|
REPLACE_MODE = 1, // replace current text
|
|
|
|
INSERT_MODE, // insert at cursor position
|
|
|
|
APPEND_MODE // insert at end of current token/command/buffer
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Pointer to what the commandline builtin considers to be the current contents of the command line
|
|
|
|
/// buffer.
|
|
|
|
static const wchar_t *current_buffer = 0;
|
|
|
|
// What the commandline builtin considers to be the current cursor position.
|
2012-08-05 06:11:43 +08:00
|
|
|
static size_t current_cursor_pos = (size_t)(-1);
|
2006-10-05 02:49:39 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
/// Returns the current commandline buffer.
|
|
|
|
static const wchar_t *get_buffer() { return current_buffer; }
|
2006-01-31 00:51:50 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
/// Returns the position of the cursor.
|
|
|
|
static size_t get_cursor_pos() { return current_cursor_pos; }
|
2006-01-31 00:51:50 +08:00
|
|
|
|
2017-01-30 10:08:00 +08:00
|
|
|
static owning_lock<wcstring_list_t> &get_transient_stack() {
|
2014-08-16 09:14:36 +08:00
|
|
|
ASSERT_IS_MAIN_THREAD();
|
2017-01-30 10:08:00 +08:00
|
|
|
static owning_lock<wcstring_list_t> s_transient_stack;
|
|
|
|
return s_transient_stack;
|
2014-08-16 09:14:36 +08:00
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
static bool get_top_transient(wcstring *out_result) {
|
2017-08-19 03:26:35 +08:00
|
|
|
auto &&locked = get_transient_stack().acquire();
|
2017-01-30 10:08:00 +08:00
|
|
|
wcstring_list_t &stack = locked.value;
|
|
|
|
if (stack.empty()) {
|
|
|
|
return false;
|
2014-08-16 09:14:36 +08:00
|
|
|
}
|
2017-01-30 10:08:00 +08:00
|
|
|
out_result->assign(stack.back());
|
|
|
|
return true;
|
2014-08-16 09:14:36 +08:00
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
builtin_commandline_scoped_transient_t::builtin_commandline_scoped_transient_t(
|
|
|
|
const wcstring &cmd) {
|
2014-08-16 09:14:36 +08:00
|
|
|
ASSERT_IS_MAIN_THREAD();
|
2017-08-19 03:26:35 +08:00
|
|
|
auto &&locked = get_transient_stack().acquire();
|
2017-01-30 10:08:00 +08:00
|
|
|
wcstring_list_t &stack = locked.value;
|
|
|
|
stack.push_back(cmd);
|
|
|
|
this->token = stack.size();
|
2014-08-16 09:14:36 +08:00
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
builtin_commandline_scoped_transient_t::~builtin_commandline_scoped_transient_t() {
|
2014-08-16 09:14:36 +08:00
|
|
|
ASSERT_IS_MAIN_THREAD();
|
2017-08-19 03:26:35 +08:00
|
|
|
auto &&locked = get_transient_stack().acquire();
|
2017-01-30 10:08:00 +08:00
|
|
|
wcstring_list_t &stack = locked.value;
|
|
|
|
assert(this->token == stack.size());
|
|
|
|
stack.pop_back();
|
2014-08-16 09:14:36 +08:00
|
|
|
}
|
2005-09-20 21:26:39 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
/// Replace/append/insert the selection with/at/after the specified string.
|
|
|
|
///
|
|
|
|
/// \param begin beginning of selection
|
|
|
|
/// \param end end of selection
|
|
|
|
/// \param insert the string to insert
|
|
|
|
/// \param append_mode can be one of REPLACE_MODE, INSERT_MODE or APPEND_MODE, affects the way the
|
|
|
|
/// test update is performed
|
|
|
|
static void replace_part(const wchar_t *begin, const wchar_t *end, const wchar_t *insert,
|
|
|
|
int append_mode) {
|
2012-11-19 08:30:30 +08:00
|
|
|
const wchar_t *buff = get_buffer();
|
|
|
|
size_t out_pos = get_cursor_pos();
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2012-02-23 03:07:34 +08:00
|
|
|
wcstring out;
|
2005-09-20 21:26:39 +08:00
|
|
|
|
2012-02-23 03:07:34 +08:00
|
|
|
out.append(buff, begin - buff);
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
switch (append_mode) {
|
|
|
|
case REPLACE_MODE: {
|
2012-11-19 16:31:03 +08:00
|
|
|
out.append(insert);
|
2016-04-20 10:49:15 +08:00
|
|
|
out_pos = wcslen(insert) + (begin - buff);
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
|
|
|
}
|
2016-04-20 10:49:15 +08:00
|
|
|
case APPEND_MODE: {
|
|
|
|
out.append(begin, end - begin);
|
2012-11-19 16:31:03 +08:00
|
|
|
out.append(insert);
|
|
|
|
break;
|
|
|
|
}
|
2016-04-20 10:49:15 +08:00
|
|
|
case INSERT_MODE: {
|
|
|
|
long cursor = get_cursor_pos() - (begin - buff);
|
2012-11-19 16:31:03 +08:00
|
|
|
out.append(begin, cursor);
|
|
|
|
out.append(insert);
|
2016-04-20 10:49:15 +08:00
|
|
|
out.append(begin + cursor, end - begin - cursor);
|
|
|
|
out_pos += wcslen(insert);
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
|
|
|
}
|
2016-10-30 08:25:48 +08:00
|
|
|
default: {
|
|
|
|
DIE("unexpected append_mode");
|
|
|
|
break;
|
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
2012-11-19 08:30:30 +08:00
|
|
|
out.append(end);
|
|
|
|
reader_set_buffer(out, out_pos);
|
2005-09-20 21:26:39 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
/// Output the specified selection.
|
|
|
|
///
|
|
|
|
/// \param begin start of selection
|
|
|
|
/// \param end end of selection
|
|
|
|
/// \param cut_at_cursor whether printing should stop at the surrent cursor position
|
|
|
|
/// \param tokenize whether the string should be tokenized, printing one string token on every line
|
|
|
|
/// and skipping non-string tokens
|
|
|
|
static void write_part(const wchar_t *begin, const wchar_t *end, int cut_at_cursor, int tokenize,
|
|
|
|
io_streams_t &streams) {
|
|
|
|
size_t pos = get_cursor_pos() - (begin - get_buffer());
|
|
|
|
|
|
|
|
if (tokenize) {
|
|
|
|
wchar_t *buff = wcsndup(begin, end - begin);
|
|
|
|
// fwprintf( stderr, L"Subshell: %ls, end char %lc\n", buff, *end );
|
2014-01-13 05:33:35 +08:00
|
|
|
wcstring out;
|
2012-11-22 09:48:35 +08:00
|
|
|
tokenizer_t tok(buff, TOK_ACCEPT_UNFINISHED);
|
2015-07-26 15:12:36 +08:00
|
|
|
tok_t token;
|
2016-04-20 10:49:15 +08:00
|
|
|
while (tok.next(&token)) {
|
|
|
|
if ((cut_at_cursor) && (token.offset + token.text.size() >= pos)) break;
|
2012-11-19 08:30:30 +08:00
|
|
|
|
2016-10-23 11:32:25 +08:00
|
|
|
if (token.type == TOK_STRING) {
|
|
|
|
wcstring tmp = token.text;
|
|
|
|
unescape_string_in_place(&tmp, UNESCAPE_INCOMPLETE);
|
|
|
|
out.append(tmp);
|
|
|
|
out.push_back(L'\n');
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
|
|
|
|
2015-09-22 02:24:49 +08:00
|
|
|
streams.out.append(out);
|
2012-11-19 08:30:30 +08:00
|
|
|
free(buff);
|
2016-04-20 10:49:15 +08:00
|
|
|
} else {
|
|
|
|
if (cut_at_cursor) {
|
2016-10-21 09:53:31 +08:00
|
|
|
streams.out.append(begin, pos);
|
|
|
|
} else {
|
|
|
|
streams.out.append(begin, end - begin);
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2016-10-21 09:53:31 +08:00
|
|
|
streams.out.push_back(L'\n');
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2005-09-20 21:26:39 +08:00
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
/// The commandline builtin. It is used for specifying a new value for the commandline.
|
|
|
|
int builtin_commandline(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
2017-06-15 03:26:05 +08:00
|
|
|
wchar_t *cmd = argv[0];
|
2016-04-20 10:49:15 +08:00
|
|
|
int buffer_part = 0;
|
|
|
|
int cut_at_cursor = 0;
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2012-11-19 08:30:30 +08:00
|
|
|
int argc = builtin_count_args(argv);
|
2016-04-20 10:49:15 +08:00
|
|
|
int append_mode = 0;
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2012-11-19 08:30:30 +08:00
|
|
|
int function_mode = 0;
|
2014-01-15 22:07:22 +08:00
|
|
|
int selection_mode = 0;
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2012-11-19 08:30:30 +08:00
|
|
|
int tokenize = 0;
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2012-11-19 08:30:30 +08:00
|
|
|
int cursor_mode = 0;
|
|
|
|
int line_mode = 0;
|
|
|
|
int search_mode = 0;
|
2014-01-18 04:53:01 +08:00
|
|
|
int paging_mode = 0;
|
2014-03-08 01:20:42 +08:00
|
|
|
const wchar_t *begin = NULL, *end = NULL;
|
2016-04-20 10:49:15 +08:00
|
|
|
|
2014-08-16 09:14:36 +08:00
|
|
|
scoped_push<const wchar_t *> saved_current_buffer(¤t_buffer);
|
|
|
|
scoped_push<size_t> saved_current_cursor_pos(¤t_cursor_pos);
|
2016-04-20 10:49:15 +08:00
|
|
|
|
2014-08-16 09:14:36 +08:00
|
|
|
wcstring transient_commandline;
|
2016-04-20 10:49:15 +08:00
|
|
|
if (get_top_transient(&transient_commandline)) {
|
2014-08-16 09:14:36 +08:00
|
|
|
current_buffer = transient_commandline.c_str();
|
|
|
|
current_cursor_pos = transient_commandline.size();
|
2016-04-20 10:49:15 +08:00
|
|
|
} else {
|
2012-11-19 08:30:30 +08:00
|
|
|
current_buffer = reader_get_buffer();
|
|
|
|
current_cursor_pos = reader_get_cursor_pos();
|
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if (!get_buffer()) {
|
|
|
|
if (is_interactive_session) {
|
|
|
|
// Prompt change requested while we don't have a prompt, most probably while reading the
|
|
|
|
// init files. Just ignore it.
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_CMD_ERROR;
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2015-09-22 02:24:49 +08:00
|
|
|
streams.err.append(argv[0]);
|
|
|
|
streams.err.append(L": Can not set commandline in non-interactive mode\n");
|
2017-06-15 03:26:05 +08:00
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_CMD_ERROR;
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2017-06-30 12:49:57 +08:00
|
|
|
static const wchar_t *short_options = L":abijpctwforhI:CLSsP";
|
2017-06-11 03:30:09 +08:00
|
|
|
static const struct woption long_options[] = {{L"append", no_argument, NULL, 'a'},
|
|
|
|
{L"insert", no_argument, NULL, 'i'},
|
|
|
|
{L"replace", no_argument, NULL, 'r'},
|
|
|
|
{L"current-job", no_argument, NULL, 'j'},
|
|
|
|
{L"current-process", no_argument, NULL, 'p'},
|
|
|
|
{L"current-token", no_argument, NULL, 't'},
|
|
|
|
{L"current-buffer", no_argument, NULL, 'b'},
|
|
|
|
{L"cut-at-cursor", no_argument, NULL, 'c'},
|
|
|
|
{L"function", no_argument, NULL, 'f'},
|
|
|
|
{L"tokenize", no_argument, NULL, 'o'},
|
|
|
|
{L"help", no_argument, NULL, 'h'},
|
|
|
|
{L"input", required_argument, NULL, 'I'},
|
|
|
|
{L"cursor", no_argument, NULL, 'C'},
|
|
|
|
{L"line", no_argument, NULL, 'L'},
|
|
|
|
{L"search-mode", no_argument, NULL, 'S'},
|
|
|
|
{L"selection", no_argument, NULL, 's'},
|
|
|
|
{L"paging-mode", no_argument, NULL, 'P'},
|
|
|
|
{NULL, 0, NULL, 0}};
|
|
|
|
|
|
|
|
int opt;
|
|
|
|
wgetopter_t w;
|
|
|
|
while ((opt = w.wgetopt_long(argc, argv, short_options, long_options, NULL)) != -1) {
|
2016-04-20 10:49:15 +08:00
|
|
|
switch (opt) {
|
|
|
|
case L'a': {
|
2012-11-19 16:31:03 +08:00
|
|
|
append_mode = APPEND_MODE;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case L'b': {
|
2012-11-19 16:31:03 +08:00
|
|
|
buffer_part = STRING_MODE;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case L'i': {
|
2012-11-19 16:31:03 +08:00
|
|
|
append_mode = INSERT_MODE;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case L'r': {
|
2012-11-19 16:31:03 +08:00
|
|
|
append_mode = REPLACE_MODE;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'c': {
|
|
|
|
cut_at_cursor = 1;
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 't': {
|
2012-11-19 16:31:03 +08:00
|
|
|
buffer_part = TOKEN_MODE;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'j': {
|
2012-11-19 16:31:03 +08:00
|
|
|
buffer_part = JOB_MODE;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'p': {
|
2012-11-19 16:31:03 +08:00
|
|
|
buffer_part = PROCESS_MODE;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'f': {
|
|
|
|
function_mode = 1;
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'o': {
|
|
|
|
tokenize = 1;
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'I': {
|
2015-07-26 09:16:00 +08:00
|
|
|
current_buffer = w.woptarg;
|
|
|
|
current_cursor_pos = wcslen(w.woptarg);
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'C': {
|
2012-11-19 16:31:03 +08:00
|
|
|
cursor_mode = 1;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'L': {
|
2012-11-19 16:31:03 +08:00
|
|
|
line_mode = 1;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'S': {
|
2012-11-19 16:31:03 +08:00
|
|
|
search_mode = 1;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 's': {
|
2014-01-15 22:07:22 +08:00
|
|
|
selection_mode = 1;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'P': {
|
2014-01-18 04:53:01 +08:00
|
|
|
paging_mode = 1;
|
|
|
|
break;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
|
|
|
case 'h': {
|
2017-06-15 03:26:05 +08:00
|
|
|
builtin_print_help(parser, streams, cmd, streams.out);
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_CMD_OK;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
2017-06-30 12:49:57 +08:00
|
|
|
case ':': {
|
2017-07-02 05:03:47 +08:00
|
|
|
builtin_missing_argument(parser, streams, cmd, argv[w.woptind - 1]);
|
2017-06-30 12:49:57 +08:00
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
2016-04-20 10:49:15 +08:00
|
|
|
case L'?': {
|
2017-06-15 03:26:05 +08:00
|
|
|
builtin_unknown_option(parser, streams, cmd, argv[w.woptind - 1]);
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_INVALID_ARGS;
|
2016-04-20 10:49:15 +08:00
|
|
|
}
|
2016-10-30 08:25:48 +08:00
|
|
|
default: {
|
2017-06-11 03:30:09 +08:00
|
|
|
DIE("unexpected retval from wgetopt_long");
|
2016-10-30 08:25:48 +08:00
|
|
|
break;
|
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if (function_mode) {
|
2012-11-19 08:30:30 +08:00
|
|
|
int i;
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
// Check for invalid switch combinations.
|
|
|
|
if (buffer_part || cut_at_cursor || append_mode || tokenize || cursor_mode || line_mode ||
|
|
|
|
search_mode || paging_mode) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_COMBO, argv[0]);
|
2017-06-15 03:26:05 +08:00
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_INVALID_ARGS;
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
2012-11-19 08:30:30 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if (argc == w.woptind) {
|
2017-07-02 05:03:47 +08:00
|
|
|
builtin_missing_argument(parser, streams, cmd, argv[0]);
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_INVALID_ARGS;
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
2016-04-20 10:49:15 +08:00
|
|
|
|
|
|
|
for (i = w.woptind; i < argc; i++) {
|
2013-04-04 09:27:18 +08:00
|
|
|
wchar_t c = input_function_get_code(argv[i]);
|
2016-04-20 10:49:15 +08:00
|
|
|
if (c != INPUT_CODE_NONE) {
|
|
|
|
// input_unreadch inserts the specified keypress or readline function at the back of
|
|
|
|
// the queue of unused keypresses.
|
2015-04-06 02:07:17 +08:00
|
|
|
input_queue_ch(c);
|
2016-04-20 10:49:15 +08:00
|
|
|
} else {
|
2017-06-15 03:26:05 +08:00
|
|
|
streams.err.append_format(_(L"%ls: Unknown input function '%ls'"), cmd, argv[i]);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_INVALID_ARGS;
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
|
|
|
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_CMD_OK;
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if (selection_mode) {
|
2014-01-18 17:18:29 +08:00
|
|
|
size_t start, len;
|
2014-01-15 22:07:22 +08:00
|
|
|
const wchar_t *buffer = reader_get_buffer();
|
2016-04-20 10:49:15 +08:00
|
|
|
if (reader_get_selection(&start, &len)) {
|
2015-09-22 02:24:49 +08:00
|
|
|
streams.out.append(buffer + start, len);
|
2014-01-15 22:07:22 +08:00
|
|
|
}
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_CMD_OK;
|
2014-01-15 22:07:22 +08:00
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
// Check for invalid switch combinations.
|
|
|
|
if ((search_mode || line_mode || cursor_mode || paging_mode) && (argc - w.woptind > 1)) {
|
2017-01-24 01:32:50 +08:00
|
|
|
streams.err.append_format(L"%ls: Too many arguments", argv[0]);
|
2017-06-15 03:26:05 +08:00
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_INVALID_ARGS;
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if ((buffer_part || tokenize || cut_at_cursor) &&
|
|
|
|
(cursor_mode || line_mode || search_mode || paging_mode)) {
|
|
|
|
streams.err.append_format(BUILTIN_ERR_COMBO, argv[0]);
|
2017-06-15 03:26:05 +08:00
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_INVALID_ARGS;
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if ((tokenize || cut_at_cursor) && (argc - w.woptind)) {
|
|
|
|
streams.err.append_format(
|
2017-06-15 03:26:05 +08:00
|
|
|
BUILTIN_ERR_COMBO2, cmd,
|
2016-04-20 10:49:15 +08:00
|
|
|
L"--cut-at-cursor and --tokenize can not be used when setting the commandline");
|
2017-06-15 03:26:05 +08:00
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_INVALID_ARGS;
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if (append_mode && !(argc - w.woptind)) {
|
|
|
|
streams.err.append_format(
|
2017-06-15 03:26:05 +08:00
|
|
|
BUILTIN_ERR_COMBO2, cmd,
|
2016-04-20 10:49:15 +08:00
|
|
|
L"insertion mode switches can not be used when not in insertion mode");
|
2017-06-15 03:26:05 +08:00
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_INVALID_ARGS;
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
// Set default modes.
|
|
|
|
if (!append_mode) {
|
2012-11-19 08:30:30 +08:00
|
|
|
append_mode = REPLACE_MODE;
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if (!buffer_part) {
|
2012-11-19 08:30:30 +08:00
|
|
|
buffer_part = STRING_MODE;
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if (cursor_mode) {
|
|
|
|
if (argc - w.woptind) {
|
2016-11-23 12:24:03 +08:00
|
|
|
long new_pos = fish_wcstol(argv[w.woptind]);
|
|
|
|
if (errno) {
|
2017-06-15 03:26:05 +08:00
|
|
|
streams.err.append_format(BUILTIN_ERR_NOT_NUMBER, cmd, argv[w.woptind]);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.err);
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
current_buffer = reader_get_buffer();
|
|
|
|
new_pos = maxi(0L, mini(new_pos, (long)wcslen(current_buffer)));
|
|
|
|
reader_set_buffer(current_buffer, (size_t)new_pos);
|
2016-04-20 10:49:15 +08:00
|
|
|
} else {
|
|
|
|
streams.out.append_format(L"%lu\n", (unsigned long)reader_get_cursor_pos());
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_CMD_OK;
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if (line_mode) {
|
2012-11-19 08:30:30 +08:00
|
|
|
size_t pos = reader_get_cursor_pos();
|
|
|
|
const wchar_t *buff = reader_get_buffer();
|
2016-04-20 10:49:15 +08:00
|
|
|
streams.out.append_format(L"%lu\n", (unsigned long)parse_util_lineno(buff, pos));
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_CMD_OK;
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if (search_mode) {
|
|
|
|
return !reader_search_mode();
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
if (paging_mode) {
|
|
|
|
return !reader_has_pager_contents();
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-04-20 10:49:15 +08:00
|
|
|
switch (buffer_part) {
|
|
|
|
case STRING_MODE: {
|
2012-11-19 16:31:03 +08:00
|
|
|
begin = get_buffer();
|
2016-04-20 10:49:15 +08:00
|
|
|
end = begin + wcslen(begin);
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
|
|
|
}
|
2016-04-20 10:49:15 +08:00
|
|
|
case PROCESS_MODE: {
|
|
|
|
parse_util_process_extent(get_buffer(), get_cursor_pos(), &begin, &end);
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
|
|
|
}
|
2016-04-20 10:49:15 +08:00
|
|
|
case JOB_MODE: {
|
|
|
|
parse_util_job_extent(get_buffer(), get_cursor_pos(), &begin, &end);
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
|
|
|
}
|
2016-04-20 10:49:15 +08:00
|
|
|
case TOKEN_MODE: {
|
|
|
|
parse_util_token_extent(get_buffer(), get_cursor_pos(), &begin, &end, 0, 0);
|
2012-11-19 16:31:03 +08:00
|
|
|
break;
|
|
|
|
}
|
2016-10-30 08:25:48 +08:00
|
|
|
default: {
|
|
|
|
DIE("unexpected buffer_part");
|
|
|
|
break;
|
|
|
|
}
|
2012-11-19 08:30:30 +08:00
|
|
|
}
|
2012-11-18 18:23:22 +08:00
|
|
|
|
2016-10-23 11:32:25 +08:00
|
|
|
int arg_count = argc - w.woptind;
|
|
|
|
if (arg_count == 0) {
|
|
|
|
write_part(begin, end, cut_at_cursor, tokenize, streams);
|
|
|
|
} else if (arg_count == 1) {
|
|
|
|
replace_part(begin, end, argv[w.woptind], append_mode);
|
|
|
|
} else {
|
|
|
|
wcstring sb = argv[w.woptind];
|
|
|
|
for (int i = w.woptind + 1; i < argc; i++) {
|
|
|
|
sb.push_back(L'\n');
|
|
|
|
sb.append(argv[i]);
|
2012-11-19 16:31:03 +08:00
|
|
|
}
|
2016-10-23 11:32:25 +08:00
|
|
|
replace_part(begin, end, sb.c_str(), append_mode);
|
2012-11-18 18:23:22 +08:00
|
|
|
}
|
|
|
|
|
2017-05-05 12:35:41 +08:00
|
|
|
return STATUS_CMD_OK;
|
2005-09-20 21:26:39 +08:00
|
|
|
}
|