blob: 2667e9bb5e2758c5fc4df2741081ff4f71361c01 [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;
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100594 char *clock_format_option = NULL;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400595
Peter Huttererf3d62272013-08-08 11:57:05 +1000596 panel = xzalloc(sizeof *panel);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400597
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100598 panel->base.configure = panel_configure;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300599 panel->window = window_create_custom(desktop->display);
Kristian Høgsberg75bc6672012-01-10 09:43:58 -0500600 panel->widget = window_add_widget(panel->window, panel);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500601 wl_list_init(&panel->launcher_list);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400602
603 window_set_title(panel->window, "panel");
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400604 window_set_user_data(panel->window, panel);
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500605
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500606 widget_set_redraw_handler(panel->widget, panel_redraw_handler);
607 widget_set_resize_handler(panel->widget, panel_resize_handler);
Michael Vetter2a18a522015-05-15 17:17:47 +0200608
Quentin Glidice7ed60f2016-06-23 18:55:24 +0200609 panel->panel_position = desktop->panel_position;
Quentin Glidic3e37b342016-06-23 18:55:22 +0200610 panel->clock_format = desktop->clock_format;
Armin Krezovićc6a55db2016-03-10 18:02:34 +0100611 if (panel->clock_format != CLOCK_FORMAT_NONE)
612 panel_add_clock(panel);
613
614 free (clock_format_option);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400615
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300616 s = weston_config_get_section(desktop->config, "shell", NULL, NULL);
Bryce Harringtone776f2a2016-07-14 18:28:03 -0700617 weston_config_section_get_color(s, "panel-color",
618 &panel->color, 0xaa000000);
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300619
620 panel_add_launchers(panel, desktop);
621
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400622 return panel;
623}
624
Philipp Brüschweiler467668c2012-08-29 10:53:36 +0200625static cairo_surface_t *
626load_icon_or_fallback(const char *icon)
627{
628 cairo_surface_t *surface = cairo_image_surface_create_from_png(icon);
Philipp Brüschweiler96386b82013-04-15 20:10:40 +0200629 cairo_status_t status;
Philipp Brüschweiler467668c2012-08-29 10:53:36 +0200630 cairo_t *cr;
631
Philipp Brüschweiler96386b82013-04-15 20:10:40 +0200632 status = cairo_surface_status(surface);
633 if (status == CAIRO_STATUS_SUCCESS)
Philipp Brüschweiler467668c2012-08-29 10:53:36 +0200634 return surface;
635
636 cairo_surface_destroy(surface);
Philipp Brüschweiler96386b82013-04-15 20:10:40 +0200637 fprintf(stderr, "ERROR loading icon from file '%s', error: '%s'\n",
638 icon, cairo_status_to_string(status));
Philipp Brüschweiler467668c2012-08-29 10:53:36 +0200639
640 /* draw fallback icon */
641 surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32,
642 20, 20);
643 cr = cairo_create(surface);
644
645 cairo_set_source_rgba(cr, 0.8, 0.8, 0.8, 1);
646 cairo_paint(cr);
647
648 cairo_set_source_rgba(cr, 0, 0, 0, 1);
649 cairo_set_line_cap(cr, CAIRO_LINE_CAP_ROUND);
650 cairo_rectangle(cr, 0, 0, 20, 20);
651 cairo_move_to(cr, 4, 4);
652 cairo_line_to(cr, 16, 16);
653 cairo_move_to(cr, 4, 16);
654 cairo_line_to(cr, 16, 4);
655 cairo_stroke(cr);
656
657 cairo_destroy(cr);
658
659 return surface;
660}
661
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400662static void
Kristian Høgsberg53880802012-01-09 11:16:50 -0500663panel_add_launcher(struct panel *panel, const char *icon, const char *path)
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400664{
Kristian Høgsberg53880802012-01-09 11:16:50 -0500665 struct panel_launcher *launcher;
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400666 char *start, *p, *eq, **ps;
667 int i, j, k;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400668
Peter Huttererf3d62272013-08-08 11:57:05 +1000669 launcher = xzalloc(sizeof *launcher);
Philipp Brüschweiler467668c2012-08-29 10:53:36 +0200670 launcher->icon = load_icon_or_fallback(icon);
U. Artie Eoff3c946772014-01-15 10:59:50 -0800671 launcher->path = xstrdup(path);
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400672
673 wl_array_init(&launcher->envp);
674 wl_array_init(&launcher->argv);
Pekka Paalanenb6df4f72012-08-03 14:39:15 +0300675 for (i = 0; environ[i]; i++) {
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400676 ps = wl_array_add(&launcher->envp, sizeof *ps);
Pekka Paalanenb6df4f72012-08-03 14:39:15 +0300677 *ps = environ[i];
Kristian Høgsbergd1936b92012-07-23 22:59:33 -0400678 }
679 j = 0;
680
681 start = launcher->path;
682 while (*start) {
683 for (p = start, eq = NULL; *p && !isspace(*p); p++)
684 if (*p == '=')
685 eq = p;
686
687 if (eq && j == 0) {
688 ps = launcher->envp.data;
689 for (k = 0; k < i; k++)
690 if (strncmp(ps[k], start, eq - start) == 0) {
691 ps[k] = start;
692 break;
693 }
694 if (k == i) {
695 ps = wl_array_add(&launcher->envp, sizeof *ps);
696 *ps = start;
697 i++;
698 }
699 } else {
700 ps = wl_array_add(&launcher->argv, sizeof *ps);
701 *ps = start;
702 j++;
703 }
704
705 while (*p && isspace(*p))
706 *p++ = '\0';
707
708 start = p;
709 }
710
711 ps = wl_array_add(&launcher->envp, sizeof *ps);
712 *ps = NULL;
713 ps = wl_array_add(&launcher->argv, sizeof *ps);
714 *ps = NULL;
715
Kristian Høgsberg53880802012-01-09 11:16:50 -0500716 launcher->panel = panel;
Kristian Høgsberg75bc6672012-01-10 09:43:58 -0500717 wl_list_insert(panel->launcher_list.prev, &launcher->link);
718
Kristian Høgsberg441338c2012-01-10 13:52:34 -0500719 launcher->widget = widget_add_widget(panel->widget, launcher);
Kristian Høgsberg53880802012-01-09 11:16:50 -0500720 widget_set_enter_handler(launcher->widget,
721 panel_launcher_enter_handler);
722 widget_set_leave_handler(launcher->widget,
723 panel_launcher_leave_handler);
724 widget_set_button_handler(launcher->widget,
725 panel_launcher_button_handler);
Rusty Lynch4384a242013-08-08 21:28:22 -0700726 widget_set_touch_down_handler(launcher->widget,
727 panel_launcher_touch_down_handler);
728 widget_set_touch_up_handler(launcher->widget,
729 panel_launcher_touch_up_handler);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500730 widget_set_redraw_handler(launcher->widget,
731 panel_launcher_redraw_handler);
Tiago Vignatti61500722012-05-23 22:06:28 +0300732 widget_set_motion_handler(launcher->widget,
733 panel_launcher_motion_handler);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400734}
735
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500736enum {
737 BACKGROUND_SCALE,
Pekka Paalanena402b052013-05-22 18:03:10 +0300738 BACKGROUND_SCALE_CROP,
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500739 BACKGROUND_TILE
740};
741
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400742static void
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500743background_draw(struct widget *widget, void *data)
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400744{
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500745 struct background *background = data;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400746 cairo_surface_t *surface, *image;
Kristian Høgsberg7e690002011-09-08 18:18:02 -0400747 cairo_pattern_t *pattern;
748 cairo_matrix_t matrix;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400749 cairo_t *cr;
Pekka Paalanena402b052013-05-22 18:03:10 +0300750 double im_w, im_h;
751 double sx, sy, s;
752 double tx, ty;
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500753 struct rectangle allocation;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400754
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500755 surface = window_get_surface(background->window);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400756
Alexander Larssonc584fa62013-05-22 14:41:32 +0200757 cr = widget_cairo_create(background->widget);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400758 cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
759 cairo_set_source_rgba(cr, 0.0, 0.0, 0.2, 1.0);
760 cairo_paint(cr);
761
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500762 widget_get_allocation(widget, &allocation);
Kristian Høgsberg8129bc02012-01-25 14:55:33 -0500763 image = NULL;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300764 if (background->image)
765 image = load_cairo_surface(background->image);
Kristian Høgsberg4c3661f2014-04-21 22:54:37 -0700766 else if (background->color == 0)
767 image = load_cairo_surface(DATADIR "/weston/pattern.png");
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500768
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300769 if (image && background->type != -1) {
Pekka Paalanena402b052013-05-22 18:03:10 +0300770 im_w = cairo_image_surface_get_width(image);
771 im_h = cairo_image_surface_get_height(image);
772 sx = im_w / allocation.width;
773 sy = im_h / allocation.height;
774
Kristian Høgsberg7e690002011-09-08 18:18:02 -0400775 pattern = cairo_pattern_create_for_surface(image);
Pekka Paalanena402b052013-05-22 18:03:10 +0300776
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300777 switch (background->type) {
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500778 case BACKGROUND_SCALE:
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500779 cairo_matrix_init_scale(&matrix, sx, sy);
780 cairo_pattern_set_matrix(pattern, &matrix);
Bill Spitzak79b7cb32014-05-08 20:00:35 -0700781 cairo_pattern_set_extend(pattern, CAIRO_EXTEND_PAD);
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500782 break;
Pekka Paalanena402b052013-05-22 18:03:10 +0300783 case BACKGROUND_SCALE_CROP:
784 s = (sx < sy) ? sx : sy;
785 /* align center */
786 tx = (im_w - s * allocation.width) * 0.5;
787 ty = (im_h - s * allocation.height) * 0.5;
788 cairo_matrix_init_translate(&matrix, tx, ty);
789 cairo_matrix_scale(&matrix, s, s);
790 cairo_pattern_set_matrix(pattern, &matrix);
Bill Spitzak79b7cb32014-05-08 20:00:35 -0700791 cairo_pattern_set_extend(pattern, CAIRO_EXTEND_PAD);
Pekka Paalanena402b052013-05-22 18:03:10 +0300792 break;
Kristian Høgsberg07f72942012-01-25 16:34:36 -0500793 case BACKGROUND_TILE:
794 cairo_pattern_set_extend(pattern, CAIRO_EXTEND_REPEAT);
795 break;
796 }
Pekka Paalanena402b052013-05-22 18:03:10 +0300797
Kristian Høgsberg7e690002011-09-08 18:18:02 -0400798 cairo_set_source(cr, pattern);
799 cairo_pattern_destroy (pattern);
Kristian Høgsberg27d38662011-10-20 13:11:12 -0400800 cairo_surface_destroy(image);
Kristian Høgsberg8129bc02012-01-25 14:55:33 -0500801 } else {
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +0300802 set_hex_color(cr, background->color);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400803 }
804
Kristian Høgsberg8129bc02012-01-25 14:55:33 -0500805 cairo_paint(cr);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400806 cairo_destroy(cr);
807 cairo_surface_destroy(surface);
Pekka Paalanen9564c752012-10-24 09:43:08 +0300808
Pekka Paalanen79346ab2013-05-22 18:03:09 +0300809 background->painted = 1;
810 check_desktop_ready(background->window);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -0400811}
812
813static void
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100814background_configure(void *data,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +0800815 struct weston_desktop_shell *desktop_shell,
Kristian Høgsbergeae5de72012-04-11 22:42:15 -0400816 uint32_t edges, struct window *window,
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100817 int32_t width, int32_t height)
818{
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500819 struct background *background =
820 (struct background *) window_get_user_data(window);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100821
Kristian Høgsberg4598f962014-01-01 16:32:09 -0800822 widget_schedule_resize(background->widget, width, height);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +0100823}
824
825static void
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500826unlock_dialog_redraw_handler(struct widget *widget, void *data)
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200827{
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500828 struct unlock_dialog *dialog = data;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200829 struct rectangle allocation;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200830 cairo_surface_t *surface;
Alexander Larssonc584fa62013-05-22 14:41:32 +0200831 cairo_t *cr;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200832 cairo_pattern_t *pat;
833 double cx, cy, r, f;
834
Alexander Larssonc584fa62013-05-22 14:41:32 +0200835 cr = widget_cairo_create(widget);
Kristian Høgsberg41c5c4e2012-03-05 20:37:51 -0500836
Kristian Høgsbergbb977002012-01-10 19:11:42 -0500837 widget_get_allocation(dialog->widget, &allocation);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200838 cairo_rectangle(cr, allocation.x, allocation.y,
839 allocation.width, allocation.height);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200840 cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
841 cairo_set_source_rgba(cr, 0, 0, 0, 0.6);
Kristian Høgsberg41c5c4e2012-03-05 20:37:51 -0500842 cairo_fill(cr);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200843
Kristian Høgsberg41c5c4e2012-03-05 20:37:51 -0500844 cairo_translate(cr, allocation.x, allocation.y);
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500845 if (dialog->button_focused)
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200846 f = 1.0;
847 else
848 f = 0.7;
849
850 cx = allocation.width / 2.0;
851 cy = allocation.height / 2.0;
852 r = (cx < cy ? cx : cy) * 0.4;
853 pat = cairo_pattern_create_radial(cx, cy, r * 0.7, cx, cy, r);
854 cairo_pattern_add_color_stop_rgb(pat, 0.0, 0, 0.86 * f, 0);
855 cairo_pattern_add_color_stop_rgb(pat, 0.85, 0.2 * f, f, 0.2 * f);
856 cairo_pattern_add_color_stop_rgb(pat, 1.0, 0, 0.86 * f, 0);
857 cairo_set_source(cr, pat);
Kristian Høgsberg41c5c4e2012-03-05 20:37:51 -0500858 cairo_pattern_destroy(pat);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200859 cairo_arc(cr, cx, cy, r, 0.0, 2.0 * M_PI);
860 cairo_fill(cr);
861
Kristian Høgsbergc51f7992012-01-08 15:09:53 -0500862 widget_set_allocation(dialog->button,
Kristian Høgsberg41c5c4e2012-03-05 20:37:51 -0500863 allocation.x + cx - r,
864 allocation.y + cy - r, 2 * r, 2 * r);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200865
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200866 cairo_destroy(cr);
867
Alexander Larssonc584fa62013-05-22 14:41:32 +0200868 surface = window_get_surface(dialog->window);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200869 cairo_surface_destroy(surface);
870}
871
872static void
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500873unlock_dialog_button_handler(struct widget *widget,
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200874 struct input *input, uint32_t time,
Daniel Stone4dbadb12012-05-30 16:31:51 +0100875 uint32_t button,
876 enum wl_pointer_button_state state, void *data)
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200877{
878 struct unlock_dialog *dialog = data;
879 struct desktop *desktop = dialog->desktop;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200880
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500881 if (button == BTN_LEFT) {
Daniel Stone4dbadb12012-05-30 16:31:51 +0100882 if (state == WL_POINTER_BUTTON_STATE_RELEASED &&
883 !dialog->closing) {
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200884 display_defer(desktop->display, &desktop->unlock_task);
885 dialog->closing = 1;
886 }
887 }
888}
889
890static void
Brian J Lovinc4df4082013-08-26 15:58:22 -0700891unlock_dialog_touch_down_handler(struct widget *widget, struct input *input,
892 uint32_t serial, uint32_t time, int32_t id,
893 float x, float y, void *data)
894{
895 struct unlock_dialog *dialog = data;
896
897 dialog->button_focused = 1;
898 widget_schedule_redraw(widget);
899}
900
901static void
902unlock_dialog_touch_up_handler(struct widget *widget, struct input *input,
903 uint32_t serial, uint32_t time, int32_t id,
904 void *data)
905{
906 struct unlock_dialog *dialog = data;
907 struct desktop *desktop = dialog->desktop;
908
909 dialog->button_focused = 0;
910 widget_schedule_redraw(widget);
911 display_defer(desktop->display, &desktop->unlock_task);
912 dialog->closing = 1;
913}
914
915static void
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200916unlock_dialog_keyboard_focus_handler(struct window *window,
917 struct input *device, void *data)
918{
919 window_schedule_redraw(window);
920}
921
Kristian Høgsbergbb901fa2012-01-09 11:22:32 -0500922static int
Kristian Høgsbergee143232012-01-09 08:42:24 -0500923unlock_dialog_widget_enter_handler(struct widget *widget,
Kristian Høgsbergeae5de72012-04-11 22:42:15 -0400924 struct input *input,
Kristian Høgsberg80680c72012-05-10 12:21:37 -0400925 float x, float y, void *data)
Kristian Høgsbergee143232012-01-09 08:42:24 -0500926{
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500927 struct unlock_dialog *dialog = data;
928
929 dialog->button_focused = 1;
Kristian Høgsbergee143232012-01-09 08:42:24 -0500930 widget_schedule_redraw(widget);
Kristian Høgsbergbb901fa2012-01-09 11:22:32 -0500931
Ander Conselvan de Oliveiradc8c8fc2012-05-25 16:01:41 +0300932 return CURSOR_LEFT_PTR;
Kristian Høgsbergee143232012-01-09 08:42:24 -0500933}
934
935static void
936unlock_dialog_widget_leave_handler(struct widget *widget,
937 struct input *input, void *data)
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200938{
Kristian Høgsbergb6323512012-01-11 00:04:42 -0500939 struct unlock_dialog *dialog = data;
940
941 dialog->button_focused = 0;
Kristian Høgsberg9a13dab2012-01-08 15:18:19 -0500942 widget_schedule_redraw(widget);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200943}
944
945static struct unlock_dialog *
Kristian Høgsberg1ec0c312011-11-15 16:39:55 -0500946unlock_dialog_create(struct desktop *desktop)
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200947{
Kristian Høgsberg1ec0c312011-11-15 16:39:55 -0500948 struct display *display = desktop->display;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200949 struct unlock_dialog *dialog;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +0800950 struct wl_surface *surface;
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200951
Peter Huttererf3d62272013-08-08 11:57:05 +1000952 dialog = xzalloc(sizeof *dialog);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200953
Kristian Høgsberg730c94d2012-06-26 21:44:35 -0400954 dialog->window = window_create_custom(display);
Jason Ekstrandee7fefc2013-10-13 19:08:38 -0500955 dialog->widget = window_frame_create(dialog->window, dialog);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200956 window_set_title(dialog->window, "Unlock your desktop");
957
958 window_set_user_data(dialog->window, dialog);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200959 window_set_keyboard_focus_handler(dialog->window,
960 unlock_dialog_keyboard_focus_handler);
Kristian Høgsberg441338c2012-01-10 13:52:34 -0500961 dialog->button = widget_add_widget(dialog->widget, dialog);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -0500962 widget_set_redraw_handler(dialog->widget,
963 unlock_dialog_redraw_handler);
Kristian Høgsbergee143232012-01-09 08:42:24 -0500964 widget_set_enter_handler(dialog->button,
965 unlock_dialog_widget_enter_handler);
966 widget_set_leave_handler(dialog->button,
967 unlock_dialog_widget_leave_handler);
Kristian Høgsberga8a0db32012-01-09 11:12:05 -0500968 widget_set_button_handler(dialog->button,
969 unlock_dialog_button_handler);
Brian J Lovinc4df4082013-08-26 15:58:22 -0700970 widget_set_touch_down_handler(dialog->button,
971 unlock_dialog_touch_down_handler);
972 widget_set_touch_up_handler(dialog->button,
973 unlock_dialog_touch_up_handler);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200974
Jonas Ådahl6d6fb612015-11-17 16:00:33 +0800975 surface = window_get_wl_surface(dialog->window);
976 weston_desktop_shell_set_lock_surface(desktop->shell, surface);
Kristian Høgsberg1ec0c312011-11-15 16:39:55 -0500977
Pekka Paalanen40e49ac2012-01-18 16:51:30 +0200978 window_schedule_resize(dialog->window, 260, 230);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200979
980 return dialog;
981}
982
983static void
984unlock_dialog_destroy(struct unlock_dialog *dialog)
985{
986 window_destroy(dialog->window);
987 free(dialog);
988}
989
990static void
991unlock_dialog_finish(struct task *task, uint32_t events)
992{
993 struct desktop *desktop =
Benjamin Franzked7759712011-11-22 12:38:48 +0100994 container_of(task, struct desktop, unlock_task);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200995
Jonas Ådahl6d6fb612015-11-17 16:00:33 +0800996 weston_desktop_shell_unlock(desktop->shell);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +0200997 unlock_dialog_destroy(desktop->unlock_dialog);
998 desktop->unlock_dialog = NULL;
999}
1000
1001static void
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001002desktop_shell_configure(void *data,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001003 struct weston_desktop_shell *desktop_shell,
Kristian Høgsbergeae5de72012-04-11 22:42:15 -04001004 uint32_t edges,
Kristian Høgsberg962342c2012-06-26 16:29:50 -04001005 struct wl_surface *surface,
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001006 int32_t width, int32_t height)
1007{
Kristian Høgsberg962342c2012-06-26 16:29:50 -04001008 struct window *window = wl_surface_get_user_data(surface);
Pekka Paalanen068ae942011-11-28 14:11:15 +02001009 struct surface *s = window_get_user_data(window);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001010
Kristian Høgsbergeae5de72012-04-11 22:42:15 -04001011 s->configure(data, desktop_shell, edges, window, width, height);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001012}
1013
Pekka Paalanen9ef3e012011-11-15 13:34:48 +02001014static void
1015desktop_shell_prepare_lock_surface(void *data,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001016 struct weston_desktop_shell *desktop_shell)
Pekka Paalanen9ef3e012011-11-15 13:34:48 +02001017{
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001018 struct desktop *desktop = data;
1019
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001020 if (!desktop->locking) {
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001021 weston_desktop_shell_unlock(desktop->shell);
Pekka Paalanenfd83b6d2011-12-08 10:06:53 +02001022 return;
1023 }
1024
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001025 if (!desktop->unlock_dialog) {
Kristian Høgsberg1ec0c312011-11-15 16:39:55 -05001026 desktop->unlock_dialog = unlock_dialog_create(desktop);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001027 desktop->unlock_dialog->desktop = desktop;
1028 }
Pekka Paalanen9ef3e012011-11-15 13:34:48 +02001029}
1030
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001031static void
1032desktop_shell_grab_cursor(void *data,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001033 struct weston_desktop_shell *desktop_shell,
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001034 uint32_t cursor)
1035{
1036 struct desktop *desktop = data;
1037
1038 switch (cursor) {
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001039 case WESTON_DESKTOP_SHELL_CURSOR_NONE:
Philipp Brüschweiler16d59d72012-08-24 15:43:55 +02001040 desktop->grab_cursor = CURSOR_BLANK;
1041 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001042 case WESTON_DESKTOP_SHELL_CURSOR_BUSY:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001043 desktop->grab_cursor = CURSOR_WATCH;
1044 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001045 case WESTON_DESKTOP_SHELL_CURSOR_MOVE:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001046 desktop->grab_cursor = CURSOR_DRAGGING;
1047 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001048 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_TOP:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001049 desktop->grab_cursor = CURSOR_TOP;
1050 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001051 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_BOTTOM:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001052 desktop->grab_cursor = CURSOR_BOTTOM;
1053 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001054 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_LEFT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001055 desktop->grab_cursor = CURSOR_LEFT;
1056 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001057 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_RIGHT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001058 desktop->grab_cursor = CURSOR_RIGHT;
1059 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001060 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_TOP_LEFT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001061 desktop->grab_cursor = CURSOR_TOP_LEFT;
1062 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001063 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_TOP_RIGHT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001064 desktop->grab_cursor = CURSOR_TOP_RIGHT;
1065 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001066 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_BOTTOM_LEFT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001067 desktop->grab_cursor = CURSOR_BOTTOM_LEFT;
1068 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001069 case WESTON_DESKTOP_SHELL_CURSOR_RESIZE_BOTTOM_RIGHT:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001070 desktop->grab_cursor = CURSOR_BOTTOM_RIGHT;
1071 break;
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001072 case WESTON_DESKTOP_SHELL_CURSOR_ARROW:
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001073 default:
1074 desktop->grab_cursor = CURSOR_LEFT_PTR;
1075 }
1076}
1077
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001078static const struct weston_desktop_shell_listener listener = {
Pekka Paalanen9ef3e012011-11-15 13:34:48 +02001079 desktop_shell_configure,
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001080 desktop_shell_prepare_lock_surface,
1081 desktop_shell_grab_cursor
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001082};
1083
U. Artie Eoff44874d92012-10-02 21:12:35 -07001084static void
1085background_destroy(struct background *background)
1086{
1087 widget_destroy(background->widget);
1088 window_destroy(background->window);
1089
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001090 free(background->image);
U. Artie Eoff44874d92012-10-02 21:12:35 -07001091 free(background);
1092}
1093
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001094static struct background *
1095background_create(struct desktop *desktop)
1096{
1097 struct background *background;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001098 struct weston_config_section *s;
1099 char *type;
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001100
Peter Huttererf3d62272013-08-08 11:57:05 +10001101 background = xzalloc(sizeof *background);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001102 background->base.configure = background_configure;
Kristian Høgsberg962342c2012-06-26 16:29:50 -04001103 background->window = window_create_custom(desktop->display);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -05001104 background->widget = window_add_widget(background->window, background);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001105 window_set_user_data(background->window, background);
Kristian Høgsbergb67e94b2012-01-10 12:23:19 -05001106 widget_set_redraw_handler(background->widget, background_draw);
Arnaud Vracfb754a02014-08-25 20:56:49 +02001107 widget_set_transparent(background->widget, 0);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001108
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001109 s = weston_config_get_section(desktop->config, "shell", NULL, NULL);
1110 weston_config_section_get_string(s, "background-image",
Kristian Høgsberg4c3661f2014-04-21 22:54:37 -07001111 &background->image, NULL);
Bryce Harringtone776f2a2016-07-14 18:28:03 -07001112 weston_config_section_get_color(s, "background-color",
1113 &background->color, 0x00000000);
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001114
1115 weston_config_section_get_string(s, "background-type",
1116 &type, "tile");
U. Artie Eoff3c946772014-01-15 10:59:50 -08001117 if (type == NULL) {
1118 fprintf(stderr, "%s: out of memory\n", program_invocation_short_name);
1119 exit(EXIT_FAILURE);
1120 }
1121
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001122 if (strcmp(type, "scale") == 0) {
1123 background->type = BACKGROUND_SCALE;
1124 } else if (strcmp(type, "scale-crop") == 0) {
1125 background->type = BACKGROUND_SCALE_CROP;
1126 } else if (strcmp(type, "tile") == 0) {
1127 background->type = BACKGROUND_TILE;
1128 } else {
1129 background->type = -1;
1130 fprintf(stderr, "invalid background-type: %s\n",
1131 type);
1132 }
1133
1134 free(type);
1135
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001136 return background;
1137}
1138
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001139static int
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001140grab_surface_enter_handler(struct widget *widget, struct input *input,
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001141 float x, float y, void *data)
1142{
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001143 struct desktop *desktop = data;
1144
1145 return desktop->grab_cursor;
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001146}
1147
1148static void
U. Artie Eoff44874d92012-10-02 21:12:35 -07001149grab_surface_destroy(struct desktop *desktop)
1150{
1151 widget_destroy(desktop->grab_widget);
1152 window_destroy(desktop->grab_window);
1153}
1154
1155static void
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001156grab_surface_create(struct desktop *desktop)
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001157{
1158 struct wl_surface *s;
1159
Ander Conselvan de Oliveira07a91cd2012-07-16 14:15:50 +03001160 desktop->grab_window = window_create_custom(desktop->display);
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001161 window_set_user_data(desktop->grab_window, desktop);
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001162
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001163 s = window_get_wl_surface(desktop->grab_window);
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001164 weston_desktop_shell_set_grab_surface(desktop->shell, s);
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001165
1166 desktop->grab_widget =
1167 window_add_widget(desktop->grab_window, desktop);
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001168 /* We set the allocation to 1x1 at 0,0 so the fake enter event
1169 * at 0,0 will go to this widget. */
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001170 widget_set_allocation(desktop->grab_widget, 0, 0, 1, 1);
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001171
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001172 widget_set_enter_handler(desktop->grab_widget,
1173 grab_surface_enter_handler);
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001174}
1175
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001176static void
U. Artie Eoff44874d92012-10-02 21:12:35 -07001177output_destroy(struct output *output)
1178{
1179 background_destroy(output->background);
Jonny Lambe67118c2014-08-12 15:07:51 +02001180 if (output->panel)
1181 panel_destroy(output->panel);
U. Artie Eoff44874d92012-10-02 21:12:35 -07001182 wl_output_destroy(output->output);
1183 wl_list_remove(&output->link);
1184
1185 free(output);
1186}
1187
1188static void
1189desktop_destroy_outputs(struct desktop *desktop)
1190{
1191 struct output *tmp;
1192 struct output *output;
1193
1194 wl_list_for_each_safe(output, tmp, &desktop->outputs, link)
1195 output_destroy(output);
1196}
1197
1198static void
Alexander Larssonc584fa62013-05-22 14:41:32 +02001199output_handle_geometry(void *data,
1200 struct wl_output *wl_output,
1201 int x, int y,
1202 int physical_width,
1203 int physical_height,
1204 int subpixel,
1205 const char *make,
1206 const char *model,
1207 int transform)
1208{
1209 struct output *output = data;
1210
Jonny Lambe67118c2014-08-12 15:07:51 +02001211 if (output->panel)
1212 window_set_buffer_transform(output->panel->window, transform);
Alexander Larssonc584fa62013-05-22 14:41:32 +02001213 window_set_buffer_transform(output->background->window, transform);
1214}
1215
1216static void
1217output_handle_mode(void *data,
1218 struct wl_output *wl_output,
1219 uint32_t flags,
1220 int width,
1221 int height,
1222 int refresh)
1223{
1224}
1225
1226static void
1227output_handle_done(void *data,
1228 struct wl_output *wl_output)
1229{
1230}
1231
1232static void
1233output_handle_scale(void *data,
1234 struct wl_output *wl_output,
Alexander Larssonedddbd12013-05-24 13:09:43 +02001235 int32_t scale)
Alexander Larssonc584fa62013-05-22 14:41:32 +02001236{
1237 struct output *output = data;
1238
Jonny Lambe67118c2014-08-12 15:07:51 +02001239 if (output->panel)
1240 window_set_buffer_scale(output->panel->window, scale);
Alexander Larssonc584fa62013-05-22 14:41:32 +02001241 window_set_buffer_scale(output->background->window, scale);
1242}
1243
1244static const struct wl_output_listener output_listener = {
1245 output_handle_geometry,
1246 output_handle_mode,
1247 output_handle_done,
1248 output_handle_scale
1249};
1250
1251static void
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001252output_init(struct output *output, struct desktop *desktop)
1253{
1254 struct wl_surface *surface;
1255
Quentin Glidic55d57012016-06-23 18:55:18 +02001256 if (desktop->want_panel) {
Jonny Lambe67118c2014-08-12 15:07:51 +02001257 output->panel = panel_create(desktop);
1258 surface = window_get_wl_surface(output->panel->window);
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001259 weston_desktop_shell_set_panel(desktop->shell,
1260 output->output, surface);
Jonny Lambe67118c2014-08-12 15:07:51 +02001261 }
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001262
1263 output->background = background_create(desktop);
1264 surface = window_get_wl_surface(output->background->window);
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001265 weston_desktop_shell_set_background(desktop->shell,
1266 output->output, surface);
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001267}
1268
1269static void
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001270create_output(struct desktop *desktop, uint32_t id)
1271{
1272 struct output *output;
1273
Bryce Harrington0d1a6222016-02-11 16:42:49 -08001274 output = zalloc(sizeof *output);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001275 if (!output)
1276 return;
1277
Kristian Høgsbergfa80e112012-10-10 21:34:26 -04001278 output->output =
Alexander Larssonc584fa62013-05-22 14:41:32 +02001279 display_bind(desktop->display, id, &wl_output_interface, 2);
Xiong Zhang83d8ee72013-10-23 13:58:35 +08001280 output->server_output_id = id;
Alexander Larssonc584fa62013-05-22 14:41:32 +02001281
1282 wl_output_add_listener(output->output, &output_listener, output);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001283
1284 wl_list_insert(&desktop->outputs, &output->link);
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001285
1286 /* On start up we may process an output global before the shell global
1287 * in which case we can't create the panel and background just yet */
1288 if (desktop->shell)
1289 output_init(output, desktop);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001290}
1291
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001292static void
Kristian Høgsbergfa80e112012-10-10 21:34:26 -04001293global_handler(struct display *display, uint32_t id,
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001294 const char *interface, uint32_t version, void *data)
1295{
1296 struct desktop *desktop = data;
1297
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001298 if (!strcmp(interface, "weston_desktop_shell")) {
Kristian Høgsbergfa80e112012-10-10 21:34:26 -04001299 desktop->shell = display_bind(desktop->display,
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001300 id,
1301 &weston_desktop_shell_interface,
1302 1);
1303 weston_desktop_shell_add_listener(desktop->shell,
1304 &listener,
1305 desktop);
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001306 } else if (!strcmp(interface, "wl_output")) {
1307 create_output(desktop, id);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001308 }
1309}
1310
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -05001311static void
Xiong Zhang83d8ee72013-10-23 13:58:35 +08001312global_handler_remove(struct display *display, uint32_t id,
1313 const char *interface, uint32_t version, void *data)
1314{
1315 struct desktop *desktop = data;
1316 struct output *output;
1317
1318 if (!strcmp(interface, "wl_output")) {
1319 wl_list_for_each(output, &desktop->outputs, link) {
1320 if (output->server_output_id == id) {
1321 output_destroy(output);
1322 break;
1323 }
1324 }
1325 }
1326}
1327
1328static void
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001329panel_add_launchers(struct panel *panel, struct desktop *desktop)
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -05001330{
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001331 struct weston_config_section *s;
1332 char *icon, *path;
1333 const char *name;
1334 int count;
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -05001335
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001336 count = 0;
1337 s = NULL;
1338 while (weston_config_next_section(desktop->config, &s, &name)) {
1339 if (strcmp(name, "launcher") != 0)
1340 continue;
1341
1342 weston_config_section_get_string(s, "icon", &icon, NULL);
1343 weston_config_section_get_string(s, "path", &path, NULL);
1344
1345 if (icon != NULL && path != NULL) {
1346 panel_add_launcher(panel, icon, path);
Rob Bradford09252d42013-07-26 16:29:45 +01001347 count++;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001348 } else {
1349 fprintf(stderr, "invalid launcher section\n");
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001350 }
1351
1352 free(icon);
1353 free(path);
Kristian Høgsbergac3a59a2011-11-14 22:43:37 -05001354 }
1355
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001356 if (count == 0) {
1357 /* add default launcher */
1358 panel_add_launcher(panel,
Kristian Høgsberg6af8eb92012-01-25 16:57:11 -05001359 DATADIR "/weston/terminal.png",
Rodney Lorrimar99ff01b2012-02-29 17:31:03 +01001360 BINDIR "/weston-terminal");
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001361 }
Kristian Høgsberg6af8eb92012-01-25 16:57:11 -05001362}
1363
Quentin Glidic55d57012016-06-23 18:55:18 +02001364static void
1365parse_panel_position(struct desktop *desktop, struct weston_config_section *s)
1366{
1367 char *position;
1368
Daniel Stone08cf24b2017-01-16 13:18:21 +00001369 desktop->want_panel = 1;
Quentin Glidic55d57012016-06-23 18:55:18 +02001370
Daniel Stone08cf24b2017-01-16 13:18:21 +00001371 weston_config_section_get_string(s, "panel-position", &position, "top");
1372 if (strcmp(position, "top") == 0) {
1373 desktop->panel_position = WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP;
1374 } else if (strcmp(position, "bottom") == 0) {
1375 desktop->panel_position = WESTON_DESKTOP_SHELL_PANEL_POSITION_BOTTOM;
1376 } else if (strcmp(position, "left") == 0) {
1377 desktop->panel_position = WESTON_DESKTOP_SHELL_PANEL_POSITION_LEFT;
1378 } else if (strcmp(position, "right") == 0) {
1379 desktop->panel_position = WESTON_DESKTOP_SHELL_PANEL_POSITION_RIGHT;
1380 } else {
1381 /* 'none' is valid here */
1382 if (strcmp(position, "none") != 0)
1383 fprintf(stderr, "Wrong panel position: %s\n", position);
1384 desktop->want_panel = 0;
Quentin Glidic55d57012016-06-23 18:55:18 +02001385 }
Daniel Stone08cf24b2017-01-16 13:18:21 +00001386 free(position);
Quentin Glidic55d57012016-06-23 18:55:18 +02001387}
1388
Quentin Glidic3e37b342016-06-23 18:55:22 +02001389static void
1390parse_clock_format(struct desktop *desktop, struct weston_config_section *s)
1391{
1392 char *clock_format;
1393
1394 weston_config_section_get_string(s, "clock-format", &clock_format, "");
1395 if (strcmp(clock_format, "minutes") == 0)
1396 desktop->clock_format = CLOCK_FORMAT_MINUTES;
1397 else if (strcmp(clock_format, "seconds") == 0)
1398 desktop->clock_format = CLOCK_FORMAT_SECONDS;
1399 else if (strcmp(clock_format, "none") == 0)
1400 desktop->clock_format = CLOCK_FORMAT_NONE;
1401 else
1402 desktop->clock_format = DEFAULT_CLOCK_FORMAT;
1403 free(clock_format);
1404}
1405
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001406int main(int argc, char *argv[])
1407{
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001408 struct desktop desktop = { 0 };
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001409 struct output *output;
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001410 struct weston_config_section *s;
Pekka Paalanen6c71aae2015-03-24 15:56:19 +02001411 const char *config_file;
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001412
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001413 desktop.unlock_task.run = unlock_dialog_finish;
Benjamin Franzked0f79ab2011-11-22 12:43:52 +01001414 wl_list_init(&desktop.outputs);
Pekka Paalanenbfbb26b2011-11-15 13:34:56 +02001415
Pekka Paalanen6c71aae2015-03-24 15:56:19 +02001416 config_file = weston_config_get_name_from_env();
1417 desktop.config = weston_config_parse(config_file);
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001418 s = weston_config_get_section(desktop.config, "shell", NULL, NULL);
1419 weston_config_section_get_bool(s, "locking", &desktop.locking, 1);
Quentin Glidic55d57012016-06-23 18:55:18 +02001420 parse_panel_position(&desktop, s);
Quentin Glidic3e37b342016-06-23 18:55:22 +02001421 parse_clock_format(&desktop, s);
Ander Conselvan de Oliveira6d75da72013-07-05 16:05:27 +03001422
Kristian Høgsberg4172f662013-02-20 15:27:49 -05001423 desktop.display = display_create(&argc, argv);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001424 if (desktop.display == NULL) {
1425 fprintf(stderr, "failed to create display: %m\n");
1426 return -1;
1427 }
1428
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001429 display_set_user_data(desktop.display, &desktop);
Kristian Høgsbergfa80e112012-10-10 21:34:26 -04001430 display_set_global_handler(desktop.display, global_handler);
Xiong Zhang83d8ee72013-10-23 13:58:35 +08001431 display_set_global_handler_remove(desktop.display, global_handler_remove);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001432
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001433 /* Create panel and background for outputs processed before the shell
1434 * global interface was processed */
Quentin Glidic55d57012016-06-23 18:55:18 +02001435 if (desktop.want_panel)
1436 weston_desktop_shell_set_panel_position(desktop.shell, desktop.panel_position);
Ander Conselvan de Oliveirae4925492013-07-05 16:05:28 +03001437 wl_list_for_each(output, &desktop.outputs, link)
1438 if (!output->panel)
1439 output_init(output, &desktop);
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001440
Ander Conselvan de Oliveirab9d2a0f2012-06-28 18:08:05 +03001441 grab_surface_create(&desktop);
Kristian Høgsbergd56bd902012-06-05 09:58:51 -04001442
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001443 signal(SIGCHLD, sigchild_handler);
1444
1445 display_run(desktop.display);
1446
U. Artie Eoff44874d92012-10-02 21:12:35 -07001447 /* Cleanup */
1448 grab_surface_destroy(&desktop);
1449 desktop_destroy_outputs(&desktop);
1450 if (desktop.unlock_dialog)
1451 unlock_dialog_destroy(desktop.unlock_dialog);
Jonas Ådahl6d6fb612015-11-17 16:00:33 +08001452 weston_desktop_shell_destroy(desktop.shell);
U. Artie Eoff44874d92012-10-02 21:12:35 -07001453 display_destroy(desktop.display);
1454
Kristian Høgsberg0c29eb22011-09-06 18:02:34 -04001455 return 0;
1456}