Click here to Skip to main content
15,881,803 members
Articles / Programming Languages / C#

Getting Caret Position Inside Any Application

Rate me:
Please Sign up or sign in to vote.
4.90/5 (27 votes)
30 Mar 2009CPOL3 min read 131.8K   5.4K   38   51
Retrieves Caret position from any application and converts it to the Screen Coordinates

Introduction 

Sometimes we need to know about the caret position inside another application for various purposes like showing Tooltip or message window along with caret. (Here the other application is an entirely different process and not connected in any way with our application.) This article describes how to show a tooltip window next to the caret in any application. The source code, attached here, is fully illustrative and easy to understand.

Background

Though there are many ways to get the caret position inside an application, the problem is that those methods are associated with the client region and non client regions and provide/use wrong handles. This association leads to wrong information about caret position inside Microsoft Office 2007 like products where the entire window results as the client area. Facing the same client and non client region problem, I spent some time looking for the possible solution and arrived at this approach. The utility is ready to use. You can enhance the UI or add some more information.

Using the Code 

To test the caret postion, just run the application and follow the process below:

  1. Open a Notepad and type some text. It will show tooltip next to caret. While typing, Tooltip moves along with caret.
  2. Now move the Notepad window and observe the continuously changing caret position inside tooltip.
  3. When Notepad window reaches the edge of screen, Tooltip changes its position (By default, it gets displayed at the bottom right to the caret).
  4. Now click on Desktop, Tooltip disappears.
  5. Now open Microsoft Word and see the Tooltip re-appear. Tooltip can be moved by pressing the left mouse button and moving the cursor onto it.

Explanation

GetCaretPosition() method populates GUI Thread information into guiInfo object. guiInfo is a structure variable of type GUIThreadInfo that is required by GetGUIThreadInfo() method of user32.dll

C#
  public void GetCaretPosition()
  {
       guiInfo = new GUITHREADINFO();
       guiInfo.cbSize = (uint)Marshal.SizeOf(guiInfo);

       // Get GuiThreadInfo into guiInfo
       GetGUIThreadInfo(0, out guiInfo);
  }   

guiInfo variable is a global variable that is further processed by EvaluateCaretPosition() method. This method fetches caret information and handle of associated window from guiInfo object and converts it to the screen coordinates.

C#
private void EvaluateCaretPosition()
 {
      caretPosition = new Point();

      // Fetch GUITHREADINFO
      GetCaretPosition();

      caretPosition.X = (int)guiInfo.rcCaret.Left + 25;
      caretPosition.Y = (int)guiInfo.rcCaret.Bottom + 25;

      ClientToScreen(guiInfo.hwndCaret, out caretPosition);

      txtCaretX.Text = (caretPosition.X).ToString();
      txtCaretY.Text = caretPosition.Y.ToString();
 }

Further the tooltip should be visible only for GUI applications and not for Windows Explorer likewise renaming any file on desktop. For this current active process is evaluated by GetActiveProcess() method. This method returns the name of currently active process to the timer event.

C#
private string GetActiveProcess()
 {
       const int nChars = 256;
       int handle = 0;
       StringBuilder Buff = new StringBuilder(nChars);
       handle = (int)GetForegroundWindow();

       // If Active window has some title info
       if (GetWindowText(handle, Buff, nChars) > 0)
        {
              uint lpdwProcessId;
              uint dwCaretID = GetWindowThreadProcessId(handle, out lpdwProcessId);
              uint dwCurrentID = (uint)Thread.CurrentThread.ManagedThreadId;
              return Process.GetProcessById((int)lpdwProcessId).ProcessName;
        }

        // Otherwise either error or non client region
        return String.Empty;
 }

In the timer event, just check whether the tooltip is foreground window (just because user might have clicked on that), if not then check whether Windows Explorer is an active process(just because of user click on desktop/ opening Wiindows Explorer), if so then hide tooltip else evaluate caret position and adjust tooltip position by AdjustUI() method.

C#
private void timer1_Tick(object sender, EventArgs e)
         {
             // If Tooltip window is active window (Suppose user clicks on the
             //  Tooltip Window)
             if (GetForegroundWindow() == this.Handle)
             {
                 // then do no processing
                 return;
             }

             // Get Current active Process
             string activeProcess = GetActiveProcess();

             // If window explorer is active window (eg. user has opened any drive)
             // Or for any failure when activeProcess is nothing
             if ((activeProcess.ToLower().Contains("explorer") |
          (activeProcess == string.Empty)))
             {
                 // Dissappear Tooltip
                 this.Visible = false;
             }
             else
             {
                 // Otherwise Calculate Caret position
                 EvaluateCaretPosition();

                 // Adjust ToolTip according to the Caret
                 AdjustUI();

                 // Display current active Process on Tooltip
                 lblCurrentApp.Text = " You are Currently inside : " + activeProcess;
                 this.Visible = true;
             }
         }

