GTK+ GUI for configuring PD Buddy Sink devices
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

pd-buddy-gtk.py 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  1. #!/usr/bin/env python3
  2. import sys
  3. import pdbuddy
  4. import gi
  5. gi.require_version('Gtk', '3.0')
  6. from gi.repository import Gtk, Gio, GObject, GLib
  7. def comms_error_dialog(parent, e):
  8. dialog = Gtk.MessageDialog(window, 0, Gtk.MessageType.ERROR,
  9. Gtk.ButtonsType.CLOSE, "Error communicating with device")
  10. dialog.format_secondary_text(e.strerror)
  11. dialog.run()
  12. dialog.destroy()
  13. class ListRowModel(GObject.GObject):
  14. def __init__(self, serport):
  15. GObject.GObject.__init__(self)
  16. self.serport = serport
  17. class SelectListStore(Gio.ListStore):
  18. def update_items(self):
  19. # Get a list of serial ports
  20. serports = list(pdbuddy.Sink.get_devices())
  21. # Mark ports to remove or add
  22. remove_list = []
  23. list_len = self.get_n_items()
  24. for i in range(list_len):
  25. remove = True
  26. for j in range(len(serports)):
  27. if serports[j] is not None and self.get_item(i).serport == serports[j]:
  28. serports[j] = None
  29. remove = False
  30. if remove:
  31. remove_list.append(i)
  32. # Remove the missing ones
  33. for i in remove_list:
  34. self.remove(i)
  35. # Add any new ports
  36. for port in serports:
  37. if port is not None:
  38. self.append(ListRowModel(port))
  39. def list_box_update_header_func(row, before, data):
  40. """Add a separator header to all rows but the first one"""
  41. if before is None:
  42. row.set_header(None)
  43. return
  44. current = row.get_header()
  45. if current is None:
  46. current = Gtk.Separator.new(Gtk.Orientation.HORIZONTAL)
  47. row.set_header(current)
  48. class SelectList(Gtk.Box):
  49. __gsignals__ = {
  50. 'row-activated': (GObject.SIGNAL_RUN_FIRST, None,
  51. (object,))
  52. }
  53. def __init__(self):
  54. Gtk.Box.__init__(self)
  55. self._model = None
  56. self._builder = Gtk.Builder()
  57. self._builder.add_from_file("data/select-stack.ui")
  58. self._builder.connect_signals(self)
  59. sl = self._builder.get_object("select-list")
  60. # Add separators to the list
  61. sl.set_header_func(list_box_update_header_func, None)
  62. self.pack_start(self._builder.get_object("select-stack"), True, True, 0)
  63. self.show_all()
  64. def bind_model(self, model, func):
  65. self._builder.get_object("select-list").bind_model(model, func)
  66. self._model = model
  67. self.reload()
  68. GLib.timeout_add(1000, self.reload)
  69. def reload(self):
  70. self._model.update_items()
  71. # Set the visible child
  72. stack = self._builder.get_object("select-stack")
  73. if self._model.get_n_items():
  74. stack.set_visible_child(self._builder.get_object("select-frame"))
  75. else:
  76. stack.set_visible_child(self._builder.get_object("select-none"))
  77. return True
  78. def on_select_list_row_activated(self, box, row):
  79. self.emit("row-activated", row.model.serport)
  80. class SelectListRow(Gtk.ListBoxRow):
  81. def __init__(self, model):
  82. Gtk.EventBox.__init__(self)
  83. self.model = model
  84. self._builder = Gtk.Builder()
  85. self._builder.add_from_file("data/select-list-row.ui")
  86. self._builder.connect_signals(self)
  87. name = self._builder.get_object("name")
  88. name.set_text('{} {} {}'.format(self.model.serport.manufacturer,
  89. self.model.serport.product,
  90. self.model.serport.serial_number))
  91. device = self._builder.get_object("device")
  92. device.set_text(self.model.serport.device)
  93. self.add(self._builder.get_object("grid"))
  94. self.show_all()
  95. def on_identify_clicked(self, button):
  96. window = self.get_toplevel()
  97. try:
  98. with pdbuddy.Sink(self.model.serport) as pdbs:
  99. pdbs.identify()
  100. except OSError as e:
  101. comms_error_dialog(window, e)
  102. return
  103. class Handler:
  104. def __init__(self, builder):
  105. self.builder = builder
  106. self.serial_port = None
  107. self.voltage = None
  108. self.current = None
  109. self.giveback = None
  110. self.selectlist = None
  111. def on_pdb_window_realize(self, *args):
  112. # Get the list
  113. sb = self.builder.get_object("select-box")
  114. self.selectlist = SelectList()
  115. sb.pack_start(self.selectlist, True, True, 0)
  116. liststore = SelectListStore()
  117. self.selectlist.bind_model(liststore, SelectListRow)
  118. self.selectlist.connect("row-activated", self.on_select_list_row_activated)
  119. # Add separators to the configuration page lists
  120. sc_list = self.builder.get_object("sink-config-list")
  121. sc_list.set_header_func(list_box_update_header_func, None)
  122. pd_list = self.builder.get_object("power-delivery-list")
  123. pd_list.set_header_func(list_box_update_header_func, None)
  124. def on_pdb_window_delete_event(self, *args):
  125. Gtk.main_quit(*args)
  126. def on_select_list_row_activated(self, selectlist, serport):
  127. # Get relevant widgets
  128. voltage = self.builder.get_object("voltage-combobox")
  129. current = self.builder.get_object("current-spinbutton")
  130. giveback = self.builder.get_object("giveback-toggle")
  131. pd_frame = self.builder.get_object("power-delivery-frame")
  132. output = self.builder.get_object("output-switch")
  133. cap_warning = self.builder.get_object("source-cap-warning")
  134. cap_label = self.builder.get_object("short-source-cap-label")
  135. self.serial_port = serport
  136. window = self.builder.get_object("pdb-window")
  137. try:
  138. with pdbuddy.Sink(self.serial_port) as pdbs:
  139. try:
  140. pdbs.load()
  141. except KeyError:
  142. # If there's no configuration, we don't want to fail. We
  143. # do want to display no configuration though
  144. self.cfg = pdbuddy.SinkConfig(
  145. status=pdbuddy.SinkStatus.VALID,
  146. flags=pdbuddy.SinkFlags.NONE, v=0, i=0)
  147. else:
  148. self.cfg = pdbs.get_tmpcfg()
  149. except OSError as e:
  150. comms_error_dialog(window, e)
  151. return
  152. self._store_device_settings()
  153. self._set_save_button_visibility()
  154. # Set giveback button state
  155. giveback.set_active(bool(self.cfg.flags & pdbuddy.SinkFlags.GIVEBACK))
  156. # Get voltage and current from device and load them into the GUI
  157. if self.cfg.v == 5000:
  158. voltage.set_active_id('voltage-five')
  159. elif self.cfg.v == 9000:
  160. voltage.set_active_id('voltage-nine')
  161. elif self.cfg.v == 12000:
  162. voltage.set_active_id('voltage-twelve')
  163. elif self.cfg.v == 15000:
  164. voltage.set_active_id('voltage-fifteen')
  165. if self.cfg.v == 20000:
  166. voltage.set_active_id('voltage-twenty')
  167. current.set_value(self.cfg.i/1000)
  168. # Set PD frame visibility and output switch state
  169. try:
  170. with pdbuddy.Sink(self.serial_port) as pdbs:
  171. output.set_state(pdbs.output)
  172. except KeyError:
  173. pd_frame.set_visible(False)
  174. else:
  175. pd_frame.set_visible(True)
  176. # TODO: do these next things repeatedly
  177. # Get the Source_Capabilities
  178. with pdbuddy.Sink(self.serial_port) as pdbs:
  179. caps = pdbs.get_source_cap()
  180. # Update the warning icon
  181. cap_warning.set_visible(not pdbuddy.follows_power_rules(caps))
  182. # Update the text in the capability label
  183. cap_label.set_text('{:g} W'.format(pdbuddy.calculate_pdp(caps)))
  184. # Show the Sink page
  185. hst = self.builder.get_object("header-stack")
  186. hsink = self.builder.get_object("header-sink")
  187. hsink.set_title('{} {} {}'.format(serport.manufacturer,
  188. serport.product,
  189. serport.serial_number))
  190. hsink.set_subtitle(serport.device)
  191. hst.set_visible_child(hsink)
  192. st = self.builder.get_object("stack")
  193. sink = self.builder.get_object("sink")
  194. st.set_visible_child(sink)
  195. # Ping the Sink repeatedly
  196. GLib.timeout_add(1000, self._ping)
  197. def _ping(self):
  198. """Ping the device we're configuring, showing to the list on failure"""
  199. if self.serial_port is None:
  200. self.selectlist.reload()
  201. self.on_header_sink_back_clicked(None)
  202. return False
  203. try:
  204. with pdbuddy.Sink(self.serial_port) as pdbs:
  205. pdbs.send_command("")
  206. return True
  207. except:
  208. self.selectlist.reload()
  209. self.on_header_sink_back_clicked(None)
  210. return False
  211. def on_header_sink_back_clicked(self, data):
  212. self.serial_port = None
  213. # Show the Select page
  214. hst = self.builder.get_object("header-stack")
  215. hselect = self.builder.get_object("header-select")
  216. hst.set_visible_child(hselect)
  217. st = self.builder.get_object("stack")
  218. select = self.builder.get_object("select")
  219. st.set_visible_child(select)
  220. def on_sink_save_clicked(self, button):
  221. window = self.builder.get_object("pdb-window")
  222. try:
  223. with pdbuddy.Sink(self.serial_port) as pdbs:
  224. pdbs.set_tmpcfg(self.cfg)
  225. pdbs.write()
  226. self._store_device_settings()
  227. self._set_save_button_visibility()
  228. except OSError as e:
  229. comms_error_dialog(window, e)
  230. self.on_header_sink_back_clicked(None)
  231. def _store_device_settings(self):
  232. """Store the settings that were loaded from the device"""
  233. self.cfg_clean = self.cfg
  234. def _set_save_button_visibility(self):
  235. """Show the save button if there are new settings to save"""
  236. # Get relevant widgets
  237. rev = self.builder.get_object("sink-save-revealer")
  238. # Set visibility
  239. rev.set_reveal_child(self.cfg != self.cfg_clean)
  240. def on_voltage_combobox_changed(self, combo):
  241. self.cfg = self.cfg._replace(v=int(combo.get_active_text()) * 1000)
  242. self._set_save_button_visibility()
  243. def on_current_spinbutton_changed(self, spin):
  244. self.cfg = self.cfg._replace(i=int(spin.get_value() * 1000))
  245. self._set_save_button_visibility()
  246. def on_giveback_toggle_toggled(self, toggle):
  247. if toggle.get_active():
  248. self.cfg = self.cfg._replace(flags=self.cfg.flags|pdbuddy.SinkFlags.GIVEBACK)
  249. else:
  250. self.cfg = self.cfg._replace(flags=self.cfg.flags&~pdbuddy.SinkFlags.GIVEBACK)
  251. self._set_save_button_visibility()
  252. def on_output_switch_state_set(self, switch, state):
  253. with pdbuddy.Sink(self.serial_port) as pdbs:
  254. pdbs.output = state
  255. def on_source_cap_row_activated(self, box, row):
  256. # Find which row was clicked
  257. sc_row = self.builder.get_object("source-cap-row")
  258. if row != sc_row:
  259. # If it's not the source-cap-row, leave
  260. return
  261. # Get the source capabilities
  262. with pdbuddy.Sink(self.serial_port) as pdbs:
  263. caps = pdbs.get_source_cap()
  264. s = ""
  265. for i, cap in enumerate(caps):
  266. s += "PDO {}: {}".format(i+1, cap)
  267. if i < len(caps) - 1:
  268. s += "\n"
  269. if not s:
  270. s = "No Source_Capabilities"
  271. flags = Gtk.DialogFlags.DESTROY_WITH_PARENT;
  272. window = self.builder.get_object("pdb-window")
  273. dialog = Gtk.MessageDialog(window,
  274. flags,
  275. Gtk.MessageType.INFO,
  276. Gtk.ButtonsType.CLOSE,
  277. None)
  278. dialog.set_markup("<span font_family='monospace'>{}</span>".format(s))
  279. dialog.run()
  280. dialog.destroy()
  281. class Application(Gtk.Application):
  282. def __init__(self, *args, **kwargs):
  283. super().__init__(*args, application_id="com.clayhobbs.pd-buddy-gtk",
  284. **kwargs)
  285. self.window = None
  286. def do_startup(self):
  287. Gtk.Application.do_startup(self)
  288. self.builder = Gtk.Builder()
  289. self.builder.add_from_file("data/pd-buddy-gtk.ui")
  290. self.builder.connect_signals(Handler(self.builder))
  291. def do_activate(self):
  292. # We only allow a single window and raise any existing ones
  293. if not self.window:
  294. # Windows are associated with the application
  295. # when the last one is closed the application shuts down
  296. self.window = self.builder.get_object("pdb-window")
  297. self.add_window(self.window)
  298. self.window.set_wmclass("PD Buddy Configuration",
  299. "PD Buddy Configuration")
  300. self.window.present()
  301. def run():
  302. app = Application()
  303. app.run(sys.argv)
  304. if __name__ == "__main__":
  305. run()