大 连 理 工 大 学 计 算 机 系
邓继光 翟丽萍 张丽滨
---- 一.Java 简 介
---- Java 技 术 是 一 个 崭 新 的 技 术, 它 从C++ 中 吸 收 了 许 多 强 有 力 的 成 分, 并 且 带 有 许 多 新 的 特 征, 如
? ? ?
---- Java 标 准 类 库 中 对Internet 和web 内 建 的 支 持, 使 得 开 发 者 可 方 便 地 开 发 网 络 应 用。
---- 自 从Java 诞 生 以 来, 众 多 的 公 司 已 致 力 于Java 开 发 平 台 的 研 制,Sun,Microsoft,Borland 和Symantec 争 相 推 出 了 自 己 的Java 开 发 平 台。JavaSoft 公 司 推 出 的JDK 以 其 灵 活 性 和 强 大 功 能 得 到 了 众 多 开 发 者 的 青 睐。
---- 二. JDK1.1.1 中 的 菜 单 功 能
---- 在JDK1.1.1 中 菜 单 功 能 具 体 由java.awt.Menu ,java.awt.MenuBar ,java.awt.MenUComponent ,java.awt.MenuItem ,java.awt.MenuShortcut 实 现, 其 继 承 关 系 如 下:
java.lang.object
java.awt.MenuComponent
java.awt.MenuItem
java.awt.Menu
java.awt.PopupMenu
java.awt.MenuBar
java.awt.MenuShortcut
---- 在java.awt.Frame 中 实 现 了java.awt.MenuContainer, 利 用 其setMenuBar 方 法 即 可 提 供 菜 单 服 务。 但 一 个Application 中 只 应 有 一 个Frame(Dialog 除 外) 同 时 存 在, 否 则 所 有 的component 不 能 构 成 一 个 整 体(Frame 可 被 自 由 拖 动), 这 样 如 仅 利 用JDK1.1.1 提 供 的 功 能,Application 中 只 能 存 在 一 个 菜 单。 由Frame 的
---- ? ? ?
---- Frame 是 一 个Container, 它 可 以 装 入 多 个Panel, 如 能 在Panel 中 实 现 菜 单 功 能, 就 可 以 在Application 中 提 供 多 菜 单 服 务 了。
---- 三. 在Panel 中 实 现 菜 单 功 能
---- 在Panel 中 实 现 菜 单 功 能 可 有 如 下 三 种 途 径
---- 1. 模 仿java.awt.Frame 的 菜 单 服 务, 在Panel 中 加 入 菜 单 功 能
---- 2. 利 用JDK 提 供 的 部 分 功 能, 和 自 己 的component 组 合 完 成
---- 3. 完 全 用 自 己 的component 实 现
---- java.awt.Frame 和java.awt.Panel 各 自 的 继 承 关 系 如 下:
java.lang.Object
java.awt.Component
java.awt.Container
java.awt.Window
java.awt.Frame
java.awt.Panel
---- 由 继 承 关 系 可 以 看 出, 要 模 仿java.awt.Frame 的 菜 单 服 务, 在Panel 中 加 入 菜 单 功 能, 是 比 较 困 难 的, 况 且,JDK 文 档 中 并 未 提 供 完 整 的 源 代 码。
---- 如 要 完 全 用 自 己 的component 实 现, 则 需 考 虑 复 杂 的 屏 幕 刷 新 和 事 件 处 理 问 题, 也 是
---- ? ? ?
---- 而 利 用JDK1.1.1 的java.awt.Component 中 提 供 的PopupMenu 功 能, 在 上 面 加 上label, 在 鼠 标 点 击label 时, 弹 出PopupMenu, 以 此 模 仿Frame 提 供 的 菜 单 服 务, 可 以 较 容 易 的 在Panel 中 加 入 菜 单 功 能。 具 体 实 现 如 下:
---- // 此listener 用 于 发 出MenuBarLabel 点 击 事 件
public interface MenuBarLabelListener
{
public void ProcessLabelClick(String labelName);
}
import java.awt.* ;
import java.awt.event.* ;
import java.util.*;
//MenuBarLabel 用 于 模 拟java.awt.MenuBar 的 功 能
public class MenuBarLabel extends
Canvas implements MouseListener
{
protected Color Foreground = Color.black ;
protected Color Background = Color.lightGray;
protected String xlabel ;
protected boolean disabled = false ;
protected MenuBarLabelListener listener=null;
public MenuBarLabel(String xlabel)
{
this.xlabel = new String(xlabel);
setFont(new Font("Courier",Font.PLAIN,16));
setBackground(Background);
addMouseListener(this);
}
public void addListener(MenuBarLabelListener listener)
{
this.listener=listener;
}
public String getName()
{
return xlabel;
}
public void paint(Graphics g)
{
Dimension dm = getSize();
int width = dm.width;
int height= dm.height;
g.setColor(Foreground);
int descent = g.getFontMetrics().getDescent();
g.drawString(xlabel,0,height-descent);
}
public void mouseClicked(MouseEvent evt){ ;}
public void mouseEntered(MouseEvent evt){ ;}
public void mousePressed(MouseEvent evt)
{
Background=Color.blue.darker().darker();
paint();
if (null !=listener)
listener.ProcessLabelClick(xlabel);
}
public void mouseExited(MouseEvent evt)
{
Background=Color.lightGray;
setBackground(Background);
}
public void mouseReleased(MouseEvent evt)
{
Background=Color.lightGray;
setBackground(Background);
}
public void setForeground(Color Foreground)
{
this.Foreground = Foreground;
repaint();
}
public void setBackground(Color Background)
{
this.Background = Background;
repaint();
}
public void paint()
{
Graphics g=getGraphics();
if (null==g) return;
Dimension dm = getSize();
int width = dm.width;
int height= dm.height;
int descent = g.getFontMetrics().getDescent();
int height1=g.getFontMetrics().getHeight();
g.setColor(Color.blue.darker().darker());
g.fillRect(0,height-height1,width,height1);
g.setColor(Color.white);
g.drawString(xlabel,0,height-descent);
g.dispose();
}
public void paintBack()
{
Graphics g=getGraphics();
if (null==g) return;
Dimension dm = getSize();
int width = dm.width;
int height= dm.height;
int descent = g.getFontMetrics().getDescent();
g.setColor(Color.lightGray);
g.fillRect(0,0,width,height);
g.setColor(Foreground);
g.drawString(xlabel,0,height-descent);
g.dispose();
}
}
// 此listener 用 于 提 供 所 有 的 菜 单 选 择 事 件
public interface MenuPanelListener
{
public void ProcessMyMenu(Object source,Object label);
//you can use source to indentify
different items with same label
}
//MenuPanel 提 供 菜 单 功 能 服 务
import java.awt.*;
import java.awt.event.*;
import java.util.*;
/**
*Pay attention:
*1.The MenuPanel′s height >==26, 26 recommended
*2.You can not change the
background of the menu because Popup
* menu′s backgroud can not be changed
*/
public class MenuPanel extends Panel implements
ActionListener,MenuBarLabelListener
{
Panel MyMenuBar=new Panel();
Panel popUpPanel=new Panel();
MenuBarLabel previousLabel=null;
MenuPanelListener listener=null;
Font labelFont=new Font("Courier",Font.PLAIN,16);
private Vector labelVector=new Vector();
private Vector popUpVector=new Vector();
int maxwidth=0;
public MenuPanel()
{
setLayout(null);
MyMenuBar.setLayout(null);
MyMenuBar.setBackground(Color.lightGray);
add(MyMenuBar);
add(popUpPanel);
MyMenuBar.setBounds(0,0,400,25);
popUpPanel.setLayout(null);
popUpPanel.setBounds(0,26,400,500);
popUpPanel.setVisible(false);
setFont(labelFont);
}
public void addListener(MenuPanelListener listener)
{
this.listener=listener;
}
private void addMenuItemListener(MenuComponent menu)
{
((MenuItem)menu).addActionListener(this);
if (menu instanceof Menu menu instanceof PopupMenu)
for (int i=0;i< ((Menu)menu).getItemCount();i++)
addMenuItemListener(((Menu)menu).getItem(i));
}
private void setMenuItemFont(MenuComponent menu)
{
menu.setFont(labelFont);
if (menu instanceof Menu menu instanceof PopupMenu)
for (int i=0;i< ((Menu)menu).getItemCount();i++)
setMenuItemFont(((Menu)menu).getItem(i));
}
public void addMenu(PopupMenu menu,String name)
{
if (menu==null name==null name.length()< =0)
return;
MenuBarLabel label=new MenuBarLabel(name);
label.setFont(labelFont);
label.addListener(this);
MyMenuBar.add(label);
labelVector.addElement(label);
setMenuItemFont(menu);
addMenuItemListener(menu);
popUpVector.addElement(menu);
popUpPanel.add(menu);
}
public void setMenuFont(Font menuFont)
{
int fontSize=menuFont.getSize();
if (menuFont.getSize() >25)
fontSize=25;
labelFont=new Font(menuFont.getName(),
menuFont.getStyle(),fontSize);
setFont(labelFont);
for (int i=0;i< popUpVector.size();i++)
setMenuItemFont(((PopupMenu)
popUpVector.elementAt(i)));
repaint();
}
public void paint(Graphics g)
{
MenuBarLabel label;
int width=0,width1;
int panelWidth=getParent().getSize().width;
if (panelWidth >0)
{
setSize(panelWidth,getSize().height);
MyMenuBar.setBounds(0,0,panelWidth,25);
popUpPanel.setBounds(0,26,panelWidth,500);
}
for (int i=0; i< labelVector.size();i++)
{
label=(MenuBarLabel)labelVector.elementAt(i);
label.setFont(labelFont);
width1=g.getFontMetrics().stringWidth(label.getName());
if (width1 >width)
width=width1;
}
if (width !=maxwidth)
{
maxwidth=width;
for (int i=0; i< labelVector.size();i++)
{
label=(MenuBarLabel)labelVector.elementAt(i);
label.setBounds(i*maxwidth+(i+1)*15,0,maxwidth,25);
}
}
}
public void ProcessLabelClick(String labelName)
{
MenuBarLabel label=null;
boolean inVector=false;
int i=0;
while (i< labelVector.size())
{
label=(MenuBarLabel)labelVector.elementAt(i);
if ((label.getName()).equals(labelName))
{
inVector=true;
break;
}
i++;
}
if (inVector)
{
if ((null !=previousLabel)&&
(!previousLabel.equals(label)))
previousLabel.paintBack();
previousLabel=label;
if (((PopupMenu)popUpVector.element
At(i)).getItemCount() >0)
{
popUpPanel.setVisible(true);
((PopupMenu)popUpVector.elementAt(i)).
show(popUpPanel,i*maxwidth+
(i+1)*15,0);
}
}
}
public void actionPerformed(ActionEvent evt)
{
if(null!=listener)
listener.ProcessMyMenu(evt.getSource(),
evt.getActionCommand());
}
}
---- 四. 应 用 举 例
---- 本 例 程 利 用 上 面 的MenuPanel, 在TestMenu 中 实 现 了 带 两 个 下 拉 菜 单 的 菜 单。
import java.awt.*;
public class TestMenu extends Frame
implements MenuPanelListener
{
public TestMenu()
{
setLayout(null);
setSize(400,300);
MenuPanel myPanel=new MenuPanel();
MenuPanel.addListener(this);
// 在myPanel 中 加 入listener
PopupMenu popupMenu1=new PopupMenu("menu1");
//"menu1" 即 为 显 示 的label
Menu subMenu1=new Menu("test1");
subMenu1.addMenuItem(new MenuItem("item1"))
subMenu1.addMenuItem(new MenuItem("item2"))
Menu subMenu2=new Menu("test2");
subMenu2.addMenuItem(new MenuItem("item3"))
subMenu2.addMenuItem(new MenuItem("item4"))
Menu subMenu3=new Menu("test3");
subMenu3.addMenuItem(new MenuItem("item5"))
subMenu3.addMenuItem(new MenuItem("item6"))
popupMenu1.addMenuItem(subMenu1);
popupMenu1.addMenuItem(subMenu2);
popupMenu1.addMenuItem(subMenu3);
myPanel.addMenu(popupMenu1);
PopupMenu popupMenu2=new PopupMenu
("menu2");//"menu2" 即 为 显 示 的label
Menu subMenu4=new Menu("test4");
subMenu4.addMenuItem(new MenuItem("item7"))
subMenu4.addMenuItem(new MenuItem("item8"))
Menu subMenu5=new Menu("test5");
subMenu5.addMenuItem(new MenuItem("item9"))
subMenu5.addMenuItem(new MenuItem("item10"))
Menu subMenu6=new Menu("test6");
subMenu6.addMenuItem(new MenuItem("item11"))
subMenu6.addMenuItem(new MenuItem("item12"))
popupMenu2.addMenuItem(subMenu4);
popupMenu2.addMenuItem(subMenu5);
popupMenu2.addMenuItem(subMenu6);
myPanel.addMenu(popupMenu2);
myPanel.setBounds(0,0,400,26);
add(MenuPanel);
}
public void ProcessMyMenu(Object source,Object label)
{ System.out.println((String)label);
}
protected void processEvent(AWTEvent evt)
{ if (evt.id == Event.WINDOW_DESTROY) System.exit(0);
}
}
---- 注: 本 文 程 序 在Win95 下, 用JDK1.1 调 试 通 过。