P /调用第三方弹出窗口

时间:2016-06-22 07:53:57

标签: c# c++ .net winforms messagebox

现状:
在我目前的项目中,我必须在C#winforms应用程序中使用旧的C ++ DLL来写入Hardlock许可加密狗。

我的应用程序是一个简单的界面,允许用户将许可证信息写入加密狗。如果加密狗已经包含数据,则C ++ dll向用户显示是/否对话框。如果信息已成功写入加密狗,则C ++ dll会显示确认对话框。

当我们的物流部门使用这个工具时,他们会写很多加密狗(一天200到800)。所以他们不想确认消息框。

在搜索解决方案以关闭第三方库的弹出窗口/消息框时,我发现了一些使用平台调用的示例。在这篇文章How to suppress a dialog box displayed by code that I can't change?中使用解决方案失败后,我尝试了以下方法:

每隔500毫秒我会检查弹出窗口是否显示,并用ENTER确认。

private void initButton_Click(object sender, EventArgs e)
{
    messageBoxCloseTimer.Interval = 500;
    messageBoxCloseTimer.Start();
    WriteDongle();
}

private void WriteDongle()
{
    // This code shows a message box        
}

private void MessageBoxCloseTimerElapsed(object sender, System.Timers.ElapsedEventArgs e)
{
    BeginInvoke(new MethodInvoker(delegate
    {
        var hwnd = FindWindow("#32770", "Information");
        if (hwnd.ToInt32() < 1)    // Window not found
            return;

        SendKeys.Send("{ENTER}");
        messageBoxCloseTimer.Stop();
    }));
}

[DllImport("user32.dll", SetLastError = true, CallingConvention = CallingConvention.StdCall)]
    static extern IntPtr FindWindow(string lpClassName, string lpWindowName);

