From b97b7fa14849ee0f6f0f0cf672f5836c60c95020 Mon Sep 17 00:00:00 2001 From: ws909 <37029098+ws909@users.noreply.github.com> Date: Wed, 4 Jan 2023 11:20:22 +0100 Subject: [PATCH] Add glfwSetApplicationIcon Closes #2041 New API: glfwSetApplication. Stubs emitting GLFW_FEATURE_UNIMPLEMENTED for X11, Wayland and Win32. Cocoa implementation selects the largest image provided. Updated the icon test to also update the application icon. --- CONTRIBUTORS.md | 1 + README.md | 1 + include/GLFW/glfw3.h | 45 ++++++++++++++++++++++++++++++++++ src/cocoa_init.m | 58 ++++++++++++++++++++++++++++++++++++++++++++ src/cocoa_platform.h | 1 + src/init.c | 5 ++++ src/internal.h | 1 + src/null_init.c | 5 ++++ src/null_platform.h | 1 + src/win32_init.c | 8 ++++++ src/win32_platform.h | 1 + src/wl_init.c | 8 ++++++ src/wl_platform.h | 1 + src/x11_init.c | 8 ++++++ src/x11_platform.h | 1 + tests/icon.c | 9 +++++++ 16 files changed, 154 insertions(+) diff --git a/CONTRIBUTORS.md b/CONTRIBUTORS.md index 47301dae..ce387a89 100644 --- a/CONTRIBUTORS.md +++ b/CONTRIBUTORS.md @@ -99,6 +99,7 @@ video tutorials. - Aaron Jacobs - JannikGM - Erik S. V. Jansson + - Andreas O. Jansen - jjYBdx4IL - Toni Jovanoski - Arseny Kapoulkine diff --git a/README.md b/README.md index 8b4a1546..37d31006 100644 --- a/README.md +++ b/README.md @@ -121,6 +121,7 @@ information on what to include when reporting a bug. ## Changelog + - Added `glfwSetApplicationIcon` to set the application's icon (#2041) - Added `GLFW_PLATFORM` init hint for runtime platform selection (#1958) - Added `GLFW_ANY_PLATFORM`, `GLFW_PLATFORM_WIN32`, `GLFW_PLATFORM_COCOA`, `GLFW_PLATFORM_WAYLAND`, `GLFW_PLATFORM_X11` and `GLFW_PLATFORM_NULL` symbols to diff --git a/include/GLFW/glfw3.h b/include/GLFW/glfw3.h index d4b40dd4..290503d0 100644 --- a/include/GLFW/glfw3.h +++ b/include/GLFW/glfw3.h @@ -2475,6 +2475,51 @@ GLFWAPI int glfwGetPlatform(void); */ GLFWAPI int glfwPlatformSupported(int platform); +/*! @brief Sets the icon for the application. + * + * This function sets the run-time icon of the application. If passed an array of + * candidate images, those of or closest to the sizes desired by the system are + * selected. If no images are specified, the window reverts to its default + * icon. + * + * The pixels are 32-bit, little-endian, non-premultiplied RGBA, i.e. eight + * bits per channel with the red channel first. They are arranged canonically + * as packed sequential rows, starting from the top-left corner. + * + * The desired image sizes varies depending on platform and system settings. + * The selected images will be rescaled as needed. Good sizes include 16x16, + * 32x32 and 48x48. + * + * @param[in] count The number of images in the specified array, or zero to + * revert to the default window icon. + * @param[in] images The images to create the icon from. This is ignored if + * count is zero. + * + * @errors Possible errors include @ref GLFW_NOT_INITIALIZED, @ref + * GLFW_INVALID_VALUE, @ref GLFW_PLATFORM_ERROR and @ref + * GLFW_FEATURE_UNAVAILABLE (see remarks). + * + * @pointer_lifetime The specified image data is copied before this function + * returns. + * + * @remark @macos Sets the Dock icon. Currently selects the largest image. + * + * @remark @win32 This function will emit @ref GLFW_FEATURE_UNIMPLEMENTED. + * + * @remark @x11 This function will emit @ref GLFW_FEATURE_UNIMPLEMENTED. + * + * @remark @wayland This function will emit @ref GLFW_FEATURE_UNIMPLEMENTED. + * + * @thread_safety This function must only be called from the main thread. + * + * @sa @ref window_icon + * + * @since Added in version 3.4. + * + * @ingroup init + */ +GLFWAPI void glfwSetApplicationIcon(int count, const GLFWimage* images); + /*! @brief Returns the currently connected monitors. * * This function returns an array of handles for all currently connected diff --git a/src/cocoa_init.m b/src/cocoa_init.m index b3831df1..581d031a 100644 --- a/src/cocoa_init.m +++ b/src/cocoa_init.m @@ -498,6 +498,7 @@ GLFWbool _glfwConnectCocoa(int platformID, _GLFWplatform* platform) GLFW_PLATFORM_COCOA, _glfwInitCocoa, _glfwTerminateCocoa, + _glfwSetApplicationIconCocoa, _glfwGetCursorPosCocoa, _glfwSetCursorPosCocoa, _glfwSetCursorModeCocoa, @@ -693,5 +694,62 @@ void _glfwTerminateCocoa(void) } // autoreleasepool } +void _glfwSetApplicationIconCocoa(int count, const GLFWimage* images) +{ + @autoreleasepool { + if (count == 0) + { + NSApp.applicationIconImage = nil; + return; + } + + // Selects the largest image. Should probably select the one with the + // size most similar to the current NSApp.applicationIconImage.size. + int imageIndex = 0; + for (int i = 1; i < count; ++i) + { + if (images[i].width > images[imageIndex].width || + images[i].height > images[imageIndex].height) + imageIndex = i; + } + + const GLFWimage* image = images + imageIndex; + + assert(image->pixels != NULL); + + NSImage* native; + NSBitmapImageRep* rep; + + rep = [[NSBitmapImageRep alloc] + initWithBitmapDataPlanes:NULL + pixelsWide:image->width + pixelsHigh:image->height + bitsPerSample:8 + samplesPerPixel:4 + hasAlpha:YES + isPlanar:NO + colorSpaceName:NSCalibratedRGBColorSpace + bitmapFormat:NSBitmapFormatAlphaNonpremultiplied + bytesPerRow:image->width * 4 + bitsPerPixel:32]; + + if (rep == nil) + { + _glfwInputError(GLFW_PLATFORM_ERROR, NULL); + return; + } + + memcpy([rep bitmapData], image->pixels, image->width * image->height * 4); + + native = [[NSImage alloc] initWithSize:NSMakeSize(image->width, image->height)]; + [native addRepresentation:rep]; + + NSApp.applicationIconImage = native; + + [native release]; + [rep release]; + } +} + #endif // _GLFW_COCOA diff --git a/src/cocoa_platform.h b/src/cocoa_platform.h index 9f7d191d..c7f88caa 100644 --- a/src/cocoa_platform.h +++ b/src/cocoa_platform.h @@ -213,6 +213,7 @@ typedef struct _GLFWcursorNS GLFWbool _glfwConnectCocoa(int platformID, _GLFWplatform* platform); int _glfwInitCocoa(void); void _glfwTerminateCocoa(void); +void _glfwSetApplicationIconCocoa(int count, const GLFWimage* images); GLFWbool _glfwCreateWindowCocoa(_GLFWwindow* window, const _GLFWwndconfig* wndconfig, const _GLFWctxconfig* ctxconfig, const _GLFWfbconfig* fbconfig); void _glfwDestroyWindowCocoa(_GLFWwindow* window); diff --git a/src/init.c b/src/init.c index d07a492e..c8054ff8 100644 --- a/src/init.c +++ b/src/init.c @@ -543,3 +543,8 @@ GLFWAPI GLFWerrorfun glfwSetErrorCallback(GLFWerrorfun cbfun) return cbfun; } +GLFWAPI void glfwSetApplicationIcon(int count, const GLFWimage* images) +{ + assert(count == 0 || images != NULL); + _glfw.platform.setApplicationIcon(count, images); +} diff --git a/src/internal.h b/src/internal.h index 781c8cdc..e1dc19d6 100644 --- a/src/internal.h +++ b/src/internal.h @@ -673,6 +673,7 @@ struct _GLFWplatform // init GLFWbool (*init)(void); void (*terminate)(void); + void (*setApplicationIcon)(int,const GLFWimage*); // input void (*getCursorPos)(_GLFWwindow*,double*,double*); void (*setCursorPos)(_GLFWwindow*,double,double); diff --git a/src/null_init.c b/src/null_init.c index de4b28f3..99374f54 100644 --- a/src/null_init.c +++ b/src/null_init.c @@ -43,6 +43,7 @@ GLFWbool _glfwConnectNull(int platformID, _GLFWplatform* platform) GLFW_PLATFORM_NULL, _glfwInitNull, _glfwTerminateNull, + _glfwSetApplicationIconNull, _glfwGetCursorPosNull, _glfwSetCursorPosNull, _glfwSetCursorModeNull, @@ -131,3 +132,7 @@ void _glfwTerminateNull(void) _glfwTerminateEGL(); } +void _glfwSetApplicationIconNull(int count, const GLFWimage* images) +{ + // Silently ignore +} diff --git a/src/null_platform.h b/src/null_platform.h index b646acb3..723df334 100644 --- a/src/null_platform.h +++ b/src/null_platform.h @@ -75,6 +75,7 @@ void _glfwPollMonitorsNull(void); GLFWbool _glfwConnectNull(int platformID, _GLFWplatform* platform); int _glfwInitNull(void); void _glfwTerminateNull(void); +void _glfwSetApplicationIconNull(int count, const GLFWimage* images); void _glfwFreeMonitorNull(_GLFWmonitor* monitor); void _glfwGetMonitorPosNull(_GLFWmonitor* monitor, int* xpos, int* ypos); diff --git a/src/win32_init.c b/src/win32_init.c index 64393e77..55889043 100644 --- a/src/win32_init.c +++ b/src/win32_init.c @@ -608,6 +608,7 @@ GLFWbool _glfwConnectWin32(int platformID, _GLFWplatform* platform) GLFW_PLATFORM_WIN32, _glfwInitWin32, _glfwTerminateWin32, + _glfwSetApplicationIconWin32, _glfwGetCursorPosWin32, _glfwSetCursorPosWin32, _glfwSetCursorModeWin32, @@ -727,5 +728,12 @@ void _glfwTerminateWin32(void) freeLibraries(); } +void _glfwSetApplicationIconWin32(int count, const GLFWimage* images) +{ + // TODO + _glfwInputError(GLFW_FEATURE_UNIMPLEMENTED, + "Win32: Application icon setting not implemented yet"); +} + #endif // _GLFW_WIN32 diff --git a/src/win32_platform.h b/src/win32_platform.h index 82b34bb9..6f502113 100644 --- a/src/win32_platform.h +++ b/src/win32_platform.h @@ -526,6 +526,7 @@ typedef struct _GLFWcursorWin32 GLFWbool _glfwConnectWin32(int platformID, _GLFWplatform* platform); int _glfwInitWin32(void); void _glfwTerminateWin32(void); +void _glfwSetApplicationIconWin32(int count, const GLFWimage* images); WCHAR* _glfwCreateWideStringFromUTF8Win32(const char* source); char* _glfwCreateUTF8FromWideStringWin32(const WCHAR* source); diff --git a/src/wl_init.c b/src/wl_init.c index 7a9157a4..1c5e963a 100644 --- a/src/wl_init.c +++ b/src/wl_init.c @@ -381,6 +381,7 @@ GLFWbool _glfwConnectWayland(int platformID, _GLFWplatform* platform) GLFW_PLATFORM_WAYLAND, _glfwInitWayland, _glfwTerminateWayland, + _glfwSetApplicationIconWayland, _glfwGetCursorPosWayland, _glfwSetCursorPosWayland, _glfwSetCursorModeWayland, @@ -793,5 +794,12 @@ void _glfwTerminateWayland(void) _glfw_free(_glfw.wl.clipboardString); } +void _glfwSetApplicationIconWayland(int count, const GLFWimage* images) +{ + // TODO + _glfwInputError(GLFW_FEATURE_UNIMPLEMENTED, + "Wayland: Application icon setting not implemented yet"); +} + #endif // _GLFW_WAYLAND diff --git a/src/wl_platform.h b/src/wl_platform.h index 238e1ed4..e1a0e64d 100644 --- a/src/wl_platform.h +++ b/src/wl_platform.h @@ -441,6 +441,7 @@ typedef struct _GLFWcursorWayland GLFWbool _glfwConnectWayland(int platformID, _GLFWplatform* platform); int _glfwInitWayland(void); void _glfwTerminateWayland(void); +void _glfwSetApplicationIconWayland(int count, const GLFWimage* images); GLFWbool _glfwCreateWindowWayland(_GLFWwindow* window, const _GLFWwndconfig* wndconfig, const _GLFWctxconfig* ctxconfig, const _GLFWfbconfig* fbconfig); void _glfwDestroyWindowWayland(_GLFWwindow* window); diff --git a/src/x11_init.c b/src/x11_init.c index a0100f2f..cf238977 100644 --- a/src/x11_init.c +++ b/src/x11_init.c @@ -1171,6 +1171,7 @@ GLFWbool _glfwConnectX11(int platformID, _GLFWplatform* platform) GLFW_PLATFORM_X11, _glfwInitX11, _glfwTerminateX11, + _glfwSetApplicationIconX11, _glfwGetCursorPosX11, _glfwSetCursorPosX11, _glfwSetCursorModeX11, @@ -1654,5 +1655,12 @@ void _glfwTerminateX11(void) } } +void _glfwSetApplicationIconX11(int count, const GLFWimage* images) +{ + // TODO + _glfwInputError(GLFW_FEATURE_UNIMPLEMENTED, + "X11: Application icon setting not implemented yet"); +} + #endif // _GLFW_X11 diff --git a/src/x11_platform.h b/src/x11_platform.h index cdea3957..1dd0fc6f 100644 --- a/src/x11_platform.h +++ b/src/x11_platform.h @@ -900,6 +900,7 @@ typedef struct _GLFWcursorX11 GLFWbool _glfwConnectX11(int platformID, _GLFWplatform* platform); int _glfwInitX11(void); void _glfwTerminateX11(void); +void _glfwSetApplicationIconX11(int count, const GLFWimage* images); GLFWbool _glfwCreateWindowX11(_GLFWwindow* window, const _GLFWwndconfig* wndconfig, const _GLFWctxconfig* ctxconfig, const _GLFWfbconfig* fbconfig); void _glfwDestroyWindowX11(_GLFWwindow* window); diff --git a/tests/icon.c b/tests/icon.c index d5baf0a0..09e27835 100644 --- a/tests/icon.c +++ b/tests/icon.c @@ -89,6 +89,7 @@ static void set_icon(GLFWwindow* window, int icon_color) } glfwSetWindowIcon(window, 1, &img); + glfwSetApplicationIcon(1, &img); } static void key_callback(GLFWwindow* window, int key, int scancode, int action, int mods) @@ -107,13 +108,21 @@ static void key_callback(GLFWwindow* window, int key, int scancode, int action, break; case GLFW_KEY_X: glfwSetWindowIcon(window, 0, NULL); + glfwSetApplicationIcon(0, NULL); break; } } +static void error_callback(int error_code, const char* description) +{ + fprintf(stderr, "Error %i: %s\n", error_code, description); +} + int main(int argc, char** argv) { GLFWwindow* window; + + glfwSetErrorCallback(error_callback); if (!glfwInit()) {