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

serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export 0.11.1
Last change on this file since b0858150 was a85d5c6, checked in by Jiri Svoboda <jiri@…>, 4 years ago

Fix window resize when not using window double-buffer

We can only create the direct mapped bitmap after the window has been
resized.

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