diff --git a/apps/image_editor.cpp b/apps/image_editor.cpp index 72d3216..1051d26 100644 --- a/apps/image_editor.cpp +++ b/apps/image_editor.cpp @@ -716,7 +716,7 @@ void InstanceCreate(EsMessage *message) { EsButton *button; - EsToolbarAddFileMenu(toolbar); + EsFileMenuAddToToolbar(toolbar); button = EsButtonCreate(toolbar, ES_BUTTON_DROPDOWN, ES_STYLE_PUSH_BUTTON_TOOLBAR_BIG, INTERFACE_STRING(ImageEditorImage)); EsButtonSetIcon(button, ES_ICON_IMAGE_X_GENERIC); button->accessKey = 'I'; diff --git a/apps/text_editor.cpp b/apps/text_editor.cpp index 5c26fa4..36f7c73 100644 --- a/apps/text_editor.cpp +++ b/apps/text_editor.cpp @@ -281,7 +281,7 @@ void ProcessApplicationMessage(EsMessage *message) { EsElement *toolbarMain = instance->toolbarMain = EsWindowGetToolbar(window, true); - EsToolbarAddFileMenu(toolbarMain); + EsFileMenuAddToToolbar(toolbarMain); button = EsButtonCreate(toolbarMain, ES_FLAGS_DEFAULT, {}, INTERFACE_STRING(CommonSearchOpen)); button->accessKey = 'S'; diff --git a/desktop/api.cpp b/desktop/api.cpp index 164867a..614152e 100644 --- a/desktop/api.cpp +++ b/desktop/api.cpp @@ -703,11 +703,14 @@ void InstanceClose(EsInstance *instance) { EsDialogAddButton(dialog, ES_FLAGS_DEFAULT, ES_STYLE_PUSH_BUTTON_DANGEROUS, INTERFACE_STRING(FileCloseWithModificationsDelete), [] (EsInstance *instance, EsElement *, EsCommand *) { + EsDialogClose(instance->window->dialogs.Last()); EsInstanceDestroy(instance); }); EsButton *button = EsDialogAddButton(dialog, ES_BUTTON_DEFAULT, 0, INTERFACE_STRING(FileCloseWithModificationsSave), [] (EsInstance *instance, EsElement *, EsCommand *) { + EsDialogClose(instance->window->dialogs.Last()); + APIInstance *apiInstance = (APIInstance *) instance->_private; if (apiInstance->startupInformation->filePathBytes) { @@ -1288,7 +1291,7 @@ EsMessage *EsMessageReceive() { } } -void InstanceSetModified(EsInstance *instance, bool modified) { +void EsInstanceSetModified(EsInstance *instance, bool modified) { EsCommandSetEnabled(EsCommandByID(instance, ES_COMMAND_SAVE), modified); uint8_t m[2]; @@ -1330,7 +1333,7 @@ void EsInstanceOpenComplete(EsMessage *message, bool success, const char *errorT EsUndoClear(instance->undoManager); } - InstanceSetModified(instance, false); + EsInstanceSetModified(instance, false); } EsAssert(!message->instanceOpen.file->operationComplete); @@ -1359,7 +1362,7 @@ void EsInstanceSaveComplete(EsMessage *message, bool success) { MessageDesktop(buffer, 1, instance->window->handle); if (success) { - InstanceSetModified(instance, false); + EsInstanceSetModified(instance, false); EsRectangle bounds = EsElementGetWindowBounds(instance->window->toolbarSwitcher); size_t messageBytes; char *message = EsStringAllocateAndFormat(&messageBytes, "Saved to %s", // TODO Localization. @@ -1763,7 +1766,7 @@ void EsUndoPush(EsUndoManager *manager, EsUndoCallback callback, const void *ite } if (manager->instance->undoManager == manager) { - InstanceSetModified(manager->instance, true); + EsInstanceSetModified(manager->instance, true); } } diff --git a/desktop/desktop.cpp b/desktop/desktop.cpp index be7c76a..2bfa57d 100644 --- a/desktop/desktop.cpp +++ b/desktop/desktop.cpp @@ -1471,7 +1471,7 @@ bool ApplicationInstanceStart(int64_t applicationID, _EsApplicationStartupInform if (desktop.inShutdown) { return false; } - + InstalledApplication *application = nullptr; for (uintptr_t i = 0; i < desktop.installedApplications.Length(); i++) { @@ -1499,6 +1499,10 @@ bool ApplicationInstanceStart(int64_t applicationID, _EsApplicationStartupInform startupInformation = &_startupInformation; } + if (instance->tab) { + EsButtonSetCheck(instance->tab->closeButton, ES_CHECK_UNCHECKED, false); + } + if (instance->tab && instance->tab->notRespondingInstance) { ApplicationInstanceClose(instance->tab->notRespondingInstance); instance->tab->notRespondingInstance = nullptr; diff --git a/desktop/gui.cpp b/desktop/gui.cpp index a9320ef..a1241d8 100644 --- a/desktop/gui.cpp +++ b/desktop/gui.cpp @@ -5526,7 +5526,7 @@ void FileMenuRename(EsInstance *_instance, EsElement *, EsCommand *) { instance->fileMenuNameTextbox->messageUser = FileMenuNameTextboxMessage; } -void FileMenuCreate(EsInstance *_instance, EsElement *element, EsCommand *) { +void EsFileMenuCreate(EsInstance *_instance, EsElement *element, uint64_t menuFlags) { // TODO Make this user-customizable? // const EsFileMenuSettings *settings = (const EsFileMenuSettings *) element->userData.p; @@ -5535,7 +5535,7 @@ void FileMenuCreate(EsInstance *_instance, EsElement *element, EsCommand *) { EsInstanceClassEditorSettings *editorSettings = &instance->editorSettings; bool newDocument = !instance->startupInformation || !instance->startupInformation->filePath; - EsMenu *menu = EsMenuCreate(element, ES_FLAGS_DEFAULT); + EsMenu *menu = EsMenuCreate(element, menuFlags); if (!menu) return; EsPanel *panel1 = EsPanelCreate(menu, ES_PANEL_HORIZONTAL | ES_CELL_H_LEFT, &styleFileMenuDocumentInformationPanel1); if (!panel1) goto show; @@ -5611,7 +5611,11 @@ void FileMenuCreate(EsInstance *_instance, EsElement *element, EsCommand *) { show: EsMenuShow(menu); } -void EsToolbarAddFileMenu(EsElement *element, const EsFileMenuSettings *settings) { +void FileMenuCreate(EsInstance *_instance, EsElement *element, EsCommand *) { + EsFileMenuCreate(_instance, element); +} + +void EsFileMenuAddToToolbar(EsElement *element, const EsFileMenuSettings *settings) { EsButton *button = EsButtonCreate(element, ES_BUTTON_DROPDOWN, 0, INTERFACE_STRING(CommonFileMenu)); if (!button) return; button->accessKey = 'F'; diff --git a/desktop/os.header b/desktop/os.header index a97c298..455f959 100644 --- a/desktop/os.header +++ b/desktop/os.header @@ -2313,6 +2313,7 @@ function void EsInstanceSetClassViewer(ES_INSTANCE_TYPE *instance, const EsInsta function EsApplicationStartupRequest EsInstanceGetStartupRequest(ES_INSTANCE_TYPE *instance); function void EsInstanceOpenComplete(EsMessage *message, bool success, STRING errorText = BLANK_STRING); function void EsInstanceSaveComplete(EsMessage *message, bool success); +function void EsInstanceSetModified(ES_INSTANCE_TYPE *instance, bool modified); function EsError EsUserTaskStart(EsUserTaskCallback callback, EsGeneric data, STRING title, uint32_t iconID); function void EsUserTaskSetProgress(EsUserTask *task, double progress, EsFileOffsetDifference bytesPerSecond); // Set bytesPerSecond to -1 if not applicable. @@ -2389,7 +2390,8 @@ function EsDialog *EsDialogShow(EsWindow *window, STRING title, STRING content, function EsButton *EsDialogAddButton(EsDialog *dialog, uint64_t flags = ES_FLAGS_DEFAULT, EsStyle *style = ES_NULL, STRING label = BLANK_STRING, EsCommandCallback callback = ES_NULL); -function void EsToolbarAddFileMenu(EsElement *element, const EsFileMenuSettings *settings = ES_NULL); +function void EsFileMenuAddToToolbar(EsElement *toolbar, const EsFileMenuSettings *settings = ES_NULL); +function void EsFileMenuCreate(ES_INSTANCE_TYPE *instance, EsElement *parent, uint64_t flags = ES_FLAGS_DEFAULT); private function EsHandle _EsWindowGetHandle(EsWindow *window); private function void _EsUISetFont(EsFontFamily id); diff --git a/shared/common.cpp b/shared/common.cpp index 14e4a49..5698b4e 100644 --- a/shared/common.cpp +++ b/shared/common.cpp @@ -1323,7 +1323,7 @@ void EsMemoryZero(void *destination, size_t bytes) { } void EsMemoryMove(void *_start, void *_end, intptr_t amount, bool zeroEmptySpace) { - // TODO Prevent this from being optimised out in the kernel./ + // TODO Prevent this from being optimised out in the kernel. uint8_t *start = (uint8_t *) _start; uint8_t *end = (uint8_t *) _end; diff --git a/util/api_table.ini b/util/api_table.ini index 8aaa1c4..2581003 100644 --- a/util/api_table.ini +++ b/util/api_table.ini @@ -363,7 +363,6 @@ EsWindowAddSizeAlternative=361 EsMenuAddCommandsFromToolbar=362 EsPanelStartMovementAnimation=363 EsElementStartTransition=364 -EsToolbarAddFileMenu=365 EsFileWriteAllFromHandle=366 EsFileWriteAllGatherFromHandle=367 EsButtonSetIconFromBits=368 @@ -462,3 +461,6 @@ EsInstanceGetStartupRequest=460 EsImageDisplayPaint=461 EsElementGetInsetBounds=462 EsDialogShow=463 +EsInstanceSetModified=464 +EsFileMenuAddToToolbar=465 +EsFileMenuCreate=466 diff --git a/util/build.c b/util/build.c index f0d2275..4ae0b07 100644 --- a/util/build.c +++ b/util/build.c @@ -334,6 +334,7 @@ void BuildUtilities() { #ifndef __APPLE__ // Luigi doesn't support macOS. BUILD_UTILITY("config_editor", "-lX11 -Wno-unused-parameter", ""); #endif +#ifndef __APPLE__ // Luigi doesn't support macOS. BUILD_UTILITY("reflect_gen", "", "designer/"); if (CheckDependencies("Utilities.DesignerHeader")) { @@ -345,8 +346,7 @@ void BuildUtilities() { } } -#ifndef __APPLE__ // Luigi doesn't support macOS. - if (CheckDependencies("Utilities.Designer1") || CheckDependencies("Utilities.Designer2")) { + if (CheckDependencies("Utilities.OldDesigner1") || CheckDependencies("Utilities.OldDesigner2")) { if (!CallSystem("gcc -MMD -o bin/designer.o -c util/designer/designer.c -g -std=c2x -fsanitize=address " WARNING_FLAGS_C) && !CallSystem("gcc -MMD -o bin/designer_luigi.o -c util/designer/designer_luigi.c -g -std=c2x " WARNING_FLAGS_C) && !CallSystem("gcc -o bin/designer -g bin/designer.o bin/designer_luigi.o -lX11 -lm -fsanitize=address ")) { @@ -354,6 +354,12 @@ void BuildUtilities() { ParseDependencies("bin/designer_luigi.d", "Utilities.Designer2", false); } } + + if (CheckDependencies("Utilities.Designer")) { + if (!CallSystem("g++ -MMD -D UI_LINUX util/designer2.cpp -o bin/designer2 -g -lX11 -Wno-unused-parameter " WARNING_FLAGS)) { + ParseDependencies("bin/designer2.d", "Utilities.Designer", false); + } + } #endif } @@ -1174,6 +1180,9 @@ void DoCommand(const char *l) { } else if (0 == strcmp(l, "designer")) { BuildUtilities(); CallSystem("bin/designer \"res/Theme Source.dat\" \"res/Themes/Theme.dat\" \"res/Cursors.png\" \"desktop/styles.header\""); + } else if (0 == strcmp(l, "designer2")) { + BuildUtilities(); + CallSystem("bin/designer2"); } else if (0 == strcmp(l, "replace-many")) { forceRebuild = true; printf("Enter the name of the replacement file: "); diff --git a/util/designer2.cpp b/util/designer2.cpp new file mode 100644 index 0000000..1028941 --- /dev/null +++ b/util/designer2.cpp @@ -0,0 +1,1587 @@ +#define UI_IMPLEMENTATION +#define ES_CRT_WITHOUT_PREFIX +#include "luigi.h" +#ifndef OS_ESSENCE +#include +#include +#endif + +// x86_64-w64-mingw32-gcc -O3 -o bin/designer2.exe -D UI_WINDOWS util/designer2.cpp -DUNICODE -lgdi32 -luser32 -lkernel32 -Wl,--subsystem,windows -fno-exceptions -fno-rtti + +// Needed to replace the old designer: +// TODO Layers containing arrays. +// TODO Export. +// TODO Proper rendering using theme.cpp. +// TODO Implement remaining objects. +// TODO Import and reorganize old theming data. + +// Additional features: +// TODO Better color modification space -- HSLuv. +// TODO Selecting multiple objects. +// TODO Resizing objects? +// TODO Find object in graph by name. +// TODO Prototyping display. (Multiple instances of each object can be placed, resized and interacted with). +// TODO Running on Essence: x86_64-essence-g++ -o root/designer2 -D UI_ESSENCE util/designer2.cpp + +////////////////////////////////////////////////////////////// + +#ifndef OS_ESSENCE + +#define ES_TEXT_H_LEFT (1 << 0) +#define ES_TEXT_H_CENTER (1 << 1) +#define ES_TEXT_H_RIGHT (1 << 2) +#define ES_TEXT_V_TOP (1 << 3) +#define ES_TEXT_V_CENTER (1 << 4) +#define ES_TEXT_V_BOTTOM (1 << 5) +#define ES_TEXT_ELLIPSIS (1 << 6) +#define ES_TEXT_WRAP (1 << 7) + +#define ES_FONT_SANS (0xFFFF) +#define ES_FONT_SERIF (0xFFFE) +#define ES_FONT_MONOSPACED (0xFFFD) + +#define EsHeap void +#define EsHeapAllocate(a, b, c) calloc(1, (a)) +#define EsHeapFree(a, b, c) free((a)) +#define EsMemoryZero(a, b) memset((a), 0, (b)) +#define EsMemoryCopy memcpy +#define EsMemoryCopyReverse memmove +#define EsPanic(...) UI_ASSERT(false) + +#define ES_MEMORY_MOVE_BACKWARDS - + +void *EsHeapReallocate(void *oldAddress, size_t newAllocationSize, bool zeroNewSpace, EsHeap *) { + UI_ASSERT(!zeroNewSpace); + return realloc(oldAddress, newAllocationSize); +} + +void EsMemoryMove(void *_start, void *_end, intptr_t amount, bool zeroEmptySpace) { + uint8_t *start = (uint8_t *) _start; + uint8_t *end = (uint8_t *) _end; + if (end < start) return; + + if (amount > 0) { + EsMemoryCopyReverse(start + amount, start, end - start); + if (zeroEmptySpace) EsMemoryZero(start, amount); + } else if (amount < 0) { + EsMemoryCopy(start + amount, start, end - start); + if (zeroEmptySpace) EsMemoryZero(end + amount, -amount); + } +} + +#endif + +const char *cursorStrings[] = { + "Normal", + "Text", + "Resize vertical", + "Resize horizontal", + "Diagonal 1", + "Diagonal 2", + "Split vertical", + "Split horizontal", + "Hand hover", + "Hand drag", + "Hand point", + "Scroll up-left", + "Scroll up", + "Scroll up-right", + "Scroll left", + "Scroll center", + "Scroll right", + "Scroll down-left", + "Scroll down", + "Scroll down-right", + "Select lines", + "Drop text", + "Cross hair pick", + "Cross hair resize", + "Move hover", + "Move drag", + "Rotate hover", + "Rotate drag", + "Blank", +}; + +void BlendPixel(uint32_t *destinationPixel, uint32_t modified, bool fullAlpha) { + if ((modified & 0xFF000000) == 0xFF000000) { + *destinationPixel = modified; + } else if ((modified & 0xFF000000) == 0x00000000) { + } else if ((*destinationPixel & 0xFF000000) != 0xFF000000 && fullAlpha) { + uint32_t original = *destinationPixel; + uint32_t alpha1 = (modified & 0xFF000000) >> 24; + uint32_t alpha2 = 255 - alpha1; + uint32_t alphaD = (original & 0xFF000000) >> 24; + uint32_t alphaD2 = alphaD * alpha2; + uint32_t alphaOut = alpha1 + (alphaD2 >> 8); + + if (alphaOut) { + uint32_t m2 = alphaD2 / alphaOut; + uint32_t m1 = (alpha1 << 8) / alphaOut; + if (m2 == 0x100) m2--; + if (m1 == 0x100) m1--; + uint32_t r2 = m2 * ((original & 0x000000FF) >> 0); + uint32_t g2 = m2 * ((original & 0x0000FF00) >> 8); + uint32_t b2 = m2 * ((original & 0x00FF0000) >> 16); + uint32_t r1 = m1 * ((modified & 0x000000FF) >> 0); + uint32_t g1 = m1 * ((modified & 0x0000FF00) >> 8); + uint32_t b1 = m1 * ((modified & 0x00FF0000) >> 16); + uint32_t result = (alphaOut << 24) + | (0x00FF0000 & ((b1 + b2) << 8)) + | (0x0000FF00 & ((g1 + g2) << 0)) + | (0x000000FF & ((r1 + r2) >> 8)); + *destinationPixel = result; + } + } else { + uint32_t original = *destinationPixel; + uint32_t alpha1 = (modified & 0xFF000000) >> 24; + uint32_t alpha2 = 255 - alpha1; + uint32_t r2 = alpha2 * ((original & 0x000000FF) >> 0); + uint32_t g2 = alpha2 * ((original & 0x0000FF00) >> 8); + uint32_t b2 = alpha2 * ((original & 0x00FF0000) >> 16); + uint32_t r1 = alpha1 * ((modified & 0x000000FF) >> 0); + uint32_t g1 = alpha1 * ((modified & 0x0000FF00) >> 8); + uint32_t b1 = alpha1 * ((modified & 0x00FF0000) >> 16); + uint32_t result = 0xFF000000 | (0x00FF0000 & ((b1 + b2) << 8)) + | (0x0000FF00 & ((g1 + g2) << 0)) + | (0x000000FF & ((r1 + r2) >> 8)); + *destinationPixel = result; + } +} + +#include "../shared/array.cpp" + +////////////////////////////////////////////////////////////// + +#define UI_SIZE_CHECKBOX_BOX (14) +#define UI_SIZE_CHECKBOX_GAP (8) + +typedef struct UICheckbox { +#define UI_CHECKBOX_ALLOW_INDETERMINATE (1 << 0) + UIElement e; +#define UI_CHECK_UNCHECKED (0) +#define UI_CHECK_CHECKED (1) +#define UI_CHECK_INDETERMINATE (2) + uint8_t check; + char *label; + ptrdiff_t labelBytes; + void (*invoke)(void *cp); +} UICheckbox; + +UICheckbox *UICheckboxCreate(UIElement *parent, uint32_t flags, const char *label, ptrdiff_t labelBytes); + +int _UICheckboxMessage(UIElement *element, UIMessage message, int di, void *dp) { + UICheckbox *box = (UICheckbox *) element; + + if (message == UI_MSG_GET_HEIGHT) { + return UI_SIZE_BUTTON_HEIGHT * element->window->scale; + } else if (message == UI_MSG_GET_WIDTH) { + int labelSize = UIMeasureStringWidth(box->label, box->labelBytes); + return (labelSize + UI_SIZE_CHECKBOX_BOX + UI_SIZE_CHECKBOX_GAP) * element->window->scale; + } else if (message == UI_MSG_PAINT) { + UIPainter *painter = (UIPainter *) dp; + uint32_t color, textColor; + _UIButtonCalculateColors(element, &color, &textColor); + int midY = (element->bounds.t + element->bounds.b) / 2; + UIRectangle boxBounds = UI_RECT_4(element->bounds.l, element->bounds.l + UI_SIZE_CHECKBOX_BOX, + midY - UI_SIZE_CHECKBOX_BOX / 2, midY + UI_SIZE_CHECKBOX_BOX / 2); + UIDrawRectangle(painter, boxBounds, color, ui.theme.border, UI_RECT_1(1)); + UIDrawString(painter, UIRectangleAdd(boxBounds, UI_RECT_4(1, 0, 0, 0)), + box->check == UI_CHECK_CHECKED ? "*" : box->check == UI_CHECK_INDETERMINATE ? "-" : " ", -1, + textColor, UI_ALIGN_CENTER, NULL); + UIDrawString(painter, UIRectangleAdd(element->bounds, UI_RECT_4(UI_SIZE_CHECKBOX_BOX + UI_SIZE_CHECKBOX_GAP, 0, 0, 0)), + box->label, box->labelBytes, textColor, UI_ALIGN_LEFT, NULL); + } else if (message == UI_MSG_UPDATE) { + UIElementRepaint(element, NULL); + } else if (message == UI_MSG_DESTROY) { + UI_FREE(box->label); + } else if (message == UI_MSG_KEY_TYPED) { + UIKeyTyped *m = (UIKeyTyped *) dp; + + if (m->textBytes == 1 && m->text[0] == ' ') { + UIElementMessage(element, UI_MSG_CLICKED, 0, 0); + UIElementRepaint(element, NULL); + } + } else if (message == UI_MSG_CLICKED) { + box->check = (box->check + 1) % ((element->flags & UI_CHECKBOX_ALLOW_INDETERMINATE) ? 3 : 2); + UIElementRepaint(element, NULL); + if (box->invoke) box->invoke(element->cp); + } + + return 0; +} + +UICheckbox *UICheckboxCreate(UIElement *parent, uint32_t flags, const char *label, ptrdiff_t labelBytes) { + UICheckbox *box = (UICheckbox *) UIElementCreate(sizeof(UICheckbox), parent, flags | UI_ELEMENT_TAB_STOP, _UICheckboxMessage, "Checkbox"); + box->label = UIStringCopy(label, (box->labelBytes = labelBytes)); + return box; +} + +////////////////////////////////////////////////////////////// + +#ifdef OS_ESSENCE +EsFileStore *fileStore; + +const EsInstanceClassEditorSettings instanceClassEditorSettings = { + "untitled.designer", -1, + "New design", -1, + ES_ICON_TEXT_CSS, +}; +#endif + +UIWindow *window; +UIElement *canvas; +UIElement *inspector; +UIElement *canvasControls; +UILabel *labelMessage; + +uint64_t selectedObjectID; + +void InspectorAnnouncePropertyChanged(uint64_t objectID, const char *cPropertyName); +void InspectorPopulate(); +void InspectorPickTargetEnd(); +void CanvasSelectObject(struct Object *object); + +////////////////////////////////////////////////////////////// + +enum PropertyType : uint8_t { + PROP_NONE, + PROP_COLOR, + PROP_INT, + PROP_OBJECT, +}; + +struct Property { + PropertyType type; +#define PROPERTY_NAME_SIZE (31) + char cName[PROPERTY_NAME_SIZE]; + + union { + int32_t integer; + uint64_t object; + }; +}; + +enum ObjectType : uint8_t { + OBJ_NONE, + + OBJ_STYLE, + + OBJ_VAR_COLOR = 0x40, + OBJ_VAR_INT, + OBJ_VAR_TEXT_STYLE, + OBJ_VAR_ICON_STYLE, + + OBJ_PAINT_OVERWRITE = 0x60, + // OBJ_PAINT_LINEAR_GRADIENT, + // OBJ_PAINT_RADIAL_GRADIENT, + + OBJ_LAYER_BOX = 0x80, + OBJ_LAYER_METRICS, + OBJ_LAYER_TEXT, + // OBJ_LAYER_PATH, + // OBJ_LAYER_GROUP, + // OBJ_LAYER_SEQUENCE, + // OBJ_LAYER_SELECTOR, + + OBJ_MOD_COLOR = 0xC0, + OBJ_MOD_MULTIPLY, +}; + +struct Object { + ObjectType type; +#define OBJECT_NAME_SIZE (47) + char cName[OBJECT_NAME_SIZE]; + uint64_t id; + Array properties; +}; + +enum StepType : uint8_t { + STEP_GROUP_MARKER, + STEP_MODIFY_PROPERTY, + STEP_RENAME_OBJECT, + STEP_ADD_OBJECT, + STEP_DELETE_OBJECT, +}; + +enum StepApplyMode { + STEP_APPLY_NORMAL, + STEP_APPLY_GROUPED, + STEP_APPLY_UNDO, + STEP_APPLY_REDO, +}; + +struct Step { + StepType type; + uint64_t objectID; + + union { + Property property; + char cName[OBJECT_NAME_SIZE]; + Object object; + }; +}; + +Array undoStack; +Array redoStack; +bool documentModified; + +// Document state: +Array objects; +uint64_t objectIDAllocator; + +Object *ObjectFind(uint64_t id) { + // TODO Use a hash table. + + for (uintptr_t i = 0; i < objects.Length(); i++) { + if (objects[i].id == id) { + return &objects[i]; + } + } + + return nullptr; +} + +Property *PropertyFind(Object *object, const char *cName, uint8_t type = 0) { + if (object) { + for (uintptr_t i = 0; i < object->properties.Length(); i++) { + if (0 == strcmp(object->properties[i].cName, cName)) { + if (type && object->properties[i].type != type) { + return nullptr; + } else { + return &object->properties[i]; + } + } + } + } + + return nullptr; +} + +Property *PropertyFindOrInherit(Object *object, const char *cName, uint8_t type = 0) { + while (object) { + Property *property = PropertyFind(object, cName); + + if (property) { + return type && property->type != type ? nullptr : property; + } + + property = PropertyFind(object, "_parent", PROP_OBJECT); + object = ObjectFind(property ? property->object : 0); + } + + return nullptr; +} + +int32_t PropertyReadInt32(Object *object, const char *cName, int32_t defaultValue = 0) { + Property *property = PropertyFind(object, cName); + return !property || property->type == PROP_OBJECT ? defaultValue : property->integer; +} + +void DocumentSave(void *) { +#ifdef OS_ESSENCE + EsBuffer buffer = { .canGrow = 1 }; +#define fwrite(a, b, c, d) EsBufferWrite(&buffer, (a), (b) * (c)) +#else + FILE *f = fopen("bin/designer2.dat", "wb"); +#endif + + uint32_t version = 1; + fwrite(&version, 1, sizeof(uint32_t), f); + uint32_t objectCount = objects.Length(); + fwrite(&objectCount, 1, sizeof(uint32_t), f); + fwrite(&objectIDAllocator, 1, sizeof(uint64_t), f); + + for (uintptr_t i = 0; i < objects.Length(); i++) { + Object copy = objects[i]; + uint32_t propertyCount = copy.properties.Length(); + copy.properties.array = nullptr; + fwrite(©, 1, sizeof(Object), f); + fwrite(&propertyCount, 1, sizeof(uint32_t), f); + fwrite(objects[i].properties.array, 1, sizeof(Property) * propertyCount, f); + } + +#ifdef OS_ESSENCE + EsFileStoreWriteAll(fileStore, buffer.out, buffer.position); + EsHeapFree(buffer.out); +#else + fclose(f); +#endif + documentModified = false; +} + +void DocumentLoad() { +#ifdef OS_ESSENCE + EsBuffer buffer = {}; + buffer.out = (uint8_t *) EsFileStoreReadAll(fileStore, &buffer.bytes); +#define fread(a, b, c, d) EsBufferReadInto(&buffer, (a), (b) * (c)) +#else + FILE *f = fopen("bin/designer2.dat", "rb"); + if (!f) return; +#endif + + uint32_t version = 1; + fread(&version, 1, sizeof(uint32_t), f); + uint32_t objectCount = 0; + fread(&objectCount, 1, sizeof(uint32_t), f); + fread(&objectIDAllocator, 1, sizeof(uint64_t), f); + + for (uintptr_t i = 0; i < objectCount; i++) { + Object object = {}; + fread(&object, 1, sizeof(Object), f); + uint32_t propertyCount = 0; + fread(&propertyCount, 1, sizeof(uint32_t), f); + object.properties.InsertMany(0, propertyCount); + fread(object.properties.array, 1, sizeof(Property) * propertyCount, f); + objects.Add(object); + } + +#ifdef OS_ESSENCE + EsHeapFree(buffer.out); +#else + fclose(f); +#endif +} + +void DocumentFree() { + for (uintptr_t i = 0; i < objects.Length(); i++) { + objects[i].properties.Free(); + } + + objects.Free(); +} + +void DocumentApplyStep(Step step, StepApplyMode mode = STEP_APPLY_NORMAL) { + bool allowMerge = false; + + if (step.type == STEP_GROUP_MARKER) { + } else if (step.type == STEP_MODIFY_PROPERTY) { + Object *object = ObjectFind(step.objectID); + UI_ASSERT(object); + + Property *property = PropertyFind(object, step.property.cName); + + if (property) { + Property oldValue = *property; + + if (step.property.type != PROP_NONE) { + // Update the property. + *property = step.property; + allowMerge = true; + } else { + // Remove the property. + for (uintptr_t i = 0; i < object->properties.Length(); i++) { + if (0 == strcmp(object->properties[i].cName, step.property.cName)) { + object->properties.DeleteSwap(i); + break; + } + } + } + + step.property = oldValue; + } else { + if (step.property.type != PROP_NONE) { + // Add the property. + object->properties.Add(step.property); + step.property.type = PROP_NONE; + } else { + // Asking to remove a property that does not exist. + // Probably from a remove broadcast. + } + } + + UIElementRepaint(canvas, nullptr); + InspectorAnnouncePropertyChanged(step.objectID, step.property.cName); + } else if (step.type == STEP_RENAME_OBJECT) { + Object *object = ObjectFind(step.objectID); + UI_ASSERT(object); + + char oldName[OBJECT_NAME_SIZE]; + strcpy(oldName, object->cName); + strcpy(object->cName, step.cName); + strcpy(step.cName, oldName); + + UIElementRepaint(canvas, nullptr); + InspectorPopulate(); + } else if (step.type == STEP_ADD_OBJECT) { + objects.Add(step.object); + selectedObjectID = step.object.id; + UIElementRepaint(canvas, nullptr); + step.objectID = step.object.id; + step.type = STEP_DELETE_OBJECT; + InspectorPopulate(); + } else if (step.type == STEP_DELETE_OBJECT) { + if (selectedObjectID == step.objectID) selectedObjectID = 0; + step.type = STEP_ADD_OBJECT; + + for (uintptr_t i = 0; i < objects.Length(); i++) { + if (objects[i].id == step.objectID) { + step.object = objects[i]; + objects.DeleteSwap(i); + break; + } + } + + UIElementRepaint(canvas, nullptr); + InspectorPopulate(); + } else { + UI_ASSERT(false); + } + + if (mode == STEP_APPLY_NORMAL || mode == STEP_APPLY_GROUPED) { + bool merge = false; + + if (allowMerge && undoStack.Length() > 2 && !redoStack.Length()) { + Step last = undoStack[undoStack.Length() - 2]; + + if (step.type == STEP_MODIFY_PROPERTY && last.type == STEP_MODIFY_PROPERTY + && last.objectID == step.objectID && 0 == strcmp(last.property.cName, step.property.cName)) { + merge = true; + } + } + + if (!merge) { + undoStack.Add(step); + + for (uintptr_t i = 0; i < redoStack.Length(); i++) { + if (redoStack[i].type == STEP_ADD_OBJECT) { + redoStack[i].object.properties.Free(); + } + } + + redoStack.Free(); + + if (mode != STEP_APPLY_GROUPED) { + Step step = {}; + step.type = STEP_GROUP_MARKER; + undoStack.Add(step); + } + } + } else if (mode == STEP_APPLY_UNDO) { + redoStack.Add(step); + } else if (mode == STEP_APPLY_REDO) { + undoStack.Add(step); + } + + documentModified = true; + InspectorPickTargetEnd(); + +#ifdef UI_ESSENCE + EsInstanceSetModified(ui.instance, true); +#endif +} + +void DocumentUndoStep(void *) { + if (!undoStack.Length()) return; + + Step marker = undoStack.Pop(); + UI_ASSERT(marker.type == STEP_GROUP_MARKER); + + do { + DocumentApplyStep(undoStack.Pop(), STEP_APPLY_UNDO); + } while (undoStack.Length() && undoStack.Last().type != STEP_GROUP_MARKER); + + redoStack.Add(marker); +} + +void DocumentRedoStep(void *) { + if (!redoStack.Length()) return; + + Step marker = redoStack.Pop(); + UI_ASSERT(marker.type == STEP_GROUP_MARKER); + + do { + DocumentApplyStep(redoStack.Pop(), STEP_APPLY_REDO); + } while (redoStack.Length() && redoStack.Last().type != STEP_GROUP_MARKER); + + undoStack.Add(marker); +} + +////////////////////////////////////////////////////////////// + +enum InspectorElementType { + INSPECTOR_INVALID_ELEMENT, + INSPECTOR_REMOVE_BUTTON, + INSPECTOR_REMOVE_BUTTON_BROADCAST, + INSPECTOR_COLOR_PICKER, + INSPECTOR_COLOR_TEXTBOX, + INSPECTOR_INTEGER_TEXTBOX, + INSPECTOR_INTEGER_TEXTBOX_BROADCAST, + INSPECTOR_LINK, + INSPECTOR_LINK_BROADCAST, + INSPECTOR_BOOLEAN_TOGGLE, + INSPECTOR_RADIO_SWITCH, + INSPECTOR_CURSOR_DROP_DOWN, +}; + +struct InspectorBindingData { + UIElement *element; + uint64_t objectID; + const char *cPropertyName; + const char *cEnablePropertyName; + InspectorElementType elementType; + int32_t radioValue; +}; + +Array inspectorBoundElements; +UIElement *inspectorActivePropertyStepElement; +InspectorBindingData *inspectorMenuData; +InspectorBindingData *inspectorPickData; + +void InspectorPickTargetEnd() { + if (inspectorPickData) { + inspectorPickData = nullptr; + UILabelSetContent(labelMessage, "", -1); + UIElementRefresh(&labelMessage->e); + UIElementRepaint(canvas, nullptr); + } +} + +void InspectorUpdateSingleElementEnable(InspectorBindingData *data) { + UI_ASSERT(data->cEnablePropertyName); + bool enabled = PropertyReadInt32(ObjectFind(data->objectID), data->cEnablePropertyName); + if (enabled) data->element->flags &= ~UI_ELEMENT_DISABLED; + else data->element->flags |= UI_ELEMENT_DISABLED; + UIElementRefresh(data->element); +} + +void InspectorUpdateSingleElement(InspectorBindingData *data) { + if (data->elementType == INSPECTOR_REMOVE_BUTTON || data->elementType == INSPECTOR_REMOVE_BUTTON_BROADCAST) { + UIButton *button = (UIButton *) data->element; + Property *property = PropertyFind(ObjectFind(data->objectID), data->cPropertyName); + if (property) button->e.flags &= ~UI_ELEMENT_DISABLED; + else button->e.flags |= UI_ELEMENT_DISABLED; + button->label[0] = property ? 'X' : '-'; + UIElementRefresh(&button->e); + } else if (data->elementType == INSPECTOR_BOOLEAN_TOGGLE) { + UICheckbox *box = (UICheckbox *) data->element; + box->check = PropertyReadInt32(ObjectFind(data->objectID), data->cPropertyName, 2); + UIElementRefresh(&box->e); + } else if (data->elementType == INSPECTOR_RADIO_SWITCH) { + UIButton *button = (UIButton *) data->element; + int32_t value = PropertyReadInt32(ObjectFind(data->objectID), data->cPropertyName); + if (value == data->radioValue) button->e.flags |= UI_BUTTON_CHECKED; + else button->e.flags &= ~UI_BUTTON_CHECKED; + UIElementRefresh(&button->e); + } else if (data->elementType == INSPECTOR_CURSOR_DROP_DOWN) { + UIButton *button = (UIButton *) data->element; + Property *property = PropertyFind(ObjectFind(data->objectID), data->cPropertyName, PROP_INT); + UI_FREE(button->label); + button->label = UIStringCopy(property ? cursorStrings[property->integer] : "---", (button->labelBytes = -1)); + UIElementRefresh(&button->e); + } else if (data->elementType == INSPECTOR_COLOR_PICKER) { + UIColorPicker *colorPicker = (UIColorPicker *) data->element; + Property *property = PropertyFind(ObjectFind(data->objectID), data->cPropertyName, PROP_COLOR); + uint32_t color = property ? property->integer : 0xFFFFFFFF; + colorPicker->opacity = (color >> 24) / 255.0f; + UIColorToHSV(color, &colorPicker->hue, &colorPicker->saturation, &colorPicker->value); + UIElementRefresh(&colorPicker->e); + } else if (data->elementType == INSPECTOR_COLOR_TEXTBOX) { + UITextbox *textbox = (UITextbox *) data->element; + Property *property = PropertyFind(ObjectFind(data->objectID), data->cPropertyName, PROP_COLOR); + char buffer[32] = ""; + if (property) snprintf(buffer, sizeof(buffer), "%.8X", (uint32_t) property->integer); + UITextboxClear(textbox, false); + UITextboxReplace(textbox, buffer, -1, false); + UIElementRefresh(&textbox->e); + } else if (data->elementType == INSPECTOR_INTEGER_TEXTBOX || data->elementType == INSPECTOR_INTEGER_TEXTBOX_BROADCAST) { + UITextbox *textbox = (UITextbox *) data->element; + Property *property = PropertyFind(ObjectFind(data->objectID), data->cPropertyName, PROP_INT); + char buffer[32] = ""; + if (property) snprintf(buffer, sizeof(buffer), "%d", property->integer); + UITextboxClear(textbox, false); + UITextboxReplace(textbox, buffer, -1, false); + UIElementRefresh(&textbox->e); + } else if (data->elementType == INSPECTOR_LINK || data->elementType == INSPECTOR_LINK_BROADCAST) { + UIButton *button = (UIButton *) data->element; + Property *property = PropertyFind(ObjectFind(data->objectID), data->cPropertyName, PROP_OBJECT); + Object *target = ObjectFind(property ? property->object : 0); + const char *string = target ? target->cName : "---"; + UI_FREE(button->label); + button->label = UIStringCopy(string, (button->labelBytes = -1)); + UIElementRefresh(&button->e); + } else { + UI_ASSERT(false); + } +} + +void InspectorAnnouncePropertyChanged(uint64_t objectID, const char *cPropertyName) { + for (uintptr_t i = 0; i < inspectorBoundElements.Length(); i++) { + InspectorBindingData *data = (InspectorBindingData *) inspectorBoundElements[i]->cp; + if (data->element == inspectorActivePropertyStepElement) continue; + if (data->objectID != objectID) continue; + + if (data->cEnablePropertyName && 0 == strcmp(data->cEnablePropertyName, cPropertyName)) { + InspectorUpdateSingleElementEnable(data); + } + + if (0 == strcmp(data->cPropertyName, cPropertyName)) { + InspectorUpdateSingleElement(data); + } + } +} + +void InspectorBroadcastStep(Step step, InspectorBindingData *data) { + if (data->elementType == INSPECTOR_INTEGER_TEXTBOX_BROADCAST || data->elementType == INSPECTOR_LINK_BROADCAST + || data->elementType == INSPECTOR_REMOVE_BUTTON_BROADCAST) { + for (char i = '1'; i <= '3'; i++) { + step.property.cName[strlen(step.property.cName) - 1] = i; + DocumentApplyStep(step, STEP_APPLY_GROUPED); + } + + strcpy(step.property.cName, data->cPropertyName); + } +} + +void InspectorCursorDropDownMenuInvoke(void *cp) { + intptr_t index = (intptr_t) cp; + Step step = {}; + step.type = STEP_MODIFY_PROPERTY; + step.objectID = inspectorMenuData->objectID; + strcpy(step.property.cName, inspectorMenuData->cPropertyName); + step.property.type = index == -1 ? PROP_NONE : PROP_INT; + step.property.integer = index; + DocumentApplyStep(step); + inspectorMenuData = nullptr; +} + +int InspectorBoundMessage(UIElement *element, UIMessage message, int di, void *dp) { + InspectorBindingData *data = (InspectorBindingData *) element->cp; + + if (message == UI_MSG_DESTROY) { + inspectorBoundElements.FindAndDeleteSwap(element, true); + free(data); + data = nullptr; + } else if (message == UI_MSG_VALUE_CHANGED) { + Step step = {}; + step.type = STEP_MODIFY_PROPERTY; + step.objectID = data->objectID; + strcpy(step.property.cName, data->cPropertyName); + + if (data->elementType == INSPECTOR_COLOR_PICKER) { + UIColorPicker *colorPicker = (UIColorPicker *) element; + uint32_t color; + UIColorToRGB(colorPicker->hue, colorPicker->saturation, colorPicker->value, &color); + color |= (uint32_t) (colorPicker->opacity * 255.0f) << 24; + step.property.type = PROP_COLOR; + step.property.integer = (int32_t) color; + } else if (data->elementType == INSPECTOR_COLOR_TEXTBOX) { + UITextbox *textbox = (UITextbox *) element; + char buffer[32]; + int length = 31 > textbox->bytes ? textbox->bytes : 31; + memcpy(buffer, textbox->string, length); + buffer[length] = 0; + step.property.type = PROP_COLOR; + step.property.integer = (int32_t) strtoul(buffer, nullptr, 16); + } else if (data->elementType == INSPECTOR_INTEGER_TEXTBOX + || data->elementType == INSPECTOR_INTEGER_TEXTBOX_BROADCAST) { + UITextbox *textbox = (UITextbox *) element; + char buffer[32]; + int length = 31 > textbox->bytes ? textbox->bytes : 31; + memcpy(buffer, textbox->string, length); + buffer[length] = 0; + step.property.type = PROP_INT; + step.property.integer = (int32_t) strtol(buffer, nullptr, 0); + InspectorBroadcastStep(step, data); + } else { + UI_ASSERT(false); + } + + inspectorActivePropertyStepElement = element; // Don't tell this element about the step. + DocumentApplyStep(step); + inspectorActivePropertyStepElement = nullptr; + } else if (message == UI_MSG_CLICKED) { + Step step = {}; + step.type = STEP_MODIFY_PROPERTY; + step.objectID = data->objectID; + strcpy(step.property.cName, data->cPropertyName); + + if (data->elementType == INSPECTOR_REMOVE_BUTTON || data->elementType == INSPECTOR_REMOVE_BUTTON_BROADCAST) { + step.property.type = PROP_NONE; // Remove the property. + InspectorBroadcastStep(step, data); + DocumentApplyStep(step); + } else if (data->elementType == INSPECTOR_LINK || data->elementType == INSPECTOR_LINK_BROADCAST) { + char *name = nullptr; + const char *dialog = "Enter the name of the new link target object:\n%t\n\n%l\n\n%f%b%b"; + const char *result = UIDialogShow(window, 0, dialog, &name, "OK", "Cancel"); + + if (0 == strcmp(result, "OK")) { + uint64_t id = 0; + + for (uintptr_t i = 0; i < objects.Length(); i++) { + if (0 == strcmp(objects[i].cName, name)) { + id = objects[i].id; + break; + } + } + + if (!id) { + UIDialogShow(window, 0, "Error: The object was not found.\n%f%b", "OK"); + } else { + step.property.type = PROP_OBJECT; + step.property.object = id; + InspectorBroadcastStep(step, data); + DocumentApplyStep(step); + } + } + + free(name); + } else if (data->elementType == INSPECTOR_RADIO_SWITCH) { + step.property.type = PROP_INT; + step.property.integer = data->radioValue; + DocumentApplyStep(step); + } else if (data->elementType == INSPECTOR_BOOLEAN_TOGGLE) { + UICheckbox *box = (UICheckbox *) element; + step.property.type = (box->check + 1) == UI_CHECK_INDETERMINATE ? PROP_NONE : PROP_INT; + step.property.integer = (box->check + 1) % 3; + DocumentApplyStep(step); + return 1; // InspectorUpdateSingleElement will update the check. + } else if (data->elementType == INSPECTOR_CURSOR_DROP_DOWN) { + UIMenu *menu = UIMenuCreate(window->pressed, UI_MENU_NO_SCROLL); + UIMenuAddItem(menu, 0, "Inherit", -1, InspectorCursorDropDownMenuInvoke, (void *) (intptr_t) -1); + + for (uintptr_t i = 0; i < sizeof(cursorStrings) / sizeof(cursorStrings[0]); i++) { + UIMenuAddItem(menu, 0, cursorStrings[i], -1, InspectorCursorDropDownMenuInvoke, (void *) (intptr_t) i); + } + + inspectorMenuData = data; + UIMenuShow(menu); + } + } else if (message == UI_MSG_UPDATE) { + if (di == UI_UPDATE_FOCUSED && element->window->focused == element + && (data->elementType == INSPECTOR_COLOR_TEXTBOX || data->elementType == INSPECTOR_INTEGER_TEXTBOX + || data->elementType == INSPECTOR_INTEGER_TEXTBOX_BROADCAST)) { + UITextbox *textbox = (UITextbox *) element; + textbox->carets[0] = 0; + textbox->carets[1] = textbox->bytes; + } + } + + return 0; +} + +InspectorBindingData *InspectorBind(UIElement *element, uint64_t objectID, const char *cPropertyName, InspectorElementType elementType, + int32_t radioValue = 0, const char *cEnablePropertyName = nullptr) { + InspectorBindingData *data = (InspectorBindingData *) calloc(1, sizeof(InspectorBindingData)); + data->element = element; + data->objectID = objectID; + data->cPropertyName = cPropertyName; + data->elementType = elementType; + data->radioValue = radioValue; + data->cEnablePropertyName = cEnablePropertyName; + element->cp = data; + element->messageUser = InspectorBoundMessage; + inspectorBoundElements.Add(element); + InspectorUpdateSingleElement(data); + if (cEnablePropertyName) InspectorUpdateSingleElementEnable(data); + return data; +} + +void InspectorRenameObject(void *) { + Step step = {}; + step.type = STEP_RENAME_OBJECT; + step.objectID = selectedObjectID; + + char *name = nullptr; + const char *result = UIDialogShow(window, 0, "Enter the new name for the object:\n%t\n\n%l\n\n%f%b%b", &name, "OK", "Cancel"); + + if (0 == strcmp(result, "OK")) { + if (!name || strlen(name) >= sizeof(step.cName) - 1) { + UIDialogShow(window, 0, "Error: Name must be between 1 and 46 characters.\n%f%b", "OK"); + } else { + strcpy(step.cName, name); + DocumentApplyStep(step); + } + } + + free(name); +} + +void InspectorPickTargetCommand(void *cp) { + if (inspectorPickData) { + InspectorPickTargetEnd(); + return; + } + + inspectorPickData = (InspectorBindingData *) cp; + UILabelSetContent(labelMessage, "** Click an object to link it. **", -1); + UIElementRefresh(&labelMessage->e); + UIElementRepaint(canvas, nullptr); +} + +void InspectorFindTargetCommand(void *cp) { + InspectorBindingData *data = (InspectorBindingData *) cp; + Property *property = PropertyFind(ObjectFind(data->objectID), data->cPropertyName, PROP_OBJECT); + Object *target = ObjectFind(property ? property->object : 0); + + if (target) { + CanvasSelectObject(target); + } else { + UIDialogShow(window, 0, "Error: The object does not exist.\n%f%b", "OK"); + } +} + +int InspectorTabPaneMessage(UIElement *element, UIMessage message, int di, void *dp) { + if (message == UI_MSG_LEFT_DOWN) { + element->messageClass(element, message, di, dp); + UIElementRefresh(inspector); + return 1; + } + + return 0; +} + +void InspectorAddLink(Object *object, const char *cLabel, const char *cPropertyName, + bool broadcast = false, const char *cEnablePropertyName = nullptr) { + if (cLabel) UILabelCreate(0, 0, cLabel, -1); + UIElement *row = &UIPanelCreate(0, UI_PANEL_HORIZONTAL)->e; + InspectorBindingData *data = InspectorBind(&UIButtonCreate(row, UI_ELEMENT_H_FILL, 0, 0)->e, + object->id, cPropertyName, broadcast ? INSPECTOR_LINK_BROADCAST : INSPECTOR_LINK, 0, cEnablePropertyName); + UIButton *pickTarget = UIButtonCreate(row, UI_BUTTON_SMALL, "Pick", -1); + pickTarget->e.cp = data; + pickTarget->invoke = InspectorPickTargetCommand; + UIButton *findTarget = UIButtonCreate(row, UI_BUTTON_SMALL, "Find", -1); + findTarget->e.cp = data; + findTarget->invoke = InspectorFindTargetCommand; + if (!broadcast) InspectorBind(&UIButtonCreate(row, UI_BUTTON_SMALL, "X", 1)->e, object->id, cPropertyName, INSPECTOR_REMOVE_BUTTON); +} + +void InspectorAddIntegerTextbox(Object *object, const char *cLabel, const char *cPropertyName, bool broadcast = false, const char *cEnablePropertyName = nullptr) { + if (cLabel) UILabelCreate(0, 0, cLabel, -1); + InspectorBind(&UITextboxCreate(0, UI_ELEMENT_H_FILL)->e, object->id, cPropertyName, + broadcast ? INSPECTOR_INTEGER_TEXTBOX_BROADCAST : INSPECTOR_INTEGER_TEXTBOX, 0, cEnablePropertyName); +} + +void InspectorAddInteger(Object *object, const char *cLabel, const char *cPropertyName) { + if (cLabel) UILabelCreate(0, 0, cLabel, -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_ELEMENT_H_FILL | UI_PANEL_HORIZONTAL); + UITabPane *tabPane = UITabPaneCreate(0, UI_ELEMENT_PARENT_PUSH | UI_ELEMENT_H_FILL, "Direct\tLink"); + tabPane->e.messageUser = InspectorTabPaneMessage; + InspectorAddIntegerTextbox(object, nullptr, cPropertyName); + InspectorAddLink(object, nullptr, cPropertyName); + UIParentPop(); + Property *property = PropertyFind(object, cPropertyName); + if (property && property->type == PROP_OBJECT) tabPane->active = 1; + InspectorBind(&UIButtonCreate(0, UI_BUTTON_SMALL, "X", 1)->e, object->id, cPropertyName, INSPECTOR_REMOVE_BUTTON); + UIParentPop(); +} + +void InspectorAddFourGroup(Object *object, const char *cLabel, const char *cPropertyName0, const char *cPropertyName1, + const char *cPropertyName2, const char *cPropertyName3, const char *cEnablePropertyName = nullptr) { + if (cLabel) UILabelCreate(0, 0, cLabel, -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_ELEMENT_H_FILL | UI_PANEL_HORIZONTAL); + UITabPane *tabPane = UITabPaneCreate(0, UI_ELEMENT_PARENT_PUSH | UI_ELEMENT_H_FILL, "Single\tIndividual\tLink"); + tabPane->e.messageUser = InspectorTabPaneMessage; + InspectorAddIntegerTextbox(object, nullptr, cPropertyName0, true /* broadcast */, cEnablePropertyName); + UIElement *row = &UIPanelCreate(0, UI_PANEL_HORIZONTAL)->e; + InspectorBind(&UITextboxCreate(row, UI_ELEMENT_H_FILL)->e, object->id, cPropertyName0, INSPECTOR_INTEGER_TEXTBOX, 0, cEnablePropertyName); + InspectorBind(&UITextboxCreate(row, UI_ELEMENT_H_FILL)->e, object->id, cPropertyName1, INSPECTOR_INTEGER_TEXTBOX, 0, cEnablePropertyName); + InspectorBind(&UITextboxCreate(row, UI_ELEMENT_H_FILL)->e, object->id, cPropertyName2, INSPECTOR_INTEGER_TEXTBOX, 0, cEnablePropertyName); + InspectorBind(&UITextboxCreate(row, UI_ELEMENT_H_FILL)->e, object->id, cPropertyName3, INSPECTOR_INTEGER_TEXTBOX, 0, cEnablePropertyName); + InspectorAddLink(object, nullptr, cPropertyName0, true /* broadcast */, cEnablePropertyName); + UIParentPop(); + Property *property = PropertyFind(object, cPropertyName0); + int32_t b0 = PropertyReadInt32(object, cPropertyName0); + int32_t b1 = PropertyReadInt32(object, cPropertyName1); + int32_t b2 = PropertyReadInt32(object, cPropertyName2); + int32_t b3 = PropertyReadInt32(object, cPropertyName3); + if (property && property->type == PROP_OBJECT) tabPane->active = 2; + else if (b0 != b1 || b1 != b2 || b2 != b3) tabPane->active = 1; + InspectorBind(&UIButtonCreate(0, UI_BUTTON_SMALL, "X", 1)->e, object->id, cPropertyName0, INSPECTOR_REMOVE_BUTTON_BROADCAST); + UIParentPop(); +} + +void InspectorAddBooleanToggle(Object *object, const char *cLabel, const char *cPropertyName) { + InspectorBind(&UICheckboxCreate(0, UI_CHECKBOX_ALLOW_INDETERMINATE, cLabel, -1)->e, object->id, cPropertyName, INSPECTOR_BOOLEAN_TOGGLE); +} + +void InspectorAddRadioSwitch(Object *object, const char *cLabel, const char *cPropertyName, int32_t radioValue) { + InspectorBind(&UIButtonCreate(0, UI_BUTTON_SMALL, cLabel, -1)->e, object->id, cPropertyName, INSPECTOR_RADIO_SWITCH, radioValue); +} + +void InspectorPopulate() { + UIElementDestroyDescendents(inspector); + UIParentPush(inspector); + + Object *object = ObjectFind(selectedObjectID); + + if (object) { + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_BORDER | UI_PANEL_MEDIUM_SPACING | UI_PANEL_EXPAND); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL); + char buffer[256]; + snprintf(buffer, sizeof(buffer), "%s (%lu)", object->cName, object->id); + UILabelCreate(0, 0, buffer, -1); + UISpacerCreate(0, UI_ELEMENT_H_FILL, 0, 0); + UIButtonCreate(0, 0, "Rename", -1)->invoke = InspectorRenameObject; + UIParentPop(); + + if (object->type != OBJ_VAR_COLOR && object->type != OBJ_VAR_INT + && object->type != OBJ_MOD_COLOR && object->type != OBJ_MOD_MULTIPLY) { + InspectorAddLink(object, "Inherit from:", "_parent"); + } + UIParentPop(); + UISpacerCreate(0, 0, 0, 10); + + if (object->type == OBJ_STYLE) { + InspectorAddLink(object, "Appearance:", "appearance"); + InspectorAddLink(object, "Metrics:", "metrics"); + InspectorAddLink(object, "Text style:", "textStyle"); + InspectorAddLink(object, "Icon style:", "iconStyle"); + InspectorAddBooleanToggle(object, "Public style", "isPublic"); + } else if (object->type == OBJ_VAR_COLOR) { + InspectorBind(&UIColorPickerCreate(&UIPanelCreate(0, 0)->e, UI_COLOR_PICKER_HAS_OPACITY)->e, object->id, "color", INSPECTOR_COLOR_PICKER); + InspectorBind(&UITextboxCreate(0, 0)->e, object->id, "color", INSPECTOR_COLOR_TEXTBOX); + InspectorAddBooleanToggle(object, "Export to theme file", "isExported"); + } else if (object->type == OBJ_VAR_INT) { + InspectorBind(&UITextboxCreate(0, 0)->e, object->id, "value", INSPECTOR_INTEGER_TEXTBOX); + InspectorAddBooleanToggle(object, "Export to theme file", "isExported"); + InspectorAddBooleanToggle(object, "Apply UI scaling factor", "isScaled"); + } else if (object->type == OBJ_LAYER_METRICS) { + UILabelCreate(0, 0, "General options:", -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL); + InspectorAddBooleanToggle(object, "Enable clipping", "clipEnabled"); + InspectorAddBooleanToggle(object, "Wrap text", "wrapText"); + InspectorAddBooleanToggle(object, "Ellipsis", "ellipsis"); + UIParentPop(); + + InspectorAddFourGroup(object, "Insets:", "insets0", "insets1", "insets2", "insets3"); + InspectorAddFourGroup(object, "Clip insets:", "clipInsets0", "clipInsets1", "clipInsets2", "clipInsets3", "clipEnabled"); + + UILabelCreate(0, 0, "Preferred size:", -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL); + InspectorAddInteger(object, nullptr, "preferredWidth"); + InspectorAddInteger(object, nullptr, "preferredHeight"); + UIParentPop(); + + UILabelCreate(0, 0, "Minimum size:", -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL); + InspectorAddInteger(object, nullptr, "minimumWidth"); + InspectorAddInteger(object, nullptr, "minimumHeight"); + UIParentPop(); + + UILabelCreate(0, 0, "Maximum size:", -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL); + InspectorAddInteger(object, nullptr, "maximumWidth"); + InspectorAddInteger(object, nullptr, "maximumHeight"); + UIParentPop(); + + UILabelCreate(0, 0, "Gaps:", -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL); + InspectorAddInteger(object, nullptr, "gapMajor"); + InspectorAddInteger(object, nullptr, "gapMinor"); + InspectorAddInteger(object, nullptr, "gapWrap"); + UIParentPop(); + + UILabelCreate(0, 0, "Horizontal text alignment:", -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL); + InspectorAddRadioSwitch(object, "Left", "horizontalTextAlign", 1); + InspectorAddRadioSwitch(object, "Center", "horizontalTextAlign", 2); + InspectorAddRadioSwitch(object, "Right", "horizontalTextAlign", 3); + InspectorBind(&UIButtonCreate(0, UI_BUTTON_SMALL, "X", 1)->e, object->id, "horizontalTextAlign", INSPECTOR_REMOVE_BUTTON); + UIParentPop(); + + UILabelCreate(0, 0, "Vertical text alignment:", -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL); + InspectorAddRadioSwitch(object, "Top", "verticalTextAlign", 1); + InspectorAddRadioSwitch(object, "Center", "verticalTextAlign", 2); + InspectorAddRadioSwitch(object, "Bottom", "verticalTextAlign", 3); + InspectorBind(&UIButtonCreate(0, UI_BUTTON_SMALL, "X", 1)->e, object->id, "verticalTextAlign", INSPECTOR_REMOVE_BUTTON); + UIParentPop(); + + UILabelCreate(0, 0, "Cursor style:", -1); + InspectorBind(&UIButtonCreate(0, UI_BUTTON_DROP_DOWN, 0, 0)->e, object->id, "cursor", INSPECTOR_CURSOR_DROP_DOWN); + } else if (object->type == OBJ_VAR_TEXT_STYLE) { + InspectorAddLink(object, "Text color:", "textColor"); + InspectorAddLink(object, "Selection background color:", "selectedBackground"); + InspectorAddLink(object, "Selection text color:", "selectedText"); + InspectorAddInteger(object, "Text size:", "textSize"); + InspectorAddInteger(object, "Font weight:", "fontWeight"); + + UILabelCreate(0, 0, "Font options:", -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL); + InspectorAddBooleanToggle(object, "Italic", "isItalic"); + UIParentPop(); + + UILabelCreate(0, 0, "Font family:", -1); + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL); + InspectorAddRadioSwitch(object, "Sans-serif", "fontFamily", ES_FONT_SANS); + InspectorAddRadioSwitch(object, "Serif", "fontFamily", ES_FONT_SERIF); + InspectorAddRadioSwitch(object, "Monospaced", "fontFamily", ES_FONT_MONOSPACED); + InspectorBind(&UIButtonCreate(0, UI_BUTTON_SMALL, "X", 1)->e, object->id, "fontFamily", INSPECTOR_REMOVE_BUTTON); + UIParentPop(); + } else if (object->type == OBJ_VAR_ICON_STYLE) { + InspectorAddLink(object, "Icon color:", "iconColor"); + InspectorAddInteger(object, "Icon size:", "iconSize"); + } else if (object->type == OBJ_LAYER_BOX) { + InspectorAddFourGroup(object, "Border sizes:", "borders0", "borders1", "borders2", "borders3"); + InspectorAddLink(object, "Fill paint:", "mainPaint"); + InspectorAddLink(object, "Border paint:", "borderPaint"); + } else if (object->type == OBJ_LAYER_TEXT) { + InspectorAddLink(object, "Text color:", "color"); + InspectorAddInteger(object, "Blur radius:", "blur"); + } else if (object->type == OBJ_PAINT_OVERWRITE) { + InspectorAddLink(object, "Color:", "color"); + } else if (object->type == OBJ_MOD_COLOR) { + InspectorAddLink(object, "Base color:", "base"); + UILabelCreate(0, 0, "Brightness:", -1); + InspectorBind(&UITextboxCreate(0, UI_ELEMENT_H_FILL)->e, object->id, "brightness", INSPECTOR_INTEGER_TEXTBOX); + } else if (object->type == OBJ_MOD_MULTIPLY) { + InspectorAddLink(object, "Base integer:", "base"); + UILabelCreate(0, 0, "Factor (%):", -1); + InspectorBind(&UITextboxCreate(0, UI_ELEMENT_H_FILL)->e, object->id, "factor", INSPECTOR_INTEGER_TEXTBOX); + } else { + // TODO. + } + } else { + UILabelCreate(0, 0, "Select an object to inspect.", -1); + } + + UIParentPop(); + UIElementRefresh(inspector); +} + +////////////////////////////////////////////////////////////// + +#define CANVAS_ALIGN (20) + +float canvasPanX, canvasPanY; +float canvasLastPanPointX, canvasLastPanPointY; +bool canvasDragging, canvasCanDrag; +int32_t canvasDragNewX, canvasDragNewY; +int32_t canvasDragOffsetX, canvasDragOffsetY; +int32_t canvasLeftDownX, canvasLeftDownY; +bool canvasShowArrows; + +UIRectangle CanvasGetObjectBounds(Object *object) { + int32_t x = PropertyReadInt32(object, "_graphX") - canvasPanX + canvas->bounds.l; + int32_t y = PropertyReadInt32(object, "_graphY") - canvasPanY + canvas->bounds.t; + int32_t w = PropertyReadInt32(object, "_graphW"); + int32_t h = PropertyReadInt32(object, "_graphH"); + + if (object->id == selectedObjectID && canvasDragging) { + x = canvasDragNewX - canvasPanX + canvas->bounds.l; + y = canvasDragNewY - canvasPanY + canvas->bounds.t; + } + + return UI_RECT_4(x, x + w, y, y + h); +} + +void CanvasSelectObject(Object *object) { + UIRectangle bounds = CanvasGetObjectBounds(object); + canvasPanX += bounds.l - UI_RECT_WIDTH(canvas->bounds) / 2; + canvasPanY += bounds.t - UI_RECT_HEIGHT(canvas->bounds) / 2; + selectedObjectID = object->id; + UIElementRepaint(canvas, nullptr); + InspectorPopulate(); +} + +void CanvasDrawColorSwatch(UIPainter *painter, UIRectangle bounds, uint32_t color) { + uint32_t colors[2] = { 0xFFFFFFFF, 0xFFC0C0C0 }; + BlendPixel(&colors[0], color, false); + BlendPixel(&colors[1], color, false); + int x = bounds.l, y = bounds.t; + bounds = UIRectangleIntersection(bounds, painter->clip); + + if (UI_RECT_VALID(bounds)) { + for (int32_t j = bounds.t; j < bounds.b; j++) { + for (int32_t i = bounds.l; i < bounds.r; i++) { + painter->bits[j * painter->width + i] = colors[(((j - y) >> 3) ^ ((i - x) >> 3)) & 1]; + } + } + } +} + +void CanvasDrawArrow(UIPainter *painter, int x0, int y0, int x1, int y1, uint32_t color) { + if (!UIDrawLine(painter, x0, y0, x1, y1, color)) return; + float angle = atan2f(y1 - y0, x1 - x0); + UIDrawLine(painter, x0, y0, x0 + cosf(angle + 0.5f) * 15, y0 + sinf(angle + 0.5f) * 15, color); + UIDrawLine(painter, x0, y0, x0 + cosf(angle - 0.5f) * 15, y0 + sinf(angle - 0.5f) * 15, color); +} + +int32_t CanvasGetInteger(Object *object, int depth = 0) { + if (!object || depth == 10) { + return 0; + } + + if (object->type == OBJ_VAR_INT) { + return PropertyReadInt32(object, "value"); + } else if (object->type == OBJ_MOD_MULTIPLY) { + Property *property = PropertyFind(object, "base", PROP_OBJECT); + int32_t base = CanvasGetInteger(ObjectFind(property ? property->object : 0), depth + 1); + int32_t factor = PropertyReadInt32(object, "factor"); + return base * factor / 100; + } else { + return 0; + } +} + +int32_t CanvasGetIntegerFromProperty(Property *property) { + if (!property) { + return 0; + } else if (property->type == PROP_INT) { + return property->integer; + } else if (property->type == PROP_OBJECT) { + return CanvasGetInteger(ObjectFind(property->object)); + } else { + return 0; + } +} + +uint32_t CanvasGetColorFromPaint(Object *object, int depth = 0) { + if (!object || depth == 10) { + return 0; + } + + if (object->type == OBJ_VAR_COLOR) { + return PropertyReadInt32(object, "color"); + } else if (object->type == OBJ_MOD_COLOR) { + Property *property = PropertyFind(object, "base", PROP_OBJECT); + uint32_t base = CanvasGetColorFromPaint(ObjectFind(property ? property->object : 0), depth + 1); + uint32_t alpha = base & 0xFF000000; + int32_t brightness = PropertyReadInt32(object, "brightness"); + float hue, saturation, value; + UIColorToHSV(base, &hue, &saturation, &value); + value += brightness / 100.0f; + if (value < 0) value = 0; + if (value > 1) value = 1; + UIColorToRGB(hue, saturation, value, &base); + return base | alpha; + } else { + return 0; + } +} + +int CanvasMessage(UIElement *element, UIMessage message, int di, void *dp) { + if (message == UI_MSG_PAINT) { + UIPainter *painter = (UIPainter *) dp; + UIDrawBlock(painter, element->bounds, 0xFFC0C0C0); + + for (uintptr_t i = 0; i < objects.Length(); i++) { + Object *object = &objects[i]; + UIRectangle bounds = CanvasGetObjectBounds(object); + + if (bounds.r < element->bounds.l || bounds.l > element->bounds.r + || bounds.b < element->bounds.t || bounds.t > element->bounds.b) { + continue; + } + + if ((object->id == selectedObjectID) == (inspectorPickData == nullptr)) { + UIDrawBorder(painter, UIRectangleAdd(bounds, UI_RECT_1I(-3)), 0xFF4092FF, UI_RECT_1(3)); + } + + UIDrawString(painter, UI_RECT_4(bounds.l, element->bounds.r, bounds.t - ui.glyphHeight, bounds.t), + object->cName, -1, 0xFF000000, UI_ALIGN_LEFT, nullptr); + + UIDrawRectangle(painter, bounds, 0xFFE0E0E0, 0xFF404040, UI_RECT_1(1)); + UIDrawBlock(painter, UI_RECT_4(bounds.l + 1, bounds.r + 1, bounds.b, bounds.b + 1), 0xFF404040); + UIDrawBlock(painter, UI_RECT_4(bounds.r, bounds.r + 1, bounds.t + 1, bounds.b + 1), 0xFF404040); + + bounds = UIRectangleAdd(bounds, UI_RECT_1I(3)); + + if (object->type == OBJ_VAR_COLOR || object->type == OBJ_MOD_COLOR) { + CanvasDrawColorSwatch(painter, bounds, CanvasGetColorFromPaint(object)); + } else if (object->type == OBJ_VAR_INT || object->type == OBJ_MOD_MULTIPLY) { + int32_t value = CanvasGetInteger(object); + char buffer[32]; + snprintf(buffer, sizeof(buffer), "%d", value); + UIDrawString(painter, bounds, buffer, -1, 0xFF000000, UI_ALIGN_CENTER, nullptr); + } else if (object->type == OBJ_VAR_TEXT_STYLE) { + // TODO Proper rendering. + UIDrawString(painter, bounds, "Text", -1, 0xFF000000, UI_ALIGN_CENTER, nullptr); + } else if (object->type == OBJ_LAYER_BOX) { + // TODO Proper rendering. + + UIRectangle borders; + borders.l = CanvasGetIntegerFromProperty(PropertyFindOrInherit(object, "borders0")); + borders.r = CanvasGetIntegerFromProperty(PropertyFindOrInherit(object, "borders1")); + borders.t = CanvasGetIntegerFromProperty(PropertyFindOrInherit(object, "borders2")); + borders.b = CanvasGetIntegerFromProperty(PropertyFindOrInherit(object, "borders3")); + + Property *mainPaintProperty = PropertyFindOrInherit(object, "mainPaint", PROP_OBJECT); + Object *mainPaint = ObjectFind(mainPaintProperty ? mainPaintProperty->object : 0); + uint32_t mainPaintColor = CanvasGetColorFromPaint(mainPaint); + + Property *borderPaintProperty = PropertyFindOrInherit(object, "borderPaint", PROP_OBJECT); + Object *borderPaint = ObjectFind(borderPaintProperty ? borderPaintProperty->object : 0); + uint32_t borderPaintColor = CanvasGetColorFromPaint(borderPaint); + + UIDrawRectangle(painter, bounds, mainPaintColor, borderPaintColor, borders); + } else { + // TODO. + } + } + + if (canvasShowArrows) { + // Draw object connections. + // TODO This will be awfully slow when there's many objects... + + for (uintptr_t i = 0; i < objects.Length(); i++) { + Object *object = &objects[i]; + UIRectangle b1 = CanvasGetObjectBounds(object); + + for (uintptr_t j = 0; j < object->properties.Length(); j++) { + if (object->properties[j].type == PROP_OBJECT) { + Object *target = ObjectFind(object->properties[j].object); + if (!target) continue; + UIRectangle b2 = CanvasGetObjectBounds(target); + CanvasDrawArrow(painter, (b2.l + b2.r) / 2, (b2.t + b2.b) / 2, (b1.l + b1.r) / 2, (b1.t + b1.b) / 2, 0xFF000000); + } + } + } + } + } else if (message == UI_MSG_LEFT_DOWN) { + canvasCanDrag = true; + selectedObjectID = 0; + + for (uintptr_t i = objects.Length(); i > 0; i--) { + Object *object = &objects[i - 1]; + UIRectangle bounds = CanvasGetObjectBounds(object); + + if (UIRectangleContains(bounds, element->window->cursorX, element->window->cursorY)) { + if (inspectorPickData) { + selectedObjectID = inspectorPickData->objectID; + canvasCanDrag = false; + + Step step = {}; + step.type = STEP_MODIFY_PROPERTY; + step.objectID = inspectorPickData->objectID; + strcpy(step.property.cName, inspectorPickData->cPropertyName); + step.property.type = PROP_OBJECT; + step.property.object = object->id; + InspectorBroadcastStep(step, inspectorPickData); + DocumentApplyStep(step); + } else { + selectedObjectID = object->id; + canvasDragOffsetX = bounds.l - element->window->cursorX; + canvasDragOffsetY = bounds.t - element->window->cursorY; + } + + break; + } + } + + canvasLeftDownX = element->window->cursorX; + canvasLeftDownY = element->window->cursorY; + + UIElementRepaint(element, nullptr); + InspectorPopulate(); + InspectorPickTargetEnd(); + } else if (message == UI_MSG_LEFT_UP && canvasDragging) { + Object *object = ObjectFind(selectedObjectID); + int32_t oldX = PropertyReadInt32(object, "_graphX"); + int32_t oldY = PropertyReadInt32(object, "_graphY"); + + if ((oldX != canvasDragNewX || oldY != canvasDragNewY) && selectedObjectID) { + Step step = {}; + step.type = STEP_MODIFY_PROPERTY; + step.objectID = selectedObjectID; + step.property.type = PROP_INT; + strcpy(step.property.cName, "_graphX"); + step.property.integer = canvasDragNewX; + DocumentApplyStep(step, STEP_APPLY_GROUPED); + strcpy(step.property.cName, "_graphY"); + step.property.integer = canvasDragNewY; + DocumentApplyStep(step); + } + + canvasDragging = false; + UIElementRepaint(element, nullptr); + } else if (message == UI_MSG_MOUSE_DRAG && element->window->pressedButton == 1 && selectedObjectID && canvasCanDrag) { + int32_t dx = canvasLeftDownX - element->window->cursorX; + int32_t dy = canvasLeftDownY - element->window->cursorY; + + if (canvasDragging || dx * dx + dy * dy > 200) { + canvasDragNewX = element->window->cursorX + canvasPanX + canvasDragOffsetX - element->bounds.l; + canvasDragNewY = element->window->cursorY + canvasPanY + canvasDragOffsetY - element->bounds.t; + canvasDragNewX -= canvasDragNewX % CANVAS_ALIGN, canvasDragNewY -= canvasDragNewY % CANVAS_ALIGN; + canvasDragging = true; + UIElementRepaint(element, nullptr); + } + } else if (message == UI_MSG_MIDDLE_DOWN) { + canvasLastPanPointX = element->window->cursorX; + canvasLastPanPointY = element->window->cursorY; + _UIWindowSetCursor(element->window, UI_CURSOR_HAND); + } else if (message == UI_MSG_MIDDLE_UP) { + _UIWindowSetCursor(element->window, UI_CURSOR_ARROW); + } else if (message == UI_MSG_MOUSE_DRAG && element->window->pressedButton == 2) { + canvasPanX -= element->window->cursorX - canvasLastPanPointX; + canvasPanY -= element->window->cursorY - canvasLastPanPointY; + canvasLastPanPointX = element->window->cursorX; + canvasLastPanPointY = element->window->cursorY; + UIElementRepaint(element, nullptr); + } else if (message == UI_MSG_LAYOUT) { + int width = UIElementMessage(canvasControls, UI_MSG_GET_WIDTH, 0, 0); + int height = UIElementMessage(canvasControls, UI_MSG_GET_HEIGHT, 0, 0); + UIRectangle bounds = UI_RECT_4(element->bounds.l + 10, element->bounds.l + 10 + width, element->bounds.b - 10 - height, element->bounds.b - 10); + UIElementMove(canvasControls, bounds, false); + } + + return 0; +} + +void CanvasToggleArrows(void *) { + canvasShowArrows = !canvasShowArrows; + UIElementRepaint(canvas, nullptr); +} + +////////////////////////////////////////////////////////////// + +void ObjectAddCommandInternal(void *cp) { + Object object = {}; + object.type = (ObjectType) (uintptr_t) cp; + strcpy(object.cName, "untitled"); + object.id = ++objectIDAllocator; + Property p; + int32_t x = canvasPanX + UI_RECT_WIDTH(canvas->bounds) / 2; + int32_t y = canvasPanY + UI_RECT_HEIGHT(canvas->bounds) / 2; + x -= x % CANVAS_ALIGN, y -= y % CANVAS_ALIGN; + p = { .type = PROP_INT, .integer = x }; strcpy(p.cName, "_graphX"); object.properties.Add(p); + p = { .type = PROP_INT, .integer = y }; strcpy(p.cName, "_graphY"); object.properties.Add(p); + p = { .type = PROP_INT, .integer = 60 }; strcpy(p.cName, "_graphW"); object.properties.Add(p); + p = { .type = PROP_INT, .integer = 60 }; strcpy(p.cName, "_graphH"); object.properties.Add(p); + + Step step = {}; + step.type = STEP_ADD_OBJECT; + step.object = object; + DocumentApplyStep(step); +} + +void ObjectAddCommand(void *) { + UIMenu *menu = UIMenuCreate(window->pressed, UI_MENU_NO_SCROLL); + UIMenuAddItem(menu, 0, "Style", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_STYLE); + UIMenuAddItem(menu, 0, "Color variable", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_VAR_COLOR); + UIMenuAddItem(menu, 0, "Integer variable", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_VAR_INT); + UIMenuAddItem(menu, 0, "Text style", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_VAR_TEXT_STYLE); + UIMenuAddItem(menu, 0, "Icon style", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_VAR_ICON_STYLE); + UIMenuAddItem(menu, 0, "Metrics", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_LAYER_METRICS); + UIMenuAddItem(menu, 0, "Overwrite paint", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_PAINT_OVERWRITE); + UIMenuAddItem(menu, 0, "Box layer", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_LAYER_BOX); + UIMenuAddItem(menu, 0, "Text layer", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_LAYER_TEXT); + UIMenuAddItem(menu, 0, "Modify color", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_MOD_COLOR); + UIMenuAddItem(menu, 0, "Modify integer", -1, ObjectAddCommandInternal, (void *) (uintptr_t) OBJ_MOD_MULTIPLY); + UIMenuShow(menu); +} + +void ObjectDeleteCommand(void *) { + if (!selectedObjectID) return; + Step step = {}; + step.type = STEP_DELETE_OBJECT; + step.objectID = selectedObjectID; + DocumentApplyStep(step); +} + +void ObjectDuplicateCommand(void *) { + if (!selectedObjectID) return; + + Object *source = ObjectFind(selectedObjectID); + UI_ASSERT(source); + + Object object = {}; + object.type = source->type; + strcpy(object.cName, "duplicate"); + object.id = ++objectIDAllocator; + object.properties.InsertMany(0, source->properties.Length()); + memcpy(object.properties.array, source->properties.array, source->properties.Length() * sizeof(Property)); + + Property *graphX = PropertyFind(&object, "_graphX", PROP_INT); + if (graphX) graphX->integer += 60; + + Step step = {}; + step.type = STEP_ADD_OBJECT; + step.object = object; + DocumentApplyStep(step); +} + +////////////////////////////////////////////////////////////// + +int WindowMessage(UIElement *element, UIMessage message, int di, void *dp) { + if (message == UI_MSG_WINDOW_CLOSE) { +#ifndef OS_ESSENCE + if (documentModified) { + const char *dialog = "Document modified. Save changes?\n%f%b%b"; + const char *result = UIDialogShow(window, 0, dialog, "Save", "Discard"); + + if (0 == strcmp(result, "Save")) { + DocumentSave(nullptr); + } + } +#endif + } + + return 0; +} + +#ifdef OS_ESSENCE +void DocumentFileMenu(void *) { + EsFileMenuCreate(ui.instance, ui.instance->window, ES_MENU_AT_CURSOR); +} +#endif + +int main() { + UIInitialise(); + ui.theme = _uiThemeClassic; + window = UIWindowCreate(0, UI_ELEMENT_PARENT_PUSH | UI_WINDOW_MAXIMIZE, "Designer", 0, 0); + window->e.messageUser = WindowMessage; + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_EXPAND); + + UIPanelCreate(0, UI_ELEMENT_PARENT_PUSH | UI_PANEL_HORIZONTAL | UI_PANEL_GRAY | UI_PANEL_SMALL_SPACING); +#ifdef OS_ESSENCE + UIButtonCreate(0, UI_BUTTON_DROP_DOWN, "File", -1)->invoke = DocumentFileMenu; +#else + UIButtonCreate(0, 0, "Save", -1)->invoke = DocumentSave; +#endif + UIButtonCreate(0, 0, "Add object", -1)->invoke = ObjectAddCommand; + UISpacerCreate(0, 0, 15, 0); + labelMessage = UILabelCreate(0, UI_ELEMENT_H_FILL, 0, 0); + UIParentPop(); + + UISpacerCreate(0, UI_SPACER_LINE, 0, 1); + + UISplitPaneCreate(0, UI_ELEMENT_PARENT_PUSH | UI_ELEMENT_V_FILL, 0.75f); + canvas = UIElementCreate(sizeof(UIElement), 0, 0, CanvasMessage, "Canvas"); + inspector = &UIPanelCreate(0, UI_PANEL_GRAY | UI_PANEL_MEDIUM_SPACING | UI_PANEL_SCROLL | UI_PANEL_EXPAND)->e; + InspectorPopulate(); + + canvasControls = &UIPanelCreate(canvas, UI_PANEL_HORIZONTAL | UI_ELEMENT_PARENT_PUSH)->e; + UIButtonCreate(0, 0, "Toggle arrows", -1)->invoke = CanvasToggleArrows; + UIParentPop(); + + UIWindowRegisterShortcut(window, UI_SHORTCUT(UI_KEYCODE_LETTER('Z'), 1 /* ctrl */, 0, 0, DocumentUndoStep, 0)); + UIWindowRegisterShortcut(window, UI_SHORTCUT(UI_KEYCODE_LETTER('Y'), 1 /* ctrl */, 0, 0, DocumentRedoStep, 0)); + UIWindowRegisterShortcut(window, UI_SHORTCUT(UI_KEYCODE_LETTER('D'), 1 /* ctrl */, 0, 0, ObjectDuplicateCommand, 0)); + UIWindowRegisterShortcut(window, UI_SHORTCUT(UI_KEYCODE_DELETE, 0, 0, 0, ObjectDeleteCommand, 0)); + +#ifdef OS_ESSENCE + EsWindowSetIcon(ui.instance->window, ES_ICON_APPLICATIONS_INTERFACEDESIGN); + EsInstanceSetClassEditor(ui.instance, &instanceClassEditorSettings); +#else + UIWindowRegisterShortcut(window, UI_SHORTCUT(UI_KEYCODE_LETTER('S'), 1 /* ctrl */, 0, 0, DocumentSave, 0)); + DocumentLoad(); +#endif + + int result = UIMessageLoop(); + DocumentFree(); + return result; +} + +#ifdef OS_ESSENCE +void _UIMessageProcess(EsMessage *message) { + if (message->type == ES_MSG_INSTANCE_OPEN) { + DocumentFree(); + fileStore = message->instanceOpen.file; + DocumentLoad(); + fileStore = nullptr; + EsInstanceOpenComplete(message, true); + } else if (message->type == ES_MSG_INSTANCE_SAVE) { + fileStore = message->instanceSave.file; + DocumentSave(nullptr); + fileStore = nullptr; + EsInstanceSaveComplete(message, true); + } +} + +void _start() { + _init(); + main(); +} +#endif diff --git a/util/designer2.ini b/util/designer2.ini new file mode 100644 index 0000000..eb3d506 --- /dev/null +++ b/util/designer2.ini @@ -0,0 +1,16 @@ +[general] +name=Designer +icon=icon_applications_interfacedesign + +[build] +source=util/designer2.cpp +compile_flags=-D UI_ESSENCE -Wno-unused-parameter + +[@file_type] +extension=designer +name=Designer file +icon=icon_text_css + +[@handler] +extension=designer +action=open diff --git a/util/luigi.h b/util/luigi.h index f89005f..52eb397 100644 --- a/util/luigi.h +++ b/util/luigi.h @@ -22,6 +22,8 @@ #define _UI_TO_STRING_2(x) _UI_TO_STRING_1(x) #ifdef UI_WINDOWS +#undef _UNICODE +#undef UNICODE #include #define UI_ASSERT(x) do { if (!(x)) { ui.assertionFailure = true; \ @@ -64,6 +66,9 @@ #define UI_CLOCK EsTimeStampMs #define UI_CLOCKS_PER_SECOND 1000 #define UI_CLOCK_T uint64_t + +// Callback to allow the application to process messages. +void _UIMessageProcess(EsMessage *message); #endif #ifdef UI_DEBUG @@ -97,7 +102,7 @@ typedef struct UITheme { #define UI_SIZE_TEXTBOX_MARGIN (3) #define UI_SIZE_TEXTBOX_WIDTH (200) -#define UI_SIZE_TEXTBOX_HEIGHT (25) +#define UI_SIZE_TEXTBOX_HEIGHT (27) #define UI_SIZE_TAB_PANE_SPACE_TOP (2) #define UI_SIZE_TAB_PANE_SPACE_LEFT (4) @@ -335,6 +340,7 @@ typedef struct UIWindow { #define UI_WINDOW_MENU (1 << 0) #define UI_WINDOW_INSPECTOR (1 << 1) #define UI_WINDOW_CENTER_IN_OWNER (1 << 2) +#define UI_WINDOW_MAXIMIZE (1 << 3) UIElement e; @@ -483,14 +489,20 @@ typedef struct UITextbox { ptrdiff_t bytes; int carets[2]; int scroll; + bool rejectNextKey; } UITextbox; -typedef struct UIMenu { #define UI_MENU_PLACE_ABOVE (1 << 0) +#define UI_MENU_NO_SCROLL (1 << 1) +#ifdef UI_ESSENCE +typedef EsMenu UIMenu; +#else +typedef struct UIMenu { UIElement e; int pointX, pointY; UIScrollBar *vScroll; } UIMenu; +#endif typedef struct UISlider { UIElement e; @@ -596,7 +608,7 @@ void UICodeInsertContent(UICode *code, const char *content, ptrdiff_t byteCount, void UIDrawBlock(UIPainter *painter, UIRectangle rectangle, uint32_t color); void UIDrawInvert(UIPainter *painter, UIRectangle rectangle); -void UIDrawLine(UIPainter *painter, int x0, int y0, int x1, int y1, uint32_t color); +bool UIDrawLine(UIPainter *painter, int x0, int y0, int x1, int y1, uint32_t color); // Returns false if the line was not visible. void UIDrawGlyph(UIPainter *painter, int x, int y, int c, uint32_t color); void UIDrawRectangle(UIPainter *painter, UIRectangle r, uint32_t mainColor, uint32_t borderColor, UIRectangle borderSize); void UIDrawBorder(UIPainter *painter, UIRectangle r, uint32_t borderColor, UIRectangle borderSize); @@ -685,6 +697,9 @@ struct { #ifdef UI_ESSENCE EsInstance *instance; + + void *menuData[256]; // HACK This limits the number of menu items to 128. + uintptr_t menuIndex; #endif #ifdef UI_FREETYPE @@ -1072,10 +1087,22 @@ void UIDrawBlock(UIPainter *painter, UIRectangle rectangle, uint32_t color) { return; } +#ifdef UI_SSE2 + __m128i color4 = _mm_set_epi32(color, color, color, color); +#endif + for (int line = rectangle.t; line < rectangle.b; line++) { uint32_t *bits = painter->bits + line * painter->width + rectangle.l; int count = UI_RECT_WIDTH(rectangle); +#ifdef UI_SSE2 + while (count >= 4) { + _mm_storeu_si128((__m128i *) bits, color4); + bits += 4; + count -= 4; + } +#endif + while (count--) { *bits++ = color; } @@ -1086,26 +1113,38 @@ void UIDrawBlock(UIPainter *painter, UIRectangle rectangle, uint32_t color) { #endif } -void UIDrawLine(UIPainter *painter, int x0, int y0, int x1, int y1, uint32_t color) { - // Clip the line to the painter's clip rectangle. +bool UIDrawLine(UIPainter *painter, int x0, int y0, int x1, int y1, uint32_t color) { + // Apply the clip. - if (x0 > x1) { int t = x0; x0 = x1; x1 = t; } - if (y0 > y1) { int t = y0; y0 = y1; y1 = t; } - UIRectangle bounds = UIRectangleIntersection(painter->clip, UI_RECT_4(x0, x1, y0, y1)); + UIRectangle c = painter->clip; int dx = x1 - x0, dy = y1 - y0; - int points[8], count = 0; - int iLY = dx ? (y0 + (bounds.l - x0) * dy / dx) : 0x7FFFFFFF; - int iRY = dx ? (y0 + (bounds.r - x0) * dy / dx) : 0x7FFFFFFF; - int iTX = dy ? (x0 + (bounds.t - y0) * dx / dy) : 0x7FFFFFFF; - int iBX = dy ? (x0 + (bounds.b - y0) * dx / dy) : 0x7FFFFFFF; - if (iLY >= bounds.t && iLY <= bounds.b) points[count + 0] = bounds.l, points[count + 1] = iLY, count += 2; - if (iRY >= bounds.t && iRY <= bounds.b) points[count + 0] = bounds.r, points[count + 1] = iRY, count += 2; - if (iTX >= bounds.l && iTX <= bounds.r) points[count + 1] = bounds.t, points[count + 0] = iTX, count += 2; - if (iBX >= bounds.l && iBX <= bounds.r) points[count + 1] = bounds.b, points[count + 0] = iBX, count += 2; - if (count < 4) return; - x0 = points[0], y0 = points[1], x1 = points[2], y1 = points[3]; - if (x0 == x1 && y0 == y1 && count > 4) x1 = points[4], y1 = points[5]; + const int p[4] = { -dx, dx, -dy, dy }; + const int q[4] = { x0 - c.l, c.r - x0, y0 - c.t, c.b - y0 }; + float t0 = 0.0f, t1 = 1.0f; // How far along the line the points end up. + + for (int i = 0; i < 4; i++) { + if (!p[i] && q[i] < 0) return false; + float r = (float) q[i] / p[i]; + if (p[i] < 0 && r > t1) return false; + if (p[i] > 0 && r < t0) return false; + if (p[i] < 0 && r > t0) t0 = r; + if (p[i] > 0 && r < t1) t1 = r; + } + + x1 = x0 + t1 * dx, y1 = y0 + t1 * dy; + x0 += t0 * dx, y0 += t0 * dy; + + // Calculate the delta X and delta Y. + + if (y1 < y0) { + int t; + t = x0, x0 = x1, x1 = t; + t = y0, y0 = y1, y1 = t; + } + dx = x1 - x0, dy = y1 - y0; + int dxs = dx < 0 ? -1 : 1; + if (dx < 0) dx = -dx; // Draw the line using Bresenham's line algorithm. @@ -1114,7 +1153,7 @@ void UIDrawLine(UIPainter *painter, int x0, int y0, int x1, int y1, uint32_t col if (dy * dy < dx * dx) { int m = 2 * dy - dx; - for (int i = 0; i < dx; i++, bits++) { + for (int i = 0; i < dx; i++, bits += dxs) { *bits = color; if (m > 0) bits += painter->width, m -= 2 * dx; m += 2 * dy; @@ -1124,10 +1163,12 @@ void UIDrawLine(UIPainter *painter, int x0, int y0, int x1, int y1, uint32_t col for (int i = 0; i < dy; i++, bits += painter->width) { *bits = color; - if (m > 0) bits++, m -= 2 * dy; + if (m > 0) bits += dxs, m -= 2 * dy; m += 2 * dx; } } + + return true; } void UIDrawInvert(UIPainter *painter, UIRectangle rectangle) { @@ -1645,6 +1686,20 @@ UIPanel *UIPanelCreate(UIElement *parent, uint32_t flags) { return panel; } +void _UIButtonCalculateColors(UIElement *element, uint32_t *color, uint32_t *textColor) { + bool disabled = element->flags & UI_ELEMENT_DISABLED; + bool focused = element == element->window->focused; + bool pressed = element == element->window->pressed; + bool hovered = element == element->window->hovered; + *color = disabled ? ui.theme.buttonDisabled + : (pressed && hovered) ? ui.theme.buttonPressed + : (pressed || hovered) ? ui.theme.buttonHovered + : focused ? ui.theme.selected : ui.theme.buttonNormal; + *textColor = disabled ? ui.theme.textDisabled + : *color == ui.theme.selected ? ui.theme.textSelected : ui.theme.text; +} + + int _UIButtonMessage(UIElement *element, UIMessage message, int di, void *dp) { UIButton *button = (UIButton *) element; bool isMenuItem = element->flags & UI_BUTTON_MENU_ITEM; @@ -1668,16 +1723,8 @@ int _UIButtonMessage(UIElement *element, UIMessage message, int di, void *dp) { } else if (message == UI_MSG_PAINT) { UIPainter *painter = (UIPainter *) dp; - bool disabled = element->flags & UI_ELEMENT_DISABLED; - bool focused = element == element->window->focused; - bool pressed = element == element->window->pressed; - bool hovered = element == element->window->hovered; - uint32_t color = disabled ? ui.theme.buttonDisabled - : (pressed && hovered) ? ui.theme.buttonPressed - : (pressed || hovered) ? ui.theme.buttonHovered - : focused ? ui.theme.selected : ui.theme.buttonNormal; - uint32_t textColor = disabled ? ui.theme.textDisabled - : color == ui.theme.selected ? ui.theme.textSelected : ui.theme.text; + uint32_t color, textColor; + _UIButtonCalculateColors(element, &color, &textColor); UIDrawRectangle(painter, element->bounds, color, ui.theme.border, UI_RECT_1(isMenuItem ? 0 : 1)); @@ -1931,6 +1978,19 @@ int _UITabPaneMessage(UIElement *element, UIMessage message, int di, void *dp) { child->flags |= UI_ELEMENT_HIDE; } + child = child->next; + index++; + } + } else if (message == UI_MSG_GET_HEIGHT) { + UIElement *child = element->children; + int index = 0; + int baseHeight = UI_SIZE_BUTTON_HEIGHT * element->window->scale; + + while (child) { + if (tabPane->active == index) { + return baseHeight + UIElementMessage(child, UI_MSG_GET_HEIGHT, di, dp); + } + child = child->next; index++; } @@ -2785,7 +2845,10 @@ int _UITextboxMessage(UIElement *element, UIMessage message, int di, void *dp) { UIKeyTyped *m = (UIKeyTyped *) dp; bool handled = true; - if (m->code == UI_KEYCODE_BACKSPACE || m->code == UI_KEYCODE_DELETE) { + if (textbox->rejectNextKey) { + textbox->rejectNextKey = false; + handled = false; + } else if (m->code == UI_KEYCODE_BACKSPACE || m->code == UI_KEYCODE_DELETE) { if (textbox->carets[0] == textbox->carets[1]) { UITextboxMoveCaret(textbox, m->code == UI_KEYCODE_BACKSPACE, element->window->ctrl); } @@ -3512,6 +3575,7 @@ bool _UIMenusClose() { return anyClosed; } +#ifndef UI_ESSENCE int _UIMenuItemMessage(UIElement *element, UIMessage message, int di, void *dp) { if (message == UI_MSG_CLICKED) { _UIMenusClose(); @@ -3556,11 +3620,12 @@ int _UIMenuMessage(UIElement *element, UIMessage message, int di, void *dp) { UIElement *child = element->children; int position = element->bounds.t + 2 - menu->vScroll->position; int totalHeight = 0; + int scrollBarSize = (menu->e.flags & UI_MENU_NO_SCROLL) ? 0 : UI_SIZE_SCROLL_BAR; while (child) { if (~child->flags & UI_ELEMENT_NON_CLIENT) { int height = UIElementMessage(child, UI_MSG_GET_HEIGHT, 0, 0); - UIElementMove(child, UI_RECT_4(element->bounds.l + 2, element->bounds.r - UI_SIZE_SCROLL_BAR - 2, + UIElementMove(child, UI_RECT_4(element->bounds.l + 2, element->bounds.r - scrollBarSize - 2, position, position + height), false); position += height; totalHeight += height; @@ -3570,7 +3635,7 @@ int _UIMenuMessage(UIElement *element, UIMessage message, int di, void *dp) { } UIRectangle scrollBarBounds = element->bounds; - scrollBarBounds.l = scrollBarBounds.r - UI_SIZE_SCROLL_BAR * element->window->scale; + scrollBarBounds.l = scrollBarBounds.r - scrollBarSize * element->window->scale; menu->vScroll->maximum = totalHeight; menu->vScroll->page = UI_RECT_HEIGHT(element->bounds); UIElementMove(&menu->vScroll->e, scrollBarBounds, true); @@ -3625,6 +3690,7 @@ UIMenu *UIMenuCreate(UIElement *parent, uint32_t flags) { return menu; } +#endif UIRectangle UIElementScreenBounds(UIElement *element) { int x = 0, y = 0; @@ -3865,7 +3931,9 @@ UIElement *_UIElementPreviousSibling(UIElement *element) { return sibling; } -void _UIWindowInputEvent(UIWindow *window, UIMessage message, int di, void *dp) { +bool _UIWindowInputEvent(UIWindow *window, UIMessage message, int di, void *dp) { + bool handled = true; + if (window->pressed) { if (message == UI_MSG_MOUSE_MOVE) { UIElementMessage(window->pressed, UI_MSG_MOUSE_DRAG, di, dp); @@ -3943,7 +4011,7 @@ void _UIWindowInputEvent(UIWindow *window, UIMessage message, int di, void *dp) element = element->parent; } } else if (message == UI_MSG_KEY_TYPED) { - bool handled = false; + handled = false; if (window->focused) { UIElement *element = window->focused; @@ -3993,6 +4061,8 @@ void _UIWindowInputEvent(UIWindow *window, UIMessage message, int di, void *dp) if (~element->flags & UI_ELEMENT_WINDOW) { UIElementFocus(element); } + + handled = true; } else if (!window->dialog) { for (intptr_t i = window->shortcutCount - 1; i >= 0; i--) { UIShortcut *shortcut = window->shortcuts + i; @@ -4000,6 +4070,7 @@ void _UIWindowInputEvent(UIWindow *window, UIMessage message, int di, void *dp) if (shortcut->code == m->code && shortcut->ctrl == window->ctrl && shortcut->shift == window->shift && shortcut->alt == window->alt) { shortcut->invoke(shortcut->cp); + handled = true; break; } } @@ -4018,6 +4089,7 @@ void _UIWindowInputEvent(UIWindow *window, UIMessage message, int di, void *dp) } end: _UIUpdate(); + return handled; } void _UIInitialiseCommon() { @@ -4259,6 +4331,11 @@ UIWindow *UIWindowCreate(UIWindow *owner, uint32_t flags, const char *cTitle, in | ButtonPressMask | ButtonReleaseMask | KeyPressMask | KeyReleaseMask | StructureNotifyMask | EnterWindowMask | LeaveWindowMask | ButtonMotionMask | KeymapStateMask | FocusChangeMask); + if (flags & UI_WINDOW_MAXIMIZE) { + Atom atoms[2] = { XInternAtom(ui.display, "_NET_WM_STATE_MAXIMIZED_HORZ", 0), XInternAtom(ui.display, "_NET_WM_STATE_MAXIMIZED_VERT", 0) }; + XChangeProperty(ui.display, window->window, XInternAtom(ui.display, "_NET_WM_STATE", 0), XA_ATOM, 32, PropModeReplace, (unsigned char *) atoms, 2); + } + if (~flags & UI_WINDOW_MENU) { XMapRaised(ui.display, window->window); } @@ -4517,6 +4594,22 @@ bool _UIProcessEvent(XEvent *event) { window->alt = true; window->altCode = event->xkey.keycode; _UIWindowInputEvent(window, UI_MSG_MOUSE_MOVE, 0, 0); + } else if (symbol == XK_KP_Left) { + m.code = UI_KEYCODE_LEFT; + } else if (symbol == XK_KP_Right) { + m.code = UI_KEYCODE_RIGHT; + } else if (symbol == XK_KP_Up) { + m.code = UI_KEYCODE_UP; + } else if (symbol == XK_KP_Down) { + m.code = UI_KEYCODE_DOWN; + } else if (symbol == XK_KP_Home) { + m.code = UI_KEYCODE_HOME; + } else if (symbol == XK_KP_End) { + m.code = UI_KEYCODE_END; + } else if (symbol == XK_KP_Enter) { + m.code = UI_KEYCODE_ENTER; + } else if (symbol == XK_KP_Delete) { + m.code = UI_KEYCODE_DELETE; } _UIWindowInputEvent(window, UI_MSG_KEY_TYPED, 0, &m); @@ -5083,14 +5176,32 @@ bool _UIMessageLoopSingle(int *result) { if (ui.animating) { // TODO. } else { - EsMessageReceive(); + _UIMessageProcess(EsMessageReceive()); } return true; } +UIMenu *UIMenuCreate(UIElement *parent, uint32_t flags) { + ui.menuIndex = 0; + return EsMenuCreate(parent->window->window, ES_MENU_AT_CURSOR); +} + +void _UIMenuItemCallback(EsMenu *menu, EsGeneric context) { + ((void (*)(void *)) ui.menuData[context.u * 2 + 0])(ui.menuData[context.u * 2 + 1]); +} + +void UIMenuAddItem(UIMenu *menu, uint32_t flags, const char *label, ptrdiff_t labelBytes, void (*invoke)(void *cp), void *cp) { + EsAssert(ui.menuIndex < 128); + ui.menuData[ui.menuIndex * 2 + 0] = (void *) invoke; + ui.menuData[ui.menuIndex * 2 + 1] = cp; + EsMenuAddItem(menu, (flags & UI_BUTTON_CHECKED) ? ES_MENU_ITEM_CHECKED : ES_FLAGS_DEFAULT, + label, labelBytes, _UIMenuItemCallback, ui.menuIndex); + ui.menuIndex++; +} + void UIMenuShow(UIMenu *menu) { - // TODO. + EsMenuShow(menu); } int _UIWindowCanvasMessage(EsElement *element, EsMessage *message) { @@ -5122,7 +5233,7 @@ int _UIWindowCanvasMessage(EsElement *element, EsMessage *message) { m.text = c; m.textBytes = EsMessageGetInputText(message, c); m.code = message->keyboard.scancode; - _UIWindowInputEvent(window, UI_MSG_KEY_TYPED, 0, &m); + return _UIWindowInputEvent(window, UI_MSG_KEY_TYPED, 0, &m) ? ES_HANDLED : 0; } else if (message->type == ES_MSG_MOUSE_LEFT_CLICK) { _UIInspectorSetFocusedWindow(window); } else if (message->type == ES_MSG_USER_START) {