This is a "cheat sheet" for "native_cursor" extension by YellowAfterlife. The extension can be found on itch.io or GM Marketplace.
Click on sections to expand/collapse them.
Quick display controls: Categories · Sections · Everything ·

Concepts

Cursors are typically 256x256 pixels or smaller, though the current Windows versions do not seem to limit the cursor size.

On select older systems, cursors above 64x64 pixels did not display at all.

It's a good idea to keep your cursors square - otherwise screen-capturing software might not display them correctly.


Hotspot (hpx/hpy) is an equivalent of an origin for the cursor.


FPS is animation speed in frames per second.

Basics
Management
Management
native_cursor_create_empty()​

Creates and returns a new empty cursor.

native_cursor_destroy(cursor)

Destroys a previously created cursor.

Buffer conversions:

native_cursor_create_from_buffer(buffer, width, height, hpx, hpy, fps)​

Creates a cursor from a buffer containing RGBA pixel data.

native_cursor_add_from_buffer(cursor, buffer, width, height, hpx, hpy)

Adds a frame to a cursor from a buffer containing RGBA pixel data.

Surface conversions:

native_cursor_create_from_surface(surface, hotspot_x, hotspot_y, ?fps)​

Creates a cursor from a surface.

native_cursor_add_from_surface(cursor, surface, hotspot_x, hotspot_y)

Adds a frame to a cursor from a surface.

Loading native cursors from files:

native_cursor_create_from_file(path)​

Loads a cursor from a file.

Note: This takes a CUR/ANI file, not an image.

Note: An ANI cursor doesn't currently get split into frames on load, so trying to append frames to it will not work like you expect.

native_cursor_add_from_file(cursor, path)​

Appends a frame to a cursor from a file.

Note: This takes a CUR/ANI file, not an image.

Note: An ANI cursor doesn't currently get split into frames on load, so trying to append frames to it will not work like you expect.

Sprite conversions:

native_cursor_create_from_sprite(sprite, ?frame, ?fps)​

Creates a cursor from a sprite.

If frame is not specified or is undefined, all frames will be added.
Otherwise a single frame is used.

If fps is not specified, game's framerate is used.

native_cursor_create_from_sprite_ext(sprite, frame, xscale, yscale, color, alpha, ?fps)​

Like above, but with scale/color/alpha.

You can use gpu_set_texfilter (GMS2) or texture_set_interpolation (GMS1) to control how the images will be scaled.

native_cursor_add_from_sprite_ext(cursor, sprite, frame, xscale, yscale, color, alpha)

Adds a sprite frame to a cursor.

State
State
native_cursor_update()

You should call this once per frame or your cursors won't animate when the mouse isn't moving.


native_cursor_set(cursor)

Switches the cursor to a specified one.

native_cursor_reset()

Reverts to default GameMaker cursor behavior.


native_cursor_get_frame(cursor)​int

Returns the current frame of a cursor.

Note that this is calculated in real-time.

native_cursor_set_frame(cursor, frame:int)

Changes the current frame for a cursor.


native_cursor_get_framerate(cursor)​number

Returns current animation speed for a cursor.

native_cursor_set_framerate(cursor, fps:int)

Changes animation speed for a cursor.

Callback
Callback

These enable you to respond to cursor position changes in a more timely manner than what would be possible by putting code in GameMaker events.

native_cursor_set_callback(?function)​

Changes the function/script that fires on mouse movement.

Resets the callback if function argument is omitted or is undefined.

The function receives x, y arguments (window mouse coordinates) that come from WM_SETCURSOR and can take an opportunity to change the cursor.

Windows may serve mouse events an arbitrary number of times per second - for example, if the player uses a gaming mouse with a 8000 Hz polling rate, your function might be called up to 133 times per frame (at 60fps).

For this reason it's best to keep your callback logic minimal or even allow the user to opt out by switching to a lower-frequency callback mode.

For example, the following creates a handful of "directional crosshair" cursors from a spr_cursor_dir sprite and then selects the appropriate one based on direction between window center and mouse position:

dir_count = 360;
dir_cursors = [];
var _surf = surface_create(64, 64);
var _texf = gpu_get_texfilter();
gpu_set_tex_filter(true);
for (var i = 0; i < dir_count; i++) {
    surface_set_target(_surf);
    draw_clear_alpha(c_white, 0);
    draw_sprite_ext(spr_cursor_dir, 0, 32, 32, 1, 1, i * 360 / dir_count, c_white, 1);
    surface_reset_target();
    dir_cursors[i] = native_cursor_create_from_surface(_surf, 32, 32);
}
surface_free(_surf);
gpu_set_tex_filter(_texf);
native_cursor_set_callback(function(_x, _y) {
    var dir = point_direction(
        window_get_width() div 2,
        window_get_height() div 2,
        _x, _y
    );
    var ind = round(dir / 360 * dir_count) % dir_count;
    if (ind < 0) ind += dir_count;
    native_cursor_set(dir_cursors[ind]);
});
native_cursor_get_callback()​

Returns the current callback.


native_cursor_set_callback_mode(mode)

Changes the callback handling mode.

The supported values are:

  • native_cursor_set_callback_mode_highp:
    The callback is executed once per WM_MOUSEMOVE event.
  • native_cursor_set_callback_mode_lowp:
    The callback is executed once per frame when you run native_cursor_update (provided that the mouse )
native_cursor_get_callback_mode()​

Returns the current callback handling mode.