MyStopWatch


/*
 *  张老师Java高级教程GUI部分视频教程最后留的练习题
 *
 * 5、 为前边讲的自定义计时器组件增加功能:允许程序设置计时器显示文本的颜色,
 *   时间文本的字体大小随组件大小的改变而改变。
 *   
 * 思路:
 * 自定义一个组件类实现题目要求的功能
 *  a 要设置文本颜色,需要提供一个设置方法,可以用列表选择框或单选按钮等来实现
 *   这里用一组单选按钮实现,将它们放在一个单独的Panel里便于管理
 *  b 要让字体大小随组件大小的改变而改变,根据张老师视频中的方法就需要不断调整填充区域的大小
 *   这里换一种方法,用布局管理器来实现大小的自动调整,所以要用一个组件来实现计时器功能
 *  c 要将计时器组件和一组单选按钮Panel放在同一个组件里边实现题目要求的功能,
 *   为了方便将类设计为Panel的子类,就可以随意添加组件了
 *  d  要处理一组单选框按钮产生的事件,为了集中管理,让自定义的类实现ItemListener接口
 *   组件重绘过程中需要不断刷新时间值,需要一个单独的线程来执行这个任务,所以让类
 *   实现Runnable接口,让功能更加集中
 *
 * 已经实现的功能:
 *   计时功能已实现,在指定区域内(单选按钮左边的区域)按下鼠标开始计时,松开结束
 *   动态显示时间信息,每过去一秒都可以看得见
 *   通过点选指定的颜色按钮,动态改变计时器的文本颜色
 *   
 * 设置字体大小很容易,但根据窗口大小动态改变还真有点难住了。
 * 先试试看哪个可以监听到大小改变事件,下面的main函数中注释有编码时的思考过程
 * 最后发现给组件添加HierarchyBoundsListener,使用它的ancestorResized方法可以处理窗口大小改变的事件
 * 但是问题又来了,字体大小到底怎么根据窗口大小决定呢?
 * 用哪个方法获取到窗口大小,然后用一个固定数字除一下得到的值作为字体大小应该可以吧,试试看
 * 找到方法了,组将的getHeight和getWidth可以得到组件的宽高
 * 要动态设置字体大小,和动态改变颜色时方法差不多,所以要增加一个变量记录字体大小
 *
 * 大功告成,呵呵
 *
 */

package test.gui.stopwatch;
a
import java.awt.BorderLayout;
import java.awt.Checkbox;
import java.awt.CheckboxGroup;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.Font;
import java.awt.Frame;
import java.awt.Graphics;
import java.awt.GridLayout;
import java.awt.Label;
import java.awt.Panel;
import java.awt.event.HierarchyBoundsAdapter;
import java.awt.event.HierarchyEvent;
import java.awt.event.ItemEvent;
import java.awt.event.ItemListener;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.awt.event.WindowStateListener;
import java.text.SimpleDateFormat;
import java.util.Date;
/*
 * 	张老师Java高级教程GUI部分视频教程最后留的练习题
 * 
 * 5、	为前边讲的自定义计时器组件增加功能:允许程序设置计时器显示文本的颜色,
 * 		时间文本的字体大小随组件大小的改变而改变。
 * 		
 * 思路:
 * 自定义一个组件类实现题目要求的功能
 * 	a	要设置文本颜色,需要提供一个设置方法,可以用列表选择框或单选按钮等来实现
 * 		这里用一组单选按钮实现,将它们放在一个单独的Panel里便于管理
 * 	b	要让字体大小随组件大小的改变而改变,根据张老师视频中的方法就需要不断调整填充区域的大小
 * 		这里换一种方法,用布局管理器来实现大小的自动调整,所以要用一个组件来实现计时器功能
 * 	c	要将计时器组件和一组单选按钮Panel放在同一个组件里边实现题目要求的功能,
 * 		为了方便将类设计为Panel的子类,就可以随意添加组件了
 * 	d 	要处理一组单选框按钮产生的事件,为了集中管理,让自定义的类实现ItemListener接口
 * 		组件重绘过程中需要不断刷新时间值,需要一个单独的线程来执行这个任务,所以让类
 * 		实现Runnable接口,让功能更加集中
 * 
 * 已经实现的功能:
 * 		计时功能已实现,在指定区域内(单选按钮左边的区域)按下鼠标开始计时,松开结束
 * 		动态显示时间信息,每过去一秒都可以看得见
 * 		通过点选指定的颜色按钮,动态改变计时器的文本颜色
 * 		
 * 设置字体大小很容易,但根据窗口大小动态改变还真有点难住了。
 * 先试试看哪个可以监听到大小改变事件,下面的main函数中注释有编码时的思考过程
 * 最后发现给组件添加HierarchyBoundsListener,使用它的ancestorResized方法可以处理窗口大小改变的事件
 * 但是问题又来了,字体大小到底怎么根据窗口大小决定呢?
 * 用哪个方法获取到窗口大小,然后用一个固定数字除一下得到的值作为字体大小应该可以吧,试试看
 * 找到方法了,组将的getHeight和getWidth可以得到组件的宽高
 * 要动态设置字体大小,和动态改变颜色时方法差不多,所以要增加一个变量记录字体大小
 * 
 * 大功告成,呵呵
 * 
 */

