Root Pane Container(四)

8.5 JApplet类

JApplet类是AWT Applet类的扩展。为了在使用Swing组件的applet中能正确的进行事件处理,我们applet必须继承JApplet,而不是Applet。

JApplet的作用与其他的实现了RootPaneContainer接口的高层窗口相同。JApplet与Applet之间一个重要的区别就是默认的布局管理器。因为我们向JApplet的内容面析添加组件,其默认的布局管理器为BorderLayout。这与Applet的默认布局管理器FlowLayout不同。另外,Swing applet还可以具有一个工具栏,或者更为特定的JMenuBar,这是applet的JRootPane的另一个属性。

如果我们计划部署一个使用Swing组件的applet,最好是使用Sun Microsystems所提供的Java插件,因为这会随运行时安装Swing库。

如查我们要扩展JApplet类,他只有一个重要的protected方法:

protected JRootPane createRootPane()

8.6 配合桌面使用

Spring提供了对一个通常窗口或是桌面内的窗体集合进行管理。正如我们在第1章所讨论的,这种管理通常被称之为MDI。窗体可以位于其他的窗体之上,或者是可以被拖动,而其外观适当当前的观感。这些窗体是JInternalFrame类的实例,而桌面是一个称之为JDesktopPane的特殊JLayeredPane。桌面内窗体的管理是DesktopManager的责任,其中所提供的默认实现是DefaultDesktopManager。桌面上的JInternalFrame的图标形式是通过JDesktopIcon的内联类JInternalFrame来表示的。同时有InternalFrmaeListener,InternalFrameAdapter以及InternalFrameEvent用于事件处理。

首先,我们来看一下构成桌面的部分,然后我们会看到使用这些部分的一个完整示例。

8.6.1 JInternalFrame类

JInternalFrame类与JFrame类类似。他是一个高层窗口,使用RootPaneContainer接口,但是并不是一个顶层窗口。我们必须将内部窗体放在另一个顶层窗口中。当我们拖动时,内部窗体会停留在其窗口的边界之内,这通常是一个JDesktopPane。另外,内部窗体是轻量级的,并且提供了一个UI委托从而使得内部窗体看起来类似当前配置的观感。

创建JInternalFrame

JInternalFrame有六个构造函数:

public JInternalFrame()
JInternalFrame frame = new JInternalFrame();
 
public JInternalFrame(String title)
JInternalFrame frame = new JInternalFrame("The Title");
 
public JInternalFrame(String title, boolean resizable)
JInternalFrame frame = new JInternalFrame("The Title", true);
 
public JInternalFrame(String title, boolean resizable, boolean closable)
JInternalFrame frame = new JInternalFrame("The Title", false, true);
 
public JInternalFrame(String title, boolean resizable, boolean
  closable, boolean maximizable)
JInternalFrame frame = new JInternalFrame("The Title", true, false, true);
 
public JInternalFrame(String title, boolean resizable, boolean
  closable, boolean maximizable, boolean iconifiable)
JInternalFrame frame = new JInternalFrame("The Title", false, true, false, true);

这些构造函数以一个向另一个添加参数的方式进行级联。无参数时,所创建的JInternalFrame没有标题,并且不能调整大小,关闭,最大化或是图标化。然而,内部窗体总是可以拖动的。

JInternalFrame属性

表8-9列出了JInternalFrame类的30个不同属性。layer属性列出了两次,因为他具有两个设置方法,其中一个用于int,而另一个用于Integer。

JInternalFrame属性

属性名

数据类型

访问性

accessibleContext

AccessibleContext

只读

closable

boolean

读写绑定

closed

boolean

读写绑定

contentPane

Container

读写绑定

defaultCloseOperation

int

读写

desktopIcon

JInternalFrame.JDesktopIcon

读写绑定

desktopPane

JDesktopPane

只读

focusCycleRoot

boolean

读写

focusCycleRootAncester

Container

只读

focusOwner

Component

只读

frameIcon

Icon

读写绑定

glassPane

Component

读写绑定

icon

boolean

读写绑定

iconifiable

boolean

读写

internalFrameListeners

InternalFrameListener[]

只读

jMenuBar

JMenuBar

读写绑定

layer

int

读写

layer

Integer

只写

layeredPane

JLayeredPane

读写绑定

layout

LayoutManager

只写

maximizable

boolean

读写绑定

maximum

boolean

读写

mostRecentFocusOwner

Component

只读

normalBounds

Rectangle

读写

resizable

boolean

读写绑定

rootPane

JRootPane

读写绑定

selected

boolean

读写绑定

title

String

读写绑定

UI

InternalFrameUI

