Merge pull request #10701 from abextm/loottracker-aggregate-opt

Optimize loottracker UI
This commit is contained in:
Adam
2020-02-07 13:09:48 -05:00
committed by GitHub
3 changed files with 116 additions and 120 deletions

View File

@@ -39,13 +39,19 @@ import java.util.List;
import java.util.function.Predicate;
import javax.swing.BorderFactory;
import javax.swing.BoxLayout;
import javax.swing.ButtonGroup;
import javax.swing.ImageIcon;
import javax.swing.JButton;
import javax.swing.JLabel;
import javax.swing.JMenuItem;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.JPopupMenu;
import javax.swing.JRadioButton;
import javax.swing.JToggleButton;
import javax.swing.border.EmptyBorder;
import javax.swing.plaf.basic.BasicButtonUI;
import javax.swing.plaf.basic.BasicToggleButtonUI;
import net.runelite.client.game.ItemManager;
import net.runelite.client.ui.ColorScheme;
import net.runelite.client.ui.FontManager;
@@ -54,6 +60,7 @@ import net.runelite.client.ui.components.PluginErrorPanel;
import net.runelite.client.util.ColorUtil;
import net.runelite.client.util.ImageUtil;
import net.runelite.client.util.QuantityFormatter;
import net.runelite.client.util.SwingUtil;
import net.runelite.http.api.loottracker.LootTrackerClient;
class LootTrackerPanel extends PluginPanel
@@ -97,11 +104,11 @@ class LootTrackerPanel extends PluginPanel
// Details and navigation
private final JPanel actionsContainer = new JPanel();
private final JLabel detailsTitle = new JLabel();
private final JLabel backBtn = new JLabel();
private final JLabel viewHiddenBtn = new JLabel();
private final JLabel singleLootBtn = new JLabel();
private final JLabel groupedLootBtn = new JLabel();
private final JLabel collapseBtn = new JLabel();
private final JButton backBtn = new JButton();
private final JToggleButton viewHiddenBtn = new JToggleButton();
private final JRadioButton singleLootBtn = new JRadioButton();
private final JRadioButton groupedLootBtn = new JRadioButton();
private final JButton collapseBtn = new JButton();
// Aggregate of all kills
private final List<LootTrackerRecord> aggregateRecords = new ArrayList<>();
@@ -173,121 +180,64 @@ class LootTrackerPanel extends PluginPanel
final JPanel viewControls = new JPanel(new GridLayout(1, 3, 10, 0));
viewControls.setBackground(ColorScheme.DARKER_GRAY_COLOR);
SwingUtil.removeButtonDecorations(collapseBtn);
collapseBtn.setIcon(EXPAND_ICON);
collapseBtn.addMouseListener(new MouseAdapter()
{
@Override
public void mousePressed(MouseEvent e)
{
changeCollapse();
}
});
singleLootBtn.setIcon(SINGLE_LOOT_VIEW);
singleLootBtn.setToolTipText("Show each kill separately");
singleLootBtn.addMouseListener(new MouseAdapter()
{
@Override
public void mousePressed(MouseEvent mouseEvent)
{
changeGrouping(false);
}
@Override
public void mouseExited(MouseEvent mouseEvent)
{
singleLootBtn.setIcon(groupLoot ? SINGLE_LOOT_VIEW_FADED : SINGLE_LOOT_VIEW);
}
@Override
public void mouseEntered(MouseEvent mouseEvent)
{
singleLootBtn.setIcon(groupLoot ? SINGLE_LOOT_VIEW_HOVER : SINGLE_LOOT_VIEW);
}
});
groupedLootBtn.setIcon(GROUPED_LOOT_VIEW);
groupedLootBtn.setToolTipText("Group loot by source");
groupedLootBtn.addMouseListener(new MouseAdapter()
{
@Override
public void mousePressed(MouseEvent mouseEvent)
{
changeGrouping(true);
}
@Override
public void mouseExited(MouseEvent mouseEvent)
{
groupedLootBtn.setIcon(groupLoot ? GROUPED_LOOT_VIEW : GROUPED_LOOT_VIEW_FADED);
}
@Override
public void mouseEntered(MouseEvent mouseEvent)
{
groupedLootBtn.setIcon(groupLoot ? GROUPED_LOOT_VIEW : GROUPED_LOOT_VIEW_HOVER);
}
});
viewHiddenBtn.setIcon(VISIBLE_ICON);
viewHiddenBtn.setToolTipText("Show ignored items");
viewHiddenBtn.addMouseListener(new MouseAdapter()
{
@Override
public void mousePressed(MouseEvent mouseEvent)
{
changeItemHiding(!hideIgnoredItems);
}
@Override
public void mouseExited(MouseEvent mouseEvent)
{
viewHiddenBtn.setIcon(hideIgnoredItems ? INVISIBLE_ICON : VISIBLE_ICON);
}
@Override
public void mouseEntered(MouseEvent mouseEvent)
{
viewHiddenBtn.setIcon(hideIgnoredItems ? INVISIBLE_ICON_HOVER : VISIBLE_ICON_HOVER);
}
});
collapseBtn.setSelectedIcon(COLLAPSE_ICON);
SwingUtil.addModalTooltip(collapseBtn, "Collapse All", "Un-Collapse All");
collapseBtn.setBackground(ColorScheme.DARKER_GRAY_COLOR);
collapseBtn.setUI(new BasicButtonUI()); // substance breaks the layout
collapseBtn.addActionListener(ev -> changeCollapse());
viewControls.add(collapseBtn);
SwingUtil.removeButtonDecorations(singleLootBtn);
singleLootBtn.setIcon(SINGLE_LOOT_VIEW_FADED);
singleLootBtn.setRolloverIcon(SINGLE_LOOT_VIEW_HOVER);
singleLootBtn.setSelectedIcon(SINGLE_LOOT_VIEW);
singleLootBtn.setToolTipText("Show each kill separately");
singleLootBtn.addActionListener(e -> changeGrouping(false));
SwingUtil.removeButtonDecorations(groupedLootBtn);
groupedLootBtn.setIcon(GROUPED_LOOT_VIEW_FADED);
groupedLootBtn.setRolloverIcon(GROUPED_LOOT_VIEW_HOVER);
groupedLootBtn.setSelectedIcon(GROUPED_LOOT_VIEW);
groupedLootBtn.setToolTipText("Group loot by source");
groupedLootBtn.addActionListener(e -> changeGrouping(true));
ButtonGroup groupSingleGroup = new ButtonGroup();
groupSingleGroup.add(singleLootBtn);
groupSingleGroup.add(groupedLootBtn);
viewControls.add(groupedLootBtn);
viewControls.add(singleLootBtn);
viewControls.add(viewHiddenBtn);
changeGrouping(true);
SwingUtil.removeButtonDecorations(viewHiddenBtn);
viewHiddenBtn.setIconTextGap(0);
viewHiddenBtn.setIcon(VISIBLE_ICON);
viewHiddenBtn.setRolloverIcon(INVISIBLE_ICON_HOVER);
viewHiddenBtn.setSelectedIcon(INVISIBLE_ICON);
viewHiddenBtn.setRolloverSelectedIcon(VISIBLE_ICON_HOVER);
viewHiddenBtn.setBackground(ColorScheme.DARKER_GRAY_COLOR);
viewHiddenBtn.setUI(new BasicToggleButtonUI()); // substance breaks the layout and the pressed icon
SwingUtil.addModalTooltip(viewHiddenBtn, "Show ignored items", "Hide ignored items");
changeItemHiding(true);
viewControls.add(viewHiddenBtn);
final JPanel leftTitleContainer = new JPanel(new BorderLayout(5, 0));
leftTitleContainer.setBackground(ColorScheme.DARKER_GRAY_COLOR);
detailsTitle.setForeground(Color.WHITE);
SwingUtil.removeButtonDecorations(backBtn);
backBtn.setIcon(BACK_ARROW_ICON);
backBtn.setRolloverIcon(BACK_ARROW_ICON_HOVER);
backBtn.setVisible(false);
backBtn.addMouseListener(new MouseAdapter()
backBtn.addActionListener(ev ->
{
@Override
public void mousePressed(MouseEvent mouseEvent)
{
currentView = null;
backBtn.setVisible(false);
detailsTitle.setText("");
rebuild();
}
@Override
public void mouseExited(MouseEvent mouseEvent)
{
backBtn.setIcon(BACK_ARROW_ICON);
}
@Override
public void mouseEntered(MouseEvent mouseEvent)
{
backBtn.setIcon(BACK_ARROW_ICON_HOVER);
}
currentView = null;
backBtn.setVisible(false);
detailsTitle.setText("");
rebuild();
});
leftTitleContainer.add(backBtn, BorderLayout.WEST);
@@ -367,16 +317,7 @@ class LootTrackerPanel extends PluginPanel
void updateCollapseText()
{
if (isAllCollapsed())
{
collapseBtn.setToolTipText("Un-Collapse All");
collapseBtn.setIcon(COLLAPSE_ICON);
}
else
{
collapseBtn.setToolTipText("Collapse All");
collapseBtn.setIcon(EXPAND_ICON);
}
collapseBtn.setSelected(isAllCollapsed());
}
private boolean isAllCollapsed()
@@ -427,9 +368,8 @@ class LootTrackerPanel extends PluginPanel
private void changeGrouping(boolean group)
{
groupLoot = group;
(group ? groupedLootBtn : singleLootBtn).setSelected(true);
rebuild();
groupedLootBtn.setIcon(group ? GROUPED_LOOT_VIEW : GROUPED_LOOT_VIEW_FADED);
singleLootBtn.setIcon(group ? SINGLE_LOOT_VIEW_FADED : SINGLE_LOOT_VIEW);
}
/**
@@ -440,8 +380,8 @@ class LootTrackerPanel extends PluginPanel
private void changeItemHiding(boolean hide)
{
hideIgnoredItems = hide;
viewHiddenBtn.setSelected(hide);
rebuild();
viewHiddenBtn.setIcon(hideIgnoredItems ? VISIBLE_ICON : INVISIBLE_ICON);
}
/**
@@ -488,7 +428,7 @@ class LootTrackerPanel extends PluginPanel
*/
private void rebuild()
{
logsContainer.removeAll();
SwingUtil.fastRemoveAll(logsContainer);
boxes.clear();
if (groupLoot)

View File

@@ -612,7 +612,7 @@ public class LootTrackerPlugin extends Plugin
}
config.setIgnoredItems(Text.toCSV(ignoredItemSet));
panel.updateIgnoredRecords();
// the config changed will update the panel
}
boolean isIgnored(String name)

