java记事本开发

今天我想要分享的是一个用java写的一个记事本程序。我知道现在市面上有各种各样的记事本了,但是我发现有的写的功能不够完善,或者是代码层次结构不够清晰,有的甚至看了之后云里雾里的,有的还不乏了非常明显的bug,我现在分享的这个记事本程序基本上把代码层次都抽分出来,并修复了一些已知bug。先看一下界面效果图,快捷键我都已经全部加上了,只是没有在界面上标明而已,一般常用的那几个快捷键都直接使用即可!


该程序主要功能有:打开、保存、另存为、退出、新建、黏贴、复制、全选、剪切、删除、查找、替换、转到、修改字体、自动换行、查看帮助等功能。目前未完善的功能有自动显示标题。

首先我是使用了swt这个插件在eclipse中开发的,该程序在跨平台使用时会有稍许不同之处!安装swt在这里就不再重复说明,我们直接来新建工程吧!


一、新建一个swt工程,命名为Note,这个时候我们说需要依赖的swt中的jar包会自动导入了,如果你的有导入jar包的问题,你也可以手工导入jar包。

之后建立3个包来存放我们的类:dialog(主要是一些弹窗类的实现)、noteUi(存放主要的实现类)、Utils(抓哟存放一些公有的工具文件)。

二、在noteUI包中新建一个住类,我们命名为Note1.java好了,额,忘记说了,这个Note1.java是一个swt的new Appliction Windows的类,在里面就可以通过design来制作这个窗口的布局,主要就是menu bar,等,具体布局方式可以看这个图:


布局写好之后,我们就可以添加各种事件了

三、添加快捷键

//快捷键部分
		open.setAccelerator(SWT.CTRL |'o');
		save_file.setAccelerator(SWT.CTRL |'s');
		quit.setAccelerator(SWT.CTRL |'p');
		select_all.setAccelerator(SWT.CTRL + 'a');
		new_file.setAccelerator(SWT.CTRL |'n');
		new_file.setAccelerator(SWT.CTRL + 'N'); //新建文件快捷键
        undo.setAccelerator(SWT.CTRL + 'Z');     //撤销快捷键
        cut.setAccelerator(SWT.CTRL + 'T');      //剪切快捷键
        copy.setAccelerator(SWT.CTRL + 'C');     //复制快捷键
        paste.setAccelerator(SWT.CTRL + 'V');    //粘贴快捷键
        delete.setAccelerator(SWT.DEL);          //删除快捷键
        find.setAccelerator(SWT.CTRL + 'F');     //查找快捷键
        find_next.setAccelerator(SWT.F3);        //查找下一处快捷键
        replace.setAccelerator(SWT.CTRL + 'H');  //替换快捷键
        go_to.setAccelerator(SWT.CTRL + 'G');    //转到快捷键

四、我们先从简单的做起吧!,先写查看帮助吧!

MenuItem look_help = new MenuItem(menu_5, SWT.NONE);
		look_help.addSelectionListener(new SelectionAdapter() {
			@Override
			public void widgetSelected(SelectionEvent e) {
				UiUtils.showMessageDialog(shell, "帮助","欢迎查看我的博客:http://blog.csdn.net/sdksdk0!");
			}
		});
		look_help.setText("查看帮助");
当然,对于这个“关于我们”的话可以使用一个弹窗来写,新建一个swt窗口文件,那么我们可以这样来调用:这个界面是具体的实现跨越看我文末提供的源码。

	MenuItem menuItem_15 = new MenuItem(menu_6, SWT.NONE);
		menuItem_15.addSelectionListener(new SelectionAdapter() {
			@Override
			public void widgetSelected(SelectionEvent e) {
				About_us as=new About_us();
				as.open();
			}
		});

五、然后我们可以来看一个字体的设置,主要是调用系统本身的api就可以了,调用这个窗体然后来使用!更改颜色的部分我暂时只加了三种颜色,其他的颜色你可以自己加上,

//字体设置
		font.addSelectionListener(new SelectionAdapter() {
			@Override
			public void widgetSelected(SelectionEvent e) {
				FontDialog fd=new FontDialog(shell,SWT.None);
				Font oldFont=getText().getFont();
				if(oldFont!=null){
					fd.setFontList(oldFont.getFontData());
				}
				FontData fontData=fd.open();
				if(fontData==null){
					return;
				}
				
				Color c=new Color(shell.getDisplay(),fd.getRGB().red,fd.getRGB().green,fd.getRGB().blue);
				text.setForeground(c);	
				final Display display=Display.getDefault();
				Font newFont=new Font(display,fontData);
				getText().setFont(newFont);
				if(oldFont!=null){
					oldFont.dispose();
				}	
			}
		});
