| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | #define _POSIX_C_SOURCE 200809L
 | 
					
						
							|  |  |  | #include <ctype.h>
 | 
					
						
							|  |  |  | #include <poll.h>
 | 
					
						
							|  |  |  | #include <stdbool.h>
 | 
					
						
							|  |  |  | #include <signal.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <strings.h>
 | 
					
						
							|  |  |  | #include <time.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | #include <sys/mman.h>
 | 
					
						
							|  |  |  | #include <sys/timerfd.h>
 | 
					
						
							|  |  |  | #include <wayland-client.h>
 | 
					
						
							|  |  |  | #include <wayland-client-protocol.h>
 | 
					
						
							|  |  |  | #include <xkbcommon/xkbcommon.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "menu.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "pango.h"
 | 
					
						
							|  |  |  | #include "render.h"
 | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | #include "wayland.h"
 | 
					
						
							| 
									
										
										
										
											2024-03-02 11:31:13 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Creates and returns a new menu.
 | 
					
						
							| 
									
										
										
										
											2024-06-09 20:30:58 -04:00
										 |  |  | struct menu *menu_create(menu_callback callback) { | 
					
						
							| 
									
										
										
										
											2024-03-02 11:31:13 -05:00
										 |  |  | 	struct menu *menu = calloc(1, sizeof(struct menu)); | 
					
						
							|  |  |  | 	menu->strncmp = strncmp; | 
					
						
							|  |  |  | 	menu->font = "monospace 10"; | 
					
						
							|  |  |  | 	menu->normalbg = 0x222222ff; | 
					
						
							|  |  |  | 	menu->normalfg = 0xbbbbbbff; | 
					
						
							|  |  |  | 	menu->promptbg = 0x005577ff; | 
					
						
							|  |  |  | 	menu->promptfg = 0xeeeeeeff; | 
					
						
							|  |  |  | 	menu->selectionbg = 0x005577ff; | 
					
						
							|  |  |  | 	menu->selectionfg = 0xeeeeeeff; | 
					
						
							| 
									
										
										
										
											2024-06-09 20:30:58 -04:00
										 |  |  | 	menu->callback = callback; | 
					
						
							| 
									
										
										
										
											2024-03-02 11:31:13 -05:00
										 |  |  | 	return menu; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-03 19:10:28 -04:00
										 |  |  | static void free_pages(struct menu *menu) { | 
					
						
							|  |  |  | 	struct page *next = menu->pages; | 
					
						
							|  |  |  | 	while (next) { | 
					
						
							|  |  |  | 		struct page *page = next; | 
					
						
							|  |  |  | 		next = page->next; | 
					
						
							|  |  |  | 		free(page); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void free_item(struct item *item) { | 
					
						
							|  |  |  | 	free(item->text); | 
					
						
							|  |  |  | 	free(item); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void free_items(struct menu *menu) { | 
					
						
							|  |  |  | 	struct item *next = menu->items; | 
					
						
							|  |  |  | 	while (next) { | 
					
						
							|  |  |  | 		struct item *item = next; | 
					
						
							|  |  |  | 		next = item->next; | 
					
						
							|  |  |  | 		free_item(item); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Destroys the menu, freeing memory associated with it.
 | 
					
						
							|  |  |  | void menu_destroy(struct menu *menu) { | 
					
						
							|  |  |  | 	free_pages(menu); | 
					
						
							|  |  |  | 	free_items(menu); | 
					
						
							|  |  |  | 	free(menu); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | static bool parse_color(const char *color, uint32_t *result) { | 
					
						
							|  |  |  | 	if (color[0] == '#') { | 
					
						
							|  |  |  | 		++color; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	size_t len = strlen(color); | 
					
						
							|  |  |  | 	if ((len != 6 && len != 8) || !isxdigit(color[0]) || !isxdigit(color[1])) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	char *ptr; | 
					
						
							|  |  |  | 	uint32_t parsed = (uint32_t)strtoul(color, &ptr, 16); | 
					
						
							|  |  |  | 	if (*ptr != '\0') { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	*result = len == 6 ? ((parsed << 8) | 0xFF) : parsed; | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-02 11:31:13 -05:00
										 |  |  | // Parse menu options from command line arguments.
 | 
					
						
							|  |  |  | void menu_getopts(struct menu *menu, int argc, char *argv[]) { | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 	const char *usage = | 
					
						
							| 
									
										
										
										
											2024-05-05 10:13:01 -04:00
										 |  |  | 		"Usage: wmenu [-biPv] [-f font] [-l lines] [-o output] [-p prompt]\n" | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 		"\t[-N color] [-n color] [-M color] [-m color] [-S color] [-s color]\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int opt; | 
					
						
							| 
									
										
										
										
											2024-05-04 21:44:59 -04:00
										 |  |  | 	while ((opt = getopt(argc, argv, "bhiPvf:l:o:p:N:n:M:m:S:s:")) != -1) { | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 		switch (opt) { | 
					
						
							|  |  |  | 		case 'b': | 
					
						
							|  |  |  | 			menu->bottom = true; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 'i': | 
					
						
							|  |  |  | 			menu->strncmp = strncasecmp; | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2024-05-04 21:44:59 -04:00
										 |  |  | 		case 'P': | 
					
						
							|  |  |  | 			menu->passwd = true; | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 		case 'v': | 
					
						
							|  |  |  | 			puts("wmenu " VERSION); | 
					
						
							|  |  |  | 			exit(EXIT_SUCCESS); | 
					
						
							|  |  |  | 		case 'f': | 
					
						
							|  |  |  | 			menu->font = optarg; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 'l': | 
					
						
							|  |  |  | 			menu->lines = atoi(optarg); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 'o': | 
					
						
							|  |  |  | 			menu->output_name = optarg; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 'p': | 
					
						
							|  |  |  | 			menu->prompt = optarg; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 'N': | 
					
						
							| 
									
										
										
										
											2024-03-02 11:31:13 -05:00
										 |  |  | 			if (!parse_color(optarg, &menu->normalbg)) { | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 				fprintf(stderr, "Invalid background color: %s", optarg); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 'n': | 
					
						
							| 
									
										
										
										
											2024-03-02 11:31:13 -05:00
										 |  |  | 			if (!parse_color(optarg, &menu->normalfg)) { | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 				fprintf(stderr, "Invalid foreground color: %s", optarg); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 'M': | 
					
						
							|  |  |  | 			if (!parse_color(optarg, &menu->promptbg)) { | 
					
						
							|  |  |  | 				fprintf(stderr, "Invalid prompt background color: %s", optarg); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 'm': | 
					
						
							|  |  |  | 			if (!parse_color(optarg, &menu->promptfg)) { | 
					
						
							|  |  |  | 				fprintf(stderr, "Invalid prompt foreground color: %s", optarg); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 'S': | 
					
						
							|  |  |  | 			if (!parse_color(optarg, &menu->selectionbg)) { | 
					
						
							|  |  |  | 				fprintf(stderr, "Invalid selection background color: %s", optarg); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case 's': | 
					
						
							|  |  |  | 			if (!parse_color(optarg, &menu->selectionfg)) { | 
					
						
							|  |  |  | 				fprintf(stderr, "Invalid selection foreground color: %s", optarg); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			fprintf(stderr, "%s", usage); | 
					
						
							|  |  |  | 			exit(EXIT_FAILURE); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (optind < argc) { | 
					
						
							|  |  |  | 		fprintf(stderr, "%s", usage); | 
					
						
							|  |  |  | 		exit(EXIT_FAILURE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int height = get_font_height(menu->font); | 
					
						
							| 
									
										
										
										
											2024-04-09 11:12:49 +03:00
										 |  |  | 	menu->line_height = height + 2; | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 	menu->height = menu->line_height; | 
					
						
							|  |  |  | 	if (menu->lines > 0) { | 
					
						
							|  |  |  | 		menu->height += menu->height * menu->lines; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	menu->padding = height / 2; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | // Add an item to the menu.
 | 
					
						
							| 
									
										
										
										
											2024-05-03 19:10:28 -04:00
										 |  |  | void menu_add_item(struct menu *menu, char *text, bool sort) { | 
					
						
							|  |  |  | 	struct item *new = calloc(1, sizeof(struct item)); | 
					
						
							|  |  |  | 	if (!new) { | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-05-03 19:10:28 -04:00
										 |  |  | 	new->text = text; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (sort) { | 
					
						
							|  |  |  | 		for (struct item **item = &menu->items; *item; item = &(*item)->next) { | 
					
						
							|  |  |  | 			int result = strcmp(new->text, (*item)->text); | 
					
						
							|  |  |  | 			if (result == 0) { | 
					
						
							|  |  |  | 				free_item(new); | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (result < 0) { | 
					
						
							|  |  |  | 				new->next = *item; | 
					
						
							|  |  |  | 				*item = new; | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (menu->lastitem) { | 
					
						
							| 
									
										
										
										
											2024-05-03 19:10:28 -04:00
										 |  |  | 		menu->lastitem->next = new; | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2024-05-03 19:10:28 -04:00
										 |  |  | 		menu->items = new; | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-05-03 19:10:28 -04:00
										 |  |  | 	menu->lastitem = new; | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | static void append_page(struct page *page, struct page **first, struct page **last) { | 
					
						
							|  |  |  | 	if (*last) { | 
					
						
							|  |  |  | 		(*last)->next = page; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		*first = page; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	page->prev = *last; | 
					
						
							|  |  |  | 	page->next = NULL; | 
					
						
							|  |  |  | 	*last = page; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void page_items(struct menu *menu) { | 
					
						
							|  |  |  | 	// Free existing pages
 | 
					
						
							|  |  |  | 	while (menu->pages != NULL) { | 
					
						
							|  |  |  | 		struct page *page = menu->pages; | 
					
						
							|  |  |  | 		menu->pages = menu->pages->next; | 
					
						
							|  |  |  | 		free(page); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!menu->matches) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Make new pages
 | 
					
						
							|  |  |  | 	if (menu->lines > 0) { | 
					
						
							|  |  |  | 		struct page *pages_end = NULL; | 
					
						
							|  |  |  | 		struct item *item = menu->matches; | 
					
						
							|  |  |  | 		while (item) { | 
					
						
							|  |  |  | 			struct page *page = calloc(1, sizeof(struct page)); | 
					
						
							|  |  |  | 			page->first = item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for (int i = 1; item && i <= menu->lines; i++) { | 
					
						
							|  |  |  | 				item->page = page; | 
					
						
							|  |  |  | 				page->last = item; | 
					
						
							|  |  |  | 				item = item->next_match; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			append_page(page, &menu->pages, &pages_end); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		// Calculate available space
 | 
					
						
							|  |  |  | 		int max_width = menu->width - menu->inputw - menu->promptw | 
					
						
							|  |  |  | 			- menu->left_arrow - menu->right_arrow; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		struct page *pages_end = NULL; | 
					
						
							|  |  |  | 		struct item *item = menu->matches; | 
					
						
							|  |  |  | 		while (item) { | 
					
						
							|  |  |  | 			struct page *page = calloc(1, sizeof(struct page)); | 
					
						
							|  |  |  | 			page->first = item; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			int total_width = 0; | 
					
						
							| 
									
										
										
										
											2024-03-25 08:20:36 -04:00
										 |  |  | 			int items = 0; | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 			while (item) { | 
					
						
							|  |  |  | 				total_width += item->width + 2 * menu->padding; | 
					
						
							| 
									
										
										
										
											2024-03-25 08:20:36 -04:00
										 |  |  | 				if (total_width > max_width && items > 0) { | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2024-03-25 08:20:36 -04:00
										 |  |  | 				items++; | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				item->page = page; | 
					
						
							|  |  |  | 				page->last = item; | 
					
						
							|  |  |  | 				item = item->next_match; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			append_page(page, &menu->pages, &pages_end); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const char *fstrstr(struct menu *menu, const char *s, const char *sub) { | 
					
						
							|  |  |  | 	for (size_t len = strlen(sub); *s; s++) { | 
					
						
							|  |  |  | 		if (!menu->strncmp(s, sub, len)) { | 
					
						
							|  |  |  | 			return s; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | static void append_match(struct item *item, struct item **first, struct item **last) { | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 	if (*last) { | 
					
						
							|  |  |  | 		(*last)->next_match = item; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		*first = item; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	item->prev_match = *last; | 
					
						
							|  |  |  | 	item->next_match = NULL; | 
					
						
							|  |  |  | 	*last = item; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void match_items(struct menu *menu) { | 
					
						
							|  |  |  | 	struct item *lexact = NULL, *exactend = NULL; | 
					
						
							|  |  |  | 	struct item *lprefix = NULL, *prefixend = NULL; | 
					
						
							|  |  |  | 	struct item *lsubstr  = NULL, *substrend = NULL; | 
					
						
							|  |  |  | 	char buf[sizeof menu->input], *tok; | 
					
						
							|  |  |  | 	char **tokv = NULL; | 
					
						
							|  |  |  | 	int i, tokc = 0; | 
					
						
							|  |  |  | 	size_t tok_len; | 
					
						
							|  |  |  | 	menu->matches = NULL; | 
					
						
							|  |  |  | 	menu->matches_end = NULL; | 
					
						
							|  |  |  | 	menu->sel = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-02 07:32:43 -05:00
										 |  |  | 	size_t input_len = strlen(menu->input); | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-02 07:32:43 -05:00
										 |  |  | 	/* tokenize input by space for matching the tokens individually */ | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 	strcpy(buf, menu->input); | 
					
						
							|  |  |  | 	tok = strtok(buf, " "); | 
					
						
							|  |  |  | 	while (tok) { | 
					
						
							|  |  |  | 		tokv = realloc(tokv, (tokc + 1) * sizeof *tokv); | 
					
						
							|  |  |  | 		if (!tokv) { | 
					
						
							|  |  |  | 			fprintf(stderr, "could not realloc %zu bytes", | 
					
						
							|  |  |  | 					(tokc + 1) * sizeof *tokv); | 
					
						
							|  |  |  | 			exit(EXIT_FAILURE); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		tokv[tokc] = tok; | 
					
						
							|  |  |  | 		tokc++; | 
					
						
							|  |  |  | 		tok = strtok(NULL, " "); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	tok_len = tokc ? strlen(tokv[0]) : 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct item *item; | 
					
						
							|  |  |  | 	for (item = menu->items; item; item = item->next) { | 
					
						
							|  |  |  | 		for (i = 0; i < tokc; i++) { | 
					
						
							|  |  |  | 			if (!fstrstr(menu, item->text, tokv[i])) { | 
					
						
							|  |  |  | 				/* token does not match */ | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (i != tokc) { | 
					
						
							|  |  |  | 			/* not all tokens match */ | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-03-02 07:32:43 -05:00
										 |  |  | 		if (!tokc || !menu->strncmp(menu->input, item->text, input_len + 1)) { | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 			append_match(item, &lexact, &exactend); | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 		} else if (!menu->strncmp(tokv[0], item->text, tok_len)) { | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 			append_match(item, &lprefix, &prefixend); | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 			append_match(item, &lsubstr, &substrend); | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-02 11:31:13 -05:00
										 |  |  | 	free(tokv); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 	if (lexact) { | 
					
						
							|  |  |  | 		menu->matches = lexact; | 
					
						
							|  |  |  | 		menu->matches_end = exactend; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (lprefix) { | 
					
						
							|  |  |  | 		if (menu->matches_end) { | 
					
						
							|  |  |  | 			menu->matches_end->next_match = lprefix; | 
					
						
							|  |  |  | 			lprefix->prev_match = menu->matches_end; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			menu->matches = lprefix; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		menu->matches_end = prefixend; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (lsubstr) { | 
					
						
							|  |  |  | 		if (menu->matches_end) { | 
					
						
							|  |  |  | 			menu->matches_end->next_match = lsubstr; | 
					
						
							|  |  |  | 			lsubstr->prev_match = menu->matches_end; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			menu->matches = lsubstr; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		menu->matches_end = substrend; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	page_items(menu); | 
					
						
							|  |  |  | 	if (menu->pages) { | 
					
						
							|  |  |  | 		menu->sel = menu->pages->first; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-02 18:45:49 -04:00
										 |  |  | // Render menu items.
 | 
					
						
							|  |  |  | void menu_render_items(struct menu *menu) { | 
					
						
							|  |  |  | 	render_menu(menu); | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 	calc_widths(menu); | 
					
						
							|  |  |  | 	match_items(menu); | 
					
						
							| 
									
										
										
										
											2024-05-02 18:45:49 -04:00
										 |  |  | 	render_menu(menu); | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | static void insert(struct menu *menu, const char *text, ssize_t len) { | 
					
						
							|  |  |  | 	if (strlen(menu->input) + len > sizeof menu->input - 1) { | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 	memmove(menu->input + menu->cursor + len, menu->input + menu->cursor, | 
					
						
							|  |  |  | 			sizeof menu->input - menu->cursor - MAX(len, 0)); | 
					
						
							|  |  |  | 	if (len > 0 && text != NULL) { | 
					
						
							|  |  |  | 		memcpy(menu->input + menu->cursor, text, len); | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 	menu->cursor += len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Add pasted text to the menu input.
 | 
					
						
							|  |  |  | void menu_paste(struct menu *menu, const char *text, ssize_t len) { | 
					
						
							|  |  |  | 	insert(menu, text, len); | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static size_t nextrune(struct menu *menu, int incr) { | 
					
						
							|  |  |  | 	size_t n, len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	len = strlen(menu->input); | 
					
						
							|  |  |  | 	for(n = menu->cursor + incr; n < len && (menu->input[n] & 0xc0) == 0x80; n += incr); | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Move the cursor to the beginning or end of the word, skipping over any preceding whitespace.
 | 
					
						
							|  |  |  | static void movewordedge(struct menu *menu, int dir) { | 
					
						
							| 
									
										
										
										
											2024-02-27 12:00:10 -05:00
										 |  |  | 	if (dir < 0) { | 
					
						
							|  |  |  | 		// Move to beginning of word
 | 
					
						
							|  |  |  | 		while (menu->cursor > 0 && menu->input[nextrune(menu, -1)] == ' ') { | 
					
						
							|  |  |  | 			menu->cursor = nextrune(menu, -1); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		while (menu->cursor > 0 && menu->input[nextrune(menu, -1)] != ' ') { | 
					
						
							|  |  |  | 			menu->cursor = nextrune(menu, -1); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		// Move to end of word
 | 
					
						
							|  |  |  | 		size_t len = strlen(menu->input); | 
					
						
							|  |  |  | 		while (menu->cursor < len && menu->input[menu->cursor] == ' ') { | 
					
						
							|  |  |  | 			menu->cursor = nextrune(menu, +1); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		while (menu->cursor < len && menu->input[menu->cursor] != ' ') { | 
					
						
							|  |  |  | 			menu->cursor = nextrune(menu, +1); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Handle a keypress.
 | 
					
						
							|  |  |  | void menu_keypress(struct menu *menu, enum wl_keyboard_key_state key_state, | 
					
						
							|  |  |  | 		xkb_keysym_t sym) { | 
					
						
							|  |  |  | 	if (key_state != WL_KEYBOARD_KEY_STATE_PRESSED) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 	struct xkb_state *state = context_get_xkb_state(menu->context); | 
					
						
							|  |  |  | 	bool ctrl = xkb_state_mod_name_is_active(state, XKB_MOD_NAME_CTRL, | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 			XKB_STATE_MODS_DEPRESSED | XKB_STATE_MODS_LATCHED); | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 	bool meta = xkb_state_mod_name_is_active(state, XKB_MOD_NAME_ALT, | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 			XKB_STATE_MODS_DEPRESSED | XKB_STATE_MODS_LATCHED); | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 	bool shift = xkb_state_mod_name_is_active(state, XKB_MOD_NAME_SHIFT, | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 			XKB_STATE_MODS_DEPRESSED | XKB_STATE_MODS_LATCHED); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	size_t len = strlen(menu->input); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ctrl) { | 
					
						
							|  |  |  | 		// Emacs-style line editing bindings
 | 
					
						
							|  |  |  | 		switch (sym) { | 
					
						
							|  |  |  | 		case XKB_KEY_a: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Home; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_b: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Left; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_c: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Escape; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_d: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Delete; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_e: | 
					
						
							|  |  |  | 			sym = XKB_KEY_End; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_f: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Right; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_g: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Escape; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_bracketleft: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Escape; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_h: | 
					
						
							|  |  |  | 			sym = XKB_KEY_BackSpace; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_i: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Tab; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_j: | 
					
						
							|  |  |  | 		case XKB_KEY_J: | 
					
						
							|  |  |  | 		case XKB_KEY_m: | 
					
						
							|  |  |  | 		case XKB_KEY_M: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Return; | 
					
						
							|  |  |  | 			ctrl = false; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_n: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Down; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_p: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Up; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case XKB_KEY_k: | 
					
						
							|  |  |  | 			// Delete right
 | 
					
						
							|  |  |  | 			menu->input[menu->cursor] = '\0'; | 
					
						
							|  |  |  | 			match_items(menu); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		case XKB_KEY_u: | 
					
						
							|  |  |  | 			// Delete left
 | 
					
						
							|  |  |  | 			insert(menu, NULL, 0 - menu->cursor); | 
					
						
							|  |  |  | 			match_items(menu); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		case XKB_KEY_w: | 
					
						
							|  |  |  | 			// Delete word
 | 
					
						
							|  |  |  | 			while (menu->cursor > 0 && menu->input[nextrune(menu, -1)] == ' ') { | 
					
						
							|  |  |  | 				insert(menu, NULL, nextrune(menu, -1) - menu->cursor); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			while (menu->cursor > 0 && menu->input[nextrune(menu, -1)] != ' ') { | 
					
						
							|  |  |  | 				insert(menu, NULL, nextrune(menu, -1) - menu->cursor); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			match_items(menu); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		case XKB_KEY_Y: | 
					
						
							|  |  |  | 			// Paste clipboard
 | 
					
						
							| 
									
										
										
										
											2024-05-02 17:03:07 -04:00
										 |  |  | 			if (!context_paste(menu->context)) { | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			match_items(menu); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		case XKB_KEY_Left: | 
					
						
							|  |  |  | 		case XKB_KEY_KP_Left: | 
					
						
							|  |  |  | 			movewordedge(menu, -1); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		case XKB_KEY_Right: | 
					
						
							|  |  |  | 		case XKB_KEY_KP_Right: | 
					
						
							|  |  |  | 			movewordedge(menu, +1); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case XKB_KEY_Return: | 
					
						
							|  |  |  | 		case XKB_KEY_KP_Enter: | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else if (meta) { | 
					
						
							|  |  |  | 		// Emacs-style line editing bindings
 | 
					
						
							|  |  |  | 		switch (sym) { | 
					
						
							|  |  |  | 		case XKB_KEY_b: | 
					
						
							|  |  |  | 			movewordedge(menu, -1); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		case XKB_KEY_f: | 
					
						
							|  |  |  | 			movewordedge(menu, +1); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		case XKB_KEY_g: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Home; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_G: | 
					
						
							|  |  |  | 			sym = XKB_KEY_End; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_h: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Up; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_j: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Next; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_k: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Prior; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case XKB_KEY_l: | 
					
						
							|  |  |  | 			sym = XKB_KEY_Down; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	char buf[8]; | 
					
						
							|  |  |  | 	switch (sym) { | 
					
						
							|  |  |  | 	case XKB_KEY_Return: | 
					
						
							|  |  |  | 	case XKB_KEY_KP_Enter: | 
					
						
							|  |  |  | 		if (shift) { | 
					
						
							| 
									
										
										
										
											2024-06-09 20:30:58 -04:00
										 |  |  | 			menu->callback(menu, menu->input, true); | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			char *text = menu->sel ? menu->sel->text : menu->input; | 
					
						
							| 
									
										
										
										
											2024-06-09 20:30:58 -04:00
										 |  |  | 			menu->callback(menu, text, !ctrl); | 
					
						
							| 
									
										
										
										
											2024-02-27 11:23:12 -05:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case XKB_KEY_Left: | 
					
						
							|  |  |  | 	case XKB_KEY_KP_Left: | 
					
						
							|  |  |  | 	case XKB_KEY_Up: | 
					
						
							|  |  |  | 	case XKB_KEY_KP_Up: | 
					
						
							|  |  |  | 		if (menu->sel && menu->sel->prev_match) { | 
					
						
							|  |  |  | 			menu->sel = menu->sel->prev_match; | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} else if (menu->cursor > 0) { | 
					
						
							|  |  |  | 			menu->cursor = nextrune(menu, -1); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case XKB_KEY_Right: | 
					
						
							|  |  |  | 	case XKB_KEY_KP_Right: | 
					
						
							|  |  |  | 	case XKB_KEY_Down: | 
					
						
							|  |  |  | 	case XKB_KEY_KP_Down: | 
					
						
							|  |  |  | 		if (menu->cursor < len) { | 
					
						
							|  |  |  | 			menu->cursor = nextrune(menu, +1); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} else if (menu->sel && menu->sel->next_match) { | 
					
						
							|  |  |  | 			menu->sel = menu->sel->next_match; | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case XKB_KEY_Prior: | 
					
						
							|  |  |  | 	case XKB_KEY_KP_Prior: | 
					
						
							|  |  |  | 		if (menu->sel && menu->sel->page->prev) { | 
					
						
							|  |  |  | 			menu->sel = menu->sel->page->prev->first; | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case XKB_KEY_Next: | 
					
						
							|  |  |  | 	case XKB_KEY_KP_Next: | 
					
						
							|  |  |  | 		if (menu->sel && menu->sel->page->next) { | 
					
						
							|  |  |  | 			menu->sel = menu->sel->page->next->first; | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case XKB_KEY_Home: | 
					
						
							|  |  |  | 	case XKB_KEY_KP_Home: | 
					
						
							|  |  |  | 		if (menu->sel == menu->matches) { | 
					
						
							|  |  |  | 			menu->cursor = 0; | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			menu->sel = menu->matches; | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case XKB_KEY_End: | 
					
						
							|  |  |  | 	case XKB_KEY_KP_End: | 
					
						
							|  |  |  | 		if (menu->cursor < len) { | 
					
						
							|  |  |  | 			menu->cursor = len; | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			menu->sel = menu->matches_end; | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case XKB_KEY_BackSpace: | 
					
						
							|  |  |  | 		if (menu->cursor > 0) { | 
					
						
							|  |  |  | 			insert(menu, NULL, nextrune(menu, -1) - menu->cursor); | 
					
						
							|  |  |  | 			match_items(menu); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case XKB_KEY_Delete: | 
					
						
							|  |  |  | 	case XKB_KEY_KP_Delete: | 
					
						
							|  |  |  | 		if (menu->cursor == len) { | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		menu->cursor = nextrune(menu, +1); | 
					
						
							|  |  |  | 		insert(menu, NULL, nextrune(menu, -1) - menu->cursor); | 
					
						
							|  |  |  | 		match_items(menu); | 
					
						
							|  |  |  | 		render_menu(menu); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case XKB_KEY_Tab: | 
					
						
							|  |  |  | 		if (!menu->sel) { | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		menu->cursor = strnlen(menu->sel->text, sizeof menu->input - 1); | 
					
						
							|  |  |  | 		memcpy(menu->input, menu->sel->text, menu->cursor); | 
					
						
							|  |  |  | 		menu->input[menu->cursor] = '\0'; | 
					
						
							|  |  |  | 		match_items(menu); | 
					
						
							|  |  |  | 		render_menu(menu); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case XKB_KEY_Escape: | 
					
						
							|  |  |  | 		menu->exit = true; | 
					
						
							|  |  |  | 		menu->failure = true; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		if (xkb_keysym_to_utf8(sym, buf, 8)) { | 
					
						
							|  |  |  | 			insert(menu, buf, strnlen(buf, 8)); | 
					
						
							|  |  |  | 			match_items(menu); | 
					
						
							|  |  |  | 			render_menu(menu); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |