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