+/*
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
+ *
+ * Copyright 2025 Olaf Wintermann. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#import "list.h"
+#import "ListDelegate.h"
+#import <objc/runtime.h>
+
+static void* getvalue_wrapper(UiList *list, void *elm, int row, int col, void *userdata, UiBool *freeResult) {
+ ui_getvaluefunc getvalue = (ui_getvaluefunc)userdata;
+ return getvalue(elm, col);
+}
+
+static void* str_getvalue(UiList *list, void *elm, int row, int col, void *userdata, UiBool *freeResult) {
+ return elm;
+}
+
+/* --------------------------- ListView --------------------------- */
+
+/*
+ * adds a NSTableViewDelegate that handles all events and calls
+ * callbacks specified in the UiListArgs
+ */
+static void add_listdelegate(UiObject *obj, NSTableView *tableview, UiListArgs *args) {
+ ListDelegate *delegate = [[ListDelegate alloc] init:tableview obj:obj];
+ delegate.onactivate = args->onactivate;
+ delegate.onactivatedata = args->onactivatedata;
+ delegate.onselection = args->onselection;
+ delegate.onselectiondata = args->onselectiondata;
+ tableview.delegate = delegate;
+ objc_setAssociatedObject(tableview, "ui_listdelegate", delegate, OBJC_ASSOCIATION_RETAIN);
+ tableview.doubleAction = @selector(activateEvent:);
+ tableview.target = delegate;
+}
+
+static void bind_list_to_tableview(UiList *list, NSTableView *tableview) {
+ list->obj = (__bridge void*)tableview;
+ list->update = ui_tableview_update;
+ list->getselection = ui_tableview_getselection;
+ list->setselection = ui_tableview_setselection;
+}
+
+UIWIDGET ui_listview_create(UiObject* obj, UiListArgs *args) {
+ NSScrollView *scrollview = [[NSScrollView alloc] init];
+
+ NSTableView *tableview = [[NSTableView alloc] init];
+ tableview.autoresizingMask = NSViewWidthSizable;
+ tableview.headerView = nil;
+
+ if(args->multiselection) {
+ tableview.allowsMultipleSelection = YES;
+ }
+
+ scrollview.documentView = tableview;
+
+ UiLayout layout = UI_INIT_LAYOUT(args);
+ ui_container_add(obj, scrollview, &layout);
+
+ add_listdelegate(obj, tableview, args);
+
+ UiVar *var = uic_widget_var(obj->ctx, obj->ctx, args->list, args->varname, UI_VAR_LIST);
+ if(var) {
+ UiList *list = var->value;
+ bind_list_to_tableview(list, tableview);
+
+ ui_getvaluefunc2 getvalue = args->getvalue2;
+ void *getvaluedata = args->getvalue2data;
+ if(!getvalue) {
+ if(args->getvalue) {
+ getvalue = getvalue_wrapper;
+ getvaluedata = (void*)args->getvalue;
+ } else {
+ getvalue = str_getvalue; // by default list values are interpreted as strings
+ }
+ }
+
+ NSTableColumn *column = [[NSTableColumn alloc] initWithIdentifier:@"column"];
+ [tableview addTableColumn:column];
+
+ ListDataSource *dataSource = [[ListDataSource alloc] init:tableview.tableColumns var:var getvalue:getvalue getvaluedata:getvaluedata];
+
+ tableview.dataSource = dataSource;
+ [tableview reloadData];
+
+ objc_setAssociatedObject(tableview, "ui_datasource", dataSource, OBJC_ASSOCIATION_RETAIN);
+ }
+
+ return (__bridge void*)scrollview;
+}
+
+/* --------------------------- TableView --------------------------- */
+
+UIWIDGET ui_table_create(UiObject* obj, UiListArgs *args) {
+ NSScrollView *scrollview = [[NSScrollView alloc] init];
+
+ NSTableView *tableview = [[NSTableView alloc] init];
+ tableview.autoresizingMask = NSViewWidthSizable;
+ tableview.columnAutoresizingStyle = NSTableViewSequentialColumnAutoresizingStyle;
+
+ if(args->multiselection) {
+ tableview.allowsMultipleSelection = YES;
+ }
+
+ UiLayout layout = UI_INIT_LAYOUT(args);
+ ui_container_add(obj, scrollview, &layout);
+
+ add_listdelegate(obj, tableview, args);
+
+ // convert model
+ NSMutableArray<NSTableColumn*> *cols = [[NSMutableArray alloc] init];
+ UiModel *model = args->model;
+ if(model) {
+ for(int i=0;i<model->columns;i++) {
+ char *title = model->titles[i];
+ UiModelType type = model->types[i];
+ int width = model->columnsize[i];
+ NSString *identifier = [[NSString alloc] initWithUTF8String:title];
+ NSTableColumn *column = [[NSTableColumn alloc] initWithIdentifier:identifier];
+ column.title = identifier;
+ column.resizingMask = NSTableColumnUserResizingMask;
+ if(width > 0) {
+ column.width = width;
+ } else if(width < 0) {
+ column.resizingMask = NSTableColumnAutoresizingMask | NSTableColumnUserResizingMask;
+ }
+ if(type >= UI_ICON) {
+ // TODO
+ }
+ [tableview addTableColumn:column];
+ [cols addObject:column];
+ }
+ }
+
+ UiVar *var = uic_widget_var(obj->ctx, obj->ctx, args->list, args->varname, UI_VAR_LIST);
+ if(var) {
+ UiList *list = var->value;
+ bind_list_to_tableview(list, tableview);
+
+ ui_getvaluefunc2 getvalue = args->getvalue2;
+ void *getvaluedata = args->getvalue2data;
+ if(!getvalue) {
+ if(args->getvalue) {
+ getvalue = getvalue_wrapper;
+ getvaluedata = (void*)args->getvalue;
+ } else {
+ fprintf(stderr, "Error: tableview requires getvalue or getvalue2 func\n");
+ return (__bridge void*)scrollview;
+ }
+ }
+
+ ListDataSource *dataSource = [[ListDataSource alloc] init:cols var:var getvalue:getvalue getvaluedata:getvaluedata];
+ if(model) {
+ dataSource.model = ui_model_copy(obj->ctx, model);
+ }
+
+ tableview.dataSource = dataSource;
+ [tableview reloadData];
+
+ objc_setAssociatedObject(tableview, "ui_datasource", dataSource, OBJC_ASSOCIATION_RETAIN);
+ }
+
+ scrollview.documentView = tableview;
+
+ return (__bridge void*)scrollview;
+}
+
+/* ------ common functions ------ */
+
+void ui_tableview_update(UiList *list, int i) {
+ NSTableView *tableview = (__bridge NSTableView*)list->obj;
+ if(i < 0) {
+ [tableview reloadData];
+ } else {
+ [tableview reloadData]; // TODO: optimize
+ }
+}
+
+UiListSelection ui_tableview_getselection(UiList *list) {
+ NSTableView *tableview = (__bridge NSTableView*)list->obj;
+ return ui_tableview_selection(tableview);
+}
+
+void ui_tableview_setselection(UiList *list, UiListSelection selection) {
+ NSTableView *tableview = (__bridge NSTableView*)list->obj;
+ NSMutableIndexSet *indexSet = [NSMutableIndexSet indexSet];
+ for(int i=0;i<selection.count;i++) {
+ [indexSet addIndex:selection.rows[i]];
+ }
+ [tableview selectRowIndexes:indexSet byExtendingSelection:NO];
+}
+
+
+/* --------------------------- DropDown --------------------------- */
+
+@implementation UiDropDown
+
+- (id)init:(UiObject*)obj {
+ _obj = obj;
+ return self;
+}
+
+- (void) comboBoxSelectionDidChange:(NSNotification *) notification {
+ int index = (int)_combobox.indexOfSelectedItem;
+
+ void *eventdata = NULL;
+ if(_var) {
+ UiList *list = _var->value;
+ if(index >= 0) {
+ eventdata = list->get(list, index);
+ }
+ } else {
+ NSString *str = _combobox.objectValueOfSelectedItem;
+ if(str) {
+ eventdata = (void*)str.UTF8String;
+ }
+ }
+
+ UiEvent event;
+ event.obj = _obj;
+ event.window = event.obj->window;
+ event.document = event.obj->ctx->document;
+ event.eventdata = eventdata;
+ event.eventdatatype = UI_EVENT_DATA_LIST_ELM;
+ event.intval = index;
+
+ if(_onselection) {
+ _onselection(&event, _onselectiondata);
+ }
+
+ if(_onactivate) {
+ _onactivate(&event, _onactivatedata);
+ }
+}
+
+@end
+
+UIWIDGET ui_combobox_create(UiObject* obj, UiListArgs *args) {
+ NSComboBox *dropdown = [[NSComboBox alloc] init];
+ dropdown.editable = NO;
+
+ UiDropDown *uidropdown = [[UiDropDown alloc] init:obj];
+ objc_setAssociatedObject(dropdown, "ui_dropdown", uidropdown, OBJC_ASSOCIATION_RETAIN);
+ uidropdown.onactivate = args->onactivate;
+ uidropdown.onactivatedata = args->onactivatedata;
+ uidropdown.onselection = args->onselection;
+ uidropdown.onselectiondata = args->onselectiondata;
+ uidropdown.combobox = dropdown;
+
+ if(!args->getvalue2) {
+ if(args->getvalue) {
+ args->getvalue2 = getvalue_wrapper;
+ args->getvalue2data = (void*)args->getvalue;
+ } else {
+ args->getvalue2 = str_getvalue;
+ }
+ }
+ uidropdown.getvalue = args->getvalue2;
+ uidropdown.getvaluedata = args->getvalue2data;
+
+ UiLayout layout = UI_INIT_LAYOUT(args);
+ ui_container_add(obj, dropdown, &layout);
+
+ UiVar *var = uic_widget_var(obj->ctx, obj->ctx, args->list, args->varname, UI_VAR_LIST);
+ if(var) {
+ UiList *list = var->value;
+ list->obj = (__bridge void*)dropdown;
+ list->update = ui_dropdown_update;
+ list->getselection = ui_dropdown_getselection;
+ list->setselection = ui_dropdown_setselection;
+ ui_dropdown_update(list, -1);
+ } else {
+ for(int i=0;i<args->static_nelm;i++) {
+ char *str = args->static_elements[i];
+ NSString *item = [[NSString alloc] initWithUTF8String:str];
+ [dropdown addItemWithObjectValue:item];
+ }
+ }
+
+ uidropdown.var = var;
+
+ return (__bridge void*)dropdown;
+}
+
+void ui_dropdown_update(UiList *list, int i) {
+ NSComboBox *combobox = (__bridge NSComboBox*)list->obj;
+ UiDropDown *dropdown = objc_getAssociatedObject(combobox, "ui_dropdown");
+ if(dropdown) {
+ [combobox removeAllItems];
+
+ ui_getvaluefunc2 getvalue = dropdown.getvalue;
+ void *getvaluedata = dropdown.getvaluedata;
+
+ int index = 0;
+ void *elm = list->first(list);
+ while(elm) {
+ UiBool freeResult = FALSE;
+ char *str = getvalue(list, elm, index, 0, getvaluedata, &freeResult);
+ if(str) {
+ NSString *item = [[NSString alloc] initWithUTF8String:str];
+ [combobox addItemWithObjectValue:item];
+ }
+ if(freeResult) {
+ free(str);
+ }
+ elm = list->next(list);
+ index++;
+ }
+ } else {
+ fprintf(stderr, "Error: obj is not a dropdown\n");
+ }
+}
+
+UiListSelection ui_dropdown_getselection(UiList *list) {
+ UiListSelection sel = { 0, NULL };
+ NSComboBox *combobox = (__bridge NSComboBox*)list->obj;
+ NSInteger index = combobox.indexOfSelectedItem;
+ if(index >= 0) {
+ sel.rows = malloc(sizeof(int));
+ sel.count = 1;
+ sel.rows[0] = (int)index;
+ }
+ return sel;
+}
+
+void ui_dropdown_setselection(UiList *list, UiListSelection selection) {
+ NSComboBox *combobox = (__bridge NSComboBox*)list->obj;
+ if(selection.count > 0) {
+ [combobox selectItemAtIndex:selection.rows[0]];
+ } else {
+ [combobox selectItemAtIndex: -1];
+ }
+}