123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889 |
- #nullable enable
- //
- // HexView.cs: A hexadecimal viewer
- //
- // TODO: Support searching and highlighting of the search result
- // TODO: Support shrinking the stream (e.g. del/backspace should work).
- //
- using System.Buffers;
- namespace Terminal.Gui;
- /// <summary>Hex viewer and editor <see cref="View"/> over a <see cref="Stream"/></summary>
- /// <remarks>
- /// <para>
- /// <see cref="HexView"/> provides a hex editor on top of a seekable <see cref="Stream"/> with the left side
- /// showing the hex values of the bytes in the <see cref="Stream"/> and the right side showing the contents
- /// (filtered
- /// to non-control sequence ASCII characters).
- /// </para>
- /// <para>Users can switch from one side to the other by using the tab key.</para>
- /// <para>
- /// To enable editing, set <see cref="AllowEdits"/> to true. When <see cref="AllowEdits"/> is true the user can
- /// make changes to the hexadecimal values of the <see cref="Stream"/>. Any changes are tracked in the
- /// <see cref="Edits"/> property (a <see cref="SortedDictionary{TKey, TValue}"/>) indicating the position where the
- /// changes were made and the new values. A convenience method, <see cref="ApplyEdits"/> will apply the edits to
- /// the <see cref="Stream"/>.
- /// </para>
- /// <para>
- /// Control the byte at the caret for editing by setting the <see cref="Address"/> property to an offset in the
- /// stream.
- /// </para>
- /// </remarks>
- public class HexView : View, IDesignable
- {
- private const int DEFAULT_ADDRESS_WIDTH = 8; // The default value for AddressWidth
- private const int NUM_BYTES_PER_HEX_COLUMN = 4;
- private const int HEX_COLUMN_WIDTH = NUM_BYTES_PER_HEX_COLUMN * 3 + 2; // 3 cols per byte + 1 for vert separator + right space
- private bool _firstNibble;
- private bool _leftSideHasFocus;
- private static readonly Rune _spaceCharRune = new (' ');
- private static readonly Rune _periodCharRune = Glyphs.DottedSquare;
- private static readonly Rune _columnSeparatorRune = Glyphs.VLineDa4;
- /// <summary>Initializes a <see cref="HexView"/> class.</summary>
- /// <param name="source">
- /// The <see cref="Stream"/> to view and edit as hex, this <see cref="Stream"/> must support seeking,
- /// or an exception will be thrown.
- /// </param>
- public HexView (Stream? source)
- {
- Source = source;
- CanFocus = true;
- CursorVisibility = CursorVisibility.Default;
- _leftSideHasFocus = true;
- _firstNibble = true;
- AddCommand (Command.Select, HandleMouseClick);
- AddCommand (Command.Left, () => MoveLeft ());
- AddCommand (Command.Right, () => MoveRight ());
- AddCommand (Command.Down, () => MoveDown (BytesPerLine));
- AddCommand (Command.Up, () => MoveUp (BytesPerLine));
- AddCommand (Command.PageUp, () => MoveUp (BytesPerLine * Viewport.Height));
- AddCommand (Command.PageDown, () => MoveDown (BytesPerLine * Viewport.Height));
- AddCommand (Command.Start, () => MoveHome ());
- AddCommand (Command.End, () => MoveEnd ());
- AddCommand (Command.LeftStart, () => MoveLeftStart ());
- AddCommand (Command.RightEnd, () => MoveEndOfLine ());
- AddCommand (Command.StartOfPage, () => MoveUp (BytesPerLine * ((int)(Address - Viewport.Y) / BytesPerLine)));
- AddCommand (
- Command.EndOfPage,
- () => MoveDown (BytesPerLine * (Viewport.Height - 1 - (int)(Address - Viewport.Y) / BytesPerLine))
- );
- AddCommand (Command.ScrollDown, () => ScrollVertical (1));
- AddCommand (Command.ScrollUp, () => ScrollVertical (-1));
- AddCommand (Command.DeleteCharLeft, () => true);
- AddCommand (Command.DeleteCharRight, () => true);
- AddCommand (Command.Insert, () => true);
- KeyBindings.Add (Key.CursorLeft, Command.Left);
- KeyBindings.Add (Key.CursorRight, Command.Right);
- KeyBindings.Add (Key.CursorDown, Command.Down);
- KeyBindings.Add (Key.CursorUp, Command.Up);
- KeyBindings.Add (Key.PageUp, Command.PageUp);
- KeyBindings.Add (Key.PageDown, Command.PageDown);
- KeyBindings.Add (Key.Home, Command.Start);
- KeyBindings.Add (Key.End, Command.End);
- KeyBindings.Add (Key.CursorLeft.WithCtrl, Command.LeftStart);
- KeyBindings.Add (Key.CursorRight.WithCtrl, Command.RightEnd);
- KeyBindings.Add (Key.CursorUp.WithCtrl, Command.StartOfPage);
- KeyBindings.Add (Key.CursorDown.WithCtrl, Command.EndOfPage);
- KeyBindings.Add (Key.Backspace, Command.DeleteCharLeft);
- KeyBindings.Add (Key.Delete, Command.DeleteCharRight);
- KeyBindings.Add (Key.InsertChar, Command.Insert);
- KeyBindings.Remove (Key.Space);
- KeyBindings.Remove (Key.Enter);
- // The Select handler deals with both single and double clicks
- MouseBindings.ReplaceCommands (MouseFlags.Button1Clicked, Command.Select);
- MouseBindings.Add (MouseFlags.Button1DoubleClicked, Command.Select);
- MouseBindings.Add (MouseFlags.WheeledUp, Command.ScrollUp);
- MouseBindings.Add (MouseFlags.WheeledDown, Command.ScrollDown);
- SubViewsLaidOut += HexViewSubViewsLaidOut;
- }
- private void HexViewSubViewsLaidOut (object? sender, LayoutEventArgs e)
- {
- SetBytesPerLine ();
- SetContentSize (new (GetLeftSideStartColumn () + BytesPerLine / NUM_BYTES_PER_HEX_COLUMN * HEX_COLUMN_WIDTH + BytesPerLine - 1, (int)((GetEditedSize ()) / BytesPerLine) + 1));
- }
- /// <summary>Initializes a <see cref="HexView"/> class.</summary>
- public HexView () : this (new MemoryStream ()) { }
- /// <summary>
- /// Gets or sets whether this <see cref="HexView"/> allows editing of the <see cref="Stream"/> of the underlying
- /// <see cref="Stream"/>.
- /// </summary>
- /// <value><c>true</c> to allow edits; otherwise, <c>false</c>.</value>
- public bool AllowEdits { get; set; } = true;
- /// <summary>Gets the current edit position.</summary>
- /// <param name="address"></param>
- public Point GetPosition (long address)
- {
- if (_source is null || BytesPerLine == 0)
- {
- return Point.Empty;
- }
- var line = address / BytesPerLine;
- var item = address % BytesPerLine;
- return new ((int)item, (int)line);
- }
- /// <summary>Gets cursor location, given an address.</summary>
- /// <param name="address"></param>
- public Point GetCursor (long address)
- {
- Point position = GetPosition (address);
- if (_leftSideHasFocus)
- {
- int block = position.X / NUM_BYTES_PER_HEX_COLUMN;
- int column = position.X % NUM_BYTES_PER_HEX_COLUMN;
- position.X = block * HEX_COLUMN_WIDTH + column * 3 + (_firstNibble ? 0 : 1);
- }
- else
- {
- position.X += BytesPerLine / NUM_BYTES_PER_HEX_COLUMN * HEX_COLUMN_WIDTH - 1;
- }
- position.X += GetLeftSideStartColumn ();
- position.Offset (-Viewport.X, -Viewport.Y);
- return position;
- }
- private void ScrollToMakeCursorVisible (Point offsetToNewCursor)
- {
- // Adjust vertical scrolling
- if (offsetToNewCursor.Y < 1)
- {
- ScrollVertical (offsetToNewCursor.Y);
- }
- else if (offsetToNewCursor.Y >= Viewport.Height)
- {
- ScrollVertical (offsetToNewCursor.Y);
- }
- if (offsetToNewCursor.X < 1)
- {
- ScrollHorizontal (offsetToNewCursor.X);
- }
- else if (offsetToNewCursor.X >= Viewport.Width)
- {
- ScrollHorizontal (offsetToNewCursor.X);
- }
- }
- ///<inheritdoc/>
- public override Point? PositionCursor ()
- {
- Point position = GetCursor (Address);
- if (HasFocus
- && position.X >= 0
- && position.X < Viewport.Width
- && position.Y >= 0
- && position.Y < Viewport.Height)
- {
- Move (position.X, position.Y);
- return position;
- }
- return null;
- }
- private SortedDictionary<long, byte> _edits = [];
- /// <summary>
- /// Gets a <see cref="SortedDictionary{TKey, TValue}"/> describing the edits done to the <see cref="HexView"/>.
- /// Each Key indicates an offset where an edit was made and the Value is the changed byte.
- /// </summary>
- /// <value>The edits.</value>
- public IReadOnlyDictionary<long, byte> Edits => _edits;
- private long GetEditedSize ()
- {
- if (_edits.Count == 0)
- {
- return _source!.Length;
- }
- long maxEditAddress = _edits.Keys.Max ();
- return Math.Max (_source!.Length, maxEditAddress + 1);
- }
- /// <summary>
- /// Applies and edits made to the <see cref="Stream"/> and resets the contents of the
- /// <see cref="Edits"/> property.
- /// </summary>
- /// <param name="stream">If provided also applies the changes to the passed <see cref="Stream"/>.</param>
- /// .
- public void ApplyEdits (Stream? stream = null)
- {
- foreach (KeyValuePair<long, byte> kv in _edits)
- {
- _source!.Position = kv.Key;
- _source.WriteByte (kv.Value);
- _source.Flush ();
- if (stream is { })
- {
- stream.Position = kv.Key;
- stream.WriteByte (kv.Value);
- stream.Flush ();
- }
- }
- _edits = new ();
- SetNeedsDraw ();
- }
- /// <summary>
- /// Discards the edits made to the <see cref="Stream"/> by resetting the contents of the
- /// <see cref="Edits"/> property.
- /// </summary>
- public void DiscardEdits () { _edits = new (); }
- private Stream? _source;
- /// <summary>
- /// Sets or gets the <see cref="Stream"/> the <see cref="HexView"/> is operating on; the stream must support
- /// seeking ( <see cref="Stream.CanSeek"/> == true).
- /// </summary>
- /// <value>The source.</value>
- public Stream? Source
- {
- get => _source;
- set
- {
- ArgumentNullException.ThrowIfNull (value);
- if (!value!.CanSeek)
- {
- throw new ArgumentException (@"The source stream must be seekable (CanSeek property)");
- }
- DiscardEdits ();
- _source = value;
- SetBytesPerLine ();
- if (Address > _source.Length)
- {
- Address = 0;
- }
- SetNeedsLayout ();
- SetNeedsDraw ();
- }
- }
- private int _bpl;
- /// <summary>The bytes length per line.</summary>
- public int BytesPerLine
- {
- get => _bpl;
- set
- {
- _bpl = value;
- RaisePositionChanged ();
- }
- }
- private long _address;
- /// <summary>Gets or sets the current byte position in the <see cref="Stream"/>.</summary>
- public long Address
- {
- get => _address;
- set
- {
- if (_address == value)
- {
- return;
- }
- long newAddress = Math.Clamp (value, 0, GetEditedSize ());
- Point offsetToNewCursor = GetCursor (newAddress);
- _address = newAddress;
- // Ensure the new cursor position is visible
- ScrollToMakeCursorVisible (offsetToNewCursor);
- RaisePositionChanged ();
- }
- }
- private int _addressWidth = DEFAULT_ADDRESS_WIDTH;
- /// <summary>
- /// Gets or sets the width of the Address column on the left. Set to 0 to hide. The default is 8.
- /// </summary>
- public int AddressWidth
- {
- get => _addressWidth;
- set
- {
- if (_addressWidth == value)
- {
- return;
- }
- _addressWidth = value;
- SetNeedsDraw ();
- SetNeedsLayout ();
- }
- }
- private int GetLeftSideStartColumn () { return AddressWidth == 0 ? 0 : AddressWidth + 1; }
- private bool? HandleMouseClick (ICommandContext? commandContext)
- {
- if (commandContext is not CommandContext<MouseBinding> { Binding.MouseEventArgs: { } } mouseCommandContext)
- {
- return false;
- }
- if (RaiseSelecting (commandContext) is true)
- {
- return true;
- }
- if (!HasFocus)
- {
- SetFocus ();
- }
- if (mouseCommandContext.Binding.MouseEventArgs.Position.X < GetLeftSideStartColumn ())
- {
- return true;
- }
- int blocks = BytesPerLine / NUM_BYTES_PER_HEX_COLUMN;
- int blocksSize = blocks * HEX_COLUMN_WIDTH;
- int blocksRightOffset = GetLeftSideStartColumn () + blocksSize - 1;
- if (mouseCommandContext.Binding.MouseEventArgs.Position.X > blocksRightOffset + BytesPerLine - 1)
- {
- return true;
- }
- bool clickIsOnLeftSide = mouseCommandContext.Binding.MouseEventArgs.Position.X >= blocksRightOffset;
- long lineStart = mouseCommandContext.Binding.MouseEventArgs.Position.Y * BytesPerLine + Viewport.Y * BytesPerLine;
- int x = mouseCommandContext.Binding.MouseEventArgs.Position.X - GetLeftSideStartColumn () + 1;
- int block = x / HEX_COLUMN_WIDTH;
- x -= block * 2;
- int empty = x % 3;
- int item = x / 3;
- if (!clickIsOnLeftSide && item > 0 && (empty == 0 || x == block * HEX_COLUMN_WIDTH + HEX_COLUMN_WIDTH - 1 - block * 2))
- {
- return true;
- }
- _firstNibble = true;
- if (clickIsOnLeftSide)
- {
- Address = Math.Min (lineStart + mouseCommandContext.Binding.MouseEventArgs.Position.X - blocksRightOffset, GetEditedSize ());
- }
- else
- {
- Address = Math.Min (lineStart + item, GetEditedSize ());
- }
- if (mouseCommandContext.Binding.MouseEventArgs.Flags == MouseFlags.Button1DoubleClicked)
- {
- _leftSideHasFocus = !clickIsOnLeftSide;
- if (_leftSideHasFocus)
- {
- _firstNibble = empty == 1;
- }
- else
- {
- _firstNibble = true;
- }
- SetNeedsDraw ();
- }
- return false;
- }
- ///<inheritdoc/>
- protected override bool OnDrawingContent ()
- {
- if (Source is null)
- {
- return true;
- }
- Attribute currentAttribute = Attribute.Default;
- Attribute current = GetFocusColor ();
- SetAttribute (current);
- Move (-Viewport.X, 0);
- long addressOfFirstLine = Viewport.Y * BytesPerLine;
- int nBlocks = BytesPerLine / NUM_BYTES_PER_HEX_COLUMN;
- var data = new byte [nBlocks * NUM_BYTES_PER_HEX_COLUMN * Viewport.Height];
- Source.Position = addressOfFirstLine;
- long bytesRead = Source!.Read (data, 0, data.Length);
- Attribute selectedAttribute = GetHotNormalColor ();
- Attribute editedAttribute = new Attribute (GetNormalColor ().Foreground.GetHighlightColor (), GetNormalColor ().Background);
- Attribute editingAttribute = new Attribute (GetFocusColor ().Background, GetFocusColor ().Foreground);
- Attribute addressAttribute = new Attribute (GetNormalColor ().Foreground.GetHighlightColor (), GetNormalColor ().Background);
- for (var line = 0; line < Viewport.Height; line++)
- {
- Move (-Viewport.X, line);
- long addressOfLine = addressOfFirstLine + line * nBlocks * NUM_BYTES_PER_HEX_COLUMN;
- if (addressOfLine <= GetEditedSize ())
- {
- SetAttribute (addressAttribute);
- }
- else
- {
- SetAttribute (new Attribute (GetNormalColor ().Background.GetHighlightColor (), addressAttribute.Background));
- }
- var address = $"{addressOfLine:x8}";
- AddStr ($"{address.Substring (8 - AddressWidth)}");
- SetAttribute (GetNormalColor ());
- if (AddressWidth > 0)
- {
- AddStr (" ");
- }
- for (var block = 0; block < nBlocks; block++)
- {
- for (var b = 0; b < NUM_BYTES_PER_HEX_COLUMN; b++)
- {
- int offset = line * nBlocks * NUM_BYTES_PER_HEX_COLUMN + block * NUM_BYTES_PER_HEX_COLUMN + b;
- byte value = GetData (data, offset, out bool edited);
- if (offset + addressOfFirstLine == Address)
- {
- // Selected
- SetAttribute (_leftSideHasFocus ? editingAttribute : (edited ? editedAttribute : selectedAttribute));
- }
- else
- {
- SetAttribute (edited ? editedAttribute : GetNormalColor ());
- }
- AddStr (offset >= bytesRead && !edited ? " " : $"{value:x2}");
- SetAttribute (GetNormalColor ());
- AddRune (_spaceCharRune);
- }
- AddStr (block + 1 == nBlocks ? " " : $"{_columnSeparatorRune} ");
- }
- for (var byteIndex = 0; byteIndex < nBlocks * NUM_BYTES_PER_HEX_COLUMN; byteIndex++)
- {
- int offset = line * nBlocks * NUM_BYTES_PER_HEX_COLUMN + byteIndex;
- byte b = GetData (data, offset, out bool edited);
- Rune c;
- var utf8BytesConsumed = 0;
- if (offset >= bytesRead && !edited)
- {
- c = _spaceCharRune;
- }
- else
- {
- switch (b)
- {
- //case < 32:
- // c = _periodCharRune;
- // break;
- case > 127:
- {
- var utf8 = GetData (data, offset, 4, out bool _);
- OperationStatus status = Rune.DecodeFromUtf8 (utf8, out c, out utf8BytesConsumed);
- while (status == OperationStatus.NeedMoreData)
- {
- status = Rune.DecodeFromUtf8 (utf8, out c, out utf8BytesConsumed);
- }
- break;
- }
- default:
- Rune.DecodeFromUtf8 (new (ref b), out c, out _);
- break;
- }
- }
- if (offset + Source.Position == Address)
- {
- // Selected
- SetAttribute (_leftSideHasFocus ? editingAttribute : (edited ? editedAttribute : selectedAttribute));
- }
- else
- {
- SetAttribute (edited ? editedAttribute : GetNormalColor ());
- }
- AddRune (c);
- for (var i = 1; i < utf8BytesConsumed; i++)
- {
- byteIndex++;
- AddRune (_periodCharRune);
- }
- }
- }
- return true;
- }
- /// <summary>Raises the <see cref="Edited"/> event.</summary>
- protected void RaiseEdited (HexViewEditEventArgs e)
- {
- OnEdited (e);
- Edited?.Invoke (this, e);
- }
- /// <summary>Event to be invoked when an edit is made on the <see cref="Stream"/>.</summary>
- public event EventHandler<HexViewEditEventArgs>? Edited;
- /// <summary>
- /// </summary>
- /// <param name="e"></param>
- protected virtual void OnEdited (HexViewEditEventArgs e) { }
- /// <summary>
- /// Call this when the position (see <see cref="GetPosition"/>) and <see cref="Address"/> have changed. Raises the
- /// <see cref="PositionChanged"/> event.
- /// </summary>
- protected void RaisePositionChanged ()
- {
- HexViewEventArgs args = new (Address, GetPosition (Address), BytesPerLine);
- OnPositionChanged (args);
- PositionChanged?.Invoke (this, args);
- }
- /// <summary>
- /// Called when the position (see <see cref="GetPosition"/>) and <see cref="Address"/> have changed.
- /// </summary>
- protected virtual void OnPositionChanged (HexViewEventArgs e) { }
- /// <summary>Raised when the position (see <see cref="GetPosition"/>) and <see cref="Address"/> have changed.</summary>
- public event EventHandler<HexViewEventArgs>? PositionChanged;
- /// <inheritdoc/>
- protected override bool OnKeyDownNotHandled (Key keyEvent)
- {
- if (!AllowEdits || _source is null)
- {
- return false;
- }
- if (keyEvent.IsAlt)
- {
- return false;
- }
- if (_leftSideHasFocus)
- {
- int value;
- var k = (char)keyEvent.KeyCode;
- if (!char.IsAsciiHexDigit ((char)keyEvent.KeyCode))
- {
- return false;
- }
- if (k is >= 'A' and <= 'F')
- {
- value = k - 'A' + 10;
- }
- else if (k is >= 'a' and <= 'f')
- {
- value = k - 'a' + 10;
- }
- else if (k is >= '0' and <= '9')
- {
- value = k - '0';
- }
- else
- {
- return false;
- }
- if (!_edits.TryGetValue (Address, out byte b))
- {
- _source.Position = Address;
- b = (byte)_source.ReadByte ();
- }
- if (_firstNibble)
- {
- _firstNibble = false;
- b = (byte)((b & 0xf) | (value << NUM_BYTES_PER_HEX_COLUMN));
- _edits [Address] = b;
- RaiseEdited (new (Address, _edits [Address]));
- }
- else
- {
- b = (byte)((b & 0xf0) | value);
- _edits [Address] = b;
- RaiseEdited (new (Address, _edits [Address]));
- MoveRight ();
- }
- return true;
- }
- keyEvent = keyEvent.NoAlt.NoCtrl;
- Rune r = keyEvent.AsRune;
- if (Rune.IsControl (r))
- {
- return false;
- }
- var utf8 = new byte [4];
- // If the rune is a wide char, encode as utf8
- if (r.TryEncodeToUtf8 (utf8, out int bytesWritten))
- {
- if (bytesWritten > 1)
- {
- bytesWritten = 4;
- }
- for (var utfIndex = 0; utfIndex < bytesWritten; utfIndex++)
- {
- _edits [Address] = utf8 [utfIndex];
- RaiseEdited (new (Address, _edits [Address]));
- MoveRight ();
- }
- }
- else
- {
- _edits [Address] = (byte)r.Value;
- RaiseEdited (new (Address, _edits [Address]));
- MoveRight ();
- }
- return true;
- }
- //
- // This is used to support editing of the buffer on a peer List<>,
- // the offset corresponds to an offset relative to DisplayStart, and
- // the buffer contains the contents of a Viewport of data, so the
- // offset is relative to the buffer.
- //
- //
- private byte GetData (byte [] buffer, int offset, out bool edited)
- {
- long pos = Viewport.Y * BytesPerLine + offset;
- if (_edits.TryGetValue (pos, out byte v))
- {
- edited = true;
- return v;
- }
- edited = false;
- return buffer [offset];
- }
- private byte [] GetData (byte [] buffer, int offset, int count, out bool edited)
- {
- var returnBytes = new byte [count];
- edited = false;
- long pos = Viewport.Y + offset;
- for (long i = pos; i < pos + count; i++)
- {
- if (_edits.TryGetValue (i, out byte v))
- {
- edited = true;
- returnBytes [i - pos] = v;
- }
- else
- {
- if (pos < buffer.Length - 1)
- {
- returnBytes [i - pos] = buffer [pos];
- }
- }
- }
- return returnBytes;
- }
- private void SetBytesPerLine ()
- {
- // Small buffers will just show the position, with the bsize field value (4 bytes)
- BytesPerLine = NUM_BYTES_PER_HEX_COLUMN;
- if (Viewport.Width - GetLeftSideStartColumn () >= HEX_COLUMN_WIDTH)
- {
- BytesPerLine = Math.Max (
- NUM_BYTES_PER_HEX_COLUMN,
- NUM_BYTES_PER_HEX_COLUMN * ((Viewport.Width - GetLeftSideStartColumn ()) / (HEX_COLUMN_WIDTH + NUM_BYTES_PER_HEX_COLUMN)));
- }
- }
- private bool MoveDown (int bytes)
- {
- if (Address + bytes < GetEditedSize ())
- {
- // We can move down lines cleanly (without extending stream)
- Address += bytes;
- }
- else if ((bytes == BytesPerLine * Viewport.Height && _source!.Length >= Viewport.Y * BytesPerLine + BytesPerLine * Viewport.Height)
- || (bytes <= BytesPerLine * Viewport.Height - BytesPerLine
- && _source!.Length <= Viewport.Y * BytesPerLine + BytesPerLine * Viewport.Height))
- {
- long p = Address;
- // This lets address go past the end of the stream one, enabling adding to the stream.
- while (p + BytesPerLine <= GetEditedSize ())
- {
- p += BytesPerLine;
- }
- Address = p;
- }
- return true;
- }
- private bool MoveEnd ()
- {
- // This lets address go past the end of the stream one, enabling adding to the stream.
- Address = GetEditedSize ();
- return true;
- }
- private bool MoveEndOfLine ()
- {
- // This lets address go past the end of the stream one, enabling adding to the stream.
- Address = Math.Min (Address / BytesPerLine * BytesPerLine + BytesPerLine - 1, GetEditedSize ());
- return true;
- }
- private bool MoveHome ()
- {
- Address = 0;
- return true;
- }
- private bool MoveLeft ()
- {
- if (_leftSideHasFocus)
- {
- if (!_firstNibble)
- {
- _firstNibble = true;
- return true;
- }
- _firstNibble = false;
- }
- if (Address == 0)
- {
- return true;
- }
- Address--;
- return true;
- }
- private bool MoveRight ()
- {
- if (_leftSideHasFocus)
- {
- if (_firstNibble)
- {
- _firstNibble = false;
- return true;
- }
- _firstNibble = true;
- }
- // This lets address go past the end of the stream one, enabling adding to the stream.
- if (Address < GetEditedSize ())
- {
- Address++;
- }
- return true;
- }
- private bool MoveLeftStart ()
- {
- Address = Address / BytesPerLine * BytesPerLine;
- return true;
- }
- private bool MoveUp (int bytes)
- {
- Address -= bytes;
- return true;
- }
- /// <inheritdoc />
- protected override bool OnAdvancingFocus (NavigationDirection direction, TabBehavior? behavior)
- {
- if (behavior is { } && behavior != TabStop)
- {
- return false;
- }
- if ((direction == NavigationDirection.Forward && _leftSideHasFocus)
- || (direction == NavigationDirection.Backward && !_leftSideHasFocus))
- {
- _leftSideHasFocus = !_leftSideHasFocus;
- _firstNibble = true;
- SetNeedsDraw ();
- return true;
- }
- return false;
- }
- /// <inheritdoc/>
- bool IDesignable.EnableForDesign ()
- {
- Source = new MemoryStream (Encoding.UTF8.GetBytes ("HexView data with wide codepoints: 𝔹Aℝ𝔽!"));
- return true;
- }
- }
|