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 bool screen_locked;
37 private int screen_lock_fd;
38 // Timestamp of the past power button pressed even (0 if the button has
40 private Posix.timeval last_down;
41 private Posix.timeval last_short_press;
43 private bool hide_after_closing_power_menu;
45 public signal void screen_lock_changed(bool state);
47 public signal void power_short_press(Posix.timeval* t);
48 public signal void power_long_press();
49 private uint button_press_timeout;
53 screen_locked = false;
55 hide_after_closing_power_menu = false;
57 last_down.tv_usec = 0;
58 last_short_press.tv_sec = 0;
59 last_short_press.tv_usec = 0;
60 button_press_timeout = 0;
62 usage = zavai.registry.sbus.get_object(
63 "org.freesmartphone.ousaged",
64 "/org/freesmartphone/Usage",
65 "org.freesmartphone.Usage");
67 zavai.input.power_button.power_button += on_power_button;
68 zavai.input.power_button.request("zavai.ui.powerbutton.power");
70 power_short_press += on_power_short_press;
71 power_long_press += on_power_long_press;
74 public void shutdown()
76 zavai.input.power_button.release("zavai.ui.powerbutton.power");
79 public void do_suspend()
84 zavai.log.error("Suspending phone: " + e.message);
87 public void do_shutdown()
91 zavai.app.run_script("shutdown -h now");
93 zavai.log.error("Shutting down phone: " + e.message);
96 public void do_reboot()
100 zavai.app.run_script("shutdown -r now");
102 zavai.log.error("Rebooting phone: " + e.message);
106 public void set_screen_lock(bool locked)
108 if (locked && screen_locked)
110 if (!locked && !screen_locked)
115 screen_lock_fd = Posix.open("/dev/input/event1", Posix.O_RDWR | Posix.O_NONBLOCK);
116 if (screen_lock_fd < 0)
118 zavai.log.error("Cannot open /dev/input/event1");
122 // FIXME: X won't see events, but it's still generating interrupts,
124 int EVIOCGRAB = 0x40044590;
125 if (Posix.ioctl(screen_lock_fd, EVIOCGRAB, locked ? 1 : 0) != 0)
127 zavai.log.error("Cannot EVIOCGRAB /dev/input/event1");
128 Posix.close(screen_lock_fd);
132 backlight.lock_screen();
134 Posix.close(screen_lock_fd);
135 backlight.unlock_screen();
137 screen_locked = locked;
141 screen_lock_changed(locked);
144 private bool on_power_button_timeout()
146 last_down.tv_sec = 0;
147 last_down.tv_usec = 0;
153 private void on_power_button(Posix.timeval* t, bool pressed)
155 bool short_press = false;
156 bool long_press = false;
160 if (last_down.tv_sec == 0)
163 button_press_timeout = Timeout.add(1000, on_power_button_timeout);
167 long diff = timediff(t, &last_down);
168 long_press = diff >= 1000000;
171 if (last_down.tv_sec == 0)
173 // Ignore: release has been simulated with the timeout
175 if (button_press_timeout != 0)
177 // Cancel the timeout
178 Source.remove(button_press_timeout);
179 button_press_timeout = 0;
181 long diff = timediff(t, &last_down);
187 last_down.tv_sec = 0;
188 last_down.tv_usec = 0;
195 last_short_press.tv_sec = 0;
196 last_short_press.tv_usec = 0;
198 if (short_press) power_short_press(t);
201 private void on_power_short_press(Posix.timeval* t)
203 long diff = timediff(t, &last_short_press);
204 bool combo = screen_locked && (diff <= 5000000);
205 last_short_press = *t;
209 // Short press: turn on backlight for a bit
214 app.toggle_visibility();
218 // Short press: toggle power menu
222 private void on_power_long_press()
225 // Long press: unlock
226 set_screen_lock(false);
228 // Long press: lock screen
229 set_screen_lock(true);
233 public class BatteryIcon : Gtk.StatusIcon
235 //public dynamic DBus.Object battery;
236 public Dkp.Device battery;
238 public BatteryIcon(Dkp.Device dev)
241 battery.changed += on_changed;
243 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);
246 battery = zavai.registry.sbus.get_object(
247 "org.freesmartphone.odeviced",
248 "/org/freesmartphone/Device/PowerSupply/battery",
249 "org.freesmartphone.Device.PowerSupply");
252 // activate += on_activate;
255 battery.PowerStatus += on_power_status;
256 battery.Capacity += on_capacity;
258 last_status = battery.GetPowerStatus();
259 last_capacity = battery.GetCapacity();
265 private void on_changed(void* obj)
271 private void on_power_status(dynamic DBus.Object bat, string status)
273 zavai.log.info("New battery status: " + status);
274 last_status = status;
278 private void on_capacity(dynamic DBus.Object bat, int val)
280 stderr.printf("NEW CAPACITY: %d\n", val);
287 private void on_activate()
292 protected void update_icon()
294 string name = zavai.config.icondir + "/battery/";
295 bool charging = false;
296 Dkp.DeviceState state = (Dkp.DeviceState)battery.state;
298 stderr.printf("New battery status: %s\n", Dkp.Device.state_to_text(state));
299 int capacity = (int)Math.round(battery.percentage/10);
302 case Dkp.DeviceState.CHARGING:
303 name += "%02d0_charging_500.png".printf(capacity);
305 case Dkp.DeviceState.FULLY_CHARGED:
306 name += "100_charging_500.png";
308 case Dkp.DeviceState.UNKNOWN:
309 case Dkp.DeviceState.DISCHARGING:
310 case Dkp.DeviceState.EMPTY:
311 case Dkp.DeviceState.PENDING_CHARGE:
312 case Dkp.DeviceState.PENDING_DISCHARGE:
313 case Dkp.DeviceState.LAST:
314 name += "%02d0.png".printf(capacity);
318 stderr.printf("Loading icon from %s\n", name);
324 public class ScreenLockButton : Gtk.Button
326 public ScreenLockButton()
328 label = "Lock screen";
329 clicked += on_clicked;
330 set_size_request(0, zavai.config.min_button_height);
333 public void on_clicked()
335 zavai.log.info("Locking screen");
336 power.set_screen_lock(true);
337 power_menu.hide_menu();
341 public class SuspendButton : Gtk.Button
343 public SuspendButton()
346 clicked += on_clicked;
347 set_size_request(0, zavai.config.min_button_height);
350 public void on_clicked()
352 zavai.log.info("Suspending the phone via FSO");
354 power_menu.hide_menu();
358 public class ShutdownButton : Gtk.Button
360 public ShutdownButton()
363 clicked += on_clicked;
364 set_size_request(0, zavai.config.min_button_height);
367 public void on_clicked()
369 zavai.log.info("Shutting down the phone via FSO");
371 power_menu.hide_menu();
375 public class RebootButton : Gtk.Button
377 public RebootButton()
380 clicked += on_clicked;
381 set_size_request(0, zavai.config.min_button_height);
384 public void on_clicked()
386 zavai.log.info("Rebooting the phone via FSO");
388 power_menu.hide_menu();
392 // For a list of dbus services, look in /etc/dbus-1/system.d/
393 public class Backlight: zavai.Service
395 public dynamic DBus.Object usage;
401 usage = zavai.registry.sbus.get_object(
402 "org.freesmartphone.ousaged",
403 "/org/freesmartphone/Usage",
404 "org.freesmartphone.Usage");
407 // Turn the backlight and then let it fade off
410 // There must be a better method
412 //display.SetBacklightPower(true);
413 //usage.SetResourcePolicy("Display", "auto");
414 //usage.RequestResource("Display");
415 //usage.ReleaseResource("Display");
416 zavai.app.run_script(zavai.config.homedir + "/display wiggle");
418 zavai.log.error("Requesting/releasing resource Display: " + e.message);
422 public void lock_screen()
427 //display.SetBacklightPower(false);
428 zavai.app.run_script(zavai.config.homedir + "/display lock_off");
430 string policy = usage.GetResourcePolicy("Display");
431 if (policy == "auto")
433 usage.SetResourcePolicy("Display", "disabled");
436 } catch (GLib.Error e) {
437 zavai.log.error(e.message);
442 public void unlock_screen()
445 //display.SetBacklightPower(true);
446 zavai.app.run_script(zavai.config.homedir + "/display defaults");
447 //usage.SetResourcePolicy("Display", "auto");
448 } catch (GLib.Error e) {
449 zavai.log.error(e.message);
454 /// Request GPS resource
455 public override void start()
459 //usage.RequestResource("Display");
460 zavai.app.run_script(zavai.config.homedir + "/display lock_on");
461 zavai.log.info("Acquired display");
463 } catch (GLib.Error e) {
464 zavai.log.error(e.message);
469 // Release usage of GPS
470 public override void stop()
472 if (!started) return;
474 //usage.ReleaseResource("Display");
475 zavai.app.run_script(zavai.config.homedir + "/display defaults");
476 zavai.log.info("Released display");
478 } catch (GLib.Error e) {
479 zavai.log.error(e.message);
485 public class PowerMenu : zavai.Resource, Gtk.Window
487 protected Gtk.VBox vbox;
488 protected ScreenLockButton act_screen_lock;
489 protected SuspendButton act_suspend;
490 protected ShutdownButton act_shutdown;
491 protected RebootButton act_reboot;
492 protected ServiceRequestLink act_backlight_on;
493 protected bool shown;
497 type = Gtk.WindowType.TOPLEVEL;
498 title = "Power Menu";
500 destroy_with_parent = true;
501 set_transient_for(zavai.app);
503 set_position(Gtk.WindowPosition.CENTER_ON_PARENT);
504 set_size_request(300, 500);
506 vbox = new Gtk.VBox(false, 0);
509 //destroy += Gtk.main_quit;
510 //set_events(get_events() | Gdk.EventMask.VISIBILITY_NOTIFY_MASK);
511 //visibility_notify_event += on_visibility;
512 set_skip_pager_hint(true);
513 set_skip_taskbar_hint(true);
514 set_type_hint(Gdk.WindowTypeHint.POPUP_MENU);
516 act_screen_lock = new ScreenLockButton();
517 vbox.pack_start(act_screen_lock, false, false, 0);
519 act_suspend = new SuspendButton();
520 vbox.pack_start(act_suspend, false, false, 0);
522 act_shutdown = new ShutdownButton();
523 vbox.pack_start(act_shutdown, false, false, 0);
525 act_reboot = new RebootButton();
526 vbox.pack_start(act_reboot, false, false, 0);
528 act_backlight_on = new ServiceRequestLink("backlight", "Keep backlight on", "Let backlight fade");
529 act_backlight_on.toggled += (src) => { this.hide_menu(); };
530 vbox.pack_start(act_backlight_on, false, false, 0);
545 // TODO: do more in case it is visible but has no visibility (is covered by others)
552 public void hide_menu()
557 public void shutdown() {}
561 public class TogglePowerMenu : Gtk.Button
563 public TogglePowerMenu()
565 label = "Toggle power menu";
566 clicked += on_clicked;
567 set_size_request(0, zavai.config.min_button_height);
570 public void on_clicked()
572 zavai.log.info("Toggling power menu");
579 PowerMenu power_menu;
580 Gee.ArrayList<BatteryIcon> battery_icons;
582 //TogglePowerMenu tpm;
587 backlight = new Backlight();
588 zavai.registry.register_service(backlight);
590 battery_icons = new Gee.ArrayList<BatteryIcon>();
591 // Enumerate batteries
592 var c = new Dkp.Client();
593 unowned GLib.PtrArray devs = c.enumerate_devices();
594 for (int i = 0; i < devs.len; ++i)
596 Dkp.Device dev = (Dkp.Device)devs.pdata[i];
597 stderr.printf("Found new device %s\n", dev.native_path);
599 stderr.printf("Rechargeable: %s\n", dev.is_rechargeable ? "yes" : "no");
600 if (!dev.is_rechargeable) continue;
601 var bi = new BatteryIcon(dev);
602 bi.set_visible(true);
603 battery_icons.add(bi);
606 power_menu = new PowerMenu();
607 zavai.registry.register_resource("powermenu", power_menu);
609 //zavai.registry.getmenu("menu.main").add_applet("menu.power");
610 //tpm = new TogglePowerMenu();
611 //zavai.registry.getmenu("menu.main").add_widget(tpm);
614 raise_icon = new RaiseIcon();
615 raise_icon.set_visible(true);
617 close_or_back = new CloseOrBack();
618 close_or_back.set_visible(true);
620 window_list = new WindowList("Current apps");
621 zavai.registry.register_applet("wm.list", window_list);
622 zavai.registry.getmenu("menu.main").add_applet("wm.list");
625 launcher = new Launcher("Run program");
627 zavai.log.error("Not running launcher: " + e.message);
631 if (launcher != null)
633 zavai.registry.register_applet("wm.launcher", launcher);
634 zavai.registry.getmenu("menu.main").add_applet("wm.launcher");