读写

UIClassID

String

只读

warningString

String

只读

对于Java 1.3及以后的版本,JInternalFrame的初始defaultCloseOperation属性设置为DISPOSE_ON_CLOSE。以前版本的默认设置为HIDE_ON_CLOSE。我们可以将这个属性设置为前面的表8-6中列出的WindowConstants的值。

normalBounds属性描述了当一个图标化的内部窗体取消息图标化时应该在哪里显示。focusOwner属性在特定的JInternalFrame被激活时提供了一个实际带有输入焦点的Component。

在Swing类中,JInternalFrame只包含四个限制属性:closed, icon, maximum以及selected。他们与四个boolean构造函数参数直接相关。每一个都可以允许我们在改变其设置时检测当前的属性状态。然而,因为属性是受限制的,当我们要设置一个属性时,我们所做的尝试必须位于一个try-catch块中,捕捉PropertyVetoException:

try {
  // Try to iconify internal frame
  internalFrame.setIcon(false);
}  catch (PropertyVetoException propertyVetoException) {
  System.out.println("Rejected");
}

为了有助于我们使用这些绑定属性,JInternalFrame类定义了一个11个常量,如表8-10所示。他们表示在PropertyChangeListener中通过PropertyChangeEvent的getPropertyName()方法返回的字符串。

JInternal属性常量

属性名常量

关联属性
CONTENT_PANE_PROPERTY

contentPane

FRAME_ICON_PROPERTY

frameIcon

GLASS_PANE_PROPERTY

glassPane

IS_CLOSED_PROPERTY

closed

IS_ICON_PROPERTY

icon

IS_MAXIMUM_PROPERTY

maximum

IS_SELECTED_PROPERTY

selected

LAYERED_PANE_PROPERTY

layeredPane

MENU_BAR_PROPERTY

jMenuBar

ROOT_PANE_PROPERTY

rootPane

TITLE_PROPERTY

title

下面的类示例演示了在PropertyChangeListener中常量的使用。

package swingstudy.ch08;
 
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
 
import javax.swing.JInternalFrame;
 
public class InternalFramePropertyChangeHandler implements
		PropertyChangeListener {
 
	@Override
	public void propertyChange(PropertyChangeEvent event) {
		// TODO Auto-generated method stub
 
		String propertyName = event.getPropertyName();
		if(propertyName.equals(JInternalFrame.IS_ICON_PROPERTY)) {
			System.out.println("Icon property changed. React.");
		}
	}
 
}

处理JInternalFrame事件

为了帮助我们像使用JFrame一样来使用JInternalFrame,有一个额外的事件监听器来负责内部窗体的打开与关闭的相关事件。这个接口名为InternalFrameListener,其定义如下。其作用类似于AWT的WindowListener接口,但是所用的JInternalFrame类,而不是AWT的Window类。

public interface InternalFrameListener extends EventListener {
  public void internalFrameActivated(InternalFrameEvent internalFrameEvent);
  public void internalFrameClosed(InternalFrameEvent internalFrameEvent);
  public void internalFrameClosing(InternalFrameEvent internalFrameEvent);
  public void internalFrameDeactivated(InternalFrameEvent internalFrameEvent);
  public void internalFrameDeiconified(InternalFrameEvent internalFrameEvent);
  public void internalFrameIconified(InternalFrameEvent internalFrameEvent);
  public void internalFrameOpened(InternalFrameEvent internalFrameEvent);
}

另外,与具有所有WindowListener方法桩的WindowApapter类类似,也有一个具有所有InternalFrameListener方法桩的InternalFrameAdapter类。如果我们并不是对JInternalFrame发生的所有事件感兴趣,我们可以继承InternalFrameAdapter类,并且只重写我们所感兴趣的方法。例如,列表8-5中所示的监听器只对图标化方法感兴趣。无需提供InternalFrameListener的其他五个方法的桩实现,我们只需要继承InternalFrameAdapter,并重写两个相关的方法。

package swingstudy.ch08;
 
import javax.swing.JInternalFrame;
import javax.swing.event.InternalFrameAdapter;
import javax.swing.event.InternalFrameEvent;
 
public class InternalFrameIconifyListener extends InternalFrameAdapter {
 
	public void internalFrameIconified(InternalFrameEvent event) {
		JInternalFrame source = (JInternalFrame)event.getSource();
		System.out.println("Iconified: "+source.getTitle());
	}
 
	public void internalFrameDeiconified(InternalFrameEvent event) {
		JInternalFrame source = (JInternalFrame)event.getSource();
		System.out.println("Deiconified: "+source.getTitle());
	}
}

