123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904 |
- #define DRAW_CONTENT
- //#define BASE_DRAW_CONTENT
- using System;
- using System.Collections.Generic;
- using System.Data;
- using System.Globalization;
- using System.Linq;
- using System.Net.Http;
- using System.Text;
- using System.Text.Json;
- using System.Text.Unicode;
- using System.Threading.Tasks;
- using Terminal.Gui;
- using static Terminal.Gui.SpinnerStyle;
- using static Terminal.Gui.TableView;
- namespace UICatalog.Scenarios;
- /// <summary>
- /// This Scenario demonstrates building a custom control (a class deriving from View) that:
- /// - Provides a "Character Map" application (like Windows' charmap.exe).
- /// - Helps test unicode character rendering in Terminal.Gui
- /// - Illustrates how to use ScrollView to do infinite scrolling
- /// </summary>
- [ScenarioMetadata ("Character Map", "Unicode viewer demonstrating the ScrollView control.")]
- [ScenarioCategory ("Text and Formatting")]
- [ScenarioCategory ("Controls")]
- [ScenarioCategory ("ScrollView")]
- public class CharacterMap : Scenario {
- CharMap _charMap;
- public Label _errorLabel;
- TableView _categoryList;
- // Don't create a Window, just return the top-level view
- public override void Init ()
- {
- Application.Init ();
- Application.Top.ColorScheme = Colors.Base;
- }
- public override void Setup ()
- {
- _charMap = new CharMap () {
- X = 0,
- Y = 1,
- Height = Dim.Fill ()
- };
- Application.Top.Add (_charMap);
- var jumpLabel = new Label ("_Jump To Code Point:") {
- X = Pos.Right (_charMap) + 1,
- Y = Pos.Y (_charMap),
- HotKeySpecifier = (Rune)'_'
- };
- Application.Top.Add (jumpLabel);
- var jumpEdit = new TextField () {
- X = Pos.Right (jumpLabel) + 1, Y = Pos.Y (_charMap), Width = 10, Caption = "e.g. 01BE3"
- };
- Application.Top.Add (jumpEdit);
- _errorLabel = new Label ("err") { X = Pos.Right (jumpEdit) + 1, Y = Pos.Y (_charMap), ColorScheme = Colors.ColorSchemes ["error"] };
- Application.Top.Add (_errorLabel);
- jumpEdit.TextChanged += JumpEdit_TextChanged;
- _categoryList = new TableView () {
- X = Pos.Right (_charMap),
- Y = Pos.Bottom (jumpLabel),
- Height = Dim.Fill ()
- };
- _categoryList.FullRowSelect = true;
- //jumpList.Style.ShowHeaders = false;
- //jumpList.Style.ShowHorizontalHeaderOverline = false;
- //jumpList.Style.ShowHorizontalHeaderUnderline = false;
- _categoryList.Style.ShowHorizontalBottomline = true;
- //jumpList.Style.ShowVerticalCellLines = false;
- //jumpList.Style.ShowVerticalHeaderLines = false;
- _categoryList.Style.AlwaysShowHeaders = true;
- bool isDescending = false;
- _categoryList.Table = CreateCategoryTable (0, isDescending);
- // if user clicks the mouse in TableView
- _categoryList.MouseClick += (s, e) => {
- _categoryList.ScreenToCell (e.MouseEvent.X, e.MouseEvent.Y, out int? clickedCol);
- if (clickedCol != null && e.MouseEvent.Flags.HasFlag (MouseFlags.Button1Clicked)) {
- var table = (EnumerableTableSource<UnicodeRange>)_categoryList.Table;
- string prevSelection = table.Data.ElementAt (_categoryList.SelectedRow).Category;
- isDescending = !isDescending;
- _categoryList.Table = CreateCategoryTable (clickedCol.Value, isDescending);
- table = (EnumerableTableSource<UnicodeRange>)_categoryList.Table;
- _categoryList.SelectedRow = table.Data
- .Select ((item, index) => new { item, index })
- .FirstOrDefault (x => x.item.Category == prevSelection)?.index ?? -1;
- }
- };
- int longestName = UnicodeRange.Ranges.Max (r => r.Category.GetColumns ());
- _categoryList.Style.ColumnStyles.Add (0, new ColumnStyle () { MaxWidth = longestName, MinWidth = longestName, MinAcceptableWidth = longestName });
- _categoryList.Style.ColumnStyles.Add (1, new ColumnStyle () { MaxWidth = 1, MinWidth = 6 });
- _categoryList.Style.ColumnStyles.Add (2, new ColumnStyle () { MaxWidth = 1, MinWidth = 6 });
- _categoryList.Width = _categoryList.Style.ColumnStyles.Sum (c => c.Value.MinWidth) + 4;
- _categoryList.SelectedCellChanged += (s, args) => {
- var table = (EnumerableTableSource<UnicodeRange>)_categoryList.Table;
- _charMap.StartCodePoint = table.Data.ToArray () [args.NewRow].Start;
- };
- Application.Top.Add (_categoryList);
- _charMap.SelectedCodePoint = 0;
- //jumpList.Refresh ();
- _charMap.SetFocus ();
- _charMap.Width = Dim.Fill () - _categoryList.Width;
- var menu = new MenuBar (new MenuBarItem [] {
- new ("_File", new MenuItem [] {
- new ("_Quit", $"{Application.QuitKey}", () => Application.RequestStop ())
- }),
- new ("_Options", new MenuItem [] {
- CreateMenuShowWidth ()
- })
- });
- Application.Top.Add (menu);
- //_charMap.Hover += (s, a) => {
- // _errorLabel.Text = $"U+{a.Item:x5} {(Rune)a.Item}";
- //};
- }
- MenuItem CreateMenuShowWidth ()
- {
- var item = new MenuItem {
- Title = "_Show Glyph Width"
- };
- item.CheckType |= MenuItemCheckStyle.Checked;
- item.Checked = _charMap?.ShowGlyphWidths;
- item.Action += () => {
- _charMap.ShowGlyphWidths = (bool)(item.Checked = !item.Checked);
- };
- return item;
- }
- EnumerableTableSource<UnicodeRange> CreateCategoryTable (int sortByColumn, bool descending)
- {
- Func<UnicodeRange, object> orderBy;
- string categorySort = string.Empty;
- string startSort = string.Empty;
- string endSort = string.Empty;
- string sortIndicator = descending ? CM.Glyphs.DownArrow.ToString () : CM.Glyphs.UpArrow.ToString ();
- switch (sortByColumn) {
- case 0:
- orderBy = r => r.Category;
- categorySort = sortIndicator;
- break;
- case 1:
- orderBy = r => r.Start;
- startSort = sortIndicator;
- break;
- case 2:
- orderBy = r => r.End;
- endSort = sortIndicator;
- break;
- default:
- throw new ArgumentException ("Invalid column number.");
- }
- var sortedRanges = descending ?
- UnicodeRange.Ranges.OrderByDescending (orderBy) :
- UnicodeRange.Ranges.OrderBy (orderBy);
- return new EnumerableTableSource<UnicodeRange> (sortedRanges, new Dictionary<string, Func<UnicodeRange, object>> () {
- { $"Category{categorySort}", s => s.Category },
- { $"Start{startSort}", s => $"{s.Start:x5}" },
- { $"End{endSort}", s => $"{s.End:x5}" }
- });
- }
- void JumpEdit_TextChanged (object sender, TextChangedEventArgs e)
- {
- var jumpEdit = sender as TextField;
- if (jumpEdit.Text.Length == 0) {
- return;
- }
- uint result = 0;
- if (jumpEdit.Text.StartsWith ("U+", StringComparison.OrdinalIgnoreCase) || jumpEdit.Text.StartsWith ("\\u")) {
- try {
- result = uint.Parse (jumpEdit.Text [2..^0], NumberStyles.HexNumber);
- } catch (FormatException) {
- _errorLabel.Text = $"Invalid hex value";
- return;
- }
- } else if (jumpEdit.Text.StartsWith ("0", StringComparison.OrdinalIgnoreCase) || jumpEdit.Text.StartsWith ("\\u")) {
- try {
- result = uint.Parse (jumpEdit.Text, NumberStyles.HexNumber);
- } catch (FormatException) {
- _errorLabel.Text = $"Invalid hex value";
- return;
- }
- } else {
- try {
- result = uint.Parse (jumpEdit.Text, NumberStyles.Integer);
- } catch (FormatException) {
- _errorLabel.Text = $"Invalid value";
- return;
- }
- }
- if (result > RuneExtensions.MaxUnicodeCodePoint) {
- _errorLabel.Text = $"Beyond maximum codepoint";
- return;
- }
- _errorLabel.Text = $"U+{result:x5}";
- var table = (EnumerableTableSource<UnicodeRange>)_categoryList.Table;
- _categoryList.SelectedRow = table.Data
- .Select ((item, index) => new { item, index })
- .FirstOrDefault (x => x.item.Start <= result && x.item.End >= result)?.index ?? -1;
- _categoryList.EnsureSelectedCellIsVisible ();
- // Ensure the typed glyph is selected
- _charMap.SelectedCodePoint = (int)result;
- }
- }
- class CharMap : ScrollView {
- /// <summary>
- /// Specifies the starting offset for the character map. The default is 0x2500
- /// which is the Box Drawing characters.
- /// </summary>
- public int StartCodePoint {
- get => _start;
- set {
- _start = value;
- SelectedCodePoint = value;
- SetNeedsDisplay ();
- }
- }
- public event EventHandler<ListViewItemEventArgs> SelectedCodePointChanged;
- /// <summary>
- /// Specifies the starting offset for the character map. The default is 0x2500
- /// which is the Box Drawing characters.
- /// </summary>
- public int SelectedCodePoint {
- get => _selected;
- set {
- _selected = value;
- int row = SelectedCodePoint / 16 * _rowHeight;
- int col = SelectedCodePoint % 16 * COLUMN_WIDTH;
- int height = Bounds.Height - (ShowHorizontalScrollIndicator ? 2 : 1);
- if (row + ContentOffset.Y < 0) {
- // Moving up.
- ContentOffset = new Point (ContentOffset.X, row);
- } else if (row + ContentOffset.Y >= height) {
- // Moving down.
- ContentOffset = new Point (ContentOffset.X, Math.Min (row, row - height + _rowHeight));
- }
- int width = Bounds.Width / COLUMN_WIDTH * COLUMN_WIDTH - (ShowVerticalScrollIndicator ? RowLabelWidth + 1 : RowLabelWidth);
- if (col + ContentOffset.X < 0) {
- // Moving left.
- ContentOffset = new Point (col, ContentOffset.Y);
- } else if (col + ContentOffset.X >= width) {
- // Moving right.
- ContentOffset = new Point (Math.Min (col, col - width + COLUMN_WIDTH), ContentOffset.Y);
- }
- SetNeedsDisplay ();
- SelectedCodePointChanged?.Invoke (this, new ListViewItemEventArgs (SelectedCodePoint, null));
- }
- }
- public event EventHandler<ListViewItemEventArgs> Hover;
- /// <summary>
- /// Gets the coordinates of the Cursor based on the SelectedCodePoint in screen coordinates
- /// </summary>
- public Point Cursor {
- get {
- int row = SelectedCodePoint / 16 * _rowHeight + ContentOffset.Y + 1;
- int col = SelectedCodePoint % 16 * COLUMN_WIDTH + ContentOffset.X + RowLabelWidth + 1; // + 1 for padding
- return new Point (col, row);
- }
- set => throw new NotImplementedException ();
- }
- public override void PositionCursor ()
- {
- if (HasFocus &&
- Cursor.X >= RowLabelWidth &&
- Cursor.X < Bounds.Width - (ShowVerticalScrollIndicator ? 1 : 0) &&
- Cursor.Y > 0 &&
- Cursor.Y < Bounds.Height - (ShowHorizontalScrollIndicator ? 1 : 0)) {
- Driver.SetCursorVisibility (CursorVisibility.Default);
- Move (Cursor.X, Cursor.Y);
- } else {
- Driver.SetCursorVisibility (CursorVisibility.Invisible);
- }
- }
- public bool ShowGlyphWidths {
- get => _rowHeight == 2;
- set {
- _rowHeight = value ? 2 : 1;
- SetNeedsDisplay ();
- }
- }
- int _start = 0;
- int _selected = 0;
- const int COLUMN_WIDTH = 3;
- int _rowHeight = 1;
- public static int MaxCodePoint => 0x10FFFF;
- static int RowLabelWidth => $"U+{MaxCodePoint:x5}".Length + 1;
- static int RowWidth => RowLabelWidth + COLUMN_WIDTH * 16;
- public CharMap ()
- {
- ColorScheme = Colors.Dialog;
- CanFocus = true;
- ContentSize = new Size (RowWidth, (int)((MaxCodePoint / 16 + (ShowHorizontalScrollIndicator ? 2 : 1)) * _rowHeight));
- AddCommand (Command.ScrollUp, () => {
- if (SelectedCodePoint >= 16) {
- SelectedCodePoint -= 16;
- }
- return true;
- });
- AddCommand (Command.ScrollDown, () => {
- if (SelectedCodePoint < MaxCodePoint - 16) {
- SelectedCodePoint += 16;
- }
- return true;
- });
- AddCommand (Command.ScrollLeft, () => {
- if (SelectedCodePoint > 0) {
- SelectedCodePoint--;
- }
- return true;
- });
- AddCommand (Command.ScrollRight, () => {
- if (SelectedCodePoint < MaxCodePoint) {
- SelectedCodePoint++;
- }
- return true;
- });
- AddCommand (Command.PageUp, () => {
- int page = (Bounds.Height / _rowHeight - 1) * 16;
- SelectedCodePoint -= Math.Min (page, SelectedCodePoint);
- return true;
- });
- AddCommand (Command.PageDown, () => {
- int page = (Bounds.Height / _rowHeight - 1) * 16;
- SelectedCodePoint += Math.Min (page, MaxCodePoint - SelectedCodePoint);
- return true;
- });
- AddCommand (Command.TopHome, () => {
- SelectedCodePoint = 0;
- return true;
- });
- AddCommand (Command.BottomEnd, () => {
- SelectedCodePoint = MaxCodePoint;
- return true;
- });
- KeyBindings.Add (Key.Enter, Command.Accept);
- AddCommand (Command.Accept, () => {
- ShowDetails ();
- return true;
- });
- MouseClick += Handle_MouseClick;
- }
- void CopyCodePoint () => Clipboard.Contents = $"U+{SelectedCodePoint:x5}";
- void CopyGlyph () => Clipboard.Contents = $"{new Rune (SelectedCodePoint)}";
- public override void OnDrawContent (Rect contentArea) =>
- //if (ShowHorizontalScrollIndicator && ContentSize.Height < (int)(MaxCodePoint / 16 + 2)) {
- // //ContentSize = new Size (CharMap.RowWidth, (int)(MaxCodePoint / 16 + 2));
- // //ContentSize = new Size (CharMap.RowWidth, (int)(MaxCodePoint / 16) * _rowHeight + 2);
- // var width = (Bounds.Width / COLUMN_WIDTH * COLUMN_WIDTH) - (ShowVerticalScrollIndicator ? RowLabelWidth + 1 : RowLabelWidth);
- // if (Cursor.X + ContentOffset.X >= width) {
- // // Snap to the selected glyph.
- // ContentOffset = new Point (
- // Math.Min (Cursor.X, Cursor.X - width + COLUMN_WIDTH),
- // ContentOffset.Y == -ContentSize.Height + Bounds.Height ? ContentOffset.Y - 1 : ContentOffset.Y);
- // } else {
- // ContentOffset = new Point (
- // ContentOffset.X - Cursor.X,
- // ContentOffset.Y == -ContentSize.Height + Bounds.Height ? ContentOffset.Y - 1 : ContentOffset.Y);
- // }
- //} else if (!ShowHorizontalScrollIndicator && ContentSize.Height > (int)(MaxCodePoint / 16 + 1)) {
- // //ContentSize = new Size (CharMap.RowWidth, (int)(MaxCodePoint / 16 + 1));
- // // Snap 1st column into view if it's been scrolled horizontally
- // ContentOffset = new Point (0, ContentOffset.Y < -ContentSize.Height + Bounds.Height ? ContentOffset.Y - 1 : ContentOffset.Y);
- //}
- base.OnDrawContent (contentArea);
- //public void CharMap_DrawContent (object s, DrawEventArgs a)
- public override void OnDrawContentComplete (Rect contentArea)
- {
- if (contentArea.Height == 0 || contentArea.Width == 0) {
- return;
- }
- var viewport = new Rect (ContentOffset,
- new Size (Math.Max (Bounds.Width - (ShowVerticalScrollIndicator ? 1 : 0), 0),
- Math.Max (Bounds.Height - (ShowHorizontalScrollIndicator ? 1 : 0), 0)));
- var oldClip = ClipToBounds ();
- if (ShowHorizontalScrollIndicator) {
- // ClipToBounds doesn't know about the scroll indicators, so if off, subtract one from height
- Driver.Clip = new Rect (Driver.Clip.Location, new Size (Driver.Clip.Width, Driver.Clip.Height - 1));
- }
- if (ShowVerticalScrollIndicator) {
- // ClipToBounds doesn't know about the scroll indicators, so if off, subtract one from width
- Driver.Clip = new Rect (Driver.Clip.Location, new Size (Driver.Clip.Width - 1, Driver.Clip.Height));
- }
- int cursorCol = Cursor.X - ContentOffset.X - RowLabelWidth - 1;
- int cursorRow = Cursor.Y - ContentOffset.Y - 1;
- Driver.SetAttribute (GetHotNormalColor ());
- Move (0, 0);
- Driver.AddStr (new string (' ', RowLabelWidth + 1));
- for (int hexDigit = 0; hexDigit < 16; hexDigit++) {
- int x = ContentOffset.X + RowLabelWidth + hexDigit * COLUMN_WIDTH;
- if (x > RowLabelWidth - 2) {
- Move (x, 0);
- Driver.SetAttribute (GetHotNormalColor ());
- Driver.AddStr (" ");
- Driver.SetAttribute (HasFocus && cursorCol + ContentOffset.X + RowLabelWidth == x ? ColorScheme.HotFocus : GetHotNormalColor ());
- Driver.AddStr ($"{hexDigit:x}");
- Driver.SetAttribute (GetHotNormalColor ());
- Driver.AddStr (" ");
- }
- }
- int firstColumnX = viewport.X + RowLabelWidth;
- for (int y = 1; y < Bounds.Height; y++) {
- // What row is this?
- int row = (y - ContentOffset.Y - 1) / _rowHeight;
- int val = row * 16;
- if (val > MaxCodePoint) {
- continue;
- }
- Move (firstColumnX + COLUMN_WIDTH, y);
- Driver.SetAttribute (GetNormalColor ());
- for (int col = 0; col < 16; col++) {
- int x = firstColumnX + COLUMN_WIDTH * col + 1;
- Move (x, y);
- if (cursorRow + ContentOffset.Y + 1 == y && cursorCol + ContentOffset.X + firstColumnX + 1 == x && !HasFocus) {
- Driver.SetAttribute (GetFocusColor ());
- }
- int scalar = val + col;
- var rune = (Rune)'?';
- if (Rune.IsValid (scalar)) {
- rune = new Rune (scalar);
- }
- int width = rune.GetColumns ();
- // are we at first row of the row?
- if (!ShowGlyphWidths || (y - ContentOffset.Y) % _rowHeight > 0) {
- if (width > 0) {
- Driver.AddRune (rune);
- } else {
- if (rune.IsCombiningMark ()) {
- // This is a hack to work around the fact that combining marks
- // a) can't be rendered on their own
- // b) that don't normalize are not properly supported in
- // any known terminal (esp Windows/AtlasEngine).
- // See Issue #2616
- var sb = new StringBuilder ();
- sb.Append ('a');
- sb.Append (rune);
- // Try normalizing after combining with 'a'. If it normalizes, at least
- // it'll show on the 'a'. If not, just show the replacement char.
- string normal = sb.ToString ().Normalize (NormalizationForm.FormC);
- if (normal.Length == 1) {
- Driver.AddRune (normal [0]);
- } else {
- Driver.AddRune (Rune.ReplacementChar);
- }
- }
- }
- } else {
- Driver.SetAttribute (ColorScheme.HotNormal);
- Driver.AddStr ($"{width}");
- }
- if (cursorRow + ContentOffset.Y + 1 == y && cursorCol + ContentOffset.X + firstColumnX + 1 == x && !HasFocus) {
- Driver.SetAttribute (GetNormalColor ());
- }
- }
- Move (0, y);
- Driver.SetAttribute (HasFocus && cursorRow + ContentOffset.Y + 1 == y ? ColorScheme.HotFocus : ColorScheme.HotNormal);
- if (!ShowGlyphWidths || (y - ContentOffset.Y) % _rowHeight > 0) {
- Driver.AddStr ($"U+{val / 16:x5}_ ");
- } else {
- Driver.AddStr (new string (' ', RowLabelWidth));
- }
- }
- Driver.Clip = oldClip;
- }
- ContextMenu _contextMenu = new ();
- void Handle_MouseClick (object sender, MouseEventEventArgs args)
- {
- var me = args.MouseEvent;
- if (me.Flags != MouseFlags.ReportMousePosition && me.Flags != MouseFlags.Button1Clicked &&
- me.Flags != MouseFlags.Button1DoubleClicked) {
- return;
- }
- if (me.Y == 0) {
- me.Y = Cursor.Y;
- }
- if (me.Y > 0) { }
- if (me.X < RowLabelWidth || me.X > RowLabelWidth + 16 * COLUMN_WIDTH - 1) {
- me.X = Cursor.X;
- }
- int row = (me.Y - 1 - ContentOffset.Y) / _rowHeight; // -1 for header
- int col = (me.X - RowLabelWidth - ContentOffset.X) / COLUMN_WIDTH;
- if (col > 15) {
- col = 15;
- }
- int val = row * 16 + col;
- if (val > MaxCodePoint) {
- return;
- }
- if (me.Flags == MouseFlags.ReportMousePosition) {
- Hover?.Invoke (this, new ListViewItemEventArgs (val, null));
- }
- if (me.Flags == MouseFlags.Button1Clicked) {
- SelectedCodePoint = val;
- return;
- }
- if (me.Flags == MouseFlags.Button1DoubleClicked) {
- SelectedCodePoint = val;
- ShowDetails ();
- return;
- }
- if (me.Flags == _contextMenu.MouseFlags) {
- SelectedCodePoint = val;
- _contextMenu = new ContextMenu (me.X + 1, me.Y + 1,
- new MenuBarItem (new MenuItem [] {
- new ("_Copy Glyph", "", () => CopyGlyph (), null, null, (KeyCode)Key.C.WithCtrl),
- new ("Copy Code _Point", "", () => CopyCodePoint (), null, null, (KeyCode)Key.C.WithCtrl.WithShift)
- }) { }
- );
- _contextMenu.Show ();
- }
- }
- public static string ToCamelCase (string str)
- {
- if (string.IsNullOrEmpty (str)) {
- return str;
- }
- var textInfo = new CultureInfo ("en-US", false).TextInfo;
- str = textInfo.ToLower (str);
- str = textInfo.ToTitleCase (str);
- return str;
- }
- void ShowDetails ()
- {
- var client = new UcdApiClient ();
- string decResponse = string.Empty;
- var waitIndicator = new Dialog (new Button ("Cancel")) {
- Title = "Getting Code Point Information",
- X = Pos.Center (),
- Y = Pos.Center (),
- Height = 7,
- Width = 50
- };
- var errorLabel = new Label () {
- Text = UcdApiClient.BaseUrl,
- AutoSize = false,
- X = 0,
- Y = 1,
- Width = Dim.Fill (),
- Height = Dim.Fill (1),
- TextAlignment = TextAlignment.Centered
- };
- var spinner = new SpinnerView () {
- X = Pos.Center (),
- Y = Pos.Center (),
- Style = new Aesthetic ()
- };
- spinner.AutoSpin = true;
- waitIndicator.Add (errorLabel);
- waitIndicator.Add (spinner);
- waitIndicator.Ready += async (s, a) => {
- try {
- decResponse = await client.GetCodepointDec ((int)SelectedCodePoint);
- } catch (HttpRequestException e) {
- (s as Dialog).Text = e.Message;
- Application.Invoke (() => {
- spinner.Visible = false;
- errorLabel.Text = e.Message;
- errorLabel.ColorScheme = Colors.ColorSchemes ["Error"];
- errorLabel.Visible = true;
- });
- }
- (s as Dialog)?.RequestStop ();
- };
- Application.Run (waitIndicator);
- if (!string.IsNullOrEmpty (decResponse)) {
- string name = string.Empty;
- using (var document = JsonDocument.Parse (decResponse)) {
- var root = document.RootElement;
- // Get a property by name and output its value
- if (root.TryGetProperty ("name", out var nameElement)) {
- name = nameElement.GetString ();
- }
- //// Navigate to a nested property and output its value
- //if (root.TryGetProperty ("property3", out JsonElement property3Element)
- //&& property3Element.TryGetProperty ("nestedProperty", out JsonElement nestedPropertyElement)) {
- // Console.WriteLine (nestedPropertyElement.GetString ());
- //}
- decResponse = JsonSerializer.Serialize (document.RootElement, new
- JsonSerializerOptions {
- WriteIndented = true
- });
- }
- string title = $"{ToCamelCase (name)} - {new Rune (SelectedCodePoint)} U+{SelectedCodePoint:x5}";
- var copyGlyph = new Button ("Copy _Glyph");
- var copyCP = new Button ("Copy Code _Point");
- var cancel = new Button ("Cancel");
- var dlg = new Dialog (copyGlyph, copyCP, cancel) {
- Title = title
- };
- copyGlyph.Clicked += (s, a) => {
- CopyGlyph ();
- dlg.RequestStop ();
- };
- copyCP.Clicked += (s, a) => {
- CopyCodePoint ();
- dlg.RequestStop ();
- };
- cancel.Clicked += (s, a) => dlg.RequestStop ();
- var rune = (Rune)SelectedCodePoint;
- var label = new Label () {
- Text = "IsAscii: ",
- X = 0,
- Y = 0
- };
- dlg.Add (label);
- label = new Label () {
- Text = $"{rune.IsAscii}",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = ", Bmp: ",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = $"{rune.IsBmp}",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = ", CombiningMark: ",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = $"{rune.IsCombiningMark ()}",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = ", SurrogatePair: ",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = $"{rune.IsSurrogatePair ()}",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = ", Plane: ",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = $"{rune.Plane}",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = "Columns: ",
- X = 0,
- Y = Pos.Bottom (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = $"{rune.GetColumns ()}",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = ", Utf16SequenceLength: ",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = $"{rune.Utf16SequenceLength}",
- X = Pos.Right (label),
- Y = Pos.Top (label)
- };
- dlg.Add (label);
- label = new Label () {
- Text = $"Code Point Information from {UcdApiClient.BaseUrl}codepoint/dec/{SelectedCodePoint}:",
- X = 0,
- Y = Pos.Bottom (label)
- };
- dlg.Add (label);
- var json = new TextView () {
- X = 0,
- Y = Pos.Bottom (label),
- Width = Dim.Fill (),
- Height = Dim.Fill (2),
- ReadOnly = true,
- Text = decResponse
- };
- dlg.Add (json);
- Application.Run (dlg);
- } else {
- MessageBox.ErrorQuery ("Code Point API", $"{UcdApiClient.BaseUrl}codepoint/dec/{SelectedCodePoint} did not return a result for\r\n {new Rune (SelectedCodePoint)} U+{SelectedCodePoint:x5}.", "Ok");
- }
- // BUGBUG: This is a workaround for some weird ScrollView related mouse grab bug
- Application.GrabMouse (this);
- }
- public override bool OnEnter (View view)
- {
- if (IsInitialized) {
- Application.Driver.SetCursorVisibility (CursorVisibility.Default);
- }
- return base.OnEnter (view);
- }
- public override bool OnLeave (View view)
- {
- Driver.SetCursorVisibility (CursorVisibility.Invisible);
- return base.OnLeave (view);
- }
- }
- public class UcdApiClient {
- static readonly HttpClient httpClient = new ();
- public const string BaseUrl = "https://ucdapi.org/unicode/latest/";
- public async Task<string> GetCodepointHex (string hex)
- {
- var response = await httpClient.GetAsync ($"{BaseUrl}codepoint/hex/{hex}");
- response.EnsureSuccessStatusCode ();
- return await response.Content.ReadAsStringAsync ();
- }
- public async Task<string> GetCodepointDec (int dec)
- {
- var response = await httpClient.GetAsync ($"{BaseUrl}codepoint/dec/{dec}");
- response.EnsureSuccessStatusCode ();
- return await response.Content.ReadAsStringAsync ();
- }
- public async Task<string> GetChars (string chars)
- {
- var response = await httpClient.GetAsync ($"{BaseUrl}chars/{Uri.EscapeDataString (chars)}");
- response.EnsureSuccessStatusCode ();
- return await response.Content.ReadAsStringAsync ();
- }
- public async Task<string> GetCharsName (string chars)
- {
- var response = await httpClient.GetAsync ($"{BaseUrl}chars/{Uri.EscapeDataString (chars)}/name");
- response.EnsureSuccessStatusCode ();
- return await response.Content.ReadAsStringAsync ();
- }
- }
- class UnicodeRange {
- public int Start;
- public int End;
- public string Category;
- public UnicodeRange (int start, int end, string category)
- {
- Start = start;
- End = end;
- Category = category;
- }
- public static List<UnicodeRange> GetRanges ()
- {
- var ranges = from r in typeof (UnicodeRanges).GetProperties (System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.Public)
- let urange = r.GetValue (null) as System.Text.Unicode.UnicodeRange
- let name = string.IsNullOrEmpty (r.Name) ? $"U+{urange.FirstCodePoint:x5}-U+{urange.FirstCodePoint + urange.Length:x5}" : r.Name
- where name != "None" && name != "All"
- select new UnicodeRange (urange.FirstCodePoint, urange.FirstCodePoint + urange.Length, name);
- // .NET 8.0 only supports BMP in UnicodeRanges: https://learn.microsoft.com/en-us/dotnet/api/system.text.unicode.unicoderanges?view=net-8.0
- var nonBmpRanges = new List<UnicodeRange> {
- new (0x1F130, 0x1F149, "Squared Latin Capital Letters"),
- new (0x12400, 0x1240f, "Cuneiform Numbers and Punctuation"),
- new (0x10000, 0x1007F, "Linear B Syllabary"),
- new (0x10080, 0x100FF, "Linear B Ideograms"),
- new (0x10100, 0x1013F, "Aegean Numbers"),
- new (0x10300, 0x1032F, "Old Italic"),
- new (0x10330, 0x1034F, "Gothic"),
- new (0x10380, 0x1039F, "Ugaritic"),
- new (0x10400, 0x1044F, "Deseret"),
- new (0x10450, 0x1047F, "Shavian"),
- new (0x10480, 0x104AF, "Osmanya"),
- new (0x10800, 0x1083F, "Cypriot Syllabary"),
- new (0x1D000, 0x1D0FF, "Byzantine Musical Symbols"),
- new (0x1D100, 0x1D1FF, "Musical Symbols"),
- new (0x1D300, 0x1D35F, "Tai Xuan Jing Symbols"),
- new (0x1D400, 0x1D7FF, "Mathematical Alphanumeric Symbols"),
- new (0x1F600, 0x1F532, "Emojis Symbols"),
- new (0x20000, 0x2A6DF, "CJK Unified Ideographs Extension B"),
- new (0x2F800, 0x2FA1F, "CJK Compatibility Ideographs Supplement"),
- new (0xE0000, 0xE007F, "Tags")
- };
- return ranges.Concat (nonBmpRanges).OrderBy (r => r.Category).ToList ();
- }
- public static List<UnicodeRange> Ranges = GetRanges ();
- }
|