You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

TreeFrameManager.java 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361
  1. /*
  2. * Copyright (c) 2006-2017 DMDirc Developers
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated
  5. * documentation files (the "Software"), to deal in the Software without restriction, including without limitation the
  6. * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to
  7. * permit persons to whom the Software is furnished to do so, subject to the following conditions:
  8. *
  9. * The above copyright notice and this permission notice shall be included in all copies or substantial portions of the
  10. * Software.
  11. *
  12. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
  13. * WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS
  14. * OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
  15. * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  16. */
  17. package com.dmdirc.addons.ui_swing.framemanager.tree;
  18. import com.dmdirc.addons.ui_swing.EdtHandlerInvocation;
  19. import com.dmdirc.addons.ui_swing.SwingController;
  20. import com.dmdirc.addons.ui_swing.SwingWindowFactory;
  21. import com.dmdirc.addons.ui_swing.UIUtilities;
  22. import com.dmdirc.addons.ui_swing.components.IconManager;
  23. import com.dmdirc.addons.ui_swing.components.TreeScroller;
  24. import com.dmdirc.addons.ui_swing.components.frames.TextFrame;
  25. import com.dmdirc.addons.ui_swing.events.SwingEventBus;
  26. import com.dmdirc.addons.ui_swing.events.SwingWindowAddedEvent;
  27. import com.dmdirc.addons.ui_swing.events.SwingWindowDeletedEvent;
  28. import com.dmdirc.addons.ui_swing.events.SwingWindowSelectedEvent;
  29. import com.dmdirc.addons.ui_swing.framemanager.FrameManager;
  30. import com.dmdirc.addons.ui_swing.interfaces.ActiveFrameManager;
  31. import com.dmdirc.config.GlobalConfig;
  32. import com.dmdirc.events.FrameIconChangedEvent;
  33. import com.dmdirc.events.UnreadStatusChangedEvent;
  34. import com.dmdirc.events.eventbus.EventBus;
  35. import com.dmdirc.interfaces.WindowModel;
  36. import com.dmdirc.interfaces.config.AggregateConfigProvider;
  37. import com.dmdirc.interfaces.config.ConfigChangeListener;
  38. import com.dmdirc.plugins.PluginDomain;
  39. import com.dmdirc.ui.WindowManager;
  40. import com.dmdirc.ui.messages.ColourManager;
  41. import com.dmdirc.util.LogUtils;
  42. import java.awt.Color;
  43. import java.awt.Rectangle;
  44. import java.awt.event.MouseEvent;
  45. import java.io.Serializable;
  46. import java.util.Collection;
  47. import java.util.HashMap;
  48. import java.util.Map;
  49. import javax.inject.Inject;
  50. import javax.swing.JComponent;
  51. import javax.swing.JScrollPane;
  52. import javax.swing.JTree;
  53. import javax.swing.ScrollPaneConstants;
  54. import javax.swing.SwingUtilities;
  55. import javax.swing.tree.DefaultMutableTreeNode;
  56. import javax.swing.tree.DefaultTreeModel;
  57. import javax.swing.tree.MutableTreeNode;
  58. import javax.swing.tree.TreeNode;
  59. import javax.swing.tree.TreePath;
  60. import net.engio.mbassy.listener.Handler;
  61. import net.engio.mbassy.listener.Invoke;
  62. import net.miginfocom.swing.MigLayout;
  63. import org.slf4j.Logger;
  64. import org.slf4j.LoggerFactory;
  65. /**
  66. * Manages open windows in the application in a tree style view.
  67. */
  68. public class TreeFrameManager implements FrameManager, Serializable, ConfigChangeListener {
  69. private static final Logger LOG = LoggerFactory.getLogger(TreeFrameManager.class);
  70. /** Serial version UID. */
  71. private static final long serialVersionUID = 5;
  72. /** node storage, used for adding and deleting nodes correctly. */
  73. private final Map<TextFrame, TreeViewNode> nodes;
  74. /** Configuration manager. */
  75. private final AggregateConfigProvider config;
  76. /** Colour manager. */
  77. private final ColourManager colourManager;
  78. /** Factory to use to retrieve swing windows. */
  79. private final SwingWindowFactory windowFactory;
  80. /** Window manage. */
  81. private final WindowManager windowManager;
  82. /** Active frame manager. */
  83. private final ActiveFrameManager activeFrameManager;
  84. /** The event bus to post errors to. */
  85. private final EventBus eventBus;
  86. /** Swing event bus. */
  87. private final SwingEventBus swingEventBus;
  88. /** Icon manager. */
  89. private final IconManager iconManager;
  90. /** display tree. */
  91. private Tree tree;
  92. /** data model. */
  93. private TreeViewModel model;
  94. /** Tree scroller. */
  95. private TreeScroller scroller;
  96. @Inject
  97. public TreeFrameManager(final WindowManager windowManager,
  98. @GlobalConfig final AggregateConfigProvider globalConfig,
  99. @GlobalConfig final ColourManager colourManager,
  100. final ActiveFrameManager activeFrameManager,
  101. final SwingWindowFactory windowFactory,
  102. @PluginDomain(SwingController.class) final String domain,
  103. final EventBus eventBus,
  104. final SwingEventBus swingEventBus,
  105. final IconManager iconManager) {
  106. this.windowFactory = windowFactory;
  107. this.windowManager = windowManager;
  108. this.nodes = new HashMap<>();
  109. this.config = globalConfig;
  110. this.colourManager = colourManager;
  111. this.activeFrameManager = activeFrameManager;
  112. this.eventBus = eventBus;
  113. this.swingEventBus = swingEventBus;
  114. this.iconManager = iconManager;
  115. UIUtilities.invokeLater(() -> {
  116. model = new TreeViewModel(config, new TreeViewNode(null, null));
  117. tree = new Tree(this, model, swingEventBus, globalConfig, domain);
  118. tree.setCellRenderer(
  119. new TreeViewTreeCellRenderer(config, colourManager, this));
  120. tree.setVisible(true);
  121. config.addChangeListener("treeview", this);
  122. config.addChangeListener("ui", "sortrootwindows", this);
  123. config.addChangeListener("ui", "sortchildwindows", this);
  124. config.addChangeListener("ui", "backgroundcolour", this);
  125. config.addChangeListener("ui", "foregroundcolour", this);
  126. });
  127. }
  128. @Override
  129. public boolean canPositionVertically() {
  130. return true;
  131. }
  132. @Override
  133. public boolean canPositionHorizontally() {
  134. return false;
  135. }
  136. @Override
  137. public void setParent(final JComponent parent) {
  138. SwingUtilities.invokeLater(() -> {
  139. final JScrollPane scrollPane = new JScrollPane(tree);
  140. scrollPane.setAutoscrolls(true);
  141. scrollPane.setHorizontalScrollBarPolicy(
  142. ScrollPaneConstants.HORIZONTAL_SCROLLBAR_NEVER);
  143. parent.setVisible(false);
  144. parent.setLayout(new MigLayout("ins 0, fill"));
  145. parent.add(scrollPane, "grow");
  146. parent.setFocusable(false);
  147. parent.setVisible(true);
  148. setColours();
  149. eventBus.subscribe(this);
  150. swingEventBus.subscribe(this);
  151. redoTreeView();
  152. });
  153. }
  154. @Handler
  155. public void doAddWindow(final SwingWindowAddedEvent event) {
  156. final TextFrame parent = event.getParentWindow().orElse(null);
  157. final TextFrame window = event.getChildWindow();
  158. if (nodes.containsKey(window)) {
  159. return;
  160. }
  161. if (parent == null) {
  162. addWindow(model.getRootNode(), window);
  163. } else {
  164. addWindow(nodes.get(parent), window);
  165. }
  166. }
  167. @Handler
  168. public void doDeleteWindow(final SwingWindowDeletedEvent event) {
  169. final TextFrame window = event.getChildWindow();
  170. UIUtilities.invokeAndWait(() -> {
  171. if (nodes.get(window) == null) {
  172. return;
  173. }
  174. final DefaultMutableTreeNode node = nodes.get(window);
  175. if (node.getLevel() == 0) {
  176. LOG.warn(LogUtils.USER_ERROR, "delServer triggered for root node {}",
  177. node, new IllegalArgumentException());
  178. } else {
  179. model.removeNodeFromParent(nodes.get(window));
  180. }
  181. synchronized (nodes) {
  182. eventBus.unsubscribe(nodes.get(window).getLabel());
  183. nodes.remove(window);
  184. }
  185. });
  186. }
  187. /**
  188. * Adds a window to the frame container.
  189. *
  190. * @param parent Parent node
  191. * @param window Window to add
  192. */
  193. public void addWindow(final MutableTreeNode parent, final TextFrame window) {
  194. UIUtilities.invokeAndWait(() -> {
  195. final NodeLabel label = new NodeLabel(window, iconManager, getForegroundColour());
  196. eventBus.subscribe(label);
  197. swingEventBus.subscribe(label);
  198. final TreeViewNode node = new TreeViewNode(label, window);
  199. synchronized (nodes) {
  200. nodes.put(window, node);
  201. }
  202. if (parent == null) {
  203. model.insertNodeInto(node, model.getRootNode());
  204. } else {
  205. model.insertNodeInto(node, parent);
  206. }
  207. tree.expandPath(new TreePath(node.getPath()).getParentPath());
  208. final Rectangle view = tree.getRowBounds(tree.getRowForPath(new TreePath(node.
  209. getPath())));
  210. if (view != null) {
  211. tree.scrollRectToVisible(new Rectangle(0, (int) view.getY(), 0, 0));
  212. }
  213. refreshNodeLabel(window, node);
  214. node.getLabel().iconChanged(new FrameIconChangedEvent(window.getContainer(),
  215. window.getContainer().getIcon()));
  216. });
  217. }
  218. /**
  219. * Returns the tree for this frame manager.
  220. *
  221. * @return Tree for the manager
  222. */
  223. public JTree getTree() {
  224. return tree;
  225. }
  226. /**
  227. * Checks for and sets a rollover node.
  228. *
  229. * @param event event to check
  230. */
  231. protected void checkRollover(final MouseEvent event) {
  232. NodeLabel node = null;
  233. if (event != null && tree.getNodeForLocation(event.getX(), event.getY()) != null) {
  234. node = tree.getNodeForLocation(event.getX(), event.getY()).getLabel();
  235. }
  236. synchronized (nodes) {
  237. for (TreeViewNode treeNode : nodes.values()) {
  238. final NodeLabel label = treeNode.getLabel();
  239. label.setRollover(label == node);
  240. }
  241. }
  242. tree.repaint();
  243. }
  244. /** Sets treeview colours. */
  245. private void setColours() {
  246. final Color foregroundColour = getForegroundColour();
  247. tree.setForeground(foregroundColour);
  248. tree.setBackground(UIUtilities.convertColour(colourManager.getColourFromString(
  249. config.getOptionString("treeview", "backgroundcolour", "ui",
  250. "backgroundcolour"), null)));
  251. tree.repaint();
  252. nodes.entrySet().forEach(pair -> {
  253. pair.getValue().getLabel().setDefaultForegroundColour(foregroundColour);
  254. refreshNodeLabel(pair.getKey(), pair.getValue());
  255. });
  256. }
  257. private Color getForegroundColour() {
  258. return UIUtilities.convertColour(colourManager.getColourFromString(
  259. config.getOptionString("treeview", "foregroundcolour", "ui",
  260. "foregroundcolour"), null));
  261. }
  262. private void refreshNodeLabel(TextFrame window, TreeViewNode node) {
  263. node.getLabel().unreadStatusChanged(new UnreadStatusChangedEvent(
  264. window.getContainer(),
  265. window.getContainer().getUnreadStatusManager(),
  266. window.getContainer().getUnreadStatusManager().getNotificationColour(),
  267. window.getContainer().getUnreadStatusManager().getUnreadLines()));
  268. }
  269. @Override
  270. public void configChanged(final String domain, final String key) {
  271. if ("sortrootwindows".equals(key) || "sortchildwindows".equals(key)) {
  272. redoTreeView();
  273. } else {
  274. setColours();
  275. }
  276. }
  277. /**
  278. * Starts the tree from scratch taking into account new sort orders.
  279. */
  280. private void redoTreeView() {
  281. UIUtilities.invokeLater(() -> {
  282. ((DefaultTreeModel) tree.getModel()).setRoot(null);
  283. ((DefaultTreeModel) tree.getModel()).setRoot(new TreeViewNode(null, null));
  284. if (scroller != null) {
  285. scroller.unregister();
  286. }
  287. scroller = new TreeTreeScroller(swingEventBus, tree);
  288. for (WindowModel window : windowManager.getRootWindows()) {
  289. addWindow(null, windowFactory.getSwingWindow(window));
  290. final Collection<WindowModel> childWindows = windowManager.getChildren(window);
  291. for (WindowModel childWindow : childWindows) {
  292. addWindow(nodes.get(windowFactory.getSwingWindow(window)),
  293. windowFactory.getSwingWindow(childWindow));
  294. }
  295. }
  296. if (activeFrameManager.getActiveFrame() != null) {
  297. selectionChanged(new SwingWindowSelectedEvent(activeFrameManager.getActiveFrame()));
  298. }
  299. });
  300. }
  301. @Handler(invocation = EdtHandlerInvocation.class)
  302. public void selectionChanged(final SwingWindowSelectedEvent event) {
  303. if (event.getWindow().isPresent()) {
  304. UIUtilities.invokeLater(() -> {
  305. final TreeNode[] treePath = ((DefaultTreeModel) tree.getModel())
  306. .getPathToRoot(nodes.get(event.getWindow().get()));
  307. if (treePath != null && treePath.length > 0) {
  308. final TreePath path = new TreePath(treePath);
  309. tree.setTreePath(path);
  310. tree.scrollPathToVisible(path);
  311. tree.repaint();
  312. }
  313. });
  314. }
  315. }
  316. @Handler(invocation = EdtHandlerInvocation.class, delivery = Invoke.Asynchronously)
  317. public void unreadStatusChanged(final UnreadStatusChangedEvent event) {
  318. synchronized (nodes) {
  319. final TreeViewNode node = nodes.get(windowFactory.getSwingWindow(event.getSource()));
  320. if (node != null) {
  321. final NodeLabel label = node.getLabel();
  322. if (label != null) {
  323. label.unreadStatusChanged(event);
  324. tree.repaint();
  325. }
  326. }
  327. }
  328. }
  329. }