mirror of
https://github.com/Kaehvaman/OAIP.git
synced 2025-01-19 00:59:12 +04:00
1022 lines
37 KiB
C
1022 lines
37 KiB
C
|
/**********************************************************************************************
|
||
|
*
|
||
|
* raylib-nuklear v5.5.0 - Nuklear GUI for Raylib.
|
||
|
*
|
||
|
* FEATURES:
|
||
|
* - Use the Nuklear immediate-mode graphical user interface in raylib.
|
||
|
*
|
||
|
* DEPENDENCIES:
|
||
|
* - raylib 5.5+ https://www.raylib.com/
|
||
|
* - Nuklear https://github.com/Immediate-Mode-UI/Nuklear
|
||
|
*
|
||
|
* LICENSE: zlib/libpng
|
||
|
*
|
||
|
* raylib-nuklear is licensed under an unmodified zlib/libpng license, which is an OSI-certified,
|
||
|
* BSD-like license that allows static linking with closed source software:
|
||
|
*
|
||
|
* Copyright (c) 2020 Rob Loach (@RobLoach)
|
||
|
*
|
||
|
* This software is provided "as-is", without any express or implied warranty. In no event
|
||
|
* will the authors be held liable for any damages arising from the use of this software.
|
||
|
*
|
||
|
* Permission is granted to anyone to use this software for any purpose, including commercial
|
||
|
* applications, and to alter it and redistribute it freely, subject to the following restrictions:
|
||
|
*
|
||
|
* 1. The origin of this software must not be misrepresented; you must not claim that you
|
||
|
* wrote the original software. If you use this software in a product, an acknowledgment
|
||
|
* in the product documentation would be appreciated but is not required.
|
||
|
*
|
||
|
* 2. Altered source versions must be plainly marked as such, and must not be misrepresented
|
||
|
* as being the original software.
|
||
|
*
|
||
|
* 3. This notice may not be removed or altered from any source distribution.
|
||
|
*
|
||
|
**********************************************************************************************/
|
||
|
|
||
|
#ifndef RAYLIB_NUKLEAR_H
|
||
|
#define RAYLIB_NUKLEAR_H
|
||
|
|
||
|
#include "raylib.h"
|
||
|
|
||
|
// Nuklear defines
|
||
|
#define NK_INCLUDE_FIXED_TYPES
|
||
|
#define NK_INCLUDE_STANDARD_VARARGS
|
||
|
#define NK_INCLUDE_STANDARD_BOOL
|
||
|
#define NK_INCLUDE_COMMAND_USERDATA
|
||
|
#define NK_KEYSTATE_BASED_INPUT
|
||
|
|
||
|
#ifndef NK_ASSERT
|
||
|
#ifdef NDEBUG
|
||
|
#define NK_ASSERT(condition) ((void)0)
|
||
|
#else
|
||
|
#define NK_ASSERT(condition) do { if (!(condition)) { TraceLog(LOG_ERROR, "NUKLEAR: Failed assert \"%s\" (%s:%i)", #condition, "nuklear.h", __LINE__); }} while (0)
|
||
|
#endif // NDEBUG
|
||
|
#endif // NK_ASSERT
|
||
|
|
||
|
#include "nuklear.h"
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
extern "C" {
|
||
|
#endif
|
||
|
|
||
|
NK_API struct nk_context* InitNuklear(int fontSize); // Initialize the Nuklear GUI context using raylib's font
|
||
|
NK_API struct nk_context* InitNuklearEx(Font font, float fontSize); // Initialize the Nuklear GUI context, with a custom font
|
||
|
NK_API Font LoadFontFromNuklear(int fontSize); // Loads the default Nuklear font
|
||
|
NK_API void UpdateNuklear(struct nk_context * ctx); // Update the input state and internal components for Nuklear
|
||
|
NK_API void UpdateNuklearEx(struct nk_context * ctx, float deltaTime); // Update the input state and internal components for Nuklear, with a custom frame time
|
||
|
NK_API void DrawNuklear(struct nk_context * ctx); // Render the Nuklear GUI on the screen
|
||
|
NK_API void UnloadNuklear(struct nk_context * ctx); // Deinitialize the Nuklear context
|
||
|
NK_API struct nk_color ColorToNuklear(Color color); // Convert a raylib Color to a Nuklear color object
|
||
|
NK_API struct nk_colorf ColorToNuklearF(Color color); // Convert a raylib Color to a Nuklear floating color
|
||
|
NK_API struct Color ColorFromNuklear(struct nk_color color); // Convert a Nuklear color to a raylib Color
|
||
|
NK_API struct Color ColorFromNuklearF(struct nk_colorf color); // Convert a Nuklear floating color to a raylib Color
|
||
|
NK_API struct Rectangle RectangleFromNuklear(struct nk_context * ctx, struct nk_rect rect); // Convert a Nuklear rectangle to a raylib Rectangle
|
||
|
NK_API struct nk_rect RectangleToNuklear(struct nk_context * ctx, Rectangle rect); // Convert a raylib Rectangle to a Nuklear Rectangle
|
||
|
NK_API struct nk_image TextureToNuklear(Texture tex); // Convert a raylib Texture to A Nuklear image
|
||
|
NK_API struct Texture TextureFromNuklear(struct nk_image img); // Convert a Nuklear image to a raylib Texture
|
||
|
NK_API struct nk_image LoadNuklearImage(const char* path); // Load a Nuklear image
|
||
|
NK_API void UnloadNuklearImage(struct nk_image img); // Unload a Nuklear image. And free its data
|
||
|
NK_API void CleanupNuklearImage(struct nk_image img); // Frees the data stored by the Nuklear image
|
||
|
NK_API void SetNuklearScaling(struct nk_context * ctx, float scaling); // Sets the scaling for the given Nuklear context
|
||
|
NK_API float GetNuklearScaling(struct nk_context * ctx); // Retrieves the scaling of the given Nuklear context
|
||
|
|
||
|
// Internal Nuklear functions
|
||
|
NK_API float nk_raylib_font_get_text_width(nk_handle handle, float height, const char *text, int len);
|
||
|
NK_API float nk_raylib_font_get_text_width_user_font(nk_handle handle, float height, const char *text, int len);
|
||
|
NK_API void nk_raylib_clipboard_paste(nk_handle usr, struct nk_text_edit *edit);
|
||
|
NK_API void nk_raylib_clipboard_copy(nk_handle usr, const char *text, int len);
|
||
|
NK_API void* nk_raylib_malloc(nk_handle unused, void *old, nk_size size);
|
||
|
NK_API void nk_raylib_mfree(nk_handle unused, void *ptr);
|
||
|
NK_API struct nk_context* InitNuklearContext(struct nk_user_font* userFont);
|
||
|
NK_API void nk_raylib_input_keyboard(struct nk_context * ctx);
|
||
|
NK_API void nk_raylib_input_mouse(struct nk_context * ctx);
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#endif // RAYLIB_NUKLEAR_H
|
||
|
|
||
|
#ifdef RAYLIB_NUKLEAR_IMPLEMENTATION
|
||
|
#ifndef RAYLIB_NUKLEAR_IMPLEMENTATION_ONCE
|
||
|
#define RAYLIB_NUKLEAR_IMPLEMENTATION_ONCE
|
||
|
|
||
|
#include <stddef.h> // NULL
|
||
|
#include <math.h> // cosf, sinf, sqrtf
|
||
|
|
||
|
// Math
|
||
|
#ifndef NK_COS
|
||
|
#define NK_COS cosf
|
||
|
#endif // NK_COS
|
||
|
#ifndef NK_SIN
|
||
|
#define NK_SIN sinf
|
||
|
#endif // NK_SIN
|
||
|
#ifndef NK_INV_SQRT
|
||
|
#define NK_INV_SQRT(value) (1.0f / sqrtf(value))
|
||
|
#endif // NK_INV_SQRT
|
||
|
|
||
|
#define NK_IMPLEMENTATION
|
||
|
#include "nuklear.h"
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
extern "C" {
|
||
|
#endif
|
||
|
|
||
|
#ifndef RAYLIB_NUKLEAR_DEFAULT_FONTSIZE
|
||
|
/**
|
||
|
* The default font size that is used when a font size is not provided.
|
||
|
*/
|
||
|
#define RAYLIB_NUKLEAR_DEFAULT_FONTSIZE 13
|
||
|
#endif // RAYLIB_NUKLEAR_DEFAULT_FONTSIZE
|
||
|
|
||
|
/*
|
||
|
* Spacing is determined by the font size multiplied by RAYLIB_NUKLEAR_FONT_SPACING_RATIO.
|
||
|
*/
|
||
|
#ifndef RAYLIB_NUKLEAR_FONT_SPACING_RATIO
|
||
|
#define RAYLIB_NUKLEAR_FONT_SPACING_RATIO 0.01f
|
||
|
#endif // RAYLIB_NUKLEAR_FONT_SPACING_RATIO
|
||
|
|
||
|
#ifndef RAYLIB_NUKLEAR_DEFAULT_ARC_SEGMENTS
|
||
|
/**
|
||
|
* The amount of segments used when drawing an arc.
|
||
|
*
|
||
|
* @see NK_COMMAND_ARC_FILLED
|
||
|
*/
|
||
|
#define RAYLIB_NUKLEAR_DEFAULT_ARC_SEGMENTS 20
|
||
|
#endif // RAYLIB_NUKLEAR_DEFAULT_ARC_SEGMENTS
|
||
|
|
||
|
#ifdef RAYLIB_NUKLEAR_INCLUDE_DEFAULT_FONT
|
||
|
#include "raylib-nuklear-font.h"
|
||
|
#endif
|
||
|
|
||
|
/**
|
||
|
* The user data that's leverages internally through Nuklear.
|
||
|
*/
|
||
|
typedef struct NuklearUserData {
|
||
|
float scaling; // The scaling of the Nuklear user interface.
|
||
|
} NuklearUserData;
|
||
|
|
||
|
/**
|
||
|
* Nuklear callback; Get the width of the given text.
|
||
|
*
|
||
|
* @internal
|
||
|
*/
|
||
|
NK_API float
|
||
|
nk_raylib_font_get_text_width(nk_handle handle, float height, const char *text, int len)
|
||
|
{
|
||
|
NK_UNUSED(handle);
|
||
|
|
||
|
if (len > 0) {
|
||
|
// Grab the text with the cropped length so that it only measures the desired string length.
|
||
|
const char* subtext = TextSubtext(text, 0, len);
|
||
|
|
||
|
// Spacing is determined by the font size multiplied by RAYLIB_NUKLEAR_FONT_SPACING_RATIO.
|
||
|
// Raylib only counts the spacing between characters, but Nuklear expects one spacing to be
|
||
|
// counter for every character in the string:
|
||
|
return (float)MeasureText(subtext, (int)height) + height * RAYLIB_NUKLEAR_FONT_SPACING_RATIO;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Nuklear callback; Get the width of the given text (userFont version)
|
||
|
*
|
||
|
* @internal
|
||
|
*/
|
||
|
NK_API float
|
||
|
nk_raylib_font_get_text_width_user_font(nk_handle handle, float height, const char *text, int len)
|
||
|
{
|
||
|
if (len > 0) {
|
||
|
// Grab the text with the cropped length so that it only measures the desired string length.
|
||
|
const char* subtext = TextSubtext(text, 0, len);
|
||
|
|
||
|
// Spacing is determined by the font size multiplied by RAYLIB_NUKLEAR_FONT_SPACING_RATIO.
|
||
|
// Raylib only counts the spacing between characters, but Nuklear expects one spacing to be
|
||
|
// counter for every character in the string:
|
||
|
return MeasureTextEx(*(Font*)handle.ptr, subtext, height, height * RAYLIB_NUKLEAR_FONT_SPACING_RATIO).x + height * RAYLIB_NUKLEAR_FONT_SPACING_RATIO;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Nuklear callback; Paste the current clipboard.
|
||
|
*
|
||
|
* @internal
|
||
|
*/
|
||
|
NK_API void
|
||
|
nk_raylib_clipboard_paste(nk_handle usr, struct nk_text_edit *edit)
|
||
|
{
|
||
|
const char *text = GetClipboardText();
|
||
|
NK_UNUSED(usr);
|
||
|
if (text != NULL) {
|
||
|
nk_textedit_paste(edit, text, (int)TextLength(text));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Nuklear callback; Copy the given text.
|
||
|
*
|
||
|
* @internal
|
||
|
*/
|
||
|
NK_API void
|
||
|
nk_raylib_clipboard_copy(nk_handle usr, const char *text, int len)
|
||
|
{
|
||
|
NK_UNUSED(usr);
|
||
|
char* trimmedText = (char*)MemAlloc((unsigned int)(sizeof(char) * (size_t)(len + 1)));
|
||
|
if(!trimmedText)
|
||
|
return;
|
||
|
nk_memcopy(trimmedText, text, (nk_size)len);
|
||
|
trimmedText[len] = 0;
|
||
|
SetClipboardText(trimmedText);
|
||
|
MemFree(trimmedText);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Nuklear callback; Allocate memory for Nuklear.
|
||
|
*
|
||
|
* @internal
|
||
|
*/
|
||
|
NK_API void*
|
||
|
nk_raylib_malloc(nk_handle unused, void *old, nk_size size)
|
||
|
{
|
||
|
NK_UNUSED(unused);
|
||
|
NK_UNUSED(old);
|
||
|
return MemAlloc((unsigned int)size);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Nuklear callback; Free memory for Nuklear.
|
||
|
*/
|
||
|
NK_API void
|
||
|
nk_raylib_mfree(nk_handle unused, void *ptr)
|
||
|
{
|
||
|
NK_UNUSED(unused);
|
||
|
MemFree(ptr);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Initialize the Nuklear context for use with Raylib, with the given Nuklear user font.
|
||
|
*
|
||
|
* @param userFont The Nuklear user font to initialize the Nuklear context with.
|
||
|
*
|
||
|
* @internal
|
||
|
*/
|
||
|
NK_API struct nk_context*
|
||
|
InitNuklearContext(struct nk_user_font* userFont)
|
||
|
{
|
||
|
struct nk_context* ctx = (struct nk_context*)MemAlloc(sizeof(struct nk_context));
|
||
|
if (ctx == NULL) {
|
||
|
TraceLog(LOG_ERROR, "NUKLEAR: Failed to initialize nuklear memory");
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
struct NuklearUserData* userData = (struct NuklearUserData*)MemAlloc(sizeof(struct NuklearUserData));
|
||
|
if (userData == NULL) {
|
||
|
TraceLog(LOG_ERROR, "NUKLEAR: Failed to initialize nuklear user data");
|
||
|
MemFree(ctx);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
// Allocator
|
||
|
struct nk_allocator alloc;
|
||
|
alloc.userdata = nk_handle_ptr(0);
|
||
|
alloc.alloc = nk_raylib_malloc;
|
||
|
alloc.free = nk_raylib_mfree;
|
||
|
|
||
|
// Initialize the context.
|
||
|
if (!nk_init(ctx, &alloc, userFont)) {
|
||
|
TraceLog(LOG_ERROR, "NUKLEAR: Failed to initialize nuklear");
|
||
|
MemFree(ctx);
|
||
|
MemFree(userData);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
// Clipboard
|
||
|
ctx->clip.copy = nk_raylib_clipboard_copy;
|
||
|
ctx->clip.paste = nk_raylib_clipboard_paste;
|
||
|
ctx->clip.userdata = nk_handle_ptr(0);
|
||
|
|
||
|
// Set the internal user data.
|
||
|
userData->scaling = 1.0f;
|
||
|
nk_handle userDataHandle;
|
||
|
userDataHandle.id = 1;
|
||
|
userDataHandle.ptr = (void*)userData;
|
||
|
nk_set_user_data(ctx, userDataHandle);
|
||
|
|
||
|
TraceLog(LOG_INFO, "NUKLEAR: Initialized GUI");
|
||
|
|
||
|
return ctx;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Initialize the Nuklear context for use with Raylib.
|
||
|
*
|
||
|
* @param fontSize The size of the font to use for GUI text. Use 0 to use the default font size of 10.
|
||
|
*
|
||
|
* @return The nuklear context, or NULL on error.
|
||
|
*/
|
||
|
NK_API struct nk_context*
|
||
|
InitNuklear(int fontSize)
|
||
|
{
|
||
|
// User font.
|
||
|
struct nk_user_font* userFont = (struct nk_user_font*)MemAlloc(sizeof(struct nk_user_font));
|
||
|
|
||
|
// Use the default font size if desired.
|
||
|
if (fontSize <= 0) {
|
||
|
fontSize = RAYLIB_NUKLEAR_DEFAULT_FONTSIZE;
|
||
|
}
|
||
|
|
||
|
userFont->height = (float)fontSize;
|
||
|
userFont->width = nk_raylib_font_get_text_width;
|
||
|
userFont->userdata = nk_handle_ptr(0);
|
||
|
|
||
|
// Nuklear context.
|
||
|
return InitNuklearContext(userFont);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Initialize the Nuklear context for use with Raylib, with a supplied custom font.
|
||
|
*
|
||
|
* @param font The custom raylib font to use with Nuklear.
|
||
|
* @param fontSize The desired size of the font. Use 0 to set the default size of 10.
|
||
|
*
|
||
|
* @return The nuklear context, or NULL on error.
|
||
|
*/
|
||
|
NK_API struct nk_context*
|
||
|
InitNuklearEx(Font font, float fontSize)
|
||
|
{
|
||
|
// Copy the font to a new raylib font pointer.
|
||
|
struct Font* newFont = (struct Font*)MemAlloc(sizeof(struct Font));
|
||
|
|
||
|
// Use the default font size if desired.
|
||
|
if (fontSize <= 0.0f) {
|
||
|
fontSize = (float)RAYLIB_NUKLEAR_DEFAULT_FONTSIZE;
|
||
|
}
|
||
|
newFont->baseSize = font.baseSize;
|
||
|
newFont->glyphCount = font.glyphCount;
|
||
|
newFont->glyphPadding = font.glyphPadding;
|
||
|
newFont->glyphs = font.glyphs;
|
||
|
newFont->recs = font.recs;
|
||
|
newFont->texture = font.texture;
|
||
|
|
||
|
// Create the nuklear user font.
|
||
|
struct nk_user_font* userFont = (struct nk_user_font*)MemAlloc(sizeof(struct nk_user_font));
|
||
|
userFont->userdata = nk_handle_ptr(newFont);
|
||
|
userFont->height = fontSize;
|
||
|
userFont->width = nk_raylib_font_get_text_width_user_font;
|
||
|
|
||
|
// Nuklear context.
|
||
|
return InitNuklearContext(userFont);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Load the default Nuklear font. Requires `RAYLIB_NUKLEAR_INCLUDE_DEFAULT_FONT` to be defined.
|
||
|
*
|
||
|
* @param size The size of the font to load (optional). Provide 0 if you'd like to use the default size from Nuklear.
|
||
|
*
|
||
|
* @return The loaded font, or an empty font on error.
|
||
|
*
|
||
|
* @code
|
||
|
* #define RAYLIB_NUKLEAR_INCLUDE_DEFAULT_FONT
|
||
|
* #include "raylib-nuklear.h"
|
||
|
* Font font = LoadFontFromNuklear(0);
|
||
|
* @endcode
|
||
|
*/
|
||
|
NK_API Font LoadFontFromNuklear(int size) {
|
||
|
#ifndef RAYLIB_NUKLEAR_INCLUDE_DEFAULT_FONT
|
||
|
(void)size;
|
||
|
TraceLog(LOG_ERROR, "NUKLEAR: RAYLIB_NUKLEAR_INCLUDE_DEFAULT_FONT must be defined to use LoadFontFromNuklear()");
|
||
|
return CLITERAL(Font) {0};
|
||
|
#else
|
||
|
if (size <= 0) {
|
||
|
size = RAYLIB_NUKLEAR_DEFAULT_FONTSIZE;
|
||
|
}
|
||
|
|
||
|
#ifndef RAYLIB_NUKLEAR_DEFAULT_FONT_GLYPHS
|
||
|
/**
|
||
|
* The amount of glyphs to load for the default font.
|
||
|
*/
|
||
|
#define RAYLIB_NUKLEAR_DEFAULT_FONT_GLYPHS 95
|
||
|
#endif
|
||
|
|
||
|
return LoadFontFromMemory(".ttf", RAYLIB_NUKLEAR_DEFAULT_FONT_NAME, RAYLIB_NUKLEAR_DEFAULT_FONT_SIZE, size, NULL, RAYLIB_NUKLEAR_DEFAULT_FONT_GLYPHS);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Convert the given Nuklear color to a raylib color.
|
||
|
*/
|
||
|
NK_API Color
|
||
|
ColorFromNuklear(struct nk_color color)
|
||
|
{
|
||
|
Color rc;
|
||
|
rc.a = color.a;
|
||
|
rc.r = color.r;
|
||
|
rc.g = color.g;
|
||
|
rc.b = color.b;
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Convert the given raylib color to a Nuklear color.
|
||
|
*/
|
||
|
NK_API struct nk_color
|
||
|
ColorToNuklear(Color color)
|
||
|
{
|
||
|
struct nk_color rc;
|
||
|
rc.a = color.a;
|
||
|
rc.r = color.r;
|
||
|
rc.g = color.g;
|
||
|
rc.b = color.b;
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Convert the given Nuklear float color to a raylib color.
|
||
|
*/
|
||
|
NK_API Color
|
||
|
ColorFromNuklearF(struct nk_colorf color)
|
||
|
{
|
||
|
return ColorFromNuklear(nk_rgba_cf(color));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Convert the given raylib color to a raylib float color.
|
||
|
*/
|
||
|
NK_API struct nk_colorf
|
||
|
ColorToNuklearF(Color color)
|
||
|
{
|
||
|
return nk_color_cf(ColorToNuklear(color));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Draw the given Nuklear context in raylib.
|
||
|
*
|
||
|
* @param ctx The nuklear context.
|
||
|
*/
|
||
|
NK_API void
|
||
|
DrawNuklear(struct nk_context * ctx)
|
||
|
{
|
||
|
// Protect against drawing when there's nothing to draw.
|
||
|
if (ctx == NULL || nk__begin(ctx) == 0) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
const struct nk_command *cmd;
|
||
|
const float scale = GetNuklearScaling(ctx);
|
||
|
|
||
|
nk_foreach(cmd, ctx) {
|
||
|
switch (cmd->type) {
|
||
|
case NK_COMMAND_NOP: {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case NK_COMMAND_SCISSOR: {
|
||
|
// TODO(RobLoach): Verify if NK_COMMAND_SCISSOR works.
|
||
|
const struct nk_command_scissor *s =(const struct nk_command_scissor*)cmd;
|
||
|
BeginScissorMode((int)(s->x * scale), (int)(s->y * scale), (int)(s->w * scale), (int)(s->h * scale));
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_LINE: {
|
||
|
const struct nk_command_line *l = (const struct nk_command_line *)cmd;
|
||
|
Color color = ColorFromNuklear(l->color);
|
||
|
Vector2 startPos = CLITERAL(Vector2) {(float)l->begin.x * scale, (float)l->begin.y * scale};
|
||
|
Vector2 endPos = CLITERAL(Vector2) {(float)l->end.x * scale, (float)l->end.y * scale};
|
||
|
DrawLineEx(startPos, endPos, l->line_thickness * scale, color);
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_CURVE: {
|
||
|
const struct nk_command_curve *q = (const struct nk_command_curve *)cmd;
|
||
|
Color color = ColorFromNuklear(q->color);
|
||
|
Vector2 begin = CLITERAL(Vector2) {(float)q->begin.x * scale, (float)q->begin.y * scale};
|
||
|
Vector2 controlPoint1 = CLITERAL(Vector2) {(float)q->ctrl[0].x * scale, (float)q->ctrl[0].y * scale};
|
||
|
Vector2 controlPoint2 = CLITERAL(Vector2) {(float)q->ctrl[1].x * scale, (float)q->ctrl[1].y * scale};
|
||
|
Vector2 end = CLITERAL(Vector2) {(float)q->end.x * scale, (float)q->end.y * scale};
|
||
|
DrawSplineSegmentBezierCubic(begin, controlPoint1, controlPoint2, end, (float)q->line_thickness * scale, color);
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_RECT: {
|
||
|
const struct nk_command_rect *r = (const struct nk_command_rect *)cmd;
|
||
|
Color color = ColorFromNuklear(r->color);
|
||
|
Rectangle rect = CLITERAL(Rectangle) {(float)r->x * scale, (float)r->y * scale, (float)r->w * scale, (float)r->h * scale};
|
||
|
float roundness = (rect.width > rect.height) ?
|
||
|
((2 * r->rounding * scale)/rect.height) : ((2 * r->rounding * scale)/rect.width);
|
||
|
roundness = NK_CLAMP(0.0f, roundness, 1.0f);
|
||
|
if (roundness > 0.0f) {
|
||
|
// DrawRectangleRoundedLines doesn't work in the same way as DrawRectangleLinesEx and it draws
|
||
|
// the outline outside the region defined by the rectangle. To compensate for that, shrink
|
||
|
// the rectangle by the thickness plus 1 (due to inconsistencies from DrawRectangleRoundedLines):
|
||
|
rect.x += ((float) r->line_thickness) * scale + 1.0f;
|
||
|
rect.y += ((float) r->line_thickness) * scale + 1.0f;
|
||
|
rect.width = NK_MAX(rect.width - (2 * ((float) r->line_thickness) * scale + 1.0f), 0.0f);
|
||
|
rect.height = NK_MAX(rect.height - (2 * ((float) r->line_thickness) * scale + 1.0f), 0.0f);
|
||
|
#if RAYLIB_VERSION_MAJOR >= 5 && RAYLIB_VERSION_MINOR == 0
|
||
|
DrawRectangleRoundedLines(rect, roundness, RAYLIB_NUKLEAR_DEFAULT_ARC_SEGMENTS, (float)r->line_thickness * scale, color);
|
||
|
#else
|
||
|
DrawRectangleRoundedLinesEx(rect, roundness, RAYLIB_NUKLEAR_DEFAULT_ARC_SEGMENTS, (float)r->line_thickness * scale, color);
|
||
|
#endif
|
||
|
}
|
||
|
else {
|
||
|
DrawRectangleLinesEx(rect, r->line_thickness * scale, color);
|
||
|
}
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_RECT_FILLED: {
|
||
|
const struct nk_command_rect_filled *r = (const struct nk_command_rect_filled *)cmd;
|
||
|
Color color = ColorFromNuklear(r->color);
|
||
|
Rectangle rect = CLITERAL(Rectangle) {(float)r->x * scale, (float)r->y * scale, (float)r->w * scale, (float)r->h * scale};
|
||
|
float roundness = (rect.width > rect.height) ?
|
||
|
((2 * r->rounding * scale)/rect.height) : ((2 * r->rounding * scale)/rect.width);
|
||
|
roundness = NK_CLAMP(0.0f, roundness, 1.0f);
|
||
|
if (roundness > 0.0f) {
|
||
|
DrawRectangleRounded(rect, roundness, RAYLIB_NUKLEAR_DEFAULT_ARC_SEGMENTS, color);
|
||
|
}
|
||
|
else {
|
||
|
DrawRectangleRec(rect, color);
|
||
|
}
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_RECT_MULTI_COLOR: {
|
||
|
const struct nk_command_rect_multi_color* rectangle = (const struct nk_command_rect_multi_color *)cmd;
|
||
|
Rectangle position = {(float)rectangle->x * scale, (float)rectangle->y * scale, (float)rectangle->w * scale, (float)rectangle->h * scale};
|
||
|
Color left = ColorFromNuklear(rectangle->left);
|
||
|
Color top = ColorFromNuklear(rectangle->top);
|
||
|
Color bottom = ColorFromNuklear(rectangle->bottom);
|
||
|
Color right = ColorFromNuklear(rectangle->right);
|
||
|
DrawRectangleGradientEx(position, left, bottom, right, top);
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_CIRCLE: {
|
||
|
const struct nk_command_circle *c = (const struct nk_command_circle *)cmd;
|
||
|
Color color = ColorFromNuklear(c->color);
|
||
|
for (unsigned short i = 0; i < c->line_thickness; i++) {
|
||
|
DrawEllipseLines((int)(c->x * scale + c->w * scale / 2.0f), (int)(c->y * scale + c->h * scale / 2.0f), c->w * scale / 2.0f - (float)i / 2.0f, c->h * scale / 2.0f - (float)i / 2.0f, color);
|
||
|
}
|
||
|
} break;
|
||
|
|
||
|
#pragma warning(disable: 4244)
|
||
|
case NK_COMMAND_CIRCLE_FILLED: {
|
||
|
const struct nk_command_circle_filled *c = (const struct nk_command_circle_filled *)cmd;
|
||
|
Color color = ColorFromNuklear(c->color);
|
||
|
DrawEllipse((int)(c->x * scale + c->w * scale / 2.0f), (int)(c->y * scale + c->h * scale / 2.0f), (int)(c->w * scale / 2), (int)(c->h * scale / 2), color);
|
||
|
} break;
|
||
|
#pragma warning(default: 4244)
|
||
|
|
||
|
case NK_COMMAND_ARC: {
|
||
|
const struct nk_command_arc *a = (const struct nk_command_arc*)cmd;
|
||
|
Color color = ColorFromNuklear(a->color);
|
||
|
Vector2 center = CLITERAL(Vector2) {(float)a->cx, (float)a->cy};
|
||
|
DrawRingLines(center, 0, a->r * scale, a->a[0] * RAD2DEG, a->a[1] * RAD2DEG, RAYLIB_NUKLEAR_DEFAULT_ARC_SEGMENTS, color);
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_ARC_FILLED: {
|
||
|
const struct nk_command_arc_filled *a = (const struct nk_command_arc_filled*)cmd;
|
||
|
Color color = ColorFromNuklear(a->color);
|
||
|
Vector2 center = CLITERAL(Vector2) {(float)a->cx * scale, (float)a->cy * scale};
|
||
|
DrawRing(center, 0, a->r * scale, a->a[0] * RAD2DEG, a->a[1] * RAD2DEG, RAYLIB_NUKLEAR_DEFAULT_ARC_SEGMENTS, color);
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_TRIANGLE: {
|
||
|
const struct nk_command_triangle *t = (const struct nk_command_triangle*)cmd;
|
||
|
Color color = ColorFromNuklear(t->color);
|
||
|
Vector2 point1 = CLITERAL(Vector2) {(float)t->b.x * scale, (float)t->b.y * scale};
|
||
|
Vector2 point2 = CLITERAL(Vector2) {(float)t->a.x * scale, (float)t->a.y * scale};
|
||
|
Vector2 point3 = CLITERAL(Vector2) {(float)t->c.x * scale, (float)t->c.y * scale};
|
||
|
|
||
|
// DrawLineEx(point1, point2, t->line_thickness * scale, color);
|
||
|
// DrawLineEx(point2, point3, t->line_thickness * scale, color);
|
||
|
// DrawLineEx(point3, point1, t->line_thickness * scale, color);
|
||
|
// TODO: Add line thickness to DrawTriangleLines(), maybe via a DrawTriangleLinesEx()?
|
||
|
DrawTriangleLines(point1, point2, point3, color);
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_TRIANGLE_FILLED: {
|
||
|
const struct nk_command_triangle_filled *t = (const struct nk_command_triangle_filled*)cmd;
|
||
|
Color color = ColorFromNuklear(t->color);
|
||
|
Vector2 point1 = CLITERAL(Vector2) {(float)t->b.x * scale, (float)t->b.y * scale};
|
||
|
Vector2 point2 = CLITERAL(Vector2) {(float)t->a.x * scale, (float)t->a.y * scale};
|
||
|
Vector2 point3 = CLITERAL(Vector2) {(float)t->c.x * scale, (float)t->c.y * scale};
|
||
|
DrawTriangle(point1, point2, point3, color);
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_POLYGON: {
|
||
|
const struct nk_command_polygon *p = (const struct nk_command_polygon*)cmd;
|
||
|
Color color = ColorFromNuklear(p->color);
|
||
|
struct Vector2* points = (struct Vector2*)MemAlloc((unsigned int)((size_t)(p->point_count + 1) * sizeof(Vector2)));
|
||
|
unsigned short i;
|
||
|
for (i = 0; i < p->point_count; i++) {
|
||
|
points[i].x = p->points[i].x * scale;
|
||
|
points[i].y = p->points[i].y * scale;
|
||
|
}
|
||
|
points[p->point_count] = points[0];
|
||
|
DrawLineStrip(points, p->point_count + 1, color);
|
||
|
MemFree(points);
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_POLYGON_FILLED: {
|
||
|
// TODO: Implement NK_COMMAND_POLYGON_FILLED
|
||
|
const struct nk_command_polygon_filled *p = (const struct nk_command_polygon_filled*)cmd;
|
||
|
Color color = ColorFromNuklear(p->color);
|
||
|
struct Vector2* points = (struct Vector2*)MemAlloc((unsigned int)((size_t)(p->point_count + 1) * sizeof(Vector2)));
|
||
|
for (unsigned short i = 0; i < p->point_count; i++) {
|
||
|
points[i].x = p->points[i].x * scale;
|
||
|
points[i].y = p->points[i].y * scale;
|
||
|
}
|
||
|
points[p->point_count] = points[0];
|
||
|
DrawLineStrip(points, p->point_count + 1, color);
|
||
|
MemFree(points);
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_POLYLINE: {
|
||
|
const struct nk_command_polyline *p = (const struct nk_command_polyline *)cmd;
|
||
|
Color color = ColorFromNuklear(p->color);
|
||
|
for (unsigned short i = 0; i < p->point_count - 1; i++) {
|
||
|
Vector2 start = {(float)p->points[i].x * scale, (float)p->points[i].y * scale};
|
||
|
Vector2 end = {(float)p->points[i + 1].x * scale, (float)p->points[i + 1].y * scale};
|
||
|
DrawLineEx(start, end, p->line_thickness * scale, color);
|
||
|
}
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_TEXT: {
|
||
|
const struct nk_command_text *text = (const struct nk_command_text*)cmd;
|
||
|
Color color = ColorFromNuklear(text->foreground);
|
||
|
float fontSize = text->font->height * scale;
|
||
|
Font* font = (Font*)text->font->userdata.ptr;
|
||
|
if (font != NULL) {
|
||
|
Vector2 position = {(float)text->x * scale, (float)text->y * scale};
|
||
|
DrawTextEx(*font, (const char*)text->string, position, fontSize, fontSize * RAYLIB_NUKLEAR_FONT_SPACING_RATIO, color);
|
||
|
}
|
||
|
else {
|
||
|
DrawText((const char*)text->string, (int)(text->x * scale), (int)(text->y * scale), (int)fontSize, color);
|
||
|
}
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_IMAGE: {
|
||
|
const struct nk_command_image *i = (const struct nk_command_image *)cmd;
|
||
|
Texture texture = *(Texture*)i->img.handle.ptr;
|
||
|
Rectangle source = CLITERAL(Rectangle) {i->img.region[0], i->img.region[1], (float)i->img.region[2], (float)i->img.region[3]};
|
||
|
Rectangle dest = CLITERAL(Rectangle) {(float)i->x * scale, (float)i->y * scale, (float)i->w * scale, (float)i->h * scale};
|
||
|
Vector2 origin = CLITERAL(Vector2) {0, 0};
|
||
|
Color tint = ColorFromNuklear(i->col);
|
||
|
DrawTexturePro(texture, source, dest, origin, 0, tint);
|
||
|
} break;
|
||
|
|
||
|
case NK_COMMAND_CUSTOM: {
|
||
|
TraceLog(LOG_WARNING, "NUKLEAR: Unverified custom callback implementation NK_COMMAND_CUSTOM");
|
||
|
const struct nk_command_custom *custom = (const struct nk_command_custom *)cmd;
|
||
|
custom->callback(NULL, (short)(custom->x * scale), (short)(custom->y * scale), (unsigned short)(custom->w * scale), (unsigned short)(custom->h * scale), custom->callback_data);
|
||
|
} break;
|
||
|
|
||
|
default: {
|
||
|
TraceLog(LOG_WARNING, "NUKLEAR: Missing implementation %i", cmd->type);
|
||
|
} break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
nk_clear(ctx);
|
||
|
}
|
||
|
|
||
|
struct nk_raylib_input_keyboard_check {
|
||
|
int key;
|
||
|
int input_key;
|
||
|
bool modifier;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Update the Nuklear context for the keyboard input from raylib.
|
||
|
*
|
||
|
* @param ctx The nuklear context.
|
||
|
*
|
||
|
* @internal
|
||
|
*/
|
||
|
NK_API void
|
||
|
nk_raylib_input_keyboard(struct nk_context * ctx)
|
||
|
{
|
||
|
bool control = IsKeyDown(KEY_LEFT_CONTROL) || IsKeyDown(KEY_RIGHT_CONTROL);
|
||
|
bool command = IsKeyDown(KEY_LEFT_SUPER);
|
||
|
bool shift = IsKeyDown(KEY_LEFT_SHIFT) || IsKeyDown(KEY_RIGHT_SHIFT);
|
||
|
#define NK_RAYLIB_INPUT_KEYBOARD_CHECK_NUM 16
|
||
|
struct nk_raylib_input_keyboard_check checks[NK_RAYLIB_INPUT_KEYBOARD_CHECK_NUM] = {
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_DELETE, NK_KEY_DEL, true},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_ENTER, NK_KEY_ENTER, true},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_BACKSPACE, NK_KEY_BACKSPACE, true},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_C, NK_KEY_COPY, (control || command)},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_V, NK_KEY_PASTE, (control || command)},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_B, NK_KEY_TEXT_LINE_START, (control || command)},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_E, NK_KEY_TEXT_LINE_END, (control || command)},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_Z, NK_KEY_TEXT_UNDO, (control || command)},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_R, NK_KEY_TEXT_REDO, (control || command)},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_A, NK_KEY_TEXT_SELECT_ALL, (control || command)},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_LEFT, NK_KEY_TEXT_WORD_LEFT, (control || command)},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_RIGHT, NK_KEY_TEXT_WORD_RIGHT, (control || command)},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_RIGHT, NK_KEY_RIGHT, true},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_LEFT, NK_KEY_LEFT, true},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_UP, NK_KEY_UP, true},
|
||
|
(struct nk_raylib_input_keyboard_check) {KEY_DOWN, NK_KEY_DOWN, true}
|
||
|
};
|
||
|
bool checked = false;
|
||
|
for (int i = 0; i < NK_RAYLIB_INPUT_KEYBOARD_CHECK_NUM; i++) {
|
||
|
struct nk_raylib_input_keyboard_check check = checks[i];
|
||
|
if (IsKeyDown(check.key) && check.modifier) {
|
||
|
nk_input_key(ctx, check.input_key, true);
|
||
|
checked = true;
|
||
|
} else {
|
||
|
nk_input_key(ctx, check.input_key, false);
|
||
|
}
|
||
|
}
|
||
|
#undef NK_RAYLIB_INPUT_KEYBOARD_CHECK_NUM
|
||
|
|
||
|
nk_input_key(ctx, NK_KEY_SHIFT, shift);
|
||
|
|
||
|
if (checked) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
nk_input_key(ctx, NK_KEY_LEFT, IsKeyDown(KEY_LEFT));
|
||
|
nk_input_key(ctx, NK_KEY_RIGHT, IsKeyDown(KEY_RIGHT));
|
||
|
nk_input_key(ctx, NK_KEY_UP, IsKeyDown(KEY_UP));
|
||
|
nk_input_key(ctx, NK_KEY_DOWN, IsKeyDown(KEY_DOWN));
|
||
|
nk_input_key(ctx, NK_KEY_TEXT_START, IsKeyDown(KEY_HOME));
|
||
|
nk_input_key(ctx, NK_KEY_TEXT_END, IsKeyDown(KEY_END));
|
||
|
nk_input_key(ctx, NK_KEY_SCROLL_START, IsKeyDown(KEY_HOME) && control);
|
||
|
nk_input_key(ctx, NK_KEY_SCROLL_END, IsKeyDown(KEY_END) && control);
|
||
|
nk_input_key(ctx, NK_KEY_SCROLL_DOWN, IsKeyDown(KEY_PAGE_DOWN));
|
||
|
nk_input_key(ctx, NK_KEY_SCROLL_UP, IsKeyDown(KEY_PAGE_UP));
|
||
|
|
||
|
// Functions
|
||
|
if (IsKeyPressed(KEY_TAB)) nk_input_unicode(ctx, 9);
|
||
|
|
||
|
// Unicode
|
||
|
int code;
|
||
|
while ((code = GetCharPressed()) != 0)
|
||
|
nk_input_unicode(ctx, code);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Update the Nuklear context for the mouse input from raylib.
|
||
|
*
|
||
|
* @param ctx The nuklear context.
|
||
|
*
|
||
|
* @internal
|
||
|
*/
|
||
|
NK_API void
|
||
|
nk_raylib_input_mouse(struct nk_context * ctx)
|
||
|
{
|
||
|
const float scale = GetNuklearScaling(ctx);
|
||
|
const int mouseX = (int)((float)GetMouseX() / scale);
|
||
|
const int mouseY = (int)((float)GetMouseY() / scale);
|
||
|
|
||
|
nk_input_motion(ctx, mouseX, mouseY);
|
||
|
nk_input_button(ctx, NK_BUTTON_LEFT, mouseX, mouseY, IsMouseButtonDown(MOUSE_LEFT_BUTTON));
|
||
|
nk_input_button(ctx, NK_BUTTON_RIGHT, mouseX, mouseY, IsMouseButtonDown(MOUSE_RIGHT_BUTTON));
|
||
|
nk_input_button(ctx, NK_BUTTON_MIDDLE, mouseX, mouseY, IsMouseButtonDown(MOUSE_MIDDLE_BUTTON));
|
||
|
|
||
|
// Mouse Wheel
|
||
|
float mouseWheel = GetMouseWheelMove();
|
||
|
if (mouseWheel != 0.0f) {
|
||
|
struct nk_vec2 mouseWheelMove;
|
||
|
mouseWheelMove.x = 0.0f;
|
||
|
mouseWheelMove.y = mouseWheel;
|
||
|
nk_input_scroll(ctx, mouseWheelMove);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Update the Nuklear context for raylib's state.
|
||
|
*
|
||
|
* @param ctx The nuklear context to act upon.
|
||
|
*/
|
||
|
NK_API inline void
|
||
|
UpdateNuklear(struct nk_context * ctx)
|
||
|
{
|
||
|
UpdateNuklearEx(ctx, GetFrameTime());
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Update the Nuklear context for raylib's state.
|
||
|
*
|
||
|
* @param ctx The nuklear context to act upon.
|
||
|
* @param deltaTime Time in seconds since last frame.
|
||
|
*/
|
||
|
NK_API void
|
||
|
UpdateNuklearEx(struct nk_context * ctx, float deltaTime)
|
||
|
{
|
||
|
// Update the time that has changed since last frame.
|
||
|
ctx->delta_time_seconds = deltaTime;
|
||
|
|
||
|
// Update the input state.
|
||
|
nk_input_begin(ctx);
|
||
|
{
|
||
|
nk_raylib_input_mouse(ctx);
|
||
|
nk_raylib_input_keyboard(ctx);
|
||
|
}
|
||
|
nk_input_end(ctx);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Unload the given Nuklear context, along with all internal raylib textures.
|
||
|
*
|
||
|
* @param ctx The nuklear context.
|
||
|
*/
|
||
|
NK_API void
|
||
|
UnloadNuklear(struct nk_context * ctx)
|
||
|
{
|
||
|
struct nk_user_font* userFont;
|
||
|
|
||
|
// Skip unloading if it's not set.
|
||
|
if (ctx == NULL) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Unload the font.
|
||
|
userFont = (struct nk_user_font*)ctx->style.font;
|
||
|
if (userFont != NULL) {
|
||
|
// Clear the raylib Font object.
|
||
|
void* fontPtr = userFont->userdata.ptr;
|
||
|
if (fontPtr != NULL) {
|
||
|
MemFree(fontPtr);
|
||
|
}
|
||
|
|
||
|
// Clear the user font.
|
||
|
MemFree(userFont);
|
||
|
ctx->style.font = NULL;
|
||
|
}
|
||
|
|
||
|
// Unload the custom user data.
|
||
|
if (ctx->userdata.ptr != NULL) {
|
||
|
MemFree(ctx->userdata.ptr);
|
||
|
}
|
||
|
|
||
|
// Unload the nuklear context.
|
||
|
nk_free(ctx);
|
||
|
MemFree(ctx);
|
||
|
TraceLog(LOG_INFO, "NUKLEAR: Unloaded GUI");
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Convert the given Nuklear rectangle to a raylib Rectangle.
|
||
|
*/
|
||
|
NK_API struct
|
||
|
Rectangle RectangleFromNuklear(struct nk_context* ctx, struct nk_rect rect)
|
||
|
{
|
||
|
float scaling = GetNuklearScaling(ctx);
|
||
|
Rectangle output;
|
||
|
output.x = rect.x * scaling;
|
||
|
output.y = rect.y * scaling;
|
||
|
output.width = rect.w * scaling;
|
||
|
output.height = rect.h * scaling;
|
||
|
return output;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Convert the given raylib Rectangle to a Nuklear rectangle.
|
||
|
*/
|
||
|
NK_API struct
|
||
|
nk_rect RectangleToNuklear(struct nk_context* ctx, Rectangle rect)
|
||
|
{
|
||
|
float scaling = GetNuklearScaling(ctx);
|
||
|
return nk_rect(rect.x / scaling, rect.y / scaling, rect.width / scaling, rect.height / scaling);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Convert the given raylib texture to a Nuklear image
|
||
|
*/
|
||
|
NK_API struct nk_image
|
||
|
TextureToNuklear(Texture tex)
|
||
|
{
|
||
|
// Declare the img to store data and allocate memory
|
||
|
// For the texture
|
||
|
struct nk_image img;
|
||
|
struct Texture* stored_tex = (struct Texture*)MemAlloc(sizeof(Texture));
|
||
|
|
||
|
// Copy the data from the texture given into the new texture
|
||
|
stored_tex->id = tex.id;
|
||
|
stored_tex->width = tex.width;
|
||
|
stored_tex->height = tex.height;
|
||
|
stored_tex->mipmaps = tex.mipmaps;
|
||
|
stored_tex->format = tex.format;
|
||
|
|
||
|
// Initialize the nk_image struct
|
||
|
img.handle.ptr = stored_tex;
|
||
|
img.w = (nk_ushort)stored_tex->width;
|
||
|
img.h = (nk_ushort)stored_tex->height;
|
||
|
|
||
|
// Set the region so we can sub-select the image later.
|
||
|
img.region[0] = (nk_ushort)0;
|
||
|
img.region[1] = (nk_ushort)0;
|
||
|
img.region[2] = img.w;
|
||
|
img.region[3] = img.h;
|
||
|
|
||
|
return img;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Convert the given Nuklear image to a raylib Texture
|
||
|
*/
|
||
|
NK_API struct Texture
|
||
|
TextureFromNuklear(struct nk_image img)
|
||
|
{
|
||
|
// Declare texture for storage
|
||
|
// And get back the stored texture
|
||
|
Texture tex;
|
||
|
Texture* stored_tex = (Texture*)img.handle.ptr;
|
||
|
|
||
|
// Copy the data from the stored texture to the texture
|
||
|
tex.id = stored_tex->id;
|
||
|
tex.width = stored_tex->width;
|
||
|
tex.height = stored_tex->height;
|
||
|
tex.mipmaps = stored_tex->mipmaps;
|
||
|
tex.format = stored_tex->format;
|
||
|
|
||
|
return tex;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Load a Nuklear image directly
|
||
|
*
|
||
|
* @param path The path to the image
|
||
|
*/
|
||
|
NK_API struct nk_image
|
||
|
LoadNuklearImage(const char* path)
|
||
|
{
|
||
|
return TextureToNuklear(LoadTexture(path));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Unload a loaded Nuklear image
|
||
|
*
|
||
|
* @param img The Nuklear image to unload
|
||
|
*/
|
||
|
NK_API void
|
||
|
UnloadNuklearImage(struct nk_image img)
|
||
|
{
|
||
|
Texture tex = TextureFromNuklear(img);
|
||
|
UnloadTexture(tex);
|
||
|
CleanupNuklearImage(img);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Cleans up memory used by a Nuklear image
|
||
|
* Does not unload the image.
|
||
|
*
|
||
|
* @param img The Nuklear image to cleanup
|
||
|
*/
|
||
|
NK_API void
|
||
|
CleanupNuklearImage(struct nk_image img)
|
||
|
{
|
||
|
MemFree(img.handle.ptr);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Sets the scaling of the given Nuklear context.
|
||
|
*
|
||
|
* @param ctx The nuklear context.
|
||
|
* @param scaling How much scale to apply to the graphical user interface.
|
||
|
*/
|
||
|
NK_API void
|
||
|
SetNuklearScaling(struct nk_context * ctx, float scaling)
|
||
|
{
|
||
|
if (ctx == NULL) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (scaling <= 0.0f) {
|
||
|
TraceLog(LOG_WARNING, "NUKLEAR: Cannot set scaling to be less than 0");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
struct NuklearUserData* userData = (struct NuklearUserData*)ctx->userdata.ptr;
|
||
|
if (userData != NULL) {
|
||
|
userData->scaling = scaling;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieves the scale value of the given Nuklear context.
|
||
|
*
|
||
|
* @return The scale value that had been set for the Nuklear context. 1.0f is the default scale value.
|
||
|
*/
|
||
|
NK_API float
|
||
|
GetNuklearScaling(struct nk_context * ctx)
|
||
|
{
|
||
|
if (ctx == NULL) {
|
||
|
return 1.0f;
|
||
|
}
|
||
|
|
||
|
struct NuklearUserData* userData = (struct NuklearUserData*)ctx->userdata.ptr;
|
||
|
if (userData != NULL) {
|
||
|
return userData->scaling;
|
||
|
}
|
||
|
|
||
|
return 1.0f;
|
||
|
}
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#endif // RAYLIB_NUKLEAR_IMPLEMENTATION_ONCE
|
||
|
#endif // RAYLIB_NUKLEAR_IMPLEMENTATION
|