六、在这个记事本的最下面一行有一个可以记录当前时间的,我们可以使用Data来写,然后通过SimpleDateFormat来格式化显示日期时间。

Label lblNewLabel = new Label(sashForm, SWT.NONE);
		SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日  HH:mm:ss");

		lblNewLabel.setText("指令汇科技欢迎您的使用,现在是:" + sdf.format(new Date()));
		sashForm.setWeights(new int[] {309, 24});

七、我们可以来看一下复制、黏贴、剪切、全选的书写,其他这个应该是最简单了了,因为这个可以直接调用以下方法。全选就直接使用text.selectAll();就可以了。

MenuItem cut = new MenuItem(menu_2, SWT.NONE);
		cut.addSelectionListener(new SelectionAdapter() {
			@Override
			public void widgetSelected(SelectionEvent e) {
				getText().cut();
			}
		});
		cut.setText("剪切");

		MenuItem copy = new MenuItem(menu_2, SWT.NONE);
		copy.addSelectionListener(new SelectionAdapter() {
			@Override
			public void widgetSelected(SelectionEvent e) {
				text.copy();
			}
		});
		copy.setText("复制");

		MenuItem paste = new MenuItem(menu_2, SWT.NONE);
		paste.addSelectionListener(new SelectionAdapter() {
			@Override
			public void widgetSelected(SelectionEvent e) {
				getText().paste();
			}
		});
		paste.setText("黏贴");

八、这个时候我们来写一下查找替换等,这个时候新建一个弹出的方法更好了,


