blob: 599295ee3b916a395a1250498a4856d29de7efeb [file] [log] [blame]
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001/*
2 * Copyright © 2011 Kristian Høgsberg
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02003 * Copyright © 2011 Collabora, Ltd.
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04004 *
Bryce Harrington1f6b0d12015-06-10 22:48:59 -07005 * Permission is hereby granted, free of charge, to any person obtaining a
6 * copy of this software and associated documentation files (the "Software"),
7 * to deal in the Software without restriction, including without limitation
8 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9 * and/or sell copies of the Software, and to permit persons to whom the
10 * Software is furnished to do so, subject to the following conditions:
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040011 *
Bryce Harrington1f6b0d12015-06-10 22:48:59 -070012 * The above copyright notice and this permission notice (including the next
13 * paragraph) shall be included in all copies or substantial portions of the
14 * Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22 * DEALINGS IN THE SOFTWARE.
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040023 */
24
U. Artie Eoff3c946772014-01-15 10:59:50 -080025#include "config.h"
26
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040027#include <stdint.h>
28#include <stdio.h>
29#include <stdlib.h>
30#include <string.h>
31#include <fcntl.h>
32#include <unistd.h>
U. Artie Eoff3c946772014-01-15 10:59:50 -080033#include <errno.h>
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040034#include <math.h>
35#include <cairo.h>
36#include <sys/wait.h>
Martin Minarik1e51a872012-06-08 00:39:11 +020037#include <sys/timerfd.h>
Michael Vetter2a18a522015-05-15 17:17:47 +020038#include <sys/epoll.h>
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040039#include <linux/input.h>
Tiago Vignatti61500722012-05-23 22:06:28 +030040#include <libgen.h>
Kristian Høgsbergd1936b92012-07-23 22:59:33 -040041#include <ctype.h>
Martin Minarik1e51a872012-06-08 00:39:11 +020042#include <time.h>
Quentin Glidicf9574f22016-06-23 18:55:21 +020043#include <assert.h>
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040044
Pekka Paalanen50719bc2011-11-22 14:18:50 +020045#include <wayland-client.h>
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040046#include "window.h"
Jon Cruz4678bab2015-06-15 15:37:07 -070047#include "shared/cairo-util.h"
48#include "shared/config-parser.h"
Jon Cruz867d50e2015-06-15 15:37:10 -070049#include "shared/helpers.h"
Bryce Harringtone99e4bf2016-03-16 14:15:18 -070050#include "shared/xalloc.h"
Bryce Harrington0d1a6222016-02-11 16:42:49 -080051#include "shared/zalloc.h"
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040052
Jonas Ådahl6d6fb612015-11-17 16:00:33 +080053#include "weston-desktop-shell-client-protocol.h"
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040054
Armin Krezovićc6a55db2016-03-10 18:02:34 +010055#define DEFAULT_CLOCK_FORMAT CLOCK_FORMAT_MINUTES
56
Pekka Paalanenb6df4f72012-08-03 14:39:15 +030057extern char **environ; /* defined by libc */
58
Quentin Glidicf9574f22016-06-23 18:55:21 +020059enum clock_format {
60 CLOCK_FORMAT_MINUTES,
61 CLOCK_FORMAT_SECONDS,
62 CLOCK_FORMAT_NONE
63};
64
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040065struct desktop {
66 struct display *display;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +080067 struct weston_desktop_shell *shell;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +020068 struct unlock_dialog *unlock_dialog;
69 struct task unlock_task;
Benjamin Franzked0f79ab2011-11-22 12:43:52 +010070 struct wl_list outputs;
Kristian Høgsbergd56bd902012-06-05 09:58:51 -040071
Quentin Glidic55d57012016-06-23 18:55:18 +020072 int want_panel;
73 enum weston_desktop_shell_panel_position panel_position;
Quentin Glidic3e37b342016-06-23 18:55:22 +020074 enum clock_format clock_format;
Quentin Glidic55d57012016-06-23 18:55:18 +020075
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +030076 struct window *grab_window;
77 struct widget *grab_widget;
78
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +030079 struct weston_config *config;
80 int locking;
81
Scott Moreauec116022012-07-22 18:23:52 -060082 enum cursor_type grab_cursor;
Pekka Paalanen79346ab2013-05-22 18:03:09 +030083
84 int painted;
Benjamin Franzked0f79ab2011-11-22 12:43:52 +010085};
86
87struct surface {
88 void (*configure)(void *data,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +080089 struct weston_desktop_shell *desktop_shell,
Kristian Høgsbergeae5de72012-04-11 22:42:15 -040090 uint32_t edges, struct window *window,
Benjamin Franzked0f79ab2011-11-22 12:43:52 +010091 int32_t width, int32_t height);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040092};
93
94struct panel {
Benjamin Franzked0f79ab2011-11-22 12:43:52 +010095 struct surface base;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -040096 struct window *window;
Kristian Høgsberg75bc6672012-01-10 09:43:58 -050097 struct widget *widget;
Kristian Høgsberg75bc6672012-01-10 09:43:58 -050098 struct wl_list launcher_list;
Martin Minarik1e51a872012-06-08 00:39:11 +020099 struct panel_clock *clock;
Pekka Paalanen79346ab2013-05-22 18:03:09 +0300100 int painted;
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200101 enum weston_desktop_shell_panel_position panel_position;
Quentin Glidicf9574f22016-06-23 18:55:21 +0200102 enum clock_format clock_format;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300103 uint32_t color;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400104};
105
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100106struct background {
107 struct surface base;
108 struct window *window;
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500109 struct widget *widget;
Pekka Paalanen79346ab2013-05-22 18:03:09 +0300110 int painted;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300111
112 char *image;
113 int type;
114 uint32_t color;
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100115};
116
117struct output {
118 struct wl_output *output;
Xiong Zhang83d8ee72013-10-23 13:58:35 +0800119 uint32_t server_output_id;
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100120 struct wl_list link;
121
122 struct panel *panel;
123 struct background *background;
124};
125
Kristian Høgsberg53880802012-01-09 11:16:50 -0500126struct panel_launcher {
Kristian Høgsbergc51f7992012-01-08 15:09:53 -0500127 struct widget *widget;
Kristian Høgsberge28d05b2011-09-20 21:43:54 -0400128 struct panel *panel;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400129 cairo_surface_t *icon;
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500130 int focused, pressed;
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400131 char *path;
Kristian Høgsberg75bc6672012-01-10 09:43:58 -0500132 struct wl_list link;
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400133 struct wl_array envp;
134 struct wl_array argv;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400135};
136
Martin Minarik1e51a872012-06-08 00:39:11 +0200137struct panel_clock {
138 struct widget *widget;
139 struct panel *panel;
Martin Minarik1e51a872012-06-08 00:39:11 +0200140 struct task clock_task;
141 int clock_fd;
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100142 char *format_string;
143 time_t refresh_timer;
Martin Minarik1e51a872012-06-08 00:39:11 +0200144};
145
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200146struct unlock_dialog {
147 struct window *window;
Kristian Høgsberg75bc6672012-01-10 09:43:58 -0500148 struct widget *widget;
Kristian Høgsbergc51f7992012-01-08 15:09:53 -0500149 struct widget *button;
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500150 int button_focused;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200151 int closing;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200152 struct desktop *desktop;
153};
154
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300155static void
156panel_add_launchers(struct panel *panel, struct desktop *desktop);
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -0500157
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400158static void
159sigchild_handler(int s)
160{
161 int status;
162 pid_t pid;
163
164 while (pid = waitpid(-1, &status, WNOHANG), pid > 0)
165 fprintf(stderr, "child %d exited\n", pid);
166}
167
Pekka Paalanen79346ab2013-05-22 18:03:09 +0300168static int
169is_desktop_painted(struct desktop *desktop)
170{
171 struct output *output;
172
173 wl_list_for_each(output, &desktop->outputs, link) {
174 if (output->panel && !output->panel->painted)
175 return 0;
176 if (output->background && !output->background->painted)
177 return 0;
178 }
179
180 return 1;
181}
182
183static void
184check_desktop_ready(struct window *window)
185{
186 struct display *display;
187 struct desktop *desktop;
188
189 display = window_get_display(window);
190 desktop = display_get_user_data(display);
191
192 if (!desktop->painted && is_desktop_painted(desktop)) {
193 desktop->painted = 1;
194
Jonas Ådahl6d6fb612015-11-17 16:00:33 +0800195 weston_desktop_shell_desktop_ready(desktop->shell);
Pekka Paalanen79346ab2013-05-22 18:03:09 +0300196 }
197}
198
Kristian Høgsbergbcee9a42011-10-12 00:36:16 -0400199static void
Kristian Høgsberg53880802012-01-09 11:16:50 -0500200panel_launcher_activate(struct panel_launcher *widget)
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400201{
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400202 char **argv;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400203 pid_t pid;
204
205 pid = fork();
206 if (pid < 0) {
207 fprintf(stderr, "fork failed: %m\n");
208 return;
209 }
210
211 if (pid)
212 return;
Benjamin Franzked7759712011-11-22 12:38:48 +0100213
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400214 argv = widget->argv.data;
Derek Foreman091c8012017-03-24 09:41:13 -0500215
216 if (setsid() == -1)
217 exit(EXIT_FAILURE);
218
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400219 if (execve(argv[0], argv, widget->envp.data) < 0) {
220 fprintf(stderr, "execl '%s' failed: %m\n", argv[0]);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400221 exit(1);
222 }
223}
224
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400225static void
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500226panel_launcher_redraw_handler(struct widget *widget, void *data)
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400227{
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500228 struct panel_launcher *launcher = data;
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500229 struct rectangle allocation;
230 cairo_t *cr;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400231
Alexander Larssonc584fa62013-05-22 14:41:32 +0200232 cr = widget_cairo_create(launcher->panel->widget);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500233
234 widget_get_allocation(widget, &allocation);
Kristian Høgsberg75bc6672012-01-10 09:43:58 -0500235 if (launcher->pressed) {
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500236 allocation.x++;
237 allocation.y++;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400238 }
239
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500240 cairo_set_source_surface(cr, launcher->icon,
241 allocation.x, allocation.y);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400242 cairo_paint(cr);
243
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500244 if (launcher->focused) {
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400245 cairo_set_source_rgba(cr, 1.0, 1.0, 1.0, 0.4);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500246 cairo_mask_surface(cr, launcher->icon,
247 allocation.x, allocation.y);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400248 }
Kristian Høgsberge28d05b2011-09-20 21:43:54 -0400249
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500250 cairo_destroy(cr);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400251}
252
Tiago Vignatti61500722012-05-23 22:06:28 +0300253static int
254panel_launcher_motion_handler(struct widget *widget, struct input *input,
255 uint32_t time, float x, float y, void *data)
256{
257 struct panel_launcher *launcher = data;
258
259 widget_set_tooltip(widget, basename((char *)launcher->path), x, y);
260
Ander Conselvan de Oliveiradc8c8fc2012-05-25 16:01:41 +0300261 return CURSOR_LEFT_PTR;
Tiago Vignatti61500722012-05-23 22:06:28 +0300262}
263
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400264static void
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -0500265set_hex_color(cairo_t *cr, uint32_t color)
266{
Michael Vetter2a18a522015-05-15 17:17:47 +0200267 cairo_set_source_rgba(cr,
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -0500268 ((color >> 16) & 0xff) / 255.0,
269 ((color >> 8) & 0xff) / 255.0,
270 ((color >> 0) & 0xff) / 255.0,
271 ((color >> 24) & 0xff) / 255.0);
272}
273
274static void
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500275panel_redraw_handler(struct widget *widget, void *data)
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400276{
277 cairo_surface_t *surface;
278 cairo_t *cr;
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500279 struct panel *panel = data;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400280
Alexander Larssonc584fa62013-05-22 14:41:32 +0200281 cr = widget_cairo_create(panel->widget);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400282 cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300283 set_hex_color(cr, panel->color);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400284 cairo_paint(cr);
285
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400286 cairo_destroy(cr);
Alexander Larssonc584fa62013-05-22 14:41:32 +0200287 surface = window_get_surface(panel->window);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400288 cairo_surface_destroy(surface);
Pekka Paalanen79346ab2013-05-22 18:03:09 +0300289 panel->painted = 1;
290 check_desktop_ready(panel->window);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400291}
292
Kristian Høgsbergbb901fa2012-01-09 11:22:32 -0500293static int
Kristian Høgsberg53880802012-01-09 11:16:50 -0500294panel_launcher_enter_handler(struct widget *widget, struct input *input,
Kristian Høgsberg80680c72012-05-10 12:21:37 -0400295 float x, float y, void *data)
Kristian Høgsbergee143232012-01-09 08:42:24 -0500296{
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500297 struct panel_launcher *launcher = data;
298
299 launcher->focused = 1;
Kristian Høgsbergee143232012-01-09 08:42:24 -0500300 widget_schedule_redraw(widget);
Kristian Høgsbergbb901fa2012-01-09 11:22:32 -0500301
Ander Conselvan de Oliveiradc8c8fc2012-05-25 16:01:41 +0300302 return CURSOR_LEFT_PTR;
Kristian Høgsbergee143232012-01-09 08:42:24 -0500303}
304
305static void
Kristian Høgsberg53880802012-01-09 11:16:50 -0500306panel_launcher_leave_handler(struct widget *widget,
307 struct input *input, void *data)
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400308{
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500309 struct panel_launcher *launcher = data;
310
311 launcher->focused = 0;
Tiago Vignatti61500722012-05-23 22:06:28 +0300312 widget_destroy_tooltip(widget);
Kristian Høgsberg9a13dab2012-01-08 15:18:19 -0500313 widget_schedule_redraw(widget);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400314}
315
316static void
Kristian Høgsberg53880802012-01-09 11:16:50 -0500317panel_launcher_button_handler(struct widget *widget,
318 struct input *input, uint32_t time,
Daniel Stone4dbadb12012-05-30 16:31:51 +0100319 uint32_t button,
320 enum wl_pointer_button_state state, void *data)
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400321{
Kristian Høgsberg53880802012-01-09 11:16:50 -0500322 struct panel_launcher *launcher;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400323
Kristian Høgsberg53880802012-01-09 11:16:50 -0500324 launcher = widget_get_user_data(widget);
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500325 widget_schedule_redraw(widget);
Daniel Stone4dbadb12012-05-30 16:31:51 +0100326 if (state == WL_POINTER_BUTTON_STATE_RELEASED)
Kristian Høgsberg53880802012-01-09 11:16:50 -0500327 panel_launcher_activate(launcher);
Rusty Lynch4384a242013-08-08 21:28:22 -0700328
329}
330
331static void
Rusty Lynch1084da52013-08-15 09:10:08 -0700332panel_launcher_touch_down_handler(struct widget *widget, struct input *input,
333 uint32_t serial, uint32_t time, int32_t id,
Rusty Lynch4384a242013-08-08 21:28:22 -0700334 float x, float y, void *data)
335{
336 struct panel_launcher *launcher;
337
338 launcher = widget_get_user_data(widget);
339 launcher->focused = 1;
340 widget_schedule_redraw(widget);
341}
342
343static void
Rusty Lynch1084da52013-08-15 09:10:08 -0700344panel_launcher_touch_up_handler(struct widget *widget, struct input *input,
Michael Vetter2a18a522015-05-15 17:17:47 +0200345 uint32_t serial, uint32_t time, int32_t id,
Rusty Lynch1084da52013-08-15 09:10:08 -0700346 void *data)
Rusty Lynch4384a242013-08-08 21:28:22 -0700347{
348 struct panel_launcher *launcher;
349
350 launcher = widget_get_user_data(widget);
351 launcher->focused = 0;
352 widget_schedule_redraw(widget);
353 panel_launcher_activate(launcher);
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500354}
355
Martin Minarik1e51a872012-06-08 00:39:11 +0200356static void
357clock_func(struct task *task, uint32_t events)
358{
359 struct panel_clock *clock =
Kristian Høgsbergbb262cf2012-06-11 11:03:03 -0400360 container_of(task, struct panel_clock, clock_task);
Kristian Høgsberg70226bb2012-06-08 16:54:52 -0400361 uint64_t exp;
Martin Minarik1e51a872012-06-08 00:39:11 +0200362
Martin Olsson8df662a2012-07-08 03:03:47 +0200363 if (read(clock->clock_fd, &exp, sizeof exp) != sizeof exp)
364 abort();
Kristian Høgsberg92a984a2012-06-11 11:10:57 -0400365 widget_schedule_redraw(clock->widget);
Martin Minarik1e51a872012-06-08 00:39:11 +0200366}
367
368static void
369panel_clock_redraw_handler(struct widget *widget, void *data)
370{
Martin Minarik1e51a872012-06-08 00:39:11 +0200371 struct panel_clock *clock = data;
372 cairo_t *cr;
373 struct rectangle allocation;
374 cairo_text_extents_t extents;
375 cairo_font_extents_t font_extents;
Martin Minarik1e51a872012-06-08 00:39:11 +0200376 time_t rawtime;
377 struct tm * timeinfo;
Kristian Høgsberg92a984a2012-06-11 11:10:57 -0400378 char string[128];
Martin Minarik1e51a872012-06-08 00:39:11 +0200379
Kristian Høgsbergbb262cf2012-06-11 11:03:03 -0400380 time(&rawtime);
381 timeinfo = localtime(&rawtime);
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100382 strftime(string, sizeof string, clock->format_string, timeinfo);
Martin Minarik1e51a872012-06-08 00:39:11 +0200383
384 widget_get_allocation(widget, &allocation);
Kristian Høgsbergbb262cf2012-06-11 11:03:03 -0400385 if (allocation.width == 0)
386 return;
Martin Minarik1e51a872012-06-08 00:39:11 +0200387
Alexander Larssonc584fa62013-05-22 14:41:32 +0200388 cr = widget_cairo_create(clock->panel->widget);
Martin Minarik1e51a872012-06-08 00:39:11 +0200389 cairo_select_font_face(cr, "sans",
390 CAIRO_FONT_SLANT_NORMAL,
391 CAIRO_FONT_WEIGHT_NORMAL);
392 cairo_set_font_size(cr, 14);
Kristian Høgsberg92a984a2012-06-11 11:10:57 -0400393 cairo_text_extents(cr, string, &extents);
Martin Minarik1e51a872012-06-08 00:39:11 +0200394 cairo_font_extents (cr, &font_extents);
Kristian Høgsbergbb262cf2012-06-11 11:03:03 -0400395 cairo_move_to(cr, allocation.x + 5,
396 allocation.y + 3 * (allocation.height >> 2) + 1);
397 cairo_set_source_rgb(cr, 0, 0, 0);
Kristian Høgsberg92a984a2012-06-11 11:10:57 -0400398 cairo_show_text(cr, string);
Kristian Høgsbergbb262cf2012-06-11 11:03:03 -0400399 cairo_move_to(cr, allocation.x + 4,
400 allocation.y + 3 * (allocation.height >> 2));
401 cairo_set_source_rgb(cr, 1, 1, 1);
Kristian Høgsberg92a984a2012-06-11 11:10:57 -0400402 cairo_show_text(cr, string);
Martin Minarik1e51a872012-06-08 00:39:11 +0200403 cairo_destroy(cr);
404}
405
406static int
407clock_timer_reset(struct panel_clock *clock)
408{
409 struct itimerspec its;
Kristian Høgsbergbb262cf2012-06-11 11:03:03 -0400410
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100411 its.it_interval.tv_sec = clock->refresh_timer;
Martin Minarik1e51a872012-06-08 00:39:11 +0200412 its.it_interval.tv_nsec = 0;
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100413 its.it_value.tv_sec = clock->refresh_timer;
Martin Minarik1e51a872012-06-08 00:39:11 +0200414 its.it_value.tv_nsec = 0;
415 if (timerfd_settime(clock->clock_fd, 0, &its, NULL) < 0) {
416 fprintf(stderr, "could not set timerfd\n: %m");
417 return -1;
418 }
419
420 return 0;
421}
422
423static void
U. Artie Eoff44874d92012-10-02 21:12:35 -0700424panel_destroy_clock(struct panel_clock *clock)
425{
426 widget_destroy(clock->widget);
427
428 close(clock->clock_fd);
429
430 free(clock);
431}
432
433static void
Martin Minarik1e51a872012-06-08 00:39:11 +0200434panel_add_clock(struct panel *panel)
435{
436 struct panel_clock *clock;
437 int timerfd;
438
439 timerfd = timerfd_create(CLOCK_MONOTONIC, TFD_CLOEXEC);
440 if (timerfd < 0) {
441 fprintf(stderr, "could not create timerfd\n: %m");
442 return;
443 }
444
Peter Huttererf3d62272013-08-08 11:57:05 +1000445 clock = xzalloc(sizeof *clock);
Martin Minarik1e51a872012-06-08 00:39:11 +0200446 clock->panel = panel;
447 panel->clock = clock;
448 clock->clock_fd = timerfd;
449
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100450 switch (panel->clock_format) {
451 case CLOCK_FORMAT_MINUTES:
452 clock->format_string = "%a %b %d, %I:%M %p";
453 clock->refresh_timer = 60;
454 break;
455 case CLOCK_FORMAT_SECONDS:
456 clock->format_string = "%a %b %d, %I:%M:%S %p";
457 clock->refresh_timer = 1;
458 break;
Quentin Glidicf9574f22016-06-23 18:55:21 +0200459 case CLOCK_FORMAT_NONE:
460 assert(!"not reached");
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100461 }
462
Martin Minarik1e51a872012-06-08 00:39:11 +0200463 clock->clock_task.run = clock_func;
Kristian Høgsbergbb262cf2012-06-11 11:03:03 -0400464 display_watch_fd(window_get_display(panel->window), clock->clock_fd,
465 EPOLLIN, &clock->clock_task);
Martin Minarik1e51a872012-06-08 00:39:11 +0200466 clock_timer_reset(clock);
467
468 clock->widget = widget_add_widget(panel->widget, clock);
Kristian Høgsbergbb262cf2012-06-11 11:03:03 -0400469 widget_set_redraw_handler(clock->widget, panel_clock_redraw_handler);
Martin Minarik1e51a872012-06-08 00:39:11 +0200470}
471
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500472static void
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500473panel_resize_handler(struct widget *widget,
474 int32_t width, int32_t height, void *data)
475{
476 struct panel_launcher *launcher;
477 struct panel *panel = data;
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200478 int bx = width / 2;
Quentin Glidic51c2c372016-06-23 18:55:23 +0200479 int by = height / 2;
480 int spacing = 10;
481 int x = spacing;
482 int y = spacing;
483 int w, h;
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200484 int horizontal = panel->panel_position == WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP || panel->panel_position == WESTON_DESKTOP_SHELL_PANEL_POSITION_BOTTOM;
Michael Vetter2a18a522015-05-15 17:17:47 +0200485
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500486 wl_list_for_each(launcher, &panel->launcher_list, link) {
487 w = cairo_image_surface_get_width(launcher->icon);
488 h = cairo_image_surface_get_height(launcher->icon);
Quentin Glidic51c2c372016-06-23 18:55:23 +0200489
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200490 if (horizontal)
Quentin Glidic51c2c372016-06-23 18:55:23 +0200491 y = by - h / 2;
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200492 else
493 x = bx - w / 2;
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500494 widget_set_allocation(launcher->widget,
Quentin Glidic51c2c372016-06-23 18:55:23 +0200495 x, y, w + 1, h + 1);
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200496 if (horizontal)
Quentin Glidic51c2c372016-06-23 18:55:23 +0200497 x += w + spacing;
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200498 else
499 y += h + spacing;
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500500 }
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100501
502 h = 20;
503
504 if (panel->clock_format == CLOCK_FORMAT_SECONDS)
505 w = 190;
506 else /* CLOCK_FORMAT_MINUTES */
507 w = 170;
Pekka Paalanen01b17252012-06-12 17:42:26 +0300508
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200509 if (horizontal) {
Quentin Glidic51c2c372016-06-23 18:55:23 +0200510 x = width - w - spacing;
511 y = by - h / 2;
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200512 } else {
513 x = bx - w / 2;
514 y = height - h - spacing;
515 }
Quentin Glidic51c2c372016-06-23 18:55:23 +0200516
Pekka Paalanen01b17252012-06-12 17:42:26 +0300517 if (panel->clock)
518 widget_set_allocation(panel->clock->widget,
Quentin Glidic51c2c372016-06-23 18:55:23 +0200519 x, y, w + 1, h + 1);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500520}
521
522static void
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100523panel_configure(void *data,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +0800524 struct weston_desktop_shell *desktop_shell,
Kristian Høgsbergeae5de72012-04-11 22:42:15 -0400525 uint32_t edges, struct window *window,
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100526 int32_t width, int32_t height)
527{
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200528 struct desktop *desktop = data;
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500529 struct surface *surface = window_get_user_data(window);
530 struct panel *panel = container_of(surface, struct panel, base);
531
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200532 switch (desktop->panel_position) {
533 case WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP:
534 case WESTON_DESKTOP_SHELL_PANEL_POSITION_BOTTOM:
Quentin Glidic51c2c372016-06-23 18:55:23 +0200535 height = 32;
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200536 break;
537 case WESTON_DESKTOP_SHELL_PANEL_POSITION_LEFT:
538 case WESTON_DESKTOP_SHELL_PANEL_POSITION_RIGHT:
539 switch (desktop->clock_format) {
540 case CLOCK_FORMAT_NONE:
541 width = 32;
542 break;
543 case CLOCK_FORMAT_MINUTES:
544 width = 170;
545 break;
546 case CLOCK_FORMAT_SECONDS:
547 width = 190;
548 break;
549 }
550 break;
551 }
Quentin Glidic51c2c372016-06-23 18:55:23 +0200552 window_schedule_resize(panel->window, width, height);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100553}
554
U. Artie Eoff44874d92012-10-02 21:12:35 -0700555static void
556panel_destroy_launcher(struct panel_launcher *launcher)
557{
558 wl_array_release(&launcher->argv);
559 wl_array_release(&launcher->envp);
560
561 free(launcher->path);
562
563 cairo_surface_destroy(launcher->icon);
564
565 widget_destroy(launcher->widget);
566 wl_list_remove(&launcher->link);
567
568 free(launcher);
569}
570
571static void
572panel_destroy(struct panel *panel)
573{
574 struct panel_launcher *tmp;
575 struct panel_launcher *launcher;
576
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100577 if (panel->clock)
578 panel_destroy_clock(panel->clock);
U. Artie Eoff44874d92012-10-02 21:12:35 -0700579
580 wl_list_for_each_safe(launcher, tmp, &panel->launcher_list, link)
581 panel_destroy_launcher(launcher);
582
583 widget_destroy(panel->widget);
584 window_destroy(panel->window);
585
586 free(panel);
587}
588
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400589static struct panel *
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300590panel_create(struct desktop *desktop)
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400591{
592 struct panel *panel;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300593 struct weston_config_section *s;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400594
Peter Huttererf3d62272013-08-08 11:57:05 +1000595 panel = xzalloc(sizeof *panel);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400596
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100597 panel->base.configure = panel_configure;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300598 panel->window = window_create_custom(desktop->display);
Kristian Høgsberg75bc6672012-01-10 09:43:58 -0500599 panel->widget = window_add_widget(panel->window, panel);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500600 wl_list_init(&panel->launcher_list);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400601
602 window_set_title(panel->window, "panel");
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400603 window_set_user_data(panel->window, panel);
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500604
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500605 widget_set_redraw_handler(panel->widget, panel_redraw_handler);
606 widget_set_resize_handler(panel->widget, panel_resize_handler);
Michael Vetter2a18a522015-05-15 17:17:47 +0200607
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200608 panel->panel_position = desktop->panel_position;
Quentin Glidic3e37b342016-06-23 18:55:22 +0200609 panel->clock_format = desktop->clock_format;
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100610 if (panel->clock_format != CLOCK_FORMAT_NONE)
611 panel_add_clock(panel);
612
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300613 s = weston_config_get_section(desktop->config, "shell", NULL, NULL);
Bryce Harringtone776f2a2016-07-14 18:28:03 -0700614 weston_config_section_get_color(s, "panel-color",
615 &panel->color, 0xaa000000);
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300616
617 panel_add_launchers(panel, desktop);
618
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400619 return panel;
620}
621
Philipp Brüschweiler467668c2012-08-29 10:53:36 +0200622static cairo_surface_t *
623load_icon_or_fallback(const char *icon)
624{
625 cairo_surface_t *surface = cairo_image_surface_create_from_png(icon);
Philipp Brüschweiler96386b82013-04-15 20:10:40 +0200626 cairo_status_t status;
Philipp Brüschweiler467668c2012-08-29 10:53:36 +0200627 cairo_t *cr;
628
Philipp Brüschweiler96386b82013-04-15 20:10:40 +0200629 status = cairo_surface_status(surface);
630 if (status == CAIRO_STATUS_SUCCESS)
Philipp Brüschweiler467668c2012-08-29 10:53:36 +0200631 return surface;
632
633 cairo_surface_destroy(surface);
Philipp Brüschweiler96386b82013-04-15 20:10:40 +0200634 fprintf(stderr, "ERROR loading icon from file '%s', error: '%s'\n",
635 icon, cairo_status_to_string(status));
Philipp Brüschweiler467668c2012-08-29 10:53:36 +0200636
637 /* draw fallback icon */
638 surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32,
639 20, 20);
640 cr = cairo_create(surface);
641
642 cairo_set_source_rgba(cr, 0.8, 0.8, 0.8, 1);
643 cairo_paint(cr);
644
645 cairo_set_source_rgba(cr, 0, 0, 0, 1);
646 cairo_set_line_cap(cr, CAIRO_LINE_CAP_ROUND);
647 cairo_rectangle(cr, 0, 0, 20, 20);
648 cairo_move_to(cr, 4, 4);
649 cairo_line_to(cr, 16, 16);
650 cairo_move_to(cr, 4, 16);
651 cairo_line_to(cr, 16, 4);
652 cairo_stroke(cr);
653
654 cairo_destroy(cr);
655
656 return surface;
657}
658
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400659static void
Kristian Høgsberg53880802012-01-09 11:16:50 -0500660panel_add_launcher(struct panel *panel, const char *icon, const char *path)
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400661{
Kristian Høgsberg53880802012-01-09 11:16:50 -0500662 struct panel_launcher *launcher;
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400663 char *start, *p, *eq, **ps;
664 int i, j, k;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400665
Peter Huttererf3d62272013-08-08 11:57:05 +1000666 launcher = xzalloc(sizeof *launcher);
Philipp Brüschweiler467668c2012-08-29 10:53:36 +0200667 launcher->icon = load_icon_or_fallback(icon);
U. Artie Eoff3c946772014-01-15 10:59:50 -0800668 launcher->path = xstrdup(path);
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400669
670 wl_array_init(&launcher->envp);
671 wl_array_init(&launcher->argv);
Pekka Paalanenb6df4f72012-08-03 14:39:15 +0300672 for (i = 0; environ[i]; i++) {
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400673 ps = wl_array_add(&launcher->envp, sizeof *ps);
Pekka Paalanenb6df4f72012-08-03 14:39:15 +0300674 *ps = environ[i];
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400675 }
676 j = 0;
677
678 start = launcher->path;
679 while (*start) {
680 for (p = start, eq = NULL; *p && !isspace(*p); p++)
681 if (*p == '=')
682 eq = p;
683
684 if (eq && j == 0) {
685 ps = launcher->envp.data;
686 for (k = 0; k < i; k++)
687 if (strncmp(ps[k], start, eq - start) == 0) {
688 ps[k] = start;
689 break;
690 }
691 if (k == i) {
692 ps = wl_array_add(&launcher->envp, sizeof *ps);
693 *ps = start;
694 i++;
695 }
696 } else {
697 ps = wl_array_add(&launcher->argv, sizeof *ps);
698 *ps = start;
699 j++;
700 }
701
702 while (*p && isspace(*p))
703 *p++ = '\0';
704
705 start = p;
706 }
707
708 ps = wl_array_add(&launcher->envp, sizeof *ps);
709 *ps = NULL;
710 ps = wl_array_add(&launcher->argv, sizeof *ps);
711 *ps = NULL;
712
Kristian Høgsberg53880802012-01-09 11:16:50 -0500713 launcher->panel = panel;
Kristian Høgsberg75bc6672012-01-10 09:43:58 -0500714 wl_list_insert(panel->launcher_list.prev, &launcher->link);
715
Kristian Høgsberg441338c2012-01-10 13:52:34 -0500716 launcher->widget = widget_add_widget(panel->widget, launcher);
Kristian Høgsberg53880802012-01-09 11:16:50 -0500717 widget_set_enter_handler(launcher->widget,
718 panel_launcher_enter_handler);
719 widget_set_leave_handler(launcher->widget,
720 panel_launcher_leave_handler);
721 widget_set_button_handler(launcher->widget,
722 panel_launcher_button_handler);
Rusty Lynch4384a242013-08-08 21:28:22 -0700723 widget_set_touch_down_handler(launcher->widget,
724 panel_launcher_touch_down_handler);
725 widget_set_touch_up_handler(launcher->widget,
726 panel_launcher_touch_up_handler);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500727 widget_set_redraw_handler(launcher->widget,
728 panel_launcher_redraw_handler);
Tiago Vignatti61500722012-05-23 22:06:28 +0300729 widget_set_motion_handler(launcher->widget,
730 panel_launcher_motion_handler);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400731}
732
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500733enum {
734 BACKGROUND_SCALE,
Pekka Paalanena402b052013-05-22 18:03:10 +0300735 BACKGROUND_SCALE_CROP,
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500736 BACKGROUND_TILE
737};
738
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400739static void
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500740background_draw(struct widget *widget, void *data)
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400741{
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500742 struct background *background = data;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400743 cairo_surface_t *surface, *image;
Kristian Høgsberg7e690002011-09-08 18:18:02 -0400744 cairo_pattern_t *pattern;
745 cairo_matrix_t matrix;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400746 cairo_t *cr;
Pekka Paalanena402b052013-05-22 18:03:10 +0300747 double im_w, im_h;
748 double sx, sy, s;
749 double tx, ty;
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500750 struct rectangle allocation;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400751
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500752 surface = window_get_surface(background->window);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400753
Alexander Larssonc584fa62013-05-22 14:41:32 +0200754 cr = widget_cairo_create(background->widget);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400755 cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
756 cairo_set_source_rgba(cr, 0.0, 0.0, 0.2, 1.0);
757 cairo_paint(cr);
758
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500759 widget_get_allocation(widget, &allocation);
Kristian Høgsberg8129bc02012-01-25 14:55:33 -0500760 image = NULL;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300761 if (background->image)
762 image = load_cairo_surface(background->image);
Kristian Høgsberg4c3661f2014-04-21 22:54:37 -0700763 else if (background->color == 0)
764 image = load_cairo_surface(DATADIR "/weston/pattern.png");
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500765
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300766 if (image && background->type != -1) {
Pekka Paalanena402b052013-05-22 18:03:10 +0300767 im_w = cairo_image_surface_get_width(image);
768 im_h = cairo_image_surface_get_height(image);
769 sx = im_w / allocation.width;
770 sy = im_h / allocation.height;
771
Kristian Høgsberg7e690002011-09-08 18:18:02 -0400772 pattern = cairo_pattern_create_for_surface(image);
Pekka Paalanena402b052013-05-22 18:03:10 +0300773
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300774 switch (background->type) {
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500775 case BACKGROUND_SCALE:
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500776 cairo_matrix_init_scale(&matrix, sx, sy);
777 cairo_pattern_set_matrix(pattern, &matrix);
Bill Spitzak79b7cb32014-05-08 20:00:35 -0700778 cairo_pattern_set_extend(pattern, CAIRO_EXTEND_PAD);
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500779 break;
Pekka Paalanena402b052013-05-22 18:03:10 +0300780 case BACKGROUND_SCALE_CROP:
781 s = (sx < sy) ? sx : sy;
782 /* align center */
783 tx = (im_w - s * allocation.width) * 0.5;
784 ty = (im_h - s * allocation.height) * 0.5;
785 cairo_matrix_init_translate(&matrix, tx, ty);
786 cairo_matrix_scale(&matrix, s, s);
787 cairo_pattern_set_matrix(pattern, &matrix);
Bill Spitzak79b7cb32014-05-08 20:00:35 -0700788 cairo_pattern_set_extend(pattern, CAIRO_EXTEND_PAD);
Pekka Paalanena402b052013-05-22 18:03:10 +0300789 break;
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500790 case BACKGROUND_TILE:
791 cairo_pattern_set_extend(pattern, CAIRO_EXTEND_REPEAT);
792 break;
793 }
Pekka Paalanena402b052013-05-22 18:03:10 +0300794
Kristian Høgsberg7e690002011-09-08 18:18:02 -0400795 cairo_set_source(cr, pattern);
796 cairo_pattern_destroy (pattern);
Kristian Høgsberg27d38662011-10-20 13:11:12 -0400797 cairo_surface_destroy(image);
Kristian Høgsberg8129bc02012-01-25 14:55:33 -0500798 } else {
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300799 set_hex_color(cr, background->color);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400800 }
801
Kristian Høgsberg8129bc02012-01-25 14:55:33 -0500802 cairo_paint(cr);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400803 cairo_destroy(cr);
804 cairo_surface_destroy(surface);
Pekka Paalanen9564c752012-10-24 09:43:08 +0300805
Pekka Paalanen79346ab2013-05-22 18:03:09 +0300806 background->painted = 1;
807 check_desktop_ready(background->window);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400808}
809
810static void
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100811background_configure(void *data,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +0800812 struct weston_desktop_shell *desktop_shell,
Kristian Høgsbergeae5de72012-04-11 22:42:15 -0400813 uint32_t edges, struct window *window,
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100814 int32_t width, int32_t height)
815{
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500816 struct background *background =
817 (struct background *) window_get_user_data(window);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100818
Kristian Høgsberg4598f962014-01-01 16:32:09 -0800819 widget_schedule_resize(background->widget, width, height);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100820}
821
822static void
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500823unlock_dialog_redraw_handler(struct widget *widget, void *data)
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200824{
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500825 struct unlock_dialog *dialog = data;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200826 struct rectangle allocation;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200827 cairo_surface_t *surface;
Alexander Larssonc584fa62013-05-22 14:41:32 +0200828 cairo_t *cr;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200829 cairo_pattern_t *pat;
830 double cx, cy, r, f;
831
Alexander Larssonc584fa62013-05-22 14:41:32 +0200832 cr = widget_cairo_create(widget);
Kristian Høgsberg41c5c4e2012-03-05 20:37:51 -0500833
Kristian Høgsbergbb977002012-01-10 19:11:42 -0500834 widget_get_allocation(dialog->widget, &allocation);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200835 cairo_rectangle(cr, allocation.x, allocation.y,
836 allocation.width, allocation.height);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200837 cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
838 cairo_set_source_rgba(cr, 0, 0, 0, 0.6);
Kristian Høgsberg41c5c4e2012-03-05 20:37:51 -0500839 cairo_fill(cr);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200840
Kristian Høgsberg41c5c4e2012-03-05 20:37:51 -0500841 cairo_translate(cr, allocation.x, allocation.y);
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500842 if (dialog->button_focused)
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200843 f = 1.0;
844 else
845 f = 0.7;
846
847 cx = allocation.width / 2.0;
848 cy = allocation.height / 2.0;
849 r = (cx < cy ? cx : cy) * 0.4;
850 pat = cairo_pattern_create_radial(cx, cy, r * 0.7, cx, cy, r);
851 cairo_pattern_add_color_stop_rgb(pat, 0.0, 0, 0.86 * f, 0);
852 cairo_pattern_add_color_stop_rgb(pat, 0.85, 0.2 * f, f, 0.2 * f);
853 cairo_pattern_add_color_stop_rgb(pat, 1.0, 0, 0.86 * f, 0);
854 cairo_set_source(cr, pat);
Kristian Høgsberg41c5c4e2012-03-05 20:37:51 -0500855 cairo_pattern_destroy(pat);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200856 cairo_arc(cr, cx, cy, r, 0.0, 2.0 * M_PI);
857 cairo_fill(cr);
858
Kristian Høgsbergc51f7992012-01-08 15:09:53 -0500859 widget_set_allocation(dialog->button,
Kristian Høgsberg41c5c4e2012-03-05 20:37:51 -0500860 allocation.x + cx - r,
861 allocation.y + cy - r, 2 * r, 2 * r);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200862
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200863 cairo_destroy(cr);
864
Alexander Larssonc584fa62013-05-22 14:41:32 +0200865 surface = window_get_surface(dialog->window);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200866 cairo_surface_destroy(surface);
867}
868
869static void
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500870unlock_dialog_button_handler(struct widget *widget,
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200871 struct input *input, uint32_t time,
Daniel Stone4dbadb12012-05-30 16:31:51 +0100872 uint32_t button,
873 enum wl_pointer_button_state state, void *data)
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200874{
875 struct unlock_dialog *dialog = data;
876 struct desktop *desktop = dialog->desktop;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200877
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500878 if (button == BTN_LEFT) {
Daniel Stone4dbadb12012-05-30 16:31:51 +0100879 if (state == WL_POINTER_BUTTON_STATE_RELEASED &&
880 !dialog->closing) {
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200881 display_defer(desktop->display, &desktop->unlock_task);
882 dialog->closing = 1;
883 }
884 }
885}
886
887static void
Brian J Lovinc4df4082013-08-26 15:58:22 -0700888unlock_dialog_touch_down_handler(struct widget *widget, struct input *input,
889 uint32_t serial, uint32_t time, int32_t id,
890 float x, float y, void *data)
891{
892 struct unlock_dialog *dialog = data;
893
894 dialog->button_focused = 1;
895 widget_schedule_redraw(widget);
896}
897
898static void
899unlock_dialog_touch_up_handler(struct widget *widget, struct input *input,
900 uint32_t serial, uint32_t time, int32_t id,
901 void *data)
902{
903 struct unlock_dialog *dialog = data;
904 struct desktop *desktop = dialog->desktop;
905
906 dialog->button_focused = 0;
907 widget_schedule_redraw(widget);
908 display_defer(desktop->display, &desktop->unlock_task);
909 dialog->closing = 1;
910}
911
912static void
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200913unlock_dialog_keyboard_focus_handler(struct window *window,
914 struct input *device, void *data)
915{
916 window_schedule_redraw(window);
917}
918
Kristian Høgsbergbb901fa2012-01-09 11:22:32 -0500919static int
Kristian Høgsbergee143232012-01-09 08:42:24 -0500920unlock_dialog_widget_enter_handler(struct widget *widget,
Kristian Høgsbergeae5de72012-04-11 22:42:15 -0400921 struct input *input,
Kristian Høgsberg80680c72012-05-10 12:21:37 -0400922 float x, float y, void *data)
Kristian Høgsbergee143232012-01-09 08:42:24 -0500923{
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500924 struct unlock_dialog *dialog = data;
925
926 dialog->button_focused = 1;
Kristian Høgsbergee143232012-01-09 08:42:24 -0500927 widget_schedule_redraw(widget);
Kristian Høgsbergbb901fa2012-01-09 11:22:32 -0500928
Ander Conselvan de Oliveiradc8c8fc2012-05-25 16:01:41 +0300929 return CURSOR_LEFT_PTR;
Kristian Høgsbergee143232012-01-09 08:42:24 -0500930}
931
932static void
933unlock_dialog_widget_leave_handler(struct widget *widget,
934 struct input *input, void *data)
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200935{
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500936 struct unlock_dialog *dialog = data;
937
938 dialog->button_focused = 0;
Kristian Høgsberg9a13dab2012-01-08 15:18:19 -0500939 widget_schedule_redraw(widget);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200940}
941
942static struct unlock_dialog *
Kristian Høgsberg1ec0c312011-11-15 16:39:55 -0500943unlock_dialog_create(struct desktop *desktop)
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200944{
Kristian Høgsberg1ec0c312011-11-15 16:39:55 -0500945 struct display *display = desktop->display;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200946 struct unlock_dialog *dialog;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +0800947 struct wl_surface *surface;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200948
Peter Huttererf3d62272013-08-08 11:57:05 +1000949 dialog = xzalloc(sizeof *dialog);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200950
Kristian Høgsberg730c94d2012-06-26 21:44:35 -0400951 dialog->window = window_create_custom(display);
Jason Ekstrandee7fefc2013-10-13 19:08:38 -0500952 dialog->widget = window_frame_create(dialog->window, dialog);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200953 window_set_title(dialog->window, "Unlock your desktop");
954
955 window_set_user_data(dialog->window, dialog);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200956 window_set_keyboard_focus_handler(dialog->window,
957 unlock_dialog_keyboard_focus_handler);
Kristian Høgsberg441338c2012-01-10 13:52:34 -0500958 dialog->button = widget_add_widget(dialog->widget, dialog);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500959 widget_set_redraw_handler(dialog->widget,
960 unlock_dialog_redraw_handler);
Kristian Høgsbergee143232012-01-09 08:42:24 -0500961 widget_set_enter_handler(dialog->button,
962 unlock_dialog_widget_enter_handler);
963 widget_set_leave_handler(dialog->button,
964 unlock_dialog_widget_leave_handler);
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500965 widget_set_button_handler(dialog->button,
966 unlock_dialog_button_handler);
Brian J Lovinc4df4082013-08-26 15:58:22 -0700967 widget_set_touch_down_handler(dialog->button,
968 unlock_dialog_touch_down_handler);
969 widget_set_touch_up_handler(dialog->button,
970 unlock_dialog_touch_up_handler);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200971
Jonas Ådahl6d6fb612015-11-17 16:00:33 +0800972 surface = window_get_wl_surface(dialog->window);
973 weston_desktop_shell_set_lock_surface(desktop->shell, surface);
Kristian Høgsberg1ec0c312011-11-15 16:39:55 -0500974
Pekka Paalanen40e49ac2012-01-18 16:51:30 +0200975 window_schedule_resize(dialog->window, 260, 230);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200976
977 return dialog;
978}
979
980static void
981unlock_dialog_destroy(struct unlock_dialog *dialog)
982{
983 window_destroy(dialog->window);
984 free(dialog);
985}
986
987static void
988unlock_dialog_finish(struct task *task, uint32_t events)
989{
990 struct desktop *desktop =
Benjamin Franzked7759712011-11-22 12:38:48 +0100991 container_of(task, struct desktop, unlock_task);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200992
Jonas Ådahl6d6fb612015-11-17 16:00:33 +0800993 weston_desktop_shell_unlock(desktop->shell);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200994 unlock_dialog_destroy(desktop->unlock_dialog);
995 desktop->unlock_dialog = NULL;
996}
997
998static void
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400999desktop_shell_configure(void *data,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001000 struct weston_desktop_shell *desktop_shell,
Kristian Høgsbergeae5de72012-04-11 22:42:15 -04001001 uint32_t edges,
Kristian Høgsberg962342c2012-06-26 16:29:50 -04001002 struct wl_surface *surface,
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001003 int32_t width, int32_t height)
1004{
Kristian Høgsberg962342c2012-06-26 16:29:50 -04001005 struct window *window = wl_surface_get_user_data(surface);
Pekka Paalanen068ae942011-11-28 14:11:15 +02001006 struct surface *s = window_get_user_data(window);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001007
Kristian Høgsbergeae5de72012-04-11 22:42:15 -04001008 s->configure(data, desktop_shell, edges, window, width, height);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001009}
1010
Pekka Paalanen9ef3e012011-11-15 13:34:48 +02001011static void
1012desktop_shell_prepare_lock_surface(void *data,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001013 struct weston_desktop_shell *desktop_shell)
Pekka Paalanen9ef3e012011-11-15 13:34:48 +02001014{
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001015 struct desktop *desktop = data;
1016
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001017 if (!desktop->locking) {
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001018 weston_desktop_shell_unlock(desktop->shell);
Pekka Paalanenfd83b6d2011-12-08 10:06:53 +02001019 return;
1020 }
1021
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001022 if (!desktop->unlock_dialog) {
Kristian Høgsberg1ec0c312011-11-15 16:39:55 -05001023 desktop->unlock_dialog = unlock_dialog_create(desktop);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001024 desktop->unlock_dialog->desktop = desktop;
1025 }
Pekka Paalanen9ef3e012011-11-15 13:34:48 +02001026}
1027
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001028static void
1029desktop_shell_grab_cursor(void *data,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001030 struct weston_desktop_shell *desktop_shell,
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001031 uint32_t cursor)
1032{
1033 struct desktop *desktop = data;
1034
1035 switch (cursor) {
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001036 case WESTON_DESKTOP_SHELL_CURSOR_NONE:
Philipp Brüschweiler16d59d72012-08-24 15:43:55 +02001037 desktop->grab_cursor = CURSOR_BLANK;
1038 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001039 case WESTON_DESKTOP_SHELL_CURSOR_BUSY:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001040 desktop->grab_cursor = CURSOR_WATCH;
1041 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001042 case WESTON_DESKTOP_SHELL_CURSOR_MOVE:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001043 desktop->grab_cursor = CURSOR_DRAGGING;
1044 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001045 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_TOP:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001046 desktop->grab_cursor = CURSOR_TOP;
1047 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001048 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_BOTTOM:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001049 desktop->grab_cursor = CURSOR_BOTTOM;
1050 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001051 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_LEFT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001052 desktop->grab_cursor = CURSOR_LEFT;
1053 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001054 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_RIGHT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001055 desktop->grab_cursor = CURSOR_RIGHT;
1056 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001057 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_TOP_LEFT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001058 desktop->grab_cursor = CURSOR_TOP_LEFT;
1059 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001060 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_TOP_RIGHT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001061 desktop->grab_cursor = CURSOR_TOP_RIGHT;
1062 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001063 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_BOTTOM_LEFT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001064 desktop->grab_cursor = CURSOR_BOTTOM_LEFT;
1065 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001066 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_BOTTOM_RIGHT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001067 desktop->grab_cursor = CURSOR_BOTTOM_RIGHT;
1068 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001069 case WESTON_DESKTOP_SHELL_CURSOR_ARROW:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001070 default:
1071 desktop->grab_cursor = CURSOR_LEFT_PTR;
1072 }
1073}
1074
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001075static const struct weston_desktop_shell_listener listener = {
Pekka Paalanen9ef3e012011-11-15 13:34:48 +02001076 desktop_shell_configure,
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001077 desktop_shell_prepare_lock_surface,
1078 desktop_shell_grab_cursor
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001079};
1080
U. Artie Eoff44874d92012-10-02 21:12:35 -07001081static void
1082background_destroy(struct background *background)
1083{
1084 widget_destroy(background->widget);
1085 window_destroy(background->window);
1086
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001087 free(background->image);
U. Artie Eoff44874d92012-10-02 21:12:35 -07001088 free(background);
1089}
1090
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001091static struct background *
1092background_create(struct desktop *desktop)
1093{
1094 struct background *background;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001095 struct weston_config_section *s;
1096 char *type;
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001097
Peter Huttererf3d62272013-08-08 11:57:05 +10001098 background = xzalloc(sizeof *background);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001099 background->base.configure = background_configure;
Kristian Høgsberg962342c2012-06-26 16:29:50 -04001100 background->window = window_create_custom(desktop->display);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -05001101 background->widget = window_add_widget(background->window, background);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001102 window_set_user_data(background->window, background);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -05001103 widget_set_redraw_handler(background->widget, background_draw);
Arnaud Vracfb754a02014-08-25 20:56:49 +02001104 widget_set_transparent(background->widget, 0);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001105
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001106 s = weston_config_get_section(desktop->config, "shell", NULL, NULL);
1107 weston_config_section_get_string(s, "background-image",
Kristian Høgsberg4c3661f2014-04-21 22:54:37 -07001108 &background->image, NULL);
Bryce Harringtone776f2a2016-07-14 18:28:03 -07001109 weston_config_section_get_color(s, "background-color",
1110 &background->color, 0x00000000);
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001111
1112 weston_config_section_get_string(s, "background-type",
1113 &type, "tile");
U. Artie Eoff3c946772014-01-15 10:59:50 -08001114 if (type == NULL) {
1115 fprintf(stderr, "%s: out of memory\n", program_invocation_short_name);
1116 exit(EXIT_FAILURE);
1117 }
1118
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001119 if (strcmp(type, "scale") == 0) {
1120 background->type = BACKGROUND_SCALE;
1121 } else if (strcmp(type, "scale-crop") == 0) {
1122 background->type = BACKGROUND_SCALE_CROP;
1123 } else if (strcmp(type, "tile") == 0) {
1124 background->type = BACKGROUND_TILE;
1125 } else {
1126 background->type = -1;
1127 fprintf(stderr, "invalid background-type: %s\n",
1128 type);
1129 }
1130
1131 free(type);
1132
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001133 return background;
1134}
1135
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001136static int
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001137grab_surface_enter_handler(struct widget *widget, struct input *input,
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001138 float x, float y, void *data)
1139{
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001140 struct desktop *desktop = data;
1141
1142 return desktop->grab_cursor;
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001143}
1144
1145static void
U. Artie Eoff44874d92012-10-02 21:12:35 -07001146grab_surface_destroy(struct desktop *desktop)
1147{
1148 widget_destroy(desktop->grab_widget);
1149 window_destroy(desktop->grab_window);
1150}
1151
1152static void
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001153grab_surface_create(struct desktop *desktop)
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001154{
1155 struct wl_surface *s;
1156
Ander Conselvan de Oliveira07a91cd2012-07-16 14:15:50 +03001157 desktop->grab_window = window_create_custom(desktop->display);
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001158 window_set_user_data(desktop->grab_window, desktop);
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001159
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001160 s = window_get_wl_surface(desktop->grab_window);
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001161 weston_desktop_shell_set_grab_surface(desktop->shell, s);
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001162
1163 desktop->grab_widget =
1164 window_add_widget(desktop->grab_window, desktop);
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001165 /* We set the allocation to 1x1 at 0,0 so the fake enter event
1166 * at 0,0 will go to this widget. */
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001167 widget_set_allocation(desktop->grab_widget, 0, 0, 1, 1);
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001168
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001169 widget_set_enter_handler(desktop->grab_widget,
1170 grab_surface_enter_handler);
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001171}
1172
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001173static void
U. Artie Eoff44874d92012-10-02 21:12:35 -07001174output_destroy(struct output *output)
1175{
1176 background_destroy(output->background);
Jonny Lambe67118c2014-08-12 15:07:51 +02001177 if (output->panel)
1178 panel_destroy(output->panel);
U. Artie Eoff44874d92012-10-02 21:12:35 -07001179 wl_output_destroy(output->output);
1180 wl_list_remove(&output->link);
1181
1182 free(output);
1183}
1184
1185static void
1186desktop_destroy_outputs(struct desktop *desktop)
1187{
1188 struct output *tmp;
1189 struct output *output;
1190
1191 wl_list_for_each_safe(output, tmp, &desktop->outputs, link)
1192 output_destroy(output);
1193}
1194
1195static void
Alexander Larssonc584fa62013-05-22 14:41:32 +02001196output_handle_geometry(void *data,
1197 struct wl_output *wl_output,
1198 int x, int y,
1199 int physical_width,
1200 int physical_height,
1201 int subpixel,
1202 const char *make,
1203 const char *model,
1204 int transform)
1205{
1206 struct output *output = data;
1207
Jonny Lambe67118c2014-08-12 15:07:51 +02001208 if (output->panel)
1209 window_set_buffer_transform(output->panel->window, transform);
Alexander Larssonc584fa62013-05-22 14:41:32 +02001210 window_set_buffer_transform(output->background->window, transform);
1211}
1212
1213static void
1214output_handle_mode(void *data,
1215 struct wl_output *wl_output,
1216 uint32_t flags,
1217 int width,
1218 int height,
1219 int refresh)
1220{
1221}
1222
1223static void
1224output_handle_done(void *data,
1225 struct wl_output *wl_output)
1226{
1227}
1228
1229static void
1230output_handle_scale(void *data,
1231 struct wl_output *wl_output,
Alexander Larssonedddbd12013-05-24 13:09:43 +02001232 int32_t scale)
Alexander Larssonc584fa62013-05-22 14:41:32 +02001233{
1234 struct output *output = data;
1235
Jonny Lambe67118c2014-08-12 15:07:51 +02001236 if (output->panel)
1237 window_set_buffer_scale(output->panel->window, scale);
Alexander Larssonc584fa62013-05-22 14:41:32 +02001238 window_set_buffer_scale(output->background->window, scale);
1239}
1240
1241static const struct wl_output_listener output_listener = {
1242 output_handle_geometry,
1243 output_handle_mode,
1244 output_handle_done,
1245 output_handle_scale
1246};
1247
1248static void
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001249output_init(struct output *output, struct desktop *desktop)
1250{
1251 struct wl_surface *surface;
1252
Quentin Glidic55d57012016-06-23 18:55:18 +02001253 if (desktop->want_panel) {
Jonny Lambe67118c2014-08-12 15:07:51 +02001254 output->panel = panel_create(desktop);
1255 surface = window_get_wl_surface(output->panel->window);
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001256 weston_desktop_shell_set_panel(desktop->shell,
1257 output->output, surface);
Jonny Lambe67118c2014-08-12 15:07:51 +02001258 }
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001259
1260 output->background = background_create(desktop);
1261 surface = window_get_wl_surface(output->background->window);
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001262 weston_desktop_shell_set_background(desktop->shell,
1263 output->output, surface);
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001264}
1265
1266static void
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001267create_output(struct desktop *desktop, uint32_t id)
1268{
1269 struct output *output;
1270
Bryce Harrington0d1a6222016-02-11 16:42:49 -08001271 output = zalloc(sizeof *output);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001272 if (!output)
1273 return;
1274
Kristian Høgsbergfa80e112012-10-10 21:34:26 -04001275 output->output =
Alexander Larssonc584fa62013-05-22 14:41:32 +02001276 display_bind(desktop->display, id, &wl_output_interface, 2);
Xiong Zhang83d8ee72013-10-23 13:58:35 +08001277 output->server_output_id = id;
Alexander Larssonc584fa62013-05-22 14:41:32 +02001278
1279 wl_output_add_listener(output->output, &output_listener, output);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001280
1281 wl_list_insert(&desktop->outputs, &output->link);
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001282
1283 /* On start up we may process an output global before the shell global
1284 * in which case we can't create the panel and background just yet */
1285 if (desktop->shell)
1286 output_init(output, desktop);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001287}
1288
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001289static void
Kristian Høgsbergfa80e112012-10-10 21:34:26 -04001290global_handler(struct display *display, uint32_t id,
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001291 const char *interface, uint32_t version, void *data)
1292{
1293 struct desktop *desktop = data;
1294
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001295 if (!strcmp(interface, "weston_desktop_shell")) {
Kristian Høgsbergfa80e112012-10-10 21:34:26 -04001296 desktop->shell = display_bind(desktop->display,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001297 id,
1298 &weston_desktop_shell_interface,
1299 1);
1300 weston_desktop_shell_add_listener(desktop->shell,
1301 &listener,
1302 desktop);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001303 } else if (!strcmp(interface, "wl_output")) {
1304 create_output(desktop, id);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001305 }
1306}
1307
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -05001308static void
Xiong Zhang83d8ee72013-10-23 13:58:35 +08001309global_handler_remove(struct display *display, uint32_t id,
1310 const char *interface, uint32_t version, void *data)
1311{
1312 struct desktop *desktop = data;
1313 struct output *output;
1314
1315 if (!strcmp(interface, "wl_output")) {
1316 wl_list_for_each(output, &desktop->outputs, link) {
1317 if (output->server_output_id == id) {
1318 output_destroy(output);
1319 break;
1320 }
1321 }
1322 }
1323}
1324
1325static void
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001326panel_add_launchers(struct panel *panel, struct desktop *desktop)
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -05001327{
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001328 struct weston_config_section *s;
1329 char *icon, *path;
1330 const char *name;
1331 int count;
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -05001332
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001333 count = 0;
1334 s = NULL;
1335 while (weston_config_next_section(desktop->config, &s, &name)) {
1336 if (strcmp(name, "launcher") != 0)
1337 continue;
1338
1339 weston_config_section_get_string(s, "icon", &icon, NULL);
1340 weston_config_section_get_string(s, "path", &path, NULL);
1341
1342 if (icon != NULL && path != NULL) {
1343 panel_add_launcher(panel, icon, path);
Rob Bradford09252d42013-07-26 16:29:45 +01001344 count++;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001345 } else {
1346 fprintf(stderr, "invalid launcher section\n");
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001347 }
1348
1349 free(icon);
1350 free(path);
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -05001351 }
1352
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001353 if (count == 0) {
1354 /* add default launcher */
1355 panel_add_launcher(panel,
Kristian Høgsberg6af8eb92012-01-25 16:57:11 -05001356 DATADIR "/weston/terminal.png",
Rodney Lorrimar99ff01b2012-02-29 17:31:03 +01001357 BINDIR "/weston-terminal");
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001358 }
Kristian Høgsberg6af8eb92012-01-25 16:57:11 -05001359}
1360
Quentin Glidic55d57012016-06-23 18:55:18 +02001361static void
1362parse_panel_position(struct desktop *desktop, struct weston_config_section *s)
1363{
1364 char *position;
1365
Daniel Stone08cf24b2017-01-16 13:18:21 +00001366 desktop->want_panel = 1;
Quentin Glidic55d57012016-06-23 18:55:18 +02001367
Daniel Stone08cf24b2017-01-16 13:18:21 +00001368 weston_config_section_get_string(s, "panel-position", &position, "top");
1369 if (strcmp(position, "top") == 0) {
1370 desktop->panel_position = WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP;
1371 } else if (strcmp(position, "bottom") == 0) {
1372 desktop->panel_position = WESTON_DESKTOP_SHELL_PANEL_POSITION_BOTTOM;
1373 } else if (strcmp(position, "left") == 0) {
1374 desktop->panel_position = WESTON_DESKTOP_SHELL_PANEL_POSITION_LEFT;
1375 } else if (strcmp(position, "right") == 0) {
1376 desktop->panel_position = WESTON_DESKTOP_SHELL_PANEL_POSITION_RIGHT;
1377 } else {
1378 /* 'none' is valid here */
1379 if (strcmp(position, "none") != 0)
1380 fprintf(stderr, "Wrong panel position: %s\n", position);
1381 desktop->want_panel = 0;
Quentin Glidic55d57012016-06-23 18:55:18 +02001382 }
Daniel Stone08cf24b2017-01-16 13:18:21 +00001383 free(position);
Quentin Glidic55d57012016-06-23 18:55:18 +02001384}
1385
Quentin Glidic3e37b342016-06-23 18:55:22 +02001386static void
1387parse_clock_format(struct desktop *desktop, struct weston_config_section *s)
1388{
1389 char *clock_format;
1390
1391 weston_config_section_get_string(s, "clock-format", &clock_format, "");
1392 if (strcmp(clock_format, "minutes") == 0)
1393 desktop->clock_format = CLOCK_FORMAT_MINUTES;
1394 else if (strcmp(clock_format, "seconds") == 0)
1395 desktop->clock_format = CLOCK_FORMAT_SECONDS;
1396 else if (strcmp(clock_format, "none") == 0)
1397 desktop->clock_format = CLOCK_FORMAT_NONE;
1398 else
1399 desktop->clock_format = DEFAULT_CLOCK_FORMAT;
1400 free(clock_format);
1401}
1402
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001403int main(int argc, char *argv[])
1404{
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001405 struct desktop desktop = { 0 };
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001406 struct output *output;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001407 struct weston_config_section *s;
Pekka Paalanen6c71aae2015-03-24 15:56:19 +02001408 const char *config_file;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001409
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001410 desktop.unlock_task.run = unlock_dialog_finish;
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001411 wl_list_init(&desktop.outputs);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001412
Pekka Paalanen6c71aae2015-03-24 15:56:19 +02001413 config_file = weston_config_get_name_from_env();
1414 desktop.config = weston_config_parse(config_file);
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001415 s = weston_config_get_section(desktop.config, "shell", NULL, NULL);
1416 weston_config_section_get_bool(s, "locking", &desktop.locking, 1);
Quentin Glidic55d57012016-06-23 18:55:18 +02001417 parse_panel_position(&desktop, s);
Quentin Glidic3e37b342016-06-23 18:55:22 +02001418 parse_clock_format(&desktop, s);
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001419
Kristian Høgsberg4172f662013-02-20 15:27:49 -05001420 desktop.display = display_create(&argc, argv);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001421 if (desktop.display == NULL) {
1422 fprintf(stderr, "failed to create display: %m\n");
1423 return -1;
1424 }
1425
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001426 display_set_user_data(desktop.display, &desktop);
Kristian Høgsbergfa80e112012-10-10 21:34:26 -04001427 display_set_global_handler(desktop.display, global_handler);
Xiong Zhang83d8ee72013-10-23 13:58:35 +08001428 display_set_global_handler_remove(desktop.display, global_handler_remove);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001429
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001430 /* Create panel and background for outputs processed before the shell
1431 * global interface was processed */
Quentin Glidic55d57012016-06-23 18:55:18 +02001432 if (desktop.want_panel)
1433 weston_desktop_shell_set_panel_position(desktop.shell, desktop.panel_position);
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001434 wl_list_for_each(output, &desktop.outputs, link)
1435 if (!output->panel)
1436 output_init(output, &desktop);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001437
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001438 grab_surface_create(&desktop);
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001439
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001440 signal(SIGCHLD, sigchild_handler);
1441
1442 display_run(desktop.display);
1443
U. Artie Eoff44874d92012-10-02 21:12:35 -07001444 /* Cleanup */
1445 grab_surface_destroy(&desktop);
1446 desktop_destroy_outputs(&desktop);
1447 if (desktop.unlock_dialog)
1448 unlock_dialog_destroy(desktop.unlock_dialog);
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001449 weston_desktop_shell_destroy(desktop.shell);
U. Artie Eoff44874d92012-10-02 21:12:35 -07001450 display_destroy(desktop.display);
1451
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001452 return 0;
1453}