mirror of
https://github.com/jaquadro/NBTExplorer.git
synced 2025-01-25 08:46:28 +00:00
3702 lines
98 KiB
C#
3702 lines
98 KiB
C#
|
using System;
|
|||
|
using System.Drawing;
|
|||
|
using System.Windows.Forms;
|
|||
|
using System.Runtime.InteropServices;
|
|||
|
using System.ComponentModel;
|
|||
|
using System.Security.Permissions;
|
|||
|
using System.Windows.Forms.VisualStyles;
|
|||
|
using System.Text;
|
|||
|
using System.Collections.Generic;
|
|||
|
|
|||
|
namespace Be.Windows.Forms
|
|||
|
{
|
|||
|
/// <summary>
|
|||
|
/// Represents a hex box control.
|
|||
|
/// </summary>
|
|||
|
[ToolboxBitmap(typeof(HexBox), "HexBox.bmp")]
|
|||
|
public class HexBox : Control
|
|||
|
{
|
|||
|
#region IKeyInterpreter interface
|
|||
|
/// <summary>
|
|||
|
/// Defines a user input handler such as for mouse and keyboard input
|
|||
|
/// </summary>
|
|||
|
interface IKeyInterpreter
|
|||
|
{
|
|||
|
/// <summary>
|
|||
|
/// Activates mouse events
|
|||
|
/// </summary>
|
|||
|
void Activate();
|
|||
|
/// <summary>
|
|||
|
/// Deactivate mouse events
|
|||
|
/// </summary>
|
|||
|
void Deactivate();
|
|||
|
/// <summary>
|
|||
|
/// Preprocesses WM_KEYUP window message.
|
|||
|
/// </summary>
|
|||
|
/// <param name="m">the Message object to process.</param>
|
|||
|
/// <returns>True, if the message was processed.</returns>
|
|||
|
bool PreProcessWmKeyUp(ref Message m);
|
|||
|
/// <summary>
|
|||
|
/// Preprocesses WM_CHAR window message.
|
|||
|
/// </summary>
|
|||
|
/// <param name="m">the Message object to process.</param>
|
|||
|
/// <returns>True, if the message was processed.</returns>
|
|||
|
bool PreProcessWmChar(ref Message m);
|
|||
|
/// <summary>
|
|||
|
/// Preprocesses WM_KEYDOWN window message.
|
|||
|
/// </summary>
|
|||
|
/// <param name="m">the Message object to process.</param>
|
|||
|
/// <returns>True, if the message was processed.</returns>
|
|||
|
bool PreProcessWmKeyDown(ref Message m);
|
|||
|
/// <summary>
|
|||
|
/// Gives some information about where to place the caret.
|
|||
|
/// </summary>
|
|||
|
/// <param name="byteIndex">the index of the byte</param>
|
|||
|
/// <returns>the position where the caret is to place.</returns>
|
|||
|
PointF GetCaretPointF(long byteIndex);
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region EmptyKeyInterpreter class
|
|||
|
/// <summary>
|
|||
|
/// Represents an empty input handler without any functionality.
|
|||
|
/// If is set ByteProvider to null, then this interpreter is used.
|
|||
|
/// </summary>
|
|||
|
class EmptyKeyInterpreter : IKeyInterpreter
|
|||
|
{
|
|||
|
HexBox _hexBox;
|
|||
|
|
|||
|
public EmptyKeyInterpreter(HexBox hexBox)
|
|||
|
{
|
|||
|
_hexBox = hexBox;
|
|||
|
}
|
|||
|
|
|||
|
#region IKeyInterpreter Members
|
|||
|
public void Activate(){}
|
|||
|
public void Deactivate(){}
|
|||
|
|
|||
|
public bool PreProcessWmKeyUp(ref Message m)
|
|||
|
{ return _hexBox.BasePreProcessMessage(ref m); }
|
|||
|
|
|||
|
public bool PreProcessWmChar(ref Message m)
|
|||
|
{ return _hexBox.BasePreProcessMessage(ref m); }
|
|||
|
|
|||
|
public bool PreProcessWmKeyDown(ref Message m)
|
|||
|
{ return _hexBox.BasePreProcessMessage(ref m); }
|
|||
|
|
|||
|
public PointF GetCaretPointF(long byteIndex)
|
|||
|
{ return new PointF (); }
|
|||
|
|
|||
|
#endregion
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region KeyInterpreter class
|
|||
|
/// <summary>
|
|||
|
/// Handles user input such as mouse and keyboard input during hex view edit
|
|||
|
/// </summary>
|
|||
|
class KeyInterpreter : IKeyInterpreter
|
|||
|
{
|
|||
|
/// <summary>
|
|||
|
/// Delegate for key-down processing.
|
|||
|
/// </summary>
|
|||
|
/// <param name="m">the message object contains key data information</param>
|
|||
|
/// <returns>True, if the message was processed</returns>
|
|||
|
delegate bool MessageDelegate(ref Message m);
|
|||
|
|
|||
|
#region Fields
|
|||
|
/// <summary>
|
|||
|
/// Contains the parent HexBox control
|
|||
|
/// </summary>
|
|||
|
protected HexBox _hexBox;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains True, if shift key is down
|
|||
|
/// </summary>
|
|||
|
protected bool _shiftDown;
|
|||
|
/// <summary>
|
|||
|
/// Contains True, if mouse is down
|
|||
|
/// </summary>
|
|||
|
bool _mouseDown;
|
|||
|
/// <summary>
|
|||
|
/// Contains the selection start position info
|
|||
|
/// </summary>
|
|||
|
BytePositionInfo _bpiStart;
|
|||
|
/// <summary>
|
|||
|
/// Contains the current mouse selection position info
|
|||
|
/// </summary>
|
|||
|
BytePositionInfo _bpi;
|
|||
|
/// <summary>
|
|||
|
/// Contains all message handlers of key interpreter key down message
|
|||
|
/// </summary>
|
|||
|
Dictionary<Keys, MessageDelegate> _messageHandlers;
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Ctors
|
|||
|
public KeyInterpreter(HexBox hexBox)
|
|||
|
{
|
|||
|
_hexBox = hexBox;
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Activate, Deactive methods
|
|||
|
public virtual void Activate()
|
|||
|
{
|
|||
|
_hexBox.MouseDown += new MouseEventHandler(BeginMouseSelection);
|
|||
|
_hexBox.MouseMove += new MouseEventHandler(UpdateMouseSelection);
|
|||
|
_hexBox.MouseUp += new MouseEventHandler(EndMouseSelection);
|
|||
|
}
|
|||
|
|
|||
|
public virtual void Deactivate()
|
|||
|
{
|
|||
|
_hexBox.MouseDown -= new MouseEventHandler(BeginMouseSelection);
|
|||
|
_hexBox.MouseMove -= new MouseEventHandler(UpdateMouseSelection);
|
|||
|
_hexBox.MouseUp -= new MouseEventHandler(EndMouseSelection);
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Mouse selection methods
|
|||
|
void BeginMouseSelection(object sender, MouseEventArgs e)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("BeginMouseSelection()", "KeyInterpreter");
|
|||
|
|
|||
|
if (e.Button != MouseButtons.Left)
|
|||
|
return;
|
|||
|
|
|||
|
_mouseDown = true;
|
|||
|
|
|||
|
if(!_shiftDown)
|
|||
|
{
|
|||
|
_bpiStart = new BytePositionInfo(_hexBox._bytePos, _hexBox._byteCharacterPos);
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
UpdateMouseSelection(this, e);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void UpdateMouseSelection(object sender, MouseEventArgs e)
|
|||
|
{
|
|||
|
if(!_mouseDown)
|
|||
|
return;
|
|||
|
|
|||
|
_bpi = GetBytePositionInfo(new Point(e.X, e.Y));
|
|||
|
long selEnd = _bpi.Index;
|
|||
|
long realselStart;
|
|||
|
long realselLength;
|
|||
|
|
|||
|
if(selEnd < _bpiStart.Index)
|
|||
|
{
|
|||
|
realselStart = selEnd;
|
|||
|
realselLength = _bpiStart.Index - selEnd;
|
|||
|
}
|
|||
|
else if(selEnd > _bpiStart.Index)
|
|||
|
{
|
|||
|
realselStart = _bpiStart.Index;
|
|||
|
realselLength = selEnd - realselStart;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
realselStart = _hexBox._bytePos;
|
|||
|
realselLength = 0;
|
|||
|
}
|
|||
|
|
|||
|
if(realselStart != _hexBox._bytePos || realselLength != _hexBox._selectionLength)
|
|||
|
{
|
|||
|
_hexBox.InternalSelect(realselStart, realselLength);
|
|||
|
_hexBox.ScrollByteIntoView(_bpi.Index);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void EndMouseSelection(object sender, MouseEventArgs e)
|
|||
|
{
|
|||
|
_mouseDown = false;
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region PrePrcessWmKeyDown methods
|
|||
|
public virtual bool PreProcessWmKeyDown(ref Message m)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("PreProcessWmKeyDown(ref Message m)", "KeyInterpreter");
|
|||
|
|
|||
|
Keys vc = (Keys)m.WParam.ToInt32();
|
|||
|
|
|||
|
Keys keyData = vc | Control.ModifierKeys;
|
|||
|
|
|||
|
// detect whether key down event should be raised
|
|||
|
var hasMessageHandler = this.MessageHandlers.ContainsKey(keyData);
|
|||
|
if (hasMessageHandler && RaiseKeyDown(keyData))
|
|||
|
return true;
|
|||
|
|
|||
|
MessageDelegate messageHandler = hasMessageHandler
|
|||
|
? this.MessageHandlers[keyData]
|
|||
|
: messageHandler = new MessageDelegate(PreProcessWmKeyDown_Default);
|
|||
|
|
|||
|
return messageHandler(ref m);
|
|||
|
}
|
|||
|
|
|||
|
protected bool PreProcessWmKeyDown_Default(ref Message m)
|
|||
|
{
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
return _hexBox.BasePreProcessMessage(ref m);
|
|||
|
}
|
|||
|
|
|||
|
protected bool RaiseKeyDown(Keys keyData)
|
|||
|
{
|
|||
|
KeyEventArgs e = new KeyEventArgs(keyData);
|
|||
|
_hexBox.OnKeyDown(e);
|
|||
|
return e.Handled;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_Left(ref Message m)
|
|||
|
{
|
|||
|
return PerformPosMoveLeft();
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_Up(ref Message m)
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if( !(pos == 0 && cp == 0) )
|
|||
|
{
|
|||
|
pos = Math.Max(-1, pos-_hexBox._iHexMaxHBytes);
|
|||
|
if(pos == -1)
|
|||
|
return true;
|
|||
|
|
|||
|
_hexBox.SetPosition(pos);
|
|||
|
|
|||
|
if(pos < _hexBox._startByte)
|
|||
|
{
|
|||
|
_hexBox.PerformScrollLineUp();
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.Invalidate();
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_Right(ref Message m)
|
|||
|
{
|
|||
|
return PerformPosMoveRight();
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_Down(ref Message m)
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if(pos == _hexBox._byteProvider.Length && cp == 0)
|
|||
|
return true;
|
|||
|
|
|||
|
pos = Math.Min(_hexBox._byteProvider.Length, pos+_hexBox._iHexMaxHBytes);
|
|||
|
|
|||
|
if(pos == _hexBox._byteProvider.Length)
|
|||
|
cp = 0;
|
|||
|
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
|
|||
|
if(pos > _hexBox._endByte-1)
|
|||
|
{
|
|||
|
_hexBox.PerformScrollLineDown();
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
_hexBox.Invalidate();
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_PageUp(ref Message m)
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if(pos == 0 && cp == 0)
|
|||
|
return true;
|
|||
|
|
|||
|
pos = Math.Max(0, pos-_hexBox._iHexMaxBytes);
|
|||
|
if(pos == 0)
|
|||
|
return true;
|
|||
|
|
|||
|
_hexBox.SetPosition(pos);
|
|||
|
|
|||
|
if(pos < _hexBox._startByte)
|
|||
|
{
|
|||
|
_hexBox.PerformScrollPageUp();
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.Invalidate();
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_PageDown(ref Message m)
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if(pos == _hexBox._byteProvider.Length && cp == 0)
|
|||
|
return true;
|
|||
|
|
|||
|
pos = Math.Min(_hexBox._byteProvider.Length, pos+_hexBox._iHexMaxBytes);
|
|||
|
|
|||
|
if(pos == _hexBox._byteProvider.Length)
|
|||
|
cp = 0;
|
|||
|
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
|
|||
|
if(pos > _hexBox._endByte-1)
|
|||
|
{
|
|||
|
_hexBox.PerformScrollPageDown();
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.Invalidate();
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_ShiftLeft(ref Message m)
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
long sel = _hexBox._selectionLength;
|
|||
|
|
|||
|
if(pos + sel < 1)
|
|||
|
return true;
|
|||
|
|
|||
|
if(pos+sel <= _bpiStart.Index)
|
|||
|
{
|
|||
|
if(pos == 0)
|
|||
|
return true;
|
|||
|
|
|||
|
pos--;
|
|||
|
sel++;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
sel = Math.Max(0, sel-1);
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
_hexBox.InternalSelect(pos, sel);
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_ShiftUp(ref Message m)
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
long sel = _hexBox._selectionLength;
|
|||
|
|
|||
|
if(pos-_hexBox._iHexMaxHBytes < 0 && pos <= _bpiStart.Index)
|
|||
|
return true;
|
|||
|
|
|||
|
if(_bpiStart.Index >= pos+sel)
|
|||
|
{
|
|||
|
pos = pos - _hexBox._iHexMaxHBytes;
|
|||
|
sel += _hexBox._iHexMaxHBytes;
|
|||
|
_hexBox.InternalSelect(pos, sel);
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
sel -= _hexBox._iHexMaxHBytes;
|
|||
|
if(sel < 0)
|
|||
|
{
|
|||
|
pos = _bpiStart.Index + sel;
|
|||
|
sel = -sel;
|
|||
|
_hexBox.InternalSelect(pos, sel);
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
sel -= _hexBox._iHexMaxHBytes;
|
|||
|
_hexBox.InternalSelect(pos, sel);
|
|||
|
_hexBox.ScrollByteIntoView(pos+sel);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_ShiftRight(ref Message m)
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
long sel = _hexBox._selectionLength;
|
|||
|
|
|||
|
if(pos+sel >= _hexBox._byteProvider.Length)
|
|||
|
return true;
|
|||
|
|
|||
|
if(_bpiStart.Index <= pos)
|
|||
|
{
|
|||
|
sel++;
|
|||
|
_hexBox.InternalSelect(pos, sel);
|
|||
|
_hexBox.ScrollByteIntoView(pos+sel);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
pos++;
|
|||
|
sel = Math.Max(0, sel-1);
|
|||
|
_hexBox.InternalSelect(pos, sel);
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
}
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_ShiftDown(ref Message m)
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
long sel = _hexBox._selectionLength;
|
|||
|
|
|||
|
long max = _hexBox._byteProvider.Length;
|
|||
|
|
|||
|
if(pos+sel+_hexBox._iHexMaxHBytes > max)
|
|||
|
return true;
|
|||
|
|
|||
|
if(_bpiStart.Index <= pos)
|
|||
|
{
|
|||
|
sel += _hexBox._iHexMaxHBytes;
|
|||
|
_hexBox.InternalSelect(pos, sel);
|
|||
|
_hexBox.ScrollByteIntoView(pos+sel);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
sel -= _hexBox._iHexMaxHBytes;
|
|||
|
if (sel < 0)
|
|||
|
{
|
|||
|
pos = _bpiStart.Index;
|
|||
|
sel = -sel;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
pos += _hexBox._iHexMaxHBytes;
|
|||
|
//sel -= _hexBox._iHexMaxHBytes;
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.InternalSelect(pos, sel);
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
}
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_Tab(ref Message m)
|
|||
|
{
|
|||
|
if(_hexBox._stringViewVisible && _hexBox._keyInterpreter.GetType() == typeof(KeyInterpreter))
|
|||
|
{
|
|||
|
_hexBox.ActivateStringKeyInterpreter();
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.Invalidate();
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
if(_hexBox.Parent == null) return true;
|
|||
|
_hexBox.Parent.SelectNextControl(_hexBox, true, true, true, true);
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_ShiftTab(ref Message m)
|
|||
|
{
|
|||
|
if(_hexBox._keyInterpreter is StringKeyInterpreter)
|
|||
|
{
|
|||
|
_shiftDown = false;
|
|||
|
_hexBox.ActivateKeyInterpreter();
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.Invalidate();
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
if(_hexBox.Parent == null) return true;
|
|||
|
_hexBox.Parent.SelectNextControl(_hexBox, false, true, true, true);
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_Back(ref Message m)
|
|||
|
{
|
|||
|
if(!_hexBox._byteProvider.SupportsDeleteBytes())
|
|||
|
return true;
|
|||
|
|
|||
|
if (_hexBox.ReadOnly)
|
|||
|
return true;
|
|||
|
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
long sel = _hexBox._selectionLength;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
long startDelete = (cp == 0 && sel == 0) ? pos-1 : pos;
|
|||
|
if(startDelete < 0 && sel < 1)
|
|||
|
return true;
|
|||
|
|
|||
|
long bytesToDelete = (sel > 0) ? sel : 1;
|
|||
|
_hexBox._byteProvider.DeleteBytes(Math.Max(0, startDelete), bytesToDelete);
|
|||
|
_hexBox.UpdateScrollSize();
|
|||
|
|
|||
|
if(sel == 0)
|
|||
|
PerformPosMoveLeftByte();
|
|||
|
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
_hexBox.Invalidate();
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_Delete(ref Message m)
|
|||
|
{
|
|||
|
if(!_hexBox._byteProvider.SupportsDeleteBytes())
|
|||
|
return true;
|
|||
|
|
|||
|
if (_hexBox.ReadOnly)
|
|||
|
return true;
|
|||
|
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
long sel = _hexBox._selectionLength;
|
|||
|
|
|||
|
if(pos >= _hexBox._byteProvider.Length)
|
|||
|
return true;
|
|||
|
|
|||
|
long bytesToDelete = (sel > 0) ? sel : 1;
|
|||
|
_hexBox._byteProvider.DeleteBytes(pos, bytesToDelete);
|
|||
|
|
|||
|
_hexBox.UpdateScrollSize();
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
_hexBox.Invalidate();
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_Home(ref Message m)
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if(pos < 1)
|
|||
|
return true;
|
|||
|
|
|||
|
pos = 0;
|
|||
|
cp = 0;
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_End(ref Message m)
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if(pos >= _hexBox._byteProvider.Length-1)
|
|||
|
return true;
|
|||
|
|
|||
|
pos = _hexBox._byteProvider.Length;
|
|||
|
cp = 0;
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_ShiftShiftKey(ref Message m)
|
|||
|
{
|
|||
|
if(_mouseDown)
|
|||
|
return true;
|
|||
|
if(_shiftDown)
|
|||
|
return true;
|
|||
|
|
|||
|
_shiftDown = true;
|
|||
|
|
|||
|
if(_hexBox._selectionLength > 0)
|
|||
|
return true;
|
|||
|
|
|||
|
_bpiStart = new BytePositionInfo(_hexBox._bytePos, _hexBox._byteCharacterPos);
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_ControlC(ref Message m)
|
|||
|
{
|
|||
|
_hexBox.Copy();
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_ControlX(ref Message m)
|
|||
|
{
|
|||
|
_hexBox.Cut();
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyDown_ControlV(ref Message m)
|
|||
|
{
|
|||
|
_hexBox.Paste();
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
#endregion
|
|||
|
|
|||
|
#region PreProcessWmChar methods
|
|||
|
public virtual bool PreProcessWmChar(ref Message m)
|
|||
|
{
|
|||
|
if(Control.ModifierKeys == Keys.Control)
|
|||
|
{
|
|||
|
return _hexBox.BasePreProcessMessage(ref m);
|
|||
|
}
|
|||
|
|
|||
|
bool sw = _hexBox._byteProvider.SupportsWriteByte();
|
|||
|
bool si = _hexBox._byteProvider.SupportsInsertBytes();
|
|||
|
bool sd = _hexBox._byteProvider.SupportsDeleteBytes();
|
|||
|
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
long sel = _hexBox._selectionLength;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if(
|
|||
|
(!sw && pos != _hexBox._byteProvider.Length) ||
|
|||
|
(!si && pos == _hexBox._byteProvider.Length))
|
|||
|
{
|
|||
|
return _hexBox.BasePreProcessMessage(ref m);
|
|||
|
}
|
|||
|
|
|||
|
char c = (char)m.WParam.ToInt32();
|
|||
|
|
|||
|
if(Uri.IsHexDigit(c))
|
|||
|
{
|
|||
|
if(RaiseKeyPress(c))
|
|||
|
return true;
|
|||
|
|
|||
|
if(_hexBox.ReadOnly)
|
|||
|
return true;
|
|||
|
|
|||
|
bool isInsertMode = (pos == _hexBox._byteProvider.Length);
|
|||
|
|
|||
|
// do insert when insertActive = true
|
|||
|
if(!isInsertMode && si && _hexBox.InsertActive && cp == 0)
|
|||
|
isInsertMode = true;
|
|||
|
|
|||
|
if(sd && si && sel > 0)
|
|||
|
{
|
|||
|
_hexBox._byteProvider.DeleteBytes(pos, sel);
|
|||
|
isInsertMode = true;
|
|||
|
cp = 0;
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
|
|||
|
byte currentByte;
|
|||
|
if(isInsertMode)
|
|||
|
currentByte = 0;
|
|||
|
else
|
|||
|
currentByte = _hexBox._byteProvider.ReadByte(pos);
|
|||
|
|
|||
|
string sCb = currentByte.ToString("X", System.Threading.Thread.CurrentThread.CurrentCulture);
|
|||
|
if(sCb.Length == 1)
|
|||
|
sCb = "0" + sCb;
|
|||
|
|
|||
|
string sNewCb = c.ToString();
|
|||
|
if(cp == 0)
|
|||
|
sNewCb += sCb.Substring(1, 1);
|
|||
|
else
|
|||
|
sNewCb = sCb.Substring(0, 1) + sNewCb;
|
|||
|
byte newcb = byte.Parse(sNewCb, System.Globalization.NumberStyles.AllowHexSpecifier, System.Threading.Thread.CurrentThread.CurrentCulture);
|
|||
|
|
|||
|
if(isInsertMode)
|
|||
|
_hexBox._byteProvider.InsertBytes(pos, new byte[]{newcb});
|
|||
|
else
|
|||
|
_hexBox._byteProvider.WriteByte(pos, newcb);
|
|||
|
|
|||
|
PerformPosMoveRight();
|
|||
|
|
|||
|
_hexBox.Invalidate();
|
|||
|
return true;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
return _hexBox.BasePreProcessMessage(ref m);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
protected bool RaiseKeyPress(char keyChar)
|
|||
|
{
|
|||
|
KeyPressEventArgs e = new KeyPressEventArgs(keyChar);
|
|||
|
_hexBox.OnKeyPress(e);
|
|||
|
return e.Handled;
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region PreProcessWmKeyUp methods
|
|||
|
public virtual bool PreProcessWmKeyUp(ref Message m)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("PreProcessWmKeyUp(ref Message m)", "KeyInterpreter");
|
|||
|
|
|||
|
Keys vc = (Keys)m.WParam.ToInt32();
|
|||
|
|
|||
|
Keys keyData = vc | Control.ModifierKeys;
|
|||
|
|
|||
|
switch(keyData)
|
|||
|
{
|
|||
|
case Keys.ShiftKey:
|
|||
|
case Keys.Insert:
|
|||
|
if(RaiseKeyUp(keyData))
|
|||
|
return true;
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
switch(keyData)
|
|||
|
{
|
|||
|
case Keys.ShiftKey:
|
|||
|
_shiftDown = false;
|
|||
|
return true;
|
|||
|
case Keys.Insert:
|
|||
|
return PreProcessWmKeyUp_Insert(ref m);
|
|||
|
default:
|
|||
|
return _hexBox.BasePreProcessMessage(ref m);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PreProcessWmKeyUp_Insert(ref Message m)
|
|||
|
{
|
|||
|
_hexBox.InsertActive = !_hexBox.InsertActive;
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected bool RaiseKeyUp(Keys keyData)
|
|||
|
{
|
|||
|
KeyEventArgs e = new KeyEventArgs(keyData);
|
|||
|
_hexBox.OnKeyUp(e);
|
|||
|
return e.Handled;
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Misc
|
|||
|
Dictionary<Keys, MessageDelegate> MessageHandlers
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
if (_messageHandlers == null)
|
|||
|
{
|
|||
|
_messageHandlers = new Dictionary<Keys, MessageDelegate>();
|
|||
|
_messageHandlers.Add(Keys.Left, new MessageDelegate(PreProcessWmKeyDown_Left)); // move left
|
|||
|
_messageHandlers.Add(Keys.Up, new MessageDelegate(PreProcessWmKeyDown_Up)); // move up
|
|||
|
_messageHandlers.Add(Keys.Right, new MessageDelegate(PreProcessWmKeyDown_Right)); // move right
|
|||
|
_messageHandlers.Add(Keys.Down, new MessageDelegate(PreProcessWmKeyDown_Down)); // move down
|
|||
|
_messageHandlers.Add(Keys.PageUp, new MessageDelegate(PreProcessWmKeyDown_PageUp)); // move pageup
|
|||
|
_messageHandlers.Add(Keys.PageDown, new MessageDelegate(PreProcessWmKeyDown_PageDown)); // move page down
|
|||
|
_messageHandlers.Add(Keys.Left | Keys.Shift, new MessageDelegate(PreProcessWmKeyDown_ShiftLeft)); // move left with selection
|
|||
|
_messageHandlers.Add(Keys.Up | Keys.Shift, new MessageDelegate(PreProcessWmKeyDown_ShiftUp)); // move up with selection
|
|||
|
_messageHandlers.Add(Keys.Right | Keys.Shift, new MessageDelegate(PreProcessWmKeyDown_ShiftRight)); // move right with selection
|
|||
|
_messageHandlers.Add(Keys.Down | Keys.Shift, new MessageDelegate(PreProcessWmKeyDown_ShiftDown)); // move down with selection
|
|||
|
_messageHandlers.Add(Keys.Tab, new MessageDelegate(PreProcessWmKeyDown_Tab)); // switch to string view
|
|||
|
_messageHandlers.Add(Keys.Back, new MessageDelegate(PreProcessWmKeyDown_Back)); // back
|
|||
|
_messageHandlers.Add(Keys.Delete, new MessageDelegate(PreProcessWmKeyDown_Delete)); // delete
|
|||
|
_messageHandlers.Add(Keys.Home, new MessageDelegate(PreProcessWmKeyDown_Home)); // move to home
|
|||
|
_messageHandlers.Add(Keys.End, new MessageDelegate(PreProcessWmKeyDown_End)); // move to end
|
|||
|
_messageHandlers.Add(Keys.ShiftKey | Keys.Shift, new MessageDelegate(PreProcessWmKeyDown_ShiftShiftKey)); // begin selection process
|
|||
|
_messageHandlers.Add(Keys.C | Keys.Control, new MessageDelegate(PreProcessWmKeyDown_ControlC)); // copy
|
|||
|
_messageHandlers.Add(Keys.X | Keys.Control, new MessageDelegate(PreProcessWmKeyDown_ControlX)); // cut
|
|||
|
_messageHandlers.Add(Keys.V | Keys.Control, new MessageDelegate(PreProcessWmKeyDown_ControlV)); // paste
|
|||
|
}
|
|||
|
return _messageHandlers;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PerformPosMoveLeft()
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
long sel = _hexBox._selectionLength;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if(sel != 0)
|
|||
|
{
|
|||
|
cp = 0;
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
if(pos == 0 && cp == 0)
|
|||
|
return true;
|
|||
|
|
|||
|
if(cp > 0)
|
|||
|
{
|
|||
|
cp--;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
pos = Math.Max(0, pos-1);
|
|||
|
cp++;
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
|
|||
|
if(pos < _hexBox._startByte)
|
|||
|
{
|
|||
|
_hexBox.PerformScrollLineUp();
|
|||
|
}
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.Invalidate();
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
return true;
|
|||
|
}
|
|||
|
protected virtual bool PerformPosMoveRight()
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
long sel = _hexBox._selectionLength;
|
|||
|
|
|||
|
if(sel != 0)
|
|||
|
{
|
|||
|
pos += sel;
|
|||
|
cp = 0;
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
if( !(pos == _hexBox._byteProvider.Length && cp == 0) )
|
|||
|
{
|
|||
|
|
|||
|
if(cp > 0)
|
|||
|
{
|
|||
|
pos = Math.Min(_hexBox._byteProvider.Length, pos+1);
|
|||
|
cp = 0;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
cp++;
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
|
|||
|
if(pos > _hexBox._endByte-1)
|
|||
|
{
|
|||
|
_hexBox.PerformScrollLineDown();
|
|||
|
}
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.Invalidate();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
return true;
|
|||
|
}
|
|||
|
protected virtual bool PerformPosMoveLeftByte()
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if(pos == 0)
|
|||
|
return true;
|
|||
|
|
|||
|
pos = Math.Max(0, pos-1);
|
|||
|
cp = 0;
|
|||
|
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
|
|||
|
if(pos < _hexBox._startByte)
|
|||
|
{
|
|||
|
_hexBox.PerformScrollLineUp();
|
|||
|
}
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
_hexBox.Invalidate();
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
protected virtual bool PerformPosMoveRightByte()
|
|||
|
{
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if(pos == _hexBox._byteProvider.Length)
|
|||
|
return true;
|
|||
|
|
|||
|
pos = Math.Min(_hexBox._byteProvider.Length, pos+1);
|
|||
|
cp = 0;
|
|||
|
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
|
|||
|
if(pos > _hexBox._endByte-1)
|
|||
|
{
|
|||
|
_hexBox.PerformScrollLineDown();
|
|||
|
}
|
|||
|
_hexBox.UpdateCaret();
|
|||
|
_hexBox.ScrollByteIntoView();
|
|||
|
_hexBox.Invalidate();
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
public virtual PointF GetCaretPointF(long byteIndex)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("GetCaretPointF()", "KeyInterpreter");
|
|||
|
|
|||
|
return _hexBox.GetBytePointF(byteIndex);
|
|||
|
}
|
|||
|
|
|||
|
protected virtual BytePositionInfo GetBytePositionInfo(Point p)
|
|||
|
{
|
|||
|
return _hexBox.GetHexBytePositionInfo(p);
|
|||
|
}
|
|||
|
#endregion
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region StringKeyInterpreter class
|
|||
|
/// <summary>
|
|||
|
/// Handles user input such as mouse and keyboard input during string view edit
|
|||
|
/// </summary>
|
|||
|
class StringKeyInterpreter : KeyInterpreter
|
|||
|
{
|
|||
|
#region Ctors
|
|||
|
public StringKeyInterpreter(HexBox hexBox) : base(hexBox)
|
|||
|
{
|
|||
|
_hexBox._byteCharacterPos = 0;
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region PreProcessWmKeyDown methods
|
|||
|
public override bool PreProcessWmKeyDown(ref Message m)
|
|||
|
{
|
|||
|
Keys vc = (Keys)m.WParam.ToInt32();
|
|||
|
|
|||
|
Keys keyData = vc | Control.ModifierKeys;
|
|||
|
|
|||
|
switch(keyData)
|
|||
|
{
|
|||
|
case Keys.Tab | Keys.Shift:
|
|||
|
case Keys.Tab:
|
|||
|
if(RaiseKeyDown(keyData))
|
|||
|
return true;
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
switch(keyData)
|
|||
|
{
|
|||
|
case Keys.Tab | Keys.Shift:
|
|||
|
return PreProcessWmKeyDown_ShiftTab(ref m);
|
|||
|
case Keys.Tab:
|
|||
|
return PreProcessWmKeyDown_Tab(ref m);
|
|||
|
default:
|
|||
|
return base.PreProcessWmKeyDown(ref m);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
protected override bool PreProcessWmKeyDown_Left(ref Message m)
|
|||
|
{
|
|||
|
return PerformPosMoveLeftByte();
|
|||
|
}
|
|||
|
|
|||
|
protected override bool PreProcessWmKeyDown_Right(ref Message m)
|
|||
|
{
|
|||
|
return PerformPosMoveRightByte();
|
|||
|
}
|
|||
|
|
|||
|
#endregion
|
|||
|
|
|||
|
#region PreProcessWmChar methods
|
|||
|
public override bool PreProcessWmChar(ref Message m)
|
|||
|
{
|
|||
|
if(Control.ModifierKeys == Keys.Control)
|
|||
|
{
|
|||
|
return _hexBox.BasePreProcessMessage(ref m);
|
|||
|
}
|
|||
|
|
|||
|
bool sw = _hexBox._byteProvider.SupportsWriteByte();
|
|||
|
bool si = _hexBox._byteProvider.SupportsInsertBytes();
|
|||
|
bool sd = _hexBox._byteProvider.SupportsDeleteBytes();
|
|||
|
|
|||
|
long pos = _hexBox._bytePos;
|
|||
|
long sel = _hexBox._selectionLength;
|
|||
|
int cp = _hexBox._byteCharacterPos;
|
|||
|
|
|||
|
if(
|
|||
|
(!sw && pos != _hexBox._byteProvider.Length) ||
|
|||
|
(!si && pos == _hexBox._byteProvider.Length))
|
|||
|
{
|
|||
|
return _hexBox.BasePreProcessMessage(ref m);
|
|||
|
}
|
|||
|
|
|||
|
char c = (char)m.WParam.ToInt32();
|
|||
|
|
|||
|
if(RaiseKeyPress(c))
|
|||
|
return true;
|
|||
|
|
|||
|
if(_hexBox.ReadOnly)
|
|||
|
return true;
|
|||
|
|
|||
|
bool isInsertMode = (pos == _hexBox._byteProvider.Length);
|
|||
|
|
|||
|
// do insert when insertActive = true
|
|||
|
if(!isInsertMode && si && _hexBox.InsertActive)
|
|||
|
isInsertMode = true;
|
|||
|
|
|||
|
if(sd && si && sel > 0)
|
|||
|
{
|
|||
|
_hexBox._byteProvider.DeleteBytes(pos, sel);
|
|||
|
isInsertMode = true;
|
|||
|
cp = 0;
|
|||
|
_hexBox.SetPosition(pos, cp);
|
|||
|
}
|
|||
|
|
|||
|
_hexBox.ReleaseSelection();
|
|||
|
|
|||
|
byte b = _hexBox.ByteCharConverter.ToByte(c);
|
|||
|
if(isInsertMode)
|
|||
|
_hexBox._byteProvider.InsertBytes(pos, new byte[]{b});
|
|||
|
else
|
|||
|
_hexBox._byteProvider.WriteByte(pos, b);
|
|||
|
|
|||
|
PerformPosMoveRightByte();
|
|||
|
_hexBox.Invalidate();
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Misc
|
|||
|
public override PointF GetCaretPointF(long byteIndex)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("GetCaretPointF()", "StringKeyInterpreter");
|
|||
|
|
|||
|
Point gp = _hexBox.GetGridBytePoint(byteIndex);
|
|||
|
return _hexBox.GetByteStringPointF(gp);
|
|||
|
}
|
|||
|
|
|||
|
protected override BytePositionInfo GetBytePositionInfo(Point p)
|
|||
|
{
|
|||
|
return _hexBox.GetStringBytePositionInfo(p);
|
|||
|
}
|
|||
|
#endregion
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Fields
|
|||
|
/// <summary>
|
|||
|
/// Contains the hole content bounds of all text
|
|||
|
/// </summary>
|
|||
|
Rectangle _recContent;
|
|||
|
/// <summary>
|
|||
|
/// Contains the line info bounds
|
|||
|
/// </summary>
|
|||
|
Rectangle _recLineInfo;
|
|||
|
/// <summary>
|
|||
|
/// Contains the hex data bounds
|
|||
|
/// </summary>
|
|||
|
Rectangle _recHex;
|
|||
|
/// <summary>
|
|||
|
/// Contains the string view bounds
|
|||
|
/// </summary>
|
|||
|
Rectangle _recStringView;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains string format information for text drawing
|
|||
|
/// </summary>
|
|||
|
StringFormat _stringFormat;
|
|||
|
/// <summary>
|
|||
|
/// Contains the width and height of a single char
|
|||
|
/// </summary>
|
|||
|
SizeF _charSize;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains the maximum of visible horizontal bytes
|
|||
|
/// </summary>
|
|||
|
int _iHexMaxHBytes;
|
|||
|
/// <summary>
|
|||
|
/// Contains the maximum of visible vertical bytes
|
|||
|
/// </summary>
|
|||
|
int _iHexMaxVBytes;
|
|||
|
/// <summary>
|
|||
|
/// Contains the maximum of visible bytes.
|
|||
|
/// </summary>
|
|||
|
int _iHexMaxBytes;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains the scroll bars minimum value
|
|||
|
/// </summary>
|
|||
|
long _scrollVmin;
|
|||
|
/// <summary>
|
|||
|
/// Contains the scroll bars maximum value
|
|||
|
/// </summary>
|
|||
|
long _scrollVmax;
|
|||
|
/// <summary>
|
|||
|
/// Contains the scroll bars current position
|
|||
|
/// </summary>
|
|||
|
long _scrollVpos;
|
|||
|
/// <summary>
|
|||
|
/// Contains a vertical scroll
|
|||
|
/// </summary>
|
|||
|
VScrollBar _vScrollBar;
|
|||
|
/// <summary>
|
|||
|
/// Contains a timer for thumbtrack scrolling
|
|||
|
/// </summary>
|
|||
|
Timer _thumbTrackTimer = new Timer();
|
|||
|
/// <summary>
|
|||
|
/// Contains the thumbtrack scrolling position
|
|||
|
/// </summary>
|
|||
|
long _thumbTrackPosition;
|
|||
|
/// <summary>
|
|||
|
/// Contains the thumptrack delay for scrolling in milliseconds.
|
|||
|
/// </summary>
|
|||
|
const int THUMPTRACKDELAY = 50;
|
|||
|
/// <summary>
|
|||
|
/// Contains the Enviroment.TickCount of the last refresh
|
|||
|
/// </summary>
|
|||
|
int _lastThumbtrack;
|
|||
|
/// <summary>
|
|||
|
/// Contains the border<65>s left shift
|
|||
|
/// </summary>
|
|||
|
int _recBorderLeft = SystemInformation.Border3DSize.Width;
|
|||
|
/// <summary>
|
|||
|
/// Contains the border<65>s right shift
|
|||
|
/// </summary>
|
|||
|
int _recBorderRight = SystemInformation.Border3DSize.Width;
|
|||
|
/// <summary>
|
|||
|
/// Contains the border<65>s top shift
|
|||
|
/// </summary>
|
|||
|
int _recBorderTop = SystemInformation.Border3DSize.Height;
|
|||
|
/// <summary>
|
|||
|
/// Contains the border bottom shift
|
|||
|
/// </summary>
|
|||
|
int _recBorderBottom = SystemInformation.Border3DSize.Height;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains the index of the first visible byte
|
|||
|
/// </summary>
|
|||
|
long _startByte;
|
|||
|
/// <summary>
|
|||
|
/// Contains the index of the last visible byte
|
|||
|
/// </summary>
|
|||
|
long _endByte;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains the current byte position
|
|||
|
/// </summary>
|
|||
|
long _bytePos = -1;
|
|||
|
/// <summary>
|
|||
|
/// Contains the current char position in one byte
|
|||
|
/// </summary>
|
|||
|
/// <example>
|
|||
|
/// "1A"
|
|||
|
/// "1" = char position of 0
|
|||
|
/// "A" = char position of 1
|
|||
|
/// </example>
|
|||
|
int _byteCharacterPos;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains string format information for hex values
|
|||
|
/// </summary>
|
|||
|
string _hexStringFormat = "X";
|
|||
|
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains the current key interpreter
|
|||
|
/// </summary>
|
|||
|
IKeyInterpreter _keyInterpreter;
|
|||
|
/// <summary>
|
|||
|
/// Contains an empty key interpreter without functionality
|
|||
|
/// </summary>
|
|||
|
EmptyKeyInterpreter _eki;
|
|||
|
/// <summary>
|
|||
|
/// Contains the default key interpreter
|
|||
|
/// </summary>
|
|||
|
KeyInterpreter _ki;
|
|||
|
/// <summary>
|
|||
|
/// Contains the string key interpreter
|
|||
|
/// </summary>
|
|||
|
StringKeyInterpreter _ski;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains True if caret is visible
|
|||
|
/// </summary>
|
|||
|
bool _caretVisible;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains true, if the find (Find method) should be aborted.
|
|||
|
/// </summary>
|
|||
|
bool _abortFind;
|
|||
|
/// <summary>
|
|||
|
/// Contains a value of the current finding position.
|
|||
|
/// </summary>
|
|||
|
long _findingPos;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Contains a state value about Insert or Write mode. When this value is true and the ByteProvider SupportsInsert is true bytes are inserted instead of overridden.
|
|||
|
/// </summary>
|
|||
|
bool _insertActive;
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Events
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of InsertActive property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of InsertActive property has changed.")]
|
|||
|
public event EventHandler InsertActiveChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of ReadOnly property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of ReadOnly property has changed.")]
|
|||
|
public event EventHandler ReadOnlyChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of ByteProvider property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of ByteProvider property has changed.")]
|
|||
|
public event EventHandler ByteProviderChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of SelectionStart property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of SelectionStart property has changed.")]
|
|||
|
public event EventHandler SelectionStartChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of SelectionLength property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of SelectionLength property has changed.")]
|
|||
|
public event EventHandler SelectionLengthChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of LineInfoVisible property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of LineInfoVisible property has changed.")]
|
|||
|
public event EventHandler LineInfoVisibleChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of StringViewVisible property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of StringViewVisible property has changed.")]
|
|||
|
public event EventHandler StringViewVisibleChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of BorderStyle property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of BorderStyle property has changed.")]
|
|||
|
public event EventHandler BorderStyleChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of BytesPerLine property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of BytesPerLine property has changed.")]
|
|||
|
public event EventHandler BytesPerLineChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of UseFixedBytesPerLine property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of UseFixedBytesPerLine property has changed.")]
|
|||
|
public event EventHandler UseFixedBytesPerLineChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of VScrollBarVisible property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of VScrollBarVisible property has changed.")]
|
|||
|
public event EventHandler VScrollBarVisibleChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of HexCasing property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of HexCasing property has changed.")]
|
|||
|
public event EventHandler HexCasingChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of HorizontalByteCount property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of HorizontalByteCount property has changed.")]
|
|||
|
public event EventHandler HorizontalByteCountChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of VerticalByteCount property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of VerticalByteCount property has changed.")]
|
|||
|
public event EventHandler VerticalByteCountChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of CurrentLine property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of CurrentLine property has changed.")]
|
|||
|
public event EventHandler CurrentLineChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when the value of CurrentPositionInLine property has changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when the value of CurrentPositionInLine property has changed.")]
|
|||
|
public event EventHandler CurrentPositionInLineChanged;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when Copy method was invoked and ClipBoardData changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when Copy method was invoked and ClipBoardData changed.")]
|
|||
|
public event EventHandler Copied;
|
|||
|
/// <summary>
|
|||
|
/// Occurs, when CopyHex method was invoked and ClipBoardData changed.
|
|||
|
/// </summary>
|
|||
|
[Description("Occurs, when CopyHex method was invoked and ClipBoardData changed.")]
|
|||
|
public event EventHandler CopiedHex;
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Ctors
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Initializes a new instance of a HexBox class.
|
|||
|
/// </summary>
|
|||
|
public HexBox()
|
|||
|
{
|
|||
|
this._vScrollBar = new VScrollBar();
|
|||
|
this._vScrollBar.Scroll += new ScrollEventHandler(_vScrollBar_Scroll);
|
|||
|
|
|||
|
this._builtInContextMenu = new BuiltInContextMenu(this);
|
|||
|
|
|||
|
BackColor = Color.White;
|
|||
|
Font = new Font("Courier New", 9F, FontStyle.Regular, GraphicsUnit.Point, ((byte)(0)));
|
|||
|
_stringFormat = new StringFormat(StringFormat.GenericTypographic);
|
|||
|
_stringFormat.FormatFlags = StringFormatFlags.MeasureTrailingSpaces;
|
|||
|
|
|||
|
ActivateEmptyKeyInterpreter();
|
|||
|
|
|||
|
SetStyle(ControlStyles.UserPaint, true);
|
|||
|
SetStyle(ControlStyles.DoubleBuffer, true);
|
|||
|
SetStyle(ControlStyles.AllPaintingInWmPaint, true);
|
|||
|
SetStyle(ControlStyles.ResizeRedraw, true);
|
|||
|
|
|||
|
_thumbTrackTimer.Interval = 50;
|
|||
|
_thumbTrackTimer.Tick += new EventHandler(PerformScrollThumbTrack);
|
|||
|
}
|
|||
|
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Scroll methods
|
|||
|
void _vScrollBar_Scroll(object sender, ScrollEventArgs e)
|
|||
|
{
|
|||
|
switch(e.Type)
|
|||
|
{
|
|||
|
case ScrollEventType.Last:
|
|||
|
break;
|
|||
|
case ScrollEventType.EndScroll:
|
|||
|
break;
|
|||
|
case ScrollEventType.SmallIncrement:
|
|||
|
PerformScrollLineDown();
|
|||
|
break;
|
|||
|
case ScrollEventType.SmallDecrement:
|
|||
|
PerformScrollLineUp();
|
|||
|
break;
|
|||
|
case ScrollEventType.LargeIncrement:
|
|||
|
PerformScrollPageDown();
|
|||
|
break;
|
|||
|
case ScrollEventType.LargeDecrement:
|
|||
|
PerformScrollPageUp();
|
|||
|
break;
|
|||
|
case ScrollEventType.ThumbPosition:
|
|||
|
long lPos = FromScrollPos(e.NewValue);
|
|||
|
PerformScrollThumpPosition(lPos);
|
|||
|
break;
|
|||
|
case ScrollEventType.ThumbTrack:
|
|||
|
// to avoid performance problems use a refresh delay implemented with a timer
|
|||
|
if (_thumbTrackTimer.Enabled) // stop old timer
|
|||
|
_thumbTrackTimer.Enabled = false;
|
|||
|
|
|||
|
// perform scroll immediately only if last refresh is very old
|
|||
|
int currentThumbTrack = System.Environment.TickCount;
|
|||
|
if (currentThumbTrack - _lastThumbtrack > THUMPTRACKDELAY)
|
|||
|
{
|
|||
|
PerformScrollThumbTrack(null, null);
|
|||
|
_lastThumbtrack = currentThumbTrack;
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
// start thumbtrack timer
|
|||
|
_thumbTrackPosition = FromScrollPos(e.NewValue);
|
|||
|
_thumbTrackTimer.Enabled = true;
|
|||
|
break;
|
|||
|
case ScrollEventType.First:
|
|||
|
break;
|
|||
|
default:
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
e.NewValue = ToScrollPos(_scrollVpos);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Performs the thumbtrack scrolling after an delay.
|
|||
|
/// </summary>
|
|||
|
void PerformScrollThumbTrack(object sender, EventArgs e)
|
|||
|
{
|
|||
|
_thumbTrackTimer.Enabled = false;
|
|||
|
PerformScrollThumpPosition(_thumbTrackPosition);
|
|||
|
_lastThumbtrack = Environment.TickCount;
|
|||
|
}
|
|||
|
|
|||
|
void UpdateScrollSize()
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("UpdateScrollSize()", "HexBox");
|
|||
|
|
|||
|
// calc scroll bar info
|
|||
|
if (VScrollBarVisible && _byteProvider != null && _byteProvider.Length > 0 && _iHexMaxHBytes != 0)
|
|||
|
{
|
|||
|
long scrollmax = (long)Math.Ceiling((double)(_byteProvider.Length + 1) / (double)_iHexMaxHBytes - (double)_iHexMaxVBytes);
|
|||
|
scrollmax = Math.Max(0, scrollmax);
|
|||
|
|
|||
|
long scrollpos = _startByte / _iHexMaxHBytes;
|
|||
|
|
|||
|
if (scrollmax < _scrollVmax)
|
|||
|
{
|
|||
|
/* Data size has been decreased. */
|
|||
|
if (_scrollVpos == _scrollVmax)
|
|||
|
/* Scroll one line up if we at bottom. */
|
|||
|
PerformScrollLineUp();
|
|||
|
}
|
|||
|
|
|||
|
if (scrollmax == _scrollVmax && scrollpos == _scrollVpos)
|
|||
|
return;
|
|||
|
|
|||
|
_scrollVmin = 0;
|
|||
|
_scrollVmax = scrollmax;
|
|||
|
_scrollVpos = Math.Min(scrollpos, scrollmax);
|
|||
|
UpdateVScroll();
|
|||
|
}
|
|||
|
else if (VScrollBarVisible)
|
|||
|
{
|
|||
|
// disable scroll bar
|
|||
|
_scrollVmin = 0;
|
|||
|
_scrollVmax = 0;
|
|||
|
_scrollVpos = 0;
|
|||
|
UpdateVScroll();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void UpdateVScroll()
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("UpdateVScroll()", "HexBox");
|
|||
|
|
|||
|
int max = ToScrollMax(_scrollVmax);
|
|||
|
|
|||
|
if(max > 0)
|
|||
|
{
|
|||
|
_vScrollBar.Minimum = 0;
|
|||
|
_vScrollBar.Maximum = max;
|
|||
|
_vScrollBar.Value = ToScrollPos(_scrollVpos);
|
|||
|
_vScrollBar.Enabled = true;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
_vScrollBar.Enabled = false;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
int ToScrollPos(long value)
|
|||
|
{
|
|||
|
int max = 65535;
|
|||
|
|
|||
|
if(_scrollVmax < max)
|
|||
|
return (int)value;
|
|||
|
else
|
|||
|
{
|
|||
|
double valperc = (double)value / (double)_scrollVmax * (double)100;
|
|||
|
int res = (int)Math.Floor((double)max / (double)100 * valperc);
|
|||
|
res = (int)Math.Max(_scrollVmin, res);
|
|||
|
res = (int)Math.Min(_scrollVmax, res);
|
|||
|
return res;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
long FromScrollPos(int value)
|
|||
|
{
|
|||
|
int max = 65535;
|
|||
|
if(_scrollVmax < max)
|
|||
|
{
|
|||
|
return (long)value;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
double valperc = (double)value / (double)max * (double)100;
|
|||
|
long res = (int)Math.Floor((double)_scrollVmax / (double)100 * valperc);
|
|||
|
return res;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
int ToScrollMax(long value)
|
|||
|
{
|
|||
|
long max = 65535;
|
|||
|
if(value > max)
|
|||
|
return (int)max;
|
|||
|
else
|
|||
|
return (int)value;
|
|||
|
}
|
|||
|
|
|||
|
void PerformScrollToLine(long pos)
|
|||
|
{
|
|||
|
if(pos < _scrollVmin || pos > _scrollVmax || pos == _scrollVpos )
|
|||
|
return;
|
|||
|
|
|||
|
_scrollVpos = pos;
|
|||
|
|
|||
|
UpdateVScroll();
|
|||
|
UpdateVisibilityBytes();
|
|||
|
UpdateCaret();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
|
|||
|
void PerformScrollLines(int lines)
|
|||
|
{
|
|||
|
long pos;
|
|||
|
if(lines > 0)
|
|||
|
{
|
|||
|
pos = Math.Min(_scrollVmax, _scrollVpos+lines);
|
|||
|
}
|
|||
|
else if(lines < 0)
|
|||
|
{
|
|||
|
pos = Math.Max(_scrollVmin, _scrollVpos+lines);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
PerformScrollToLine(pos);
|
|||
|
}
|
|||
|
|
|||
|
void PerformScrollLineDown()
|
|||
|
{
|
|||
|
this.PerformScrollLines(1);
|
|||
|
}
|
|||
|
|
|||
|
void PerformScrollLineUp()
|
|||
|
{
|
|||
|
this.PerformScrollLines(-1);
|
|||
|
}
|
|||
|
|
|||
|
void PerformScrollPageDown()
|
|||
|
{
|
|||
|
this.PerformScrollLines(_iHexMaxVBytes);
|
|||
|
}
|
|||
|
|
|||
|
void PerformScrollPageUp()
|
|||
|
{
|
|||
|
this.PerformScrollLines(-_iHexMaxVBytes);
|
|||
|
}
|
|||
|
|
|||
|
void PerformScrollThumpPosition(long pos)
|
|||
|
{
|
|||
|
// Bug fix: Scroll to end, do not scroll to end
|
|||
|
int difference = (_scrollVmax > 65535) ? 10 : 9;
|
|||
|
|
|||
|
if(ToScrollPos(pos) == ToScrollMax(_scrollVmax)-difference)
|
|||
|
pos = _scrollVmax;
|
|||
|
// End Bug fix
|
|||
|
|
|||
|
|
|||
|
PerformScrollToLine(pos);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Scrolls the selection start byte into view
|
|||
|
/// </summary>
|
|||
|
public void ScrollByteIntoView()
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("ScrollByteIntoView()", "HexBox");
|
|||
|
|
|||
|
ScrollByteIntoView(_bytePos);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Scrolls the specific byte into view
|
|||
|
/// </summary>
|
|||
|
/// <param name="index">the index of the byte</param>
|
|||
|
public void ScrollByteIntoView(long index)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("ScrollByteIntoView(long index)", "HexBox");
|
|||
|
|
|||
|
if(_byteProvider == null || _keyInterpreter == null)
|
|||
|
return;
|
|||
|
|
|||
|
if(index < _startByte)
|
|||
|
{
|
|||
|
long line = (long)Math.Floor((double)index / (double)_iHexMaxHBytes);
|
|||
|
PerformScrollThumpPosition(line);
|
|||
|
}
|
|||
|
else if(index > _endByte)
|
|||
|
{
|
|||
|
long line = (long)Math.Floor((double)index / (double)_iHexMaxHBytes);
|
|||
|
line -= _iHexMaxVBytes-1;
|
|||
|
PerformScrollThumpPosition(line);
|
|||
|
}
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Selection methods
|
|||
|
void ReleaseSelection()
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("ReleaseSelection()", "HexBox");
|
|||
|
|
|||
|
if(_selectionLength == 0)
|
|||
|
return;
|
|||
|
_selectionLength = 0;
|
|||
|
OnSelectionLengthChanged(EventArgs.Empty);
|
|||
|
|
|||
|
if(!_caretVisible)
|
|||
|
CreateCaret();
|
|||
|
else
|
|||
|
UpdateCaret();
|
|||
|
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Returns true if Select method could be invoked.
|
|||
|
/// </summary>
|
|||
|
public bool CanSelectAll()
|
|||
|
{
|
|||
|
if (!this.Enabled)
|
|||
|
return false;
|
|||
|
if (_byteProvider == null)
|
|||
|
return false;
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Selects all bytes.
|
|||
|
/// </summary>
|
|||
|
public void SelectAll()
|
|||
|
{
|
|||
|
if (this.ByteProvider == null)
|
|||
|
return;
|
|||
|
this.Select(0, this.ByteProvider.Length);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Selects the hex box.
|
|||
|
/// </summary>
|
|||
|
/// <param name="start">the start index of the selection</param>
|
|||
|
/// <param name="length">the length of the selection</param>
|
|||
|
public void Select(long start, long length)
|
|||
|
{
|
|||
|
if (this.ByteProvider == null)
|
|||
|
return;
|
|||
|
if (!this.Enabled)
|
|||
|
return;
|
|||
|
|
|||
|
InternalSelect(start, length);
|
|||
|
ScrollByteIntoView();
|
|||
|
}
|
|||
|
|
|||
|
void InternalSelect(long start, long length)
|
|||
|
{
|
|||
|
long pos = start;
|
|||
|
long sel = length;
|
|||
|
int cp = 0;
|
|||
|
|
|||
|
if(sel > 0 && _caretVisible)
|
|||
|
DestroyCaret();
|
|||
|
else if(sel == 0 && !_caretVisible)
|
|||
|
CreateCaret();
|
|||
|
|
|||
|
SetPosition(pos, cp);
|
|||
|
SetSelectionLength(sel);
|
|||
|
|
|||
|
UpdateCaret();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Key interpreter methods
|
|||
|
void ActivateEmptyKeyInterpreter()
|
|||
|
{
|
|||
|
if(_eki == null)
|
|||
|
_eki = new EmptyKeyInterpreter(this);
|
|||
|
|
|||
|
if(_eki == _keyInterpreter)
|
|||
|
return;
|
|||
|
|
|||
|
if(_keyInterpreter != null)
|
|||
|
_keyInterpreter.Deactivate();
|
|||
|
|
|||
|
_keyInterpreter = _eki;
|
|||
|
_keyInterpreter.Activate();
|
|||
|
}
|
|||
|
|
|||
|
void ActivateKeyInterpreter()
|
|||
|
{
|
|||
|
if(_ki == null)
|
|||
|
_ki = new KeyInterpreter(this);
|
|||
|
|
|||
|
if(_ki == _keyInterpreter)
|
|||
|
return;
|
|||
|
|
|||
|
if(_keyInterpreter != null)
|
|||
|
_keyInterpreter.Deactivate();
|
|||
|
|
|||
|
_keyInterpreter = _ki;
|
|||
|
_keyInterpreter.Activate();
|
|||
|
}
|
|||
|
|
|||
|
void ActivateStringKeyInterpreter()
|
|||
|
{
|
|||
|
if(_ski == null)
|
|||
|
_ski = new StringKeyInterpreter(this);
|
|||
|
|
|||
|
if(_ski == _keyInterpreter)
|
|||
|
return;
|
|||
|
|
|||
|
if(_keyInterpreter != null)
|
|||
|
_keyInterpreter.Deactivate();
|
|||
|
|
|||
|
_keyInterpreter = _ski;
|
|||
|
_keyInterpreter.Activate();
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Caret methods
|
|||
|
void CreateCaret()
|
|||
|
{
|
|||
|
if(_byteProvider == null || _keyInterpreter == null || _caretVisible || !this.Focused)
|
|||
|
return;
|
|||
|
|
|||
|
System.Diagnostics.Debug.WriteLine("CreateCaret()", "HexBox");
|
|||
|
|
|||
|
// define the caret width depending on InsertActive mode
|
|||
|
int caretWidth = (this.InsertActive) ? 1 : (int)_charSize.Width;
|
|||
|
int caretHeight = (int)_charSize.Height;
|
|||
|
NativeMethods.CreateCaret(Handle, IntPtr.Zero, caretWidth, caretHeight);
|
|||
|
|
|||
|
UpdateCaret();
|
|||
|
|
|||
|
NativeMethods.ShowCaret(Handle);
|
|||
|
|
|||
|
_caretVisible = true;
|
|||
|
}
|
|||
|
|
|||
|
void UpdateCaret()
|
|||
|
{
|
|||
|
if (_byteProvider == null || _keyInterpreter == null)
|
|||
|
return;
|
|||
|
|
|||
|
System.Diagnostics.Debug.WriteLine("UpdateCaret()", "HexBox");
|
|||
|
|
|||
|
long byteIndex =_bytePos - _startByte;
|
|||
|
PointF p = _keyInterpreter.GetCaretPointF(byteIndex);
|
|||
|
p.X += _byteCharacterPos*_charSize.Width;
|
|||
|
NativeMethods.SetCaretPos((int)p.X, (int)p.Y);
|
|||
|
}
|
|||
|
|
|||
|
void DestroyCaret()
|
|||
|
{
|
|||
|
if (!_caretVisible)
|
|||
|
return;
|
|||
|
|
|||
|
System.Diagnostics.Debug.WriteLine("DestroyCaret()", "HexBox");
|
|||
|
|
|||
|
NativeMethods.DestroyCaret();
|
|||
|
_caretVisible = false;
|
|||
|
}
|
|||
|
|
|||
|
void SetCaretPosition(Point p)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("SetCaretPosition()", "HexBox");
|
|||
|
|
|||
|
if (_byteProvider == null || _keyInterpreter == null)
|
|||
|
return;
|
|||
|
|
|||
|
long pos = _bytePos;
|
|||
|
int cp = _byteCharacterPos;
|
|||
|
|
|||
|
if(_recHex.Contains(p))
|
|||
|
{
|
|||
|
BytePositionInfo bpi = GetHexBytePositionInfo(p);
|
|||
|
pos = bpi.Index;
|
|||
|
cp = bpi.CharacterPosition;
|
|||
|
|
|||
|
SetPosition(pos, cp);
|
|||
|
|
|||
|
ActivateKeyInterpreter();
|
|||
|
UpdateCaret();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
else if(_recStringView.Contains(p))
|
|||
|
{
|
|||
|
BytePositionInfo bpi = GetStringBytePositionInfo(p);
|
|||
|
pos = bpi.Index;
|
|||
|
cp = bpi.CharacterPosition;
|
|||
|
|
|||
|
SetPosition(pos, cp);
|
|||
|
|
|||
|
ActivateStringKeyInterpreter();
|
|||
|
UpdateCaret();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
BytePositionInfo GetHexBytePositionInfo(Point p)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("GetHexBytePositionInfo()", "HexBox");
|
|||
|
|
|||
|
long bytePos;
|
|||
|
int byteCharaterPos;
|
|||
|
|
|||
|
float x = ((float)(p.X - _recHex.X) / _charSize.Width);
|
|||
|
float y = ((float)(p.Y - _recHex.Y) / _charSize.Height);
|
|||
|
int iX = (int)x;
|
|||
|
int iY = (int)y;
|
|||
|
|
|||
|
int hPos = (iX / 3 + 1);
|
|||
|
|
|||
|
bytePos = Math.Min(_byteProvider.Length,
|
|||
|
_startByte + (_iHexMaxHBytes * (iY+1) - _iHexMaxHBytes) + hPos - 1);
|
|||
|
byteCharaterPos = (iX % 3);
|
|||
|
if(byteCharaterPos > 1)
|
|||
|
byteCharaterPos = 1;
|
|||
|
|
|||
|
if(bytePos == _byteProvider.Length)
|
|||
|
byteCharaterPos = 0;
|
|||
|
|
|||
|
if(bytePos < 0)
|
|||
|
return new BytePositionInfo(0, 0);
|
|||
|
return new BytePositionInfo(bytePos, byteCharaterPos);
|
|||
|
}
|
|||
|
|
|||
|
BytePositionInfo GetStringBytePositionInfo(Point p)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("GetStringBytePositionInfo()", "HexBox");
|
|||
|
|
|||
|
long bytePos;
|
|||
|
int byteCharacterPos;
|
|||
|
|
|||
|
float x = ((float)(p.X - _recStringView.X) / _charSize.Width);
|
|||
|
float y = ((float)(p.Y - _recStringView.Y) / _charSize.Height);
|
|||
|
int iX = (int)x;
|
|||
|
int iY = (int)y;
|
|||
|
|
|||
|
int hPos = iX+1;
|
|||
|
|
|||
|
bytePos = Math.Min(_byteProvider.Length,
|
|||
|
_startByte + (_iHexMaxHBytes * (iY+1) - _iHexMaxHBytes) + hPos - 1);
|
|||
|
byteCharacterPos = 0;
|
|||
|
|
|||
|
if(bytePos < 0)
|
|||
|
return new BytePositionInfo(0, 0);
|
|||
|
return new BytePositionInfo(bytePos, byteCharacterPos);
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region PreProcessMessage methods
|
|||
|
/// <summary>
|
|||
|
/// Preprocesses windows messages.
|
|||
|
/// </summary>
|
|||
|
/// <param name="m">the message to process.</param>
|
|||
|
/// <returns>true, if the message was processed</returns>
|
|||
|
[SecurityPermission(SecurityAction.LinkDemand, UnmanagedCode=true), SecurityPermission(SecurityAction.InheritanceDemand, UnmanagedCode=true)]
|
|||
|
public override bool PreProcessMessage(ref Message m)
|
|||
|
{
|
|||
|
switch(m.Msg)
|
|||
|
{
|
|||
|
case NativeMethods.WM_KEYDOWN:
|
|||
|
return _keyInterpreter.PreProcessWmKeyDown(ref m);
|
|||
|
case NativeMethods.WM_CHAR:
|
|||
|
return _keyInterpreter.PreProcessWmChar(ref m);
|
|||
|
case NativeMethods.WM_KEYUP:
|
|||
|
return _keyInterpreter.PreProcessWmKeyUp(ref m);
|
|||
|
default:
|
|||
|
return base.PreProcessMessage (ref m);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
bool BasePreProcessMessage(ref Message m)
|
|||
|
{
|
|||
|
return base.PreProcessMessage(ref m);
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Find methods
|
|||
|
/// <summary>
|
|||
|
/// Searches the current ByteProvider
|
|||
|
/// </summary>
|
|||
|
/// <param name="bytes">the array of bytes to find</param>
|
|||
|
/// <param name="startIndex">the start index</param>
|
|||
|
/// <returns>the SelectionStart property value if find was successfull or
|
|||
|
/// -1 if there is no match
|
|||
|
/// -2 if Find was aborted.</returns>
|
|||
|
public long Find(byte[] bytes, long startIndex)
|
|||
|
{
|
|||
|
int match = 0;
|
|||
|
int bytesLength = bytes.Length;
|
|||
|
|
|||
|
_abortFind = false;
|
|||
|
|
|||
|
for(long pos = startIndex; pos < _byteProvider.Length; pos++)
|
|||
|
{
|
|||
|
if(_abortFind)
|
|||
|
return -2;
|
|||
|
|
|||
|
if(pos % 1000 == 0) // for performance reasons: DoEvents only 1 times per 1000 loops
|
|||
|
Application.DoEvents();
|
|||
|
|
|||
|
if(_byteProvider.ReadByte(pos) != bytes[match])
|
|||
|
{
|
|||
|
pos -= match;
|
|||
|
match = 0;
|
|||
|
_findingPos = pos;
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
match++;
|
|||
|
|
|||
|
if(match == bytesLength)
|
|||
|
{
|
|||
|
long bytePos = pos-bytesLength+1;
|
|||
|
Select(bytePos, bytesLength);
|
|||
|
ScrollByteIntoView(_bytePos+_selectionLength);
|
|||
|
ScrollByteIntoView(_bytePos);
|
|||
|
|
|||
|
return bytePos;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return -1;
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Aborts a working Find method.
|
|||
|
/// </summary>
|
|||
|
public void AbortFind()
|
|||
|
{
|
|||
|
_abortFind = true;
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets a value that indicates the current position during Find method execution.
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
|||
|
public long CurrentFindingPosition
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
return _findingPos;
|
|||
|
}
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Copy, Cut and Paste methods
|
|||
|
byte[] GetCopyData()
|
|||
|
{
|
|||
|
if (!CanCopy()) return new byte[0];
|
|||
|
|
|||
|
// put bytes into buffer
|
|||
|
byte[] buffer = new byte[_selectionLength];
|
|||
|
int id = -1;
|
|||
|
for (long i = _bytePos; i < _bytePos + _selectionLength; i++)
|
|||
|
{
|
|||
|
id++;
|
|||
|
|
|||
|
buffer[id] = _byteProvider.ReadByte(i);
|
|||
|
}
|
|||
|
return buffer;
|
|||
|
}
|
|||
|
/// <summary>
|
|||
|
/// Copies the current selection in the hex box to the Clipboard.
|
|||
|
/// </summary>
|
|||
|
public void Copy()
|
|||
|
{
|
|||
|
if(!CanCopy()) return;
|
|||
|
|
|||
|
// put bytes into buffer
|
|||
|
byte[] buffer = GetCopyData();
|
|||
|
|
|||
|
DataObject da = new DataObject();
|
|||
|
|
|||
|
// set string buffer clipbard data
|
|||
|
string sBuffer = System.Text.Encoding.ASCII.GetString(buffer, 0, buffer.Length);
|
|||
|
da.SetData(typeof(string), sBuffer);
|
|||
|
|
|||
|
//set memorystream (BinaryData) clipboard data
|
|||
|
System.IO.MemoryStream ms = new System.IO.MemoryStream(buffer, 0, buffer.Length, false, true);
|
|||
|
da.SetData("BinaryData", ms);
|
|||
|
|
|||
|
Clipboard.SetDataObject(da, true);
|
|||
|
UpdateCaret();
|
|||
|
ScrollByteIntoView();
|
|||
|
Invalidate();
|
|||
|
|
|||
|
OnCopied(EventArgs.Empty);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Return true if Copy method could be invoked.
|
|||
|
/// </summary>
|
|||
|
public bool CanCopy()
|
|||
|
{
|
|||
|
if(_selectionLength < 1 || _byteProvider == null)
|
|||
|
return false;
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Moves the current selection in the hex box to the Clipboard.
|
|||
|
/// </summary>
|
|||
|
public void Cut()
|
|||
|
{
|
|||
|
if(!CanCut()) return;
|
|||
|
|
|||
|
Copy();
|
|||
|
|
|||
|
_byteProvider.DeleteBytes(_bytePos, _selectionLength);
|
|||
|
_byteCharacterPos = 0;
|
|||
|
UpdateCaret();
|
|||
|
ScrollByteIntoView();
|
|||
|
ReleaseSelection();
|
|||
|
Invalidate();
|
|||
|
Refresh();
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Return true if Cut method could be invoked.
|
|||
|
/// </summary>
|
|||
|
public bool CanCut()
|
|||
|
{
|
|||
|
if (ReadOnly || !this.Enabled)
|
|||
|
return false;
|
|||
|
if(_byteProvider == null)
|
|||
|
return false;
|
|||
|
if(_selectionLength < 1 || !_byteProvider.SupportsDeleteBytes())
|
|||
|
return false;
|
|||
|
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Replaces the current selection in the hex box with the contents of the Clipboard.
|
|||
|
/// </summary>
|
|||
|
public void Paste()
|
|||
|
{
|
|||
|
if(!CanPaste()) return;
|
|||
|
|
|||
|
if(_selectionLength > 0)
|
|||
|
_byteProvider.DeleteBytes(_bytePos, _selectionLength);
|
|||
|
|
|||
|
byte[] buffer = null;
|
|||
|
IDataObject da = Clipboard.GetDataObject();
|
|||
|
if(da.GetDataPresent("BinaryData"))
|
|||
|
{
|
|||
|
System.IO.MemoryStream ms = (System.IO.MemoryStream)da.GetData("BinaryData");
|
|||
|
buffer = new byte[ms.Length];
|
|||
|
ms.Read(buffer, 0, buffer.Length);
|
|||
|
}
|
|||
|
else if(da.GetDataPresent(typeof(string)))
|
|||
|
{
|
|||
|
string sBuffer = (string)da.GetData(typeof(string));
|
|||
|
buffer = System.Text.Encoding.ASCII.GetBytes(sBuffer);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
_byteProvider.InsertBytes(_bytePos, buffer);
|
|||
|
|
|||
|
SetPosition(_bytePos + buffer.Length, 0);
|
|||
|
|
|||
|
ReleaseSelection();
|
|||
|
ScrollByteIntoView();
|
|||
|
UpdateCaret();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Return true if Paste method could be invoked.
|
|||
|
/// </summary>
|
|||
|
public bool CanPaste()
|
|||
|
{
|
|||
|
if (ReadOnly || !this.Enabled) return false;
|
|||
|
|
|||
|
if(_byteProvider == null || !_byteProvider.SupportsInsertBytes())
|
|||
|
return false;
|
|||
|
|
|||
|
if(!_byteProvider.SupportsDeleteBytes() && _selectionLength > 0)
|
|||
|
return false;
|
|||
|
|
|||
|
IDataObject da = Clipboard.GetDataObject();
|
|||
|
if(da.GetDataPresent("BinaryData"))
|
|||
|
return true;
|
|||
|
else if(da.GetDataPresent(typeof(string)))
|
|||
|
return true;
|
|||
|
else
|
|||
|
return false;
|
|||
|
}
|
|||
|
/// <summary>
|
|||
|
/// Return true if PasteHex method could be invoked.
|
|||
|
/// </summary>
|
|||
|
public bool CanPasteHex()
|
|||
|
{
|
|||
|
if (!CanPaste()) return false;
|
|||
|
|
|||
|
byte[] buffer = null;
|
|||
|
IDataObject da = Clipboard.GetDataObject();
|
|||
|
if (da.GetDataPresent(typeof(string)))
|
|||
|
{
|
|||
|
string hexString = (string)da.GetData(typeof(string));
|
|||
|
buffer = ConvertHexToBytes(hexString);
|
|||
|
return (buffer != null);
|
|||
|
}
|
|||
|
return false;
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Replaces the current selection in the hex box with the hex string data of the Clipboard.
|
|||
|
/// </summary>
|
|||
|
public void PasteHex()
|
|||
|
{
|
|||
|
if (!CanPaste()) return;
|
|||
|
|
|||
|
byte[] buffer = null;
|
|||
|
IDataObject da = Clipboard.GetDataObject();
|
|||
|
if (da.GetDataPresent(typeof(string)))
|
|||
|
{
|
|||
|
string hexString = (string)da.GetData(typeof(string));
|
|||
|
buffer = ConvertHexToBytes(hexString);
|
|||
|
if (buffer == null)
|
|||
|
return;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
if (_selectionLength > 0)
|
|||
|
_byteProvider.DeleteBytes(_bytePos, _selectionLength);
|
|||
|
|
|||
|
_byteProvider.InsertBytes(_bytePos, buffer);
|
|||
|
|
|||
|
SetPosition(_bytePos + buffer.Length, 0);
|
|||
|
|
|||
|
ReleaseSelection();
|
|||
|
ScrollByteIntoView();
|
|||
|
UpdateCaret();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Copies the current selection in the hex box to the Clipboard in hex format.
|
|||
|
/// </summary>
|
|||
|
public void CopyHex()
|
|||
|
{
|
|||
|
if (!CanCopy()) return;
|
|||
|
|
|||
|
// put bytes into buffer
|
|||
|
byte[] buffer = GetCopyData();
|
|||
|
|
|||
|
DataObject da = new DataObject();
|
|||
|
|
|||
|
// set string buffer clipbard data
|
|||
|
string hexString = ConvertBytesToHex(buffer); ;
|
|||
|
da.SetData(typeof(string), hexString);
|
|||
|
|
|||
|
//set memorystream (BinaryData) clipboard data
|
|||
|
System.IO.MemoryStream ms = new System.IO.MemoryStream(buffer, 0, buffer.Length, false, true);
|
|||
|
da.SetData("BinaryData", ms);
|
|||
|
|
|||
|
Clipboard.SetDataObject(da, true);
|
|||
|
UpdateCaret();
|
|||
|
ScrollByteIntoView();
|
|||
|
Invalidate();
|
|||
|
|
|||
|
OnCopiedHex(EventArgs.Empty);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Paint methods
|
|||
|
/// <summary>
|
|||
|
/// Paints the background.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">A PaintEventArgs that contains the event data.</param>
|
|||
|
protected override void OnPaintBackground(PaintEventArgs e)
|
|||
|
{
|
|||
|
switch(_borderStyle)
|
|||
|
{
|
|||
|
case BorderStyle.Fixed3D:
|
|||
|
{
|
|||
|
if(TextBoxRenderer.IsSupported)
|
|||
|
{
|
|||
|
VisualStyleElement state = VisualStyleElement.TextBox.TextEdit.Normal;
|
|||
|
Color backColor = this.BackColor;
|
|||
|
|
|||
|
if (this.Enabled)
|
|||
|
{
|
|||
|
if (this.ReadOnly)
|
|||
|
state = VisualStyleElement.TextBox.TextEdit.ReadOnly;
|
|||
|
else if (this.Focused)
|
|||
|
state = VisualStyleElement.TextBox.TextEdit.Focused;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
state = VisualStyleElement.TextBox.TextEdit.Disabled;
|
|||
|
backColor = this.BackColorDisabled;
|
|||
|
}
|
|||
|
|
|||
|
VisualStyleRenderer vsr = new VisualStyleRenderer(state);
|
|||
|
vsr.DrawBackground(e.Graphics, this.ClientRectangle);
|
|||
|
|
|||
|
Rectangle rectContent = vsr.GetBackgroundContentRectangle(e.Graphics, this.ClientRectangle);
|
|||
|
e.Graphics.FillRectangle(new SolidBrush(backColor), rectContent);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
// draw background
|
|||
|
e.Graphics.FillRectangle(new SolidBrush(BackColor), ClientRectangle);
|
|||
|
|
|||
|
// draw default border
|
|||
|
ControlPaint.DrawBorder3D(e.Graphics, ClientRectangle, Border3DStyle.Sunken);
|
|||
|
}
|
|||
|
|
|||
|
break;
|
|||
|
}
|
|||
|
case BorderStyle.FixedSingle:
|
|||
|
{
|
|||
|
// draw background
|
|||
|
e.Graphics.FillRectangle(new SolidBrush(BackColor), ClientRectangle);
|
|||
|
|
|||
|
// draw fixed single border
|
|||
|
ControlPaint.DrawBorder(e.Graphics, ClientRectangle, Color.Black, ButtonBorderStyle.Solid);
|
|||
|
break;
|
|||
|
}
|
|||
|
default:
|
|||
|
{
|
|||
|
// draw background
|
|||
|
e.Graphics.FillRectangle(new SolidBrush(BackColor), ClientRectangle);
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Paints the hex box.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">A PaintEventArgs that contains the event data.</param>
|
|||
|
protected override void OnPaint(PaintEventArgs e)
|
|||
|
{
|
|||
|
base.OnPaint(e);
|
|||
|
|
|||
|
if(_byteProvider == null)
|
|||
|
return;
|
|||
|
|
|||
|
System.Diagnostics.Debug.WriteLine("OnPaint " + DateTime.Now.ToString(), "HexBox");
|
|||
|
|
|||
|
// draw only in the content rectangle, so exclude the border and the scrollbar.
|
|||
|
Region r = new Region(ClientRectangle);
|
|||
|
r.Exclude(_recContent);
|
|||
|
e.Graphics.ExcludeClip(r);
|
|||
|
|
|||
|
UpdateVisibilityBytes();
|
|||
|
|
|||
|
if(_lineInfoVisible)
|
|||
|
PaintLineInfo(e.Graphics, _startByte, _endByte);
|
|||
|
|
|||
|
if(!_stringViewVisible)
|
|||
|
{
|
|||
|
PaintHex(e.Graphics, _startByte, _endByte);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
PaintHexAndStringView(e.Graphics, _startByte, _endByte);
|
|||
|
if(_shadowSelectionVisible)
|
|||
|
PaintCurrentBytesSign(e.Graphics);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void PaintLineInfo(Graphics g, long startByte, long endByte)
|
|||
|
{
|
|||
|
// Ensure endByte isn't > length of array.
|
|||
|
endByte = Math.Min(_byteProvider.Length-1, endByte);
|
|||
|
|
|||
|
Color lineInfoColor = (this.LineInfoForeColor != Color.Empty) ? this.LineInfoForeColor : this.ForeColor;
|
|||
|
Brush brush = new SolidBrush(lineInfoColor);
|
|||
|
|
|||
|
int maxLine = GetGridBytePoint(endByte-startByte).Y+1;
|
|||
|
|
|||
|
for(int i = 0; i < maxLine; i++)
|
|||
|
{
|
|||
|
long firstLineByte = (startByte + (_iHexMaxHBytes)*i) + _lineInfoOffset;
|
|||
|
|
|||
|
PointF bytePointF = GetBytePointF(new Point(0, 0+i));
|
|||
|
string info = firstLineByte.ToString(_hexStringFormat, System.Threading.Thread.CurrentThread.CurrentCulture);
|
|||
|
int nulls = 8-info.Length;
|
|||
|
string formattedInfo;
|
|||
|
if(nulls > -1)
|
|||
|
{
|
|||
|
formattedInfo = new string('0', 8-info.Length) + info;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
formattedInfo = new string('~', 8);
|
|||
|
}
|
|||
|
|
|||
|
g.DrawString(formattedInfo, Font, brush, new PointF(_recLineInfo.X, bytePointF.Y), _stringFormat);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void PaintHex(Graphics g, long startByte, long endByte)
|
|||
|
{
|
|||
|
Brush brush = new SolidBrush(GetDefaultForeColor());
|
|||
|
Brush selBrush = new SolidBrush(_selectionForeColor);
|
|||
|
Brush selBrushBack = new SolidBrush(_selectionBackColor);
|
|||
|
|
|||
|
int counter = -1;
|
|||
|
long intern_endByte = Math.Min(_byteProvider.Length-1, endByte+_iHexMaxHBytes);
|
|||
|
|
|||
|
bool isKeyInterpreterActive = _keyInterpreter == null || _keyInterpreter.GetType() == typeof(KeyInterpreter);
|
|||
|
|
|||
|
for(long i = startByte; i < intern_endByte+1; i++)
|
|||
|
{
|
|||
|
counter++;
|
|||
|
Point gridPoint = GetGridBytePoint(counter);
|
|||
|
byte b = _byteProvider.ReadByte(i);
|
|||
|
|
|||
|
bool isSelectedByte = i >= _bytePos && i <= (_bytePos + _selectionLength-1) && _selectionLength != 0;
|
|||
|
|
|||
|
if(isSelectedByte && isKeyInterpreterActive)
|
|||
|
{
|
|||
|
PaintHexStringSelected(g, b, selBrush, selBrushBack, gridPoint);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
PaintHexString(g, b, brush, gridPoint);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void PaintHexString(Graphics g, byte b, Brush brush, Point gridPoint)
|
|||
|
{
|
|||
|
PointF bytePointF = GetBytePointF(gridPoint);
|
|||
|
|
|||
|
string sB = ConvertByteToHex(b);
|
|||
|
|
|||
|
g.DrawString(sB.Substring(0,1), Font, brush, bytePointF, _stringFormat);
|
|||
|
bytePointF.X += _charSize.Width;
|
|||
|
g.DrawString(sB.Substring(1,1), Font, brush, bytePointF, _stringFormat);
|
|||
|
}
|
|||
|
|
|||
|
void PaintHexStringSelected(Graphics g, byte b, Brush brush, Brush brushBack, Point gridPoint)
|
|||
|
{
|
|||
|
string sB = b.ToString(_hexStringFormat, System.Threading.Thread.CurrentThread.CurrentCulture);
|
|||
|
if(sB.Length == 1)
|
|||
|
sB = "0" + sB;
|
|||
|
|
|||
|
PointF bytePointF = GetBytePointF(gridPoint);
|
|||
|
|
|||
|
bool isLastLineChar = (gridPoint.X+1 == _iHexMaxHBytes);
|
|||
|
float bcWidth = (isLastLineChar) ? _charSize.Width*2 : _charSize.Width*3;
|
|||
|
|
|||
|
g.FillRectangle(brushBack, bytePointF.X, bytePointF.Y, bcWidth, _charSize.Height);
|
|||
|
g.DrawString(sB.Substring(0,1), Font, brush, bytePointF, _stringFormat);
|
|||
|
bytePointF.X += _charSize.Width;
|
|||
|
g.DrawString(sB.Substring(1,1), Font, brush, bytePointF, _stringFormat);
|
|||
|
}
|
|||
|
|
|||
|
void PaintHexAndStringView(Graphics g, long startByte, long endByte)
|
|||
|
{
|
|||
|
Brush brush = new SolidBrush(GetDefaultForeColor());
|
|||
|
Brush selBrush = new SolidBrush(_selectionForeColor);
|
|||
|
Brush selBrushBack = new SolidBrush(_selectionBackColor);
|
|||
|
|
|||
|
int counter = -1;
|
|||
|
long intern_endByte = Math.Min(_byteProvider.Length-1, endByte+_iHexMaxHBytes);
|
|||
|
|
|||
|
bool isKeyInterpreterActive = _keyInterpreter == null || _keyInterpreter.GetType() == typeof(KeyInterpreter);
|
|||
|
bool isStringKeyInterpreterActive = _keyInterpreter != null && _keyInterpreter.GetType() == typeof(StringKeyInterpreter);
|
|||
|
|
|||
|
for(long i = startByte; i < intern_endByte+1; i++)
|
|||
|
{
|
|||
|
counter++;
|
|||
|
Point gridPoint = GetGridBytePoint(counter);
|
|||
|
PointF byteStringPointF = GetByteStringPointF(gridPoint);
|
|||
|
byte b = _byteProvider.ReadByte(i);
|
|||
|
|
|||
|
bool isSelectedByte = i >= _bytePos && i <= (_bytePos + _selectionLength-1) && _selectionLength != 0;
|
|||
|
|
|||
|
if(isSelectedByte && isKeyInterpreterActive)
|
|||
|
{
|
|||
|
PaintHexStringSelected(g, b, selBrush, selBrushBack, gridPoint);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
PaintHexString(g, b, brush, gridPoint);
|
|||
|
}
|
|||
|
|
|||
|
string s = new String(ByteCharConverter.ToChar(b), 1);
|
|||
|
|
|||
|
if(isSelectedByte && isStringKeyInterpreterActive)
|
|||
|
{
|
|||
|
g.FillRectangle(selBrushBack, byteStringPointF.X, byteStringPointF.Y, _charSize.Width, _charSize.Height);
|
|||
|
g.DrawString(s, Font, selBrush, byteStringPointF, _stringFormat);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
g.DrawString(s, Font, brush, byteStringPointF, _stringFormat);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void PaintCurrentBytesSign(Graphics g)
|
|||
|
{
|
|||
|
if(_keyInterpreter != null && Focused && _bytePos != -1 && Enabled)
|
|||
|
{
|
|||
|
if(_keyInterpreter.GetType() == typeof(KeyInterpreter))
|
|||
|
{
|
|||
|
if(_selectionLength == 0)
|
|||
|
{
|
|||
|
Point gp = GetGridBytePoint(_bytePos - _startByte);
|
|||
|
PointF pf = GetByteStringPointF(gp);
|
|||
|
Size s = new Size((int)_charSize.Width, (int)_charSize.Height);
|
|||
|
Rectangle r = new Rectangle((int)pf.X, (int)pf.Y, s.Width, s.Height);
|
|||
|
if(r.IntersectsWith(_recStringView))
|
|||
|
{
|
|||
|
r.Intersect(_recStringView);
|
|||
|
PaintCurrentByteSign(g, r);
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
int lineWidth = (int)(_recStringView.Width-_charSize.Width);
|
|||
|
|
|||
|
Point startSelGridPoint = GetGridBytePoint(_bytePos-_startByte);
|
|||
|
PointF startSelPointF = GetByteStringPointF(startSelGridPoint);
|
|||
|
|
|||
|
Point endSelGridPoint = GetGridBytePoint(_bytePos-_startByte+_selectionLength-1);
|
|||
|
PointF endSelPointF = GetByteStringPointF(endSelGridPoint);
|
|||
|
|
|||
|
int multiLine = endSelGridPoint.Y - startSelGridPoint.Y;
|
|||
|
if(multiLine == 0)
|
|||
|
{
|
|||
|
Rectangle singleLine = new Rectangle(
|
|||
|
(int)startSelPointF.X,
|
|||
|
(int)startSelPointF.Y,
|
|||
|
(int)(endSelPointF.X-startSelPointF.X+_charSize.Width),
|
|||
|
(int)_charSize.Height);
|
|||
|
if(singleLine.IntersectsWith(_recStringView))
|
|||
|
{
|
|||
|
singleLine.Intersect(_recStringView);
|
|||
|
PaintCurrentByteSign(g, singleLine);
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
Rectangle firstLine = new Rectangle(
|
|||
|
(int)startSelPointF.X,
|
|||
|
(int)startSelPointF.Y,
|
|||
|
(int)(_recStringView.X+lineWidth-startSelPointF.X+_charSize.Width),
|
|||
|
(int)_charSize.Height);
|
|||
|
if(firstLine.IntersectsWith(_recStringView))
|
|||
|
{
|
|||
|
firstLine.Intersect(_recStringView);
|
|||
|
PaintCurrentByteSign(g, firstLine);
|
|||
|
}
|
|||
|
|
|||
|
if(multiLine > 1)
|
|||
|
{
|
|||
|
Rectangle betweenLines = new Rectangle(
|
|||
|
_recStringView.X,
|
|||
|
(int)(startSelPointF.Y+_charSize.Height),
|
|||
|
(int)(_recStringView.Width),
|
|||
|
(int)(_charSize.Height*(multiLine-1)));
|
|||
|
if(betweenLines.IntersectsWith(_recStringView))
|
|||
|
{
|
|||
|
betweenLines.Intersect(_recStringView);
|
|||
|
PaintCurrentByteSign(g, betweenLines);
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
Rectangle lastLine = new Rectangle(
|
|||
|
_recStringView.X,
|
|||
|
(int)endSelPointF.Y,
|
|||
|
(int)(endSelPointF.X-_recStringView.X+_charSize.Width),
|
|||
|
(int)_charSize.Height);
|
|||
|
if(lastLine.IntersectsWith(_recStringView))
|
|||
|
{
|
|||
|
lastLine.Intersect(_recStringView);
|
|||
|
PaintCurrentByteSign(g, lastLine);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
if(_selectionLength == 0)
|
|||
|
{
|
|||
|
Point gp = GetGridBytePoint(_bytePos - _startByte);
|
|||
|
PointF pf = GetBytePointF(gp);
|
|||
|
Size s = new Size((int)_charSize.Width * 2, (int)_charSize.Height);
|
|||
|
Rectangle r = new Rectangle((int)pf.X, (int)pf.Y, s.Width, s.Height);
|
|||
|
PaintCurrentByteSign(g, r);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
int lineWidth = (int)(_recHex.Width-_charSize.Width*5);
|
|||
|
|
|||
|
Point startSelGridPoint = GetGridBytePoint(_bytePos-_startByte);
|
|||
|
PointF startSelPointF = GetBytePointF(startSelGridPoint);
|
|||
|
|
|||
|
Point endSelGridPoint = GetGridBytePoint(_bytePos-_startByte+_selectionLength-1);
|
|||
|
PointF endSelPointF = GetBytePointF(endSelGridPoint);
|
|||
|
|
|||
|
int multiLine = endSelGridPoint.Y - startSelGridPoint.Y;
|
|||
|
if(multiLine == 0)
|
|||
|
{
|
|||
|
Rectangle singleLine = new Rectangle(
|
|||
|
(int)startSelPointF.X,
|
|||
|
(int)startSelPointF.Y,
|
|||
|
(int)(endSelPointF.X-startSelPointF.X+_charSize.Width*2),
|
|||
|
(int)_charSize.Height);
|
|||
|
if(singleLine.IntersectsWith(_recHex))
|
|||
|
{
|
|||
|
singleLine.Intersect(_recHex);
|
|||
|
PaintCurrentByteSign(g, singleLine);
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
Rectangle firstLine = new Rectangle(
|
|||
|
(int)startSelPointF.X,
|
|||
|
(int)startSelPointF.Y,
|
|||
|
(int)(_recHex.X+lineWidth-startSelPointF.X+_charSize.Width*2),
|
|||
|
(int)_charSize.Height);
|
|||
|
if(firstLine.IntersectsWith(_recHex))
|
|||
|
{
|
|||
|
firstLine.Intersect(_recHex);
|
|||
|
PaintCurrentByteSign(g, firstLine);
|
|||
|
}
|
|||
|
|
|||
|
if(multiLine > 1)
|
|||
|
{
|
|||
|
Rectangle betweenLines = new Rectangle(
|
|||
|
_recHex.X,
|
|||
|
(int)(startSelPointF.Y+_charSize.Height),
|
|||
|
(int)(lineWidth+_charSize.Width*2),
|
|||
|
(int)(_charSize.Height*(multiLine-1)));
|
|||
|
if(betweenLines.IntersectsWith(_recHex))
|
|||
|
{
|
|||
|
betweenLines.Intersect(_recHex);
|
|||
|
PaintCurrentByteSign(g, betweenLines);
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
Rectangle lastLine = new Rectangle(
|
|||
|
_recHex.X,
|
|||
|
(int)endSelPointF.Y,
|
|||
|
(int)(endSelPointF.X-_recHex.X+_charSize.Width*2),
|
|||
|
(int)_charSize.Height);
|
|||
|
if(lastLine.IntersectsWith(_recHex))
|
|||
|
{
|
|||
|
lastLine.Intersect(_recHex);
|
|||
|
PaintCurrentByteSign(g, lastLine);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void PaintCurrentByteSign(Graphics g, Rectangle rec)
|
|||
|
{
|
|||
|
// stack overflowexception on big files - workaround
|
|||
|
if(rec.Top < 0 || rec.Left < 0 || rec.Width <= 0 || rec.Height <= 0)
|
|||
|
return;
|
|||
|
|
|||
|
Bitmap myBitmap = new Bitmap(rec.Width, rec.Height);
|
|||
|
Graphics bitmapGraphics = Graphics.FromImage(myBitmap);
|
|||
|
|
|||
|
SolidBrush greenBrush = new SolidBrush(_shadowSelectionColor);
|
|||
|
|
|||
|
bitmapGraphics.FillRectangle(greenBrush, 0,
|
|||
|
0, rec.Width, rec.Height);
|
|||
|
|
|||
|
g.CompositingQuality = System.Drawing.Drawing2D.CompositingQuality.GammaCorrected;
|
|||
|
|
|||
|
g.DrawImage(myBitmap, rec.Left, rec.Top);
|
|||
|
}
|
|||
|
|
|||
|
Color GetDefaultForeColor()
|
|||
|
{
|
|||
|
if(Enabled)
|
|||
|
return ForeColor;
|
|||
|
else
|
|||
|
return Color.Gray;
|
|||
|
}
|
|||
|
void UpdateVisibilityBytes()
|
|||
|
{
|
|||
|
if(_byteProvider == null || _byteProvider.Length == 0)
|
|||
|
return;
|
|||
|
|
|||
|
_startByte = (_scrollVpos+1) * _iHexMaxHBytes - _iHexMaxHBytes;
|
|||
|
_endByte = (long)Math.Min(_byteProvider.Length - 1, _startByte + _iHexMaxBytes);
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Positioning methods
|
|||
|
void UpdateRectanglePositioning()
|
|||
|
{
|
|||
|
// calc char size
|
|||
|
SizeF charSize = this.CreateGraphics().MeasureString("A", Font, 100, _stringFormat);
|
|||
|
_charSize = new SizeF((float)Math.Ceiling(charSize.Width), (float)Math.Ceiling(charSize.Height));
|
|||
|
|
|||
|
// calc content bounds
|
|||
|
_recContent = ClientRectangle;
|
|||
|
_recContent.X += _recBorderLeft;
|
|||
|
_recContent.Y += _recBorderTop;
|
|||
|
_recContent.Width -= _recBorderRight+_recBorderLeft;
|
|||
|
_recContent.Height -= _recBorderBottom+_recBorderTop;
|
|||
|
|
|||
|
if(_vScrollBarVisible)
|
|||
|
{
|
|||
|
_recContent.Width -= _vScrollBar.Width;
|
|||
|
_vScrollBar.Left = _recContent.X+_recContent.Width;
|
|||
|
_vScrollBar.Top = _recContent.Y;
|
|||
|
_vScrollBar.Height = _recContent.Height;
|
|||
|
}
|
|||
|
|
|||
|
int marginLeft = 4;
|
|||
|
|
|||
|
// calc line info bounds
|
|||
|
if(_lineInfoVisible)
|
|||
|
{
|
|||
|
_recLineInfo = new Rectangle(_recContent.X+marginLeft,
|
|||
|
_recContent.Y,
|
|||
|
(int)(_charSize.Width*10),
|
|||
|
_recContent.Height);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
_recLineInfo = Rectangle.Empty;
|
|||
|
_recLineInfo.X = marginLeft;
|
|||
|
}
|
|||
|
|
|||
|
// calc hex bounds and grid
|
|||
|
_recHex = new Rectangle(_recLineInfo.X + _recLineInfo.Width,
|
|||
|
_recLineInfo.Y,
|
|||
|
_recContent.Width - _recLineInfo.Width,
|
|||
|
_recContent.Height);
|
|||
|
|
|||
|
if(UseFixedBytesPerLine)
|
|||
|
{
|
|||
|
SetHorizontalByteCount(_bytesPerLine);
|
|||
|
_recHex.Width = (int)Math.Floor(((double)_iHexMaxHBytes)*_charSize.Width*3+(2*_charSize.Width));
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
int hmax = (int)Math.Floor((double)_recHex.Width/(double)_charSize.Width);
|
|||
|
if(hmax > 1)
|
|||
|
SetHorizontalByteCount((int)Math.Floor((double)hmax/3));
|
|||
|
else
|
|||
|
SetHorizontalByteCount(hmax);
|
|||
|
}
|
|||
|
|
|||
|
if(_stringViewVisible)
|
|||
|
{
|
|||
|
_recStringView = new Rectangle(_recHex.X + _recHex.Width,
|
|||
|
_recHex.Y,
|
|||
|
(int)(_charSize.Width*_iHexMaxHBytes),
|
|||
|
_recHex.Height);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
_recStringView = Rectangle.Empty;
|
|||
|
}
|
|||
|
|
|||
|
int vmax = (int)Math.Floor((double)_recHex.Height/(double)_charSize.Height);
|
|||
|
SetVerticalByteCount(vmax);
|
|||
|
|
|||
|
_iHexMaxBytes = _iHexMaxHBytes * _iHexMaxVBytes;
|
|||
|
|
|||
|
UpdateScrollSize();
|
|||
|
}
|
|||
|
|
|||
|
PointF GetBytePointF(long byteIndex)
|
|||
|
{
|
|||
|
Point gp = GetGridBytePoint(byteIndex);
|
|||
|
|
|||
|
return GetBytePointF(gp);
|
|||
|
}
|
|||
|
|
|||
|
PointF GetBytePointF(Point gp)
|
|||
|
{
|
|||
|
float x = (3 * _charSize.Width) * gp.X + _recHex.X;
|
|||
|
float y = (gp.Y+1)*_charSize.Height-_charSize.Height+_recHex.Y;
|
|||
|
|
|||
|
return new PointF(x,y);
|
|||
|
}
|
|||
|
|
|||
|
PointF GetByteStringPointF(Point gp)
|
|||
|
{
|
|||
|
float x = (_charSize.Width) * gp.X + _recStringView.X;
|
|||
|
float y = (gp.Y+1)*_charSize.Height-_charSize.Height+_recStringView.Y;
|
|||
|
|
|||
|
return new PointF(x,y);
|
|||
|
}
|
|||
|
|
|||
|
Point GetGridBytePoint(long byteIndex)
|
|||
|
{
|
|||
|
int row = (int)Math.Floor((double)byteIndex/(double)_iHexMaxHBytes);
|
|||
|
int column = (int)(byteIndex+_iHexMaxHBytes-_iHexMaxHBytes*(row+1));
|
|||
|
|
|||
|
Point res = new Point(column, row);
|
|||
|
return res;
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Overridden properties
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the background color for the control.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue(typeof(Color), "White")]
|
|||
|
public override Color BackColor
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
return base.BackColor;
|
|||
|
}
|
|||
|
set
|
|||
|
{
|
|||
|
base.BackColor = value;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// The font used to display text in the hexbox.
|
|||
|
/// </summary>
|
|||
|
public override Font Font
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
return base.Font;
|
|||
|
}
|
|||
|
set
|
|||
|
{
|
|||
|
base.Font = value;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Not used.
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), EditorBrowsable(EditorBrowsableState.Never), Bindable(false)]
|
|||
|
public override string Text
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
return base.Text;
|
|||
|
}
|
|||
|
set
|
|||
|
{
|
|||
|
base.Text = value;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Not used.
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), EditorBrowsable(EditorBrowsableState.Never), Bindable(false)]
|
|||
|
public override RightToLeft RightToLeft
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
return base.RightToLeft;
|
|||
|
}
|
|||
|
set
|
|||
|
{
|
|||
|
base.RightToLeft = value;
|
|||
|
}
|
|||
|
}
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Properties
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the background color for the disabled control.
|
|||
|
/// </summary>
|
|||
|
[Category("Appearance"), DefaultValue(typeof(Color), "WhiteSmoke")]
|
|||
|
public Color BackColorDisabled
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
return _backColorDisabled;
|
|||
|
}
|
|||
|
set
|
|||
|
{
|
|||
|
_backColorDisabled = value;
|
|||
|
}
|
|||
|
} Color _backColorDisabled = Color.FromName("WhiteSmoke");
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets if the count of bytes in one line is fix.
|
|||
|
/// </summary>
|
|||
|
/// <remarks>
|
|||
|
/// When set to True, BytesPerLine property determine the maximum count of bytes in one line.
|
|||
|
/// </remarks>
|
|||
|
[DefaultValue(false), Category("Hex"), Description("Gets or sets if the count of bytes in one line is fix.")]
|
|||
|
public bool ReadOnly
|
|||
|
{
|
|||
|
get { return _readOnly; }
|
|||
|
set
|
|||
|
{
|
|||
|
if(_readOnly == value)
|
|||
|
return;
|
|||
|
|
|||
|
_readOnly = value;
|
|||
|
OnReadOnlyChanged(EventArgs.Empty);
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
} bool _readOnly;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the maximum count of bytes in one line.
|
|||
|
/// </summary>
|
|||
|
/// <remarks>
|
|||
|
/// UsedFixedBytesPerLine property must set to true
|
|||
|
/// </remarks>
|
|||
|
[DefaultValue(16), Category("Hex"), Description("Gets or sets the maximum count of bytes in one line.")]
|
|||
|
public int BytesPerLine
|
|||
|
{
|
|||
|
get { return _bytesPerLine; }
|
|||
|
set
|
|||
|
{
|
|||
|
if(_bytesPerLine == value)
|
|||
|
return;
|
|||
|
|
|||
|
_bytesPerLine = value;
|
|||
|
OnBytesPerLineChanged(EventArgs.Empty);
|
|||
|
|
|||
|
UpdateRectanglePositioning();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
} int _bytesPerLine = 16;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets if the count of bytes in one line is fix.
|
|||
|
/// </summary>
|
|||
|
/// <remarks>
|
|||
|
/// When set to True, BytesPerLine property determine the maximum count of bytes in one line.
|
|||
|
/// </remarks>
|
|||
|
[DefaultValue(false), Category("Hex"), Description("Gets or sets if the count of bytes in one line is fix.")]
|
|||
|
public bool UseFixedBytesPerLine
|
|||
|
{
|
|||
|
get { return _useFixedBytesPerLine; }
|
|||
|
set
|
|||
|
{
|
|||
|
if(_useFixedBytesPerLine == value)
|
|||
|
return;
|
|||
|
|
|||
|
_useFixedBytesPerLine = value;
|
|||
|
OnUseFixedBytesPerLineChanged(EventArgs.Empty);
|
|||
|
|
|||
|
UpdateRectanglePositioning();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
} bool _useFixedBytesPerLine;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the visibility of a vertical scroll bar.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue(false), Category("Hex"), Description("Gets or sets the visibility of a vertical scroll bar.")]
|
|||
|
public bool VScrollBarVisible
|
|||
|
{
|
|||
|
get { return this._vScrollBarVisible; }
|
|||
|
set
|
|||
|
{
|
|||
|
if(_vScrollBarVisible == value)
|
|||
|
return;
|
|||
|
|
|||
|
_vScrollBarVisible = value;
|
|||
|
|
|||
|
if(_vScrollBarVisible)
|
|||
|
Controls.Add(_vScrollBar);
|
|||
|
else
|
|||
|
Controls.Remove(_vScrollBar);
|
|||
|
|
|||
|
UpdateRectanglePositioning();
|
|||
|
UpdateScrollSize();
|
|||
|
|
|||
|
OnVScrollBarVisibleChanged(EventArgs.Empty);
|
|||
|
}
|
|||
|
} bool _vScrollBarVisible;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the ByteProvider.
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
|||
|
public IByteProvider ByteProvider
|
|||
|
{
|
|||
|
get { return _byteProvider; }
|
|||
|
set
|
|||
|
{
|
|||
|
if(_byteProvider == value)
|
|||
|
return;
|
|||
|
|
|||
|
if(value == null)
|
|||
|
ActivateEmptyKeyInterpreter();
|
|||
|
else
|
|||
|
ActivateKeyInterpreter();
|
|||
|
|
|||
|
if(_byteProvider != null)
|
|||
|
_byteProvider.LengthChanged -= new EventHandler(_byteProvider_LengthChanged);
|
|||
|
|
|||
|
_byteProvider = value;
|
|||
|
if(_byteProvider != null)
|
|||
|
_byteProvider.LengthChanged += new EventHandler(_byteProvider_LengthChanged);
|
|||
|
|
|||
|
OnByteProviderChanged(EventArgs.Empty);
|
|||
|
|
|||
|
if(value == null) // do not raise events if value is null
|
|||
|
{
|
|||
|
_bytePos = -1;
|
|||
|
_byteCharacterPos = 0;
|
|||
|
_selectionLength = 0;
|
|||
|
|
|||
|
DestroyCaret();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
SetPosition(0, 0);
|
|||
|
SetSelectionLength(0);
|
|||
|
|
|||
|
if(_caretVisible && Focused)
|
|||
|
UpdateCaret();
|
|||
|
else
|
|||
|
CreateCaret();
|
|||
|
}
|
|||
|
|
|||
|
CheckCurrentLineChanged();
|
|||
|
CheckCurrentPositionInLineChanged();
|
|||
|
|
|||
|
_scrollVpos = 0;
|
|||
|
|
|||
|
UpdateVisibilityBytes();
|
|||
|
UpdateRectanglePositioning();
|
|||
|
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
IByteProvider _byteProvider;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the visibility of a line info.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue(false), Category("Hex"), Description("Gets or sets the visibility of a line info.")]
|
|||
|
public bool LineInfoVisible
|
|||
|
{
|
|||
|
get { return _lineInfoVisible; }
|
|||
|
set
|
|||
|
{
|
|||
|
if(_lineInfoVisible == value)
|
|||
|
return;
|
|||
|
|
|||
|
_lineInfoVisible = value;
|
|||
|
OnLineInfoVisibleChanged(EventArgs.Empty);
|
|||
|
|
|||
|
UpdateRectanglePositioning();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
} bool _lineInfoVisible;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the offset of a line info.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue((long)0), Category("Hex"), Description("Gets or sets the offset of the line info.")]
|
|||
|
public long LineInfoOffset
|
|||
|
{
|
|||
|
get { return _lineInfoOffset; }
|
|||
|
set
|
|||
|
{
|
|||
|
if (_lineInfoOffset == value)
|
|||
|
return;
|
|||
|
|
|||
|
_lineInfoOffset = value;
|
|||
|
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
} long _lineInfoOffset;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the hex box<6F>s border style.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue(typeof(BorderStyle), "Fixed3D"), Category("Hex"), Description("Gets or sets the hex box<6F>s border style.")]
|
|||
|
public BorderStyle BorderStyle
|
|||
|
{
|
|||
|
get { return _borderStyle;}
|
|||
|
set
|
|||
|
{
|
|||
|
if(_borderStyle == value)
|
|||
|
return;
|
|||
|
|
|||
|
_borderStyle = value;
|
|||
|
switch(_borderStyle)
|
|||
|
{
|
|||
|
case BorderStyle.None:
|
|||
|
_recBorderLeft = _recBorderTop = _recBorderRight = _recBorderBottom = 0;
|
|||
|
break;
|
|||
|
case BorderStyle.Fixed3D:
|
|||
|
_recBorderLeft = _recBorderRight = SystemInformation.Border3DSize.Width;
|
|||
|
_recBorderTop = _recBorderBottom = SystemInformation.Border3DSize.Height;
|
|||
|
break;
|
|||
|
case BorderStyle.FixedSingle:
|
|||
|
_recBorderLeft = _recBorderTop = _recBorderRight = _recBorderBottom = 1;
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
UpdateRectanglePositioning();
|
|||
|
|
|||
|
OnBorderStyleChanged(EventArgs.Empty);
|
|||
|
|
|||
|
}
|
|||
|
} BorderStyle _borderStyle = BorderStyle.Fixed3D;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the visibility of the string view.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue(false), Category("Hex"), Description("Gets or sets the visibility of the string view.")]
|
|||
|
public bool StringViewVisible
|
|||
|
{
|
|||
|
get { return _stringViewVisible; }
|
|||
|
set
|
|||
|
{
|
|||
|
if(_stringViewVisible == value)
|
|||
|
return;
|
|||
|
|
|||
|
_stringViewVisible = value;
|
|||
|
OnStringViewVisibleChanged(EventArgs.Empty);
|
|||
|
|
|||
|
UpdateRectanglePositioning();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
} bool _stringViewVisible;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets whether the HexBox control displays the hex characters in upper or lower case.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue(typeof(HexCasing), "Upper"), Category("Hex"), Description("Gets or sets whether the HexBox control displays the hex characters in upper or lower case.")]
|
|||
|
public HexCasing HexCasing
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
if(_hexStringFormat == "X")
|
|||
|
return HexCasing.Upper;
|
|||
|
else
|
|||
|
return HexCasing.Lower;
|
|||
|
}
|
|||
|
set
|
|||
|
{
|
|||
|
string format;
|
|||
|
if(value == HexCasing.Upper)
|
|||
|
format = "X";
|
|||
|
else
|
|||
|
format = "x";
|
|||
|
|
|||
|
if(_hexStringFormat == format)
|
|||
|
return;
|
|||
|
|
|||
|
_hexStringFormat = format;
|
|||
|
OnHexCasingChanged(EventArgs.Empty);
|
|||
|
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets and sets the starting point of the bytes selected in the hex box.
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
|||
|
public long SelectionStart
|
|||
|
{
|
|||
|
get { return _bytePos; }
|
|||
|
set
|
|||
|
{
|
|||
|
SetPosition(value, 0);
|
|||
|
ScrollByteIntoView();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets and sets the number of bytes selected in the hex box.
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
|||
|
public long SelectionLength
|
|||
|
{
|
|||
|
get { return _selectionLength; }
|
|||
|
set
|
|||
|
{
|
|||
|
SetSelectionLength(value);
|
|||
|
ScrollByteIntoView();
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
} long _selectionLength;
|
|||
|
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the line info color. When this property is null, then ForeColor property is used.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue(typeof(Color), "Empty"), Category("Hex"), Description("Gets or sets the line info color. When this property is null, then ForeColor property is used.")]
|
|||
|
public Color LineInfoForeColor
|
|||
|
{
|
|||
|
get { return _lineInfoForeColor; }
|
|||
|
set { _lineInfoForeColor = value; Invalidate(); }
|
|||
|
} Color _lineInfoForeColor = Color.Empty;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the background color for the selected bytes.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue(typeof(Color), "Blue"), Category("Hex"), Description("Gets or sets the background color for the selected bytes.")]
|
|||
|
public Color SelectionBackColor
|
|||
|
{
|
|||
|
get { return _selectionBackColor; }
|
|||
|
set { _selectionBackColor = value; Invalidate(); }
|
|||
|
} Color _selectionBackColor = Color.Blue;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the foreground color for the selected bytes.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue(typeof(Color), "White"), Category("Hex"), Description("Gets or sets the foreground color for the selected bytes.")]
|
|||
|
public Color SelectionForeColor
|
|||
|
{
|
|||
|
get { return _selectionForeColor; }
|
|||
|
set { _selectionForeColor = value; Invalidate(); }
|
|||
|
} Color _selectionForeColor = Color.White;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the visibility of a shadow selection.
|
|||
|
/// </summary>
|
|||
|
[DefaultValue(true), Category("Hex"), Description("Gets or sets the visibility of a shadow selection.")]
|
|||
|
public bool ShadowSelectionVisible
|
|||
|
{
|
|||
|
get { return _shadowSelectionVisible; }
|
|||
|
set
|
|||
|
{
|
|||
|
if(_shadowSelectionVisible == value)
|
|||
|
return;
|
|||
|
_shadowSelectionVisible = value;
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
} bool _shadowSelectionVisible = true;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the color of the shadow selection.
|
|||
|
/// </summary>
|
|||
|
/// <remarks>
|
|||
|
/// A alpha component must be given!
|
|||
|
/// Default alpha = 100
|
|||
|
/// </remarks>
|
|||
|
[Category("Hex"), Description("Gets or sets the color of the shadow selection.")]
|
|||
|
public Color ShadowSelectionColor
|
|||
|
{
|
|||
|
get { return _shadowSelectionColor; }
|
|||
|
set { _shadowSelectionColor = value; Invalidate(); }
|
|||
|
} Color _shadowSelectionColor = Color.FromArgb(100, 60, 188, 255);
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets the number bytes drawn horizontally.
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
|||
|
public int HorizontalByteCount
|
|||
|
{
|
|||
|
get { return _iHexMaxHBytes; }
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets the number bytes drawn vertically.
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
|||
|
public int VerticalByteCount
|
|||
|
{
|
|||
|
get { return _iHexMaxVBytes; }
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets the current line
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
|||
|
public long CurrentLine
|
|||
|
{
|
|||
|
get { return _currentLine; }
|
|||
|
} long _currentLine;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets the current position in the current line
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
|||
|
public long CurrentPositionInLine
|
|||
|
{
|
|||
|
get { return _currentPositionInLine; }
|
|||
|
} int _currentPositionInLine;
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets the a value if insertion mode is active or not.
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
|||
|
public bool InsertActive
|
|||
|
{
|
|||
|
get { return _insertActive; }
|
|||
|
set
|
|||
|
{
|
|||
|
if (_insertActive == value)
|
|||
|
return;
|
|||
|
|
|||
|
_insertActive = value;
|
|||
|
|
|||
|
// recreate caret
|
|||
|
DestroyCaret();
|
|||
|
CreateCaret();
|
|||
|
|
|||
|
// raise change event
|
|||
|
OnInsertActiveChanged(EventArgs.Empty);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the built-in context menu.
|
|||
|
/// </summary>
|
|||
|
[DesignerSerializationVisibility(DesignerSerializationVisibility.Content)]
|
|||
|
public BuiltInContextMenu BuiltInContextMenu
|
|||
|
{
|
|||
|
get { return _builtInContextMenu; }
|
|||
|
} BuiltInContextMenu _builtInContextMenu;
|
|||
|
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Gets or sets the converter that will translate between byte and character values.
|
|||
|
/// </summary>
|
|||
|
[Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
|||
|
public IByteCharConverter ByteCharConverter
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
if (_byteCharConverter == null)
|
|||
|
_byteCharConverter = new DefaultByteCharConverter();
|
|||
|
return _byteCharConverter;
|
|||
|
}
|
|||
|
set
|
|||
|
{
|
|||
|
if (value != null && value != _byteCharConverter)
|
|||
|
{
|
|||
|
_byteCharConverter = value;
|
|||
|
Invalidate();
|
|||
|
}
|
|||
|
}
|
|||
|
} IByteCharConverter _byteCharConverter;
|
|||
|
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Misc
|
|||
|
/// <summary>
|
|||
|
/// Converts a byte array to a hex string. For example: {10,11} = "0A 0B"
|
|||
|
/// </summary>
|
|||
|
/// <param name="data">the byte array</param>
|
|||
|
/// <returns>the hex string</returns>
|
|||
|
string ConvertBytesToHex(byte[] data)
|
|||
|
{
|
|||
|
StringBuilder sb = new StringBuilder();
|
|||
|
foreach (byte b in data)
|
|||
|
{
|
|||
|
string hex = ConvertByteToHex(b);
|
|||
|
sb.Append(hex);
|
|||
|
sb.Append(" ");
|
|||
|
}
|
|||
|
if (sb.Length > 0)
|
|||
|
sb.Remove(sb.Length - 1, 1);
|
|||
|
string result = sb.ToString();
|
|||
|
return result;
|
|||
|
}
|
|||
|
/// <summary>
|
|||
|
/// Converts the byte to a hex string. For example: "10" = "0A";
|
|||
|
/// </summary>
|
|||
|
/// <param name="b">the byte to format</param>
|
|||
|
/// <returns>the hex string</returns>
|
|||
|
string ConvertByteToHex(byte b)
|
|||
|
{
|
|||
|
string sB = b.ToString(_hexStringFormat, System.Threading.Thread.CurrentThread.CurrentCulture);
|
|||
|
if (sB.Length == 1)
|
|||
|
sB = "0" + sB;
|
|||
|
return sB;
|
|||
|
}
|
|||
|
/// <summary>
|
|||
|
/// Converts the hex string to an byte array. The hex string must be separated by a space char ' '. If there is any invalid hex information in the string the result will be null.
|
|||
|
/// </summary>
|
|||
|
/// <param name="hex">the hex string separated by ' '. For example: "0A 0B 0C"</param>
|
|||
|
/// <returns>the byte array. null if hex is invalid or empty</returns>
|
|||
|
byte[] ConvertHexToBytes(string hex)
|
|||
|
{
|
|||
|
if (string.IsNullOrEmpty(hex))
|
|||
|
return null;
|
|||
|
hex = hex.Trim();
|
|||
|
var hexArray = hex.Split(' ');
|
|||
|
var byteArray = new byte[hexArray.Length];
|
|||
|
|
|||
|
for(int i = 0; i < hexArray.Length; i++)
|
|||
|
{
|
|||
|
var hexValue = hexArray[i];
|
|||
|
|
|||
|
byte b;
|
|||
|
var isByte = ConvertHexToByte(hexValue, out b);
|
|||
|
if (!isByte)
|
|||
|
return null;
|
|||
|
byteArray[i] = b;
|
|||
|
}
|
|||
|
|
|||
|
return byteArray;
|
|||
|
}
|
|||
|
|
|||
|
bool ConvertHexToByte(string hex, out byte b)
|
|||
|
{
|
|||
|
bool isByte = byte.TryParse(hex, System.Globalization.NumberStyles.HexNumber, System.Threading.Thread.CurrentThread.CurrentCulture, out b);
|
|||
|
return isByte;
|
|||
|
}
|
|||
|
|
|||
|
void SetPosition(long bytePos)
|
|||
|
{
|
|||
|
SetPosition(bytePos, _byteCharacterPos);
|
|||
|
}
|
|||
|
|
|||
|
void SetPosition(long bytePos, int byteCharacterPos)
|
|||
|
{
|
|||
|
if(_byteCharacterPos != byteCharacterPos)
|
|||
|
{
|
|||
|
_byteCharacterPos = byteCharacterPos;
|
|||
|
}
|
|||
|
|
|||
|
if(bytePos != _bytePos)
|
|||
|
{
|
|||
|
_bytePos = bytePos;
|
|||
|
CheckCurrentLineChanged();
|
|||
|
CheckCurrentPositionInLineChanged();
|
|||
|
|
|||
|
OnSelectionStartChanged(EventArgs.Empty);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void SetSelectionLength(long selectionLength)
|
|||
|
{
|
|||
|
if(selectionLength != _selectionLength)
|
|||
|
{
|
|||
|
_selectionLength = selectionLength;
|
|||
|
OnSelectionLengthChanged(EventArgs.Empty);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void SetHorizontalByteCount(int value)
|
|||
|
{
|
|||
|
if(_iHexMaxHBytes == value)
|
|||
|
return;
|
|||
|
|
|||
|
_iHexMaxHBytes = value;
|
|||
|
OnHorizontalByteCountChanged(EventArgs.Empty);
|
|||
|
}
|
|||
|
|
|||
|
void SetVerticalByteCount(int value)
|
|||
|
{
|
|||
|
if(_iHexMaxVBytes == value)
|
|||
|
return;
|
|||
|
|
|||
|
_iHexMaxVBytes = value;
|
|||
|
OnVerticalByteCountChanged(EventArgs.Empty);
|
|||
|
}
|
|||
|
|
|||
|
void CheckCurrentLineChanged()
|
|||
|
{
|
|||
|
long currentLine = (long)Math.Floor((double)_bytePos / (double)_iHexMaxHBytes) + 1;
|
|||
|
|
|||
|
if(_byteProvider == null && _currentLine != 0)
|
|||
|
{
|
|||
|
_currentLine = 0;
|
|||
|
OnCurrentLineChanged(EventArgs.Empty);
|
|||
|
}
|
|||
|
else if(currentLine != _currentLine)
|
|||
|
{
|
|||
|
_currentLine = currentLine;
|
|||
|
OnCurrentLineChanged(EventArgs.Empty);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void CheckCurrentPositionInLineChanged()
|
|||
|
{
|
|||
|
Point gb = GetGridBytePoint(_bytePos);
|
|||
|
int currentPositionInLine = gb.X + 1;
|
|||
|
|
|||
|
if(_byteProvider == null && _currentPositionInLine != 0)
|
|||
|
{
|
|||
|
_currentPositionInLine = 0;
|
|||
|
OnCurrentPositionInLineChanged(EventArgs.Empty);
|
|||
|
}
|
|||
|
else if(currentPositionInLine != _currentPositionInLine)
|
|||
|
{
|
|||
|
_currentPositionInLine = currentPositionInLine;
|
|||
|
OnCurrentPositionInLineChanged(EventArgs.Empty);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the InsertActiveChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnInsertActiveChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(InsertActiveChanged != null)
|
|||
|
InsertActiveChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the ReadOnlyChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnReadOnlyChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(ReadOnlyChanged != null)
|
|||
|
ReadOnlyChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the ByteProviderChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnByteProviderChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(ByteProviderChanged != null)
|
|||
|
ByteProviderChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the SelectionStartChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnSelectionStartChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(SelectionStartChanged != null)
|
|||
|
SelectionStartChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the SelectionLengthChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnSelectionLengthChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(SelectionLengthChanged != null)
|
|||
|
SelectionLengthChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the LineInfoVisibleChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnLineInfoVisibleChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(LineInfoVisibleChanged != null)
|
|||
|
LineInfoVisibleChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the StringViewVisibleChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnStringViewVisibleChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(StringViewVisibleChanged != null)
|
|||
|
StringViewVisibleChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the BorderStyleChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnBorderStyleChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(BorderStyleChanged != null)
|
|||
|
BorderStyleChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the UseFixedBytesPerLineChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnUseFixedBytesPerLineChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(UseFixedBytesPerLineChanged != null)
|
|||
|
UseFixedBytesPerLineChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the BytesPerLineChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnBytesPerLineChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(BytesPerLineChanged != null)
|
|||
|
BytesPerLineChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the VScrollBarVisibleChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnVScrollBarVisibleChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(VScrollBarVisibleChanged != null)
|
|||
|
VScrollBarVisibleChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the HexCasingChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnHexCasingChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(HexCasingChanged != null)
|
|||
|
HexCasingChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the HorizontalByteCountChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnHorizontalByteCountChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(HorizontalByteCountChanged != null)
|
|||
|
HorizontalByteCountChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the VerticalByteCountChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnVerticalByteCountChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(VerticalByteCountChanged != null)
|
|||
|
VerticalByteCountChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the CurrentLineChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnCurrentLineChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(CurrentLineChanged != null)
|
|||
|
CurrentLineChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the CurrentPositionInLineChanged event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnCurrentPositionInLineChanged(EventArgs e)
|
|||
|
{
|
|||
|
if(CurrentPositionInLineChanged != null)
|
|||
|
CurrentPositionInLineChanged(this, e);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the Copied event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnCopied(EventArgs e)
|
|||
|
{
|
|||
|
if (Copied != null)
|
|||
|
Copied(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the CopiedHex event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected virtual void OnCopiedHex(EventArgs e)
|
|||
|
{
|
|||
|
if (CopiedHex != null)
|
|||
|
CopiedHex(this, e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the MouseDown event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected override void OnMouseDown(MouseEventArgs e)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("OnMouseDown()", "HexBox");
|
|||
|
|
|||
|
if(!Focused)
|
|||
|
Focus();
|
|||
|
|
|||
|
if(e.Button == MouseButtons.Left)
|
|||
|
SetCaretPosition(new Point(e.X, e.Y));
|
|||
|
|
|||
|
base.OnMouseDown (e);
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the MouseWhell event
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected override void OnMouseWheel(MouseEventArgs e)
|
|||
|
{
|
|||
|
int linesToScroll = -(e.Delta * SystemInformation.MouseWheelScrollLines / 120);
|
|||
|
this.PerformScrollLines(linesToScroll);
|
|||
|
|
|||
|
base.OnMouseWheel (e);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the Resize event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected override void OnResize(EventArgs e)
|
|||
|
{
|
|||
|
base.OnResize (e);
|
|||
|
UpdateRectanglePositioning();
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the GotFocus event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected override void OnGotFocus(EventArgs e)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("OnGotFocus()", "HexBox");
|
|||
|
|
|||
|
base.OnGotFocus (e);
|
|||
|
|
|||
|
CreateCaret();
|
|||
|
}
|
|||
|
|
|||
|
/// <summary>
|
|||
|
/// Raises the LostFocus event.
|
|||
|
/// </summary>
|
|||
|
/// <param name="e">An EventArgs that contains the event data.</param>
|
|||
|
protected override void OnLostFocus(EventArgs e)
|
|||
|
{
|
|||
|
System.Diagnostics.Debug.WriteLine("OnLostFocus()", "HexBox");
|
|||
|
|
|||
|
base.OnLostFocus (e);
|
|||
|
|
|||
|
DestroyCaret();
|
|||
|
}
|
|||
|
|
|||
|
void _byteProvider_LengthChanged(object sender, EventArgs e)
|
|||
|
{
|
|||
|
UpdateScrollSize();
|
|||
|
}
|
|||
|
#endregion
|
|||
|
}
|
|||
|
}
|