1*4882a593SmuzhiyunFrom b342bc6139ea8a765084eacfc0ca365561fcdd30 Mon Sep 17 00:00:00 2001 2*4882a593SmuzhiyunFrom: Jeffy Chen <jeffy.chen@rock-chips.com> 3*4882a593SmuzhiyunDate: Wed, 27 Apr 2022 18:05:28 +0800 4*4882a593SmuzhiyunSubject: [PATCH 13/93] Support setting output flow direction 5*4882a593Smuzhiyun 6*4882a593SmuzhiyunSet env "WESTON_OUTPUT_FLOW" to change output flow direction: 7*4882a593Smuzhiyunhorizontal: 8*4882a593Smuzhiyun Place outputs horizontal. 9*4882a593Smuzhiyunvertical: 10*4882a593Smuzhiyun Place outputs vertical. 11*4882a593Smuzhiyunsame-as: 12*4882a593Smuzhiyun Place outputs at (0,0). 13*4882a593Smuzhiyun 14*4882a593SmuzhiyunSigned-off-by: Jeffy Chen <jeffy.chen@rock-chips.com> 15*4882a593Smuzhiyun--- 16*4882a593Smuzhiyun clients/desktop-shell.c | 20 +++++++++++++++ 17*4882a593Smuzhiyun compositor/main.c | 11 ++++++++ 18*4882a593Smuzhiyun desktop-shell/shell.c | 17 ++++++++++--- 19*4882a593Smuzhiyun include/libweston/libweston.h | 11 ++++++++ 20*4882a593Smuzhiyun libweston/compositor.c | 48 +++++++++++++++-------------------- 21*4882a593Smuzhiyun 5 files changed, 77 insertions(+), 30 deletions(-) 22*4882a593Smuzhiyun 23*4882a593Smuzhiyundiff --git a/clients/desktop-shell.c b/clients/desktop-shell.c 24*4882a593Smuzhiyunindex b450a01..6eb9775 100644 25*4882a593Smuzhiyun--- a/clients/desktop-shell.c 26*4882a593Smuzhiyun+++ b/clients/desktop-shell.c 27*4882a593Smuzhiyun@@ -135,6 +135,8 @@ struct output { 28*4882a593Smuzhiyun int y; 29*4882a593Smuzhiyun struct panel *panel; 30*4882a593Smuzhiyun struct background *background; 31*4882a593Smuzhiyun+ 32*4882a593Smuzhiyun+ struct desktop *desktop; 33*4882a593Smuzhiyun }; 34*4882a593Smuzhiyun 35*4882a593Smuzhiyun struct panel_launcher { 36*4882a593Smuzhiyun@@ -1227,10 +1229,27 @@ output_handle_geometry(void *data, 37*4882a593Smuzhiyun int transform) 38*4882a593Smuzhiyun { 39*4882a593Smuzhiyun struct output *output = data; 40*4882a593Smuzhiyun+ struct desktop *desktop = output->desktop; 41*4882a593Smuzhiyun+ struct wl_surface *surface; 42*4882a593Smuzhiyun 43*4882a593Smuzhiyun output->x = x; 44*4882a593Smuzhiyun output->y = y; 45*4882a593Smuzhiyun 46*4882a593Smuzhiyun+ if (y && output->panel) { 47*4882a593Smuzhiyun+ /* HACK: Re-set the panel to destroy it */ 48*4882a593Smuzhiyun+ surface = window_get_wl_surface(output->panel->window); 49*4882a593Smuzhiyun+ weston_desktop_shell_set_panel(desktop->shell, 50*4882a593Smuzhiyun+ output->output, surface); 51*4882a593Smuzhiyun+ } 52*4882a593Smuzhiyun+ 53*4882a593Smuzhiyun+ if (!y && desktop->want_panel && !output->panel) { 54*4882a593Smuzhiyun+ /* based on output_init() */ 55*4882a593Smuzhiyun+ output->panel = panel_create(desktop, output); 56*4882a593Smuzhiyun+ surface = window_get_wl_surface(output->panel->window); 57*4882a593Smuzhiyun+ weston_desktop_shell_set_panel(desktop->shell, 58*4882a593Smuzhiyun+ output->output, surface); 59*4882a593Smuzhiyun+ } 60*4882a593Smuzhiyun+ 61*4882a593Smuzhiyun if (output->panel) 62*4882a593Smuzhiyun window_set_buffer_transform(output->panel->window, transform); 63*4882a593Smuzhiyun if (output->background) 64*4882a593Smuzhiyun@@ -1300,6 +1319,7 @@ create_output(struct desktop *desktop, uint32_t id) 65*4882a593Smuzhiyun if (!output) 66*4882a593Smuzhiyun return; 67*4882a593Smuzhiyun 68*4882a593Smuzhiyun+ output->desktop = desktop; 69*4882a593Smuzhiyun output->output = 70*4882a593Smuzhiyun display_bind(desktop->display, id, &wl_output_interface, 2); 71*4882a593Smuzhiyun output->server_output_id = id; 72*4882a593Smuzhiyundiff --git a/compositor/main.c b/compositor/main.c 73*4882a593Smuzhiyunindex 631355e..57a52b6 100644 74*4882a593Smuzhiyun--- a/compositor/main.c 75*4882a593Smuzhiyun+++ b/compositor/main.c 76*4882a593Smuzhiyun@@ -3546,6 +3546,7 @@ wet_main(int argc, char *argv[], const struct weston_testsuite_data *test_data) 77*4882a593Smuzhiyun struct weston_log_subscriber *flight_rec = NULL; 78*4882a593Smuzhiyun sigset_t mask; 79*4882a593Smuzhiyun struct sigaction action; 80*4882a593Smuzhiyun+ char *buf; 81*4882a593Smuzhiyun 82*4882a593Smuzhiyun bool wait_for_debugger = false; 83*4882a593Smuzhiyun struct wl_protocol_logger *protologger = NULL; 84*4882a593Smuzhiyun@@ -3699,6 +3700,16 @@ wet_main(int argc, char *argv[], const struct weston_testsuite_data *test_data) 85*4882a593Smuzhiyun goto out; 86*4882a593Smuzhiyun } 87*4882a593Smuzhiyun 88*4882a593Smuzhiyun+ wet.compositor->output_flow = WESTON_OUTPUT_FLOW_HORIZONTAL; 89*4882a593Smuzhiyun+ 90*4882a593Smuzhiyun+ buf = getenv("WESTON_OUTPUT_FLOW"); 91*4882a593Smuzhiyun+ if (buf) { 92*4882a593Smuzhiyun+ if (!strcmp(buf, "vertical")) 93*4882a593Smuzhiyun+ wet.compositor->output_flow = WESTON_OUTPUT_FLOW_VERTICAL; 94*4882a593Smuzhiyun+ else if (!strcmp(buf, "same-as")) 95*4882a593Smuzhiyun+ wet.compositor->output_flow = WESTON_OUTPUT_FLOW_SAME_AS; 96*4882a593Smuzhiyun+ } 97*4882a593Smuzhiyun+ 98*4882a593Smuzhiyun protocol_scope = 99*4882a593Smuzhiyun weston_log_ctx_add_log_scope(log_ctx, "proto", 100*4882a593Smuzhiyun "Wayland protocol dump for all clients.\n", 101*4882a593Smuzhiyundiff --git a/desktop-shell/shell.c b/desktop-shell/shell.c 102*4882a593Smuzhiyunindex 7363ee3..79e20ad 100644 103*4882a593Smuzhiyun--- a/desktop-shell/shell.c 104*4882a593Smuzhiyun+++ b/desktop-shell/shell.c 105*4882a593Smuzhiyun@@ -1020,7 +1020,7 @@ constrain_position(struct weston_move_grab *move, int *cx, int *cy) 106*4882a593Smuzhiyun y = wl_fixed_to_int(pointer->y + move->dy); 107*4882a593Smuzhiyun 108*4882a593Smuzhiyun if (shsurf->shell->panel_position == 109*4882a593Smuzhiyun- WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP) { 110*4882a593Smuzhiyun+ WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP && !surface->output->y) { 111*4882a593Smuzhiyun get_output_work_area(shsurf->shell, surface->output, &area); 112*4882a593Smuzhiyun geometry = 113*4882a593Smuzhiyun weston_desktop_surface_get_geometry(shsurf->desktop_surface); 114*4882a593Smuzhiyun@@ -2658,6 +2658,18 @@ desktop_shell_set_panel(struct wl_client *client, 115*4882a593Smuzhiyun wl_resource_get_user_data(surface_resource); 116*4882a593Smuzhiyun struct weston_view *view, *next; 117*4882a593Smuzhiyun struct shell_output *sh_output; 118*4882a593Smuzhiyun+ struct weston_output *output; 119*4882a593Smuzhiyun+ 120*4882a593Smuzhiyun+ output = weston_head_from_resource(output_resource)->output; 121*4882a593Smuzhiyun+ sh_output = find_shell_output_from_weston_output(shell, output); 122*4882a593Smuzhiyun+ 123*4882a593Smuzhiyun+ if (surface == sh_output->panel_surface) { 124*4882a593Smuzhiyun+ /* HACK: Re-set to destroy output panel */ 125*4882a593Smuzhiyun+ weston_desktop_shell_send_configure(resource, 0, 126*4882a593Smuzhiyun+ surface_resource, 127*4882a593Smuzhiyun+ 0, 0); 128*4882a593Smuzhiyun+ return; 129*4882a593Smuzhiyun+ } 130*4882a593Smuzhiyun 131*4882a593Smuzhiyun if (surface->committed) { 132*4882a593Smuzhiyun wl_resource_post_error(surface_resource, 133*4882a593Smuzhiyun@@ -2673,10 +2685,9 @@ desktop_shell_set_panel(struct wl_client *client, 134*4882a593Smuzhiyun surface->committed = panel_committed; 135*4882a593Smuzhiyun surface->committed_private = shell; 136*4882a593Smuzhiyun weston_surface_set_label_func(surface, panel_get_label); 137*4882a593Smuzhiyun- surface->output = weston_head_from_resource(output_resource)->output; 138*4882a593Smuzhiyun+ surface->output = output; 139*4882a593Smuzhiyun weston_view_set_output(view, surface->output); 140*4882a593Smuzhiyun 141*4882a593Smuzhiyun- sh_output = find_shell_output_from_weston_output(shell, surface->output); 142*4882a593Smuzhiyun if (sh_output->panel_surface) { 143*4882a593Smuzhiyun /* The output already has a panel, tell our helper 144*4882a593Smuzhiyun * there is no need for another one. */ 145*4882a593Smuzhiyundiff --git a/include/libweston/libweston.h b/include/libweston/libweston.h 146*4882a593Smuzhiyunindex a3a23bc..103c293 100644 147*4882a593Smuzhiyun--- a/include/libweston/libweston.h 148*4882a593Smuzhiyun+++ b/include/libweston/libweston.h 149*4882a593Smuzhiyun@@ -1184,6 +1184,12 @@ struct weston_color_manager; 150*4882a593Smuzhiyun struct weston_dmabuf_feedback; 151*4882a593Smuzhiyun struct weston_dmabuf_feedback_format_table; 152*4882a593Smuzhiyun 153*4882a593Smuzhiyun+enum weston_output_flow { 154*4882a593Smuzhiyun+ WESTON_OUTPUT_FLOW_HORIZONTAL, 155*4882a593Smuzhiyun+ WESTON_OUTPUT_FLOW_VERTICAL, 156*4882a593Smuzhiyun+ WESTON_OUTPUT_FLOW_SAME_AS, 157*4882a593Smuzhiyun+}; 158*4882a593Smuzhiyun+ 159*4882a593Smuzhiyun /** Main object, container-like structure which aggregates all other objects. 160*4882a593Smuzhiyun * 161*4882a593Smuzhiyun * \ingroup compositor 162*4882a593Smuzhiyun@@ -1317,6 +1323,8 @@ struct weston_compositor { 163*4882a593Smuzhiyun /* One-time warning about a view appearing in the layer list when it 164*4882a593Smuzhiyun * or its surface are not mapped. */ 165*4882a593Smuzhiyun bool warned_about_unmapped_surface_or_view; 166*4882a593Smuzhiyun+ 167*4882a593Smuzhiyun+ enum weston_output_flow output_flow; 168*4882a593Smuzhiyun }; 169*4882a593Smuzhiyun 170*4882a593Smuzhiyun struct weston_solid_buffer_values { 171*4882a593Smuzhiyun@@ -2347,6 +2355,9 @@ struct weston_color_profile * 172*4882a593Smuzhiyun weston_compositor_load_icc_file(struct weston_compositor *compositor, 173*4882a593Smuzhiyun const char *path); 174*4882a593Smuzhiyun 175*4882a593Smuzhiyun+void 176*4882a593Smuzhiyun+weston_compositor_reflow_outputs(struct weston_compositor *compositor); 177*4882a593Smuzhiyun+ 178*4882a593Smuzhiyun #ifdef __cplusplus 179*4882a593Smuzhiyun } 180*4882a593Smuzhiyun #endif 181*4882a593Smuzhiyundiff --git a/libweston/compositor.c b/libweston/compositor.c 182*4882a593Smuzhiyunindex 55751a7..23316ef 100644 183*4882a593Smuzhiyun--- a/libweston/compositor.c 184*4882a593Smuzhiyun+++ b/libweston/compositor.c 185*4882a593Smuzhiyun@@ -258,10 +258,6 @@ weston_mode_switch_finish(struct weston_output *output, 186*4882a593Smuzhiyun mode_changed, scale_changed); 187*4882a593Smuzhiyun } 188*4882a593Smuzhiyun 189*4882a593Smuzhiyun-static void 190*4882a593Smuzhiyun-weston_compositor_reflow_outputs(struct weston_compositor *compositor, 191*4882a593Smuzhiyun- struct weston_output *resized_output, int delta_width); 192*4882a593Smuzhiyun- 193*4882a593Smuzhiyun /** 194*4882a593Smuzhiyun * \ingroup output 195*4882a593Smuzhiyun */ 196*4882a593Smuzhiyun@@ -272,7 +268,6 @@ weston_output_mode_set_native(struct weston_output *output, 197*4882a593Smuzhiyun { 198*4882a593Smuzhiyun int ret; 199*4882a593Smuzhiyun int mode_changed = 0, scale_changed = 0; 200*4882a593Smuzhiyun- int32_t old_width; 201*4882a593Smuzhiyun 202*4882a593Smuzhiyun if (!output->switch_mode) 203*4882a593Smuzhiyun return -1; 204*4882a593Smuzhiyun@@ -288,14 +283,13 @@ weston_output_mode_set_native(struct weston_output *output, 205*4882a593Smuzhiyun } 206*4882a593Smuzhiyun } 207*4882a593Smuzhiyun 208*4882a593Smuzhiyun- old_width = output->width; 209*4882a593Smuzhiyun output->native_mode = mode; 210*4882a593Smuzhiyun output->native_scale = scale; 211*4882a593Smuzhiyun 212*4882a593Smuzhiyun weston_mode_switch_finish(output, mode_changed, scale_changed); 213*4882a593Smuzhiyun 214*4882a593Smuzhiyun if (mode_changed || scale_changed) { 215*4882a593Smuzhiyun- weston_compositor_reflow_outputs(output->compositor, output, output->width - old_width); 216*4882a593Smuzhiyun+ weston_compositor_reflow_outputs(output->compositor); 217*4882a593Smuzhiyun 218*4882a593Smuzhiyun wl_signal_emit(&output->compositor->output_resized_signal, output); 219*4882a593Smuzhiyun } 220*4882a593Smuzhiyun@@ -6468,30 +6462,29 @@ weston_head_get_destroy_listener(struct weston_head *head, 221*4882a593Smuzhiyun static void 222*4882a593Smuzhiyun weston_output_set_position(struct weston_output *output, int x, int y); 223*4882a593Smuzhiyun 224*4882a593Smuzhiyun-/* Move other outputs when one is resized so the space remains contiguous. */ 225*4882a593Smuzhiyun-static void 226*4882a593Smuzhiyun-weston_compositor_reflow_outputs(struct weston_compositor *compositor, 227*4882a593Smuzhiyun- struct weston_output *resized_output, int delta_width) 228*4882a593Smuzhiyun+WL_EXPORT void 229*4882a593Smuzhiyun+weston_compositor_reflow_outputs(struct weston_compositor *compositor) 230*4882a593Smuzhiyun { 231*4882a593Smuzhiyun struct weston_output *output; 232*4882a593Smuzhiyun- bool start_resizing = false; 233*4882a593Smuzhiyun+ int x, y, next_x, next_y; 234*4882a593Smuzhiyun 235*4882a593Smuzhiyun if (compositor->output_flow_dirty) 236*4882a593Smuzhiyun return; 237*4882a593Smuzhiyun 238*4882a593Smuzhiyun- if (!delta_width) 239*4882a593Smuzhiyun- return; 240*4882a593Smuzhiyun- 241*4882a593Smuzhiyun+ next_x = next_y = 0; 242*4882a593Smuzhiyun wl_list_for_each(output, &compositor->output_list, link) { 243*4882a593Smuzhiyun- if (output == resized_output) { 244*4882a593Smuzhiyun- start_resizing = true; 245*4882a593Smuzhiyun+ if (output->destroying) 246*4882a593Smuzhiyun continue; 247*4882a593Smuzhiyun- } 248*4882a593Smuzhiyun 249*4882a593Smuzhiyun- if (start_resizing) { 250*4882a593Smuzhiyun- weston_output_set_position(output, output->x + delta_width, output->y); 251*4882a593Smuzhiyun- output->dirty = 1; 252*4882a593Smuzhiyun- } 253*4882a593Smuzhiyun+ x = next_x; 254*4882a593Smuzhiyun+ y = next_y; 255*4882a593Smuzhiyun+ 256*4882a593Smuzhiyun+ if (compositor->output_flow == WESTON_OUTPUT_FLOW_HORIZONTAL) 257*4882a593Smuzhiyun+ next_x += output->width; 258*4882a593Smuzhiyun+ else if (compositor->output_flow == WESTON_OUTPUT_FLOW_VERTICAL) 259*4882a593Smuzhiyun+ next_y += output->height; 260*4882a593Smuzhiyun+ 261*4882a593Smuzhiyun+ weston_output_set_position(output, x, y); 262*4882a593Smuzhiyun } 263*4882a593Smuzhiyun } 264*4882a593Smuzhiyun 265*4882a593Smuzhiyun@@ -6583,6 +6576,8 @@ weston_output_transform_scale_init(struct weston_output *output, uint32_t transf 266*4882a593Smuzhiyun static void 267*4882a593Smuzhiyun weston_output_init_geometry(struct weston_output *output, int x, int y) 268*4882a593Smuzhiyun { 269*4882a593Smuzhiyun+ output->dirty = 1; 270*4882a593Smuzhiyun+ 271*4882a593Smuzhiyun output->x = x; 272*4882a593Smuzhiyun output->y = y; 273*4882a593Smuzhiyun 274*4882a593Smuzhiyun@@ -6616,8 +6611,6 @@ weston_output_set_position(struct weston_output *output, int x, int y) 275*4882a593Smuzhiyun 276*4882a593Smuzhiyun weston_output_init_geometry(output, x, y); 277*4882a593Smuzhiyun 278*4882a593Smuzhiyun- output->dirty = 1; 279*4882a593Smuzhiyun- 280*4882a593Smuzhiyun /* Move views on this output. */ 281*4882a593Smuzhiyun wl_signal_emit(&output->compositor->output_moved_signal, output); 282*4882a593Smuzhiyun 283*4882a593Smuzhiyun@@ -6909,7 +6902,7 @@ weston_compositor_remove_output(struct weston_output *output) 284*4882a593Smuzhiyun 285*4882a593Smuzhiyun weston_presentation_feedback_discard_list(&output->feedback_list); 286*4882a593Smuzhiyun 287*4882a593Smuzhiyun- weston_compositor_reflow_outputs(compositor, output, -output->width); 288*4882a593Smuzhiyun+ weston_compositor_reflow_outputs(compositor); 289*4882a593Smuzhiyun 290*4882a593Smuzhiyun wl_list_remove(&output->link); 291*4882a593Smuzhiyun wl_list_insert(compositor->pending_output_list.prev, &output->link); 292*4882a593Smuzhiyun@@ -6983,7 +6976,7 @@ weston_output_set_transform(struct weston_output *output, 293*4882a593Smuzhiyun 294*4882a593Smuzhiyun weston_output_init_geometry(output, output->x, output->y); 295*4882a593Smuzhiyun 296*4882a593Smuzhiyun- output->dirty = 1; 297*4882a593Smuzhiyun+ weston_compositor_reflow_outputs(output->compositor); 298*4882a593Smuzhiyun 299*4882a593Smuzhiyun /* Notify clients of the change for output transform. */ 300*4882a593Smuzhiyun wl_list_for_each(head, &output->head_list, output_link) { 301*4882a593Smuzhiyun@@ -7346,7 +7339,6 @@ weston_output_enable(struct weston_output *output) 302*4882a593Smuzhiyun /* Make sure we have a transform set */ 303*4882a593Smuzhiyun assert(output->transform != UINT32_MAX); 304*4882a593Smuzhiyun 305*4882a593Smuzhiyun- output->dirty = 1; 306*4882a593Smuzhiyun output->original_scale = output->scale; 307*4882a593Smuzhiyun 308*4882a593Smuzhiyun wl_signal_init(&output->frame_signal); 309*4882a593Smuzhiyun@@ -7385,6 +7377,8 @@ weston_output_enable(struct weston_output *output) 310*4882a593Smuzhiyun output->name, head_names); 311*4882a593Smuzhiyun free(head_names); 312*4882a593Smuzhiyun 313*4882a593Smuzhiyun+ weston_compositor_reflow_outputs(output->compositor); 314*4882a593Smuzhiyun+ 315*4882a593Smuzhiyun return 0; 316*4882a593Smuzhiyun } 317*4882a593Smuzhiyun 318*4882a593Smuzhiyun-- 319*4882a593Smuzhiyun2.20.1 320*4882a593Smuzhiyun 321