View File

@@ -26,11 +26,16 @@ package net.runelite.client.util;
import java.awt.AWTException;
import java.awt.Color;
import java.awt.Component;
import java.awt.Container;
import java.awt.EventQueue;
import java.awt.Font;
import java.awt.Frame;
import java.awt.Image;
import java.awt.Insets;
import java.awt.SecondaryLoop;
import java.awt.SystemTray;
import java.awt.Toolkit;
import java.awt.TrayIcon;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
@@ -50,6 +55,7 @@ import javax.swing.JMenuItem;
import javax.swing.JOptionPane;
import javax.swing.JPopupMenu;
import javax.swing.LookAndFeel;
import javax.swing.SwingUtilities;
import javax.swing.ToolTipManager;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
@@ -293,4 +299,54 @@ public class SwingUtil
{
button.addItemListener(l -> button.setToolTipText(button.isSelected() ? on : off));
}
/**
* Removes all of a component's children faster than calling removeAll() on it in many cases
*/
public static void fastRemoveAll(Container c)
{
// If we are not on the EDT this will deadlock, in addition to being totally unsafe
assert SwingUtilities.isEventDispatchThread();
// when a component is removed it has to be resized for some reason, but only if it's valid
// so we make sure to invalidate everything before removing it
c.invalidate();
for (int i = 0; i < c.getComponentCount(); i++)
{
Component ic = c.getComponent(i);
// removeAll and removeNotify are both recursive, so we have to recurse before them
if (ic instanceof Container)
{
fastRemoveAll((Container) ic);
}
// each removeNotify needs to remove anything from the event queue that is for that widget
// this however requires taking a lock, and is moderately slow, so we just execute all of
// those events with a secondary event loop
pumpPendingEvents();
// call removeNotify early; this is most of the work in removeAll, and generates events that
// the next secondaryLoop will pickup
ic.removeNotify();
}
// Actually remove anything
c.removeAll();
}
/**
* Run any events currently in the event queue
*/
public static void pumpPendingEvents()
{
EventQueue eq = Toolkit.getDefaultToolkit().getSystemEventQueue();
if (eq.peekEvent() != null)
{
SecondaryLoop l = eq.createSecondaryLoop();
SwingUtilities.invokeLater(l::exit);
l.enter();
}
}
}