Skip to content

Commit

Permalink
Add canse insensitive tilde completion
Browse files Browse the repository at this point in the history
darcs-hash:20080113193221-75c98-169804fe128001f73eaee6fab28bfb99dca7c93e.gz
  • Loading branch information
liljencrantz committed Jan 13, 2008
1 parent c2b2806 commit f5540ff
Show file tree
Hide file tree
Showing 3 changed files with 151 additions and 60 deletions.
44 changes: 30 additions & 14 deletions complete.c
Original file line number Diff line number Diff line change
Expand Up @@ -58,7 +58,7 @@ These functions are used for storing and retrieving tab-completion data, as well
/**
Description for ~USER completion
*/
#define COMPLETE_USER_DESC _( L"Home for %s" )
#define COMPLETE_USER_DESC _( L"Home for %ls" )

/**
Description for short variables. The value is concatenated to this description
Expand Down Expand Up @@ -1613,8 +1613,8 @@ static void complete_param_expand( wchar_t *str,
Complete the specified string as an environment variable
*/
static int complete_variable( const wchar_t *whole_var,
int start_offset,
array_list_t *comp_list )
int start_offset,
array_list_t *comp_list )
{
int i;
const wchar_t *var = &whole_var[start_offset];
Expand Down Expand Up @@ -1796,24 +1796,40 @@ static int try_complete_user( const wchar_t *cmd,
{
if( wcsncmp( user_name, pw_name, name_len )==0 )
{
string_buffer_t desc;
string_buffer_t name;

sb_init( &name );
sb_printf( &name,
L"%ls/",
&pw_name[name_len] );
string_buffer_t desc;

sb_init( &desc );
sb_printf( &desc,
COMPLETE_USER_DESC,
pw->pw_gecos );
pw_name );

completion_allocate( comp,
(wchar_t *)name.buff,
(wchar_t *)desc.buff,
0 );
&pw_name[name_len],
(wchar_t *)desc.buff,
COMPLETE_NO_SPACE );

res=1;

sb_destroy( &desc );
}
else if( wcsncasecmp( user_name, pw_name, name_len )==0 )
{
string_buffer_t name;
string_buffer_t desc;

sb_init( &name );
sb_init( &desc );
sb_printf( &name,
L"~%ls",
pw_name );
sb_printf( &desc,
COMPLETE_USER_DESC,
pw_name );

completion_allocate( comp,
(wchar_t *)name.buff,
(wchar_t *)desc.buff,
COMPLETE_NO_CASE | COMPLETE_DONT_ESCAPE | COMPLETE_NO_SPACE );
res=1;

sb_destroy( &desc );
Expand Down
7 changes: 7 additions & 0 deletions complete.h
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,13 @@
*/
#define COMPLETE_AUTO_SPACE 8

/**
This completion should be inserted as-is, without escaping.
*/
#define COMPLETE_DONT_ESCAPE 16



typedef struct
{

Expand Down
160 changes: 114 additions & 46 deletions reader.c
Original file line number Diff line number Diff line change
Expand Up @@ -998,7 +998,10 @@ static void completion_insert( const wchar_t *val, int flags )

wchar_t quote;
int add_space = !(flags & COMPLETE_NO_SPACE);
int do_replace = (flags&COMPLETE_NO_CASE);
int do_replace = (flags & COMPLETE_NO_CASE);
int do_escape = !(flags & COMPLETE_DONT_ESCAPE);

// debug( 0, L"Insert completion %ls with flags %d", val, flags);

if( do_replace )
{
Expand All @@ -1017,11 +1020,18 @@ static void completion_insert( const wchar_t *val, int flags )
sb_init( &sb );
sb_append_substring( &sb, data->buff, begin - data->buff );

escaped = escape( val, ESCAPE_ALL | ESCAPE_NO_QUOTED );

sb_append( &sb, escaped );
free( escaped );
if( do_escape )
{
escaped = escape( val, ESCAPE_ALL | ESCAPE_NO_QUOTED );
sb_append( &sb, escaped );
free( escaped );
}
else
{
sb_append( &sb, val );
}


if( add_space )
{
sb_append( &sb, L" " );
Expand All @@ -1039,52 +1049,60 @@ static void completion_insert( const wchar_t *val, int flags )
else
{

get_param( data->buff,
if( do_escape )
{

get_param( data->buff,
data->buff_pos,
&quote,
0, 0, 0 );

if( quote == L'\0' )
{
replaced = escape( val, ESCAPE_ALL | ESCAPE_NO_QUOTED );
}
else
{
int unescapable=0;

const wchar_t *pin;
wchar_t *pout;
if( quote == L'\0' )
{
replaced = escape( val, ESCAPE_ALL | ESCAPE_NO_QUOTED );
}
else
{
int unescapable=0;

replaced = pout =
malloc( sizeof(wchar_t)*(wcslen(val) + 1) );
const wchar_t *pin;
wchar_t *pout;

replaced = pout =
malloc( sizeof(wchar_t)*(wcslen(val) + 1) );

for( pin=val; *pin; pin++ )
{
switch( *pin )
for( pin=val; *pin; pin++ )
{
case L'\n':
case L'\t':
case L'\b':
case L'\r':
unescapable=1;
break;
default:
*pout++ = *pin;
break;
switch( *pin )
{
case L'\n':
case L'\t':
case L'\b':
case L'\r':
unescapable=1;
break;
default:
*pout++ = *pin;
break;
}
}
if( unescapable )
{
free( replaced );
wchar_t *tmp = escape( val, ESCAPE_ALL | ESCAPE_NO_QUOTED );
replaced = wcsdupcat( L" ", tmp );
free( tmp);
replaced[0]=quote;
}
else
*pout = 0;
}
if( unescapable )
{
free( replaced );
wchar_t *tmp = escape( val, ESCAPE_ALL | ESCAPE_NO_QUOTED );
replaced = wcsdupcat( L" ", tmp );
free( tmp);
replaced[0]=quote;
}
else
*pout = 0;
}

else
{
replaced = wcsdup(val);
}

if( insert_str( replaced ) )
{
/*
Expand Down Expand Up @@ -1292,6 +1310,27 @@ static void reader_flash()

}

#define UNCLEAN L"$*?({})"

int reader_can_replace( const wchar_t *in )
{

const wchar_t * str = in;

CHECK( in, 1 );

/*
Test characters that have a special meaning in any character position
*/
while( *str )
{
if( wcschr( UNCLEAN, *str ) )
return 0;
str++;
}

return 1;
}

/**
Handle the list of completions. This means the following:
Expand Down Expand Up @@ -1327,24 +1366,39 @@ static int handle_completions( array_list_t *comp )
context = halloc( 0, 0 );
tok = halloc_wcsndup( context, begin, end-begin );

/*
Check trivial cases
*/
switch( al_get_count( comp ) )
{
/*
No suitable completions found, flash screen and retur
*/
case 0:
{
reader_flash();
done = 1;
break;
}


/*
Exactly one suitable completion found - insert it
*/
case 1:
{

completion_t *c = (completion_t *)al_get( comp, 0 );

if( !(c->flags & COMPLETE_NO_CASE) || expand_is_clean( tok ) )
/*
If this is a replacement completion, check
that we know how to replace it, e.g. that
the token doesn't contain evil operators
like {}
*/
if( !(c->flags & COMPLETE_NO_CASE) || reader_can_replace( tok ) )
{
completion_insert( c->completion,
c->flags );
c->flags );
}
done = 1;
len = 1;
Expand All @@ -1355,12 +1409,17 @@ static int handle_completions( array_list_t *comp )

if( !done )
{

/*
Try to find something to insert whith the correct case
*/
for( i=0; i<al_get_count( comp ); i++ )
{
completion_t *c = (completion_t *)al_get( comp, i );
int new_len;

/*
Ignore case insensitive completions for now
*/
if( c->flags & COMPLETE_NO_CASE )
continue;

Expand All @@ -1379,6 +1438,9 @@ static int handle_completions( array_list_t *comp )
}
}

/*
If we found something to insert, do it.
*/
if( len > 0 )
{
if( count > 1 )
Expand All @@ -1390,14 +1452,18 @@ static int handle_completions( array_list_t *comp )
}
}



if( !done && base == 0 )
{
/*
Try to find something to insert ignoring case
*/

if( begin )
{

if( expand_is_clean( tok ) )
if( reader_can_replace( tok ) )
{
int offset = wcslen( tok );

Expand Down Expand Up @@ -2602,9 +2668,11 @@ wchar_t *reader_readline()

comp = al_halloc( 0 );
data->complete_func( buffcpy, comp );


sort_completion_list( comp );
remove_duplicates( comp );


free( buffcpy );
comp_empty = handle_completions( comp );
Expand Down

0 comments on commit f5540ff

Please sign in to comment.