gui.py 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. import os.path
  2. import json
  3. from subprocess import Popen
  4. from typing import Set
  5. import gi
  6. from marxbook import Store
  7. gi.require_version("Gtk", "3.0")
  8. from gi.repository import Gtk, Gdk # noqa: E402
  9. BROWSER_CMDLINE = ["firefox"]
  10. class MyWindow(Gtk.Window):
  11. def __init__(self, mxbstore: Store):
  12. super().__init__()
  13. self.builder = Gtk.Builder.new_from_file(
  14. os.path.join(os.path.dirname(__file__), "gui.ui")
  15. )
  16. self.add(self.builder.get_object("root"))
  17. self.my_accelerators = Gtk.AccelGroup()
  18. self.add_accel_group(self.my_accelerators)
  19. self.mxbstore = mxbstore
  20. self.store_dirs = Gtk.TreeStore(str, str) # dirname, basename
  21. self.store_marks = Gtk.ListStore(
  22. str, str, str, str, str
  23. ) # title, description, tags, URL, path
  24. self.mxb_import()
  25. self.filter_marks = self.store_marks.filter_new()
  26. self.filter_marks_dir: Set[str] = set()
  27. self.filter_marks.set_visible_func(self.filter_func, data=None)
  28. self.builder.get_object("tree_marks").set_model(self.filter_marks)
  29. self.builder.get_object("tree_dirs").set_model(self.store_dirs)
  30. self.init_marks_view()
  31. self.init_dirs_view()
  32. self.connect("destroy", Gtk.main_quit)
  33. self.builder.connect_signals(self)
  34. self.builder.get_object("search").grab_focus()
  35. self._add_accelerator("<ctrl>space", self.on_focus_switch)
  36. self._add_accelerator("<ctrl>a", self.on_focus_switch)
  37. self.show_all()
  38. def init_marks_view(self):
  39. renderer = Gtk.CellRendererText()
  40. for i, col in enumerate(("Title", "Description", "Tag", "URL")):
  41. # TODO: special renderer for Tag
  42. column = Gtk.TreeViewColumn(col, renderer, text=i)
  43. column.set_expand(True)
  44. column.set_resizable(True)
  45. column.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
  46. column.set_max_width(500) # XXX: determine by window size?
  47. self.builder.get_object("tree_marks").append_column(column)
  48. column = Gtk.TreeViewColumn("Path", renderer, text=3)
  49. column.set_visible(False)
  50. self.builder.get_object("tree_marks").append_column(column)
  51. targets = [("text/plain", 0, 1), ("TEXT", 0, 2)]
  52. self.builder.get_object("tree_marks").enable_model_drag_source(
  53. Gdk.ModifierType.BUTTON1_MASK, targets, Gdk.DragAction.COPY
  54. )
  55. self.builder.get_object("tree_dirs").enable_model_drag_dest(
  56. targets, Gdk.DragAction.COPY
  57. )
  58. def init_dirs_view(self):
  59. renderer = Gtk.CellRendererText()
  60. column = Gtk.TreeViewColumn("Folder", renderer, text=1)
  61. column.set_expand(True)
  62. column.set_resizable(True)
  63. column.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
  64. column.set_max_width(300) # XXX: determine by window size?
  65. column.set_min_width(200) # XXX: determine by window size?
  66. self.builder.get_object("tree_dirs").append_column(column)
  67. column = Gtk.TreeViewColumn("Parent", renderer, text=0)
  68. column.set_visible(False)
  69. self.builder.get_object("tree_marks").append_column(column)
  70. def filter_func(self, store, treeiter, data):
  71. text = self.builder.get_object("search").get_text().lower()
  72. row = store[treeiter]
  73. title, description, tags, url, path = row
  74. dirpath = os.path.dirname(path).lower()
  75. tags = json.loads(tags)
  76. if self.filter_marks_dir:
  77. if not any(path.startswith(dir + "/") for dir in self.filter_marks_dir):
  78. return False
  79. for query in text.split():
  80. if not any(
  81. query in part
  82. for part in (
  83. title.lower(),
  84. description.lower(),
  85. tags,
  86. url.lower(),
  87. dirpath,
  88. )
  89. ):
  90. return False
  91. return True
  92. def mxb_import(self):
  93. dirs = set()
  94. for bookmark in self.mxbstore:
  95. dirs.add(os.path.dirname(bookmark["Path"]))
  96. self.store_marks.append(
  97. [
  98. bookmark["Title"],
  99. bookmark["Description"],
  100. json.dumps(bookmark["Tag"]),
  101. bookmark["Url"],
  102. bookmark["Path"],
  103. ]
  104. )
  105. for d in sorted(dirs):
  106. parts = d.split("/")
  107. for n in range(len(parts)):
  108. dirs.add("/".join(parts[0 : n + 1]))
  109. iters = {}
  110. for d in sorted(dirs):
  111. parent, me = os.path.split(d)
  112. if not parent:
  113. parentiter = None
  114. else:
  115. parentiter = iters[parent]
  116. iters[d] = self.store_dirs.append(parentiter, [parent, me])
  117. def _add_accelerator(self, accelerator: str, callback):
  118. """Add a keyboard shortcut."""
  119. key, mod = Gtk.accelerator_parse(accelerator)
  120. self.my_accelerators.connect(key, mod, Gtk.AccelFlags.VISIBLE, callback)
  121. def on_search_changed(self, *args, **kwargs):
  122. self.filter_marks.refilter()
  123. def on_search_activate(self, *args, **kwargs):
  124. self.builder.get_object("tree_marks").grab_focus()
  125. def on_tree_marks_row_activated(self, view, treepath, column):
  126. row = self.filter_marks[treepath]
  127. url = row[3]
  128. cmd = BROWSER_CMDLINE + [url]
  129. Popen(cmd, preexec_fn=os.setpgrp)
  130. def tree_dirs_select_row(self, treepaths: list):
  131. rows = [self.store_dirs[path] for path in treepaths]
  132. self.filter_marks_dir = set(
  133. filter(bool, ("/".join([row[0], row[1]]).lstrip("/") for row in rows))
  134. )
  135. self.filter_marks.refilter()
  136. def on_tree_dirs_sel_changed(self, selection):
  137. model, treepaths = selection.get_selected_rows()
  138. if not treepaths:
  139. return
  140. self.tree_dirs_select_row(treepaths)
  141. def on_tree_dirs_row_activated(self, view, treepath, column):
  142. self.tree_dirs_select_row([treepath])
  143. self.builder.get_object("search").grab_focus()
  144. view.get_parent().hide()
  145. def on_tree_dirs_key_press_event(self, view, eventkey):
  146. key = eventkey.keyval
  147. model, treepath = view.get_selection().get_selected_rows()
  148. if not treepath or len(treepath) > 1:
  149. return
  150. treepath = treepath[0]
  151. if key in (Gdk.KEY_Right, Gdk.KEY_space):
  152. view.expand_row(treepath, False)
  153. return
  154. if key in (Gdk.KEY_Left, Gdk.KEY_BackSpace):
  155. view.collapse_row(treepath)
  156. return
  157. def on_focus_switch(self, *args, **kwargs):
  158. tree_dirs = self.builder.get_object("tree_dirs")
  159. if not tree_dirs.has_focus():
  160. tree_dirs.get_parent().show()
  161. tree_dirs.grab_focus()
  162. else:
  163. self.builder.get_object("search").grab_focus()
  164. tree_dirs.get_parent().hide()
  165. def on_tree_dirs_focus(self, *args):
  166. """Avoid giving focus to tree_dirs."""
  167. return True
  168. # DND {{{
  169. def on_tree_marks_drag_data_get(self, widget, drag_context, selection, info, time):
  170. source_widget = self.builder.get_object("tree_marks")
  171. treeselection = source_widget.get_selection()
  172. model, treepaths = treeselection.get_selected_rows()
  173. bookmarks = []
  174. for path in treepaths:
  175. treeiter = model.get_iter(path)
  176. bookmark_path = model.get_value(treeiter, 4)
  177. bookmarks.append((bookmark_path, path.to_string()))
  178. selection.set_text(json.dumps(bookmarks, indent=1), -1)
  179. source_widget.stop_emission_by_name("drag-data-get")
  180. def on_tree_dirs_drag_data_received(
  181. self, source_widget, drag_context, x, y, selection_data, info, time
  182. ):
  183. target_widget = self.builder.get_object("tree_dirs")
  184. bookmarks = json.loads(selection_data.get_text())
  185. if not bookmarks:
  186. return
  187. model, treepaths = target_widget.get_selection().get_selected_rows()
  188. drop_info = target_widget.get_dest_row_at_pos(x, y)
  189. if drop_info:
  190. path, position = drop_info
  191. treeiter = model.get_iter(path)
  192. parent = model.get_value(treeiter, 0)
  193. folder = model.get_value(treeiter, 1)
  194. if position == Gtk.TreeViewDropPosition.BEFORE:
  195. new_folder = parent
  196. else:
  197. new_folder = os.path.join(parent, folder)
  198. for fspath, treepath in bookmarks:
  199. self.mxbstore.move(fspath, new_folder)
  200. new_path = os.path.join(new_folder, os.path.basename(fspath))
  201. treeiter = self.filter_marks.get_iter(treepath)
  202. self.filter_marks.set_value(treeiter, 4, new_path)
  203. self.filter_marks.refilter()
  204. target_widget.stop_emission_by_name("drag-data-received")
  205. # }}}
  206. def main():
  207. s = Store()
  208. w = MyWindow(s) # noqa: F841
  209. # w.show_all()
  210. Gtk.main()
  211. if __name__ == "__main__":
  212. main()