/**
	 * Create contents of the dialog.
	 */
	private void createContents() {
		shell = new Shell(getParent(), getStyle());
		shell.setSize(445, 137);
		shell.setLocation(780, 300);
		shell.setText("\u67E5\u627E");
		
		textContent = new Text(shell, SWT.BORDER);
		textContent.addKeyListener(new KeyAdapter() {
			/**
			 * 为查找编辑框添加事件,以同步查找按钮的更新
			 * 有内容则显示查找,否则显示为不可点击
			 */
			public void keyReleased(KeyEvent e) {
				if(textContent.getText() != ""){
					find_next.setEnabled(true);
					chazhao = true;
				}else{
					find_next.setEnabled(false);
					chazhao = false;
				}
			}
		});
		textContent.setBounds(99, 10, 204, 22);
		
		find_next = new Button(shell, SWT.NONE);
		find_next.setEnabled(false);
		find_next.addSelectionListener(new SelectionAdapter() {
			@Override
			public void widgetSelected(SelectionEvent e) {
				pareStr = Util.shell.getText_1(); //获取主窗口文本框内容  	 true
				index = pareStr.indexOf(textContent.getText(),index);  //获取查找串在母串中的从index处开始的位置
				if(index == -1){
					JOptionPane.showMessageDialog(null, "找不到\""+textContent.getText()+"\"","记事本",  JOptionPane.ERROR_MESSAGE); 
				}
				int end = index+textContent.getText().length();//获得渲染的结束位置
				Util.shell.selectContent(index, end);
				index = index+textContent.getText().length();	
			}
		});
		find_next.setBounds(329, 6, 99, 27);
		find_next.setText("\u67E5\u627E\u4E0B\u4E00\u5904(F)");
		
		Label lblNewLabel = new Label(shell, SWT.NONE);
		lblNewLabel.setBounds(10, 10, 85, 22);
		lblNewLabel.setText("\u67E5\u627E\u5185\u5BB9(&N):");
		
		Button quxiao = new Button(shell, SWT.NONE);
		quxiao.addSelectionListener(new SelectionAdapter() {
			/**
			 * 为取消按钮设置事件
			 */
			public void widgetSelected(SelectionEvent e) {
				shell.setVisible(false);
			}
		});
		quxiao.setBounds(329, 65, 100, 27);
		quxiao.setText("\u53D6\u6D88");
		
		Button qufen = new Button(shell, SWT.CHECK);
		qufen.addSelectionListener(new SelectionAdapter() {
			/**
			 * 判断是否区分大小写,若是,则改变
			 */
			public void widgetSelected(SelectionEvent e) {
				qf = qufen.getSelection();
			}
		});
		qufen.setBounds(10, 65, 149, 27);
		qufen.setText("\u533A\u5206\u5927\u5C0F\u5199(&C)");
		
		Label label1 = new Label(shell, SWT.NONE);
		label1.setText("\u66FF\u6362\u4E3A(&F):");
		label1.setBounds(10, 38, 72, 22);
		
		replaceText = new Text(shell, SWT.BORDER);
		replaceText.addKeyListener(new KeyAdapter() {
			/**
			 * 为替换编辑框添加事件,以同步替换按钮的更新
			 * 有内容则显示替换,否则显示为不可点击
			 */
			public void keyReleased(KeyEvent e) {
				if(replaceText.getText() != ""){
					replaceBtn.setEnabled(true);
				}else{
					find_next.setEnabled(false);
				}
			}
		});
		replaceText.setBounds(99, 38, 204, 22);
		
		replaceBtn = new Button(shell, SWT.NONE);
		replaceBtn.addSelectionListener(new SelectionAdapter() {
			/**
			 * 替换按钮事件
			 */
			public void widgetSelected(SelectionEvent e) {
				pareStr = Util.shell.getText_1(); //获取主窗口文本框内容  	 true
				pareStr = pareStr.replaceFirst(textContent.getText(), replaceText.getText());
				Util.shell.showText(pareStr);
			}
		});
		replaceBtn.setText("\u66FF\u6362(&R):");
		replaceBtn.setEnabled(false);
		replaceBtn.setBounds(329, 32, 99, 27);

我们在之前的主类,即Note1中添加相应的调用就可以了,例如:
replace.addSelectionListener(new SelectionAdapter() {
			@Override
			public void widgetSelected(SelectionEvent e) {
				FindWindow fw = new FindWindow(shell,SWT.DIALOG_TRIM);
				fw.open();
			}
		});
查找替换等都是一样的,这里就不再重复说明。

九、至于跳转的话这样就可以了:

	queding.addSelectionListener(new SelectionAdapter() {
			/**
			 * 确定跳转到某行
			 */
			public void widgetSelected(SelectionEvent e) {
				int n = Integer.parseInt(goto_line.getText());
				goto_line.setText(""+ n + "");
				Util.shell.gotoOneLine(1);
			}
		});

十、接下来就是最重要的文件打开 和保存等操作了,一般这个地方的bug最多

那我们就先来看看打开文件的操作吧,我使用的是gbk编码来读取文件的,我之前发现如果默认编码的话读取文件有乱码现象,所以我就直接加gbk了,简单粗暴 

// 打开
		open.addSelectionListener(new SelectionAdapter() {
			@Override
			public void widgetSelected(SelectionEvent e) {
				FileDialog fd = new FileDialog(shell, SWT.None);
				String file = fd.open();
				File f = new File(file);
				BufferedInputStream bis = null;
				if (f.exists() && f.isFile()) {
					try {
						bis = new BufferedInputStream(new FileInputStream(f));
						byte[] bs = new byte[1024];
						int length = 0;
						StringBuffer sb = new StringBuffer();
						while ((length = bis.read(bs, 0, bs.length)) != -1) {
							sb.append(new String(bs, 0, length,"gbk"));
						}
						text.setText(sb.toString());
					} catch (Exception e1) {
						e1.printStackTrace();
						UiUtils.showMessageDialog(shell, "出错了",e1.getMessage());
					} finally {
						if(bis!=null){
							try {
								bis.close();
							} catch (IOException e1) {
								UiUtils.showMessageDialog(shell, "出错了",e1.getMessage());
							}
						}
					}

				} else {
					UiUtils.showMessageDialog(shell, "出错了",
							"文件" + f.getName() + "不存在");
				}
			}
		});

十一、之后就是写保存文件的代码了,保存的时候有两种方法,一种是直接保存,另一种是另存为。我们采用FileOutputStream来从程序写出到文件中。

public class SaveMethod {
	/**
	 * 将保存的两种方法分装成类,方便调用
	 * 保存 
	 * 另存为
	 */
	public void Save(){
		
		/**
		 * 保存文件
		 */
		if(Util.shell.fileDir!=null ){ //表示该文件有目录,保存时直接保存,不需要弹窗
			Util.shell.shell.setText((new File( Util.shell.fileDir.trim())).getName());//获取文件名(不含路径)用于设置title
			FileOutputStream fileWriter;
				try {
					fileWriter = new FileOutputStream(Util.shell.fileDir);
					BufferedWriter out = new BufferedWriter(new OutputStreamWriter(fileWriter,"gbk"));
					out.write(Util.shell.getText().getText());
					out.close();
					fileWriter.close();
				} catch (IOException e1) {
					e1.printStackTrace();
				}
		}
		else{//否则为从未保存过,则调用另存为窗口
			SaveAs();
		}
	}
	
	public void SaveAs(){
		/**
		 * 将另存为这个方法进行封装,方便调用
		 */
		FileDialog dialog = new FileDialog(Util.shell.shell,SWT.SAVE);
	     dialog.setFilterPath(System.getProperty("C:\\Documents and Settings"));//设置打开默认的路径
	     dialog.setFilterExtensions(new String[]{"*.txt","*.sql","*.java","*.doc"});
	     String file = dialog.open();//打开窗口,返回用户所选的文件目录
	     if(file != null){
	    	 Util.shell.fileDir = file;//将文件目录保存下来,方面之后的使用
	     }
	     if ( file != null )
	     {
	    	 Util.shell.shell.setText((new File( file.trim())).getName());//获取文件名(不含路径)用于设置title
	    	 FileOutputStream fileWriter;
			try {
				fileWriter = new FileOutputStream(file);
				BufferedWriter out = new BufferedWriter(new OutputStreamWriter(fileWriter,"gbk"));
				out.write(Util.shell.getText().getText());
				out.close();
				fileWriter.close();
			} catch (IOException e1) {
				e1.printStackTrace();
			}
	     }
	}
}

十二、最后我们写一下退出吧,如果你想偷懒的话直接用System.exit(0);也是可以实现这个功能的,但是我觉得吧!这样直接退出不太友好吧!所以我写了下面这种优化的退出方法!

若是保存则说明存在fileDir,直接调用保存的函数即可,若不存在fileDir则说明文件时新打开的,还没确定过路径,因此跳转另存为窗体。 
首先定义一个全局变量String fileDir = null;将其赋值为空,默认所有的文件都未保存 
接着不管是打开文件,还是另存为文件,都将文件所在目录(包括文件名以及后缀名)记录在fileDir中,最后在保存事件中确定fileDir是否为空即可。

public void quit(){
		/**
		 * 退出函数退出时进行判断是否需要保存
		 */
		String tips;
		if(Util.shell.fileDir != null || Util.shell.getText().getText() != ""){  //文件目录不为空,说明有打开着的文件,需要询问是否保存
			if(Util.shell.fileDir == null){
				tips = "文件  无标题   的文字已经改变。\n"+"想保存文件吗";
			}else{
				tips = "文件  " +Util.shell.fileDir + "  的文字已经改变。\n"+"想保存文件吗";
			}
			int n=JOptionPane.showConfirmDialog(null, tips,"记事本",JOptionPane.YES_NO_CANCEL_OPTION);
			if(n==0){  //是 返回0  否返回1 取消 返回2 
				SaveMethod savemethod = new SaveMethod();
				savemethod.Save(); //点是,则保存文件然后打开文件
			}else if(n==1){ //返回否
				System.exit(0);
			}else if(n==2){  //返回取消
				return ;
			}
			System.exit(0);
		}else{ 
			if(Util.shell.getText().getText() != ""){ //若目录为空,但是内容不为空,说明有内容但未保存,提示之
				tips = "文件  无标题   的文字已经改变!\n"+"想保存文件吗?";
				int n=JOptionPane.showConfirmDialog(null, tips,"记事本",JOptionPane.YES_NO_CANCEL_OPTION);
				if(n==0){  //是 返回0  否返回1 取消 返回2 
					SaveMethod savemethod = new SaveMethod();
					savemethod.Save(); //点是,则保存文件然后打开文件
				}else if(n==1){ //返回否
					System.exit(0);
				}else if(n==2){  //返回取消
					return ;
				}
				System.exit(0);
			}else{
				
			}
		}
		System.exit(0);
	}


写在最后,我们这里还封装了一个弹窗的utils类

public class UiUtils {

		/**
		 * 显示普通对话框的工具类
		 * @param shell
		 * @param title
		 * @param message
		 */
		public static void showMessageDialog(   Shell shell, String title, String message){
			MessageBox mb=new MessageBox(   shell, SWT.None);
			mb.setText(title);
			mb.setMessage(message);
			mb.open();
		}
	}
总结:技术源于思维,我觉得首先在编程之前还是要有一个整体的想法构思,不要一上来就直接写代码了,首先应该想想你要做什么东西,怎么个做法来实现!对于可重用的代码尽量封装起来,让代码整体看起来更简洁明了。坚持学习,坚持分享!

源码下载地址:点击打开链接http://download.csdn.net/detail/sdksdk0/9501286




posted @ 2016-04-24 21:43  朱培  阅读(412)  评论(0编辑  收藏  举报