InternalFrameEvent是AWTEvent的子类。为了定义由AWTEvent的public int getID()方法返回的值,InternalFrameEvent每一个可用的特定事件子类定义了一个常量。表8-11列出了九个常量。我们也可以通过事件的getInternalFrame()方法来获得实际的JInternalFrame。

InternalFrameEvent事件子类型

事件子类型ID

关联的接口方法

INTERNAL_FRAME_ACTIVATED

internalFrameActivated

INTERNAL_FRAME_CLOSED

internalFrameClosed

INTERNAL_FRAME_CLOSING

internalFrameClosing

INTERNAL_FRAME_DEACTIVATED

internalFrameDeactivated

INTERNAL_FRAME_DEICONIFIED

internalFrameDeiconified

INTERNAL_FRAME_FIRST

N/A

INTERNAL_FRAME_ICONIFIED

internalFrameIconified

INTERNAL_FRAME_LAST

N/A

INTERNAL_FRAME_OPENED

internalFrameOpened

自定义JInternalFrame观感

因为JInternalFrame是一个轻量级组件,他具有可安装的观感。每一个可安装的Swing观感提供了一个不同的JInternalFrame外观以及默认的UIResource值集合。图8-5预安装的观感类型集合的JWindow窗口外观。

Swing_8_5_motif

Swing_8_5_ocean

Swing_8_5_windows

表8-12中列出了JInternalFrame的可用UIResource相关属性的集合。对于JInternalFrame常量,有60个不同的属性,包括内部窗体的标题面板的属性。

JInternalFrame UIResource元素

属性字符串

对象类型

InternalFrame.actionMap

ActionMap

InternalFrame.activeBroderColor

Color

InternalFrame.activeTitleBackground

Color

InternaleFrame.activeTitleForeground

Color

InternalFrame.activeTitleGradient

List

InternalFrame.border

Border

InternalFrame.borderColor

Color

InternalFrame.borderDarkShadow

Color

InternalFrame.borderHighlight

Color

InterenalFrame.borderLight

Color

InternaleFrame.borderShadow

Color

InternaleFrame.borderWidth

Integer

InternalFrame.closeButtonToolTip

String

InternalFrame.closeIcon

Icon

InternalFrmae.closeSound

String

InternalFrame.icon

Icon

InternalFrame.iconButtonToolTip

String

InternalFrame.iconifyIcon

Icon

InternalFrame.inactiveBorderColor

Color

InternalFrame.inactiveTitleBackground

Color

InternalFrame.inactiveTitleForeground

Color

InternalFrame.inactiveTitleGradient

List

InternalFrame.layoutTitlePaneAtOrigin

Boolean

InternalFrame.maxButtonToolTip

String

InternalFrame.maximizeIcon

Icon

InternalFrame.maximizeSound

String

InternalFrame.minimizeIcon

Icon

InternalFrame.minimizeIconBackground

Color

InternalFrame.minimizeSound

String

InternalFrame.optionDialogBorder

Border

InternalFrame.paletteBorder

Border

InternalFrame.paletteCloseIcon

Icon

InternalFrame.paletteTitleHeight

Integer

InternaleFrame.resizeIconHighlight

Color

InternalFrame.resizeIconShadow

Color

InternalFrame.restoreButtonToolTip

String

InternalFrame.restoreDownSound

String

InternalFrame.restoreUpSound

String

InternalFrame.titlebuttonHeight

Integer

InternalFrame.titleButtonWidth

Integer

InternalFrame.titleFont

Font

InternalFrame.titlePaneHeight

Integer

InternalFrame.useTaskBar

Boolean

InternalFrame.windowBindings

Object[]

InternalFrameTitlePane.closebuttonAccessibleName

String

InternalFrameTitlePane.closebuttonText

String

InternalFrameTitlePane.closeIcon

Icon

InternalFrameTitlePane.iconifyButtonAccessibleName

String

InternalFrameTitlePane.iconifyIcon

Icon

InternalFrameTitlePane.maximizeButtonAccessiblName

String

InternalFrameTitlePane.maximizeButtonText

String

InternalFrameTitlePane.minimizeIcon

Icon

InternalFrameTitlePane.moveButtonText

String

InternalFrameTitlePane.restoreButtonText

String

InternalFrameTitlePane.sizeButtonText

String

InternalFrameTitlePane.titlePaneLayout

LayoutManager

InternalFrameTitlePaneUI

String

InternalFrameUI

String

除了表8-12中许多可配置属性以外,对于Metal观感,我们还可以通过特殊的客户端属性JInternalFrame.isPalette来将内部窗体设计为一个palette。当设置为Boolean.TRUE时,内部窗体的外观会与其他窗体略微不同,并且具有较短的标题栏,如图8-6所示。

