407 lines
11 KiB
C
407 lines
11 KiB
C
/*++
|
|
|
|
Copyright (c) 1988-1999 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
complete.c
|
|
|
|
Abstract:
|
|
|
|
File/path name completion support
|
|
|
|
--*/
|
|
|
|
#include "cmd.h"
|
|
|
|
//
|
|
// Upon the first completion, pCompleteBuffer is a pointer to an array
|
|
// of matching full path names
|
|
//
|
|
|
|
TCHAR **pCompleteBuffer = NULL;
|
|
|
|
//
|
|
// The count of strings stored in pCompleteBuffer
|
|
//
|
|
|
|
int nBufSize;
|
|
|
|
//
|
|
// The index in pCompleteBuffer of the current match displayed
|
|
//
|
|
|
|
int nCurrentSpec;
|
|
|
|
|
|
//
|
|
// There are two types of completion matching, path and directory. This is the current
|
|
// matching being done.
|
|
//
|
|
|
|
int bCurrentSpecType;
|
|
|
|
//
|
|
// When called for completion, the location of the beginning of the path is found
|
|
// and stored in nCurrentSpecPathStart. This is relative to the buffer passed
|
|
// in to DoComplete
|
|
//
|
|
|
|
int nCurrentSpecPathStart;
|
|
|
|
int CompleteDir( TCHAR *pFileSpecBuffer, int, int );
|
|
|
|
//
|
|
// Characters that CMD uses for its own grammar. To use these in filenames
|
|
// requires quoting
|
|
//
|
|
|
|
TCHAR szSpecialFileCharsToQuote[] = TEXT(" &()[]{}^=;!%'+,`~");
|
|
|
|
void
|
|
FreeCompleteBuffers( VOID )
|
|
{
|
|
int i;
|
|
|
|
if (pCompleteBuffer != NULL) {
|
|
for (i = 0; i < nBufSize; i++ ){
|
|
free( pCompleteBuffer[i] );
|
|
}
|
|
|
|
free( pCompleteBuffer );
|
|
|
|
pCompleteBuffer = NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
DoCompleteInitialize( VOID )
|
|
{
|
|
FreeCompleteBuffers( );
|
|
nBufSize = 0;
|
|
bCurrentSpecType = 0;
|
|
nCurrentSpecPathStart = 0;
|
|
nCurrentSpec = 0;
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
int
|
|
DoComplete(
|
|
TCHAR *buffer,
|
|
int len,
|
|
int maxlen,
|
|
int bForward,
|
|
int bPathCompletion,
|
|
int bTouched)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This is used whenever a path completion character is seen on input. It updates the
|
|
input buffer with the next matching text and returns the updated size.
|
|
|
|
Arguments:
|
|
|
|
buffer - input string that contains the prefix of the path to match at the end.
|
|
|
|
len - length of the input string.
|
|
|
|
maxlen - maximum string length that can be stored in buffer.
|
|
|
|
bForward - TRUE => matching goes forward through the storted match list. Otherwise
|
|
move backwards through the list.
|
|
|
|
bPathCompletion - TRUE => we match ONLY directories and not files+directories.
|
|
|
|
bTouched - TRUE => the user has edited the path. This usually means that we need to
|
|
begin the matching process anew.
|
|
|
|
Return Value:
|
|
|
|
Zero if no matching entries were found, otherwise the length of the updated buffer.
|
|
|
|
--*/
|
|
{
|
|
PTCHAR pFileSpecBuffer;
|
|
int nBufPos;
|
|
int nPathStart;
|
|
int nFileStart;
|
|
int k;
|
|
BOOL bWildSeen;
|
|
|
|
//
|
|
// Allocate the temp buffer. Unfortunately, some of the internal
|
|
// routines will issue a longjmp, so we need to free the buffer in a try/finally
|
|
//
|
|
|
|
pFileSpecBuffer = mkstr( LBUFLEN * sizeof( TCHAR ));
|
|
if (pFileSpecBuffer == NULL) {
|
|
return 0;
|
|
}
|
|
|
|
try {
|
|
|
|
//
|
|
// If the user edited the previous match or if the form of completion (dir vs
|
|
// dir/file) changed, then we must rebuild the matching information
|
|
//
|
|
|
|
if ( bTouched || (bCurrentSpecType != bPathCompletion)) {
|
|
|
|
BOOL InQuotes = FALSE;
|
|
|
|
//
|
|
// The following code was shipped in NT 4 and Windows 2000. It presented
|
|
// a usability problem when changing matching forms in mid-stream. We will
|
|
// now treat a simple change of completion type the same as being touched
|
|
// by the user: rebuild the matching database from where we are presently.
|
|
//
|
|
|
|
// //
|
|
// // if the buffer was left alone but the matching style
|
|
// // was changed, then start the matching process at the
|
|
// // beginning of the path
|
|
// //
|
|
//
|
|
// if (!bTouched && (bCurrentSpecType != bPathCompletion)) {
|
|
// buffer[nCurrentSpecPathStart] = NULLC;
|
|
// len = nCurrentSpecPathStart;
|
|
// }
|
|
|
|
//
|
|
// Determine the beginning of the path and file name. We
|
|
// need to take into account the presence of quotes and the
|
|
// need for CMD to introduce quoting as well
|
|
//
|
|
|
|
nPathStart = 0;
|
|
nFileStart = -1;
|
|
bWildSeen = FALSE;
|
|
for ( k = 0; k < len; k++ ) {
|
|
if (buffer[k] == SWITCHAR) {
|
|
nPathStart = k + 1;
|
|
bWildSeen = FALSE;
|
|
|
|
} else if ( buffer[k] == QUOTE ) {
|
|
if ( !InQuotes )
|
|
nPathStart = k;
|
|
|
|
InQuotes = !InQuotes;
|
|
} else if ( !InQuotes &&
|
|
_tcschr(szSpecialFileCharsToQuote, buffer[k]) != NULL
|
|
) {
|
|
nPathStart = k+1;
|
|
bWildSeen = FALSE;
|
|
} else if (buffer[k] == COLON ||
|
|
buffer[k] == BSLASH
|
|
) {
|
|
nFileStart = k+1;
|
|
bWildSeen = FALSE;
|
|
} else if (buffer[k] == STAR || buffer[k] == QMARK) {
|
|
bWildSeen = TRUE;
|
|
}
|
|
}
|
|
|
|
if (nFileStart == -1 || nFileStart < nPathStart)
|
|
nFileStart = nPathStart;
|
|
|
|
_tcsncpy( pFileSpecBuffer, &(buffer[nPathStart]), len-nPathStart );
|
|
if (!bWildSeen) {
|
|
pFileSpecBuffer[len-nPathStart+0] = TEXT('*');
|
|
pFileSpecBuffer[len-nPathStart+1] = TEXT('\0');
|
|
} else {
|
|
pFileSpecBuffer[len-nPathStart+0] = TEXT('\0');
|
|
}
|
|
|
|
// do the DIR into a buffer
|
|
nBufSize = CompleteDir( pFileSpecBuffer, bPathCompletion, nFileStart - nPathStart );
|
|
|
|
// reset the current completion string
|
|
nCurrentSpec = nBufSize;
|
|
nCurrentSpecPathStart = nPathStart;
|
|
bCurrentSpecType = bPathCompletion;
|
|
}
|
|
|
|
// if no matches, do nothing.
|
|
if ( nBufSize == 0 ) {
|
|
leave;;
|
|
}
|
|
|
|
// find our postion in the completion buffer.
|
|
if ( bForward ) {
|
|
nCurrentSpec++;
|
|
if ( nCurrentSpec >= nBufSize )
|
|
nCurrentSpec = 0;
|
|
} else {
|
|
nCurrentSpec--;
|
|
if ( nCurrentSpec < 0 )
|
|
nCurrentSpec = nBufSize - 1;
|
|
|
|
}
|
|
|
|
// Return nothing if buffer not big enough
|
|
if ((int)(nCurrentSpecPathStart+_tcslen(pCompleteBuffer[nCurrentSpec])) >= maxlen) {
|
|
nBufSize = 0;
|
|
leave;
|
|
}
|
|
|
|
// copy the completion path onto the end of the command line
|
|
_tcscpy(&buffer[nCurrentSpecPathStart], pCompleteBuffer[nCurrentSpec] );
|
|
} finally {
|
|
FreeStr( pFileSpecBuffer );
|
|
}
|
|
|
|
return nBufSize;
|
|
}
|
|
|
|
|
|
int
|
|
CompleteDir(
|
|
TCHAR *pFileSpecBuffer,
|
|
int bPathCompletion,
|
|
int nFileStart
|
|
)
|
|
{
|
|
PFS pfsCur;
|
|
PSCREEN pscr;
|
|
DRP drpCur = {0, 0, 0, 0,
|
|
{{0,0}, {0,0}, {0,0}, {0,0}, {0,0}, {0,0}},
|
|
0, 0, NULL, 0, 0, 0, 0} ;
|
|
int hits = 0;
|
|
int i, j, nFileLen;
|
|
unsigned Err;
|
|
TCHAR *s, *d, *pszFileStart;
|
|
BOOLEAN bNeedQuotes;
|
|
ULONG rgfAttribs, rgfAttribsOnOff;
|
|
|
|
FreeCompleteBuffers( );
|
|
|
|
// fake up a screen to print into
|
|
pscr = (PSCREEN)gmkstr(sizeof(SCREEN));
|
|
pscr->ccol = 2048;
|
|
|
|
rgfAttribs = 0;
|
|
rgfAttribsOnOff = 0;
|
|
if (bPathCompletion) {
|
|
rgfAttribs = FILE_ATTRIBUTE_DIRECTORY;
|
|
rgfAttribsOnOff = FILE_ATTRIBUTE_DIRECTORY;
|
|
}
|
|
|
|
ParseDirParms(pFileSpecBuffer, &drpCur);
|
|
if ( (drpCur.patdscFirst.pszPattern == NULL) ||
|
|
(SetFsSetSaveDir(drpCur.patdscFirst.pszPattern) == (PCPYINFO) FAILURE) ||
|
|
(BuildFSFromPatterns(&drpCur, FALSE, TRUE, &pfsCur) == FAILURE) ) {
|
|
RestoreSavedDirectory( );
|
|
return( 0 );
|
|
}
|
|
|
|
Err =
|
|
ExpandAndApplyToFS( pfsCur,
|
|
pscr,
|
|
rgfAttribs,
|
|
rgfAttribsOnOff,
|
|
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL );
|
|
|
|
if (Err) {
|
|
RestoreSavedDirectory( );
|
|
return 0;
|
|
}
|
|
|
|
//
|
|
// Make sure there is something to sort, then sort
|
|
//
|
|
if (pfsCur->cff) {
|
|
qsort( pfsCur->prgpff,
|
|
pfsCur->cff,
|
|
sizeof(PTCHAR),
|
|
CmpName
|
|
);
|
|
}
|
|
|
|
s = pFileSpecBuffer;
|
|
d = s;
|
|
bNeedQuotes = FALSE;
|
|
while (*s) {
|
|
if (*s == QUOTE) {
|
|
bNeedQuotes = TRUE;
|
|
s += 1;
|
|
if (nFileStart >= (s-pFileSpecBuffer))
|
|
nFileStart -= 1;
|
|
if (*s == QUOTE)
|
|
*d++ = *s++;
|
|
}
|
|
else {
|
|
if (_tcschr(szSpecialFileCharsToQuote, *s) != NULL)
|
|
bNeedQuotes = TRUE;
|
|
|
|
*d++ = *s++;
|
|
}
|
|
}
|
|
*d = NULLC;
|
|
|
|
hits = pfsCur->cff;
|
|
pCompleteBuffer = calloc( sizeof(TCHAR *), hits );
|
|
if (pCompleteBuffer == NULL) {
|
|
RestoreSavedDirectory( );
|
|
return 0;
|
|
}
|
|
|
|
for(i=0, j=0; i<hits; i++) {
|
|
if (!_tcscmp((TCHAR *)(pfsCur->prgpff[i]->data.cFileName), TEXT(".") )
|
|
|| !_tcscmp((TCHAR *)(pfsCur->prgpff[i]->data.cFileName), TEXT("..") )) {
|
|
continue;
|
|
}
|
|
nFileLen = _tcslen( (TCHAR *)(pfsCur->prgpff[i]->data.cFileName) );
|
|
pCompleteBuffer[j] = (TCHAR *)calloc( (nFileStart + nFileLen + 4) , sizeof( TCHAR ));
|
|
|
|
if (pCompleteBuffer[j] == NULL) {
|
|
continue;
|
|
}
|
|
|
|
if (!bNeedQuotes) {
|
|
s = (TCHAR *)(pfsCur->prgpff[i]->data.cFileName);
|
|
while (*s) {
|
|
if (_tcschr(szSpecialFileCharsToQuote, *s) != NULL)
|
|
bNeedQuotes = TRUE;
|
|
s += 1;
|
|
}
|
|
}
|
|
else
|
|
s = NULL;
|
|
|
|
d = pCompleteBuffer[j];
|
|
if (bNeedQuotes)
|
|
*d++ = QUOTE;
|
|
_tcsncpy( d, pFileSpecBuffer, nFileStart );
|
|
d += nFileStart;
|
|
_tcsncpy( d, (TCHAR *)(pfsCur->prgpff[i]->data.cFileName), nFileLen );
|
|
d += nFileLen;
|
|
|
|
if (bNeedQuotes) {
|
|
*d++ = QUOTE;
|
|
if (s)
|
|
bNeedQuotes = FALSE;
|
|
}
|
|
*d++ = NULLC;
|
|
|
|
j++;
|
|
}
|
|
|
|
hits = j;
|
|
|
|
FreeStr((PTCHAR)(pfsCur->pff));
|
|
FreeStr(pfsCur->pszDir);
|
|
FreeStr((PTCHAR)pfsCur);
|
|
|
|
RestoreSavedDirectory( );
|
|
|
|
return hits;
|
|
}
|