1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29 #include <stdio.h>
30 #include <stdlib.h>
31 #include <string.h>
32 #include <inttypes.h>
33
34 #include "tree.h"
35
36 #include "container.h"
37 #include "../common/object.h"
38 #include "../common/context.h"
39 #include <cx/utils.h>
40 #include <cx/compare.h>
41 #include <cx/printf.h>
42
43 UIWIDGET ui_table_var(UiObject *obj, UiVar *var, UiModel *model, UiListCallbacks cb) {
44
45
46 Arg args[
32];
47 int n =
0;
48
49
50 UiContainer *ct = uic_get_current_container(obj);
51 Widget parent = ct->prepare(ct, args, &n,
TRUE);
52
53 XtSetArg(args[n], XmNscrollingPolicy, XmAUTOMATIC);
54 n++;
55 XtSetArg(args[n], XmNshadowThickness,
0);
56 n++;
57 Widget scrollw = XmCreateScrolledWindow(parent,
"scroll_win", args, n);
58 ct->add(ct, scrollw);
59 XtManageChild(scrollw);
60
61
62 XmStringTable header = (XmStringTable)XtMalloc(
63 model->columns *
sizeof(XmString));
64 for(
int i=
0;i<model->columns;i++) {
65 header[i] = XmStringCreateLocalized(model->titles[i]);
66 }
67 n =
0;
68 XtSetArg(args[n], XmNdetailColumnHeading, header);
69 n++;
70 XtSetArg(args[n], XmNdetailColumnHeadingCount, model->columns);
71 n++;
72
73
74 XtSetArg(args[n], XmNlayoutType, XmDETAIL);
75 n++;
76 XtSetArg(args[n], XmNentryViewType, XmSMALL_ICON);
77 n++;
78 XtSetArg(args[n], XmNselectionPolicy, XmSINGLE_SELECT);
79 n++;
80 XtSetArg(args[n], XmNwidth,
600);
81 n++;
82
83
84
85
86
87 Widget container = XmCreateContainer(scrollw,
"table", args, n);
88 XtManageChild(container);
89
90
91 UiTreeEventData *event = ui_malloc(obj->ctx,
sizeof(UiTreeEventData));
92 event->obj = obj;
93 event->activate = cb.activate;
94 event->selection = cb.selection;
95 event->userdata = cb.userdata;
96 event->last_selection =
NULL;
97 if(cb.selection) {
98 XtAddCallback(
99 container,
100 XmNselectionCallback,
101 (XtCallbackProc)ui_table_select_callback,
102 event);
103 }
104 if(cb.activate) {
105 XtAddCallback(
106 container,
107 XmNdefaultActionCallback,
108 (XtCallbackProc)ui_table_action_callback,
109 event);
110 }
111
112
113 UiList *list = var->value;
114 void *data = list->first(list);
115 int width =
0;
116 while(data) {
117 int w = ui_add_icon_gadget(container, model, data);
118 if(w > width) {
119 width = w;
120 }
121 data = list->next(list);
122 }
123
124 UiTableView *tableview = cxMalloc(obj->ctx->allocator,
sizeof(UiTableView));
125 tableview->widget = container;
126 tableview->var = var;
127 tableview->model = model;
128
129
130 XtVaSetValues(container, XmNwidth, width,
NULL);
131
132
133 for(
int i=
0;i<model->columns;i++) {
134 XmStringFree(header[i]);
135 }
136 XtFree((
char*)header);
137
138 return scrollw;
139 }
140
141 UIWIDGET ui_table(UiObject *obj, UiList *data, UiModel *model, UiListCallbacks cb) {
142 UiVar *var = malloc(
sizeof(UiVar));
143 var->value = data;
144 var->type =
UI_VAR_SPECIAL;
145 return ui_table_var(obj, var, model, cb);
146 }
147
148 void ui_table_update(UiEvent *event, UiTableView *view) {
149
150 Widget *children;
151 int nc;
152
153 XtVaGetValues(
154 view->widget,
155 XmNchildren,
156 &children,
157 XmNnumChildren,
158 &nc,
159 NULL);
160
161 for(
int i=
0;i<nc;i++) {
162 XtDestroyWidget(children[i]);
163 }
164
165 UiList *list = view->var->value;
166
167 void *data = list->first(list);
168 int width =
0;
169 while(data) {
170 int w = ui_add_icon_gadget(view->widget, view->model, data);
171 if(w > width) {
172 width = w;
173 }
174 data = list->next(list);
175 }
176
177 }
178
179 #define UI_COL_CHAR_WIDTH 12
180
181 int ui_add_icon_gadget(Widget container, UiModel *model,
void *data) {
182 int width =
50;
183
184 if(model->columns ==
0) {
185 return width;
186 }
187
188 XmString label =
NULL;
189 Arg args[
8];
190 Boolean f;
191
192 if(model->types[
0] !=
12345678) {
193 char *str = ui_type_to_string(
194 model->types[
0],
195 model->getvalue(data,
0),
196 &f);
197
198
199 width += strlen(str) *
UI_COL_CHAR_WIDTH;
200
201
202 XmString label = XmStringCreateLocalized(str);
203 XtSetArg(args[
0], XmNlabelString, label);
204 if(f) {
205 free(str);
206 }
207 }
else {
208
209 }
210
211
212 XmStringTable details = (XmStringTable)XtMalloc(
213 (model->columns -
1) *
sizeof(XmString));
214 for(
int i=
1;i<model->columns;i++) {
215 char *str = ui_type_to_string(
216 model->types[i],
217 model->getvalue(data, i),
218 &f);
219
220
221 width += strlen(str) *
UI_COL_CHAR_WIDTH;
222
223 details[i -
1] = XmStringCreateLocalized(str);
224 if(f) {
225 free(str);
226 }
227 }
228 XtSetArg(args[
1], XmNdetail, details);
229 XtSetArg(args[
2], XmNdetailCount, model->columns -
1);
230 XtSetArg(args[
3], XmNshadowThickness,
0);
231
232 Widget item = XmCreateIconGadget(container,
"table_item", args,
4);
233 XtManageChild(item);
234
235
236 XmStringFree(label);
237 for(
int i=
0;i<model->columns-
1;i++) {
238 XmStringFree(details[i]);
239 }
240 XtFree((
char*)details);
241
242 return width;
243 }
244
245 char* ui_type_to_string(UiModelType type,
void *data, Boolean *free) {
246 switch(type) {
247 case UI_STRING: *free =
FALSE;
return data;
248 case UI_INTEGER: {
249 *free =
TRUE;
250 int *val = data;
251 cxmutstr str = cx_asprintf(
"%d", *val);
252 return str.ptr;
253 }
254 case UI_ICON:
break;
255 case UI_ICON_TEXT:
break;
256 }
257 *free =
FALSE;
258 return NULL;
259 }
260
261 void ui_table_action_callback(
262 Widget widget,
263 UiTreeEventData *event,
264 XmContainerSelectCallbackStruct *sel)
265 {
266 UiListSelection *selection = ui_list_selection(sel);
267
268 UiEvent e;
269 e.obj = event->obj;
270 e.window = event->obj->window;
271 e.document = event->obj->ctx->document;
272 e.eventdata = selection;
273 e.intval = selection->count >
0 ? selection->rows[
0] : -
1;
274 event->activate(&e, event->userdata);
275
276 free(event->last_selection->rows);
277 free(event->last_selection);
278 event->last_selection = selection;
279 }
280
281 void ui_table_select_callback(
282 Widget widget,
283 UiTreeEventData *event,
284 XmContainerSelectCallbackStruct *sel)
285 {
286 UiListSelection *selection = ui_list_selection(sel);
287 if(!ui_compare_list_selection(selection, event->last_selection)) {
288 UiEvent e;
289 e.obj = event->obj;
290 e.window = event->obj->window;
291 e.document = event->obj->ctx->document;
292 e.eventdata = selection;
293 e.intval = selection->count >
0 ? selection->rows[
0] : -
1;
294 event->selection(&e, event->userdata);
295 }
296 if(event->last_selection) {
297 free(event->last_selection->rows);
298 free(event->last_selection);
299 }
300 event->last_selection = selection;
301 }
302
303 UiListSelection* ui_list_selection(XmContainerSelectCallbackStruct *xs) {
304 UiListSelection *selection = malloc(
sizeof(UiListSelection));
305 selection->count = xs->selected_item_count;
306 selection->rows = calloc(selection->count,
sizeof(
int));
307 for(
int i=
0;i<selection->count;i++) {
308 int index;
309 XtVaGetValues(xs->selected_items[i], XmNpositionIndex, &index,
NULL);
310 selection->rows[i] = index;
311 }
312 return selection;
313 }
314
315 Boolean ui_compare_list_selection(UiListSelection *s1, UiListSelection *s2) {
316 if(!s1 || !s2) {
317 return FALSE;
318 }
319 if(s1->count != s2->count) {
320 return FALSE;
321 }
322 for(
int i=
0;i<s1->count;i++) {
323 if(s1->rows[i] != s2->rows[i]) {
324 return FALSE;
325 }
326 }
327 return TRUE;
328 }
329