Swing_8_6

如果我们同时在桌面的PALETTE_LAYER上添加了一个内部窗体,则这个窗体会位于其他所有窗体之上(如图8-6所示):

JInternalFrame palette = new JInternalFrame("Palette", true, false, true, false);
palette.setBounds(150, 0, 100, 100);
palette.putClientProperty("JInternalFrame.isPalette", Boolean.TRUE);
desktop.add(palette, JDesktopPane.PALETTE_LAYER);

创建图8-6所示的程序的完整代码显示在本章稍后的列表8-6中。

修改JDesktopIcon

JInternalFrame依赖一个内联类JDesktopIcon来为JInternalFrame的图标化显示提供UI委托。这个类只是用来这种功能的一个特殊的JComponent,而不是如其名字暗示的一个特殊的Icon实现。事实上,JDesktopIcon类的注释表明这个类是临时的,所以我们不应直接对其进行自定义。(当然,这个类会存在一段时间。)

如果我们需要自定义JDesktopIcon,我们可以修改一些UIResource相关的属性。表8-13列出了JDesktopIcon组件的八个UIResource相关属性。

JInternalFrame.JDesktopIcon UIResource元素

属性字符串

对象类型

DesktopIcon.background

Color

DesktopIcon.border

Border

DesktopIcon.font

Font

DesktopIcon.foreground

Color

DesktopIcon.icon

Icon

DesktopIcon.width

Integer

DesktopIcon.windowBindings

Object[]

DesktopIconUI

String

8.6.2 JDesktopPane类

与内部窗体组合配合使用的另一个类就是JDesktopPane类。桌面面板的目的就是包含内部窗体集合。当内部窗体被包含在一个桌面面板中时,他们将其行为的大部分委托给桌面面板的桌面管理器。我们将会在本章稍后详细了解DesktopManager接口。

创建JDesktopPane

JDesktopPane只有一个无参数的构造函数。一旦创建,我们通常将其放在由BorderLayout管理的容器的中部。这可以保证桌面占据容器的所有空间。

将内部窗体添加到JDesktopPane

JDesktopPane并没有实现RootPaneContainer。我们并不能直接将组件添加到JRootPane内的不同面板中,而是直接将其添加到JDesktopPane:

desktop.add(anInternalFrame);

JDesktopPane属性

如表8-14所示,JDesktopPane有八个属性。位于allFrames属性数组索引0外的JInternalFrame是位于桌面前面的内部窗体(JInternalFrame f = desktop.getAllFrames()[0])。除了获取JDesktopPane中的所有窗体以外,我们还可以仅获取特定层的窗体:public JInternalFrame[] getAllFramesInLayer(int layer)。

可用的dragMode属性设置可以为类的LIVE_DRAG_MODE与OUTLINE_DRAG_MODE常量。

JDesktopPane属性

属性名
数据类型

访问性

accessibleContext
AccessibleContext

只读

allFrames
JInternalFrame[]

只读

desktopManager
DesktopManager

读写

dragMode
int

读写绑定

opaque
boolean

只读

selectedFrame
JInternalFrame

读写

UI
DesktopPaneUI

读写

UIClassID
String

只读

自定义JDesktopPane观感

回到图8-5,我们可以看到JDesktopPane中的JInternalFrame对象。JDesktopPane的基本观感与每一个观感相同。如表8-15所示,对JDesktopPane并没有太多可以配置的UIResource相关属性。

JDesktopPane UIResource元素

属性字符串

对象类型

desktop

Color

Desktop.ancestorInputMap

InputMap

Desktop.background

Color

Desktop.windowBindings

Object[]

DesktopPane.actionMap

ActionMap

DesktopPaneUI

String

完整的桌面示例

现在我们已经了解了主要的桌面相关类,现在我们来看一下完整的示例。基本的过程包括创建一组JInternalFrame对象,然后将放在一个JDesktopPane中。如果需要,可以对每一个内部窗体的单个组件进行事件处理,也可以对单个窗体进行事件处理。在这个示例中简单的使用了前面的列表8-5中所给出的InternalFrameIconifyListener类来监听正在图标化和取消图标化的内容窗体。

图8-6显示了程序启动时的样子。一个特定的内部窗体被设计为palette,并且允许了拖放模式。

列表8-6显示了这个示例的完整代码。

package swingstudy.ch08;
 
import java.awt.BorderLayout;
import java.awt.EventQueue;
 
import javax.swing.JDesktopPane;
import javax.swing.JFrame;
import javax.swing.JInternalFrame;
import javax.swing.JLabel;
import javax.swing.event.InternalFrameListener;
 
