com.jidesoft.dialog.AbstractPage Maven / Gradle / Ivy
/*
* @(#)AbstractPage.java
*
* Copyright 2002 - 2003 JIDE Software. All rights reserved.
*/
package com.jidesoft.dialog;
import javax.swing.*;
import java.awt.*;
import java.util.logging.Level;
import java.util.logging.Logger;
/**
* AbstractPage is an abstract base class that provides functionality to defer populating a JPanel object until it is
* actually viewed. This is very useful when using CardLayout and tab panel views which have several pages. Delaying the
* construction means it will start up fast. Sometimes delay means never.
*
* If subclasses choose to override any of the following methods, it is their responsibility to ensure their overridden
* methods call the parent's method first. The methods are:
*
* - public void paint (Graphics)
- public void paintComponents(Graphics)
- public void paintAll (Graphics)
*
- public void repaint ()
- public void repaint (long)
- public void repaint (int, int, int, int)
- public
* void repaint (long, int, int, int, int)
- public void update (Graphics)
*
*
* By default, if any of the methods is called, the panel will be populated. However user can setInvokeCondition() to
* customize when the panel be populated. See javadoc of setInvokeContion() for details.
*
* The idea of the lazy panel is from an article on JavaWorld at http://www.javaworld.com/javatips/jw-javatip90_p.html.
* The credit should be given to Mark Roulo. We modified the code he provided in the article to add additional things as
* we need. Things added are - Added setInvokeCondition()
- Added addPageListener(), removePageListener() etc
* so that subclass can fire {@link PageEvent}
*/
public abstract class AbstractPage extends JPanel implements Laziness {
private static final long serialVersionUID = 3566789990192642869L;
/**
* Used by setInvokeCondition(). This value means initialize will be called in all paint/repaint/update methods.
*/
public static int INVOKE_ON_ALL = 0xFFFFFFFF;
/**
* Used by setInvokeCondition(). This value means initialize will not be called. You have to call it manually.
*/
public static int INVOKE_ON_NONE = 0x0;
/**
* Used by setInvokeCondition(). This value means initialize will be called with paint() is called.
*/
public static int INVOKE_ON_PAINT = 0x1;
/**
* Used by setInvokeCondition(). This value means initialize will be called with repaint() is called.
*/
public static int INVOKE_ON_REPAINT = 0x2;
/**
* Used by setInvokeCondition(). This value means initialize will be called with update() is called.
*/
public static int INVOKE_ON_UPDATE = 0x4;
/**
* Used by setInvokeCondition(). This value means initialize will be called with invalidate(), revalidate() is
* called.
*/
public static int INVOKE_ON_VALIDATE = 0x8;
private boolean _allowClosing = true;
private int _invokeCondition = INVOKE_ON_PAINT | INVOKE_ON_REPAINT | INVOKE_ON_UPDATE;
/**
* Only one DataChangeEvent
is needed per model instance since the event's only (read-only) state is
* the source property. The source of events generated here is always "this".
*/
protected transient PageEvent _pageEvent = null;
// We want to call the lazyConstructor only once.
private boolean _lazyConstructorCalled = false;
// Some versions of Swing called paint() before
// the components were added to their containers.
private static final Logger LOGGER_EVENT = Logger.getLogger(PageEvent.class.getName());
/**
* Creates an AbstractPage.
*/
protected AbstractPage() {
}
/**
* Gets the invoke condition. Invoke condition defines how lazy the page is. By default, the lazyInitialize() will
* be called on any update, paint or repaint method. However you can change the invoke condition to INVOKE_ON_PAINT.
* If so, lazyInitialize() will be called only when paint() method is called. You can even set the invoke condition
* to INVOKE_ON_NONE. If so, you will be responsible to call lazyInitialize() since none of those methods methods
* mentioned above will call lazyInitialize().
*
* @return the invocation condition
*/
public int getInvokeCondition() {
return _invokeCondition;
}
/**
* Sets the invoke condition.
*
* @param invokeCondition the invoke condition.
*/
public void setInvokeCondition(int invokeCondition) {
_invokeCondition = invokeCondition;
}
@Override
public void invalidate() {
if ((getInvokeCondition() & INVOKE_ON_VALIDATE) != 0) {
initialize();
}
super.invalidate();
}
@Override
public void revalidate() {
if ((getInvokeCondition() & INVOKE_ON_VALIDATE) != 0) {
initialize();
}
super.revalidate();
}
@Override
public void paint(Graphics g) {
if ((getInvokeCondition() & INVOKE_ON_PAINT) != 0) {
initialize();
}
super.paint(g);
}
@Override
public void paintAll(Graphics g) {
if ((getInvokeCondition() & INVOKE_ON_PAINT) != 0) {
initialize();
}
super.paintAll(g);
}
@Override
public void paintComponents(Graphics g) {
if ((getInvokeCondition() & INVOKE_ON_PAINT) != 0) {
initialize();
}
super.paintComponents(g);
}
@Override
public void repaint() {
if ((getInvokeCondition() & INVOKE_ON_REPAINT) != 0) {
initialize();
}
super.repaint();
}
@Override
public void repaint(long l) {
if ((getInvokeCondition() & INVOKE_ON_REPAINT) != 0) {
initialize();
}
super.repaint(l);
}
@Override
public void repaint(int i1, int i2, int i3, int i4) {
if ((getInvokeCondition() & INVOKE_ON_REPAINT) != 0) {
initialize();
}
super.repaint(i1, i2, i3, i4);
}
@Override
public void repaint(long l, int i1, int i2, int i3, int i4) {
if ((getInvokeCondition() & INVOKE_ON_REPAINT) != 0) {
initialize();
}
super.repaint(l, i1, i2, i3, i4);
}
@Override
public void update(Graphics g) {
if ((getInvokeCondition() & INVOKE_ON_UPDATE) != 0) {
initialize();
}
super.update(g);
}
/**
* Force the lazyInitialize() method implemented in the child class to be called. If this method is called more than
* once on a given object, all calls but the first do nothing.
*/
public synchronized final void initialize() {
if (!_lazyConstructorCalled) {
_lazyConstructorCalled = true;
lazyInitialize();
validate();
}
}
/**
* Resets the page which will result all child components being removed and the method {@link #initialize()} being
* invoked again.
*
* @since 3.2.2
*/
public synchronized void reset() {
_lazyConstructorCalled = false;
removeAll();
}
/**
* Adds a PageListener
to the page.
*
* @param l the PageListener
to be added
*/
public void addPageListener(PageListener l) {
listenerList.add(PageListener.class, l);
}
/**
* Removes a PageListener
from the page.
*
* @param l the PageListener
to be removed
*/
public void removePageListener(PageListener l) {
listenerList.remove(PageListener.class, l);
}
/**
* Returns an array of all the PageListener
s added to this Page
with
* addPageListener
.
*
* @return all of the PageListener
s added, or an empty array if no listeners have been added
*/
public PageListener[] getPageListeners() {
return listenerList.getListeners(
PageListener.class);
}
/**
* Runs each PageListener
's pageEventFired
method.
*
* @param id event id.
*/
public void firePageEvent(int id) {
firePageEvent(this, id);
}
/**
* Runs each PageListener
's pageEventFired
method.
*
* @param source of this event
* @param id event id.
*/
public void firePageEvent(Object source, int id) {
// make sure the page is initialized because user might add page listener in it.
// If the page is not initialized, what's the point firing event
initialize();
if (source == null) { // set the source to this if it's null
source = this;
}
if (LOGGER_EVENT.isLoggable(Level.FINE)) {
switch (id) {
case PageEvent.PAGE_OPENED:
LOGGER_EVENT.fine("Page \"" + this + " is opened, source is " + source.getClass().getName());
break;
case PageEvent.PAGE_CLOSING:
LOGGER_EVENT.fine("Page \"" + this + " is closing, source is " + source.getClass().getName());
break;
case PageEvent.PAGE_CLOSED:
LOGGER_EVENT.fine("Page \"" + this + " is closed, source is " + source.getClass().getName());
break;
default:
break;
}
}
Object[] listeners = listenerList.getListenerList();
for (int i = listeners.length - 2; i >= 0; i -= 2) {
if (listeners[i] == PageListener.class) {
_pageEvent = new PageEvent(source, id);
((PageListener) listeners[i + 1]).pageEventFired(_pageEvent);
}
}
}
/**
* Sets allow closing. If true, the document cannot be closed. user can change the value in documentClosing() to
* prevent document from being closed.
*
* @param allowClosing true or false.
*/
public void setAllowClosing(boolean allowClosing) {
_allowClosing = allowClosing;
}
/**
* Allow this document closing. By default it return true. User can override this method to return based on
* condition. A typical user case is: add a DocumentComponentListener. In documentComponentClosing, make this method
* return to false to prevent it from being closed.
*
* @return whether allow closing
*/
public boolean allowClosing() {
return _allowClosing;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy