source: mainline/uspace/lib/ui/src/window.c@ 344f8b9

Last change on this file since 344f8b9 was 344f8b9, checked in by Jiri Svoboda <jiri@…>, 4 years ago

Open menu in separate popup window (WIP)

  • Property mode set to 100644
File size: 22.6 KB
Line 
1/*
2 * Copyright (c) 2021 Jiri Svoboda
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @addtogroup libui
30 * @{
31 */
32/**
33 * @file Window
34 */
35
36#include <congfx/console.h>
37#include <display.h>
38#include <errno.h>
39#include <gfx/bitmap.h>
40#include <gfx/context.h>
41#include <gfx/render.h>
42#include <io/kbd_event.h>
43#include <io/pos_event.h>
44#include <mem.h>
45#include <memgfx/memgc.h>
46#include <stdlib.h>
47#include <ui/control.h>
48#include <ui/resource.h>
49#include <ui/ui.h>
50#include <ui/wdecor.h>
51#include <ui/window.h>
52#include "../private/control.h"
53#include "../private/dummygc.h"
54#include "../private/resource.h"
55#include "../private/ui.h"
56#include "../private/wdecor.h"
57#include "../private/window.h"
58
59static void dwnd_close_event(void *);
60static void dwnd_focus_event(void *);
61static void dwnd_kbd_event(void *, kbd_event_t *);
62static void dwnd_pos_event(void *, pos_event_t *);
63static void dwnd_resize_event(void *, gfx_rect_t *);
64static void dwnd_unfocus_event(void *);
65
66static display_wnd_cb_t dwnd_cb = {
67 .close_event = dwnd_close_event,
68 .focus_event = dwnd_focus_event,
69 .kbd_event = dwnd_kbd_event,
70 .pos_event = dwnd_pos_event,
71 .resize_event = dwnd_resize_event,
72 .unfocus_event = dwnd_unfocus_event
73};
74
75static void wd_close(ui_wdecor_t *, void *);
76static void wd_move(ui_wdecor_t *, void *, gfx_coord2_t *);
77static void wd_resize(ui_wdecor_t *, void *, ui_wdecor_rsztype_t,
78 gfx_coord2_t *);
79static void wd_set_cursor(ui_wdecor_t *, void *, ui_stock_cursor_t);
80
81static ui_wdecor_cb_t wdecor_cb = {
82 .close = wd_close,
83 .move = wd_move,
84 .resize = wd_resize,
85 .set_cursor = wd_set_cursor
86};
87
88static void ui_window_invalidate(void *, gfx_rect_t *);
89static void ui_window_update(void *);
90static void ui_window_app_invalidate(void *, gfx_rect_t *);
91static void ui_window_app_update(void *);
92static void ui_window_expose_cb(void *);
93
94/** Initialize window parameters structure.
95 *
96 * Window parameters structure must always be initialized using this function
97 * first. By default, the window will be decorated. To get a non-decorated
98 * window, one needs to clear ui_wds_decorated
99 * (e.g. params->style &= ~ui_wds_decorated).
100 *
101 * @param params Window parameters structure
102 */
103void ui_wnd_params_init(ui_wnd_params_t *params)
104{
105 memset(params, 0, sizeof(ui_wnd_params_t));
106
107 /* Make window decorated by default. */
108 params->style = ui_wds_decorated;
109}
110
111/** Create new window.
112 *
113 * @param ui User interface
114 * @param params Window parameters
115 * @param rwindow Place to store pointer to new window
116 * @return EOK on success or an error code
117 */
118errno_t ui_window_create(ui_t *ui, ui_wnd_params_t *params,
119 ui_window_t **rwindow)
120{
121 ui_window_t *window;
122 display_info_t info;
123 gfx_coord2_t pos;
124 gfx_coord2_t scr_dims;
125 display_wnd_params_t dparams;
126 display_window_t *dwindow = NULL;
127 gfx_context_t *gc = NULL;
128 ui_resource_t *res = NULL;
129 ui_wdecor_t *wdecor = NULL;
130 dummy_gc_t *dgc = NULL;
131 gfx_bitmap_params_t bparams;
132 gfx_bitmap_alloc_t alloc;
133 gfx_bitmap_t *bmp = NULL;
134 mem_gc_t *memgc = NULL;
135 errno_t rc;
136
137 window = calloc(1, sizeof(ui_window_t));
138 if (window == NULL)
139 return ENOMEM;
140
141 display_wnd_params_init(&dparams);
142 dparams.rect = params->rect;
143 /* Only allow making the window larger */
144 gfx_rect_dims(&params->rect, &dparams.min_size);
145
146 if (ui->display != NULL) {
147 if (params->placement != ui_wnd_place_default) {
148 rc = display_get_info(ui->display, &info);
149 if (rc != EOK)
150 goto error;
151 }
152
153 if (params->placement == ui_wnd_place_full_screen) {
154 /* Make window the size of the screen */
155 gfx_rect_dims(&info.rect, &scr_dims);
156 gfx_coord2_add(&dparams.rect.p0, &scr_dims,
157 &dparams.rect.p1);
158 }
159
160 rc = display_window_create(ui->display, &dparams, &dwnd_cb,
161 (void *) window, &dwindow);
162 if (rc != EOK)
163 goto error;
164
165 if (params->placement != ui_wnd_place_default) {
166 pos.x = 0;
167 pos.y = 0;
168
169 switch (params->placement) {
170 case ui_wnd_place_default:
171 assert(false);
172 case ui_wnd_place_top_left:
173 case ui_wnd_place_full_screen:
174 pos.x = info.rect.p0.x - params->rect.p0.x;
175 pos.y = info.rect.p0.y - params->rect.p0.y;
176 break;
177 case ui_wnd_place_top_right:
178 pos.x = info.rect.p1.x - params->rect.p1.x;
179 pos.y = info.rect.p0.y - params->rect.p0.y;
180 break;
181 case ui_wnd_place_bottom_left:
182 pos.x = info.rect.p0.x - params->rect.p0.x;
183 pos.y = info.rect.p1.y - params->rect.p1.y;
184 break;
185 case ui_wnd_place_bottom_right:
186 pos.x = info.rect.p1.x - params->rect.p1.x;
187 pos.y = info.rect.p1.y - params->rect.p1.y;
188 break;
189 }
190
191 rc = display_window_move(dwindow, &pos);
192 if (rc != EOK)
193 goto error;
194 }
195
196 rc = display_window_get_gc(dwindow, &gc);
197 if (rc != EOK)
198 goto error;
199 } else if (ui->console != NULL) {
200 gc = console_gc_get_ctx(ui->cgc);
201 } else {
202 /* Needed for unit tests */
203 rc = dummygc_create(&dgc);
204 if (rc != EOK)
205 goto error;
206
207 gc = dummygc_get_ctx(dgc);
208 }
209
210#ifdef CONFIG_UI_CS_RENDER
211 /* Create window bitmap */
212 gfx_bitmap_params_init(&bparams);
213#ifndef CONFIG_WIN_DOUBLE_BUF
214 /* Console does not support direct output */
215 if (ui->display != NULL)
216 bparams.flags |= bmpf_direct_output;
217#endif
218
219 /* Move rectangle so that top-left corner is 0,0 */
220 gfx_rect_rtranslate(&params->rect.p0, &params->rect, &bparams.rect);
221
222 rc = gfx_bitmap_create(gc, &bparams, NULL, &bmp);
223 if (rc != EOK)
224 goto error;
225
226 /* Create memory GC */
227 rc = gfx_bitmap_get_alloc(bmp, &alloc);
228 if (rc != EOK) {
229 gfx_bitmap_destroy(window->app_bmp);
230 return rc;
231 }
232
233 rc = mem_gc_create(&bparams.rect, &alloc, ui_window_invalidate,
234 ui_window_update, (void *) window, &memgc);
235 if (rc != EOK) {
236 gfx_bitmap_destroy(window->app_bmp);
237 return rc;
238 }
239
240 window->bmp = bmp;
241 window->mgc = memgc;
242 window->gc = mem_gc_get_ctx(memgc);
243 window->realgc = gc;
244#else
245 (void) ui_window_update;
246 (void) ui_window_invalidate;
247 (void) alloc;
248 (void) bparams;
249 window->gc = gc;
250#endif
251
252 rc = ui_resource_create(window->gc, ui_is_textmode(ui), &res);
253 if (rc != EOK)
254 goto error;
255
256 rc = ui_wdecor_create(res, params->caption, params->style, &wdecor);
257 if (rc != EOK)
258 goto error;
259
260 ui_wdecor_set_rect(wdecor, &dparams.rect);
261 ui_wdecor_set_cb(wdecor, &wdecor_cb, (void *) window);
262 ui_wdecor_paint(wdecor);
263
264 ui_resource_set_expose_cb(res, ui_window_expose_cb, (void *) window);
265
266 window->ui = ui;
267 window->dwindow = dwindow;
268 window->rect = dparams.rect;
269
270 window->res = res;
271 window->wdecor = wdecor;
272 window->cursor = ui_curs_arrow;
273 *rwindow = window;
274
275 list_append(&window->lwindows, &ui->windows);
276 return EOK;
277error:
278 if (wdecor != NULL)
279 ui_wdecor_destroy(wdecor);
280 if (res != NULL)
281 ui_resource_destroy(res);
282 if (memgc != NULL)
283 mem_gc_delete(memgc);
284 if (bmp != NULL)
285 gfx_bitmap_destroy(bmp);
286 if (dgc != NULL)
287 dummygc_destroy(dgc);
288 if (dwindow != NULL)
289 display_window_destroy(dwindow);
290 free(window);
291 return rc;
292}
293
294/** Destroy window.
295 *
296 * @param window Window or @c NULL
297 */
298void ui_window_destroy(ui_window_t *window)
299{
300 ui_t *ui;
301
302 if (window == NULL)
303 return;
304
305 ui = window->ui;
306
307 list_remove(&window->lwindows);
308 ui_control_destroy(window->control);
309 ui_wdecor_destroy(window->wdecor);
310 ui_resource_destroy(window->res);
311 if (0 && window->app_mgc != NULL)
312 mem_gc_delete(window->app_mgc);
313 if (0 && window->app_bmp != NULL)
314 gfx_bitmap_destroy(window->app_bmp);
315 if (window->mgc != NULL) {
316 mem_gc_delete(window->mgc);
317 window->gc = NULL;
318 }
319 if (window->bmp != NULL)
320 gfx_bitmap_destroy(window->bmp);
321 gfx_context_delete(window->gc);
322 if (window->dwindow != NULL)
323 display_window_destroy(window->dwindow);
324
325 free(window);
326
327 /* Need to repaint if windows are emulated */
328 if (ui_is_fullscreen(ui)) {
329 ui_paint(ui);
330 }
331}
332
333/** Add control to window.
334 *
335 * Only one control can be added to a window. If more than one control
336 * is added, the results are undefined.
337 *
338 * @param window Window
339 * @param control Control
340 * @return EOK on success, ENOMEM if out of memory
341 */
342void ui_window_add(ui_window_t *window, ui_control_t *control)
343{
344 assert(window->control == NULL);
345
346 window->control = control;
347 control->elemp = (void *) window;
348}
349
350/** Remove control from window.
351 *
352 * @param window Window
353 * @param control Control
354 */
355void ui_window_remove(ui_window_t *window, ui_control_t *control)
356{
357 assert(window->control == control);
358 assert((ui_window_t *) control->elemp == window);
359
360 window->control = NULL;
361 control->elemp = NULL;
362}
363
364/** Get active window (only valid in fullscreen mode).
365 *
366 * @param ui User interface
367 * @return Active window
368 */
369ui_window_t *ui_window_get_active(ui_t *ui)
370{
371 link_t *link;
372
373 link = list_last(&ui->windows);
374 if (link == NULL)
375 return NULL;
376
377 return list_get_instance(link, ui_window_t, lwindows);
378}
379
380/** Resize/move window.
381 *
382 * Resize window to the dimensions of @a rect. If @a rect.p0 is not 0,0,
383 * the top-left corner of the window will move on the screen accordingly.
384 *
385 * @param window Window
386 * @param rect Rectangle
387 *
388 * @return EOK on success or an error code
389 */
390errno_t ui_window_resize(ui_window_t *window, gfx_rect_t *rect)
391{
392 gfx_coord2_t offs;
393 gfx_rect_t nrect;
394 gfx_rect_t arect;
395 gfx_bitmap_t *app_bmp = NULL;
396 gfx_bitmap_t *win_bmp = NULL;
397 gfx_bitmap_params_t app_params;
398 gfx_bitmap_params_t win_params;
399 gfx_bitmap_alloc_t app_alloc;
400 gfx_bitmap_alloc_t win_alloc;
401 errno_t rc;
402
403 /*
404 * Move rect so that p0=0,0 - keep window's coordinate system origin
405 * locked to top-left corner of the window.
406 */
407 offs = rect->p0;
408 gfx_rect_rtranslate(&offs, rect, &nrect);
409
410 /* mgc != NULL iff client-side rendering */
411 if (window->mgc != NULL) {
412#ifdef CONFIG_WIN_DOUBLE_BUF
413 /*
414 * Create new window bitmap in advance. If direct mapping,
415 * will need do it after resizing the window.
416 */
417 assert(window->bmp != NULL);
418 gfx_bitmap_params_init(&win_params);
419 win_params.rect = nrect;
420
421 rc = gfx_bitmap_create(window->realgc, &win_params, NULL,
422 &win_bmp);
423 if (rc != EOK)
424 goto error;
425
426 rc = gfx_bitmap_get_alloc(win_bmp, &win_alloc);
427 if (rc != EOK)
428 goto error;
429#endif
430 }
431
432 /* Application area GC? */
433 if (window->app_gc != NULL) {
434 /* Resize application bitmap */
435 assert(window->app_bmp != NULL);
436
437 gfx_bitmap_params_init(&app_params);
438
439 /*
440 * The bitmap will have the same dimensions as the
441 * application rectangle, but start at 0,0.
442 */
443 ui_wdecor_app_from_rect(window->wdecor->style, &nrect, &arect);
444 gfx_rect_rtranslate(&arect.p0, &arect, &app_params.rect);
445
446 rc = gfx_bitmap_create(window->gc, &app_params, NULL,
447 &app_bmp);
448 if (rc != EOK)
449 goto error;
450
451 rc = gfx_bitmap_get_alloc(app_bmp, &app_alloc);
452 if (rc != EOK)
453 goto error;
454 }
455
456 /* dwindow can be NULL in case of unit tests */
457 if (window->dwindow != NULL) {
458 rc = display_window_resize(window->dwindow, &offs, &nrect);
459 if (rc != EOK)
460 goto error;
461 }
462
463 /* Client side rendering? */
464 if (window->mgc != NULL) {
465#ifndef CONFIG_WIN_DOUBLE_BUF
466 /* Window is resized, now we can map the window bitmap again */
467 gfx_bitmap_params_init(&win_params);
468 win_params.flags |= bmpf_direct_output;
469 win_params.rect = nrect;
470
471 rc = gfx_bitmap_create(window->realgc, &win_params, NULL,
472 &win_bmp);
473 if (rc != EOK)
474 goto error;
475
476 rc = gfx_bitmap_get_alloc(win_bmp, &win_alloc);
477 if (rc != EOK)
478 goto error;
479#endif
480
481 mem_gc_retarget(window->mgc, &win_params.rect, &win_alloc);
482
483 gfx_bitmap_destroy(window->bmp);
484 window->bmp = win_bmp;
485 }
486
487 ui_wdecor_set_rect(window->wdecor, &nrect);
488 ui_wdecor_paint(window->wdecor);
489 gfx_update(window->gc);
490
491 /* Application area GC? */
492 if (window->app_gc != NULL) {
493 mem_gc_retarget(window->app_mgc, &app_params.rect, &app_alloc);
494
495 gfx_bitmap_destroy(window->app_bmp);
496 window->app_bmp = app_bmp;
497 }
498
499 return EOK;
500error:
501 if (app_bmp != NULL)
502 gfx_bitmap_destroy(app_bmp);
503 if (win_bmp != NULL)
504 gfx_bitmap_destroy(win_bmp);
505 return rc;
506}
507
508/** Set window callbacks.
509 *
510 * @param window Window
511 * @param cb Window callbacks
512 * @param arg Callback argument
513 */
514void ui_window_set_cb(ui_window_t *window, ui_window_cb_t *cb, void *arg)
515{
516 window->cb = cb;
517 window->arg = arg;
518}
519
520/** Get UI resource from window.
521 *
522 * @param window Window
523 * @return UI resource
524 */
525ui_resource_t *ui_window_get_res(ui_window_t *window)
526{
527 return window->res;
528}
529
530/** Get window GC.
531 *
532 * @param window Window
533 * @return GC (relative to window)
534 */
535gfx_context_t *ui_window_get_gc(ui_window_t *window)
536{
537 return window->gc;
538}
539
540/** Get window application area GC
541 *
542 * @param window Window
543 * @param rgc Place to store GC (relative to application area)
544 * @return EOK on success or an error code
545 */
546errno_t ui_window_get_app_gc(ui_window_t *window, gfx_context_t **rgc)
547{
548 gfx_bitmap_params_t params;
549 gfx_bitmap_alloc_t alloc;
550 gfx_rect_t rect;
551 mem_gc_t *memgc;
552 errno_t rc;
553
554 if (window->app_gc == NULL) {
555 assert(window->app_bmp == NULL);
556
557 gfx_bitmap_params_init(&params);
558
559 /*
560 * The bitmap will have the same dimensions as the
561 * application rectangle, but start at 0,0.
562 */
563 ui_window_get_app_rect(window, &rect);
564 gfx_rect_rtranslate(&rect.p0, &rect, &params.rect);
565
566 rc = gfx_bitmap_create(window->gc, &params, NULL,
567 &window->app_bmp);
568 if (rc != EOK)
569 return rc;
570
571 rc = gfx_bitmap_get_alloc(window->app_bmp, &alloc);
572 if (rc != EOK) {
573 gfx_bitmap_destroy(window->app_bmp);
574 return rc;
575 }
576
577 rc = mem_gc_create(&params.rect, &alloc, ui_window_app_invalidate,
578 ui_window_app_update, (void *) window, &memgc);
579 if (rc != EOK) {
580 gfx_bitmap_destroy(window->app_bmp);
581 return rc;
582 }
583
584 window->app_mgc = memgc;
585 window->app_gc = mem_gc_get_ctx(memgc);
586 }
587
588 *rgc = window->app_gc;
589 return EOK;
590}
591
592/** Get window application rectangle
593 *
594 * @param window Window
595 * @param rect Place to store application rectangle
596 */
597void ui_window_get_app_rect(ui_window_t *window, gfx_rect_t *rect)
598{
599 ui_wdecor_geom_t geom;
600
601 ui_wdecor_get_geom(window->wdecor, &geom);
602 *rect = geom.app_area_rect;
603}
604
605/** Paint window
606 *
607 * @param window Window
608 * @return EOK on success or an error code
609 */
610errno_t ui_window_paint(ui_window_t *window)
611{
612 return ui_window_send_paint(window);
613}
614
615/** Handle window close event. */
616static void dwnd_close_event(void *arg)
617{
618 ui_window_t *window = (ui_window_t *) arg;
619
620 ui_window_send_close(window);
621}
622
623/** Handle window focus event. */
624static void dwnd_focus_event(void *arg)
625{
626 ui_window_t *window = (ui_window_t *) arg;
627
628 if (window->wdecor != NULL) {
629 ui_wdecor_set_active(window->wdecor, true);
630 ui_wdecor_paint(window->wdecor);
631 }
632
633 ui_window_send_focus(window);
634}
635
636/** Handle window keyboard event */
637static void dwnd_kbd_event(void *arg, kbd_event_t *kbd_event)
638{
639 ui_window_t *window = (ui_window_t *) arg;
640
641 (void) window;
642 ui_window_send_kbd(window, kbd_event);
643}
644
645/** Handle window position event */
646static void dwnd_pos_event(void *arg, pos_event_t *event)
647{
648 ui_window_t *window = (ui_window_t *) arg;
649
650 /* Make sure we don't process events until fully initialized */
651 if (window->wdecor == NULL)
652 return;
653
654 ui_wdecor_pos_event(window->wdecor, event);
655 ui_window_send_pos(window, event);
656}
657
658/** Handle window resize event */
659static void dwnd_resize_event(void *arg, gfx_rect_t *rect)
660{
661 ui_window_t *window = (ui_window_t *) arg;
662
663 /* Make sure we don't process events until fully initialized */
664 if (window->wdecor == NULL)
665 return;
666
667 if ((window->wdecor->style & ui_wds_resizable) == 0)
668 return;
669
670 (void) ui_window_resize(window, rect);
671 (void) ui_window_paint(window);
672}
673
674/** Handle window unfocus event. */
675static void dwnd_unfocus_event(void *arg)
676{
677 ui_window_t *window = (ui_window_t *) arg;
678
679 if (window->wdecor != NULL) {
680 ui_wdecor_set_active(window->wdecor, false);
681 ui_wdecor_paint(window->wdecor);
682 }
683
684 ui_window_send_unfocus(window);
685}
686
687/** Window decoration requested window closure.
688 *
689 * @param wdecor Window decoration
690 * @param arg Argument (window)
691 */
692static void wd_close(ui_wdecor_t *wdecor, void *arg)
693{
694 ui_window_t *window = (ui_window_t *) arg;
695
696 ui_window_send_close(window);
697}
698
699/** Window decoration requested window move.
700 *
701 * @param wdecor Window decoration
702 * @param arg Argument (window)
703 * @param pos Position where the title bar was pressed
704 */
705static void wd_move(ui_wdecor_t *wdecor, void *arg, gfx_coord2_t *pos)
706{
707 ui_window_t *window = (ui_window_t *) arg;
708
709 if (window->dwindow != NULL)
710 (void) display_window_move_req(window->dwindow, pos);
711}
712
713/** Window decoration requested window resize.
714 *
715 * @param wdecor Window decoration
716 * @param arg Argument (window)
717 * @param rsztype Resize type
718 * @param pos Position where the button was pressed
719 */
720static void wd_resize(ui_wdecor_t *wdecor, void *arg,
721 ui_wdecor_rsztype_t rsztype, gfx_coord2_t *pos)
722{
723 ui_window_t *window = (ui_window_t *) arg;
724
725 if (window->dwindow != NULL)
726 (void) display_window_resize_req(window->dwindow, rsztype, pos);
727}
728
729/** Window decoration requested changing cursor.
730 *
731 * @param wdecor Window decoration
732 * @param arg Argument (window)
733 * @param cursor Cursor to set
734 */
735static void wd_set_cursor(ui_wdecor_t *wdecor, void *arg,
736 ui_stock_cursor_t cursor)
737{
738 ui_window_t *window = (ui_window_t *) arg;
739 display_stock_cursor_t dcursor;
740
741 if (cursor == window->cursor)
742 return;
743
744 dcursor = dcurs_arrow;
745
746 switch (cursor) {
747 case ui_curs_arrow:
748 dcursor = dcurs_arrow;
749 break;
750 case ui_curs_size_ud:
751 dcursor = dcurs_size_ud;
752 break;
753 case ui_curs_size_lr:
754 dcursor = dcurs_size_lr;
755 break;
756 case ui_curs_size_uldr:
757 dcursor = dcurs_size_uldr;
758 break;
759 case ui_curs_size_urdl:
760 dcursor = dcurs_size_urdl;
761 break;
762 }
763
764 if (window->dwindow != NULL)
765 (void) display_window_set_cursor(window->dwindow, dcursor);
766 window->cursor = cursor;
767}
768
769/** Send window close event.
770 *
771 * @param window Window
772 */
773void ui_window_send_close(ui_window_t *window)
774{
775 if (window->cb != NULL && window->cb->close != NULL)
776 window->cb->close(window, window->arg);
777}
778
779/** Send window focus event.
780 *
781 * @param window Window
782 */
783void ui_window_send_focus(ui_window_t *window)
784{
785 if (window->cb != NULL && window->cb->focus != NULL)
786 window->cb->focus(window, window->arg);
787}
788
789/** Send window keyboard event.
790 *
791 * @param window Window
792 */
793void ui_window_send_kbd(ui_window_t *window, kbd_event_t *kbd)
794{
795 if (window->cb != NULL && window->cb->kbd != NULL)
796 window->cb->kbd(window, window->arg, kbd);
797}
798
799/** Send window paint event.
800 *
801 * @param window Window
802 */
803errno_t ui_window_send_paint(ui_window_t *window)
804{
805 if (window->cb != NULL && window->cb->paint != NULL)
806 return window->cb->paint(window, window->arg);
807 else
808 return ui_window_def_paint(window);
809}
810
811/** Send window position event.
812 *
813 * @param window Window
814 */
815void ui_window_send_pos(ui_window_t *window, pos_event_t *pos)
816{
817 if (window->cb != NULL && window->cb->pos != NULL)
818 window->cb->pos(window, window->arg, pos);
819 else
820 ui_window_def_pos(window, pos);
821}
822
823/** Send window unfocus event.
824 *
825 * @param window Window
826 */
827void ui_window_send_unfocus(ui_window_t *window)
828{
829 if (window->cb != NULL && window->cb->unfocus != NULL)
830 window->cb->unfocus(window, window->arg);
831 else
832 return ui_window_def_unfocus(window);
833}
834
835/** Default window paint routine.
836 *
837 * @param window Window
838 * @return EOK on success or an error code
839 */
840errno_t ui_window_def_paint(ui_window_t *window)
841{
842 gfx_rect_t app_rect;
843 errno_t rc;
844
845 rc = gfx_set_color(window->gc, window->res->wnd_face_color);
846 if (rc != EOK)
847 return rc;
848
849 ui_window_get_app_rect(window, &app_rect);
850
851 rc = gfx_fill_rect(window->gc, &app_rect);
852 if (rc != EOK)
853 return rc;
854
855 if (window->control != NULL)
856 return ui_control_paint(window->control);
857
858 rc = gfx_update(window->res->gc);
859 if (rc != EOK)
860 return rc;
861
862 return EOK;
863}
864
865/** Default window position event routine.
866 *
867 * @param window Window
868 */
869void ui_window_def_pos(ui_window_t *window, pos_event_t *pos)
870{
871 if (window->control != NULL)
872 ui_control_pos_event(window->control, pos);
873}
874
875/** Default window unfocus routine.
876 *
877 * @param window Window
878 * @return EOK on success or an error code
879 */
880void ui_window_def_unfocus(ui_window_t *window)
881{
882 if (window->control != NULL)
883 ui_control_unfocus(window->control);
884}
885
886/** Window invalidate callback
887 *
888 * @param arg Argument (ui_window_t *)
889 * @param rect Rectangle to update
890 */
891static void ui_window_invalidate(void *arg, gfx_rect_t *rect)
892{
893 ui_window_t *window = (ui_window_t *) arg;
894 gfx_rect_t env;
895
896 gfx_rect_envelope(&window->dirty_rect, rect, &env);
897 window->dirty_rect = env;
898}
899
900/** Window update callback
901 *
902 * @param arg Argument (ui_window_t *)
903 */
904static void ui_window_update(void *arg)
905{
906 ui_window_t *window = (ui_window_t *) arg;
907
908 if (!gfx_rect_is_empty(&window->dirty_rect))
909 (void) gfx_bitmap_render(window->bmp, &window->dirty_rect, NULL);
910
911 window->dirty_rect.p0.x = 0;
912 window->dirty_rect.p0.y = 0;
913 window->dirty_rect.p1.x = 0;
914 window->dirty_rect.p1.y = 0;
915}
916
917/** Application area invalidate callback
918 *
919 * @param arg Argument (ui_window_t *)
920 * @param rect Rectangle to update
921 */
922static void ui_window_app_invalidate(void *arg, gfx_rect_t *rect)
923{
924 ui_window_t *window = (ui_window_t *) arg;
925 gfx_rect_t arect;
926
927 ui_window_get_app_rect(window, &arect);
928
929 /* Render bitmap rectangle inside the application area */
930 (void) gfx_bitmap_render(window->app_bmp, rect, &arect.p0);
931 /*
932 * TODO Update applications to call gfx_update(), then
933 * we can defer update to ui_window_app_update().
934 */
935 (void) gfx_update(window->res->gc);
936}
937
938/** Application area update callback
939 *
940 * @param arg Argument (ui_window_t *)
941 */
942static void ui_window_app_update(void *arg)
943{
944 ui_window_t *window = (ui_window_t *) arg;
945
946 /*
947 * Not used since display is updated immediately
948 * in ui_window_app_invalidate
949 */
950 (void) window;
951}
952
953/** Window expose callback. */
954static void ui_window_expose_cb(void *arg)
955{
956 ui_window_t *window = (ui_window_t *) arg;
957
958 ui_window_paint(window);
959}
960
961/** @}
962 */
Note: See TracBrowser for help on using the repository browser.