Отправка сообщения CTRL-S в окно - PullRequest
3 голосов
/ 28 февраля 2011

Я хочу сохранить окно TextPad, используя код C #; Я могу узнать дескриптор окна, но не уверен, как отправить CTRL - S в это окно. Я хочу использовать P / Invoke API для достижения этой цели. Кроме того, это окно TextPad будет неактивным, потому что мое приложение будет активным в это время.

[DllImport("user32.dll")]
private static extern int SendMessage(IntPtr hwnd, int msg, int wParam, int lParam);

Отправить Ctrl + Вверх до окна

Я посмотрел на это обсуждение, которое очень похоже на мою проблему. Я логически понимаю, что я должен сделать 4 отправки сообщения, как показано ниже

  • KeyDown CTRL key
  • Ключ KeyDown S
  • Ключ KeyUp S
  • KeyUp CTRL key

Я не уверен, как отправить правильные параметры в SendMessage. Чтобы закрыть окно я использую

SendMessage(hWnd, 0x0010, 0, 0);

Я получил это из библиотеки MSDN.

Не могли бы вы указать мне какую-нибудь ссылку, которая говорит мне шестнадцатеричное значение для клавиш на клавиатуре и объясняет, что представляют собой последние два параметра?

ОБНОВЛЕНИЕ - 1
Используя spy ++, я нахожу эти события сгенерированными, нажимая CTRL-S в окне Блокнота

1. WM_KEYDOWN nVirtKey:VK_Control, 
2. WM_KEYDOWN nVirtKey:'S' .. some other messates ..
3. WM_KEYUP nVirtKey:VK_Control. 
4. WM_KEYUP nVirtKey:'S'. 

ОБНОВЛЕНИЕ - 2


       private IntPtr startnotepad() {
            ProcessStartInfo psi = new ProcessStartInfo();
            psi.FileName = @"notepad.exe";
            String fileName = baseDirectory + textbox1.Text;
            psi.Arguments = @"""" + fileName + @"""";
            psi.WindowStyle = ProcessWindowStyle.Minimized;
            Process p = Process.Start(psi); 
            return p.MainWindowHandle;
        }

        private void SaveNotepad(IntPtr handle)
        {
            IntPtr handle = GetWindow(handle, 5); // get the keyboard focus handle
            // verified the handle with Spy ++.
            SendMessage(handle, 0x0100, 0x11, 0); // keydown ctrl
            SendMessage(handle, 0x0100, 0x53, 0); // keydown S
            //SendMessage(handle, 0x0101, 0x11, 0); --- I tried keeping "Keyup Ctrl" here as well.
            SendMessage(handle, 0x0101, 0x53, 0); // keyup s
            SendMessage(handle, 0x0101, 0x11, 0); // keyup ctrl            
        }

Этот код не работает (сохранить его часть), хотя я могу видеть, что WM_KEYDOWN - CTRL .. WM_KEYDOWN - 'S' .. KEYUP 'S' и KEYUP CTRL отправляются в правое окно в spy ++. Кто-нибудь может прокомментировать, что не так с этим кодом? Или любые предложения, если я делаю что-то действительно глупое.

ОБНОВЛЕНИЕ 3

Мне следует использовать PostMessage вместо SendMessage, как @Hans предложил в своих комментариях.


 [DllImport("user32.dll")]
 public static extern IntPtr PostMessage(IntPtr hWnd, uint Msg, IntPtr wParam, IntPtr lParam);  

        private void Save(IntPtr mainWindowHandle)
        {               
            IntPtr handle = GetWindow(mainWindowHandle, 5); // getChild window
            IntPtr CTRL_KEY = new IntPtr(0x11);
            uint KEY_DOWN = 0x0100;
            uint KEY_UP = 0x0101;
            IntPtr S_KEY = new IntPtr(0x53);

            //SetForegroundWindow(p.MainWindowHandle);
            PostMessage(handle, KEY_DOWN, CTRL_KEY, IntPtr.Zero);
            PostMessage(handle, KEY_DOWN, S_KEY, IntPtr.Zero);
            PostMessage(handle, KEY_UP, S_KEY, IntPtr.Zero);
            PostMessage(handle, KEY_UP, CTRL_KEY, IntPtr.Zero);                      
        } 

Приведенный выше код вместо CTRL + S отправляет CTRL и S в окно Блокнота. Итак, я вижу две буквы "S" в блокноте. Глядя на реальное сообщение, генерируемое, когда мы нажимаем CTRL + S, я вижу, что последний параметр моего Postmessage неверен. Это должно быть что-то вроде (для клавиши KEY UP CTRL "C01F0001")

Подскажите, пожалуйста, как передать этот шестнадцатеричный код в качестве последнего параметра сообщения?