public class MyStopWatch extends Panel implements ItemListener, Runnable
{
	//用Label作为计时器的主体,直接设置里边的文本即可
	Label label = new Label();
	//记录开始结束时间,默认为0
	long startTime, endTime;
	//记录计时器字体大小,便于动态调整字体大小,增加这个变量的原因下边注释有说明
	int fontSize = 25;
	//构造函数,让类对象一创建出来就具备想要的一些效果和功能
	public MyStopWatch()
	{
		//让自定义组件使用边界布局
		setLayout(new BorderLayout());
		//再用一个Label组件显示提示信息
		Label notic = new Label("在黑色区域按下鼠标开始计时,松开停止,单选按钮改变文字颜色");
		//设置提示文字颜色
		notic.setForeground(Color.RED);
		//设置提示文字对齐方式,居中
		notic.setAlignment(Label.CENTER);
		//设置提示文字字体
		notic.setFont(new Font("楷体_GB2312", Font.BOLD, 12));
		//将提示文字Label加入到自定义组件上方
		add(notic, "North");
		//将计时器主体Label加到自定义组件的中间
		add(label, "Center");
		//设置计时器的文本对齐方式
		label.setAlignment(Label.CENTER);
		//定义一个选择框组,管理用于改变颜色的一组单选按钮
		CheckboxGroup checkgroup = new CheckboxGroup();
		//创建3个颜色单选按钮,添加到同一个组中
		Checkbox cb_green = new Checkbox("绿色", false, checkgroup);
		Checkbox cb_red = new Checkbox("红色", true, checkgroup);
		Checkbox cb_white = new Checkbox("白色", false, checkgroup);
		//定义一个单独的Panel,集中存放单选按钮
		Panel checks = new Panel();
		//设置单选按钮区域的前景色和背景色
		checks.setBackground(Color.BLACK);
		checks.setForeground(Color.GREEN);
		//设置单选按钮区域的字体
		checks.setFont(new Font("楷体_GB2312", Font.BOLD, 15));
		//设置单选按钮面板的布局方式为网格布局,添加一列三行按钮
		checks.setLayout(new GridLayout(3, 1));
		checks.add(cb_green);
		checks.add(cb_red);
		checks.add(cb_white);
		//将颜色选择面板加入到自定义组件的右边
		add(checks, "East");
		//为3个单选按钮添加事件监听器,自定义组件本身已经实现了ItemListener接口
		//其本类对象就可以作为监听器对象使用
		cb_green.addItemListener(this);
		cb_red.addItemListener(this);
		cb_white.addItemListener(this);
		
		//**为计时器添加一个监听器,用来检测大小改变,以实现动态改变文字大小
		//查来查去发现好像这个才能实现吧。试过后确实可以,开工了
		label.addHierarchyBoundsListener(new HierarchyBoundsAdapter()
		{
			public void ancestorResized(HierarchyEvent e)
			{
				//System.out.println("大小变了");
				//看看哪个方法可以获取到窗口大小,找到了,第一次width:321,
				//字体大小也需要定义一个变量进行保存,和设置颜色类似,初始为25
				//int height = label.getHeight();
				//得到计时器的宽度,这里根据宽度计算字体大小
				int width = label.getWidth();
				//System.out.println("Width:"+width+"--Height:"+height);
				fontSize = width/12;
				//字体大小改变后,进行重绘
				repaint();
			}			
		});
		
		//为计时器添加鼠标监听器,实现计时功能		
		label.addMouseListener(new MouseAdapter()
		{
			//鼠标按下时的处理方式
			public void mousePressed(MouseEvent e)
			{
				//记录一下,表示鼠标已经按下,可以刷新时间值了
				isPressed = true;
				//开启线程,不断刷新时间值,以便实现时间动态显示的效果
				new Thread(MyStopWatch.this).start();
				//开始计时
				startTime = endTime = System.currentTimeMillis();
				//重绘组件,以免上次的时间值遗留在组件表面
				repaint();
			}
			//鼠标释放时的处理方式
			public void mouseReleased(MouseEvent e)
			{
				//将结束时间置为当前时间
				endTime = System.currentTimeMillis();
				//改变标记值,停止刷新
				isPressed = false;
				//再次重绘,将最后的时间值刷一下
				repaint();
			}
		});
	}
	//定义一个记录颜色值的变量,默认红色,便于颜色的动态改变
	Color color = Color.RED;
	//覆盖paint方法,实现自定义的重绘效果
	public void paint(Graphics g)
	{
		//定义SimpleDateFormat对象,便于将时间值进行格式化
		SimpleDateFormat sdf = new SimpleDateFormat("HH:mm:ss");
		//计算鼠标按下到释放耗用的时间,因为计算机内部时间表示方式的差异,
		//与我们想要的效果(刚开始显示00:00:00)相差了8个小时,所以减掉8个小时
		Date costTime = new Date(endTime-startTime-8*60*60*1000);
		//将耗用的时间值格式化为我们想要的字符串形式
		String strTime = sdf.format(costTime);
		//设置计算器的背景色为黑色,前景色动态获取
		label.setBackground(Color.BLACK);
		label.setForeground(color);
		//设置计时器表面字体
		label.setFont(new Font("楷体_GB2312", Font.BOLD, fontSize));
		//将时间值显示在计算器表面
		label.setText(strTime);
	}
	//实现ItemListener接口中的方法,让自定义组件具有监听器功能
	public void itemStateChanged(ItemEvent e)
	{
		//定义一个Checkbox变量,记录产生事件的具体对象
		Checkbox cb = (Checkbox)e.getItemSelectable();
		//获取事件源表面的文字信息,便于对比和进行相应设置
		String strColor = cb.getLabel();
		if (strColor!=null)//如果获取到的信息不为空,应该不会空吧
		{
			//根据单选按钮表面的文字信息,动态修改计时器的前景色
			if (strColor.equals("绿色"))
			{
				color = Color.GREEN;
				label.setForeground(Color.GREEN);
			}
			else if (strColor.equals("白色"))
			{
				color = Color.WHITE;
				label.setForeground(Color.WHITE);
			}
			else
			{
				color = Color.RED;
				label.setForeground(Color.RED);
			}	
		}
	}
	//定义标记变量,记录鼠标按下状态,默认没有按下
	boolean isPressed = false;
	//实现Runnable接口,复写其中的run方法,让自定义组件具有自动运行指定任务的功能
	public void run()
	{
		//判断鼠标状态,如果按下就开始执行任务
		while (isPressed)
		{
			//让线程稍微休息一会,不能休息太长时间了,不然就不能实现较好的动态效果了
			try
			{
				Thread.sleep(500);
			} catch (InterruptedException e)
			{
				e.printStackTrace();
			}
			//将结束时间修改为当前时间,以便重绘后达到动态显示时间的效果
			endTime = System.currentTimeMillis();
			//对组件进行重绘,更新时间值
			repaint();
		}
	}
	
