123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666 |
- using NStack;
- using System;
- using System.Collections;
- using System.Collections.Generic;
- using System.ComponentModel;
- using System.Reflection;
- using System.Runtime.CompilerServices;
- using System.Text;
- using Terminal.Gui;
- namespace UICatalog.Scenarios {
- [ScenarioMetadata (Name: "Dynamic StatusBar", Description: "Demonstrates how to add and remove a StatusBar and change items dynamically.")]
- [ScenarioCategory ("Top Level Windows")]
- public class DynamicStatusBar : Scenario {
- public override void Init (Toplevel top, ColorScheme colorScheme)
- {
- Application.Init ();
- Top = Application.Top;
- Top.Add (new DynamicStatusBarSample ($"CTRL-Q to Close - Scenario: {GetName ()}"));
- }
- public class DynamicStatusItemList {
- public ustring Title { get; set; }
- public StatusItem StatusItem { get; set; }
- public DynamicStatusItemList () { }
- public DynamicStatusItemList (ustring title, StatusItem statusItem)
- {
- Title = title;
- StatusItem = statusItem;
- }
- public override string ToString () => $"{Title}, {StatusItem}";
- }
- public class DynamicStatusItem {
- public ustring title = "New";
- public ustring action = "";
- public ustring shortcut;
- public DynamicStatusItem () { }
- public DynamicStatusItem (ustring title)
- {
- this.title = title;
- }
- public DynamicStatusItem (ustring title, ustring action, ustring shortcut = null)
- {
- this.title = title;
- this.action = action;
- this.shortcut = shortcut;
- }
- }
- public class DynamicStatusBarSample : Window {
- StatusBar _statusBar;
- StatusItem _currentStatusItem;
- int _currentSelectedStatusBar = -1;
- StatusItem _currentEditStatusItem;
- ListView _lstItems;
- public DynamicStatusItemModel DataContext { get; set; }
- public DynamicStatusBarSample (ustring title) : base (title)
- {
- DataContext = new DynamicStatusItemModel ();
- var _frmDelimiter = new FrameView ("Shortcut Delimiter:") {
- X = Pos.Center (),
- Y = 0,
- Width = 25,
- Height = 4
- };
- var _txtDelimiter = new TextField (StatusBar.ShortcutDelimiter.ToString ()) {
- X = Pos.Center (),
- Width = 2,
- };
- _txtDelimiter.TextChanged += (_) => StatusBar.ShortcutDelimiter = _txtDelimiter.Text;
- _frmDelimiter.Add (_txtDelimiter);
- Add (_frmDelimiter);
- var _frmStatusBar = new FrameView ("Items:") {
- Y = 5,
- Width = Dim.Percent (50),
- Height = Dim.Fill (2)
- };
- var _btnAddStatusBar = new Button ("Add a StatusBar") {
- Y = 1,
- };
- _frmStatusBar.Add (_btnAddStatusBar);
- var _btnRemoveStatusBar = new Button ("Remove a StatusBar") {
- Y = 1
- };
- _btnRemoveStatusBar.X = Pos.AnchorEnd () - (Pos.Right (_btnRemoveStatusBar) - Pos.Left (_btnRemoveStatusBar));
- _frmStatusBar.Add (_btnRemoveStatusBar);
- var _btnAdd = new Button (" Add ") {
- Y = Pos.Top (_btnRemoveStatusBar) + 2,
- };
- _btnAdd.X = Pos.AnchorEnd () - (Pos.Right (_btnAdd) - Pos.Left (_btnAdd));
- _frmStatusBar.Add (_btnAdd);
- _lstItems = new ListView (new List<DynamicStatusItemList> ()) {
- ColorScheme = Colors.Dialog,
- Y = Pos.Top (_btnAddStatusBar) + 2,
- Width = Dim.Fill () - Dim.Width (_btnAdd) - 1,
- Height = Dim.Fill (),
- };
- _frmStatusBar.Add (_lstItems);
- var _btnRemove = new Button ("Remove") {
- X = Pos.Left (_btnAdd),
- Y = Pos.Top (_btnAdd) + 1
- };
- _frmStatusBar.Add (_btnRemove);
- var _btnUp = new Button ("^") {
- X = Pos.Right (_lstItems) + 2,
- Y = Pos.Top (_btnRemove) + 2
- };
- _frmStatusBar.Add (_btnUp);
- var _btnDown = new Button ("v") {
- X = Pos.Right (_lstItems) + 2,
- Y = Pos.Top (_btnUp) + 1
- };
- _frmStatusBar.Add (_btnDown);
- Add (_frmStatusBar);
- var _frmStatusBarDetails = new DynamicStatusBarDetails ("StatusBar Item Details:") {
- X = Pos.Right (_frmStatusBar),
- Y = Pos.Top (_frmStatusBar),
- Width = Dim.Fill (),
- Height = Dim.Fill (4)
- };
- Add (_frmStatusBarDetails);
- _btnUp.Clicked += () => {
- var i = _lstItems.SelectedItem;
- var statusItem = DataContext.Items.Count > 0 ? DataContext.Items [i].StatusItem : null;
- if (statusItem != null) {
- var items = _statusBar.Items;
- if (i > 0) {
- items [i] = items [i - 1];
- items [i - 1] = statusItem;
- DataContext.Items [i] = DataContext.Items [i - 1];
- DataContext.Items [i - 1] = new DynamicStatusItemList (statusItem.Title, statusItem);
- _lstItems.SelectedItem = i - 1;
- _statusBar.SetNeedsDisplay ();
- }
- }
- };
- _btnDown.Clicked += () => {
- var i = _lstItems.SelectedItem;
- var statusItem = DataContext.Items.Count > 0 ? DataContext.Items [i].StatusItem : null;
- if (statusItem != null) {
- var items = _statusBar.Items;
- if (i < items.Length - 1) {
- items [i] = items [i + 1];
- items [i + 1] = statusItem;
- DataContext.Items [i] = DataContext.Items [i + 1];
- DataContext.Items [i + 1] = new DynamicStatusItemList (statusItem.Title, statusItem);
- _lstItems.SelectedItem = i + 1;
- _statusBar.SetNeedsDisplay ();
- }
- }
- };
- var _btnOk = new Button ("Ok") {
- X = Pos.Right (_frmStatusBar) + 20,
- Y = Pos.Bottom (_frmStatusBarDetails),
- };
- Add (_btnOk);
- var _btnCancel = new Button ("Cancel") {
- X = Pos.Right (_btnOk) + 3,
- Y = Pos.Top (_btnOk),
- };
- _btnCancel.Clicked += () => {
- SetFrameDetails (_currentEditStatusItem);
- };
- Add (_btnCancel);
- _lstItems.SelectedItemChanged += (e) => {
- SetFrameDetails ();
- };
- _btnOk.Clicked += () => {
- if (ustring.IsNullOrEmpty (_frmStatusBarDetails._txtTitle.Text) && _currentEditStatusItem != null) {
- MessageBox.ErrorQuery ("Invalid title", "Must enter a valid title!.", "Ok");
- } else if (_currentEditStatusItem != null) {
- _frmStatusBarDetails._txtTitle.Text = SetTitleText (
- _frmStatusBarDetails._txtTitle.Text, _frmStatusBarDetails._txtShortcut.Text);
- var statusItem = new DynamicStatusItem (_frmStatusBarDetails._txtTitle.Text,
- _frmStatusBarDetails._txtAction.Text,
- _frmStatusBarDetails._txtShortcut.Text);
- UpdateStatusItem (_currentEditStatusItem, statusItem, _lstItems.SelectedItem);
- }
- };
- _btnAdd.Clicked += () => {
- if (StatusBar == null) {
- MessageBox.ErrorQuery ("StatusBar Bar Error", "Must add a StatusBar first!", "Ok");
- _btnAddStatusBar.SetFocus ();
- return;
- }
- var frameDetails = new DynamicStatusBarDetails ();
- var item = frameDetails.EnterStatusItem ();
- if (item == null) {
- return;
- }
- StatusItem newStatusItem = CreateNewStatusBar (item);
- _currentSelectedStatusBar++;
- _statusBar.AddItemAt (_currentSelectedStatusBar, newStatusItem);
- DataContext.Items.Add (new DynamicStatusItemList (newStatusItem.Title, newStatusItem));
- _lstItems.MoveDown ();
- SetFrameDetails ();
- };
- _btnRemove.Clicked += () => {
- var statusItem = DataContext.Items.Count > 0 ? DataContext.Items [_lstItems.SelectedItem].StatusItem : null;
- if (statusItem != null) {
- _statusBar.RemoveItem (_currentSelectedStatusBar);
- DataContext.Items.RemoveAt (_lstItems.SelectedItem);
- if (_lstItems.Source.Count > 0 && _lstItems.SelectedItem > _lstItems.Source.Count - 1) {
- _lstItems.SelectedItem = _lstItems.Source.Count - 1;
- }
- _lstItems.SetNeedsDisplay ();
- SetFrameDetails ();
- }
- };
- _lstItems.Enter += (_) => {
- var statusItem = DataContext.Items.Count > 0 ? DataContext.Items [_lstItems.SelectedItem].StatusItem : null;
- SetFrameDetails (statusItem);
- };
- _btnAddStatusBar.Clicked += () => {
- if (_statusBar != null) {
- return;
- }
- _statusBar = new StatusBar ();
- Add (_statusBar);
- };
- _btnRemoveStatusBar.Clicked += () => {
- if (_statusBar == null) {
- return;
- }
- Remove (_statusBar);
- _statusBar = null;
- DataContext.Items = new List<DynamicStatusItemList> ();
- _currentStatusItem = null;
- _currentSelectedStatusBar = -1;
- SetListViewSource (_currentStatusItem, true);
- SetFrameDetails (null);
- };
- SetFrameDetails ();
- var ustringConverter = new UStringValueConverter ();
- var listWrapperConverter = new ListWrapperConverter ();
- var lstItems = new Binding (this, "Items", _lstItems, "Source", listWrapperConverter);
- void SetFrameDetails (StatusItem statusItem = null)
- {
- StatusItem newStatusItem;
- if (statusItem == null) {
- newStatusItem = DataContext.Items.Count > 0 ? DataContext.Items [_lstItems.SelectedItem].StatusItem : null;
- } else {
- newStatusItem = statusItem;
- }
- _currentEditStatusItem = newStatusItem;
- _frmStatusBarDetails.EditStatusItem (newStatusItem);
- var f = _btnOk.Enabled == _frmStatusBarDetails.Enabled;
- if (!f) {
- _btnOk.Enabled = _frmStatusBarDetails.Enabled;
- _btnCancel.Enabled = _frmStatusBarDetails.Enabled;
- }
- }
- void SetListViewSource (StatusItem _currentStatusItem, bool fill = false)
- {
- DataContext.Items = new List<DynamicStatusItemList> ();
- var statusItem = _currentStatusItem;
- if (!fill) {
- return;
- }
- if (statusItem != null) {
- foreach (var si in _statusBar.Items) {
- DataContext.Items.Add (new DynamicStatusItemList (si.Title, si));
- }
- }
- }
- StatusItem CreateNewStatusBar (DynamicStatusItem item)
- {
- var newStatusItem = new StatusItem (ShortcutHelper.GetShortcutFromTag (
- item.shortcut, StatusBar.ShortcutDelimiter),
- item.title, _frmStatusBarDetails.CreateAction (item));
- return newStatusItem;
- }
- void UpdateStatusItem (StatusItem _currentEditStatusItem, DynamicStatusItem statusItem, int index)
- {
- _currentEditStatusItem = CreateNewStatusBar (statusItem);
- _statusBar.Items [index] = _currentEditStatusItem;
- if (DataContext.Items.Count == 0) {
- DataContext.Items.Add (new DynamicStatusItemList (_currentEditStatusItem.Title, _currentEditStatusItem));
- }
- DataContext.Items [index] = new DynamicStatusItemList (_currentEditStatusItem.Title, _currentEditStatusItem);
- SetFrameDetails (_currentEditStatusItem);
- }
- //_frmStatusBarDetails.Initialized += (s, e) => _frmStatusBarDetails.Enabled = false;
- }
- public static ustring SetTitleText (ustring title, ustring shortcut)
- {
- var txt = title;
- var split = title.ToString ().Split ('~');
- if (split.Length > 1) {
- txt = split [2].Trim (); ;
- }
- if (string.IsNullOrEmpty (shortcut.ToString ())) {
- return txt;
- }
- return $"~{shortcut}~ {txt}";
- }
- }
- public class DynamicStatusBarDetails : FrameView {
- public StatusItem _statusItem;
- public TextField _txtTitle;
- public TextView _txtAction;
- public TextField _txtShortcut;
- public DynamicStatusBarDetails (StatusItem statusItem = null) : this (statusItem == null ? "Adding New StatusBar Item." : "Editing StatusBar Item.")
- {
- _statusItem = statusItem;
- }
- public DynamicStatusBarDetails (ustring title) : base (title)
- {
- var _lblTitle = new Label ("Title:") {
- Y = 1
- };
- Add (_lblTitle);
- _txtTitle = new TextField () {
- X = Pos.Right (_lblTitle) + 4,
- Y = Pos.Top (_lblTitle),
- Width = Dim.Fill ()
- };
- Add (_txtTitle);
- var _lblAction = new Label ("Action:") {
- X = Pos.Left (_lblTitle),
- Y = Pos.Bottom (_lblTitle) + 1
- };
- Add (_lblAction);
- _txtAction = new TextView () {
- X = Pos.Left (_txtTitle),
- Y = Pos.Top (_lblAction),
- Width = Dim.Fill (),
- Height = 5
- };
- Add (_txtAction);
- var _lblShortcut = new Label ("Shortcut:") {
- X = Pos.Left (_lblTitle),
- Y = Pos.Bottom (_txtAction) + 1
- };
- Add (_lblShortcut);
- _txtShortcut = new TextField () {
- X = Pos.X (_txtAction),
- Y = Pos.Y (_lblShortcut),
- Width = Dim.Fill (),
- ReadOnly = true
- };
- _txtShortcut.KeyDown += (e) => {
- if (!ProcessKey (e.KeyEvent)) {
- return;
- }
- var k = ShortcutHelper.GetModifiersKey (e.KeyEvent);
- if (CheckShortcut (k, true)) {
- e.Handled = true;
- }
- };
- bool ProcessKey (KeyEvent ev)
- {
- switch (ev.Key) {
- case Key.CursorUp:
- case Key.CursorDown:
- case Key.Tab:
- case Key.BackTab:
- return false;
- }
- return true;
- }
- bool CheckShortcut (Key k, bool pre)
- {
- var m = _statusItem != null ? _statusItem : new StatusItem (k, "", null);
- if (pre && !ShortcutHelper.PreShortcutValidation (k)) {
- _txtShortcut.Text = "";
- return false;
- }
- if (!pre) {
- if (!ShortcutHelper.PostShortcutValidation (ShortcutHelper.GetShortcutFromTag (
- _txtShortcut.Text, StatusBar.ShortcutDelimiter))) {
- _txtShortcut.Text = "";
- return false;
- }
- return true;
- }
- _txtShortcut.Text = ShortcutHelper.GetShortcutTag (k, StatusBar.ShortcutDelimiter);
- return true;
- }
- _txtShortcut.KeyUp += (e) => {
- var k = ShortcutHelper.GetModifiersKey (e.KeyEvent);
- if (CheckShortcut (k, false)) {
- e.Handled = true;
- }
- };
- Add (_txtShortcut);
- var _btnShortcut = new Button ("Clear Shortcut") {
- X = Pos.X (_lblShortcut),
- Y = Pos.Bottom (_txtShortcut) + 1
- };
- _btnShortcut.Clicked += () => {
- _txtShortcut.Text = "";
- };
- Add (_btnShortcut);
- }
- public DynamicStatusItem EnterStatusItem ()
- {
- var valid = false;
- if (_statusItem == null) {
- var m = new DynamicStatusItem ();
- _txtTitle.Text = m.title;
- _txtAction.Text = m.action;
- } else {
- EditStatusItem (_statusItem);
- }
- var _btnOk = new Button ("Ok") {
- IsDefault = true,
- };
- _btnOk.Clicked += () => {
- if (ustring.IsNullOrEmpty (_txtTitle.Text)) {
- MessageBox.ErrorQuery ("Invalid title", "Must enter a valid title!.", "Ok");
- } else {
- if (!ustring.IsNullOrEmpty (_txtShortcut.Text)) {
- _txtTitle.Text = DynamicStatusBarSample.SetTitleText (
- _txtTitle.Text, _txtShortcut.Text);
- }
- valid = true;
- Application.RequestStop ();
- }
- };
- var _btnCancel = new Button ("Cancel");
- _btnCancel.Clicked += () => {
- _txtTitle.Text = ustring.Empty;
- Application.RequestStop ();
- };
- var _dialog = new Dialog ("Please enter the item details.", _btnOk, _btnCancel);
- Width = Dim.Fill ();
- Height = Dim.Fill () - 1;
- _dialog.Add (this);
- _txtTitle.SetFocus ();
- _txtTitle.CursorPosition = _txtTitle.Text.Length;
- Application.Run (_dialog);
- if (valid) {
- return new DynamicStatusItem (_txtTitle.Text, _txtAction.Text, _txtShortcut.Text);
- } else {
- return null;
- }
- }
- public void EditStatusItem (StatusItem statusItem)
- {
- if (statusItem == null) {
- Enabled = false;
- CleanEditStatusItem ();
- return;
- } else {
- Enabled = true;
- }
- _statusItem = statusItem;
- _txtTitle.Text = statusItem?.Title ?? "";
- _txtAction.Text = statusItem != null && statusItem.Action != null ? GetTargetAction (statusItem.Action) : ustring.Empty;
- _txtShortcut.Text = ShortcutHelper.GetShortcutTag (statusItem.Shortcut, StatusBar.ShortcutDelimiter) ?? "";
- }
- void CleanEditStatusItem ()
- {
- _txtTitle.Text = "";
- _txtAction.Text = "";
- _txtShortcut.Text = "";
- }
- ustring GetTargetAction (Action action)
- {
- var me = action.Target;
- if (me == null) {
- throw new ArgumentException ();
- }
- object v = new object ();
- foreach (var field in me.GetType ().GetFields ()) {
- if (field.Name == "item") {
- v = field.GetValue (me);
- }
- }
- return v == null || !(v is DynamicStatusItem item) ? ustring.Empty : item.action;
- }
- public Action CreateAction (DynamicStatusItem item)
- {
- return new Action (() => MessageBox.ErrorQuery (item.title, item.action, "Ok"));
- }
- }
- public class DynamicStatusItemModel : INotifyPropertyChanged {
- public event PropertyChangedEventHandler PropertyChanged;
- private ustring statusBar;
- private List<DynamicStatusItemList> items;
- public ustring StatusBar {
- get => statusBar;
- set {
- if (value != statusBar) {
- statusBar = value;
- PropertyChanged?.Invoke (this, new PropertyChangedEventArgs (GetPropertyName ()));
- }
- }
- }
- public List<DynamicStatusItemList> Items {
- get => items;
- set {
- if (value != items) {
- items = value;
- PropertyChanged?.Invoke (this, new PropertyChangedEventArgs (GetPropertyName ()));
- }
- }
- }
- public DynamicStatusItemModel ()
- {
- Items = new List<DynamicStatusItemList> ();
- }
- public string GetPropertyName ([CallerMemberName] string propertyName = null)
- {
- return propertyName;
- }
- }
- public interface IValueConverter {
- object Convert (object value, object parameter = null);
- }
- public class Binding {
- public View Target { get; private set; }
- public View Source { get; private set; }
- public string SourcePropertyName { get; private set; }
- public string TargetPropertyName { get; private set; }
- private object sourceDataContext;
- private PropertyInfo sourceBindingProperty;
- private IValueConverter valueConverter;
- public Binding (View source, string sourcePropertyName, View target, string targetPropertyName, IValueConverter valueConverter = null)
- {
- Target = target;
- Source = source;
- SourcePropertyName = sourcePropertyName;
- TargetPropertyName = targetPropertyName;
- sourceDataContext = Source.GetType ().GetProperty ("DataContext").GetValue (Source);
- sourceBindingProperty = sourceDataContext.GetType ().GetProperty (SourcePropertyName);
- this.valueConverter = valueConverter;
- UpdateTarget ();
- var notifier = ((INotifyPropertyChanged)sourceDataContext);
- if (notifier != null) {
- notifier.PropertyChanged += (s, e) => {
- if (e.PropertyName == SourcePropertyName) {
- UpdateTarget ();
- }
- };
- }
- }
- private void UpdateTarget ()
- {
- try {
- var sourceValue = sourceBindingProperty.GetValue (sourceDataContext);
- if (sourceValue == null) {
- return;
- }
- var finalValue = valueConverter?.Convert (sourceValue) ?? sourceValue;
- var targetProperty = Target.GetType ().GetProperty (TargetPropertyName);
- targetProperty.SetValue (Target, finalValue);
- } catch (Exception ex) {
- MessageBox.ErrorQuery ("Binding Error", $"Binding failed: {ex}.", "Ok");
- }
- }
- }
- public class ListWrapperConverter : IValueConverter {
- public object Convert (object value, object parameter = null)
- {
- return new ListWrapper ((IList)value);
- }
- }
- public class UStringValueConverter : IValueConverter {
- public object Convert (object value, object parameter = null)
- {
- var data = Encoding.ASCII.GetBytes (value.ToString ());
- return ustring.Make (data);
- }
- }
- }
- }
|