Changes in uspace/srv/hid/display/window.c [0d00e53:1215db9] in mainline
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
uspace/srv/hid/display/window.c
r0d00e53 r1215db9 1 1 /* 2 * Copyright (c) 202 4Jiri Svoboda2 * Copyright (c) 2021 Jiri Svoboda 3 3 * All rights reserved. 4 4 * … … 44 44 #include <memgfx/memgc.h> 45 45 #include <stdlib.h> 46 #include <str.h>47 #include <wndmgt.h>48 46 #include "client.h" 49 47 #include "display.h" 50 48 #include "seat.h" 51 49 #include "window.h" 52 #include "wmclient.h"53 50 54 51 static void ds_window_invalidate_cb(void *, gfx_rect_t *); … … 67 64 * @param client Client owning the window 68 65 * @param params Window parameters 69 * @param r wnd Place to store pointer to new window.66 * @param rgc Place to store pointer to new GC. 70 67 * 71 68 * @return EOK on success or an error code 72 69 */ 73 70 errno_t ds_window_create(ds_client_t *client, display_wnd_params_t *params, 74 ds_window_t **r wnd)71 ds_window_t **rgc) 75 72 { 76 73 ds_window_t *wnd = NULL; … … 88 85 } 89 86 90 /* Caption */91 wnd->caption = str_dup(params->caption);92 if (wnd->caption == NULL) {93 rc = ENOMEM;94 goto error;95 }96 97 wnd->flags = params->flags;98 99 87 ds_client_add_window(client, wnd); 100 88 ds_display_add_window(client->display, wnd); … … 102 90 gfx_bitmap_params_init(&bparams); 103 91 bparams.rect = params->rect; 104 105 /* Allocate window bitmap */106 92 107 93 dgc = ds_display_get_gc(wnd->display); … … 136 122 wnd->gc = mem_gc_get_ctx(wnd->mgc); 137 123 wnd->cursor = wnd->display->cursor[dcurs_arrow]; 124 wnd->flags = params->flags; 138 125 139 126 if ((params->flags & wndf_setpos) != 0) { … … 146 133 } 147 134 148 /* Determine which seat should own the window */ 149 if (params->idev_id != 0) 150 seat = ds_display_seat_by_idev(wnd->display, params->idev_id); 135 seat = ds_display_first_seat(client->display); 136 137 if ((params->flags & wndf_popup) != 0) 138 ds_seat_set_popup(seat, wnd); 151 139 else 152 seat = ds_display_default_seat(wnd->display); 153 154 /* Is this a popup window? */ 155 if ((params->flags & wndf_popup) != 0) { 156 ds_seat_set_popup(seat, wnd); 157 } else { 158 if ((params->flags & wndf_nofocus) == 0) 159 ds_seat_set_focus(seat, wnd); 160 } 161 162 /* Is this window a panel? */ 163 if ((params->flags & wndf_avoid) != 0) 164 ds_display_update_max_rect(wnd->display); 140 ds_seat_set_focus(seat, wnd); 165 141 166 142 (void) ds_display_paint(wnd->display, NULL); 167 143 168 *r wnd= wnd;144 *rgc = wnd; 169 145 return EOK; 170 146 error: 171 147 if (wnd != NULL) { 172 ds_client_remove_window(wnd);173 ds_display_remove_window(wnd);174 if (wnd->mgc != NULL)175 mem_gc_delete(wnd->mgc);176 148 if (wnd->bitmap != NULL) 177 149 gfx_bitmap_destroy(wnd->bitmap); 178 if (wnd->caption != NULL)179 free(wnd->caption);180 150 free(wnd); 181 151 } … … 193 163 194 164 disp = wnd->display; 195 196 ds_window_unfocus(wnd);197 165 198 166 ds_client_remove_window(wnd); 199 167 ds_display_remove_window(wnd); 200 168 201 if ((wnd->flags & wndf_avoid) != 0)202 ds_display_update_max_rect(disp);203 204 169 mem_gc_delete(wnd->mgc); 205 170 … … 207 172 gfx_bitmap_destroy(wnd->bitmap); 208 173 209 free(wnd->caption);210 174 free(wnd); 211 175 … … 219 183 void ds_window_bring_to_top(ds_window_t *wnd) 220 184 { 221 ds_display_window_to_top(wnd); 185 ds_display_t *disp = wnd->display; 186 187 ds_display_remove_window(wnd); 188 ds_display_add_window(disp, wnd); 222 189 (void) ds_display_paint(wnd->display, NULL); 223 190 } … … 231 198 { 232 199 return wnd->gc; 233 }234 235 /** Determine if window is visible.236 *237 * @param wnd Window238 * @return @c true iff window is visible239 */240 bool ds_window_is_visible(ds_window_t *wnd)241 {242 return (wnd->flags & wndf_minimized) == 0;243 200 } 244 201 … … 255 212 gfx_rect_t crect; 256 213 257 log_msg(LOG_DEFAULT, LVL_DEBUG2, "ds_window_paint"); 258 259 /* Skip painting the window if not visible */ 260 if (!ds_window_is_visible(wnd)) 261 return EOK; 214 log_msg(LOG_DEFAULT, LVL_DEBUG, "ds_window_paint"); 262 215 263 216 if (rect != NULL) { … … 404 357 bool newr; 405 358 406 log_msg(LOG_DEFAULT, LVL_DEBUG 2, "ds_window_repaint_preview");359 log_msg(LOG_DEFAULT, LVL_DEBUG, "ds_window_repaint_preview"); 407 360 408 361 /* … … 448 401 * @param wnd Window 449 402 * @param pos Position where mouse button was pressed 450 * @param pos_id Positioning device ID 451 */ 452 static void ds_window_start_move(ds_window_t *wnd, gfx_coord2_t *pos, 453 sysarg_t pos_id) 403 */ 404 static void ds_window_start_move(ds_window_t *wnd, gfx_coord2_t *pos) 454 405 { 455 406 log_msg(LOG_DEFAULT, LVL_DEBUG, "ds_window_start_move (%d, %d)", … … 460 411 461 412 wnd->orig_pos = *pos; 462 wnd->orig_pos_id = pos_id;463 413 wnd->state = dsw_moving; 464 414 wnd->preview_pos = wnd->dpos; … … 490 440 wnd->dpos = nwpos; 491 441 wnd->state = dsw_idle; 492 wnd->orig_pos_id = 0;493 442 494 443 (void) ds_display_paint(wnd->display, NULL); … … 506 455 gfx_rect_t old_rect; 507 456 508 log_msg(LOG_DEFAULT, LVL_DEBUG 2, "ds_window_update_move (%d, %d)",457 log_msg(LOG_DEFAULT, LVL_DEBUG, "ds_window_update_move (%d, %d)", 509 458 (int) pos->x, (int) pos->y); 510 459 … … 525 474 * @param rsztype Resize type (which part of window is being dragged) 526 475 * @param pos Position where mouse button was pressed 527 * @param pos_id Positioning device ID528 476 */ 529 477 static void ds_window_start_resize(ds_window_t *wnd, 530 display_wnd_rsztype_t rsztype, gfx_coord2_t *pos , sysarg_t pos_id)478 display_wnd_rsztype_t rsztype, gfx_coord2_t *pos) 531 479 { 532 480 ds_seat_t *seat; … … 539 487 return; 540 488 541 /* Determine which seat started the resize */542 seat = ds_display_seat_by_idev(wnd->display, pos_id);543 if (seat == NULL)544 return;545 546 489 wnd->orig_pos = *pos; 547 wnd->orig_pos_id = pos_id;548 490 wnd->state = dsw_resizing; 549 491 wnd->rsztype = rsztype; 550 492 wnd->preview_rect = wnd->rect; 551 493 494 // XXX Need client to tell us which seat started the resize! 495 seat = ds_display_first_seat(wnd->display); 552 496 ctype = display_cursor_from_wrsz(rsztype); 553 497 ds_seat_set_wm_cursor(seat, wnd->display->cursor[ctype]); … … 579 523 ds_client_post_resize_event(wnd->client, wnd, &nrect); 580 524 581 /* Determine which seat started the resize */ 582 seat = ds_display_seat_by_idev(wnd->display, wnd->orig_pos_id); 583 if (seat != NULL) 584 ds_seat_set_wm_cursor(seat, NULL); 585 586 wnd->orig_pos_id = 0; 525 // XXX Need to know which seat started the resize! 526 seat = ds_display_first_seat(wnd->display); 527 ds_seat_set_wm_cursor(seat, NULL); 587 528 588 529 (void) ds_display_paint(wnd->display, NULL); … … 600 541 gfx_rect_t old_rect; 601 542 602 log_msg(LOG_DEFAULT, LVL_DEBUG 2, "ds_window_update_resize (%d, %d)",543 log_msg(LOG_DEFAULT, LVL_DEBUG, "ds_window_update_resize (%d, %d)", 603 544 (int) pos->x, (int) pos->y); 604 545 … … 639 580 * @param wnd Window 640 581 * @param event Position event 641 *642 * @return EOK on success or an error code643 582 */ 644 583 errno_t ds_window_post_pos_event(ds_window_t *wnd, pos_event_t *event) … … 646 585 pos_event_t tevent; 647 586 gfx_coord2_t pos; 648 sysarg_t pos_id;649 587 gfx_rect_t drect; 650 588 bool inside; 651 589 652 log_msg(LOG_DEFAULT, LVL_DEBUG 2,590 log_msg(LOG_DEFAULT, LVL_DEBUG, 653 591 "ds_window_post_pos_event type=%d pos=%d,%d", event->type, 654 592 (int) event->hpos, (int) event->vpos); … … 656 594 pos.x = event->hpos; 657 595 pos.y = event->vpos; 658 pos_id = event->pos_id;659 596 gfx_rect_translate(&wnd->dpos, &wnd->rect, &drect); 660 597 inside = gfx_pix_inside_rect(&pos, &drect); 661 598 662 if (event->type == POS_PRESS && event->btn_num == 2 && inside && 663 (wnd->flags & wndf_maximized) == 0) { 664 ds_window_start_move(wnd, &pos, pos_id); 599 if (event->type == POS_PRESS && event->btn_num == 2 && inside) { 600 ds_window_start_move(wnd, &pos); 665 601 return EOK; 666 602 } 667 603 668 604 if (event->type == POS_RELEASE) { 669 /* Finish move/resize if they were started by the same seat */ 670 if (wnd->state == dsw_moving && 671 ds_window_orig_seat(wnd, pos_id)) { 605 if (wnd->state == dsw_moving) { 672 606 ds_window_finish_move(wnd, &pos); 673 607 return EOK; 674 608 } 675 609 676 if (wnd->state == dsw_resizing && 677 ds_window_orig_seat(wnd, pos_id)) { 610 if (wnd->state == dsw_resizing) { 678 611 ds_window_finish_resize(wnd, &pos); 679 612 return EOK; … … 682 615 683 616 if (event->type == POS_UPDATE) { 684 /* Update move/resize if they were started by the same seat */ 685 if (wnd->state == dsw_moving && 686 ds_window_orig_seat(wnd, pos_id)) { 617 if (wnd->state == dsw_moving) { 687 618 ds_window_update_move(wnd, &pos); 688 619 return EOK; 689 620 } 690 621 691 if (wnd->state == dsw_resizing && 692 ds_window_orig_seat(wnd, pos_id)) { 622 if (wnd->state == dsw_resizing) { 693 623 ds_window_update_resize(wnd, &pos); 694 624 return EOK; … … 707 637 * 708 638 * @param wnd Window 709 * @return EOK on success or an error code710 639 */ 711 640 errno_t ds_window_post_focus_event(ds_window_t *wnd) 712 641 { 713 display_wnd_focus_ev_t efocus;714 errno_t rc;715 ds_wmclient_t *wmclient;716 717 642 log_msg(LOG_DEFAULT, LVL_DEBUG, "ds_window_post_focus_event"); 718 643 719 /* Increase focus counter */ 720 ++wnd->nfocus; 721 efocus.nfocus = wnd->nfocus; 722 723 rc = ds_client_post_focus_event(wnd->client, wnd, &efocus); 724 if (rc != EOK) 725 return rc; 726 727 /* Notify window managers about window information change */ 728 wmclient = ds_display_first_wmclient(wnd->display); 729 while (wmclient != NULL) { 730 ds_wmclient_post_wnd_changed_event(wmclient, wnd->id); 731 wmclient = ds_display_next_wmclient(wmclient); 732 } 733 734 return EOK; 644 return ds_client_post_focus_event(wnd->client, wnd); 735 645 } 736 646 … … 738 648 * 739 649 * @param wnd Window 740 * @return EOK on success or an error code741 650 */ 742 651 errno_t ds_window_post_unfocus_event(ds_window_t *wnd) 743 652 { 744 display_wnd_unfocus_ev_t eunfocus;745 errno_t rc;746 ds_wmclient_t *wmclient;747 748 653 log_msg(LOG_DEFAULT, LVL_DEBUG, "ds_window_post_unfocus_event"); 749 654 750 /* Decrease focus counter */ 751 --wnd->nfocus; 752 eunfocus.nfocus = wnd->nfocus; 753 754 rc = ds_client_post_unfocus_event(wnd->client, wnd, &eunfocus); 755 if (rc != EOK) 756 return rc; 757 758 /* Notify window managers about window information change */ 759 wmclient = ds_display_first_wmclient(wnd->display); 760 while (wmclient != NULL) { 761 ds_wmclient_post_wnd_changed_event(wmclient, wnd->id); 762 wmclient = ds_display_next_wmclient(wmclient); 763 } 764 765 return EOK; 655 return ds_client_post_unfocus_event(wnd->client, wnd); 766 656 } 767 657 … … 771 661 * @param pos Position where the pointer was when the move started 772 662 * relative to the window 773 * @param pos_id Positioning device ID774 663 * @param event Button press event 775 664 */ 776 void ds_window_move_req(ds_window_t *wnd, gfx_coord2_t *pos , sysarg_t pos_id)665 void ds_window_move_req(ds_window_t *wnd, gfx_coord2_t *pos) 777 666 { 778 667 gfx_coord2_t orig_pos; … … 782 671 783 672 gfx_coord2_add(&wnd->dpos, pos, &orig_pos); 784 ds_window_start_move(wnd, &orig_pos , pos_id);673 ds_window_start_move(wnd, &orig_pos); 785 674 } 786 675 … … 802 691 { 803 692 *dpos = wnd->dpos; 804 }805 806 /** Get maximized window rectangle.807 *808 * @param wnd Window809 */810 void ds_window_get_max_rect(ds_window_t *wnd, gfx_rect_t *rect)811 {812 *rect = wnd->display->max_rect;813 693 } 814 694 … … 819 699 * @param pos Position where the pointer was when the resize started 820 700 * relative to the window 821 * @param pos_id Positioning device ID822 701 * @param event Button press event 823 702 */ 824 703 void ds_window_resize_req(ds_window_t *wnd, display_wnd_rsztype_t rsztype, 825 gfx_coord2_t *pos , sysarg_t pos_id)704 gfx_coord2_t *pos) 826 705 { 827 706 gfx_coord2_t orig_pos; 828 707 829 log_msg(LOG_DEFAULT, LVL_DEBUG, "ds_window_resize_req (%d, %d, %d , %d)",830 (int) rsztype, (int)pos->x, (int)pos->y, (int)pos_id);708 log_msg(LOG_DEFAULT, LVL_DEBUG, "ds_window_resize_req (%d, %d, %d)", 709 (int) rsztype, (int) pos->x, (int) pos->y); 831 710 832 711 gfx_coord2_add(&wnd->dpos, pos, &orig_pos); 833 ds_window_start_resize(wnd, rsztype, &orig_pos , pos_id);712 ds_window_start_resize(wnd, rsztype, &orig_pos); 834 713 } 835 714 … … 837 716 * 838 717 * @param wnd Window 839 * @return EOK on success or an error code840 718 */ 841 719 errno_t ds_window_resize(ds_window_t *wnd, gfx_coord2_t *offs, … … 888 766 wnd->rect = *nrect; 889 767 890 if ((wnd->flags & wndf_avoid) != 0)891 ds_display_update_max_rect(wnd->display);892 893 768 (void) ds_display_paint(wnd->display, NULL); 894 return EOK;895 }896 897 /** Minimize window.898 *899 * @param wnd Window900 * @return EOK on success or an error code901 */902 errno_t ds_window_minimize(ds_window_t *wnd)903 {904 /* If already minimized, do nothing and return success. */905 if ((wnd->flags & wndf_minimized) != 0)906 return EOK;907 908 ds_window_unfocus(wnd);909 910 wnd->flags |= wndf_minimized;911 (void) ds_display_paint(wnd->display, NULL);912 return EOK;913 }914 915 /** Unminimize window.916 *917 * @param wnd Window918 * @return EOK on success or an error code919 */920 errno_t ds_window_unminimize(ds_window_t *wnd)921 {922 /* If not minimized, do nothing and return success. */923 if ((wnd->flags & wndf_minimized) == 0)924 return EOK;925 926 wnd->flags &= ~wndf_minimized;927 (void) ds_display_paint(wnd->display, NULL);928 return EOK;929 }930 931 /** Maximize window.932 *933 * @param wnd Window934 * @return EOK on success or an error code935 */936 errno_t ds_window_maximize(ds_window_t *wnd)937 {938 gfx_coord2_t old_dpos;939 gfx_rect_t old_rect;940 gfx_coord2_t offs;941 gfx_rect_t max_rect;942 gfx_rect_t nrect;943 errno_t rc;944 945 /* If already maximized, do nothing and return success. */946 if ((wnd->flags & wndf_maximized) != 0)947 return EOK;948 949 /* Remember the old window rectangle and display position */950 old_rect = wnd->rect;951 old_dpos = wnd->dpos;952 953 ds_window_get_max_rect(wnd, &max_rect);954 955 /* Keep window contents on the same position on the screen */956 offs.x = max_rect.p0.x - wnd->dpos.x;957 offs.y = max_rect.p0.y - wnd->dpos.y;958 959 /* Maximized window's coordinates will start at 0,0 */960 gfx_rect_rtranslate(&max_rect.p0, &max_rect, &nrect);961 962 rc = ds_window_resize(wnd, &offs, &nrect);963 if (rc != EOK)964 return rc;965 966 /* Set window flags, remember normal rectangle */967 wnd->flags |= wndf_maximized;968 wnd->normal_rect = old_rect;969 wnd->normal_dpos = old_dpos;970 971 return EOK;972 }973 974 /** Unmaximize window.975 *976 * @param wnd Window977 * @return EOK on success or an error code978 */979 errno_t ds_window_unmaximize(ds_window_t *wnd)980 {981 gfx_coord2_t offs;982 errno_t rc;983 984 /* If not maximized, do nothing and return success. */985 if ((wnd->flags & wndf_maximized) == 0)986 return EOK;987 988 /* Keep window contents on the same position on the screen */989 offs.x = wnd->normal_dpos.x - wnd->dpos.x;990 offs.y = wnd->normal_dpos.y - wnd->dpos.y;991 992 rc = ds_window_resize(wnd, &offs, &wnd->normal_rect);993 if (rc != EOK)994 return rc;995 996 /* Clear maximized flag */997 wnd->flags &= ~wndf_maximized;998 999 769 return EOK; 1000 770 } … … 1042 812 * 1043 813 * @param wnd Window 1044 * @param cursor New cursor1045 814 * @return EOK on success, EINVAL if @a cursor is invalid 1046 815 */ … … 1056 825 } 1057 826 1058 /** Set window caption.1059 *1060 * @param wnd Window1061 * @param caption New caption1062 *1063 * @return EOK on success, EINVAL if @a cursor is invalid1064 */1065 errno_t ds_window_set_caption(ds_window_t *wnd, const char *caption)1066 {1067 char *dcaption;1068 ds_wmclient_t *wmclient;1069 1070 dcaption = str_dup(caption);1071 if (dcaption == NULL)1072 return ENOMEM;1073 1074 free(wnd->caption);1075 wnd->caption = dcaption;1076 1077 /* Notify window managers about window information change */1078 wmclient = ds_display_first_wmclient(wnd->display);1079 while (wmclient != NULL) {1080 ds_wmclient_post_wnd_changed_event(wmclient, wnd->id);1081 wmclient = ds_display_next_wmclient(wmclient);1082 }1083 1084 return EOK;1085 }1086 1087 /** Find alternate window with the allowed flags.1088 *1089 * An alternate window is a *different* window that is preferably previous1090 * in the display order and only has the @a allowed flags.1091 *1092 * @param wnd Window1093 * @param allowed_flags Bitmask of flags that the window is allowed to have1094 *1095 * @return Alternate window matching the criteria or @c NULL if there is none1096 */1097 ds_window_t *ds_window_find_prev(ds_window_t *wnd,1098 display_wnd_flags_t allowed_flags)1099 {1100 ds_window_t *nwnd;1101 1102 /* Try preceding windows in display order */1103 nwnd = ds_display_next_window(wnd);1104 while (nwnd != NULL && (nwnd->flags & ~allowed_flags) != 0) {1105 nwnd = ds_display_next_window(nwnd);1106 }1107 1108 /* Do we already have a matching window? */1109 if (nwnd != NULL && (nwnd->flags & ~allowed_flags) == 0) {1110 return nwnd;1111 }1112 1113 /* Try succeeding windows in display order */1114 nwnd = ds_display_first_window(wnd->display);1115 while (nwnd != NULL && nwnd != wnd &&1116 (nwnd->flags & ~allowed_flags) != 0) {1117 nwnd = ds_display_next_window(nwnd);1118 }1119 1120 if (nwnd == wnd)1121 return NULL;1122 1123 return nwnd;1124 }1125 1126 /** Find alternate window with the allowed flags.1127 *1128 * An alternate window is a *different* window that is preferably previous1129 * in the display order and only has the @a allowed flags.1130 *1131 * @param wnd Window1132 * @param allowed_flags Bitmask of flags that the window is allowed to have1133 *1134 * @return Alternate window matching the criteria or @c NULL if there is none1135 */1136 ds_window_t *ds_window_find_next(ds_window_t *wnd,1137 display_wnd_flags_t allowed_flags)1138 {1139 ds_window_t *nwnd;1140 1141 /* Try preceding windows in display order */1142 nwnd = ds_display_prev_window(wnd);1143 while (nwnd != NULL && (nwnd->flags & ~allowed_flags) != 0) {1144 nwnd = ds_display_prev_window(nwnd);1145 }1146 1147 /* Do we already have a matching window? */1148 if (nwnd != NULL && (nwnd->flags & ~allowed_flags) == 0) {1149 return nwnd;1150 }1151 1152 /* Try succeeding windows in display order */1153 nwnd = ds_display_last_window(wnd->display);1154 while (nwnd != NULL && nwnd != wnd &&1155 (nwnd->flags & ~allowed_flags) != 0) {1156 nwnd = ds_display_prev_window(nwnd);1157 }1158 1159 if (nwnd == wnd)1160 return NULL;1161 1162 return nwnd;1163 }1164 1165 /** Remove focus from window.1166 *1167 * Used to switch focus to another window when closing or minimizing window.1168 *1169 * @param wnd Window1170 */1171 void ds_window_unfocus(ds_window_t *wnd)1172 {1173 ds_seat_t *seat;1174 1175 /* Make sure window is no longer focused in any seat */1176 seat = ds_display_first_seat(wnd->display);1177 while (seat != NULL) {1178 ds_seat_unfocus_wnd(seat, wnd);1179 seat = ds_display_next_seat(seat);1180 }1181 }1182 1183 /** Determine if input device belongs to the same seat as the original device.1184 *1185 * Compare the seat ownning @a idev_id with the seat owning @a wnd->orig_pos_id1186 * (the device that started the window move or resize).1187 *1188 * This is used to make sure that, when two seats focus the same window,1189 * only devices owned by the seat that started the resize or move can1190 * affect it. Otherwise moving the other pointer(s) would disrupt the1191 * resize or move operation.1192 *1193 * @param wnd Window (that is currently being resized or moved)1194 * @param idev_id Input device ID1195 * @return @c true iff idev_id is owned by the same seat as the input1196 * device that started the resize or move1197 */1198 bool ds_window_orig_seat(ds_window_t *wnd, sysarg_t idev_id)1199 {1200 ds_seat_t *orig_seat;1201 ds_seat_t *seat;1202 1203 /* Window must be in state such that wnd->orig_pos_id is valid */1204 assert(wnd->state == dsw_moving || wnd->state == dsw_resizing);1205 1206 orig_seat = ds_display_seat_by_idev(wnd->display, wnd->orig_pos_id);1207 seat = ds_display_seat_by_idev(wnd->display, idev_id);1208 1209 return seat == orig_seat;1210 }1211 1212 827 /** Window memory GC invalidate callback. 1213 828 *
Note:
See TracChangeset
for help on using the changeset viewer.