	//主函数,测试使用,也可以直接新建一个类,在Frame中加入一个此类对象
	public static void main(String[] args)
	{
		Frame frame = new Frame("My own StopWatch");
		//在窗口中加入一个上边的自定义组件
		frame.add(new MyStopWatch());
		frame.setSize(380, 180);
		//设置窗口最小值,以免用户调的太小,组件都挤不下难看死了
		frame.setMinimumSize(new Dimension(380, 180));
		//固定大小,用户不可调整
		//frame.setResizable(false);
		//给窗口添加窗口监听器,处理窗口事件
		frame.addWindowListener(new WindowAdapter()
		{
			//指定默认关闭事件处理方式
			public void windowClosing(WindowEvent e)
			{
				System.exit(0);				
			}
			//想通过这个事件来动态调整计时器内的文本大小,发现这个好像没什么效果,
			//不知道具体有什么用,文档中只是简单地说状态i改变,到底什么状态呢
			public void windowStateChanged(WindowEvent e)
			{
				System.out.println("窗口大小改变了");
			}
		});
		//发现这个只有在最大化、最小化时起效果
		frame.addWindowStateListener(new WindowStateListener()
		{
			public void windowStateChanged(WindowEvent e)
			{
				System.out.println("窗口状态改变了");
			}
		});
		//设置窗口左上角的小图标
		//frame.setIconImage(frame.getToolkit().createImage("c:/logo.jpg"));
		frame.setVisible(true);
	}

}


 

 

你可能感兴趣的:(学习笔记,GUI,秒表)