Avoid notification flickering by moving windows in reverse order

This commit is contained in:
Reinhard Pointner 2016-03-11 08:16:54 +00:00
parent d53dd1ffec
commit 0a039d943b
5 changed files with 46 additions and 47 deletions

View File

@ -23,7 +23,7 @@ public class NotificationHandler extends Handler {
private final NotificationManager manager; private final NotificationManager manager;
public NotificationHandler(String title) { public NotificationHandler(String title) {
this(title, 2500, new NotificationManager(new QueueNotificationLayout(NORTH, SOUTH))); this(title, 2500, new NotificationManager(new QueueNotificationLayout(NORTH, SOUTH), 5));
} }
public NotificationHandler(String title, int timeout, NotificationManager manager) { public NotificationHandler(String title, int timeout, NotificationManager manager) {

View File

@ -4,11 +4,12 @@
package net.filebot.util.ui.notification; package net.filebot.util.ui.notification;
public interface NotificationLayout { public interface NotificationLayout {
public void add(NotificationWindow notification); public void add(NotificationWindow notification);
public void remove(NotificationWindow notification); public void remove(NotificationWindow notification);
public int size();
} }

View File

@ -4,38 +4,35 @@
package net.filebot.util.ui.notification; package net.filebot.util.ui.notification;
import java.awt.event.WindowAdapter; import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent; import java.awt.event.WindowEvent;
import net.filebot.util.ui.SwingUI; import net.filebot.util.ui.SwingUI;
public class NotificationManager { public class NotificationManager {
private final NotificationLayout layout; private final NotificationLayout layout;
private final int limit;
public NotificationManager() { public NotificationManager() {
this(new QueueNotificationLayout()); this(new QueueNotificationLayout(), 5);
} }
public NotificationManager(NotificationLayout layout, int limit) {
public NotificationManager(NotificationLayout layout) {
this.layout = layout; this.layout = layout;
this.limit = limit;
} }
public void show(NotificationWindow notification) { public void show(NotificationWindow notification) {
SwingUI.checkEventDispatchThread(); SwingUI.checkEventDispatchThread();
notification.addWindowListener(new RemoveListener()); if (layout.size() < limit) {
layout.add(notification); layout.add(notification);
notification.addWindowListener(new RemoveListener());
notification.setVisible(true); notification.setVisible(true);
}
} }
private class RemoveListener extends WindowAdapter { private class RemoveListener extends WindowAdapter {
@Override @Override

View File

@ -4,7 +4,6 @@
package net.filebot.util.ui.notification; package net.filebot.util.ui.notification;
import static net.filebot.util.ui.notification.Direction.*; import static net.filebot.util.ui.notification.Direction.*;
import java.awt.Dimension; import java.awt.Dimension;
@ -13,9 +12,9 @@ import java.awt.Insets;
import java.awt.Point; import java.awt.Point;
import java.awt.Toolkit; import java.awt.Toolkit;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Iterator;
import java.util.List; import java.util.List;
public class QueueNotificationLayout implements NotificationLayout { public class QueueNotificationLayout implements NotificationLayout {
private final List<NotificationWindow> notifications = new ArrayList<NotificationWindow>(); private final List<NotificationWindow> notifications = new ArrayList<NotificationWindow>();
@ -24,26 +23,22 @@ public class QueueNotificationLayout implements NotificationLayout {
private final Direction direction; private final Direction direction;
private final Direction growAnchor; private final Direction growAnchor;
public QueueNotificationLayout() { public QueueNotificationLayout() {
this(SOUTH_EAST, WEST); this(SOUTH_EAST, WEST);
} }
public QueueNotificationLayout(Direction alignment, Direction direction) { public QueueNotificationLayout(Direction alignment, Direction direction) {
this.alignment = alignment; this.alignment = alignment;
this.growAnchor = alignment; this.growAnchor = alignment;
this.direction = direction; this.direction = direction;
} }
public QueueNotificationLayout(Direction orientation, Direction direction, Direction growAnchor) { public QueueNotificationLayout(Direction orientation, Direction direction, Direction growAnchor) {
this.alignment = orientation; this.alignment = orientation;
this.direction = direction; this.direction = direction;
this.growAnchor = growAnchor; this.growAnchor = growAnchor;
} }
private Point getBaseAnchor(Dimension screen, Insets insets) { private Point getBaseAnchor(Dimension screen, Insets insets) {
Point p = new Point(); Point p = new Point();
@ -59,7 +54,6 @@ public class QueueNotificationLayout implements NotificationLayout {
return p; return p;
} }
private Point getLocation(Point anchor, Dimension size) { private Point getLocation(Point anchor, Dimension size) {
Point p = new Point(); Point p = new Point();
@ -69,7 +63,6 @@ public class QueueNotificationLayout implements NotificationLayout {
return p; return p;
} }
private Point getNextAnchor(Point anchor, Dimension size) { private Point getNextAnchor(Point anchor, Dimension size) {
Point p = new Point(); Point p = new Point();
@ -79,30 +72,34 @@ public class QueueNotificationLayout implements NotificationLayout {
return p; return p;
} }
@Override @Override
public void add(NotificationWindow notification) { public void add(NotificationWindow notification) {
notifications.add(notification); notifications.add(notification);
align(notification.getGraphicsConfiguration()); align(notification.getGraphicsConfiguration());
} }
private void align(GraphicsConfiguration gc) { private void align(GraphicsConfiguration gc) {
Dimension screen = Toolkit.getDefaultToolkit().getScreenSize(); Dimension screen = Toolkit.getDefaultToolkit().getScreenSize();
Insets insets = Toolkit.getDefaultToolkit().getScreenInsets(gc); Insets insets = Toolkit.getDefaultToolkit().getScreenInsets(gc);
// avoid flickering by moving windows in reverse order
Point anchor = getBaseAnchor(screen, insets); Point anchor = getBaseAnchor(screen, insets);
align(anchor, notifications.iterator());
for (NotificationWindow window : notifications) {
Dimension size = window.getSize();
Point p = getLocation(anchor, size);
window.setLocation(p);
anchor = getNextAnchor(anchor, size);
}
} }
private void align(Point anchor, Iterator<NotificationWindow> seq) {
if (!seq.hasNext()) {
return;
}
NotificationWindow window = seq.next();
Dimension size = window.getSize();
Point p = getLocation(anchor, size);
align(getNextAnchor(anchor, size), seq);
window.setLocation(p);
}
@Override @Override
public void remove(NotificationWindow notification) { public void remove(NotificationWindow notification) {
@ -111,4 +108,9 @@ public class QueueNotificationLayout implements NotificationLayout {
} }
} }
@Override
public int size() {
return notifications.size();
}
} }

View File

@ -4,7 +4,6 @@
package net.filebot.util.ui.notification; package net.filebot.util.ui.notification;
import static net.filebot.util.ui.notification.Direction.*; import static net.filebot.util.ui.notification.Direction.*;
import java.awt.Dimension; import java.awt.Dimension;
@ -12,23 +11,19 @@ import java.awt.Insets;
import java.awt.Point; import java.awt.Point;
import java.awt.Toolkit; import java.awt.Toolkit;
public class SimpleNotificationLayout implements NotificationLayout { public class SimpleNotificationLayout implements NotificationLayout {
private NotificationWindow currentNotification; private NotificationWindow current;
private Direction alignment; private Direction alignment;
public SimpleNotificationLayout() { public SimpleNotificationLayout() {
this(NORTH); this(NORTH);
} }
public SimpleNotificationLayout(Direction alignment) { public SimpleNotificationLayout(Direction alignment) {
this.alignment = alignment; this.alignment = alignment;
} }
private Point getBaseAnchor(Dimension screen, Insets insets) { private Point getBaseAnchor(Dimension screen, Insets insets) {
Point p = new Point(); Point p = new Point();
@ -44,7 +39,6 @@ public class SimpleNotificationLayout implements NotificationLayout {
return p; return p;
} }
private Point getLocation(Point anchor, Dimension size) { private Point getLocation(Point anchor, Dimension size) {
Point p = new Point(); Point p = new Point();
@ -54,7 +48,6 @@ public class SimpleNotificationLayout implements NotificationLayout {
return p; return p;
} }
@Override @Override
public void add(NotificationWindow notification) { public void add(NotificationWindow notification) {
Dimension screen = Toolkit.getDefaultToolkit().getScreenSize(); Dimension screen = Toolkit.getDefaultToolkit().getScreenSize();
@ -64,17 +57,23 @@ public class SimpleNotificationLayout implements NotificationLayout {
Point anchor = getBaseAnchor(screen, insets); Point anchor = getBaseAnchor(screen, insets);
notification.setLocation(getLocation(anchor, size)); notification.setLocation(getLocation(anchor, size));
if (currentNotification != null) { if (current != null) {
currentNotification.close(); current.close();
} }
current = notification;
currentNotification = notification;
} }
@Override @Override
public void remove(NotificationWindow notification) { public void remove(NotificationWindow notification) {
if (current != null && current == notification) {
current.close();
}
current = null;
}
@Override
public int size() {
return current == null ? 0 : 1;
} }
} }