-
Bug
-
Resolution: Duplicate
-
P4
-
None
-
1.1.4, 1.1.6
-
None
-
other, x86
-
generic, solaris_2.5.1, windows_95, windows_nt
0, 10 , CENTER);
}
public void setLabel(String label)
{
newLabel(label);
measure();
repaint();
}
public void addNotify()
{
super.addNotify();
measure();
}
public Dimension preferredSize()
{
return new Dimension(max_width + 2*mw,
numlines * lheight + 2 *mh);
}
public Dimension minimumSize()
{
return new Dimension(max_width,numlines * lheight);
}
public void paint(Graphics g)
{
int x, y;
Dimension d= this.size();
y = lascent + (d.height - numlines * lheight)/2;
for ( int i =0; i<numlines; i++, y +=lheight)
{
switch(alignment)
{
case LEFT:
x = mw;
break;
case CENTER:
default:
x = (d.width - lwidth[i])/2;
break;
case RIGHT:
x = d.width - mw - lwidth[i];
break;
}
g.drawString(lines[i],x,y);
}
}
}
Here are the code changes necessary to fix the problem:
DELTA: src/share/java/java/awt/EventDispatchThread.java (a112 1.2)
*** 1.1 Fri Sep 12 11:11:16 1997
--- 1.2 Fri Sep 12 11:11:17 1997
***************
*** 47,52 ****
--- 47,65 ----
public void stopDispatching() {
doDispatch = false;
+ // Interrupt the dispatch thread so we won't get stuck here /*ibm.2888*/
+ // if the thread is waiting for an event. /*ibm.2888*/
+ // There is a potential side-effect here of an event handler /*ibm.2888*/
+ // routine being interrupted. /*ibm.2888*/
+ interrupt(); /*ibm.2888*/
+ while (true) { /*ibm.2888*/
+ try { /*ibm.2888*/
+ join(); /*ibm.2888*/
+ break; /*ibm.2888*/
+ } /*ibm.2888*/
+ catch (InterruptedException e) { /*ibm.2888*/
+ } /*ibm.2888*/
+ } /*ibm.2888*/
}
public void run() {
***************
*** 72,77 ****
--- 85,92 ----
} catch (ThreadDeath death) {
return;
+ // Simply loop if we get interrupted /* ibm.2888 */
+ } catch (InterruptedException interr) { /* ibm.2888 */
} catch (Throwable e) {
System.err.println(
"Exception occurred during event dispatching:");
======================================================================
Name: mf23781 Date: 09/12/97
Race condition when modal dialog boxes are closed
Platform: All platforms
Version: 1.1.2 - and I believe it is still present in 1.1.4
Abstract: Race condition when modal dialog boxes are closed can
cause horrible symptoms.
When a modal dialog box is closed, the following symptoms have been
observed on AIX:
1. a 'postEvent' method not found error.
2. a null pointer exception, like this:
Exception occurred during event dispatching:
java.lang.NullPointerException: sun.awt.motif.MFramePeer@300c8d48
at sun.awt.motif.MComponentPeer.repaint(MComponentPeer.java:134)
at java.awt.Component.repaint(Component.java:1159)
at java.awt.Component.repaint(Component.java:1109)
at java.awt.Window.dispatchEventImpl(Window.java:389)
at java.awt.Component.dispatchEvent(Component.java:1393)
at java.awt.EventDispatchThread.run(EventDispatchThread.java:63)
3. core dumps have also been reported.
During my investigations, I discovered that there is a race
condition that can occur when a modal dialog box is closed.
In src/share/java/java/awt/Dialog.java, in the show() method,
a new event dispatching thread is created for the modal dialog
box. The peer's show method is called, which blocks until
the dialog box is hidden. At this point, stopDispatching() is
called against the Dialog box's own dispatching thread, and
then control returns to the caller. If the caller of
Dialog.show() had been an event handler, then control goes
straight back to the original dispatching thread. The problem
with this is that the Dialog's thread has actually not
finished yet, so there is a potential race condition here.
One situation in which this causes a problem is a test case
in which a modal dialog box is opened from the event handler of
a button in a frame. This event handler calls the dialog's
show method and then disposes the frame. The idea is that when
the dialog box is closed, the frame underneath it is also
closed.
The dialog box is closed by a button method which disposes the
dialog. Now - a dispose starts by issuing a hide, which will
cause the modal dialog peer's show loop to exit - but, the
dispose is still happening. Control is returned to the caller
at this point, and the caller disposes the fame. Now, because
the dialog is a child of the frame, disposing the frame will
also dispose the dialog. Therefore, the dialog is disposed by
two threads at the same time.
The fix I have done here changes the stopDispatching() method
of java.awt.EventDispatchThread to block until the thread has
actually finished doing its job. This means that now no two
events can be processed at the same time. A dialog box's
removeNotify() method (actually in java.awt.Component) checks
the peer before disposing, so, as long as AWT objects are
only disposed by one thread at a time, there is no problem if
you try to dispose an object twice (the second dispose will
have no effect). So, with this fix, the test case will work
reliably.
Here is the test case (wMapAttrs). I'm sure it could be tidied
up a bit!
To use this test case, click on 'Create Mappings', and then
when the dialog box comes up, click on OK. Unfortunately the
problem happens quite rarely!
By mucking about with the AWT code, you can make it happen
more frequently. I would suggest adding a delay after each
pHide method in the C code (on Unix-derived platforms).
------ wMapAttrs.java
import java.awt.*;
import java.util.*;
import java.applet.*;
public
class wMapAttrs extends Frame {
private static Applet applet;
Hashtable menuitems = new Hashtable();
Button pbtnCreateMap, pbtnCancel;
//wMapAttrsgbxDEDSMap pgbxDEDSMap;
// Initial size in logical units
Dimension initialSize = new Dimension(335, 252);
Point initialLocation;
// zpb_begin wMapAttrsUserVars
//RefinedDataSource pRDS;
//RefinedDataEntity pRDE;
// zpb_end
public static void main(String argv[])
{
new wMapAttrs(null, "Hello").setVisible(true);
}
// zpb_begin wMapAttrsJavadocConstructor
/**
* Creates a specialized window with the specified app and title.
*/
// zpb_end
wMapAttrs(Applet app, String title) {
super(title);
applet = app;
// zpb_begin wMapAttrsConstructor_1
// zpb_end
setBackground(Color.lightGray);
setFont(new Font("Sansserif", Font.PLAIN, 12));
setLayout(new FlowLayout());
//pgbxDEDSMap = new wMapAttrsgbxDEDSMap(applet, (Frame)getParent(), "");
//add(pgbxDEDSMap);
//pgbxDEDSMap.setFont(new Font("Sansserif", Font.PLAIN, 12));
pbtnCreateMap = new Button(" Create Mappings ");
add(pbtnCreateMap);
pbtnCancel = new Button(" Cancel ");
add(pbtnCancel);
// Add the MenuBar, Menus, and Menuitems
MenuBar mbMenu1 = new MenuBar();
MenuItem mi;
this.setMenuBar(mbMenu1);
// zpb_begin wMapAttrsConstructor_2
pbtnCreateMap.setLabel("Create Mappings");
pbtnCancel.setLabel("Cancel");
// zpb_end
pack();
// zpb_begin wMapAttrsConstructor_3
// zpb_end
}
// zpb_begin wMapAttrsJavadocHandleEvent
/**
* Handle events that occur with components in this container.
*
* @return Returns a boolean which specifies whether the event
* handler should stop at this level or continue up the
* parent hierarchy.
*/
// zpb_end
public boolean handleEvent(Event e) {
if (e.id == Event.WINDOW_DESTROY) {
// zpb_begin wMapAttrsDestroy
// zpb_end
dispose();
return true;
}
// zpb_begin wMapAttrsHandleEvent
// zpb_end
return super.handleEvent(e);
}
// zpb_begin wMapAttrsJavadocAction
/**
* Action handling routine.
*
* @return Returns a boolean specifying whether the event should be handled
* furthur up the parent hierarchy.
*/
// zpb_end
public boolean action(Event evt, Object obj) {
if (evt.target instanceof MenuItem) {
return false;
}
if (evt.target == pbtnCreateMap) {
// zpb_begin wMapAttrsbtnCreateMapClicked
boolean rc;
BDS_Context.showError(this);
dispose();
return true;
// zpb_end
}
else if (evt.target == pbtnCancel) {
// zpb_begin wMapAttrsbtnCancelClicked
dispose();
return false;
// zpb_end
}
// zpb_begin wMapAttrsAction
// zpb_end
return false;
}
// zpb_begin wMapAttrsUserMethods
}
------ BDS_Context.java
import java.awt.*;
public class BDS_Context
{
public static void showError(Frame frame)
{
// TEMPORARY FIX: DON"T SHOW ANY MORE ERRORS IF COMMS FAILURE !!
String newtitle="Hello, there!";
String newmessage="And this is the message to be displayed.";
//********************************************************
// Make AIX error messages modal and Windows error messages
// non-modal due to bugs in both JDK's
//*********************************************************
boolean modality ;
String osName = System.getProperty("os.name");
modality = true ;
dlgError pdlgError = new dlgError(null,
frame,
newtitle,
modality);
pdlgError.setErrorMsg(newmessage);
pdlgError.show();
}
}
------ dlgError.java
import java.awt.*;
import java.util.*;
import java.applet.*;
//import com.roguewave.widgets.*;
// zpb_begin UserImports
//import ilsTextTable;
// zpb_end
/**
* Class dlgError.
*
*
* @author BDS Development Team
* @version 1.0 12/05/96 10:46:02am
* @copyright IBM
*/
public
class dlgError extends Dialog {
private static Applet applet;
Button pButton1;
// Initial size in logical units
Dimension initialSize = new Dimension(200, 200);
Point initialLocation;
// zpb_begin dlgErrorUserVars
GridBagLayout gridbag;
GridBagConstraints gbc;
TextBox ptxtErr;
// zpb_end
// zpb_begin dlgErrorJavadocConstructor
/**
* Creates a specialized dialog with the specified app, parent, title and
* modality.
*/
// zpb_end
dlgError(Applet app, Frame parent, String title, boolean modal) {
super(parent, title, modal);
applet = app;
// zpb_begin dlgErrorConstructor_1
setResizable(true);
/***********
// zpb_end
// zpb_begin dlgErrorConstructor_2
***********/
gridbag = new GridBagLayout();
gbc = new GridBagConstraints();
setFont(new Font("Sansserif", Font.PLAIN, 12));
setLayout(gridbag);
/***********
// zpb_end
// Size and Position in logical units
// zpb_begin dlgErrorConstructor_3
************/
pack();
// zpb_end
}
// zpb_begin dlgErrorJavadocMinimumSize
/**
* @return Returns a Dimension with the minimum width and height
* of this container.
*/
// zpb_end
public Dimension minimumSize() {
return gridbag.minimumLayoutSize(this);
}
public Dimension preferredSize() {
return gridbag.preferredLayoutSize(this);
}
// zpb_begin dlgErrorJavadocHandleEvent
/**
* Handle events that occur with components in this container.
*
* @return Returns a boolean which specifies whether the event
* handler should stop at this level or continue up the
* parent hierarchy.
*/
// zpb_end
public boolean handleEvent(Event e) {
if (e.id == Event.WINDOW_DESTROY) {
// zpb_begin dlgErrorDestroy
// zpb_end
dispose();
return true;
}
// zpb_begin dlgErrorHandleEvent
// zpb_end
return super.handleEvent(e);
}
// zpb_begin dlgErrorJavadocAction
/**
* Action handling routine.
*
* @return Returns a boolean specifying whether the event should be handled
* furthur up the parent hierarchy.
*/
// zpb_end
public boolean action(Event evt, Object obj) {
if (evt.target == pButton1) {
// zpb_begin dlgErrorButton1Clicked
// zpb_end
dispose();
return false;
}
// zpb_begin dlgErrorAction
// zpb_end
return false;
}
// zpb_begin dlgErrorUserMethods
void setErrorMsg(String msg)
{
gbc.anchor = GridBagConstraints.CENTER;
gbc.fill = GridBagConstraints.HORIZONTAL;
gbc.ipadx = 50;
gbc.ipady = 50;
ptxtErr = new TextBox(msg);
constrain(ptxtErr,
GridBagConstraints.REMAINDER,
1.0,1.0,
1,1,0,1);
pButton1 = new Button("OK");
Panel p = new Panel();
p.setLayout(new FlowLayout(FlowLayout.CENTER,10,10));
p.add(pButton1);
pButton1.setLabel("OK");
constrain(p,
GridBagConstraints.REMAINDER,
0.0,0.0,
0,0,0,0);
initialLocation = gridbag.location(15,28);
pack();
move(initialLocation.x, initialLocation.y);
}
void constrain(Component comp,
int w,
double wx, double wy,
int top, int left, int bottom, int right)
{
gbc.gridwidth = w;
// wx and wy are use to determine how much will a
// component grow if the window is resized
gbc.weightx = wx; gbc.weighty = wy;
if (top+bottom+left+right > 0)
gbc.insets = new Insets(top,left,bottom,right);
gridbag.setConstraints(comp, gbc);
add(comp);
}
// zpb_end
}
------ TextBox.java
import java.awt.*;
import java.util.*;
public class TextBox extends Canvas {
public static final int LEFT = 0;
public static final int CENTER = 1;
public static final int RIGHT = 2;
protected int alignment = CENTER;
protected String[] lines;
protected int numlines;
protected int mw, mh; // margins
protected int lheight, lascent;
protected int[] lwidth;
protected int max_width;
// newLabel: breaks a label into an array of lines
//
protected void newLabel(String label) {
StringTokenizer t =new StringTokenizer(label, "\n");
numlines = t.countTokens();
lines = new String[numlines];
lwidth = new int[numlines];
for(int i=0; i< numlines; i++)
lines[i] = t.nextToken();
}
// measure : figures out how large the font is and how
// wide each line of the label is.
protected void measure() {
FontMetrics fm= getFontMetrics(getFont());
if (fm != null)
{
lheight = fm.getHeight();
lascent = fm.getAscent();
max_width = 0;
for (int i=0; i< numlines; i++) {
lwidth[i] = fm.stringWidth(lines[i]);
if (lwidth[i] >max_width)
max_width = lwidth[i];
}
}
}
// TextBox : Break up label into separate lines
//
public TextBox(String label, int w, int h , int alignment)
{
newLabel(label);
mw = w;
mh = h;
this.alignment = alignment;
}
public TextBox(String label, int w, int h)
{
this(label, w, h , CENTER);
}
public TextBox(String label, int alignment)
{
this(label, 10, 10 , alignment);
}
public TextBox(String label)
{
this(label, 1
}
public void setLabel(String label)
{
newLabel(label);
measure();
repaint();
}
public void addNotify()
{
super.addNotify();
measure();
}
public Dimension preferredSize()
{
return new Dimension(max_width + 2*mw,
numlines * lheight + 2 *mh);
}
public Dimension minimumSize()
{
return new Dimension(max_width,numlines * lheight);
}
public void paint(Graphics g)
{
int x, y;
Dimension d= this.size();
y = lascent + (d.height - numlines * lheight)/2;
for ( int i =0; i<numlines; i++, y +=lheight)
{
switch(alignment)
{
case LEFT:
x = mw;
break;
case CENTER:
default:
x = (d.width - lwidth[i])/2;
break;
case RIGHT:
x = d.width - mw - lwidth[i];
break;
}
g.drawString(lines[i],x,y);
}
}
}
Here are the code changes necessary to fix the problem:
DELTA: src/share/java/java/awt/EventDispatchThread.java (a112 1.2)
*** 1.1 Fri Sep 12 11:11:16 1997
--- 1.2 Fri Sep 12 11:11:17 1997
***************
*** 47,52 ****
--- 47,65 ----
public void stopDispatching() {
doDispatch = false;
+ // Interrupt the dispatch thread so we won't get stuck here /*ibm.2888*/
+ // if the thread is waiting for an event. /*ibm.2888*/
+ // There is a potential side-effect here of an event handler /*ibm.2888*/
+ // routine being interrupted. /*ibm.2888*/
+ interrupt(); /*ibm.2888*/
+ while (true) { /*ibm.2888*/
+ try { /*ibm.2888*/
+ join(); /*ibm.2888*/
+ break; /*ibm.2888*/
+ } /*ibm.2888*/
+ catch (InterruptedException e) { /*ibm.2888*/
+ } /*ibm.2888*/
+ } /*ibm.2888*/
}
public void run() {
***************
*** 72,77 ****
--- 85,92 ----
} catch (ThreadDeath death) {
return;
+ // Simply loop if we get interrupted /* ibm.2888 */
+ } catch (InterruptedException interr) { /* ibm.2888 */
} catch (Throwable e) {
System.err.println(
"Exception occurred during event dispatching:");
======================================================================
Name: mf23781 Date: 09/12/97
Race condition when modal dialog boxes are closed
Platform: All platforms
Version: 1.1.2 - and I believe it is still present in 1.1.4
Abstract: Race condition when modal dialog boxes are closed can
cause horrible symptoms.
When a modal dialog box is closed, the following symptoms have been
observed on AIX:
1. a 'postEvent' method not found error.
2. a null pointer exception, like this:
Exception occurred during event dispatching:
java.lang.NullPointerException: sun.awt.motif.MFramePeer@300c8d48
at sun.awt.motif.MComponentPeer.repaint(MComponentPeer.java:134)
at java.awt.Component.repaint(Component.java:1159)
at java.awt.Component.repaint(Component.java:1109)
at java.awt.Window.dispatchEventImpl(Window.java:389)
at java.awt.Component.dispatchEvent(Component.java:1393)
at java.awt.EventDispatchThread.run(EventDispatchThread.java:63)
3. core dumps have also been reported.
During my investigations, I discovered that there is a race
condition that can occur when a modal dialog box is closed.
In src/share/java/java/awt/Dialog.java, in the show() method,
a new event dispatching thread is created for the modal dialog
box. The peer's show method is called, which blocks until
the dialog box is hidden. At this point, stopDispatching() is
called against the Dialog box's own dispatching thread, and
then control returns to the caller. If the caller of
Dialog.show() had been an event handler, then control goes
straight back to the original dispatching thread. The problem
with this is that the Dialog's thread has actually not
finished yet, so there is a potential race condition here.
One situation in which this causes a problem is a test case
in which a modal dialog box is opened from the event handler of
a button in a frame. This event handler calls the dialog's
show method and then disposes the frame. The idea is that when
the dialog box is closed, the frame underneath it is also
closed.
The dialog box is closed by a button method which disposes the
dialog. Now - a dispose starts by issuing a hide, which will
cause the modal dialog peer's show loop to exit - but, the
dispose is still happening. Control is returned to the caller
at this point, and the caller disposes the fame. Now, because
the dialog is a child of the frame, disposing the frame will
also dispose the dialog. Therefore, the dialog is disposed by
two threads at the same time.
The fix I have done here changes the stopDispatching() method
of java.awt.EventDispatchThread to block until the thread has
actually finished doing its job. This means that now no two
events can be processed at the same time. A dialog box's
removeNotify() method (actually in java.awt.Component) checks
the peer before disposing, so, as long as AWT objects are
only disposed by one thread at a time, there is no problem if
you try to dispose an object twice (the second dispose will
have no effect). So, with this fix, the test case will work
reliably.
Here is the test case (wMapAttrs). I'm sure it could be tidied
up a bit!
To use this test case, click on 'Create Mappings', and then
when the dialog box comes up, click on OK. Unfortunately the
problem happens quite rarely!
By mucking about with the AWT code, you can make it happen
more frequently. I would suggest adding a delay after each
pHide method in the C code (on Unix-derived platforms).
------ wMapAttrs.java
import java.awt.*;
import java.util.*;
import java.applet.*;
public
class wMapAttrs extends Frame {
private static Applet applet;
Hashtable menuitems = new Hashtable();
Button pbtnCreateMap, pbtnCancel;
//wMapAttrsgbxDEDSMap pgbxDEDSMap;
// Initial size in logical units
Dimension initialSize = new Dimension(335, 252);
Point initialLocation;
// zpb_begin wMapAttrsUserVars
//RefinedDataSource pRDS;
//RefinedDataEntity pRDE;
// zpb_end
public static void main(String argv[])
{
new wMapAttrs(null, "Hello").setVisible(true);
}
// zpb_begin wMapAttrsJavadocConstructor
/**
* Creates a specialized window with the specified app and title.
*/
// zpb_end
wMapAttrs(Applet app, String title) {
super(title);
applet = app;
// zpb_begin wMapAttrsConstructor_1
// zpb_end
setBackground(Color.lightGray);
setFont(new Font("Sansserif", Font.PLAIN, 12));
setLayout(new FlowLayout());
//pgbxDEDSMap = new wMapAttrsgbxDEDSMap(applet, (Frame)getParent(), "");
//add(pgbxDEDSMap);
//pgbxDEDSMap.setFont(new Font("Sansserif", Font.PLAIN, 12));
pbtnCreateMap = new Button(" Create Mappings ");
add(pbtnCreateMap);
pbtnCancel = new Button(" Cancel ");
add(pbtnCancel);
// Add the MenuBar, Menus, and Menuitems
MenuBar mbMenu1 = new MenuBar();
MenuItem mi;
this.setMenuBar(mbMenu1);
// zpb_begin wMapAttrsConstructor_2
pbtnCreateMap.setLabel("Create Mappings");
pbtnCancel.setLabel("Cancel");
// zpb_end
pack();
// zpb_begin wMapAttrsConstructor_3
// zpb_end
}
// zpb_begin wMapAttrsJavadocHandleEvent
/**
* Handle events that occur with components in this container.
*
* @return Returns a boolean which specifies whether the event
* handler should stop at this level or continue up the
* parent hierarchy.
*/
// zpb_end
public boolean handleEvent(Event e) {
if (e.id == Event.WINDOW_DESTROY) {
// zpb_begin wMapAttrsDestroy
// zpb_end
dispose();
return true;
}
// zpb_begin wMapAttrsHandleEvent
// zpb_end
return super.handleEvent(e);
}
// zpb_begin wMapAttrsJavadocAction
/**
* Action handling routine.
*
* @return Returns a boolean specifying whether the event should be handled
* furthur up the parent hierarchy.
*/
// zpb_end
public boolean action(Event evt, Object obj) {
if (evt.target instanceof MenuItem) {
return false;
}
if (evt.target == pbtnCreateMap) {
// zpb_begin wMapAttrsbtnCreateMapClicked
boolean rc;
BDS_Context.showError(this);
dispose();
return true;
// zpb_end
}
else if (evt.target == pbtnCancel) {
// zpb_begin wMapAttrsbtnCancelClicked
dispose();
return false;
// zpb_end
}
// zpb_begin wMapAttrsAction
// zpb_end
return false;
}
// zpb_begin wMapAttrsUserMethods
}
------ BDS_Context.java
import java.awt.*;
public class BDS_Context
{
public static void showError(Frame frame)
{
// TEMPORARY FIX: DON"T SHOW ANY MORE ERRORS IF COMMS FAILURE !!
String newtitle="Hello, there!";
String newmessage="And this is the message to be displayed.";
//********************************************************
// Make AIX error messages modal and Windows error messages
// non-modal due to bugs in both JDK's
//*********************************************************
boolean modality ;
String osName = System.getProperty("os.name");
modality = true ;
dlgError pdlgError = new dlgError(null,
frame,
newtitle,
modality);
pdlgError.setErrorMsg(newmessage);
pdlgError.show();
}
}
------ dlgError.java
import java.awt.*;
import java.util.*;
import java.applet.*;
//import com.roguewave.widgets.*;
// zpb_begin UserImports
//import ilsTextTable;
// zpb_end
/**
* Class dlgError.
*
*
* @author BDS Development Team
* @version 1.0 12/05/96 10:46:02am
* @copyright IBM
*/
public
class dlgError extends Dialog {
private static Applet applet;
Button pButton1;
// Initial size in logical units
Dimension initialSize = new Dimension(200, 200);
Point initialLocation;
// zpb_begin dlgErrorUserVars
GridBagLayout gridbag;
GridBagConstraints gbc;
TextBox ptxtErr;
// zpb_end
// zpb_begin dlgErrorJavadocConstructor
/**
* Creates a specialized dialog with the specified app, parent, title and
* modality.
*/
// zpb_end
dlgError(Applet app, Frame parent, String title, boolean modal) {
super(parent, title, modal);
applet = app;
// zpb_begin dlgErrorConstructor_1
setResizable(true);
/***********
// zpb_end
// zpb_begin dlgErrorConstructor_2
***********/
gridbag = new GridBagLayout();
gbc = new GridBagConstraints();
setFont(new Font("Sansserif", Font.PLAIN, 12));
setLayout(gridbag);
/***********
// zpb_end
// Size and Position in logical units
// zpb_begin dlgErrorConstructor_3
************/
pack();
// zpb_end
}
// zpb_begin dlgErrorJavadocMinimumSize
/**
* @return Returns a Dimension with the minimum width and height
* of this container.
*/
// zpb_end
public Dimension minimumSize() {
return gridbag.minimumLayoutSize(this);
}
public Dimension preferredSize() {
return gridbag.preferredLayoutSize(this);
}
// zpb_begin dlgErrorJavadocHandleEvent
/**
* Handle events that occur with components in this container.
*
* @return Returns a boolean which specifies whether the event
* handler should stop at this level or continue up the
* parent hierarchy.
*/
// zpb_end
public boolean handleEvent(Event e) {
if (e.id == Event.WINDOW_DESTROY) {
// zpb_begin dlgErrorDestroy
// zpb_end
dispose();
return true;
}
// zpb_begin dlgErrorHandleEvent
// zpb_end
return super.handleEvent(e);
}
// zpb_begin dlgErrorJavadocAction
/**
* Action handling routine.
*
* @return Returns a boolean specifying whether the event should be handled
* furthur up the parent hierarchy.
*/
// zpb_end
public boolean action(Event evt, Object obj) {
if (evt.target == pButton1) {
// zpb_begin dlgErrorButton1Clicked
// zpb_end
dispose();
return false;
}
// zpb_begin dlgErrorAction
// zpb_end
return false;
}
// zpb_begin dlgErrorUserMethods
void setErrorMsg(String msg)
{
gbc.anchor = GridBagConstraints.CENTER;
gbc.fill = GridBagConstraints.HORIZONTAL;
gbc.ipadx = 50;
gbc.ipady = 50;
ptxtErr = new TextBox(msg);
constrain(ptxtErr,
GridBagConstraints.REMAINDER,
1.0,1.0,
1,1,0,1);
pButton1 = new Button("OK");
Panel p = new Panel();
p.setLayout(new FlowLayout(FlowLayout.CENTER,10,10));
p.add(pButton1);
pButton1.setLabel("OK");
constrain(p,
GridBagConstraints.REMAINDER,
0.0,0.0,
0,0,0,0);
initialLocation = gridbag.location(15,28);
pack();
move(initialLocation.x, initialLocation.y);
}
void constrain(Component comp,
int w,
double wx, double wy,
int top, int left, int bottom, int right)
{
gbc.gridwidth = w;
// wx and wy are use to determine how much will a
// component grow if the window is resized
gbc.weightx = wx; gbc.weighty = wy;
if (top+bottom+left+right > 0)
gbc.insets = new Insets(top,left,bottom,right);
gridbag.setConstraints(comp, gbc);
add(comp);
}
// zpb_end
}
------ TextBox.java
import java.awt.*;
import java.util.*;
public class TextBox extends Canvas {
public static final int LEFT = 0;
public static final int CENTER = 1;
public static final int RIGHT = 2;
protected int alignment = CENTER;
protected String[] lines;
protected int numlines;
protected int mw, mh; // margins
protected int lheight, lascent;
protected int[] lwidth;
protected int max_width;
// newLabel: breaks a label into an array of lines
//
protected void newLabel(String label) {
StringTokenizer t =new StringTokenizer(label, "\n");
numlines = t.countTokens();
lines = new String[numlines];
lwidth = new int[numlines];
for(int i=0; i< numlines; i++)
lines[i] = t.nextToken();
}
// measure : figures out how large the font is and how
// wide each line of the label is.
protected void measure() {
FontMetrics fm= getFontMetrics(getFont());
if (fm != null)
{
lheight = fm.getHeight();
lascent = fm.getAscent();
max_width = 0;
for (int i=0; i< numlines; i++) {
lwidth[i] = fm.stringWidth(lines[i]);
if (lwidth[i] >max_width)
max_width = lwidth[i];
}
}
}
// TextBox : Break up label into separate lines
//
public TextBox(String label, int w, int h , int alignment)
{
newLabel(label);
mw = w;
mh = h;
this.alignment = alignment;
}
public TextBox(String label, int w, int h)
{
this(label, w, h , CENTER);
}
public TextBox(String label, int alignment)
{
this(label, 10, 10 , alignment);
}
public TextBox(String label)
{
this(label, 1
- duplicates
-
JDK-4122683 hiding modal Dialog on NT can deadlock, freezing GUI
-
- Closed
-