You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							496 lines
						
					
					
						
							13 KiB
						
					
					
				
			
		
		
	
	
							496 lines
						
					
					
						
							13 KiB
						
					
					
				| /*
 | |
|  * Copyright © 2011, 2012 Intel Corporation
 | |
|  *
 | |
|  * Permission to use, copy, modify, distribute, and sell this software and its
 | |
|  * documentation for any purpose is hereby granted without fee, provided that
 | |
|  * the above copyright notice appear in all copies and that both that copyright
 | |
|  * notice and this permission notice appear in supporting documentation, and
 | |
|  * that the name of the copyright holders not be used in advertising or
 | |
|  * publicity pertaining to distribution of the software without specific,
 | |
|  * written prior permission.  The copyright holders make no representations
 | |
|  * about the suitability of this software for any purpose.  It is provided "as
 | |
|  * is" without express or implied warranty.
 | |
|  *
 | |
|  * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
 | |
|  * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
 | |
|  * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
 | |
|  * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
 | |
|  * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
 | |
|  * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
 | |
|  * OF THIS SOFTWARE.
 | |
|  */
 | |
| 
 | |
| #include <stdint.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <sys/wait.h>
 | |
| 
 | |
| #include "window.h"
 | |
| #include "../shared/cairo-util.h"
 | |
| #include "../shared/config-parser.h"
 | |
| 
 | |
| #include "tablet-shell-client-protocol.h"
 | |
| 
 | |
| struct tablet {
 | |
| 	struct display *display;
 | |
| 	struct tablet_shell *tablet_shell;
 | |
| 	struct rectangle allocation;
 | |
| 	struct window *switcher;
 | |
| 
 | |
| 	struct homescreen *homescreen;
 | |
| 	struct lockscreen *lockscreen;
 | |
| };
 | |
| 
 | |
| struct homescreen {
 | |
| 	struct window *window;
 | |
| 	struct widget *widget;
 | |
| 	struct wl_list launcher_list;
 | |
| };
 | |
| 
 | |
| struct lockscreen {
 | |
| 	struct window *window;
 | |
| 	struct widget *widget;
 | |
| };
 | |
| 
 | |
| struct launcher {
 | |
| 	struct widget *widget;
 | |
| 	struct homescreen *homescreen;
 | |
| 	cairo_surface_t *icon;
 | |
| 	int focused, pressed;
 | |
| 	char *path;
 | |
| 	struct wl_list link;
 | |
| };
 | |
| 
 | |
| static char *key_lockscreen_icon;
 | |
| static char *key_lockscreen_background;
 | |
| static char *key_homescreen_background;
 | |
| static char *key_launcher_icon;
 | |
| static char *key_launcher_path;
 | |
| static void launcher_section_done(void *data);
 | |
| 
 | |
| static const struct config_key shell_config_keys[] = {
 | |
| 	{ "lockscreen-icon", CONFIG_KEY_STRING, &key_lockscreen_icon },
 | |
| 	{ "lockscreen", CONFIG_KEY_STRING, &key_lockscreen_background },
 | |
| 	{ "homescreen", CONFIG_KEY_STRING, &key_homescreen_background },
 | |
| };
 | |
| 
 | |
| static const struct config_key launcher_config_keys[] = {
 | |
| 	{ "icon", CONFIG_KEY_STRING, &key_launcher_icon },
 | |
| 	{ "path", CONFIG_KEY_STRING, &key_launcher_path },
 | |
| };
 | |
| 
 | |
| static const struct config_section config_sections[] = {
 | |
| 	{ "shell",
 | |
| 	  shell_config_keys, ARRAY_LENGTH(shell_config_keys) },
 | |
| 	{ "launcher",
 | |
| 	  launcher_config_keys, ARRAY_LENGTH(launcher_config_keys),
 | |
| 	  launcher_section_done }
 | |
| };
 | |
| 
 | |
| static void
 | |
| sigchild_handler(int s)
 | |
| {
 | |
| 	int status;
 | |
| 	pid_t pid;
 | |
| 
 | |
| 	while (pid = waitpid(-1, &status, WNOHANG), pid > 0)
 | |
| 		fprintf(stderr, "child %d exited\n", pid);
 | |
| }
 | |
| 
 | |
| static void
 | |
