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 "pch.h"
30
31 #include "text.h"
32
33 #include "../common/context.h"
34 #include "../common/object.h"
35
36 #include <cx/string.h>
37 #include <cx/allocator.h>
38
39 #include "util.h"
40 #include "container.h"
41
42
43
44 using namespace winrt;
45 using namespace Microsoft::
UI::Xaml;
46 using namespace Microsoft::
UI::Xaml::Controls;
47 using namespace Windows::
UI::Xaml::Interop;
48 using namespace winrt::Windows::Foundation;
49 using namespace Microsoft::
UI::Xaml::Markup;
50 using namespace Microsoft::
UI::Xaml::Media;
51 using namespace winrt::Microsoft::
UI::Xaml::Controls::Primitives;
52 using namespace winrt::Windows::
UI::Xaml::Input;
53
54
55 UIEXPORT UIWIDGET ui_textarea_create(UiObject *obj, UiTextAreaArgs args) {
56 UiObject* current = uic_current_obj(obj);
57
58
59 TextBox textarea = TextBox();
60 textarea.AcceptsReturn(true);
61 ScrollViewer::SetVerticalScrollBarVisibility(textarea, ScrollBarVisibility::Auto);
62 UIElement elm = textarea;
63 UiWidget* widget = new UiWidget(elm);
64 ui_context_add_widget_destructor(current->ctx, widget);
65 ui_set_widget_groups(current->ctx, widget, args.groups);
66
67 UiVar* var = uic_widget_var(obj->ctx, current->ctx, args.value, args.varname,
UI_VAR_TEXT);
68 if (var) {
69 UiText* value = (UiText*)var->value;
70 value->obj = widget;
71 value->undomgr =
NULL;
72 value->set = ui_textarea_set;
73 value->get = ui_textarea_get;
74 value->getsubstr = ui_textarea_getsubstr;
75 value->insert = ui_textarea_insert;
76 value->setposition = ui_textarea_setposition;
77 value->position = ui_textarea_position;
78 value->selection = ui_textarea_selection;
79 value->length = ui_textarea_length;
80 value->remove = ui_textarea_remove;
81 }
82
83
84 UI_APPLY_LAYOUT1(current, args);
85
86 current->container->Add(textarea, true);
87
88 return widget;
89 }
90
91 UIEXPORT UIWIDGET ui_textarea_gettextwidget(
UIWIDGET textarea) {
92 return textarea;
93 }
94
95 UIEXPORT void ui_text_undo(UiText *value) {
96
97 }
98
99 UIEXPORT void ui_text_redo(UiText *value) {
100
101 }
102
103
104
105 char* ui_wtext_get(UiText *text, std::wstring &value) {
106 if (text->value.ptr) {
107 text->value.free(text->value.ptr);
108 }
109
110 text->value.ptr = wchar2utf8(value.c_str(), value.length());
111 text->value.free = free;
112
113 return text->value.ptr;
114 }
115
116 std::wstring ui_wtext_set(UiText *text,
const char* value) {
117 if (text->value.ptr) {
118 text->value.free(text->value.ptr);
119 }
120
121 text->value.ptr = _strdup(value);
122 text->value.free = free;
123
124 int len;
125 wchar_t* wstr = str2wstr(value, &len);
126 std::wstring s(wstr);
127 free(wstr);
128
129 return s;
130 }
131
132 extern "C" char* ui_textarea_get(UiText *text) {
133 UiWidget* widget = (UiWidget*)text->obj;
134 TextBox box = widget->uielement.as<TextBox>();
135 std::wstring wstr(box.Text());
136 return ui_wtext_get(text, wstr);
137 }
138
139 extern "C" void ui_textarea_set(UiText *text,
const char *newvalue) {
140 UiWidget* widget = (UiWidget*)text->obj;
141 TextBox box = widget->uielement.as<TextBox>();
142 box.Text(ui_wtext_set(text, newvalue));
143 }
144
145 extern "C" char* ui_textarea_getsubstr(UiText *text,
int begin,
int end) {
146 return NULL;
147 }
148
149 extern "C" void ui_textarea_insert(UiText *text,
int pos,
char *str) {
150
151 }
152
153 extern "C" void ui_textarea_setposition(UiText *text,
int pos) {
154
155 }
156
157 extern "C" int ui_textarea_position(UiText *text) {
158 return 0;
159 }
160
161 extern "C" void ui_textarea_selection(UiText *text,
int *begin,
int *end) {
162
163 }
164
165 extern "C" int ui_textarea_length(UiText *text) {
166 return 0;
167 }
168
169 extern "C" void ui_textarea_remove(UiText *text,
int begin,
int end) {
170
171 }
172
173
174
175
176 UIWIDGET ui_textfield_create(UiObject* obj, UiTextFieldArgs args) {
177 UiObject* current = uic_current_obj(obj);
178
179
180 TextBox textfield = TextBox();
181 UIElement elm = textfield;
182 UiWidget* widget = new UiWidget(elm);
183 ui_context_add_widget_destructor(current->ctx, widget);
184 ui_set_widget_groups(current->ctx, widget, args.groups);
185
186 UiVar* var = uic_widget_var(obj->ctx, current->ctx, args.value, args.varname,
UI_VAR_STRING);
187 if (var) {
188 UiString* value = (UiString*)var->value;
189 value->obj = widget;
190 value->get = ui_textfield_get;
191 value->set = ui_textfield_set;
192
193
194
195 }
196
197
198 UI_APPLY_LAYOUT1(current, args);
199
200 current->container->Add(textfield, false);
201
202 return widget;
203 }
204
205 UIWIDGET ui_frameless_textfield_create(UiObject* obj, UiTextFieldArgs args) {
206 return ui_textfield_create(obj, args);
207 }
208
209 UIWIDGET ui_passwordfield_create(UiObject* obj, UiTextFieldArgs args) {
210 UiObject* current = uic_current_obj(obj);
211
212
213 PasswordBox textfield = PasswordBox();
214 UIElement elm = textfield;
215 UiWidget* widget = new UiWidget(elm);
216 ui_context_add_widget_destructor(current->ctx, widget);
217 ui_set_widget_groups(current->ctx, widget, args.groups);
218
219 UiVar* var = uic_widget_var(obj->ctx, current->ctx, args.value, args.varname,
UI_VAR_STRING);
220 if (var) {
221 UiString* value = (UiString*)var->value;
222 value->obj = widget;
223 value->get = ui_passwordfield_get;
224 value->set = ui_passwordfield_set;
225
226
227
228 }
229
230
231 UI_APPLY_LAYOUT1(current, args);
232
233 current->container->Add(textfield, false);
234
235 return widget;
236 }
237
238
239
240
241 char* ui_wstring_get(UiString* str, std::wstring &value) {
242 if (str->value.ptr) {
243 str->value.free(str->value.ptr);
244 }
245
246 str->value.ptr = wchar2utf8(value.c_str(), value.length());
247 str->value.free = free;
248
249 return str->value.ptr;
250 }
251
252 std::wstring ui_wstring_set(UiString* str,
const char* value) {
253 if (str->value.ptr) {
254 str->value.free(str->value.ptr);
255 }
256
257 str->value.ptr = _strdup(value);
258 str->value.free = free;
259
260 int len;
261 wchar_t* wstr = str2wstr(value, &len);
262 std::wstring s(wstr);
263 free(wstr);
264
265 return s;
266 }
267
268 char* ui_textfield_get(UiString * str) {
269 UiWidget* widget = (UiWidget*)str->obj;
270 TextBox box = widget->uielement.as<TextBox>();
271 std::wstring wstr(box.Text());
272 return ui_wstring_get(str, wstr);
273 }
274
275 void ui_textfield_set(UiString * str,
const char* newvalue) {
276 UiWidget* widget = (UiWidget*)str->obj;
277 TextBox box = widget->uielement.as<TextBox>();
278 box.Text(ui_wstring_set(str, newvalue));
279 }
280
281
282 char* ui_passwordfield_get(UiString * str) {
283 UiWidget* widget = (UiWidget*)str->obj;
284 PasswordBox box = widget->uielement.as<PasswordBox>();
285 std::wstring wstr(box.Password());
286 return ui_wstring_get(str, wstr);
287 }
288
289 void ui_passwordfield_set(UiString * str,
const char* newvalue) {
290 UiWidget* widget = (UiWidget*)str->obj;
291 PasswordBox box = widget->uielement.as<PasswordBox>();
292 box.Password(ui_wstring_set(str, newvalue));
293 }
294
295
296
297
298 extern "C" static void destroy_ui_pathtextfield(
void* ptr) {
299 UiPathTextField* pb = (UiPathTextField*)ptr;
300 delete pb;
301 }
302
303 static void ui_context_add_pathtextfield_destructor(UiContext* ctx, UiPathTextField* pb) {
304 cxMempoolRegister(ctx->mp, pb, destroy_ui_pathtextfield);
305 }
306
307 static void ui_pathtextfield_clear(StackPanel& buttons) {
308 for (
int i = buttons.Children().Size() -
1; i >=
0; i--) {
309 buttons.Children().RemoveAt(i);
310 }
311 }
312
313 static void ui_pathfield_free_pathelms(UiPathElm* elms,
size_t nelm) {
314 if (!elms) {
315 return;
316 }
317 for (
int i =
0; i < nelm; i++) {
318 UiPathElm e = elms[i];
319 free(e.name);
320 free(e.path);
321 }
322 free(elms);
323 }
324
325 UiPathTextField::~UiPathTextField() {
326 ui_pathfield_free_pathelms(this->current_path, this->current_path_nelms);
327 }
328
329 static UiPathElm* default_pathelm_func(
const char* full_path,
size_t len,
size_t* ret_nelm,
void* data) {
330 cxstring *pathelms;
331 size_t nelm = cx_strsplit_a(cxDefaultAllocator, cx_strn(full_path, len),
CX_STR(
"/"),
4096, &pathelms);
332
333 if (nelm ==
0) {
334 *ret_nelm =
0;
335 return nullptr;
336 }
337
338 UiPathElm* elms = (UiPathElm*)calloc(nelm,
sizeof(UiPathElm));
339 size_t n = nelm;
340 int j =
0;
341 for (
int i =
0; i < nelm; i++) {
342 cxstring c = pathelms[i];
343 if (c.length ==
0) {
344 if (i ==
0) {
345 c.length =
1;
346 }
347 else {
348 n--;
349 continue;
350 }
351 }
352
353 cxmutstr m = cx_strdup(c);
354 elms[j].name = m.ptr;
355 elms[j].name_len = m.length;
356
357 size_t elm_path_len = c.ptr + c.length - full_path;
358 cxmutstr elm_path = cx_strdup(cx_strn(full_path, elm_path_len));
359 elms[j].path = elm_path.ptr;
360 elms[j].path_len = elm_path.length;
361
362 j++;
363 }
364 *ret_nelm = n;
365
366 return elms;
367 }
368
369 int ui_pathtextfield_update(UiPathTextField* pb,
const char *full_path) {
370 Grid grid = pb->grid;
371
372 ui_pathelm_func getpathelm = pb->getpathelm;
373 void* getpathelmdata = pb->getpathelmdata;
374
375 size_t full_path_len = full_path ? strlen(full_path) :
0;
376
377 size_t nelm =
0;
378 UiPathElm* path_elm = getpathelm(full_path, full_path_len, &nelm, getpathelmdata);
379 if (!path_elm) {
380 return 1;
381 }
382
383
384 pb->textbox.Visibility(Visibility::Collapsed);
385 pb->buttons.Visibility(Visibility::Visible);
386
387
388 ui_pathtextfield_clear(pb->buttons);
389
390 ui_pathfield_free_pathelms(pb->current_path, pb->current_path_nelms);
391 pb->current_path = path_elm;
392 pb->current_path_nelms = nelm;
393
394
395 int j =
0;
396 for (
int i =
0; i < nelm;i++) {
397 UiPathElm elm = path_elm[i];
398 wchar_t* wstr = str2wstr_len(elm.name, elm.name_len, nullptr);
399 Button button = Button();
400 button.Content(box_value(wstr));
401 free(wstr);
402
403 if (pb->onactivate) {
404 button.Click([pb, j, elm](IInspectable
const& sender, RoutedEventArgs) {
405
406 cxmutstr elmpath = cx_strdup(cx_strn(elm.path, elm.path_len));
407
408 UiEvent evt;
409 evt.obj = pb->obj;
410 evt.window = evt.obj->window;
411 evt.document = evt.obj->ctx->document;
412 evt.eventdata = elmpath.ptr;
413 evt.intval = j;
414 pb->onactivate(&evt, pb->onactivatedata);
415
416 free(elmpath.ptr);
417 });
418 }
419
420 Thickness t = {
0,
0,
1,
0 };
421 CornerRadius c = {
0 ,
0,
0,
0 };
422 button.BorderThickness(t);
423 button.CornerRadius(c);
424
425 pb->buttons.Children().Append(button);
426
427 j++;
428 }
429
430 return 0;
431 }
432
433 char* ui_path_textfield_get(UiString * str) {
434 UiPathTextField* widget = (UiPathTextField*)str->obj;
435 TextBox box = widget->textbox;
436 std::wstring wstr(box.Text());
437 return ui_wstring_get(str, wstr);
438 }
439
440 void ui_path_textfield_set(UiString* str,
const char* newvalue) {
441 UiPathTextField* widget = (UiPathTextField*)str->obj;
442 TextBox box = widget->textbox;
443 box.Text(ui_wstring_set(str, newvalue));
444 ui_pathtextfield_update(widget, newvalue);
445 }
446
447 UIEXPORT UIWIDGET ui_path_textfield_create(UiObject* obj, UiPathTextFieldArgs args) {
448 UiObject* current = uic_current_obj(obj);
449
450
451 Border pathbar = Border();
452
453 IInspectable bgRes = Application::Current().Resources().Lookup(box_value(
"TextControlBackground"L));
454 IInspectable borderThicknessRes = Application::Current().Resources().Lookup(box_value(
"TextControlBorderThemeThickness"L));
455 IInspectable borderBrushRes = Application::Current().Resources().Lookup(box_value(
"TextControlBorderBrush"L));
456
457
458 Brush bgBrush = unbox_value<Brush>(bgRes);
459 Thickness border = unbox_value<Thickness>(borderThicknessRes);
460 Brush borderBrush = unbox_value<Brush>(borderBrushRes);
461 CornerRadius cornerRadius = {
4,
4,
4,
4 };
462
463 pathbar.Background(bgBrush);
464 pathbar.BorderBrush(borderBrush);
465 pathbar.BorderThickness(border);
466 pathbar.CornerRadius(cornerRadius);
467
468 Grid content = Grid();
469 pathbar.Child(content);
470
471 GridLength gl;
472 gl.Value =
0;
473 gl.GridUnitType = GridUnitType::Auto;
474
475 ColumnDefinition coldef = ColumnDefinition();
476 coldef.Width(gl);
477 content.ColumnDefinitions().Append(coldef);
478
479 gl.Value =
1;
480 gl.GridUnitType = GridUnitType::Star;
481
482 ColumnDefinition coldef2 = ColumnDefinition();
483 coldef2.Width(gl);
484 content.ColumnDefinitions().Append(coldef2);
485
486 TextBox pathTextBox = TextBox();
487 Thickness t = {
0,
0,
0,
0 };
488 CornerRadius c = {
0 ,
0,
0,
0 };
489 pathTextBox.BorderThickness(t);
490
491
492
493 pathTextBox.HorizontalAlignment(HorizontalAlignment::Stretch);
494 content.SetColumn(pathTextBox,
0);
495 content.SetColumnSpan(pathTextBox,
2);
496
497 content.Children().Append(pathTextBox);
498
499
500 StackPanel buttons = StackPanel();
501 buttons.Orientation(Orientation::Horizontal);
502 buttons.Visibility(Visibility::Collapsed);
503 content.SetColumn(buttons,
0);
504 content.Children().Append(buttons);
505
506 TextBlock filler = TextBlock();
507 filler.VerticalAlignment(VerticalAlignment::Stretch);
508
509
510 filler.HorizontalAlignment(HorizontalAlignment::Stretch);
511 filler.VerticalAlignment(VerticalAlignment::Stretch);
512 content.SetColumn(filler,
1);
513 content.Children().Append(filler);
514
515 filler.PointerPressed(
516 winrt::Microsoft::
UI::Xaml::Input::PointerEventHandler(
517 [=](IInspectable
const& sender, winrt::Microsoft::
UI::Xaml::Input::PointerRoutedEventArgs
const& args) {
518 pathTextBox.Visibility(Visibility::Visible);
519 buttons.Visibility(Visibility::Collapsed);
520 filler.Visibility(Visibility::Collapsed);
521 pathTextBox.SelectionStart(pathTextBox.Text().size());
522 pathTextBox.SelectionLength(
0);
523 pathTextBox.Focus(FocusState::Keyboard);
524 })
525 );
526
527
528
529 UiPathTextField* uipathbar = new UiPathTextField;
530 ui_context_add_pathtextfield_destructor(current->ctx, uipathbar);
531 uipathbar->grid = content;
532 uipathbar->buttons = buttons;
533 uipathbar->textbox = pathTextBox;
534 uipathbar->filler = filler;
535 uipathbar->obj = obj;
536 uipathbar->getpathelm = args.getpathelm ? args.getpathelm : default_pathelm_func;
537 uipathbar->getpathelmdata = args.getpathelmdata;
538 uipathbar->onactivate = args.onactivate;
539 uipathbar->onactivatedata = args.onactivatedata;
540 uipathbar->ondragstart = args.ondragstart;
541 uipathbar->ondragstartdata = args.ondragstartdata;
542 uipathbar->ondragcomplete = args.ondragcomplete;
543 uipathbar->ondragcompletedata = args.ondragcompletedata;
544 uipathbar->ondrop = args.ondrop;
545 uipathbar->ondropdata = args.ondropsdata;
546
547
548 pathTextBox.KeyDown(
549 winrt::Microsoft::
UI::Xaml::Input::KeyEventHandler(
550 [=](winrt::Windows::Foundation::IInspectable
const& sender, winrt::Microsoft::
UI::Xaml::Input::KeyRoutedEventArgs
const& e) {
551 auto key = e.Key();
552 bool showButtons = false;
553 bool update = false;
554 if (key == Windows::System::VirtualKey::Escape) {
555 showButtons = true;
556 }
557 else if (key == Windows::System::VirtualKey::Enter) {
558 showButtons = true;
559 update = true;
560 }
561
562 if (showButtons) {
563 pathTextBox.Visibility(Visibility::Collapsed);
564 buttons.Visibility(Visibility::Visible);
565 filler.Visibility(Visibility::Visible);
566 if (update) {
567 std::wstring value(pathTextBox.Text());
568 char* full_path = wchar2utf8(value.c_str(), value.length());
569
570 if (!ui_pathtextfield_update(uipathbar, full_path)) {
571 UiEvent evt;
572 evt.obj = obj;
573 evt.window = obj->window;
574 evt.document = obj->ctx->document;
575 evt.eventdata = full_path;
576 evt.intval = -
1;
577 args.onactivate(&evt, args.onactivatedata);
578 }
579
580 free(full_path);
581 }
582
583
584 }
585 })
586 );
587
588
589 UIElement elm = pathbar;
590 UiWidget* widget = new UiWidget(elm);
591 widget->data1 = uipathbar;
592 ui_context_add_widget_destructor(current->ctx, widget);
593
594
595 UiVar* var = uic_widget_var(obj->ctx, current->ctx, args.value, args.varname,
UI_VAR_STRING);
596 if (var) {
597 UiString* value = (UiString*)var->value;
598 value->obj = uipathbar;
599 value->get = ui_path_textfield_get;
600 value->set = ui_path_textfield_set;
601 }
602
603
604 UI_APPLY_LAYOUT1(current, args);
605
606 current->container->Add(pathbar, false);
607
608 return widget;
609 }
610