2 * app_power - zavai power handling
4 * Copyright (C) 2009 Enrico Zini <enrico@enricozini.org>
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
27 // Compute a-b in microseconds
28 static long timediff(Posix.timeval* a, Posix.timeval* b)
30 return (a->tv_sec - b->tv_sec) * 1000000 + (a->tv_usec - b->tv_usec);
33 public class Power : zavai.Resource, Object
35 public dynamic DBus.Object usage;
36 public dynamic DBus.Object gsm_device;
37 public bool screen_locked;
38 private int screen_lock_fd;
39 // Timestamp of the past power button pressed even (0 if the button has
41 private Posix.timeval last_down;
42 private Posix.timeval last_short_press;
44 private bool hide_after_closing_power_menu;
46 public signal void screen_lock_changed(bool state);
48 public signal void power_short_press(Posix.timeval* t);
49 public signal void power_long_press();
50 private uint button_press_timeout;
54 screen_locked = false;
56 hide_after_closing_power_menu = false;
58 last_down.tv_usec = 0;
59 last_short_press.tv_sec = 0;
60 last_short_press.tv_usec = 0;
61 button_press_timeout = 0;
63 usage = zavai.registry.sbus.get_object(
64 "org.freesmartphone.ousaged",
65 "/org/freesmartphone/Usage",
66 "org.freesmartphone.Usage");
67 gsm_device = zavai.registry.sbus.get_object(
68 "org.freesmartphone.ogsmd",
69 "/org/freesmartphone/GSM/Device",
70 "org.freesmartphone.Resource");
72 zavai.input.power_button.power_button += on_power_button;
73 zavai.input.power_button.request("zavai.ui.powerbutton.power");
75 power_short_press += on_power_short_press;
76 power_long_press += on_power_long_press;
79 public void shutdown()
81 zavai.input.power_button.release("zavai.ui.powerbutton.power");
84 public void do_suspend()
93 zavai.log.error("Suspending phone with ousaged: " + e.message);
98 // From http://lindi.iki.fi/lindi/openmoko/susp
100 gsm_device.Suspend();
102 zavai.log.error("Cannot tell GSM to suspend (but never mind): " + e.message);
104 // amixer -q -d sset "Amp Spk" mute
106 // echo 0 | sudo tee /proc/sysrq-trigger
108 // Limit the scope of state, so that it's
109 // closed before we resume
110 FileStream state = FileStream.open("/sys/power/state", "w");
117 // amixer -q -d sset "Amp Spk" unmute
121 zavai.log.error("Cannot tell GSM to resume (but never mind): " + e.message);
126 public void do_shutdown()
130 zavai.app.run_script("shutdown -h now");
132 zavai.log.error("Shutting down phone: " + e.message);
135 public void do_reboot()
139 zavai.app.run_script("shutdown -r now");
141 zavai.log.error("Rebooting phone: " + e.message);
145 public void set_screen_lock(bool locked)
147 if (locked && screen_locked)
149 if (!locked && !screen_locked)
154 screen_lock_fd = Posix.open("/dev/input/event1", Posix.O_RDWR | Posix.O_NONBLOCK);
155 if (screen_lock_fd < 0)
157 zavai.log.error("Cannot open /dev/input/event1");
161 // FIXME: X won't see events, but it's still generating interrupts,
163 int EVIOCGRAB = 0x40044590;
164 if (Posix.ioctl(screen_lock_fd, EVIOCGRAB, locked ? 1 : 0) != 0)
166 zavai.log.error("Cannot EVIOCGRAB /dev/input/event1");
167 Posix.close(screen_lock_fd);
171 backlight.lock_screen();
173 Posix.close(screen_lock_fd);
174 backlight.unlock_screen();
176 screen_locked = locked;
180 screen_lock_changed(locked);
183 private bool on_power_button_timeout()
185 last_down.tv_sec = 0;
186 last_down.tv_usec = 0;
192 private void on_power_button(Posix.timeval* t, bool pressed)
194 bool short_press = false;
195 bool long_press = false;
199 if (last_down.tv_sec == 0)
202 button_press_timeout = Timeout.add(1000, on_power_button_timeout);
206 long diff = timediff(t, &last_down);
207 long_press = diff >= 1000000;
210 if (last_down.tv_sec == 0)
212 // Ignore: release has been simulated with the timeout
214 if (button_press_timeout != 0)
216 // Cancel the timeout
217 Source.remove(button_press_timeout);
218 button_press_timeout = 0;
220 long diff = timediff(t, &last_down);
226 last_down.tv_sec = 0;
227 last_down.tv_usec = 0;
234 last_short_press.tv_sec = 0;
235 last_short_press.tv_usec = 0;
237 if (short_press) power_short_press(t);
240 private void on_power_short_press(Posix.timeval* t)
242 long diff = timediff(t, &last_short_press);
243 bool combo = screen_locked && (diff <= 5000000);
244 last_short_press = *t;
248 // Short press: turn on backlight for a bit
253 app.toggle_visibility();
257 // Short press: toggle power menu
261 private void on_power_long_press()
264 // Long press: unlock
265 set_screen_lock(false);
267 // Long press: lock screen
268 set_screen_lock(true);
272 public class BatteryIcon : Gtk.StatusIcon
274 public Dkp.Device battery;
276 public BatteryIcon(Dkp.Device dev)
279 battery.changed += on_changed;
281 stderr.printf("New battery icon for %s online %s perc %f isrec %s tte %lld ttf %lld\n", dev.native_path, dev.online ? "yes" : "no", dev.percentage, dev.is_rechargeable ? "yes" : "no", dev.time_to_empty, dev.time_to_full);
286 private void on_changed(void* obj)
291 protected void update_icon()
293 string name = zavai.config.icondir + "/battery/";
294 Dkp.DeviceState state = (Dkp.DeviceState)battery.state;
296 stderr.printf("New battery status: %s\n", Dkp.Device.state_to_text(state));
297 int capacity = (int)Math.round(battery.percentage/10);
300 case Dkp.DeviceState.CHARGING:
301 name += "%02d0_charging_500.png".printf(capacity);
303 case Dkp.DeviceState.FULLY_CHARGED:
304 name += "100_charging_500.png";
306 case Dkp.DeviceState.UNKNOWN:
307 case Dkp.DeviceState.DISCHARGING:
308 case Dkp.DeviceState.EMPTY:
309 case Dkp.DeviceState.PENDING_CHARGE:
310 case Dkp.DeviceState.PENDING_DISCHARGE:
311 case Dkp.DeviceState.LAST:
312 name += "%02d0.png".printf(capacity);
316 stderr.printf("Loading icon from %s\n", name);
322 public class ScreenLockButton : Gtk.Button
324 public ScreenLockButton()
326 label = "Lock screen";
327 clicked += on_clicked;
328 set_size_request(0, zavai.config.min_button_height);
331 public void on_clicked()
333 zavai.log.info("Locking screen");
334 power.set_screen_lock(true);
335 power_menu.hide_menu();
339 public class SuspendButton : Gtk.Button
341 public SuspendButton()
344 clicked += on_clicked;
345 set_size_request(0, zavai.config.min_button_height);
348 public void on_clicked()
350 zavai.log.info("Suspending the phone");
352 power_menu.hide_menu();
356 public class ShutdownButton : Gtk.Button
358 public ShutdownButton()
361 clicked += on_clicked;
362 set_size_request(0, zavai.config.min_button_height);
365 public void on_clicked()
367 zavai.log.info("Shutting down the phone");
369 power_menu.hide_menu();
373 public class RebootButton : Gtk.Button
375 public RebootButton()
378 clicked += on_clicked;
379 set_size_request(0, zavai.config.min_button_height);
382 public void on_clicked()
384 zavai.log.info("Rebooting the phone");
386 power_menu.hide_menu();
390 // For a list of dbus services, look in /etc/dbus-1/system.d/
391 public class Backlight: zavai.Service
398 // Turn the backlight on and then let it fade off
402 zavai.app.run_script(zavai.config.homedir + "/display wiggle");
404 zavai.log.error("Requesting/releasing resource Display: " + e.message);
408 public void lock_screen()
413 zavai.app.run_script(zavai.config.homedir + "/display lock_off");
414 } catch (GLib.Error e) {
415 zavai.log.error(e.message);
420 public void unlock_screen()
423 zavai.app.run_script(zavai.config.homedir + "/display defaults");
424 } catch (GLib.Error e) {
425 zavai.log.error(e.message);
429 public override void start()
433 zavai.app.run_script(zavai.config.homedir + "/display lock_on");
434 zavai.log.info("Acquired display");
436 } catch (GLib.Error e) {
437 zavai.log.error(e.message);
442 public override void stop()
444 if (!started) return;
446 zavai.app.run_script(zavai.config.homedir + "/display defaults");
447 zavai.log.info("Released display");
449 } catch (GLib.Error e) {
450 zavai.log.error(e.message);
456 public class PowerMenu : zavai.Resource, Gtk.Window
458 protected Gtk.VBox vbox;
459 protected ScreenLockButton act_screen_lock;
460 protected SuspendButton act_suspend;
461 protected ShutdownButton act_shutdown;
462 protected RebootButton act_reboot;
463 protected ServiceRequestLink act_backlight_on;
464 protected bool shown;
468 type = Gtk.WindowType.TOPLEVEL;
469 title = "Power Menu";
471 destroy_with_parent = true;
472 set_transient_for(zavai.app);
474 set_position(Gtk.WindowPosition.CENTER_ON_PARENT);
475 set_size_request(300, 500);
477 vbox = new Gtk.VBox(false, 0);
480 //destroy += Gtk.main_quit;
481 //set_events(get_events() | Gdk.EventMask.VISIBILITY_NOTIFY_MASK);
482 //visibility_notify_event += on_visibility;
483 set_skip_pager_hint(true);
484 set_skip_taskbar_hint(true);
485 set_type_hint(Gdk.WindowTypeHint.POPUP_MENU);
487 act_screen_lock = new ScreenLockButton();
488 vbox.pack_start(act_screen_lock, false, false, 0);
490 act_suspend = new SuspendButton();
491 vbox.pack_start(act_suspend, false, false, 0);
493 act_shutdown = new ShutdownButton();
494 vbox.pack_start(act_shutdown, false, false, 0);
496 act_reboot = new RebootButton();
497 vbox.pack_start(act_reboot, false, false, 0);
499 act_backlight_on = new ServiceRequestLink("backlight", "Keep backlight on", "Let backlight fade");
500 act_backlight_on.toggled += (src) => { this.hide_menu(); };
501 vbox.pack_start(act_backlight_on, false, false, 0);
516 // TODO: do more in case it is visible but has no visibility (is covered by others)
523 public void hide_menu()
528 public void shutdown() {}
532 public class TogglePowerMenu : Gtk.Button
534 public TogglePowerMenu()
536 label = "Toggle power menu";
537 clicked += on_clicked;
538 set_size_request(0, zavai.config.min_button_height);
541 public void on_clicked()
543 zavai.log.info("Toggling power menu");
550 PowerMenu power_menu;
551 Gee.ArrayList<BatteryIcon> battery_icons;
553 //TogglePowerMenu tpm;
558 backlight = new Backlight();
559 zavai.registry.register_service(backlight);
561 battery_icons = new Gee.ArrayList<BatteryIcon>();
562 // Enumerate batteries
563 var c = new Dkp.Client();
564 unowned GLib.PtrArray devs = c.enumerate_devices();
565 for (int i = 0; i < devs.len; ++i)
567 Dkp.Device dev = (Dkp.Device)devs.pdata[i];
568 stderr.printf("Found new device %s\n", dev.native_path);
570 stderr.printf("Rechargeable: %s\n", dev.is_rechargeable ? "yes" : "no");
571 if (!dev.is_rechargeable) continue;
572 var bi = new BatteryIcon(dev);
573 bi.set_visible(true);
574 battery_icons.add(bi);
577 power_menu = new PowerMenu();
578 zavai.registry.register_resource("powermenu", power_menu);
580 //zavai.registry.getmenu("menu.main").add_applet("menu.power");
581 //tpm = new TogglePowerMenu();
582 //zavai.registry.getmenu("menu.main").add_widget(tpm);
585 raise_icon = new RaiseIcon();
586 raise_icon.set_visible(true);
588 close_or_back = new CloseOrBack();
589 close_or_back.set_visible(true);
591 window_list = new WindowList("Current apps");
592 zavai.registry.register_applet("wm.list", window_list);
593 zavai.registry.getmenu("menu.main").add_applet("wm.list");
596 launcher = new Launcher("Run program");
598 zavai.log.error("Not running launcher: " + e.message);
602 if (launcher != null)
604 zavai.registry.register_applet("wm.launcher", launcher);
605 zavai.registry.getmenu("menu.main").add_applet("wm.launcher");