// Copyright 2012-2022 David Robillard // SPDX-License-Identifier: ISC #include "internal.h" #include "types.h" #include "pugl/pugl.h" #include #include #include #include void puglEnsureHint(PuglView* const view, const PuglViewHint hint, const int value) { if (view->hints[hint] == PUGL_DONT_CARE) { view->hints[hint] = value; } } PuglStatus puglSetBlob(PuglBlob* const dest, const void* const data, const size_t len) { if (data) { void* const newData = realloc(dest->data, len + 1); if (!newData) { free(dest->data); dest->len = 0; return PUGL_NO_MEMORY; } memcpy(newData, data, len); ((char*)newData)[len] = 0; dest->len = len; dest->data = newData; } else { dest->len = 0; dest->data = NULL; } return PUGL_SUCCESS; } void puglSetString(char** dest, const char* string) { if (*dest == string) { return; } const size_t len = string ? strlen(string) : 0U; if (!len) { free(*dest); *dest = NULL; } else { *dest = (char*)realloc(*dest, len + 1U); strncpy(*dest, string, len + 1U); } } uint32_t puglDecodeUTF8(const uint8_t* buf) { #define FAIL_IF(cond) \ do { \ if (cond) \ return 0xFFFD; \ } while (0) // http://en.wikipedia.org/wiki/UTF-8 if (buf[0] < 0x80) { return buf[0]; } if (buf[0] < 0xC2) { return 0xFFFD; } if (buf[0] < 0xE0) { FAIL_IF((buf[1] & 0xC0U) != 0x80); return ((uint32_t)buf[0] << 6U) + buf[1] - 0x3080U; } if (buf[0] < 0xF0) { FAIL_IF((buf[1] & 0xC0U) != 0x80); FAIL_IF(buf[0] == 0xE0 && buf[1] < 0xA0); FAIL_IF((buf[2] & 0xC0U) != 0x80); return ((uint32_t)buf[0] << 12U) + // ((uint32_t)buf[1] << 6U) + // ((uint32_t)buf[2] - 0xE2080U); } if (buf[0] < 0xF5) { FAIL_IF((buf[1] & 0xC0U) != 0x80); FAIL_IF(buf[0] == 0xF0 && buf[1] < 0x90); FAIL_IF(buf[0] == 0xF4 && buf[1] >= 0x90); FAIL_IF((buf[2] & 0xC0U) != 0x80U); FAIL_IF((buf[3] & 0xC0U) != 0x80U); return (((uint32_t)buf[0] << 18U) + // ((uint32_t)buf[1] << 12U) + // ((uint32_t)buf[2] << 6U) + // ((uint32_t)buf[3] - 0x3C82080U)); } return 0xFFFD; } static inline bool isValidSize(const double width, const double height) { return width > 0.0 && height > 0.0; } PuglStatus puglPreRealize(PuglView* const view) { // Ensure that a backend with at least a configure method has been set if (!view->backend || !view->backend->configure) { return PUGL_BAD_BACKEND; } // Ensure that the view has an event handler if (!view->eventFunc) { return PUGL_BAD_CONFIGURATION; } // Ensure that the default size is set to a valid size const PuglViewSize defaultSize = view->sizeHints[PUGL_DEFAULT_SIZE]; if (!isValidSize(defaultSize.width, defaultSize.height)) { return PUGL_BAD_CONFIGURATION; } return PUGL_SUCCESS; } PuglStatus puglDispatchSimpleEvent(PuglView* view, const PuglEventType type) { assert(type == PUGL_REALIZE || type == PUGL_UNREALIZE || type == PUGL_UPDATE || type == PUGL_CLOSE || type == PUGL_LOOP_ENTER || type == PUGL_LOOP_LEAVE); const PuglEvent event = {{type, 0}}; return puglDispatchEvent(view, &event); } static inline bool puglMustConfigure(PuglView* view, const PuglConfigureEvent* configure) { return !!memcmp(configure, &view->lastConfigure, sizeof(PuglConfigureEvent)); } PuglStatus puglConfigure(PuglView* view, const PuglEvent* event) { PuglStatus st = PUGL_SUCCESS; assert(event->type == PUGL_CONFIGURE); if (puglMustConfigure(view, &event->configure)) { st = view->eventFunc(view, event); view->lastConfigure = event->configure; } return st; } PuglStatus puglExpose(PuglView* view, const PuglEvent* event) { return (event->expose.width > 0.0 && event->expose.height > 0.0) ? view->eventFunc(view, event) : PUGL_SUCCESS; } PuglStatus puglDispatchEvent(PuglView* view, const PuglEvent* event) { PuglStatus st0 = PUGL_SUCCESS; PuglStatus st1 = PUGL_SUCCESS; switch (event->type) { case PUGL_NOTHING: break; case PUGL_REALIZE: assert(view->stage == PUGL_VIEW_STAGE_ALLOCATED); if (!(st0 = view->backend->enter(view, NULL))) { st0 = view->eventFunc(view, event); st1 = view->backend->leave(view, NULL); } view->stage = PUGL_VIEW_STAGE_REALIZED; break; case PUGL_UNREALIZE: assert(view->stage >= PUGL_VIEW_STAGE_REALIZED); if (!(st0 = view->backend->enter(view, NULL))) { st0 = view->eventFunc(view, event); st1 = view->backend->leave(view, NULL); } view->stage = PUGL_VIEW_STAGE_ALLOCATED; break; case PUGL_CONFIGURE: if (puglMustConfigure(view, &event->configure)) { if (!(st0 = view->backend->enter(view, NULL))) { st0 = puglConfigure(view, event); st1 = view->backend->leave(view, NULL); } } if (view->stage == PUGL_VIEW_STAGE_REALIZED) { view->stage = PUGL_VIEW_STAGE_CONFIGURED; } break; case PUGL_EXPOSE: assert(view->stage == PUGL_VIEW_STAGE_CONFIGURED); if (!(st0 = view->backend->enter(view, &event->expose))) { st0 = puglExpose(view, event); st1 = view->backend->leave(view, &event->expose); } break; default: st0 = view->eventFunc(view, event); } return st0 ? st0 : st1; }