multitasking - Stop programs from stealing focus in Windows 7

25
2014-05
  • cirkef

    This question already has an answer here:

  • Answers
  • Moses

    Judging by at least the research I've done, it seems there used to be a registry key in XP that could be modified to accomplish this, but since Windows 7, that registry change has been ineffective (there is a very heated and relevant conversation on the Microsoft forums about this.)

    I even found an interesting blog article our very own Jeff Atwood wrote about this in 2007, though his example was in XP.

    The Microsoft moderators on the forums seem to suggest that by default in Windows 7, properly written applications shouldn't steal focus (they argue that the registry tweak from XP was built in), but at the end of the day, the community believes they are simply wrong and some applications just do.

    Discussions here on Superuser related to XP have ended up, as time has passed, having questions in comments being raised to how solutions could be appled to Windows 7. HarryMC posted a great answer that sheds some light there.

    At the end of the day, it would appear that there is simply no way to accomplish this universally across Windows 7, as it seems to do more with the applications themselves. After a little testing, I discovered that some of my applications do call for focus and get it, while some do not.

    Alternative Solution

    One alternative that I use every day is Always On Top (alternative link). It binds the Ctrl + Space hotkey to make whatever window is in focus always on top.

    I use it when working in almost any program that I know I don't want interupted, as it will not allow any window over it. However, this may be more annoying to some people. I just make it a habit to hit the hotkey after I'm done with a particular program, or minimize it.

    Alternative to the Alternative (My Opinion)

    A shell replacement such as those HarryMC suggested in his answer from above are a viable solution.

    Though in my mind, if you're willing to completely replace Aero, or completely replace Explorer, or willing to completely replace the Windows shell to get this functionality (and I don't blame you), you could just as easily replace Windows.

    That's what I did the beginning of this year. Learning ubuntu has been interesting, but it was for the better.


  • Related Question

    windows 7 - Preventing applications from stealing focus
  • svandragt

    Are there any solutions to prevent applications stealing focus from the active window?

    This is especially annoying when I'm starting an application, switch to do something else and the new application starts receiving half a sentence of text.


  • Related Answers
  • Simon P Stevens

    There is an option in TweakUI which does this. It prevents most of the usual tricks dubious software developers employ to force focus on their app.

    It's an ongoing arms war though, so I don't know if it works for everything.

    Update: According to EndangeredMassa, TweakUI does not work on Windows 7.

  • Oliver Salzburg

    A while ago I've done extensive research on solving this issue once and for all (and failed). The result of my research can be found on the annoyance project page.

    The project also includes an application that repeatedly tries to grab focus by calling:

    switch( message ) {
      case WM_TIMER:
        if( hWnd != NULL ) {
          // Start off easy
          // SetForegroundWindow will not move the window to the foreground,
          // but it will invoke FlashWindow internally and, thus, show the
          // taskbar.
          SetForegroundWindow( hWnd );
    
          // Our application is awesome! It must have your focus!
          SetActiveWindow( hWnd );
    
          // Flash that button!
          FlashWindow( hWnd, TRUE );
        }
        break;
    

    As we can see from this snippet, my research was also focused on other aspects of user interface behavior I don't like.

    The way I tried to solve this was to load a DLL into every new process and hook the API calls that cause another windows to be activated.
    The last part is the easy one, thanks to awesome API hooking libraries out there. I used the very great mhook library:

    #include "stdafx.h"
    #include "mhook-2.2/mhook-lib/mhook.h"
    
    typedef NTSTATUS( WINAPI* PNT_QUERY_SYSTEM_INFORMATION ) ( 
      __in       SYSTEM_INFORMATION_CLASS SystemInformationClass,     
      __inout    PVOID SystemInformation, 
      __in       ULONG SystemInformationLength, 
      __out_opt  PULONG ReturnLength    
    );
    
    // Originals
    PNT_QUERY_SYSTEM_INFORMATION OriginalFlashWindow   = 
      (PNT_QUERY_SYSTEM_INFORMATION)::GetProcAddress( 
      ::GetModuleHandle( L"user32" ), "FlashWindow" );
    
    PNT_QUERY_SYSTEM_INFORMATION OriginalFlashWindowEx = 
      (PNT_QUERY_SYSTEM_INFORMATION)::GetProcAddress( 
      ::GetModuleHandle( L"user32" ), "FlashWindowEx" );
    
    PNT_QUERY_SYSTEM_INFORMATION OriginalSetForegroundWindow = 
      (PNT_QUERY_SYSTEM_INFORMATION)::GetProcAddress( 
      ::GetModuleHandle( L"user32" ), "SetForegroundWindow" );
    
    // Hooks
    BOOL WINAPI
    HookedFlashWindow(
      __in  HWND hWnd,
      __in  BOOL bInvert
      ) {
      return 0;
    }
    
    BOOL WINAPI 
    HookedFlashWindowEx(
      __in  PFLASHWINFO pfwi
      ) {
      return 0;
    }
    
    BOOL WINAPI 
    HookedSetForegroundWindow(
      __in  HWND hWnd
      ) {
      // Pretend window was brought to foreground
      return 1;
    }
    
    
    BOOL APIENTRY 
    DllMain( 
      HMODULE hModule,
      DWORD   ul_reason_for_call,
      LPVOID  lpReserved
      ) {
      switch( ul_reason_for_call ) {
        case DLL_PROCESS_ATTACH:
          Mhook_SetHook( (PVOID*)&OriginalFlashWindow,         HookedFlashWindow );
          Mhook_SetHook( (PVOID*)&OriginalFlashWindowEx,       HookedFlashWindowEx );
          Mhook_SetHook( (PVOID*)&OriginalSetForegroundWindow, HookedSetForegroundWindow );
          break;
    
        case DLL_PROCESS_DETACH:
          Mhook_Unhook( (PVOID*)&OriginalFlashWindow );
          Mhook_Unhook( (PVOID*)&OriginalFlashWindowEx );
          Mhook_Unhook( (PVOID*)&OriginalSetForegroundWindow );
          break;
      }
      return TRUE;
    }
    

    From my tests back then, this worked great. Except for the part of loading the DLL into every new process. As one might imagine, that's nothing to take too lightly. I used the AppInit_DLLs approach back then (which is simply not sufficient).

    Basically, this works great. But I never found the time to write something that properly injects my DLL into new processes. And the time invested in this largely overshadows the annoyance the focus stealing causes me.

    In addition to the DLL injection problem, there is also a focus stealing method which I didn't cover in the implementation on Google Code. A co-worker actually did some additional research and covered that method. The problem was discussed on SO: http://stackoverflow.com/questions/7430864/windows-7-prevent-application-from-losing-focus

  • Tom Wijsman

    In Windows 7, the ForegroundLockTimeout registry entry is no longer checked, you can verify this with Process Monitor. In fact, in Windows 7 they disallow you from changing the foreground window. Go and read about its details, it has even been there since Windows 2000.

    However, the documentation sucks and they chase each other and find ways around that.

    So, there is something buggy going on with SetForegroundWindow, or similar API functions...

    The only way to really do this properly is to make a small application which periodically calls LockSetForegroundWindow, virtually disabling any calls to our buggy API function.

    If that's not enough (another buggy API call?) you can go even further and do some API monitoring to see what's going on, and then you simply hook the API calls on every process after which you can get rid of any calls that mess up the foreground. However, ironically, this is discouraged by Microsoft...

  • harrymc

    I believe that some confusion may exist, as there are two ways of "stealing focus" : (1) a window coming to the foreground, and (2) the window receiving keystrokes.

    The problem referred to here is probably the second one, where a windows claims the focus by bringing itself to the foreground - without the user's request or permission.

    The discussion must split here between XP and 7.

    Windows XP

    In XP there is a registry hack that makes XP work the same as Windows 7 in preventing applications from stealing focus :

    1. Use regedit to go to: HKEY_CURRENT_USER\Control Panel\Desktop.
    2. Double-click on ForegroundLockTimeout and set its value in hexadecimal to 30d40.
    3. Press OK and exit regedit.
    4. Reboot your PC for the changes to take effect.

    Windows 7

    (The discussion below mostly applies to XP as well.)

    Please understand that there is no way in which Windows can totally block applications from stealing the focus and remain functional. For example, if during a file-copy your anti-virus detected a possible threat and would like to pop-up a window asking you for the action to take, if this window is blocked then you would never understand why the copy never terminates.

    In Windows 7 there is only one modification possible to the behavior of Windows itself, which is to use the MS-Windows focus-follows-mouse Registry hacks, where the focus and/or activation goes always to the windows under the cursor. A delay can be added to avoid applications popping up all over the desktop.
    See this article : Windows 7 - Mouse Hover Makes Window Active - Enable.

    Otherwise, one must detect and neutralize the guilty program : If this is always the same application that is getting the focus, then this application is programmed to take the focus and preventing this may be done by either disabling it from starting with the computer, or use some setting supplied by that application to avoid this behavior.

    You could use the VBS script included in VB Code which identifies who's stealing focus, which the author used to identify the culprit as a "call home" updater for a printer software.

    A desperate measure when all else fails, and if you have identified this badly-programmed application, is to minimize it and hope that will not then bring itself to the front. A stronger form of minimization is to the tray by using one of the free products listed in Best Free Application Minimizer.

    Last idea in the order of desperation is to fracture your desktop virtually by using a product such as Desktops or Dexpot, and do your work in another desktop than the default.

  • Ivo Flipse

    Ghacks has a possible solution:

    It happens several times a day that some applications steal the focus of the active window by popping up. This can happen for a number of reasons, when I extract files or a transfer finishes for instance. It does not matter most of the time when this happens but sometimes I’m writing an article and it does not only mean that I have to type some words again but also that I lost concentration and have to click to regain focus.

    The Pro Reviewer website has a tip on how to prevent this from happening. The easiest way of preventing focus stealing is to use Tweak UI which has a setting that is called “Prevent applications from stealing focus”. Checking this option prevents that other applications pop up suddenly and steal the focus of the window you are currently working in.

    This only works when the application has been minimized before. Instead of stealing the focus it will flash a number of times which can be defined in the same menu in Tweak UI. If you do not want to use Tweak UI you can change the setting in the Windows Registry.

    Navigate to the Registry key HKEY_CURRENT_USER > Control Panel > Desktop and change the ForegroundLockTimeout value to 30d40 (Hexadecimal) or 200000 (Decimal). The key ForeGroundFlashCount defines the amount of flashes of a window to alert the user where 0 means unlimited.

  • avirk

    PowerMenu will do waht you want to do in windows 7. I have Windows-7 32 bit so don't know it will work on 64 bit properly or not. But on my version its working like a charm for me. After installing this just right click on the top of window and select the option you want to.

    enter image description here

    And select the Always on Top option and t will do the rest.

    EDIT

    OK I got it!

    You can use these two more programs which are easy to use. Always on Top will work good just you have to launch it after installing it and then just drag the hand button to the desired app to make it on top always and you can toggle it through the Ctrl+F8 and lose focus from active windows you can use Ctrl+Spacebar.

    enter image description here

    Second tool is the Deskpin just install it and then click on the pin icon in the taskbar tray icon and then click it on the windows top which you want to keep on top.