The AdjustUI method is responsible to adjust UI of tooltip to keep it inside user screen.

C#
private void AdjustUI()
          {
              // Get Current Screen Resolution
              Rectangle workingArea = SystemInformation.WorkingArea;

              // If current caret position throws Tooltip outside of screen area
              // then do some UI adjustment.
              if (caretPosition.X + this.Width > workingArea.Width)
              {
                  caretPosition.X = caretPosition.X - this.Width - 50;
              }

              if (caretPosition.Y + this.Height > workingArea.Height)
              {
                  caretPosition.Y = caretPosition.Y - this.Height - 50;
              }

              this.Left = caretPosition.X;
              this.Top = caretPosition.Y;
          }

Points of Interest

After spending some time on Google to get some clue about it, I found some people saying that it is impossible for Microsoft Office 2007 products because it has its own paint logic. Then I tried to resolve this problem and reached the solution shared here that works for any application (Including Microsoft Office 2007 & Visual Studio).

History

  • 30th March, 2009: Initial post

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)


Written By
Software Developer (Senior)
India India
This member has not yet provided a Biography. Assume it's interesting and varied, and probably something to do with programming.

Comments and Discussions

 
AnswerRe: Your work is very helpful. Pin
Saurabh Singh _4-Aug-12 8:46
Saurabh Singh _4-Aug-12 8:46 
QuestionChrome issue Pin
whatnextkenn5-Mar-12 7:53
whatnextkenn5-Mar-12 7:53 
AnswerRe: Chrome issue Pin
Saurabh Singh _25-May-12 0:21
Saurabh Singh _25-May-12 0:21 
QuestionMost challenging!! get caret position under WinWord when some text is selected Pin
Dallas Cao18-Dec-11 6:43
Dallas Cao18-Dec-11 6:43 
GeneralMy vote of 5 Pin
Nimeshji11-Jan-11 1:47
Nimeshji11-Jan-11 1:47 
QuestionIs there a VB Classic equivalent code? Pin
balajiallam11-Aug-10 14:58
balajiallam11-Aug-10 14:58 
GeneralIt works on most but not all open windows Pin
mazrabul21-Jun-10 5:08
mazrabul21-Jun-10 5:08 
GeneralRe: It works on most but not all open windows Pin
Saurabh Singh _8-Jul-10 20:39
Saurabh Singh _8-Jul-10 20:39 
WPF uses DirectX to render UI that's why caret is not accessible.
GeneralGetting the word Pin
Gazatteer3412-Apr-10 22:12
Gazatteer3412-Apr-10 22:12 
GeneralFrom what i've found about the api function GetCaretPos Pin
talavi8-Apr-10 2:02
talavi8-Apr-10 2:02 
GeneralRe: From what i've found about the api function GetCaretPos Pin
Gopalakrishna Palem15-Jul-11 15:56
Gopalakrishna Palem15-Jul-11 15:56 
QuestionIf you find support for browsers and any application? Pin
talavi8-Apr-10 1:59
talavi8-Apr-10 1:59 
AnswerRe: If you find support for browsers and any application? Pin
GRTR25-Mar-11 9:14
GRTR25-Mar-11 9:14 
GeneralThanks Pin
Member 457916711-Mar-10 17:17
Member 457916711-Mar-10 17:17 
GeneralUsing my own text content in the tooltip Pin
bmnot4-Feb-10 12:19
bmnot4-Feb-10 12:19 
AnswerRe: Using my own text content in the tooltip Pin
Saurabh Singh _3-Mar-10 20:29
Saurabh Singh _3-Mar-10 20:29 
GeneralBrowser Support Pin
avrankou29-Jan-10 18:45
avrankou29-Jan-10 18:45 
GeneralRe: Browser Support Pin
lydevoux4-Mar-10 17:30
lydevoux4-Mar-10 17:30 
GeneralGood Article Pin
satishnuni1-Jan-10 0:39
satishnuni1-Jan-10 0:39 
Questionhow to attach tooltip to the caret? Pin
Member 384027927-Dec-09 9:41
Member 384027927-Dec-09 9:41 
GeneralUsing this with AutoHotkey Pin
Ivo Roper8-Jul-09 15:28
Ivo Roper8-Jul-09 15:28 
GeneralRe: Using this with AutoHotkey Pin
Gilbert Premo24-Sep-09 7:50
Gilbert Premo24-Sep-09 7:50 
GeneralRe: Using this with AutoHotkey Pin
Ivo Roper24-Sep-09 14:41
Ivo Roper24-Sep-09 14:41 
GeneralRe: Using this with AutoHotkey [modified] Pin
Gilbert Premo16-Jul-10 12:22
Gilbert Premo16-Jul-10 12:22 
GeneralRe: Using this with AutoHotkey Pin
Ivo Roper16-Jul-10 12:48
Ivo Roper16-Jul-10 12:48 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.