private void processLinuxInput(TermKeyKey key)
        {
            // If any special button has been pressed (Tab, Enter, etc)
            // we should convert its code to INPUT_RECORD.KeyEvent
            // Because INPUT_RECORD.KeyEvent depends on Windows' scan codes,
            // we convert codes retrieved from LibTermKey to Windows virtual scan codes
            // In the future, this logic may be changed (for example, both Windows and Linux
            // raw codes can be converted into ConsoleFramework's own abstract enum)
            if (key.type == TermKeyType.TERMKEY_TYPE_KEYSYM)
            {
                INPUT_RECORD inputRecord = new INPUT_RECORD();
                inputRecord.EventType             = EventType.KEY_EVENT;
                inputRecord.KeyEvent.bKeyDown     = true;
                inputRecord.KeyEvent.wRepeatCount = 1;
                switch (key.code.sym)
                {
                case TermKeySym.TERMKEY_SYM_TAB:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Tab;
                    break;

                case TermKeySym.TERMKEY_SYM_ENTER:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Return;
                    break;

                // in gnome-terminal it is backspace by default
                // (see default compatibility settings in Profile's settings)
                case TermKeySym.TERMKEY_SYM_DEL:
                case TermKeySym.TERMKEY_SYM_BACKSPACE:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Back;
                    break;

                case TermKeySym.TERMKEY_SYM_DELETE:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Delete;
                    break;

                case TermKeySym.TERMKEY_SYM_HOME:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Home;
                    break;

                case TermKeySym.TERMKEY_SYM_END:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.End;
                    break;

                case TermKeySym.TERMKEY_SYM_PAGEUP:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Prior;
                    break;

                case TermKeySym.TERMKEY_SYM_PAGEDOWN:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Next;
                    break;

                case TermKeySym.TERMKEY_SYM_SPACE:
                    inputRecord.KeyEvent.UnicodeChar     = ' ';
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Space;
                    break;

                case TermKeySym.TERMKEY_SYM_ESCAPE:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Escape;
                    break;

                case TermKeySym.TERMKEY_SYM_INSERT:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Insert;
                    break;

                case TermKeySym.TERMKEY_SYM_UP:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Up;
                    break;

                case TermKeySym.TERMKEY_SYM_DOWN:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Down;
                    break;

                case TermKeySym.TERMKEY_SYM_LEFT:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Left;
                    break;

                case TermKeySym.TERMKEY_SYM_RIGHT:
                    inputRecord.KeyEvent.wVirtualKeyCode = VirtualKeys.Right;
                    break;

                default:
                    throw new NotSupportedException("Not supported keyboard code detected: " + key.code.sym);
                }
                inputRecord.KeyEvent.dwControlKeyState = 0;
                if ((key.modifiers & 4) == 4)
                {
                    inputRecord.KeyEvent.dwControlKeyState |= ControlKeyState.LEFT_CTRL_PRESSED;
                }
                if ((key.modifiers & 2) == 2)
                {
                    inputRecord.KeyEvent.dwControlKeyState |= ControlKeyState.LEFT_ALT_PRESSED;
                }
                processInputEvent(inputRecord);
            }
            else if (key.type == TermKeyType.TERMKEY_TYPE_UNICODE)
            {
                byte[] data = new byte[7];
                data[0] = key.utf8_0;
                data[1] = key.utf8_1;
                data[2] = key.utf8_2;
                data[3] = key.utf8_3;
                data[4] = key.utf8_4;
                data[5] = key.utf8_5;
                data[6] = key.utf8_6;
                string       d = System.Text.Encoding.UTF8.GetString(data);
                char         unicodeCharacter = d[0];
                INPUT_RECORD inputRecord      = new INPUT_RECORD();
                inputRecord.EventType                  = EventType.KEY_EVENT;
                inputRecord.KeyEvent.bKeyDown          = true;
                inputRecord.KeyEvent.wRepeatCount      = 1;
                inputRecord.KeyEvent.UnicodeChar       = unicodeCharacter;
                inputRecord.KeyEvent.dwControlKeyState = 0;
                if (char.IsLetterOrDigit(unicodeCharacter))
                {
                    if (char.IsDigit(unicodeCharacter))
                    {
                        inputRecord.KeyEvent.wVirtualKeyCode =
                            (VirtualKeys)(unicodeCharacter - '0' + (int)VirtualKeys.N0);
                    }
                    else
                    {
                        char lowercased = char.ToLowerInvariant(unicodeCharacter);

                        // Only english characters can be converted to VirtualKeys
                        if (lowercased >= 'a' && lowercased <= 'z')
                        {
                            inputRecord.KeyEvent.wVirtualKeyCode =
                                (VirtualKeys)(lowercased - 'a' + (int)VirtualKeys.A);
                        }
                    }
                }
                if ((key.modifiers & 4) == 4)
                {
                    inputRecord.KeyEvent.dwControlKeyState |= ControlKeyState.LEFT_CTRL_PRESSED;
                }
                if ((key.modifiers & 2) == 2)
                {
                    inputRecord.KeyEvent.dwControlKeyState |= ControlKeyState.LEFT_ALT_PRESSED;
                }
                // todo : remove hardcoded exit combo after testing
                if (unicodeCharacter == 'd' && key.modifiers == 4)
                {
                    Exit();
                }
                processInputEvent(inputRecord);
                //
            }
            else if (key.type == TermKeyType.TERMKEY_TYPE_MOUSE)
            {
                TermKeyMouseEvent ev;
                int button;
                int line, col;
                LibTermKey.termkey_interpret_mouse(termkeyHandle, ref key, out ev, out button, out line, out col);
                //
                INPUT_RECORD inputRecord = new INPUT_RECORD();
                inputRecord.EventType = EventType.MOUSE_EVENT;
                if (ev == TermKeyMouseEvent.TERMKEY_MOUSE_PRESS || ev == TermKeyMouseEvent.TERMKEY_MOUSE_RELEASE)
                {
                    inputRecord.MouseEvent.dwEventFlags = MouseEventFlags.PRESSED_OR_RELEASED;
                }
                if (ev == TermKeyMouseEvent.TERMKEY_MOUSE_DRAG)
                {
                    inputRecord.MouseEvent.dwEventFlags = MouseEventFlags.MOUSE_MOVED;
                }
                inputRecord.MouseEvent.dwMousePosition = new COORD((short)(col - 1), (short)(line - 1));
                if (ev == TermKeyMouseEvent.TERMKEY_MOUSE_RELEASE)
                {
                    inputRecord.MouseEvent.dwButtonState = 0;
                }
                else if (ev == TermKeyMouseEvent.TERMKEY_MOUSE_DRAG || ev == TermKeyMouseEvent.TERMKEY_MOUSE_PRESS)
                {
                    if (1 == button)
                    {
                        inputRecord.MouseEvent.dwButtonState = MOUSE_BUTTON_STATE.FROM_LEFT_1ST_BUTTON_PRESSED;
                    }
                    else if (2 == button)
                    {
                        inputRecord.MouseEvent.dwButtonState = MOUSE_BUTTON_STATE.FROM_LEFT_2ND_BUTTON_PRESSED;
                    }
                    else if (3 == button)
                    {
                        inputRecord.MouseEvent.dwButtonState = MOUSE_BUTTON_STATE.RIGHTMOST_BUTTON_PRESSED;
                    }
                }
                //
                processInputEvent(inputRecord);
            }
        }
