ReactOS 0.4.16-dev-13-ge2fc578
cmdinput.c File Reference
#include "precomp.h"
Include dependency graph for cmdinput.c:

Go to the source code of this file.

Functions

static VOID ClearCommandLine (LPTSTR str, INT maxlen, SHORT orgx, SHORT orgy)
 
BOOL ReadCommand (LPTSTR str, INT maxlen)
 

Variables

TCHAR AutoCompletionChar = 0x20
 
TCHAR PathCompletionChar = 0x20
 
SHORT maxx
 
SHORT maxy
 
static BOOL bInsert = TRUE
 

Function Documentation

◆ ClearCommandLine()

static VOID ClearCommandLine ( LPTSTR  str,
INT  maxlen,
SHORT  orgx,
SHORT  orgy 
)
static

Definition at line 125 of file cmdinput.c.

126{
127 INT count;
128
129 SetCursorXY (orgx, orgy);
130 for (count = 0; count < (INT)_tcslen (str); count++)
131 ConOutChar (_T(' '));
132 _tcsnset (str, _T('\0'), maxlen);
133 SetCursorXY (orgx, orgy);
134}
VOID SetCursorXY(SHORT x, SHORT y)
Definition: console.c:187
GLuint GLuint GLsizei count
Definition: gl.h:1545
#define _tcsnset
Definition: tchar.h:1419
#define INT
Definition: polytest.cpp:20
VOID ConOutChar(TCHAR c)
Definition: util.c:233
const WCHAR * str
int32_t INT
Definition: typedefs.h:58
#define _T(x)
Definition: vfdio.h:22
#define _tcslen
Definition: xmlstorage.h:198

Referenced by ReadCommand().

◆ ReadCommand()

BOOL ReadCommand ( LPTSTR  str,
INT  maxlen 
)

Definition at line 138 of file cmdinput.c.

