90% времени я не могу запуститься osk.exe
от процесса на 32 бита на Win7 x64
. Первоначально код просто использовал:
Process.Launch("osk.exe");
Который не будет работать над x64 из-за виртуализации каталога. Не проблема я думал, я просто отключу виртуализацию, запущу приложение и включу ее снова, который я думал, был корректный способ сделать вещи. Я также добавил некоторый код для возвращения клавиатуры, если это было минимизировано (который хорошо работает) - код (в демонстрационном приложении WPF) теперь смотрит следующим образом:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;using System.Diagnostics;
using System.Runtime.InteropServices;
namespace KeyboardTest
{
/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
[DllImport("kernel32.dll", SetLastError = true)]
private static extern bool Wow64DisableWow64FsRedirection(ref IntPtr ptr);
[DllImport("kernel32.dll", SetLastError = true)]
public static extern bool Wow64RevertWow64FsRedirection(IntPtr ptr);
private const UInt32 WM_SYSCOMMAND = 0x112;
private const UInt32 SC_RESTORE = 0xf120;
[DllImport("user32.dll", CharSet = CharSet.Auto)]
static extern IntPtr SendMessage(IntPtr hWnd, UInt32 Msg, IntPtr wParam, IntPtr lParam);
private string OnScreenKeyboadApplication = "osk.exe";
public MainWindow()
{
InitializeComponent();
}
private void KeyboardButton_Click(object sender, RoutedEventArgs e)
{
// Get the name of the On screen keyboard
string processName = System.IO.Path.GetFileNameWithoutExtension(OnScreenKeyboadApplication);
// Check whether the application is not running
var query = from process in Process.GetProcesses()
where process.ProcessName == processName
select process;
var keyboardProcess = query.FirstOrDefault();
// launch it if it doesn't exist
if (keyboardProcess == null)
{
IntPtr ptr = new IntPtr(); ;
bool sucessfullyDisabledWow64Redirect = false;
// Disable x64 directory virtualization if we're on x64,
// otherwise keyboard launch will fail.
if (System.Environment.Is64BitOperatingSystem)
{
sucessfullyDisabledWow64Redirect = Wow64DisableWow64FsRedirection(ref ptr);
}
// osk.exe is in windows/system folder. So we can directky call it without path
using (Process osk = new Process())
{
osk.StartInfo.FileName = OnScreenKeyboadApplication;
osk.Start();
osk.WaitForInputIdle(2000);
}
// Re-enable directory virtualisation if it was disabled.
if (System.Environment.Is64BitOperatingSystem)
if (sucessfullyDisabledWow64Redirect)
Wow64RevertWow64FsRedirection(ptr);
}
else
{
// Bring keyboard to the front if it's already running
var windowHandle = keyboardProcess.MainWindowHandle;
SendMessage(windowHandle, WM_SYSCOMMAND, new IntPtr(SC_RESTORE), new IntPtr(0));
}
}
}
}
Но этот код, большую часть времени, выдает следующее исключение на osk.Start()
:
Указанная процедура не могла быть найдена в System. Диагностика. Процесс. StartWithShellExecuteEx (ProcessStartInfo startInfo)
Я попытался поместить длинную резьбу. Сон управляет приблизительно в osk. Запустите строку, только чтобы удостовериться, что это не было состояние состязания, но та же проблема сохраняется. Кто-либо может определить, где я делаю что-то не так или предоставляю альтернативное решение для этого? Это, кажется, хорошо работает, запуская Блокнот, это просто не будет сотрудничать с экранной клавиатурой.
У меня нет убедительного объяснения того, какое именно сообщение об ошибке вы получаете. Но отключение перенаправления приведет к нарушению работы .NET framework. По умолчанию Process.Start() P/Invokes the ShellExecuteEx() API function to start the process. Эта функция находится в shell32.dll, библиотеке DLL, которую, возможно, придется загрузить, если это не было сделано ранее. При отключении перенаправления вы получите неправильный вариант.
Обходным решением для этого является установка ProcessStartInfo.UseShellExecute в false. Здесь это не нужно.
Очевидно, что отключение перенаправления - это рискованный подход с побочными эффектами, которые невозможно предсказать. Существует множество библиотек DLL, которые загружаются по требованию. Очень маленький вспомогательный EXE, который вы компилируете с Platform Target = Any CPU, может решить вашу проблему.
Под капотом происходят некоторые вещи, которые требуют запуска osk.exe из потока MTA. Причина, похоже, в том, что вызов Wow64DisableWow64FsRedirection
влияет только на текущий поток. Однако при определенных условиях Process.Start
будет создавать новый процесс из отдельного потока, например, когда UseShellExecute
имеет значение false, а также при вызове из потока STA, как кажется.
Код ниже проверяет состояние апартаментов и затем убеждается в запуске экранной клавиатуры из потока MTA:
using System;
using System.Diagnostics;
using System.IO;
using System.Runtime.InteropServices;
using System.Threading;
class Program
{
[DllImport("kernel32.dll", SetLastError = true)]
private static extern bool Wow64DisableWow64FsRedirection(ref IntPtr ptr);
[DllImport("kernel32.dll", SetLastError = true)]
public static extern bool Wow64RevertWow64FsRedirection(IntPtr ptr);
[DllImport("user32.dll", CharSet = CharSet.Auto)]
static extern IntPtr SendMessage(IntPtr hWnd,
UInt32 Msg,
IntPtr wParam,
IntPtr lParam);
private const UInt32 WM_SYSCOMMAND = 0x112;
private const UInt32 SC_RESTORE = 0xf120;
private const string OnScreenKeyboardExe = "osk.exe";
[STAThread]
static void Main(string[] args)
{
Process[] p = Process.GetProcessesByName(
Path.GetFileNameWithoutExtension(OnScreenKeyboardExe));
if (p.Length == 0)
{
// we must start osk from an MTA thread
if (Thread.CurrentThread.GetApartmentState() == ApartmentState.STA)
{
ThreadStart start = new ThreadStart(StartOsk);
Thread thread = new Thread(start);
thread.SetApartmentState(ApartmentState.MTA);
thread.Start();
thread.Join();
}
else
{
StartOsk();
}
}
else
{
// there might be a race condition if the process terminated
// meanwhile -> proper exception handling should be added
//
SendMessage(p[0].MainWindowHandle,
WM_SYSCOMMAND, new IntPtr(SC_RESTORE), new IntPtr(0));
}
}
static void StartOsk()
{
IntPtr ptr = new IntPtr(); ;
bool sucessfullyDisabledWow64Redirect = false;
// Disable x64 directory virtualization if we're on x64,
// otherwise keyboard launch will fail.
if (System.Environment.Is64BitOperatingSystem)
{
sucessfullyDisabledWow64Redirect =
Wow64DisableWow64FsRedirection(ref ptr);
}
ProcessStartInfo psi = new ProcessStartInfo();
psi.FileName = OnScreenKeyboardExe;
// We must use ShellExecute to start osk from the current thread
// with psi.UseShellExecute = false the CreateProcessWithLogon API
// would be used which handles process creation on a separate thread
// where the above call to Wow64DisableWow64FsRedirection would not
// have any effect.
//
psi.UseShellExecute = true;
Process.Start(psi);
// Re-enable directory virtualisation if it was disabled.
if (System.Environment.Is64BitOperatingSystem)
if (sucessfullyDisabledWow64Redirect)
Wow64RevertWow64FsRedirection(ptr);
}
}