All Downloads are FREE. Search and download functionalities are using the official Maven repository.

dorkbox.util.swing.SwingActiveRender Maven / Gradle / Ivy

package dorkbox.util.swing;

import dorkbox.util.ActionHandlerLong;

import javax.swing.JComponent;
import javax.swing.JFrame;
import java.awt.Component;
import java.awt.EventQueue;
import java.util.ArrayDeque;
import java.util.Collections;
import java.util.Deque;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;

/**
 * Contains all of the appropriate logic to setup and render via "Active" rendering (instead of "Passive" rendering). This permits us to
 * render JFrames (and their contents), OFF of the EDT - even though there are other frames/components that are ON the EDT. 
Because we * still want to react to mouse events, etc on the EDT, we do not completely remove the EDT -- we merely allow us to "synchronize" the EDT * object to our thread. It's a little bit hacky, but it works beautifully, and permits MUCH nicer animations.
*

* It is also important to REMEMBER -- if you add a component to an actively managed JFrame, YOU MUST make sure to call {@link * JComponent#setIgnoreRepaint(boolean)} otherwise this component will "fight" on the EDT for updates. */ public final class SwingActiveRender { private static Thread activeRenderThread = null; static final List activeRenders = new CopyOnWriteArrayList(); static final List activeRenderEvents = new CopyOnWriteArrayList(); // volatile, so that access triggers thread synchrony, since 1.6. See the Java Language Spec, Chapter 17 static volatile boolean hasActiveRenders = false; private static final Runnable renderLoop = new ActiveRenderLoop(); private SwingActiveRender() { } /** * Enables the jFrame to to added to an "Active Render" thread, at a target "Frames-per-second". This is to support smooth, swing-based * animations.
This works by removing this object from EDT updates, and instead manually calls paint(g) on the jFrame, updating it * on our own thread. * * @param jFrame * the jFrame to add to the ActiveRender thread. */ public static void addActiveRender(final JFrame jFrame) { // this should be on the EDT if (!EventQueue.isDispatchThread()) { throw new RuntimeException("adding a swing JFrame to be actively rendered, must be done on the EDT."); } // setup double-buffering, so we can properly use Active-Rendering, so the animations will be smooth jFrame.createBufferStrategy(2); // have to specify ALL children in jFrame to ignore EDT paint requests Deque components = new ArrayDeque(); components.add(jFrame); Component[] c; Component pop; while ((pop = components.poll()) != null) { pop.setIgnoreRepaint(true); if (pop instanceof JComponent) { c = ((JComponent) pop).getComponents(); Collections.addAll(components, c); } } synchronized (activeRenders) { if (!hasActiveRenders) { setupActiveRenderThread(); } hasActiveRenders = true; activeRenders.add(jFrame); } } /** * Specifies an ActionHandler to be called when the ActiveRender thread starts to render at each tick. * * @param handler * the handler to add */ public static void addActiveRenderFrameStart(final ActionHandlerLong handler) { synchronized (activeRenders) { activeRenderEvents.add(handler); } } /** * Potentially SLOW calculation, as it compares each entry in a queue for equality * * @param handler * this is the handler to check * * @return true if this handler already exists in the active render, on-frame-start queue */ public static boolean containsActiveRenderFrameStart(final ActionHandlerLong handler) { synchronized (activeRenders) { return activeRenderEvents.contains(handler); } } /** * Removes the handler from the on-frame-start queue * * @param handler * the handler to remove */ public static void removeActiveRenderFrameStart(final ActionHandlerLong handler) { synchronized (activeRenders) { activeRenderEvents.remove(handler); } } /** * Removes a jFrame from the ActiveRender queue. This should happen when the jFrame is closed. * * @param jFrame * the jFrame to remove */ public static void removeActiveRender(final JFrame jFrame) { synchronized (activeRenders) { activeRenders.remove(jFrame); final boolean hadActiveRenders = !activeRenders.isEmpty(); hasActiveRenders = hadActiveRenders; if (!hadActiveRenders) { activeRenderThread = null; } } } /** * Creates (if necessary) the active-render thread. When there are no active-render targets, this thread will exit */ private static void setupActiveRenderThread() { if (activeRenderThread != null) { return; } SynchronizedEventQueue.install(); activeRenderThread = new Thread(renderLoop, "AWT-ActiveRender"); activeRenderThread.setDaemon(true); activeRenderThread.start(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy