清华大学计算机系
王玉刚
---- Java AWT(抽象视窗工具库)提供了许多用户接口组件,JFC(Java 1.2推出的基础类) 虽扩展了工具库,但是标准组件仍然不能满足应用程序所要的功能,这就要我们自己开发我们自己的组件。
---- 设计定制组件类必须三思而后行,其中主要的问题是事件的处理。设计定制组件有三种方法。第一种方法是直接生成 Component 的子类,让子类完成所有的绘制功能和事件处理。这种方法定制的组件有全新的外观和感受。第二种方法是生成 Container或其它容器的子类,并在子类中加入标准组件,这使得组件可以交互工作,提供更高级的功能。第三种方法是生成标准非容器类组件的子类,如Button,TextArea等,从而加强继承的性能。下面我们就通过第一种方法的一个实例来看一看在定制组件过程中要注意的问题和如何处理事件。
---- 这个例子是在组件上画一个椭圆及椭圆上的一个点,该点和椭圆的两个焦点相连。首先要考虑的是椭圆的参数,椭圆要有长半轴和短半轴,而且还要经过一个点,这个点可以用鼠标取得,那么只要有椭圆的短半轴和长半轴的比就可以确定椭圆了。所以我们用辅助类 EllipseParam 来包装椭圆的短半轴和长半轴的比和要经过的那个点。
---- 清单1
---- 这个类将scale和point声明为专用数据,并提供了公共的访问器和设置器。这时可以给我们要定制的组件Ellipse类提供EllipseParam类型的专用实例变量value,以及设置和取得这个值的公共接口。
----
其次要考虑的是用户激活Ellipse组件是发出的事件类型。如果定制的组件发送标准事件类型,则比较方便。如果在标准的事件类型中没有合适的,则就要自己创造一个事件。本例中由于数据的类型是EllipseParam,所以我们只好自己创造一个事件来传输EllipseParam 类型的数据。
---- 在Java中,自己创造的事件一般是AWTEvent的子类。而且在构造方法中,传入上级类构造方法的第二个参数为事件的ID,编程人员生成新事件时不能用AWTEvent.RESERVER_ID_MAX以内的数值。
---- 清单2
EllipseEvent.java
package java2.ellipse;
import java.awt.AWTEvent;
public class EllipseEvent extends AWTEvent
{ private EllipseParam ellipseValue;
public EllipseEvent(Ellipse source,EllipseParam p)
{ super(source,AWTEvent.RESERVED_ID_MAX+1);
ellipseValue=p; }
public EllipseParam getEllipseValue()
{ return ellipseValue; }
public void setEllipseValue(EllipseParam p)
{ ellipseValue=p; }
}
EllipseEvent类很简单,但新建一个事件还需要事件监听器EllipseListener和在组件Ellipse 中加入和去除事件监听器的方法。事件监听器如下:
---- 清单3
EllipseListener.java
package java2.ellipse;
public interface EllipseListener extends java.util.EventListener
{ public void ellipseValueChanged(EllipseEvent ee);}
---- 这样任何接收Ellipse组件通知的对象都可以实现EllipseListener接口并提供ellipseValueChanged() 方法。在组件Ellipse中应提供addEllipseListener(EllipseListener listener)和removeEllipseListener (EllipseListener listener)方法。一种跟踪监听器的常用方法是利用向量,我们向Ellipse类提供一个 Vector的实例listeners。当用户通过鼠标单击或拖动时,事件处理器一方面要更新value的值,调用repaint() 方法,另一方面还要通知所有监听器,方法是调用方法notifyAllListener()。notifyAllListener()方法见清单4。
---- 清单4
---- 所有代码编译通过并在IE4.0上运行成功。