2019-07-10 01:51:41 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
|
|
|
|
from gi.repository import GObject, Gtk
|
|
|
|
from .drawing import Drawing
|
|
|
|
from .svg import JsonSvg
|
2019-07-16 04:56:02 +02:00
|
|
|
from .config import Config
|
2019-07-10 01:51:41 +02:00
|
|
|
|
2019-07-15 07:39:19 +02:00
|
|
|
import time
|
2019-07-10 01:51:41 +02:00
|
|
|
import gi
|
2019-07-16 07:28:51 +02:00
|
|
|
import logging
|
|
|
|
|
2019-07-10 01:51:41 +02:00
|
|
|
gi.require_version("Gtk", "3.0")
|
|
|
|
|
2019-07-11 12:17:38 +02:00
|
|
|
logging.basicConfig(level=logging.DEBUG)
|
|
|
|
logger = logging.getLogger('drawingperspective')
|
|
|
|
|
2019-07-16 07:28:51 +02:00
|
|
|
|
2019-07-15 07:39:19 +02:00
|
|
|
def relative_time(seconds):
|
|
|
|
MIN = 60
|
|
|
|
H = 60 * MIN
|
|
|
|
DAY = 24 * H
|
|
|
|
WEEK = 7 * DAY
|
|
|
|
|
|
|
|
if seconds < 30:
|
2019-07-16 07:28:51 +02:00
|
|
|
return 'just now'
|
2019-07-15 07:39:19 +02:00
|
|
|
if seconds < 5 * MIN:
|
2019-07-16 07:28:51 +02:00
|
|
|
return 'a few minutes ago'
|
2019-07-15 07:39:19 +02:00
|
|
|
if seconds < H:
|
2019-07-16 07:28:51 +02:00
|
|
|
return f'{int(seconds/MIN/10) * 10} minutes ago'
|
2019-07-15 07:39:19 +02:00
|
|
|
if seconds < DAY:
|
2019-07-16 07:28:51 +02:00
|
|
|
return f'{int(seconds/H)} hours ago'
|
2019-07-15 07:39:19 +02:00
|
|
|
if seconds < 4 * WEEK:
|
2019-07-16 07:28:51 +02:00
|
|
|
return f'{int(seconds/DAY)} days ago'
|
|
|
|
|
2019-07-15 07:39:19 +02:00
|
|
|
return 'a long time ago'
|
|
|
|
|
2019-07-10 01:51:41 +02:00
|
|
|
|
|
|
|
@Gtk.Template(resource_path="/org/freedesktop/TuhiGui/ui/DrawingPerspective.ui")
|
|
|
|
class DrawingPerspective(Gtk.Stack):
|
|
|
|
__gtype_name__ = "DrawingPerspective"
|
|
|
|
|
|
|
|
image_battery = Gtk.Template.Child()
|
|
|
|
flowbox_drawings = Gtk.Template.Child()
|
2019-07-15 07:39:19 +02:00
|
|
|
spinner_sync = Gtk.Template.Child()
|
|
|
|
label_last_sync = Gtk.Template.Child()
|
2019-07-10 01:51:41 +02:00
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
2019-07-11 08:05:41 +02:00
|
|
|
self.known_drawings = []
|
2019-07-15 07:39:19 +02:00
|
|
|
self.last_sync_time = 0
|
|
|
|
self._sync_label_timer = GObject.timeout_add_seconds(60, self._update_sync_label)
|
|
|
|
self._update_sync_label()
|
2019-07-16 04:56:02 +02:00
|
|
|
Config.load().connect('notify::orientation', self._on_orientation_changed)
|
|
|
|
|
|
|
|
def _on_orientation_changed(self, config, pspec):
|
|
|
|
# When the orientation changes, we just re-generate all SVGs. This
|
|
|
|
# isn't something that should happen very often anyway so meh.
|
|
|
|
self.known_drawings = []
|
|
|
|
child = self.flowbox_drawings.get_child_at_index(0)
|
|
|
|
while child is not None:
|
|
|
|
self.flowbox_drawings.remove(child)
|
|
|
|
child = self.flowbox_drawings.get_child_at_index(0)
|
|
|
|
|
2019-07-16 05:52:54 +02:00
|
|
|
self._update_drawings(Config.load(), None)
|
2019-07-10 01:51:41 +02:00
|
|
|
|
2019-07-16 05:52:54 +02:00
|
|
|
def _cache_drawings(self, device, pspec):
|
|
|
|
# The config backend filters duplicates anyway, so don't care here
|
|
|
|
for ts in self.device.drawings_available:
|
|
|
|
json_string = self.device.json(ts)
|
|
|
|
Config.load().add_drawing(ts, json_string)
|
|
|
|
|
|
|
|
def _update_drawings(self, config, pspec):
|
|
|
|
for js in config.drawings:
|
|
|
|
if js in self.known_drawings:
|
2019-07-11 08:05:41 +02:00
|
|
|
continue
|
|
|
|
|
2019-07-16 05:52:54 +02:00
|
|
|
self.known_drawings.append(js)
|
|
|
|
|
2019-07-10 01:51:41 +02:00
|
|
|
svg = JsonSvg(js)
|
|
|
|
drawing = Drawing(svg)
|
2019-07-16 05:52:54 +02:00
|
|
|
|
|
|
|
# We don't know which order we get drawings from the device, so
|
|
|
|
# let's do a sorted insert here
|
|
|
|
index = 0
|
|
|
|
child = self.flowbox_drawings.get_child_at_index(index)
|
|
|
|
while child is not None:
|
|
|
|
if child.get_child().timestamp < drawing.timestamp:
|
|
|
|
break
|
|
|
|
index += 1
|
|
|
|
child = self.flowbox_drawings.get_child_at_index(index)
|
|
|
|
|
|
|
|
self.flowbox_drawings.insert(drawing, index)
|
2019-07-10 01:51:41 +02:00
|
|
|
|
|
|
|
@GObject.Property
|
|
|
|
def device(self):
|
|
|
|
return self._device
|
|
|
|
|
|
|
|
@device.setter
|
|
|
|
def device(self, device):
|
|
|
|
self._device = device
|
|
|
|
|
2019-07-11 08:05:41 +02:00
|
|
|
device.connect('notify::connected', self._on_connected)
|
|
|
|
device.connect('notify::listening', self._on_listening_stopped)
|
2019-07-15 07:39:19 +02:00
|
|
|
device.connect('notify::sync-state', self._on_sync_state)
|
2019-07-16 02:28:55 +02:00
|
|
|
device.connect('notify::battery-percent', self._on_battery_changed)
|
|
|
|
device.connect('notify::battery-state', self._on_battery_changed)
|
2019-07-16 05:52:54 +02:00
|
|
|
|
|
|
|
# This is a bit convoluted. We need to cache all drawings
|
|
|
|
# because Tuhi doesn't have guaranteed storage. So any json that
|
|
|
|
# comes in from Tuhi, we pass to our config backend to save
|
|
|
|
# somewhere.
|
|
|
|
# The config backend adds the json file and emits a notify for the
|
|
|
|
# json itself (once cached) that we then actually use for SVG
|
|
|
|
# generation.
|
|
|
|
device.connect('notify::drawings-available', self._cache_drawings)
|
|
|
|
Config.load().connect('notify::drawings', self._update_drawings)
|
2019-07-11 08:05:41 +02:00
|
|
|
|
2019-07-16 02:28:55 +02:00
|
|
|
self._on_battery_changed(device, None)
|
|
|
|
|
2019-07-16 05:52:54 +02:00
|
|
|
self._update_drawings(Config.load(), None)
|
2019-07-16 02:28:55 +02:00
|
|
|
|
|
|
|
# We always want to sync on startup
|
|
|
|
logger.debug(f'{device.name} - starting to listen')
|
|
|
|
device.start_listening()
|
|
|
|
|
|
|
|
@GObject.Property
|
|
|
|
def name(self):
|
|
|
|
return "drawing_perspective"
|
|
|
|
|
|
|
|
def _on_battery_changed(self, device, pspec):
|
2019-07-15 07:43:49 +02:00
|
|
|
if device.battery_percent > 80:
|
2019-07-16 03:06:57 +02:00
|
|
|
fill = 'full'
|
2019-07-15 07:43:49 +02:00
|
|
|
elif device.battery_percent > 40:
|
2019-07-16 03:06:57 +02:00
|
|
|
fill = 'good'
|
2019-07-15 07:43:49 +02:00
|
|
|
elif device.battery_percent > 10:
|
2019-07-16 03:06:57 +02:00
|
|
|
fill = 'low'
|
2019-07-15 07:43:49 +02:00
|
|
|
else:
|
2019-07-16 03:06:57 +02:00
|
|
|
fill = 'caution'
|
2019-07-15 07:43:49 +02:00
|
|
|
|
2019-07-10 01:51:41 +02:00
|
|
|
if device.battery_state == 1:
|
|
|
|
state = '-charging'
|
|
|
|
else:
|
|
|
|
state = ''
|
2019-07-16 03:06:57 +02:00
|
|
|
batt_icon_name = f'battery-{fill}{state}-symbolic'
|
2019-07-10 01:51:41 +02:00
|
|
|
_, isize = self.image_battery.get_icon_name()
|
|
|
|
self.image_battery.set_from_icon_name(batt_icon_name, isize)
|
2019-07-16 03:04:46 +02:00
|
|
|
self.image_battery.set_tooltip_text(f'{device.battery_percent}%')
|
2019-07-10 01:51:41 +02:00
|
|
|
|
2019-07-15 07:39:19 +02:00
|
|
|
def _on_sync_state(self, device, pspec):
|
|
|
|
if device.sync_state:
|
|
|
|
self.spinner_sync.start()
|
|
|
|
else:
|
|
|
|
self.spinner_sync.stop()
|
|
|
|
self.last_sync_time = time.time()
|
|
|
|
self._update_sync_label()
|
|
|
|
|
|
|
|
def _update_sync_label(self):
|
|
|
|
now = time.time()
|
|
|
|
self.label_last_sync.set_text(f'{relative_time(now - self.last_sync_time)}')
|
|
|
|
return True
|
|
|
|
|
2019-07-11 08:05:41 +02:00
|
|
|
def _on_connected(self, device, pspec):
|
2019-07-11 12:18:17 +02:00
|
|
|
# Turns out we don't really care about whether the device is
|
|
|
|
# connected or not, it has little effect on how we work here
|
2019-07-11 08:05:41 +02:00
|
|
|
pass
|
|
|
|
|
|
|
|
def _on_listening_stopped(self, device, pspec):
|
2019-07-11 12:18:17 +02:00
|
|
|
if not device.listening:
|
|
|
|
logger.debug(f'{device.name} - listening stopped, restarting')
|
|
|
|
# We never want to stop listening
|
|
|
|
device.start_listening()
|