Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Do not autocomplete if only whitespace precedes cursor #410

Open
wants to merge 9 commits into
base: develop
Choose a base branch
from

Conversation

krageon
Copy link
Contributor

@krageon krageon commented Aug 30, 2017

If we are on the first character of a line and the input buffer is empty, return a tab character when autocompleting.

This should fix #254

@krageon
Copy link
Contributor Author

krageon commented Aug 30, 2017

Travis will probably fail on old php versions as I run only 7.1 locally, I'll fix that after I have lunch.

@krageon krageon force-pushed the feature/tab-paste branch from 8c7e5ef to 07f723c Compare August 30, 2017 15:26
@krageon krageon changed the base branch from master to develop August 30, 2017 15:27
* @param array $info readline_info() data
*
* @return array
*/
public function processCallback($input, $index, $info = array())
public function processCallback($input, $start, $end, $info = array())
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

unfortunately this is a public method, so we won't be able to break backwards compatibility. right now.

but! $info['end'] is supposed to be a thing already, and we're using it below. we could work the $end param from the callback into the $info array to augment and/or replace that one, or we could just use that one?

or we could add the $end param to the end here, but that's just a bit awkward.

or we could rename this method and add a deprecated stub of a method called processCallback that still takes the old arguments. i think i like this third option best :)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's actually the function passed to readline_completion_function that needs the change, which is also public. Refactoring that in a non-breaking manner gives me a large headache, so I reverted it back to the old (and admittedly broken) way. I use point now, which is more in line with what should happen anyway.


$trimmedInput = trim($precedingInput);

return empty($trimmedInput);
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

don't use empty, use === ''. because php.

>>> empty('0')
=> true

*
* @return array
*/
public function callback($input, $index)
public function callback($input, $start, $end)
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

do all the possible combinations of readline and libedit and php versions all provide $end for this callback? documentation is sadly lacking here :(

if not, should we fall back to the end in readline_info if it's available?

Copy link
Contributor Author

@krageon krageon Aug 31, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I looked here: https://github.com/php/php-src/blob/master/ext/readline/readline.c#L490

And I saw that params always contains int start and int end.

For editline / libedit, editline/readline.h is included. This has typedef char **rl_completion_func_t(const char *, int, int); and extern rl_completion_func_t *rl_attempted_completion_function;

in editline's readline.c I see this:

/*
 * complete word at current point
 */
/* ARGSUSED */
int
rl_complete(int ignore __attribute__((__unused__)), int invoking_key)
{
	static ct_buffer_t wbreak_conv, sprefix_conv;
	char *breakchars;

	if (h == NULL || e == NULL)
		rl_initialize();

	if (rl_inhibit_completion) {
		char arr[2];
		arr[0] = (char)invoking_key;
		arr[1] = '\0';
		el_insertstr(e, arr);
		return CC_REFRESH;
	}

	if (rl_completion_word_break_hook != NULL)
		breakchars = (*rl_completion_word_break_hook)();
	else
		breakchars = rl_basic_word_break_characters;

	_rl_update_pos();

	/* Just look at how many global variables modify this operation! */
	return fn_complete(e,
	    (rl_compentry_func_t *)rl_completion_entry_function,
	    rl_attempted_completion_function,
	    ct_decode_string(rl_basic_word_break_characters, &wbreak_conv),
	    ct_decode_string(breakchars, &sprefix_conv),
	    _rl_completion_append_character_function,
	    (size_t)rl_completion_query_items,
	    &rl_completion_type, &rl_attempted_completion_over,
	    &rl_point, &rl_end);


}

Where rl_point and rl_end are updated as follows:

static void
_rl_update_pos(void)
{
	const LineInfo *li = el_line(e);

	rl_point = (int)(li->cursor - li->buffer);
	rl_end = (int)(li->lastchar - li->buffer);
}

Which leads me to believe that it does the same thing (http://www.delorie.com/gnu/docs/readline/rlman_28.html tells me that rl_end contains the number of characters in the line).

Even if that is in question, from the readline.c file in php-src, we can see that rl_end (the value passed to the callback as end) and the end key output by readline_info are the same:

https://github.com/php/php-src/blob/master/ext/readline/readline.c#L257

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

However, having dived into it this far, I cannot help but wonder how this works - I had interpreted end differently than what the documentation suggests it does, and I saw it work in that way as well. The variable I actually need is rl_point, which is always in readline_info but not passed as a callback argument.

@krageon
Copy link
Contributor Author

krageon commented Aug 31, 2017

The issue that I was fixing is already fixed, so I'm closing this PR.

@krageon krageon closed this Aug 31, 2017
@bobthecow
Copy link
Owner

I think this is still valuable for people who don't have a PHP or terminal or readline version that supports bracketed paste… IIRC PHP with libedit doesn't?

@krageon
Copy link
Contributor Author

krageon commented Sep 2, 2017 via email

@krageon krageon reopened this Sep 4, 2017
@krageon
Copy link
Contributor Author

krageon commented Sep 12, 2017

There. That should be the tests fixed :)

@@ -62,6 +62,10 @@ public function processCallback($input, $index, $info = array())
$line = $input;
}

if ($this->shouldInsertOnlyTab($line, $info['point'])) {
return array("\t");
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

because of the magic space inserted after tab completion, this inserts five spaces for the first tab stop, then nine spaces for each additional tab stop. why don't we hijack this and return three spaces? then it's always four space indents :)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It inserts a tab character, not spaces. I'm definitely not a fan of replacing it with space characters, as those are different (no matter what you think of tabs vs spaces) and substituting one for the other silently can never be what you want.

The tab + space behaviour is sort of vexing, but it's not something I can realistically solve in a way that doesn't break everything in environments that do work properly (for example ones that have the path from php/php-src#2720 applied).

What I can probably do is return array("\t" . chr(8));. chr(8) is the backspace character, which should work.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I could add in a setting for this and apply it to every autocompletion, thinking about it. Until the patch I mentioned works that's a sort of viable workaround to get autocompletion as you'd expect from an IDE.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wait no, that won't work. The space is appended after the completion - so a backspace that is inserted will actually just remove the tab character entirely.

Copy link
Owner

@bobthecow bobthecow Sep 13, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i understand that tabs and spaces are different. but practically, one looks like you'd expect, and the other looks really bad. so ¯\_(ツ)_/¯

@GrahamCampbell
Copy link
Contributor

What's the status here?

@bobthecow bobthecow closed this Mar 15, 2020
@bobthecow bobthecow reopened this Mar 15, 2020
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants