October 26th, 2022

What can or should I do with the cursor handle returned by SetCursor?

A customer was a bit confused by the return value of the Set­Cursor function. Why does it return the previous cursor? Does the caller own it now? Is the caller responsible for destroying it?

First we’ll answer the questions: Ownership of the cursor does not change. Whoever was responsible for the cursor before you called Set­Cursor is still responsible for it.

Okay, with the answers out of the way, let’s take a step back.

There are two general use cases for Set­Cursor.

One of them is for setting the cursor in response to a WM_SET­CURSOR message. The system is telling you, “Okay, you’re in charge of the cursor. Pick something.” In this case, you set the cursor, and you don’t care what the previous cursor was, because you’re choosing the cursor now. Any old cursors are losers.

The other pattern is where you are temporarily changing the cursor (typically to an hourglass), and you want to change it back when you’re done.

void DoLongRunningThingOnTheUIThread()
{
    HCURSOR oldCursor = SetCursor(hourglass);
    /* Do stuff that DOES NOT PUMP MESSAGES */
    SetCursor(oldCursor);
}

It is essential that you not pump messages because

  1. If you pump messages and the user moves the mouse, then the WM_SET­CURSOR message will change the cursor, and your hourglass will be lost. Even worse, you will restore the wrong cursor.
  2. If you pump messages, then that creates the opportunity for the cursor owner to destroy the old cursor while you still have a handle to it.

The point is that the code that had set the cursor (the cursor you are temporarily replacing) has to keep the cursor handle valid until it gets a chance to change the cursor to something else. And as long as you don’t pump messages, that code is not going to get another WM_SET­CURSOR message, and therefore won’t get a chance to change the cursor.

Mind you, locking up the UI thread for a long period of time is not a great idea, so even though it is a common pattern, it’s a sign that your program should probably be moving the expensive work to a background thread.

Bonus reading: The effect of SetCursor lasts only until the next SetCursor.

Topics
Code

Author

Raymond has been involved in the evolution of Windows for more than 30 years. In 2003, he began a Web site known as The Old New Thing which has grown in popularity far beyond his wildest imagination, a development which still gives him the heebie-jeebies. The Web site spawned a book, coincidentally also titled The Old New Thing (Addison Wesley 2007). He occasionally appears on the Windows Dev Docs Twitter account to tell stories which convey no useful information.

3 comments

Discussion is closed. Login to edit/delete existing comments.

Newest
Newest
Popular
Oldest
  • Yuri Khan

    Bonus lore: SetCursor and WM_SETCURSOR are from the times when multitasking was coöperative and multithreading did not exist (in Windows, at least).

  • 紅樓鍮

    If you pump messages and the user moves the mouse… your hourglass will be lost.

    Is this the reason why I practically never see the “background hourglass” cursor?

    • Ian Boyd

      Then i first insalled Windows NT 3.1, the whole thing just exuded power.

      – the hardware cursor was smooth and responsive
      – it showed full window contents while dragging

      And while launching a program from Program Manager, the cursor would change to an “App Start” – the system remained responsive while a program launched in the background.

      And I don’t remember seeing the app start cursor since.

Feedback