139{
141 SHORT orgx; /* origin x/y */
142 SHORT orgy;
143 SHORT curx; /*current x/y cursor position*/
144 SHORT cury;
145 SIZE_T tempscreen;
146 INT count; /*used in some for loops*/
147 INT current = 0; /*the position of the cursor in the string (str)*/
148 INT charcount = 0;/*chars in the string (str)*/
149 INPUT_RECORD ir;
150 DWORD dwControlKeyState;
151#ifdef FEATURE_UNIX_FILENAME_COMPLETION
152 WORD wLastKey = 0;
153#endif
154 TCHAR ch;
155 BOOL bReturn = FALSE;
156 BOOL bCharInput;
157#ifdef FEATURE_4NT_FILENAME_COMPLETION
159#endif
160#ifdef FEATURE_HISTORY
161 //BOOL bContinue=FALSE;/*is TRUE the second case will not be executed*/
162 TCHAR PreviousChar;
163#endif
164
166 {
167 /* No console */
169 DWORD dwRead;
170 CHAR chr;
171 do
172 {
173 if (!ReadFile(hStdin, &chr, 1, &dwRead, NULL) || !dwRead)
174 return FALSE;
175#ifdef _UNICODE
176 MultiByteToWideChar(InputCodePage, 0, &chr, 1, &str[charcount++], 1);
177#endif
178 } while (chr != '\n' && charcount < maxlen);
179 str[charcount] = _T('\0');
180 return TRUE;
181 }
182
183 /* get screen size */
184 maxx = csbi.dwSize.X;
185 maxy = csbi.dwSize.Y;
186
187 curx = orgx = csbi.dwCursorPosition.X;
188 cury = orgy = csbi.dwCursorPosition.Y;
189
190 memset (str, 0, maxlen * sizeof (TCHAR));
191
193
194 do
195 {
196 bReturn = FALSE;
197 ConInKey (&ir);
198
199 dwControlKeyState = ir.Event.KeyEvent.dwControlKeyState;
200
201 if (dwControlKeyState &
204 {
205 switch (ir.Event.KeyEvent.wVirtualKeyCode)
206 {
207#ifdef FEATURE_HISTORY
208 case _T('K'):
209 /* add the current command line to the history */
210 if (dwControlKeyState &
212 {
213 if (str[0])
214 History(0,str);
215
216 ClearCommandLine (str, maxlen, orgx, orgy);
217 current = charcount = 0;
218 curx = orgx;
219 cury = orgy;
220 //bContinue=TRUE;
221 }
222 break;
223
224 case _T('D'):
225 /* delete current history entry */
226 if (dwControlKeyState &
228 {
229 ClearCommandLine (str, maxlen, orgx, orgy);
231 current = charcount = _tcslen (str);
232 ConOutPrintf (_T("%s"), str);
233 GetCursorXY (&curx, &cury);
234 //bContinue=TRUE;
235 }
236 break;
237#endif /*FEATURE_HISTORY*/
238
239 case _T('M'):
240 /* ^M does the same as return */
241 if (dwControlKeyState &
243 {
244 /* end input, return to main */
245#ifdef FEATURE_HISTORY
246 /* add to the history */
247 if (str[0])
248 History(0, str);
249#endif /*FEATURE_HISTORY*/
250 str[charcount++] = _T('\n');
251 str[charcount] = _T('\0');
252 ConOutChar (_T('\n'));
253 bReturn = TRUE;
254 }
255 break;
256
257 case _T('H'): /* ^H does the same as VK_BACK */
258 if (dwControlKeyState & (LEFT_CTRL_PRESSED | RIGHT_CTRL_PRESSED))
259 {
260 bCharInput = FALSE;
261 goto DoBackSpace;
262 }
263 break;
264 }
265 }
266
267 bCharInput = FALSE;
268
269 switch (ir.Event.KeyEvent.wVirtualKeyCode)
270 {
271 case VK_BACK:
272 DoBackSpace:
273 /* <BACKSPACE> - delete character to left of cursor */
274 if (current > 0 && charcount > 0)
275 {
276 if (current == charcount)
277 {
278 /* if at end of line */
279 str[current - 1] = _T('\0');
280 if (GetCursorX () != 0)
281 {
282 ConOutPrintf (_T("\b \b"));
283 curx--;
284 }
285 else
286 {
287 SetCursorXY ((SHORT)(maxx - 1), (SHORT)(GetCursorY () - 1));
288 ConOutChar (_T(' '));
289 SetCursorXY ((SHORT)(maxx - 1), (SHORT)(GetCursorY () - 1));
290 cury--;
291 curx = maxx - 1;
292 }
293 }
294 else
295 {
296 for (count = current - 1; count < charcount; count++)
297 str[count] = str[count + 1];
298 if (GetCursorX () != 0)
299 {
300 SetCursorXY ((SHORT)(GetCursorX () - 1), GetCursorY ());
301 curx--;
302 }
303 else
304 {
305 SetCursorXY ((SHORT)(maxx - 1), (SHORT)(GetCursorY () - 1));
306 cury--;
307 curx = maxx - 1;
308 }
309 GetCursorXY (&curx, &cury);
310 ConOutPrintf (_T("%s "), &str[current - 1]);
311 SetCursorXY (curx, cury);
312 }
313 charcount--;
314 current--;
315 }
316 break;
317
318 case VK_INSERT:
319 /* toggle insert/overstrike mode */
320 bInsert ^= TRUE;
322 break;
323
324 case VK_DELETE:
325 /* delete character under cursor */
326 if (current != charcount && charcount > 0)
327 {
328 for (count = current; count < charcount; count++)
329 str[count] = str[count + 1];
330 charcount--;
331 GetCursorXY (&curx, &cury);
332 ConOutPrintf (_T("%s "), &str[current]);
333 SetCursorXY (curx, cury);
334 }
335 break;
336
337 case VK_HOME:
338 /* goto beginning of string */
339 if (current != 0)
340 {
341 SetCursorXY (orgx, orgy);
342 curx = orgx;
343 cury = orgy;
344 current = 0;
345 }
346 break;
347
348 case VK_END:
349 /* goto end of string */
350 if (current != charcount)
351 {
352 SetCursorXY (orgx, orgy);
353 ConOutPrintf (_T("%s"), str);
354 GetCursorXY (&curx, &cury);
355 current = charcount;
356 }
357 break;
358
359 case VK_TAB:
360#ifdef FEATURE_UNIX_FILENAME_COMPLETION
361 /* expand current file name */
362 if ((current == charcount) ||
363 (current == charcount - 1 &&
364 str[current] == _T('"'))) /* only works at end of line*/
365 {
366 if (wLastKey != VK_TAB)
367 {
368 /* if first TAB, complete filename*/
369 tempscreen = charcount;
370 CompleteFilename (str, charcount);
371 charcount = _tcslen (str);
372 current = charcount;
373
374 SetCursorXY (orgx, orgy);
375 ConOutPrintf (_T("%s"), str);
376
377 if (tempscreen > charcount)
378 {
379 GetCursorXY (&curx, &cury);
380 for (count = tempscreen - charcount; count--; )
381 ConOutChar (_T(' '));
382 SetCursorXY (curx, cury);
383 }
384 else
385 {
386 if (((charcount + orgx) / maxx) + orgy > maxy - 1)
387 orgy += maxy - ((charcount + orgx) / maxx + orgy + 1);
388 }
389
390 /* set cursor position */
391 SetCursorXY ((orgx + current) % maxx,
392 orgy + (orgx + current) / maxx);
393 GetCursorXY (&curx, &cury);
394 }
395 else
396 {
397 /*if second TAB, list matches*/
398 if (ShowCompletionMatches (str, charcount))
399 {
400 PrintPrompt();
401 GetCursorXY(&orgx, &orgy);
402 ConOutPrintf(_T("%s"), str);
403
404 /* set cursor position */
405 SetCursorXY((orgx + current) % maxx,
406 orgy + (orgx + current) / maxx);
407 GetCursorXY(&curx, &cury);
408 }
409
410 }
411 }
412 else
413 {
414 MessageBeep(-1);
415 }
416#endif
417#ifdef FEATURE_4NT_FILENAME_COMPLETION
418 /* used to later see if we went down to the next line */
419 tempscreen = charcount;
420 szPath[0]=_T('\0');
421
422 /* str is the whole things that is on the current line
423 that is and and out. arg 2 is weather it goes back
424 one file or forward one file */
426 /* Attempt to clear the line */
427 ClearCommandLine (str, maxlen, orgx, orgy);
428 curx = orgx;
429 cury = orgy;
430 current = charcount = 0;
431
432 /* Everything is deleted, lets add it back in */
434
435 /* Figure out where cusor is going to be after we print it */
436 charcount = _tcslen(str);
437 current = charcount;
438
439 SetCursorXY(orgx, orgy);
440 /* Print out what we have now */
441 ConOutPrintf(_T("%s"), str);
442
443 /* Move cursor accordingly */
444 if (tempscreen > charcount)
445 {
446 GetCursorXY(&curx, &cury);
447 for(count = tempscreen - charcount; count--; )
448 ConOutChar(_T(' '));
449 SetCursorXY(curx, cury);
450 }
451 else
452 {
453 if (((charcount + orgx) / maxx) + orgy > maxy - 1)
454 orgy += maxy - ((charcount + orgx) / maxx + orgy + 1);
455 }
456 SetCursorXY((short)(((int)orgx + current) % maxx), (short)((int)orgy + ((int)orgx + current) / maxx));
457 GetCursorXY(&curx, &cury);
458#endif
459 break;
460
461 case _T('C'):
462 if ((ir.Event.KeyEvent.dwControlKeyState &
464 {
465 /* Ignore the Ctrl-C key event if it has already been handled */
466 if (!bCtrlBreak)
467 break;
468
469 /*
470 * Fully print the entered string
471 * so the command prompt would not overwrite it.
472 */
473 SetCursorXY(orgx, orgy);
474 ConOutPrintf(_T("%s"), str);
475
476 /*
477 * A Ctrl-C. Do not clear the command line,
478 * but return an empty string in str.
479 */
480 str[0] = _T('\0');
481 curx = orgx;
482 cury = orgy;
483 current = charcount = 0;
484 bReturn = TRUE;
485 }
486 else
487 {
488 /* Just a normal 'C' character */
489 bCharInput = TRUE;
490 }
491 break;
492
493 case VK_RETURN:
494 /* end input, return to main */
495#ifdef FEATURE_HISTORY
496 /* add to the history */
497 if (str[0])
498 History(0, str);
499#endif
500 str[charcount++] = _T('\n');
501 str[charcount] = _T('\0');
502 ConOutChar(_T('\n'));
503 bReturn = TRUE;
504 break;
505
506 case VK_ESCAPE:
507 /* clear str Make this callable! */
508 ClearCommandLine (str, maxlen, orgx, orgy);
509 curx = orgx;
510 cury = orgy;
511 current = charcount = 0;
512 break;
513
514#ifdef FEATURE_HISTORY
515 case VK_F3:
517#endif
518 case VK_UP:
519#ifdef FEATURE_HISTORY
520 /* get previous command from buffer */
521 ClearCommandLine (str, maxlen, orgx, orgy);
522 History(-1, str);
523 current = charcount = _tcslen (str);
524 if (((charcount + orgx) / maxx) + orgy > maxy - 1)
525 orgy += maxy - ((charcount + orgx) / maxx + orgy + 1);
526 ConOutPrintf (_T("%s"), str);
527 GetCursorXY (&curx, &cury);
528#endif
529 break;
530
531 case VK_DOWN:
532#ifdef FEATURE_HISTORY
533 /* get next command from buffer */
534 ClearCommandLine (str, maxlen, orgx, orgy);
535 History(1, str);
536 current = charcount = _tcslen (str);
537 if (((charcount + orgx) / maxx) + orgy > maxy - 1)
538 orgy += maxy - ((charcount + orgx) / maxx + orgy + 1);
539 ConOutPrintf (_T("%s"), str);
540 GetCursorXY (&curx, &cury);
541#endif
542 break;
543
544 case VK_LEFT:
545 if (dwControlKeyState & (RIGHT_CTRL_PRESSED | LEFT_CTRL_PRESSED))
546 {
547 /* move cursor to the previous word */
548 if (current > 0)
549 {
550 while (current > 0 && str[current - 1] == _T(' '))
551 {
552 current--;
553 if (curx == 0)
554 {
555 cury--;
556 curx = maxx -1;
557 }
558 else
559 {
560 curx--;
561 }
562 }
563
564 while (current > 0 && str[current -1] != _T(' '))
565 {
566 current--;
567 if (curx == 0)
568 {
569 cury--;
570 curx = maxx -1;
571 }
572 else
573 {
574 curx--;
575 }
576 }
577
578 SetCursorXY(curx, cury);
579 }
580 }
581 else
582 {
583 /* move cursor left */
584 if (current > 0)
585 {
586 current--;
587 if (GetCursorX () == 0)
588 {
589 SetCursorXY ((SHORT)(maxx - 1), (SHORT)(GetCursorY () - 1));
590 curx = maxx - 1;
591 cury--;
592 }
593 else
594 {
595 SetCursorXY ((SHORT)(GetCursorX () - 1), GetCursorY ());
596 curx--;
597 }
598 }
599 else
600 {
601 MessageBeep (-1);
602 }
603 }
604 break;
605
606 case VK_RIGHT:
607 if (dwControlKeyState & (RIGHT_CTRL_PRESSED | LEFT_CTRL_PRESSED))
608 {
609 /* move cursor to the next word */
610 if (current != charcount)
611 {
612 while (current != charcount && str[current] != _T(' '))
613 {
614 current++;
615 if (curx == maxx - 1)
616 {
617 cury++;
618 curx = 0;
619 }
620 else
621 {
622 curx++;
623 }
624 }
625
626 while (current != charcount && str[current] == _T(' '))
627 {
628 current++;
629 if (curx == maxx - 1)
630 {
631 cury++;
632 curx = 0;
633 }
634 else
635 {
636 curx++;
637 }
638 }
639
640 SetCursorXY(curx, cury);
641 }
642 }
643 else
644 {
645 /* move cursor right */
646 if (current != charcount)
647 {
648 current++;
649 if (GetCursorX () == maxx - 1)
650 {
651 SetCursorXY (0, (SHORT)(GetCursorY () + 1));
652 curx = 0;
653 cury++;
654 }
655 else
656 {
657 SetCursorXY ((SHORT)(GetCursorX () + 1), GetCursorY ());
658 curx++;
659 }
660 }
661#ifdef FEATURE_HISTORY
662 else
663 {
665 if (last && charcount < (INT)_tcslen (last))
666 {
667 PreviousChar = last[current];
668 ConOutChar(PreviousChar);
669 GetCursorXY(&curx, &cury);
670 str[current++] = PreviousChar;
671 charcount++;
672 }
673 }
674#endif
675 }
676 break;
677
678 default:
679 /* This input is just a normal char */
680 bCharInput = TRUE;
681
682 }
683#ifdef _UNICODE
685 if (ch >= 32 && (charcount != (maxlen - 2)) && bCharInput)
686#else
688 if ((UCHAR)ch >= 32 && (charcount != (maxlen - 2)) && bCharInput)
689#endif /* _UNICODE */
690 {
691 /* insert character into string... */
692 if (bInsert && current != charcount)
693 {
694 /* If this character insertion will cause screen scrolling,
695 * adjust the saved origin of the command prompt. */
696 tempscreen = _tcslen(str + current) + curx;
697 if ((tempscreen % maxx) == (maxx - 1) &&
698 (tempscreen / maxx) + cury == (maxy - 1))
699 {
700 orgy--;
701 cury--;
702 }
703
704 for (count = charcount; count > current; count--)
705 str[count] = str[count - 1];
706 str[current++] = ch;
707 if (curx == maxx - 1)
708 curx = 0, cury++;
709 else
710 curx++;
711 ConOutPrintf (_T("%s"), &str[current - 1]);
712 SetCursorXY (curx, cury);
713 charcount++;
714 }
715 else
716 {
717 if (current == charcount)
718 charcount++;
719 str[current++] = ch;
720 if (GetCursorX () == maxx - 1 && GetCursorY () == maxy - 1)
721 orgy--, cury--;
722 if (GetCursorX () == maxx - 1)
723 curx = 0, cury++;
724 else
725 curx++;
726 ConOutChar (ch);
727 }
728 }
729
730 //wLastKey = ir.Event.KeyEvent.wVirtualKeyCode;
731 }
732 while (!bReturn);
733
735
736#ifdef FEATURE_ALIASES
737 /* expand all aliases */
738 ExpandAlias (str, maxlen);
739#endif /* FEATURE_ALIAS */
740 return TRUE;
741}
HANDLE WINAPI GetStdHandle(IN DWORD nStdHandle)
Definition: console.c:203
BOOL WINAPI GetConsoleScreenBufferInfo(IN HANDLE hConsoleOutput, OUT PCONSOLE_SCREEN_BUFFER_INFO lpConsoleScreenBufferInfo)
Definition: console.c:595
VOID ExpandAlias(LPTSTR cmd, INT maxlen)
Definition: alias.c:89
VOID PrintPrompt(VOID)
Definition: prompt.c:134
SHORT GetCursorY(VOID)
Definition: console.c:214
UINT InputCodePage
Definition: console.c:25
SHORT GetCursorX(VOID)
Definition: console.c:206
VOID GetCursorXY(PSHORT x, PSHORT y)
Definition: console.c:196
VOID ConInKey(PINPUT_RECORD lpBuffer)
Definition: console.c:61
VOID SetCursorType(BOOL bInsert, BOOL bVisible)
Definition: console.c:222
#define ConOutPrintf(szStr,...)
Definition: console.h:41
VOID History_move_to_bottom(VOID)
Definition: history.c:280
VOID History(INT dir, LPTSTR commandline)
Definition: history.c:326
VOID History_del_current_entry(LPTSTR str)
Definition: history.c:175
LPCTSTR PeekHistory(INT dir)
Definition: history.c:287
SHORT maxx
Definition: cmdinput.c:115
static BOOL bInsert
Definition: cmdinput.c:121
static VOID ClearCommandLine(LPTSTR str, INT maxlen, SHORT orgx, SHORT orgy)
Definition: cmdinput.c:125
SHORT maxy
Definition: cmdinput.c:116
HANDLE hStdin
Definition: ctm.c:55
#define NULL
Definition: types.h:112
#define TRUE
Definition: types.h:120
#define FALSE
Definition: types.h:117
#define ReadFile(a, b, c, d, e)
Definition: compat.h:742
#define MAX_PATH
Definition: compat.h:34
#define MultiByteToWideChar
Definition: compat.h:110
VOID CompleteFilename(LPTSTR strIN, BOOL bNext, LPTSTR strOut, UINT cusor)
Definition: filecomp.c:545
unsigned int BOOL
Definition: ntddk_ex.h:94
unsigned long DWORD
Definition: ntddk_ex.h:95
unsigned short WORD
Definition: ntddk_ex.h:93
#define _tcscpy
Definition: tchar.h:623
if(dx< 0)
Definition: linetemp.h:194
struct task_struct * current
Definition: linux.c:32
LPCWSTR szPath
Definition: env.c:37
static UINT UINT last
Definition: font.c:45
short SHORT
Definition: pedump.c:59
BOOL bCtrlBreak
Definition: util.c:11
#define memset(x, y, z)
Definition: compat.h:39
union _INPUT_RECORD::@3282 Event
KEY_EVENT_RECORD KeyEvent
Definition: wincon.h:275
DWORD dwControlKeyState
Definition: wincon.h:248
WORD wVirtualKeyCode
Definition: wincon.h:242
union _KEY_EVENT_RECORD::@3281 uChar
WCHAR UnicodeChar
Definition: wincon.h:245
SHORT Y
Definition: blue.h:27
SHORT X
Definition: blue.h:26
ULONG_PTR SIZE_T
Definition: typedefs.h:80
#define STD_OUTPUT_HANDLE
Definition: winbase.h:268
#define STD_INPUT_HANDLE
Definition: winbase.h:267
#define LEFT_CTRL_PRESSED
Definition: wincon.h:140
#define SHIFT_PRESSED
Definition: wincon.h:141
#define RIGHT_CTRL_PRESSED
Definition: wincon.h:139
#define RIGHT_ALT_PRESSED
Definition: wincon.h:137
#define LEFT_ALT_PRESSED
Definition: wincon.h:138
#define VK_TAB
Definition: winuser.h:2202
#define VK_UP
Definition: winuser.h:2228
BOOL WINAPI MessageBeep(_In_ UINT uType)
#define VK_RETURN
Definition: winuser.h:2204
#define VK_END
Definition: winuser.h:2225
#define VK_HOME
Definition: winuser.h:2226
#define VK_BACK
Definition: winuser.h:2201
#define VK_F3
Definition: winuser.h:2260
#define VK_LEFT
Definition: winuser.h:2227
#define VK_RIGHT
Definition: winuser.h:2229
#define VK_DOWN
Definition: winuser.h:2230
#define VK_DELETE
Definition: winuser.h:2236
#define VK_ESCAPE
Definition: winuser.h:2217
#define VK_INSERT
Definition: winuser.h:2235
char TCHAR
Definition: xmlstorage.h:189
unsigned char UCHAR
Definition: xmlstorage.h:181
const CHAR * LPCTSTR
Definition: xmlstorage.h:193
char CHAR
Definition: xmlstorage.h:175

Variable Documentation

◆ AutoCompletionChar

TCHAR AutoCompletionChar = 0x20

Definition at line 111 of file cmdinput.c.

Referenced by Initialize(), and LoadRegistrySettings().

◆ bInsert

◆ maxx

SHORT maxx

Definition at line 115 of file cmdinput.c.

Referenced by CommandScreen(), GetScreenSize(), and ReadCommand().

◆ maxy

◆ PathCompletionChar

TCHAR PathCompletionChar = 0x20

Definition at line 112 of file cmdinput.c.

Referenced by Initialize(), and LoadRegistrySettings().