Пример #2
0
        public static void Main(string[] args)
        {
            Thread thread = new Thread(new ThreadStart(() => {
                Thread.Sleep(TimeSpan.FromSeconds(5));
                Console.WriteLine("Message from thread");
                int res = LibTermKey.writeInt64(eventfd, 1);
                Console.WriteLine("write(1) returned {0}\n", res);
                if (res == -1)
                {
                    int lastError = System.Runtime.InteropServices.Marshal.GetLastWin32Error();
                    Console.WriteLine("Last error is {0}\n", lastError);
                }
            }));

            thread.IsBackground = true;
            thread.Start();

            IntPtr handle = LibTermKey.termkey_new(0, TermKeyFlag.TERMKEY_FLAG_SPACESYMBOL);

            Console.Write("\x1B[?1002h");
            pollfd fd = new pollfd();

            fd.fd     = 0;
            fd.events = POLL_EVENTS.POLLIN;

            pollfd[] fds = new pollfd[2];
            fds[0] = fd;

            fds[1]  = new pollfd();
            eventfd = LibTermKey.eventfd(0, EVENTFD_FLAGS.EFD_CLOEXEC);
            if (eventfd == -1)
            {
                Console.WriteLine("Cannot create eventfd\n");
                int lastError = System.Runtime.InteropServices.Marshal.GetLastWin32Error();
                Console.WriteLine("Last error is {0}\n", lastError);
            }
            fds[1].fd     = eventfd;
            fds[1].events = POLL_EVENTS.POLLIN;

            TermKeyKey key = new TermKeyKey();

            while (true)
            {
                int pollRes = LibTermKey.poll(fds, 2, -1);
                if (0 == pollRes)
                {
                    // timed out
                    Console.WriteLine("Timed out");
                    if (LibTermKey.termkey_getkey_force(handle, ref key) == TermKeyResult.TERMKEY_RES_KEY)
                    {
                        Console.WriteLine("got TERMKEY_RES_KEY");
                    }
                }
                else if (-1 == pollRes)
                {
                    int errorCode = System.Runtime.InteropServices.Marshal.GetLastWin32Error();
                    Console.WriteLine(string.Format("ErrorCode = {0}", errorCode));
                }
                Console.WriteLine(string.Format("PollRes is {0}", pollRes));

                for (int i = 0; i < 2; i++)
                {
                    if (fds[i].revents != POLL_EVENTS.NONE)
                    {
                        if (i == 1)
                        {
                            UInt64 u;
                            LibTermKey.readInt64(fds[i].fd, out u);
                            Console.WriteLine("Readed eventfd counter : {0}\n", u);
                        }
                    }
                }

                if ((fds[0].revents & POLL_EVENTS.POLLIN) == POLL_EVENTS.POLLIN ||
                    (fds[0].revents & POLL_EVENTS.POLLHUP) == POLL_EVENTS.POLLHUP ||
                    (fds[0].revents & POLL_EVENTS.POLLERR) == POLL_EVENTS.POLLERR)
                {
                    // todo : log return value
                    LibTermKey.termkey_advisereadable(handle);
                }

                TermKeyResult result;
                while ((result = LibTermKey.termkey_getkey(handle, ref key)) == TermKeyResult.TERMKEY_RES_KEY)
                {
                    Console.WriteLine("Received some key.");
                    string descr = String.Format("Type : {0} Modifiers: {1} Utf8 bytes: {2}{3}{4}{5}{6}{7}{8}",
                                                 key.type, key.modifiers, key.utf8_0,
                                                 key.utf8_1, key.utf8_2, key.utf8_3,
                                                 key.utf8_4, key.utf8_5, key.utf8_6);
                    //dump the retrieved structure
                    //byte[] buffer = new byte[30];
                    //IntPtr nativeBuffer = System.Runtime.InteropServices.Marshal.AllocHGlobal(30);
                    //System.Runtime.InteropServices.Marshal.StructureToPtr(key, nativeBuffer, false);
                    //System.Runtime.InteropServices.Marshal.Copy(nativeBuffer, buffer, 0, 30);
                    //for (int i = 0; i < 30; i++ ) {
                    //	Console.Write("{0} ", buffer[i]);
                    //	if ((i + 1) % 10 == 0)
                    //		Console.WriteLine();
                    //}
                    //System.Runtime.InteropServices.Marshal.FreeHGlobal(nativeBuffer);
                    Console.WriteLine(descr);
                    if (key.type == TermKeyType.TERMKEY_TYPE_UNICODE)
                    {
                        byte[] data = new byte[7];
                        data[0] = key.utf8_0;
                        data[1] = key.utf8_1;
                        data[2] = key.utf8_2;
                        data[3] = key.utf8_3;
                        data[4] = key.utf8_4;
                        data[5] = key.utf8_5;
                        data[6] = key.utf8_6;
                        string d = System.Text.Encoding.UTF8.GetString(data);
                        Console.WriteLine(String.Format("Unicode symbol : {0}", d));
                    }
                    else if (key.type == TermKeyType.TERMKEY_TYPE_MOUSE)
                    {
                        TermKeyMouseEvent ev;
                        int button;
                        int line, col;
                        LibTermKey.termkey_interpret_mouse(handle, ref key, out ev, out button, out line, out col);
                        Console.WriteLine("MouseEvent : {0} (button {1}) at {2}:{3}", ev, button, line, col);
                    }
                }
            }

            LibTermKey.termkey_destroy(handle);
            LibTermKey.close(eventfd);
        }