public class DesktopSample {
 
	/**
	 * @param args
	 */
	public static void main(final String[] args) {
		// TODO Auto-generated method stub
 
		Runnable runner = new Runnable() {
			public void run() {
				String title = (args.length==0 ? "Desktop Sample" : args[0]);
				JFrame frame = new JFrame(title);
				frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
 
				JDesktopPane desktop = new JDesktopPane();
				JInternalFrame internalFrames[] = {
						new JInternalFrame("Can Do All", true, true, true, true),
						new JInternalFrame("Not Resizable", false, true, true, true),
						new JInternalFrame("Not Closable", true, false, true, true),
						new JInternalFrame("Not Maximizable", true, true, false, true),
						new JInternalFrame("Not Iconifiable", true, true, true, false)
				};
 
				InternalFrameListener internalFrameListener = new InternalFrameIconifyListener();
 
				int pos = 0;
				for(JInternalFrame internalFrame: internalFrames) {
					// Add to desktop
					desktop.add(internalFrame);
 
					// Position and size
					internalFrame.setBounds(pos*25, pos*25, 200, 100);
					pos++;
 
					// Add listener for iconification events
					internalFrame.addInternalFrameListener(internalFrameListener);
 
					JLabel label = new JLabel(internalFrame.getTitle(), JLabel.CENTER);
					internalFrame.add(label, BorderLayout.CENTER);
 
					// Make visible
					internalFrame.setVisible(true);
 
				}
 
				JInternalFrame palette = new JInternalFrame("Palette", true, false, true, false);
				palette.setBounds(350, 150, 100, 100);
				palette.putClientProperty("JInternalFrame.isPalette", Boolean.TRUE);
				desktop.add(palette, JDesktopPane.PALETTE_LAYER);
				palette.setVisible(true);
 
				desktop.setDragMode(JDesktopPane.OUTLINE_DRAG_MODE);
 
				frame.add(desktop, BorderLayout.CENTER);
				frame.setSize(500, 300);
				frame.setVisible(true);
			}
		};
 
		EventQueue.invokeLater(runner);
	}
 
}

DesktopManager接口

使用桌面的最后一部分就是桌面管理器了,他是DesktopManager接口的实现,其定义如下:

当JInternalFrame位于JDe

public interface DesktopManager {
  public void activateFrame(JInternalFrame frame);
  public void beginDraggingFrame(JComponent frame);
  public void beginResizingFrame(JComponent frame, int direction);
  public void closeFrame(JInternalFrame frame);
  public void deactivateFrame(JInternalFrame frame);
  public void deiconifyFrame(JInternalFrame frame);
  public void dragFrame(JComponent frame, int newX, int newY);
  public void endDraggingFrame(JComponent frame);
  public void endResizingFrame(JComponent frame);
  public void iconifyFrame(JInternalFrame frame);
  public void maximizeFrame(JInternalFrame frame);
  public void minimizeFrame(JInternalFrame frame);
  public void openFrame(JInternalFrame frame);
  public void resizeFrame(JComponent frame, int newX, int newY, int newWidth,
    int newHeight);
  public void setBoundsForFrame(JComponent frame, int newX, int newY, int newWidth,
    int newHeight);
}

sktopPane中时,他们不应尝试例如图标化或是最大化的操作。相反,他们应该请求他们所安装在的桌面面板的桌面管理器来执行这些操作:

getDesktopPane().getDesktopManager().iconifyFrame(anInternalFrame);

DefaultDesktopManager类提供了DesktopManager的一个实现。如果默认实现还足够,观感会提供他们自己的DesktopManager实现类,例如Windows观感的WindowsDesktopManager。我们也可以定义自己的管理器,但是通常并不需要这样。

8.7 小结

在本章中,我们探讨了JRootPane类,以及如何实现依据JRootPane对内部组件进行管理的RootPaneContainer接口。我们同时了解了在Swing中我们如何使用JFrame, JDialog, JWindow, JApplet或是JInternalFrame类的JRootPane。根面板可以借助JLayeredPane来布局组件,其方式是工具提示文本以及弹出菜单总是显示在相关联组件的上面。

JInternalFrame同时也可以存在于桌面环境中,在这种情况下,JDesktopPane以及DesktopManager管理如何以及在哪里放置并显示内部窗体。我们还可以通过将InternalFrameListener实现也JInternalFrame关联来响应内部窗体事件。

在第9章中,我们将会探讨Swing库中的特定弹出组件:JColorChooser, JFileChooser, JOptionPane以及ProgressMonitor。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值