mirror of
				https://gitlab.freedesktop.org/wayland/wayland.git
				synced 2025-11-03 09:01:42 -05:00 
			
		
		
		
	Let clients override the suggested size on resize.
This lets gears enforce a square aspect ration and minimum size of 300x300 and the terminal now properly resizes in steps of character cells.
This commit is contained in:
		
							parent
							
								
									1584c57edc
								
							
						
					
					
						commit
						221067654c
					
				
					 4 changed files with 104 additions and 40 deletions
				
			
		
							
								
								
									
										15
									
								
								gears.c
									
										
									
									
									
								
							
							
						
						
									
										15
									
								
								gears.c
									
										
									
									
									
								
							| 
						 | 
					@ -243,8 +243,19 @@ draw_gears(struct gears *gears)
 | 
				
			||||||
static void
 | 
					static void
 | 
				
			||||||
resize_window(struct gears *gears)
 | 
					resize_window(struct gears *gears)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	window_draw(gears->window);
 | 
						/* Constrain child size to be square and at least 300x300 */
 | 
				
			||||||
	window_get_child_rectangle(gears->window, &gears->rectangle);
 | 
						window_get_child_rectangle(gears->window, &gears->rectangle);
 | 
				
			||||||
 | 
						if (gears->rectangle.width > gears->rectangle.height)
 | 
				
			||||||
 | 
							gears->rectangle.height = gears->rectangle.width;
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							gears->rectangle.width = gears->rectangle.height;
 | 
				
			||||||
 | 
						if (gears->rectangle.width < 300) {
 | 
				
			||||||
 | 
							gears->rectangle.width = 300;
 | 
				
			||||||
 | 
							gears->rectangle.height = 300;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						window_set_child_size(gears->window, &gears->rectangle);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						window_draw(gears->window);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (gears->buffer != NULL)
 | 
						if (gears->buffer != NULL)
 | 
				
			||||||
		buffer_destroy(gears->buffer, gears->fd);
 | 
							buffer_destroy(gears->buffer, gears->fd);
 | 
				
			||||||
| 
						 | 
					@ -269,7 +280,7 @@ resize_window(struct gears *gears)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void
 | 
					static void
 | 
				
			||||||
resize_handler(struct window *window, struct rectangle *rectangle, void *data)
 | 
					resize_handler(struct window *window, void *data)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct gears *gears = data;
 | 
						struct gears *gears = data;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										108
									
								
								terminal.c
									
										
									
									
									
								
							
							
						
						
									
										108
									
								
								terminal.c
									
										
									
									
									
								
							| 
						 | 
					@ -66,6 +66,55 @@ struct terminal {
 | 
				
			||||||
	int margin;
 | 
						int margin;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void
 | 
				
			||||||
 | 
					terminal_resize(struct terminal *terminal, int width, int height)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						size_t size;
 | 
				
			||||||
 | 
						char *data;
 | 
				
			||||||
 | 
						int i, l, total_rows, row, tail;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (terminal->width == width && terminal->height == height)
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						size = (width + 1) * height;
 | 
				
			||||||
 | 
						data = malloc(size);
 | 
				
			||||||
 | 
						memset(data, 0, size);
 | 
				
			||||||
 | 
						if (terminal->data) {
 | 
				
			||||||
 | 
							if (width > terminal->width)
 | 
				
			||||||
 | 
								l = terminal->width;
 | 
				
			||||||
 | 
							else
 | 
				
			||||||
 | 
								l = width;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (terminal->total_rows > height) {
 | 
				
			||||||
 | 
								total_rows = height;
 | 
				
			||||||
 | 
								tail = terminal->tail + terminal->total_rows - height;
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								total_rows = terminal->total_rows;
 | 
				
			||||||
 | 
								tail = terminal->tail;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							for (i = 0; i < total_rows; i++) {
 | 
				
			||||||
 | 
								row = (tail + i) % terminal->height;
 | 
				
			||||||
 | 
								memcpy(data + (width + 1) * i,
 | 
				
			||||||
 | 
								       &terminal->data[row * (terminal->width + 1)], l);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							free(terminal->data);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							total_rows = 1;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						terminal->width = width;
 | 
				
			||||||
 | 
						terminal->height = height;
 | 
				
			||||||
 | 
						terminal->data = data;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						terminal->total_rows = total_rows;
 | 
				
			||||||
 | 
						terminal->row = total_rows - 1;
 | 
				
			||||||
 | 
						if (terminal->column >= terminal->width)
 | 
				
			||||||
 | 
							terminal->column = terminal->width - 1;
 | 
				
			||||||
 | 
						terminal->tail = 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void
 | 
					static void
 | 
				
			||||||
terminal_draw_contents(struct terminal *terminal)
 | 
					terminal_draw_contents(struct terminal *terminal)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
| 
						 | 
					@ -111,6 +160,33 @@ terminal_draw_contents(struct terminal *terminal)
 | 
				
			||||||
static void
 | 
					static void
 | 
				
			||||||
terminal_draw(struct terminal *terminal)
 | 
					terminal_draw(struct terminal *terminal)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						struct rectangle rectangle;
 | 
				
			||||||
 | 
						cairo_surface_t *surface;
 | 
				
			||||||
 | 
						cairo_font_extents_t extents;
 | 
				
			||||||
 | 
						cairo_t *cr;
 | 
				
			||||||
 | 
						int width, height;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						window_get_child_rectangle(terminal->window, &rectangle);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, 0, 0);
 | 
				
			||||||
 | 
						cr = cairo_create(surface);
 | 
				
			||||||
 | 
						cairo_select_font_face (cr, "mono",
 | 
				
			||||||
 | 
									CAIRO_FONT_SLANT_NORMAL,
 | 
				
			||||||
 | 
									CAIRO_FONT_WEIGHT_NORMAL);
 | 
				
			||||||
 | 
						cairo_set_font_size(cr, 14);
 | 
				
			||||||
 | 
						cairo_font_extents(cr, &extents);
 | 
				
			||||||
 | 
						cairo_destroy(cr);
 | 
				
			||||||
 | 
						cairo_surface_destroy(surface);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						width = (rectangle.width - 2 * terminal->margin) / (int32_t) extents.max_x_advance;
 | 
				
			||||||
 | 
						height = (rectangle.height - 2 * terminal->margin) / (int32_t) extents.height;
 | 
				
			||||||
 | 
						terminal_resize(terminal, width, height);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						rectangle.width = terminal->width * extents.max_x_advance + 2 * terminal->margin;
 | 
				
			||||||
 | 
						rectangle.height = terminal->height * extents.height + 2 * terminal->margin;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						window_set_child_size(terminal->window, &rectangle);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	window_draw(terminal->window);
 | 
						window_draw(terminal->window);
 | 
				
			||||||
	terminal_draw_contents(terminal);
 | 
						terminal_draw_contents(terminal);
 | 
				
			||||||
	wl_display_commit(terminal->display, 0);
 | 
						wl_display_commit(terminal->display, 0);
 | 
				
			||||||
| 
						 | 
					@ -234,29 +310,9 @@ terminal_data(struct terminal *terminal, const char *data, size_t length)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void
 | 
					static void
 | 
				
			||||||
resize_handler(struct window *window, struct rectangle *rectangle, void *data)
 | 
					resize_handler(struct window *window, void *data)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct terminal *terminal = data;
 | 
						struct terminal *terminal = data;
 | 
				
			||||||
	cairo_surface_t *surface;
 | 
					 | 
				
			||||||
	cairo_font_extents_t extents;
 | 
					 | 
				
			||||||
	cairo_t *cr;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	/* Adjust the size to an integer number of character cells.
 | 
					 | 
				
			||||||
	 * Maybe this is better done in the redraw path, as we're
 | 
					 | 
				
			||||||
	 * creating the cr and setting the font there anyway. */
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, 0, 0);
 | 
					 | 
				
			||||||
	cr = cairo_create(surface);
 | 
					 | 
				
			||||||
	cairo_select_font_face (cr, "mono",
 | 
					 | 
				
			||||||
				CAIRO_FONT_SLANT_NORMAL,
 | 
					 | 
				
			||||||
				CAIRO_FONT_WEIGHT_NORMAL);
 | 
					 | 
				
			||||||
	cairo_set_font_size(cr, 14);
 | 
					 | 
				
			||||||
	cairo_font_extents(cr, &extents);
 | 
					 | 
				
			||||||
	cairo_destroy(cr);
 | 
					 | 
				
			||||||
	cairo_surface_destroy(surface);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	rectangle->width -= (rectangle->width - 2 * terminal->margin) % (int32_t) extents.max_x_advance;
 | 
					 | 
				
			||||||
	rectangle->height -= (rectangle->height - 2 * terminal->margin) % (int32_t) extents.height;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	terminal_schedule_redraw(terminal);
 | 
						terminal_schedule_redraw(terminal);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -390,7 +446,6 @@ static struct terminal *
 | 
				
			||||||
terminal_create(struct wl_display *display, int fd)
 | 
					terminal_create(struct wl_display *display, int fd)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct terminal *terminal;
 | 
						struct terminal *terminal;
 | 
				
			||||||
	int size;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	terminal = malloc(sizeof *terminal);
 | 
						terminal = malloc(sizeof *terminal);
 | 
				
			||||||
	if (terminal == NULL)
 | 
						if (terminal == NULL)
 | 
				
			||||||
| 
						 | 
					@ -402,18 +457,14 @@ terminal_create(struct wl_display *display, int fd)
 | 
				
			||||||
					 500, 100, 500, 400);
 | 
										 500, 100, 500, 400);
 | 
				
			||||||
	terminal->display = display;
 | 
						terminal->display = display;
 | 
				
			||||||
	terminal->redraw_scheduled = 1;
 | 
						terminal->redraw_scheduled = 1;
 | 
				
			||||||
	terminal->width = 80;
 | 
					 | 
				
			||||||
	terminal->height = 25;
 | 
					 | 
				
			||||||
	terminal->total_rows = 1;
 | 
					 | 
				
			||||||
	terminal->margin = 5;
 | 
						terminal->margin = 5;
 | 
				
			||||||
	size = (terminal->width + 1) * terminal->height;
 | 
					 | 
				
			||||||
	terminal->data = malloc(size);
 | 
					 | 
				
			||||||
	memset(terminal->data, 0, size);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	window_set_resize_handler(terminal->window, resize_handler, terminal);
 | 
						window_set_resize_handler(terminal->window, resize_handler, terminal);
 | 
				
			||||||
	window_set_acknowledge_handler(terminal->window, acknowledge_handler, terminal);
 | 
						window_set_acknowledge_handler(terminal->window, acknowledge_handler, terminal);
 | 
				
			||||||
	window_set_key_handler(terminal->window, key_handler, terminal);
 | 
						window_set_key_handler(terminal->window, key_handler, terminal);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						terminal_draw(terminal);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return terminal;
 | 
						return terminal;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -486,7 +537,6 @@ int main(int argc, char *argv[])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	terminal = terminal_create(display, fd);
 | 
						terminal = terminal_create(display, fd);
 | 
				
			||||||
	terminal_run(terminal, "/bin/bash");
 | 
						terminal_run(terminal, "/bin/bash");
 | 
				
			||||||
	terminal_draw(terminal);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	g_main_loop_run(loop);
 | 
						g_main_loop_run(loop);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										16
									
								
								window.c
									
										
									
									
									
								
							
							
						
						
									
										16
									
								
								window.c
									
										
									
									
									
								
							| 
						 | 
					@ -247,20 +247,12 @@ event_handler(struct wl_display *display,
 | 
				
			||||||
		case WINDOW_RESIZING_LOWER_RIGHT:
 | 
							case WINDOW_RESIZING_LOWER_RIGHT:
 | 
				
			||||||
			window->width = window->drag_x + x;
 | 
								window->width = window->drag_x + x;
 | 
				
			||||||
			window->height = window->drag_y + y;
 | 
								window->height = window->drag_y + y;
 | 
				
			||||||
			if (window->width < window->minimum_width)
 | 
					 | 
				
			||||||
				window->width = window->minimum_width;
 | 
					 | 
				
			||||||
			if (window->height < window->minimum_height)
 | 
					 | 
				
			||||||
				window->height = window->minimum_height;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
			window_get_child_rectangle(window, &rectangle);
 | 
								window_get_child_rectangle(window, &rectangle);
 | 
				
			||||||
			if (window->resize_handler)
 | 
								if (window->resize_handler)
 | 
				
			||||||
				(*window->resize_handler)(window,
 | 
									(*window->resize_handler)(window,
 | 
				
			||||||
							  &rectangle,
 | 
					 | 
				
			||||||
							  window->user_data);
 | 
												  window->user_data);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			window->width = rectangle.width + 20;
 | 
					 | 
				
			||||||
			window->height = rectangle.height + 60;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			break;
 | 
								break;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	} else if (opcode == 1) {
 | 
						} else if (opcode == 1) {
 | 
				
			||||||
| 
						 | 
					@ -316,6 +308,14 @@ window_get_child_rectangle(struct window *window,
 | 
				
			||||||
	rectangle->height = window->height - 60;
 | 
						rectangle->height = window->height - 60;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void
 | 
				
			||||||
 | 
					window_set_child_size(struct window *window,
 | 
				
			||||||
 | 
							      struct rectangle *rectangle)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						window->width = rectangle->width + 20;
 | 
				
			||||||
 | 
						window->height = rectangle->height + 60;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void
 | 
					void
 | 
				
			||||||
window_copy(struct window *window,
 | 
					window_copy(struct window *window,
 | 
				
			||||||
	    struct rectangle *rectangle,
 | 
						    struct rectangle *rectangle,
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										5
									
								
								window.h
									
										
									
									
									
								
							
							
						
						
									
										5
									
								
								window.h
									
										
									
									
									
								
							| 
						 | 
					@ -32,7 +32,7 @@ struct rectangle {
 | 
				
			||||||
	int32_t height;
 | 
						int32_t height;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
typedef void (*window_resize_handler_t)(struct window *window, struct rectangle *rectangle, void *data);
 | 
					typedef void (*window_resize_handler_t)(struct window *window, void *data);
 | 
				
			||||||
typedef void (*window_frame_handler_t)(struct window *window, uint32_t frame, uint32_t timestamp, void *data);
 | 
					typedef void (*window_frame_handler_t)(struct window *window, uint32_t frame, uint32_t timestamp, void *data);
 | 
				
			||||||
typedef void (*window_acknowledge_handler_t)(struct window *window, uint32_t key, void *data);
 | 
					typedef void (*window_acknowledge_handler_t)(struct window *window, uint32_t key, void *data);
 | 
				
			||||||
typedef void (*window_key_handler_t)(struct window *window, uint32_t key, uint32_t state, void *data);
 | 
					typedef void (*window_key_handler_t)(struct window *window, uint32_t key, uint32_t state, void *data);
 | 
				
			||||||
| 
						 | 
					@ -52,6 +52,9 @@ void
 | 
				
			||||||
window_get_child_rectangle(struct window *window,
 | 
					window_get_child_rectangle(struct window *window,
 | 
				
			||||||
			   struct rectangle *rectangle);
 | 
								   struct rectangle *rectangle);
 | 
				
			||||||
void
 | 
					void
 | 
				
			||||||
 | 
					window_set_child_size(struct window *window,
 | 
				
			||||||
 | 
							      struct rectangle *rectangle);
 | 
				
			||||||
 | 
					void
 | 
				
			||||||
window_copy(struct window *window,
 | 
					window_copy(struct window *window,
 | 
				
			||||||
	    struct rectangle *rectangle,
 | 
						    struct rectangle *rectangle,
 | 
				
			||||||
	    uint32_t name, uint32_t stride);
 | 
						    uint32_t name, uint32_t stride);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue