会计考友 发表于 2012-8-4 12:37:27

用Java制作十六进制编辑器

http://www.examw.com/java/Files/2011-11/17/118109550.jpg
分析:对于同一段数据值,用两种方式来显示和编辑,则用MVC(模型-视图-控制器)模式来作为主结构是再合适不过的了。模型的作用是保存真实的数据值,同时提供若干提取和修改数据的方法;视图是数据在用户界面上的表示,控制器定义用户界面对用户输入的响应方式,即把用户的键盘动作和鼠标动作解释成模型中的数据操作方法。出于简化的考虑,本例中把视图和控制器合并到了一起。

模型的设计:编辑器必须能处理任意字节块,所以考虑模型内用字节数组来存储数据;要提供在指定偏移处增加、修改和删除字节块的操作;当模型内的数据被改变时要及时通知视图来刷新用户界面或其它感兴趣的对象;

视图和控制器的设计:

对于常规编辑的视图,只需把模型中的字节数组转化成String,使用一个文本区域组件JTextArea来显示即可。JTextArea本身也是一个遵循MVC模式的Swing组件,它的控制器即可被用来作为我们自己的控制器,监听它的文本增加、修改和删除事件,从而控制我们自己的数据模型;

对于16进制编辑的视图,同样可以用JTextArea来显示,只是在显示之前,要对模型中的数据进行若干加工,如每行显示16个字节,每行都要加表示偏移量的头,行尾要加这一行数据的字符串表示形式。它的控制器则不能简单的利用JTextArea的控制器了,为了保证显示格式不被打乱,需要监视它的所有光标移动事件、键盘击键事件等。同时,为了保持与UltraEdit的十六进制编辑器的功能一致性,对它的数据的增加、删除功能提供两个按钮,询问用户操作的字节数,如果增加n个字节,则在输入光标处插入n个十六进制值为20的字符(字符空格)。

十六进制编辑器的主要结构请参见下图,由于篇幅关系,图中只列出了十六进制编辑部分,常规编辑部分请读者自行设计:

http://www.examw.com/java/Files/2011-11/17/118115958.jpg
下面分别就几个主要的方法的功能和主要流程加以说明。

HexPane.displayValue方法,它主要完成数据的显示工作:

public void diplayValue() {
    canvas.setText("");

    byte[] data = model.getData();
    int dataLen = data.length;

    // 把字节数组按每16个字节为一块进行分块;
    int lines = dataLen / 16 + 1;
    int tails = dataLen % 16;
    int offset = 0;
    for(int i = 0; i < lines; i ++) {
      // 在canvas的新行上加上行标,如“000020:”,表示这是第3行;
      canvas.append(lineHead(i));
      canvas.append(" ");

      // 把数据块的字节值用Integer.toHexString()转化成长度为48的字符串,
      // 数据块不足16个字节的,在字符串后用空格补足;把字符串加入canvas的当前行;
      for(offset = 0; offset < 16; offset ++) {
            canvas.append((i < lines - 1 || offset < tails) ?
                                                byteHex(data) : "");
            canvas.append(" ");
      }
      canvas.append("| ");

      // 把数据块构造成字符串,添在canvas的行尾;
      canvas.append(bytesToStr(data, i * 16, (i == lines - 1) ? tails : 16));
      if(i < lines - 1) ta.append("n");
    };
}
这里的bytesToStr方法有两点特别需要注意的地方,一是不可见字符,如果不屏蔽这些字符,则我们的编辑器的显示格式会被搞得乱七八糟,一般可以把ASCII值0到0×1F和0×7F的33个字符全部替换成0×2E,即字符小数点。二是中文字符,因为每个中文字符是2个字节,如果数据块的起始字节一个中文字符的一半(可以用ASCII值大于0×7F来判断)的时候,将会显示一串乱字符,处理方法是不显示该字节。

为叙述方便,我们把canvas中显示每一行的行标的区域称为标号区,它宽度固定为8个字符(6个字符显示标号,一个冒号和一个空格);把canvas中显示十六进制数据的区域称为数据区,宽度固定为48个字符(每字节用十六进制显示为2字符宽,两两之间有一个空格,则总宽为16×3);把canvas中每行以字符串形式显示数据的区域称为字串值区,宽度不定(最短为8个字符――全中文状态,最长为16个字符――全英文状态)。

我们的canvas是一个Swing的文本组件,我们不但用它显示数据,还显示标号和字串值,而只有数据才是允许被编辑的,所以我们给canvas增加了CaretListener和KeyListener,当输入光标落在不允许编辑的区域时,我们要把光标自动移到最近的允许编辑的地方去。

public void caretUpdate(CaretEvent e) {
    // 这个方法在输入光标移动时被触发

    // 输入光标相对canvas第0行第0个字符的偏移量
    int pos = canvas.getCaretPosition();
    int line = 0;
    int startPos = 0;
    try {
      // 输入光标位于第几行
      line = canvas.getLineOfOffset(pos);

      // 当前行的第0个字符相对canvas第0行第0个字符的偏移量
      startPos = canvas.getLineStartOffset(line);
    }catch(BadLocationException exception) { }

    if(pos - startPos < 8)// 输入光标在标号区
      // 移动到数据区第0个字符
      canvas.setCaretPosition(startPos + 8);
    else if(pos - startPos > 54) // 输入光标在字串值区
      // 移动到数据区最后一个字节
      canvas.setCaretPosition(startPos + 54);
    else if((pos - startPos - 8) % 3 == 2) { // 在数据区的间隙空格上
      // 往前移一个字符
      canvas.setCaretPosition(pos - 1);
    }
}

会计考友 发表于 2012-8-4 12:37:28

用Java制作十六进制编辑器

public void keyPressed(KeyEvent e) { // 当键盘被按下时触发
    int key = e.getKeyCode();
    switch(key) {// 如果是方向键则移动输入光标
      case KeyEvent.VK_LEFT:
            setCaretPrev();
            break;
      case KeyEvent.VK_RIGHT:
            setCaretNext();
            break;
      case KeyEvent.VK_UP:
            setCaretPrevLine();
            break;
      case KeyEvent.VK_DOWN:
            setCaretNextLine();
            break;
      default:
            return;
    }
}

public void keyTyped(KeyEvent e) {
    // 在键盘的可见字符被输入时触发
    char ch = e.getKeyChar();
    if((ch >= ‘0′ && ch = ‘a’ && ch = ‘A’ && ch
页: [1]
查看完整版本: 用Java制作十六进制编辑器