1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS Console Server DLL 4 * FILE: win32ss/user/winsrv/consrv/frontends/input.c 5 * PURPOSE: Common Front-Ends Input functions 6 * PROGRAMMERS: Jeffrey Morlan 7 * Hermes Belusca-Maito (hermes.belusca@sfr.fr) 8 */ 9 10 /* INCLUDES *******************************************************************/ 11 12 #include "consrv.h" 13 #include "include/term.h" 14 #include "coninput.h" 15 16 #define NDEBUG 17 #include <debug.h> 18 19 20 /* PRIVATE FUNCTIONS **********************************************************/ 21 22 static DWORD 23 ConioGetShiftState(PBYTE KeyState, LPARAM lParam) 24 { 25 DWORD ssOut = 0; 26 27 if (KeyState[VK_CAPITAL] & 0x01) 28 ssOut |= CAPSLOCK_ON; 29 30 if (KeyState[VK_NUMLOCK] & 0x01) 31 ssOut |= NUMLOCK_ON; 32 33 if (KeyState[VK_SCROLL] & 0x01) 34 ssOut |= SCROLLLOCK_ON; 35 36 if (KeyState[VK_SHIFT] & 0x80) 37 // || (KeyState[VK_LSHIFT] & 0x80) || (KeyState[VK_RSHIFT] & 0x80) 38 ssOut |= SHIFT_PRESSED; 39 40 if (KeyState[VK_LCONTROL] & 0x80) 41 ssOut |= LEFT_CTRL_PRESSED; 42 if (KeyState[VK_RCONTROL] & 0x80) 43 ssOut |= RIGHT_CTRL_PRESSED; 44 // if (KeyState[VK_CONTROL] & 0x80) { ... } 45 46 if (KeyState[VK_LMENU] & 0x80) 47 ssOut |= LEFT_ALT_PRESSED; 48 if (KeyState[VK_RMENU] & 0x80) 49 ssOut |= RIGHT_ALT_PRESSED; 50 // if (KeyState[VK_MENU] & 0x80) { ... } 51 52 /* See WM_CHAR MSDN documentation for instance */ 53 if (HIWORD(lParam) & KF_EXTENDED) 54 ssOut |= ENHANCED_KEY; 55 56 return ssOut; 57 } 58 59 VOID NTAPI 60 ConioProcessKey(PCONSRV_CONSOLE Console, MSG* msg) 61 { 62 static BYTE KeyState[256] = { 0 }; 63 /* MSDN mentions that you should use the last virtual key code received 64 * when putting a virtual key identity to a WM_CHAR message since multiple 65 * or translated keys may be involved. */ 66 static UINT LastVirtualKey = 0; 67 DWORD ShiftState; 68 WCHAR UnicodeChar; 69 UINT VirtualKeyCode; 70 UINT VirtualScanCode; 71 BOOL Down; 72 BOOLEAN Fake; // Synthesized, not a real event 73 BOOLEAN NotChar; // Message should not be used to return a character 74 75 INPUT_RECORD er; 76 77 if (Console == NULL) 78 { 79 DPRINT1("No Active Console!\n"); 80 return; 81 } 82 83 VirtualScanCode = HIWORD(msg->lParam) & 0xFF; 84 Down = msg->message == WM_KEYDOWN || msg->message == WM_CHAR || 85 msg->message == WM_SYSKEYDOWN || msg->message == WM_SYSCHAR; 86 87 GetKeyboardState(KeyState); 88 ShiftState = ConioGetShiftState(KeyState, msg->lParam); 89 90 if (msg->message == WM_CHAR || msg->message == WM_SYSCHAR) 91 { 92 VirtualKeyCode = LastVirtualKey; 93 UnicodeChar = msg->wParam; 94 } 95 else 96 { 97 WCHAR Chars[2]; 98 INT RetChars = 0; 99 100 VirtualKeyCode = msg->wParam; 101 RetChars = ToUnicodeEx(VirtualKeyCode, 102 VirtualScanCode, 103 KeyState, 104 Chars, 105 2, 106 0, 107 NULL); 108 UnicodeChar = (RetChars == 1 ? Chars[0] : 0); 109 } 110 111 Fake = UnicodeChar && 112 (msg->message != WM_CHAR && msg->message != WM_SYSCHAR && 113 msg->message != WM_KEYUP && msg->message != WM_SYSKEYUP); 114 NotChar = (msg->message != WM_CHAR && msg->message != WM_SYSCHAR); 115 if (NotChar) LastVirtualKey = msg->wParam; 116 117 DPRINT("CONSRV: %s %s %s %s %02x %02x '%lc' %04x\n", 118 Down ? "down" : "up ", 119 (msg->message == WM_CHAR || msg->message == WM_SYSCHAR) ? 120 "char" : "key ", 121 Fake ? "fake" : "real", 122 NotChar ? "notc" : "char", 123 VirtualScanCode, 124 VirtualKeyCode, 125 (UnicodeChar >= L' ') ? UnicodeChar : L'.', 126 ShiftState); 127 128 if (Fake) return; 129 130 // 131 // FIXME: Scrolling via keyboard shortcuts must be done differently, 132 // without touching the internal VirtualY variable. 133 // 134 #if 0 135 if ( (ShiftState & (RIGHT_ALT_PRESSED | LEFT_ALT_PRESSED)) != 0 && 136 (VirtualKeyCode == VK_UP || VirtualKeyCode == VK_DOWN) ) 137 { 138 if (!Down) return; 139 140 /* Scroll up or down */ 141 if (VirtualKeyCode == VK_UP) 142 { 143 /* Only scroll up if there is room to scroll up into */ 144 if (Console->ActiveBuffer->CursorPosition.Y != Console->ActiveBuffer->ScreenBufferSize.Y - 1) 145 { 146 Console->ActiveBuffer->VirtualY = (Console->ActiveBuffer->VirtualY + 147 Console->ActiveBuffer->ScreenBufferSize.Y - 1) % 148 Console->ActiveBuffer->ScreenBufferSize.Y; 149 Console->ActiveBuffer->CursorPosition.Y++; 150 } 151 } 152 else 153 { 154 /* Only scroll down if there is room to scroll down into */ 155 if (Console->ActiveBuffer->CursorPosition.Y != 0) 156 { 157 Console->ActiveBuffer->VirtualY = (Console->ActiveBuffer->VirtualY + 1) % 158 Console->ActiveBuffer->ScreenBufferSize.Y; 159 Console->ActiveBuffer->CursorPosition.Y--; 160 } 161 } 162 163 ConioDrawConsole(Console); 164 return; 165 } 166 #endif 167 168 /* Send the key press to the console driver */ 169 170 er.EventType = KEY_EVENT; 171 er.Event.KeyEvent.bKeyDown = Down; 172 er.Event.KeyEvent.wRepeatCount = 1; 173 er.Event.KeyEvent.wVirtualKeyCode = VirtualKeyCode; 174 er.Event.KeyEvent.wVirtualScanCode = VirtualScanCode; 175 er.Event.KeyEvent.uChar.UnicodeChar = UnicodeChar; 176 er.Event.KeyEvent.dwControlKeyState = ShiftState; 177 178 ConioProcessInputEvent(Console, &er); 179 } 180 181 DWORD 182 ConioEffectiveCursorSize(PCONSRV_CONSOLE Console, DWORD Scale) 183 { 184 DWORD Size = (Console->ActiveBuffer->CursorInfo.dwSize * Scale + 99) / 100; 185 /* If line input in progress, perhaps adjust for insert toggle */ 186 if (Console->LineBuffer && !Console->LineComplete && (Console->InsertMode ? !Console->LineInsertToggle : Console->LineInsertToggle)) 187 return (Size * 2 <= Scale) ? (Size * 2) : (Size / 2); 188 return Size; 189 } 190 191 /* EOF */ 192