| paint_background(cairo_t *cr, const char *path, struct rectangle *allocation)
 | |
| {
 | |
| 	cairo_surface_t *image = NULL;
 | |
| 	cairo_pattern_t *pattern;
 | |
| 	cairo_matrix_t matrix;
 | |
| 	double sx, sy;
 | |
| 
 | |
| 	cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
 | |
| 	if (path)
 | |
| 		image = load_cairo_surface(path);
 | |
| 	if (image) {
 | |
| 		pattern = cairo_pattern_create_for_surface(image);
 | |
| 		sx = (double) cairo_image_surface_get_width(image) /
 | |
| 			allocation->width;
 | |
| 		sy = (double) cairo_image_surface_get_height(image) /
 | |
| 			allocation->height;
 | |
| 		cairo_matrix_init_scale(&matrix, sx, sy);
 | |
| 		cairo_pattern_set_matrix(pattern, &matrix);
 | |
| 		cairo_set_source(cr, pattern);
 | |
| 		cairo_pattern_destroy (pattern);
 | |
| 		cairo_surface_destroy(image);
 | |
| 		cairo_paint(cr);
 | |
| 	} else {
 | |
| 		fprintf(stderr, "couldn't load background image: %s\n", path);
 | |
| 		cairo_set_source_rgb(cr, 0.2, 0, 0);
 | |
| 		cairo_paint(cr);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void
 | |
| homescreen_draw(struct widget *widget, void *data)
 | |
| {
 | |
| 	struct homescreen *homescreen = data;
 | |
| 	cairo_surface_t *surface;
 | |
| 	struct rectangle allocation;
 | |
| 	cairo_pattern_t *pattern;
 | |
| 	cairo_matrix_t matrix;
 | |
| 	cairo_t *cr;
 | |
| 	struct launcher *launcher;
 | |
| 	const int rows = 4, columns = 5, icon_width = 128, icon_height = 128;
 | |
| 	int x, y, i, width, height, vmargin, hmargin, vpadding, hpadding;
 | |
| 
 | |
| 	surface = window_get_surface(homescreen->window);
 | |
| 	cr = cairo_create(surface);
 | |
| 
 | |
| 	widget_get_allocation(widget, &allocation);
 | |
| 	paint_background(cr, key_homescreen_background, &allocation);
 | |
| 
 | |
| 	cairo_set_operator(cr, CAIRO_OPERATOR_OVER);
 | |
| 
 | |
| 	width = allocation.width - columns * icon_width;
 | |
| 	hpadding = width / (columns + 1);
 | |
| 	hmargin = (width - hpadding * (columns - 1)) / 2;
 | |
| 
 | |
| 	height = allocation.height - rows * icon_height;
 | |
| 	vpadding = height / (rows + 1);
 | |
| 	vmargin = (height - vpadding * (rows - 1)) / 2;
 | |
| 
 | |
| 	x = hmargin;
 | |
| 	y = vmargin;
 | |
| 	i = 0;
 | |
| 
 | |
| 	wl_list_for_each(launcher, &homescreen->launcher_list, link) {
 | |
| 		widget_set_allocation(launcher->widget,
 | |
| 				      x, y, icon_width, icon_height);
 | |
| 		x += icon_width + hpadding;
 | |
| 		i++;
 | |
| 		if (i == columns) {
 | |
| 			x = hmargin;
 | |
| 			y += icon_height + vpadding;
 | |
| 			i = 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	cairo_destroy(cr);
 | |
| 	cairo_surface_destroy(surface);
 | |
| }
 | |
| 
 | |
| static void
 | |
| lockscreen_draw(struct widget *widget, void *data)
 | |
| {
 | |
| 	struct lockscreen *lockscreen = data;
 | |
| 	cairo_surface_t *surface;
 | |
| 	cairo_surface_t *icon;
 | |
| 	struct rectangle allocation;
 | |
| 	cairo_t *cr;
 | |
| 	int width, height;
 | |
| 
 | |
| 	surface = window_get_surface(lockscreen->window);
 | |
| 	cr = cairo_create(surface);
 | |
| 
 | |
| 	widget_get_allocation(widget, &allocation);
 | |
| 	paint_background(cr, key_lockscreen_background, &allocation);
 | |
| 
 | |
| 	cairo_set_operator(cr, CAIRO_OPERATOR_OVER);
 | |
| 	icon = load_cairo_surface(key_lockscreen_icon);
 | |
| 	if (icon) {
 | |
| 		width = cairo_image_surface_get_width(icon);
 | |
| 		height = cairo_image_surface_get_height(icon);
 | |
| 		cairo_set_source_surface(cr, icon,
 | |
| 			allocation.x + (allocation.width - width) / 2,
 | |
| 			allocation.y + (allocation.height - height) / 2);
 | |
| 	} else {
 | |
| 		fprintf(stderr, "couldn't load lockscreen icon: %s\n",
 | |
| 				 key_lockscreen_icon);
 | |
| 		cairo_set_source_rgb(cr, 0.2, 0, 0);
 | |
| 	}
 | |
| 	cairo_paint(cr);
 | |
| 	cairo_destroy(cr);
 | |
| 	cairo_surface_destroy(icon);
 | |
| 	cairo_surface_destroy(surface);
 | |
| }
 | |
| 
 | |
| static void
 | |
| lockscreen_button_handler(struct widget *widget,
 | |
| 			  struct input *input, uint32_t time,
 | |
| 			  uint32_t button,
 | |
| 			  enum wl_pointer_button_state state, void *data)
 | |
| {
 | |
| 	struct lockscreen *lockscreen = data;
 | |
| 
 | |
| 	if (state == WL_POINTER_BUTTON_STATE_PRESSED && lockscreen->window) {
 | |
| 		window_destroy(lockscreen->window);
 | |
| 		lockscreen->window = NULL;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static struct homescreen *
 | |
| homescreen_create(struct tablet *tablet)
 | |
| {
 | |
| 	struct homescreen *homescreen;
 | |
| 
 | |
| 	homescreen = malloc (sizeof *homescreen);
 | |
| 	memset(homescreen, 0, sizeof *homescreen);
 | |
| 
 | |
| 	homescreen->window = window_create(tablet->display);
 | |
| 	homescreen->widget =
 | |
| 		window_add_widget(homescreen->window, homescreen);
 | |
| 	window_set_custom(homescreen->window);
 | |
| 	window_set_user_data(homescreen->window, homescreen);
 | |
| 	window_set_title(homescreen->window, "homescreen");
 | |
| 	widget_set_redraw_handler(homescreen->widget, homescreen_draw);
 | |
| 
 | |
| 	return homescreen;
 | |
| }
 | |
| 
 | |
| static struct lockscreen *
 | |
| lockscreen_create(struct tablet *tablet)
 | |
| {
 | |
| 	struct lockscreen *lockscreen;
 | |
| 
 | |
| 	lockscreen = malloc (sizeof *lockscreen);
 | |
| 	memset(lockscreen, 0, sizeof *lockscreen);
 | |
| 
 | |
| 	lockscreen->window = window_create(tablet->display);
 | |
| 	lockscreen->widget =
 | |
| 		window_add_widget(lockscreen->window, lockscreen);
 | |
| 	window_set_user_data(lockscreen->window, lockscreen);
 | |
| 	window_set_title(lockscreen->window, "lockscreen");
 | |
| 	window_set_custom(lockscreen->window);
 | |
| 	widget_set_redraw_handler(lockscreen->widget, lockscreen_draw);
 | |
| 	widget_set_button_handler(lockscreen->widget,
 | |
| 				  lockscreen_button_handler);
 | |
| 
 | |
| 	return lockscreen;
 | |
| }
 | |
| 
 | |
| static void
 | |
| show_lockscreen(void *data, struct tablet_shell *tablet_shell)
 | |
| {
 | |
| 	struct tablet *tablet = data;
 | |
| 
 | |
| 	tablet->lockscreen = lockscreen_create(tablet);
 | |
| 	tablet_shell_set_lockscreen(tablet->tablet_shell,
 | |
| 			window_get_wl_surface(tablet->lockscreen->window));
 | |
| 
 | |
| 	widget_schedule_resize(tablet->lockscreen->widget,
 | |
| 			       tablet->allocation.width,
 | |
| 			       tablet->allocation.height);
 | |
| }
 | |
| 
 | |
| static void
 | |
| show_switcher(void *data, struct tablet_shell *tablet_shell)
 | |
| {
 | |
| 	struct tablet *tablet = data;
 | |
| 
 | |
| 	tablet->switcher = window_create(tablet->display);
 | |
| 	window_set_user_data(tablet->switcher, tablet);
 | |
| 	window_set_custom(tablet->switcher);
 | |
| 	tablet_shell_set_switcher(tablet->tablet_shell,
 | |
| 				  window_get_wl_surface(tablet->switcher));
 | |
| }
 | |
| 
 | |
| static void
 | |
| hide_switcher(void *data, struct tablet_shell *tablet_shell)
 | |
| {
 | |
| }
 | |
| 
 | |
| static const struct tablet_shell_listener tablet_shell_listener = {
 | |
| 	show_lockscreen,
 | |
| 	show_switcher,
 | |
| 	hide_switcher
 | |
| };
 | |
| 
 | |
| static int
 | |
| launcher_enter_handler(struct widget *widget, struct input *input,
 | |
| 			     float x, float y, void *data)
 | |
| {
 | |
| 	struct launcher *launcher = data;
 | |
| 
 | |
| 	launcher->focused = 1;
 | |
| 	widget_schedule_redraw(widget);
 | |
| 
 | |
| 	return CURSOR_LEFT_PTR;
 | |
| }
 | |
| 
 | |
| static void
 | |
| launcher_leave_handler(struct widget *widget,
 | |
| 			     struct input *input, void *data)
 | |
| {
 | |
| 	struct launcher *launcher = data;
 | |
| 
 | |
| 	launcher->focused = 0;
 | |
| 	widget_schedule_redraw(widget);
 | |
| }
 | |
| 
 | |
| static void
 | |
| launcher_activate(struct launcher *widget)
 | |
| {
 | |
| 	pid_t pid;
 | |
| 
 | |
| 	pid = fork();
 | |
| 	if (pid < 0) {
 | |
| 		fprintf(stderr, "fork failed: %m\n");
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (pid)
 | |
| 		return;
 | |
| 
 | |
| 	if (execl(widget->path, widget->path, NULL) < 0) {
 | |
| 		fprintf(stderr, "execl '%s' failed: %m\n", widget->path);
 | |
| 		exit(1);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void
 | |
| launcher_button_handler(struct widget *widget,
 | |
| 			      struct input *input, uint32_t time,
 | |
| 			      uint32_t button,
 | |
| 			      enum wl_pointer_button_state state, void *data)
 | |
| {
 | |
| 	struct launcher *launcher;
 | |
| 
 | |
| 	launcher = widget_get_user_data(widget);
 | |
| 	widget_schedule_redraw(widget);
 | |
| 	if (state == WL_POINTER_BUTTON_STATE_RELEASED) {
 | |
| 		launcher_activate(launcher);
 | |
| 		launcher->pressed = 0;
 | |
| 	} else if (state == WL_POINTER_BUTTON_STATE_PRESSED) 
 | |
| 		launcher->pressed = 1;
 | |
| }
 | |
| 
 | |
| static void
 | |
| launcher_redraw_handler(struct widget *widget, void *data)
 | |
| {
 | |
| 	struct launcher *launcher = data;
 | |
| 	cairo_surface_t *surface;
 | |
| 	struct rectangle allocation;
 | |
| 	cairo_t *cr;
 | |
| 
 | |
| 	surface = window_get_surface(launcher->homescreen->window);
 | |
| 	cr = cairo_create(surface);
 | |
| 
 | |
| 	widget_get_allocation(widget, &allocation);
 | |
| 	if (launcher->pressed) {
 | |
| 		allocation.x++;
 | |
| 		allocation.y++;
 | |
| 	}
 | |
| 
 | |
| 	cairo_set_source_surface(cr, launcher->icon,
 | |
| 				 allocation.x, allocation.y);
 | |
| 	cairo_paint(cr);
 | |
| 
 | |
| 	if (launcher->focused) {
 | |
| 		cairo_set_source_rgba(cr, 1.0, 1.0, 1.0, 0.4);
 | |
| 		cairo_mask_surface(cr, launcher->icon,
 | |
| 				   allocation.x, allocation.y);
 | |
| 	}
 | |
| 
 | |
| 	cairo_destroy(cr);
 | |
| }
 | |
| 
 | |
| static void
 | |
| tablet_shell_add_launcher(struct tablet *tablet,
 | |
| 			  const char *icon, const char *path)
 | |
| {
 | |
| 	struct launcher *launcher;
 | |
| 	struct homescreen *homescreen = tablet->homescreen;
 | |
| 
 | |
| 	launcher = malloc(sizeof *launcher);
 | |
| 	launcher->path = strdup(path);
 | |
| 	launcher->icon = load_cairo_surface(icon);
 | |
| 	if (cairo_surface_status (launcher->icon) != CAIRO_STATUS_SUCCESS) {
 | |
| 		fprintf(stderr, "couldn't load %s\n", icon);
 | |
| 		free(launcher);
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	launcher->homescreen = homescreen;
 | |
| 	launcher->widget = widget_add_widget(homescreen->widget, launcher);
 | |
| 	widget_set_enter_handler(launcher->widget,
 | |
| 				 launcher_enter_handler);
 | |
| 	widget_set_leave_handler(launcher->widget,
 | |
| 				 launcher_leave_handler);
 | |
| 	widget_set_button_handler(launcher->widget,
 | |
| 				  launcher_button_handler);
 | |
| 	widget_set_redraw_handler(launcher->widget,
 | |
| 				  launcher_redraw_handler);
 | |
| 
 | |
| 	wl_list_insert(&homescreen->launcher_list, &launcher->link);
 | |
| }
 | |
| 
 | |
| static void
 | |
| launcher_section_done(void *data)
 | |
| {
 | |
| 	struct tablet *tablet = data;
 | |
| 
 | |
| 	if (key_launcher_icon == NULL || key_launcher_path == NULL) {
 | |
| 		fprintf(stderr, "invalid launcher section\n");
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	tablet_shell_add_launcher(tablet, key_launcher_icon, key_launcher_path);
 | |
| 
 | |
| 	free(key_launcher_icon);
 | |
| 	key_launcher_icon = NULL;
 | |
| 	free(key_launcher_path);
 | |
| 	key_launcher_path = NULL;
 | |
| }
 | |
| 
 | |
| static void
 | |
| global_handler(struct wl_display *display, uint32_t id,
 | |
| 		const char *interface, uint32_t version, void *data)
 | |
| {
 | |
| 	struct tablet *tablet = data;
 | |
| 
 | |
| 	if (!strcmp(interface, "tablet_shell")) {
 | |
| 		tablet->tablet_shell =
 | |
| 			wl_display_bind(display, id, &tablet_shell_interface);
 | |
| 		tablet_shell_add_listener(tablet->tablet_shell,
 | |
| 				&tablet_shell_listener, tablet);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int main(int argc, char *argv[])
 | |
| {
 | |
| 	struct tablet tablet = { 0 };
 | |
| 	struct display *display;
 | |
| 	char *config_file;
 | |
| 	struct output *output;
 | |
| 
 | |
| 	display = display_create(argc, argv);
 | |
| 	if (display == NULL) {
 | |
| 		fprintf(stderr, "failed to create display: %m\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	tablet.display = display;
 | |
| 
 | |
| 	wl_display_add_global_listener(display_get_display(tablet.display),
 | |
| 			global_handler, &tablet);
 | |
| 
 | |
| 	tablet.homescreen = homescreen_create(&tablet);
 | |
| 	tablet_shell_set_homescreen(tablet.tablet_shell,
 | |
| 			window_get_wl_surface(tablet.homescreen->window));
 | |
| 	wl_list_init(&tablet.homescreen->launcher_list);
 | |
| 
 | |
| 	config_file = config_file_path("weston.ini");
 | |
| 	parse_config_file(config_file,
 | |
| 			  config_sections, ARRAY_LENGTH(config_sections),
 | |
| 			  &tablet);
 | |
| 	free(config_file);
 | |
| 
 | |
| 	signal(SIGCHLD, sigchild_handler);
 | |
| 
 | |
| 	output = display_get_output(tablet.display);
 | |
| 	output_get_allocation(output, &tablet.allocation);
 | |
| 	widget_schedule_resize(tablet.homescreen->widget,
 | |
| 			tablet.allocation.width,
 | |
| 			tablet.allocation.height);
 | |
| 	display_run(display);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 |