// C++ dll which is used to write the dongle
[DllImport("License.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
    public static extern Int16 peak_BurnDongle(
        Int16 iType,
        [MarshalAs(UnmanagedType.LPStr)] ref string pszSysID,
        int iProgPort,
        Int16 iSigPort);

问题:
有时我得到一个AccessViolationException。 不幸的是,这个错误不是确定性的。在一些加密狗写入之后它发生或不发生。导致此错误的最佳机会是,如果我快速单击按钮(执行上面的代码),例如每2秒钟。

是谁有人看到一个明显的错误?谢谢!

Windows EventLog:

Anwendung: VizLicInitializer.exe
Frameworkversion: v4.0.30319
Beschreibung: Der Prozess wurde aufgrund eines Ausnahmefehlers beendet.
Ausnahmeinformationen: System.AccessViolationException
Stapel:
   bei System.Drawing.SafeNativeMethods+Gdip.GdipCreateBitmapFromStream(IStream, IntPtr ByRef)
   bei System.Drawing.Bitmap..ctor(System.Type, System.String)
   bei System.Windows.Forms.ThreadExceptionDialog..ctor(System.Exception)
   bei System.Windows.Forms.Application+ThreadContext.OnThreadException(System.Exception)
   bei System.Windows.Forms.Application+ThreadContext.PreTranslateMessage(MSG ByRef)
   bei System.Windows.Forms.Application+ThreadContext.System.Windows.Forms.UnsafeNativeMethods.IMsoComponent.FPreTranslateMessage(MSG ByRef)
   bei System.Windows.Forms.Application+ComponentManager.System.Windows.Forms.UnsafeNativeMethods.IMsoComponentManager.FPushMessageLoop(IntPtr, Int32, Int32)
   bei System.Windows.Forms.Application+ThreadContext.RunMessageLoopInner(Int32, System.Windows.Forms.ApplicationContext)
   bei System.Windows.Forms.Application+ThreadContext.RunMessageLoop(Int32, System.Windows.Forms.ApplicationContext)
   bei System.Windows.Forms.Application.Run(System.Windows.Forms.Form)
   bei VizLicInitializer.Program.Main()

修改:
感谢Luc给我发送了以下解决方案(仍然没有解决我的问题,但仍然是比我更好的解决方案):

        internal static void RegisterWindowCreatedEvent()
        {
            var hHook = SetWindowsHookEx(HookType.WH_SHELL, winDelegate, IntPtr.Zero, AppDomain.GetCurrentThreadId());
        }

        private static int ShellHookProcDelegate(int code, IntPtr wParam, IntPtr lParam)
        {
            if (code != HSHELL_WINDOWCREATED)
            {
                return CallNextHookEx(IntPtr.Zero, code, wParam, lParam);
            }

            StringBuilder wCaption = new StringBuilder();
            GetWindowCaption(wParam, wCaption, wCaption.Capacity);
            if (wCaption.ToString() == "Information")
            {
                SendMessage(wParam, 0x0010, IntPtr.Zero, IntPtr.Zero);
            }
            return CallNextHookEx(IntPtr.Zero, code, wParam, lParam);
        }


        private delegate int HookProc(int code, IntPtr wParam, IntPtr lParam);
        private static readonly HookProc winDelegate = ShellHookProcDelegate;
        [DllImport("user32.dll", SetLastError = true)]
        private static extern IntPtr SetWindowsHookEx(HookType hook, HookProc callback, IntPtr hMod, int dwThreadId);

        [DllImport("user32.dll")]
        private static extern int CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, IntPtr lParam);
        const int HSHELL_WINDOWCREATED = 1;

        [DllImport("user32.dll", EntryPoint = "GetWindowText",CharSet = CharSet.Auto)]
        static extern IntPtr GetWindowCaption(IntPtr hwnd,StringBuilder lpString, int maxCount);

        [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
        static extern IntPtr SendMessage(IntPtr hWnd, UInt32 msg, IntPtr wParam, IntPtr lParam);

遵循Hans Passant的方法,我使用Application Verifier来获取以下错误详细信息。

在堆栈跟踪的开头,我看到一些&#34; USER32&#34;和#34; SendMessage&#34;线。我使用SendMessage向MessageBox窗口发送关闭消息。但为什么会这样呢? 我必须说我不知道​​堆栈跟踪的其余部分意味着什么。

<avrf:logfile xmlns:avrf="Application Verifier">
    <avrf:logSession Version="2" PID="4024" TimeStarted="2016-06-22 : 15:38:26">
        <avrf:logEntry Severity="Error" StopCode="0x13" LayerName="Heaps" Time="2016-06-22 : 15:38:54">
            <avrf:message>First chance access violation for current stack trace.</avrf:message>
            <avrf:parameter1>0 - Invalid address causing the exception.</avrf:parameter1>
            <avrf:parameter2>94e5d8f - Code address executing the invalid access.</avrf:parameter2>
            <avrf:parameter3>2ad984 - Exception record.</avrf:parameter3>
            <avrf:parameter4>2ad9a0 - Context record.</avrf:parameter4>
            <avrf:stackTrace>
                <avrf:trace>vrfcore!VerifierDisableVerifier+73c ( @ 0)</avrf:trace>
                <avrf:trace>vfbasics!+6c7462f0 ( @ 0)</avrf:trace>
                <avrf:trace>vfbasics!+6c747982 ( @ 0)</avrf:trace>
                <avrf:trace>vfbasics!+6c74728a ( @ 0)</avrf:trace>
                <avrf:trace>ntdll!RtlReleasePrivilege+114 ( @ 0)</avrf:trace>
                <avrf:trace>ntdll!RtlGetGroupSecurityDescriptor+30c ( @ 0)</avrf:trace>
                <avrf:trace>ntdll!RtlGetGroupSecurityDescriptor+211 ( @ 0)</avrf:trace>
                <avrf:trace>ntdll!KiUserExceptionDispatcher+f ( @ 0)</avrf:trace>
                <avrf:trace>HLPROG!GetListOfParPorts+f86a ( @ 0)</avrf:trace>
                <avrf:trace>verifier!VerifierGetProviderHelper+ce0d ( @ 0)</avrf:trace>
                <avrf:trace>vrfcore!VerifierTlsSetValue+431 ( @ 0)</avrf:trace>
                <avrf:trace>vfbasics!+6c746ba5 ( @ 0)</avrf:trace>
                <avrf:trace>ntdll!wcsncmp+58 ( @ 0)</avrf:trace>
                <avrf:trace>ntdll!EtwEventRegister+135 ( @ 0)</avrf:trace>
                <avrf:trace>ntdll!RtlImageNtHeader+2b1 ( @ 0)</avrf:trace>
                <avrf:trace>ntdll!LdrLoadDll+66 ( @ 0)</avrf:trace>
                <avrf:trace>vfbasics!+6c746fd7 ( @ 0)</avrf:trace>
                <avrf:trace>KERNELBASE!GetDiskFreeSpaceExA+2bd0 ( @ 0)</avrf:trace>
                <avrf:trace>clr!StrongNameFreeBuffer+201d1 ( @ 0)</avrf:trace>
                <avrf:trace>clr!StrongNameFreeBuffer+20234 ( @ 0)</avrf:trace>
                <avrf:trace>clr!PreBindAssemblyEx+1747a ( @ 0)</avrf:trace>
                <avrf:trace>clr!PreBindAssemblyEx+17edb ( @ 0)</avrf:trace>
                <avrf:trace>clr!CreateAssemblyNameObject+2f57 ( @ 0)</avrf:trace>
                <avrf:trace>clr!CreateAssemblyNameObject+2ec2 ( @ 0)</avrf:trace>
                <avrf:trace>clr!CreateAssemblyNameObject+2dcb ( @ 0)</avrf:trace>
                <avrf:trace>clr!CreateAssemblyNameObject+2d64 ( @ 0)</avrf:trace>
                <avrf:trace>clr!DllCanUnloadNowInternal+173 ( @ 0)</avrf:trace>
                <avrf:trace>clr!+5ba12a0c ( @ 0)</avrf:trace>
                <avrf:trace>????????+07F73967</avrf:trace>
                <avrf:trace>????????+07F73645</avrf:trace>
                <avrf:trace>????????+07F73579</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57cb918a ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57cbb71c ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+5827c720 ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+582590cc ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+585b8aae ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+585b7841 ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57cc1d00 ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57cc6bb1 ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57cc6b99 ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57cc6ae0 ( @ 0)</avrf:trace>
                <avrf:trace>????????+0035A093</avrf:trace>
                <avrf:trace>USER32!gapfnScSendMessage+1cf ( @ 0)</avrf:trace>
                <avrf:trace>USER32!gapfnScSendMessage+2cf ( @ 0)</avrf:trace>
                <avrf:trace>USER32!gapfnScSendMessage+908 ( @ 0)</avrf:trace>
                <avrf:trace>USER32!DispatchMessageW+f ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57d1b6bc ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57cd5701 ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57cd5389 ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57cd5202 ( @ 0)</avrf:trace>
                <avrf:trace>System.Windows.Forms.ni!+57cb15e1 ( @ 0)</avrf:trace>
            </avrf:stackTrace>
        </avrf:logEntry>
    </avrf:logSession>
</avrf:logfile>

1 个答案:

答案 0 :(得分:1)

我认为你的代码在关闭弹出窗口之前会尝试写入加密狗。

考虑一下: 1.启动计时器并写入加密狗。 2.计时器触发,但弹出窗口还没有,所以没有任何反应。 3.再次单击该按钮时,弹出窗口显示。 4.再次触发计时器(并弹出关闭)之前,请调用WriteDongle。

尝试在启动计时器之前禁用该按钮,并在关闭弹出窗口时启用它。

或者,您可能尝试挂钩某个窗口创建事件,过滤掉您需要的那个并在那里调用您的close弹出代码(从来没有自己完成,所以这只是一点脑力激荡)。弹出窗口显示时,您仍需要注意禁用写入加密狗。 How can I be notified when a new window is created on Win32?