// does not work
PostMessage(handle, KEY_UP, CTRL_KEY, new IntPtr(0xC01F0001);

Обновление 4: я думаю, что мы не можем отправлять сообщения «горячих клавиш» в свернутое окно. используя сообщение
Этот код ниже работает с SendInput: но при этом появится всплывающее окно, которое не соответствует цели. В любом случае, просто хотел обновить эту тему.


 private void Save_Notepad(IntPtr mainWindowHandle) {
            //SetActiveWindow(mainWindowHandle);
            //SetFocus(GetWindow(mainWindowHandle, 5));
            ShowWindow(mainWindowHandle, 1); // show window 
            SetForegroundWindow(mainWindowHandle);
            //SetActiveWindow(mainWindowHandle);
            //SetFocus(GetWindow(mainWindowHandle, 5));
            //IntPtr returnvalue = SetFocus(mainWindowHandle);
            uint intReturn;
            INPUT structInput;
            structInput = new INPUT();
            structInput.type = 1;// keyboard input
            // key down
            structInput.ki.wScan = 0x1D;
            structInput.ki.time = 0;
            structInput.ki.dwFlags = 0;
            // key down Ctrl
            structInput.ki.wVk = 0x11;  //0x1D; //
            intReturn = SendInput(1, ref structInput, Marshal.SizeOf(new INPUT()));
            // key down S
            structInput.ki.wScan = 0x1F;
            structInput.ki.wVk = 0x53; //0x41;//0x53;  //0x1F;//
            intReturn = SendInput(1, ref structInput, Marshal.SizeOf(new INPUT()));
            // key up 
            structInput.ki.dwFlags = 0x0002; // key up
            // key up S
            intReturn = SendInput(1, ref structInput, Marshal.SizeOf(typeof(INPUT)));
            // key up CTRL
            structInput.ki.wVk = 0x11;  //0x1D; //
            intReturn = SendInput(1, ref structInput, Marshal.SizeOf(new INPUT()));
            //ShowWindow(mainWindowHandle, 2); // minimize it again 
        }

После CTRL + S я могу свернуть окно, но тогда оно создаст вспышку. Кто-нибудь может подсказать, смогу ли я добиться этой функциональности без "FLASH" (хотя бы)?
ОБНОВЛЕНИЕ 5: использование WM_SYSCOMMAND


IntPtr handle = GetWindow(mainWindowHandle, 5);
// send Alt F message to Notepad.
// http://msdn.microsoft.com/en-us/library/ms646360(v=VS.85).aspx
// I can see this is working.
PostMessage(handle, 0x0112, 0xF100, 'f');  // send WM_SYSCOMMAND
// how to send s on this window ??
// I have tried things which I have learned so far. 

Мне просто нужно отправить S-сообщения в файловое меню, которое появится в результате WM_SYSCOMMAND. Может кто-нибудь указать мне документацию о том, как это сделать?


PostMessage(handle, KEY_DOWN, S_KEY, 0x1F0001); // POST does not work
// tried with mainwindowhandle as well
PostMessage(handle, 0x111, 'S', 0); // WM_COMMAND does not work
PostMessage(handle, 0x0112, 0xF100, 's');  // WM_SYSCOMMAND does not work (and does not make sence either)

Окончательное обновление

Мне не удалось отправить сообщение ^ S в свернутое окно. Я использую sendkeys (^ s) каждые 2-3 секунды, когда основное внимание уделяется редактору.

- Karephul

Ответы [ 2 ]

5 голосов
/ 28 февраля 2011

Вы не можете использовать SendMessage (или PostMessage, правильный) для имитации состояния клавиш-модификаторов, например CTRL.Вы должны использовать SendInput ().

Нажатие клавиши, например Ctrl + S, нетипично переводится в сообщение WM_COMMAND.Используйте инструмент Spy ++, чтобы увидеть, что происходит, когда вы нажимаете Ctrl + S вручную.Если вы видите WM_COMMAND, то вы золотой, вы можете использовать SendMessage () для отправки этого сообщения.Это, конечно, будет работать только для определенного процесса.

3 голосов
/ 18 февраля 2013

Мне нужно было сделать именно то, что вам нужно, вызвать Ctrl + S в некотором фоновом окне, после примерно одного дня исследований, кажется, есть много способов достичь этого.

Например, чтобы нажать Alt-F, введите 'S', которую вы можете вызвать:

PostMessage(handle, 0x0112, 0xF100, 0x0046);
PostMessage(handle, 0x0102, 0x0053, 0);

Часть Alt-F работает на фоновых окнах, в то время как последующая буква S не работает.

Еще один способ сделать это с помощью WM_COMMAND и WM_MENUSELECT, MF_MOUSESELECT:

IntPtr menu = GetMenu(handle);
IntPtr subMenu = GetSubMenu(menu, 0);//0 = first menu item
uint menuItemID = GetMenuItemID(subMenu, 2);//2 = second item in submenu
SendMessage(handle, 0x0111, 0x20000000 + menuItemID, menu);

Наконец, несколько иронично, самое простое решение - вызвать WM_COMMAND с помощью menuItemID:

PostMessage(handle, 0x0111, 0x0003, 0x0);

0x0003 (сохранить в блокноте) определяется приложением и его структурой меню. Вы можете получить этот код, прослушивая WM_COMMAND в spy ++ в окне, в то время как вы используете комбинацию клавиш или нажимаете кнопку мыши в меню. команда.

Кажется, также возможно использовать WM_SYSKEYUP / DOWN и WM_MENUCOMMAND, также такие клавиши, как Ctrl-C, Ctrl-V имеют постоянные определенные значения и могут передаваться как один символ, но только в приложения, которые слушают эти жестко закодированные символы. .

Спасибо за отправную точку.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...