123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420 |
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using Terminal.Gui;
- namespace UICatalog.Scenarios;
- [ScenarioMetadata ("Content Scrolling", "Demonstrates using View.Viewport and View.GetContentSize () to scroll content.")]
- [ScenarioCategory ("Layout")]
- [ScenarioCategory ("Drawing")]
- [ScenarioCategory ("Scrolling")]
- public class ContentScrolling : Scenario
- {
- private ViewDiagnosticFlags _diagnosticFlags;
- public class ScrollingDemoView : FrameView
- {
- public ScrollingDemoView ()
- {
- Width = Dim.Fill ();
- Height = Dim.Fill ();
- ColorScheme = Colors.ColorSchemes ["Base"];
- Text =
- "Text (ScrollingDemoView.Text). This is long text.\nThe second line.\n3\n4\n5th line\nLine 6. This is a longer line that should wrap automatically.";
- CanFocus = true;
- BorderStyle = LineStyle.Rounded;
- Arrangement = ViewArrangement.Fixed;
- SetContentSize (new (60, 40));
- ViewportSettings |= ViewportSettings.ClearContentOnly;
- ViewportSettings |= ViewportSettings.ClipContentOnly;
- // Things this view knows how to do
- AddCommand (Command.ScrollDown, () => ScrollVertical (1));
- AddCommand (Command.ScrollUp, () => ScrollVertical (-1));
- AddCommand (Command.ScrollRight, () => ScrollHorizontal (1));
- AddCommand (Command.ScrollLeft, () => ScrollHorizontal (-1));
- // Default keybindings for all ListViews
- KeyBindings.Add (Key.CursorUp, Command.ScrollUp);
- KeyBindings.Add (Key.CursorDown, Command.ScrollDown);
- KeyBindings.Add (Key.CursorLeft, Command.ScrollLeft);
- KeyBindings.Add (Key.CursorRight, Command.ScrollRight);
- // Add a status label to the border that shows Viewport and ContentSize values. Bit of a hack.
- // TODO: Move to Padding with controls
- Border.Add (new Label { X = 20 });
- LayoutComplete += VirtualDemoView_LayoutComplete;
- MouseEvent += VirtualDemoView_MouseEvent;
- }
- private void VirtualDemoView_MouseEvent (object sender, MouseEventEventArgs e)
- {
- if (e.MouseEvent.Flags == MouseFlags.WheeledDown)
- {
- ScrollVertical (1);
- return;
- }
- if (e.MouseEvent.Flags == MouseFlags.WheeledUp)
- {
- ScrollVertical (-1);
- return;
- }
- if (e.MouseEvent.Flags == MouseFlags.WheeledRight)
- {
- ScrollHorizontal (1);
- return;
- }
- if (e.MouseEvent.Flags == MouseFlags.WheeledLeft)
- {
- ScrollHorizontal (-1);
- }
- }
- private void VirtualDemoView_LayoutComplete (object sender, LayoutEventArgs e)
- {
- Label status = Border.Subviews.OfType<Label> ().FirstOrDefault ();
- if (status is { })
- {
- status.Title = $"Frame: {Frame}\n\nViewport: {Viewport}, ContentSize = {GetContentSize ()}";
- status.Width = Border.Frame.Width - status.Frame.X - Border.Thickness.Right;
- status.Height = Border.Thickness.Top;
- }
- SetNeedsDisplay ();
- }
- }
- public override void Main ()
- {
- Application.Init ();
- _diagnosticFlags = View.Diagnostics;
- Window app = new ()
- {
- Title = GetQuitKeyAndName (),
- // Use a different colorscheme so ViewSettings.ClearContentOnly is obvious
- ColorScheme = Colors.ColorSchemes ["Toplevel"]
- };
- var editor = new AdornmentsEditor
- {
- AutoSelectViewToEdit = true
- };
- app.Add (editor);
- var view = new ScrollingDemoView
- {
- Title = "Demo View",
- X = Pos.Right (editor),
- Width = Dim.Fill (),
- Height = Dim.Fill ()
- };
- app.Add (view);
- // Add Scroll Setting UI to Padding
- view.Padding.Thickness = new (0, 3, 0, 0);
- view.Padding.ColorScheme = Colors.ColorSchemes ["Error"];
- view.Padding.CanFocus = true;
- var cbAllowNegativeX = new CheckBox
- {
- Title = "Allow _X < 0",
- Y = 0,
- CanFocus = true
- };
- cbAllowNegativeX.CheckedState = view.ViewportSettings.HasFlag(ViewportSettings.AllowNegativeX) ? CheckState.Checked : CheckState.UnChecked;
- cbAllowNegativeX.CheckedStateChanging += AllowNegativeX_Toggle;
- void AllowNegativeX_Toggle (object sender, CancelEventArgs<CheckState> e)
- {
- if (e.NewValue == CheckState.Checked)
- {
- view.ViewportSettings |= ViewportSettings.AllowNegativeX;
- }
- else
- {
- view.ViewportSettings &= ~ViewportSettings.AllowNegativeX;
- }
- }
- view.Padding.Add (cbAllowNegativeX);
- var cbAllowNegativeY = new CheckBox
- {
- Title = "Allow _Y < 0",
- X = Pos.Right (cbAllowNegativeX) + 1,
- Y = 0,
- CanFocus = true
- };
- cbAllowNegativeY.CheckedState = view.ViewportSettings.HasFlag(ViewportSettings.AllowNegativeY) ? CheckState.Checked : CheckState.UnChecked;
- cbAllowNegativeY.CheckedStateChanging += AllowNegativeY_Toggle;
- void AllowNegativeY_Toggle (object sender, CancelEventArgs<CheckState> e)
- {
- if (e.NewValue == CheckState.Checked)
- {
- view.ViewportSettings |= ViewportSettings.AllowNegativeY;
- }
- else
- {
- view.ViewportSettings &= ~ViewportSettings.AllowNegativeY;
- }
- }
- view.Padding.Add (cbAllowNegativeY);
- var cbAllowXGreaterThanContentWidth = new CheckBox
- {
- Title = "All_ow X > Content",
- Y = Pos.Bottom (cbAllowNegativeX),
- CanFocus = true
- };
- cbAllowXGreaterThanContentWidth.CheckedState = view.ViewportSettings.HasFlag(ViewportSettings.AllowXGreaterThanContentWidth) ? CheckState.Checked : CheckState.UnChecked;
- cbAllowXGreaterThanContentWidth.CheckedStateChanging += AllowXGreaterThanContentWidth_Toggle;
- void AllowXGreaterThanContentWidth_Toggle (object sender, CancelEventArgs<CheckState> e)
- {
- if (e.NewValue == CheckState.Checked)
- {
- view.ViewportSettings |= ViewportSettings.AllowXGreaterThanContentWidth;
- }
- else
- {
- view.ViewportSettings &= ~ViewportSettings.AllowXGreaterThanContentWidth;
- }
- }
- view.Padding.Add (cbAllowXGreaterThanContentWidth);
- var cbAllowYGreaterThanContentHeight = new CheckBox
- {
- Title = "Allo_w Y > Content",
- X = Pos.Right (cbAllowXGreaterThanContentWidth) + 1,
- Y = Pos.Bottom (cbAllowNegativeX),
- CanFocus = true
- };
- cbAllowYGreaterThanContentHeight.CheckedState = view.ViewportSettings.HasFlag(ViewportSettings.AllowYGreaterThanContentHeight) ? CheckState.Checked : CheckState.UnChecked;
- cbAllowYGreaterThanContentHeight.CheckedStateChanging += AllowYGreaterThanContentHeight_Toggle;
- void AllowYGreaterThanContentHeight_Toggle (object sender, CancelEventArgs<CheckState> e)
- {
- if (e.NewValue == CheckState.Checked)
- {
- view.ViewportSettings |= ViewportSettings.AllowYGreaterThanContentHeight;
- }
- else
- {
- view.ViewportSettings &= ~ViewportSettings.AllowYGreaterThanContentHeight;
- }
- }
- view.Padding.Add (cbAllowYGreaterThanContentHeight);
- var labelContentSize = new Label
- {
- Title = "_ContentSize:",
- Y = Pos.Bottom (cbAllowYGreaterThanContentHeight)
- };
- NumericUpDown<int> contentSizeWidth = new NumericUpDown<int>
- {
- Value = view.GetContentSize ().Width,
- X = Pos.Right (labelContentSize) + 1,
- Y = Pos.Top (labelContentSize),
- CanFocus = true
- };
- contentSizeWidth.ValueChanging += ContentSizeWidth_ValueChanged;
- void ContentSizeWidth_ValueChanged (object sender, CancelEventArgs<int> e)
- {
- if (e.NewValue < 0)
- {
- e.Cancel = true;
- return;
- }
- // BUGBUG: set_ContentSize is supposed to be `protected`.
- view.SetContentSize (view.GetContentSize () with { Width = e.NewValue });
- }
- var labelComma = new Label
- {
- Title = ",",
- X = Pos.Right (contentSizeWidth),
- Y = Pos.Top (labelContentSize)
- };
- NumericUpDown<int> contentSizeHeight = new NumericUpDown<int>
- {
- Value = view.GetContentSize ().Height,
- X = Pos.Right (labelComma) + 1,
- Y = Pos.Top (labelContentSize),
- CanFocus = true
- };
- contentSizeHeight.ValueChanging += ContentSizeHeight_ValueChanged;
- void ContentSizeHeight_ValueChanged (object sender, CancelEventArgs<int> e)
- {
- if (e.NewValue < 0)
- {
- e.Cancel = true;
- return;
- }
- // BUGBUG: set_ContentSize is supposed to be `protected`.
- view.SetContentSize (view.GetContentSize () with { Height = e.NewValue });
- }
- var cbClearContentOnly = new CheckBox
- {
- Title = "ClearContentOnly",
- X = Pos.Right (contentSizeHeight) + 1,
- Y = Pos.Top (labelContentSize),
- CanFocus = true
- };
- cbClearContentOnly.CheckedState = view.ViewportSettings.HasFlag(ViewportSettings.ClearContentOnly) ? CheckState.Checked : CheckState.UnChecked;
- cbClearContentOnly.CheckedStateChanging += ClearContentOnly_Toggle;
- void ClearContentOnly_Toggle (object sender, CancelEventArgs<CheckState> e)
- {
- if (e.NewValue == CheckState.Checked)
- {
- view.ViewportSettings |= ViewportSettings.ClearContentOnly;
- }
- else
- {
- view.ViewportSettings &= ~ViewportSettings.ClearContentOnly;
- }
- }
- var cbClipContentOnly = new CheckBox
- {
- Title = "ClipContentOnly",
- X = Pos.Right (cbClearContentOnly) + 1,
- Y = Pos.Top (labelContentSize),
- CanFocus = true
- };
- cbClipContentOnly.CheckedState = view.ViewportSettings.HasFlag (ViewportSettings.ClipContentOnly) ? CheckState.Checked : CheckState.UnChecked;
- cbClipContentOnly.CheckedStateChanging += ClipContentOnlyOnly_Toggle;
- void ClipContentOnlyOnly_Toggle (object sender, CancelEventArgs<CheckState> e)
- {
- if (e.NewValue == CheckState.Checked)
- {
- view.ViewportSettings |= ViewportSettings.ClipContentOnly;
- }
- else
- {
- view.ViewportSettings &= ~ViewportSettings.ClipContentOnly;
- }
- }
- view.Padding.Add (labelContentSize, contentSizeWidth, labelComma, contentSizeHeight, cbClearContentOnly, cbClipContentOnly);
- // Add demo views to show that things work correctly
- var textField = new TextField { X = 20, Y = 7, Width = 15, Text = "Test TextField" };
- var colorPicker = new ColorPicker16 { Title = "BG", BoxHeight = 1, BoxWidth = 1, X = Pos.AnchorEnd (), Y = 10 };
- colorPicker.BorderStyle = LineStyle.RoundedDotted;
- colorPicker.ColorChanged += (s, e) =>
- {
- colorPicker.SuperView.ColorScheme = new (colorPicker.SuperView.ColorScheme)
- {
- Normal = new (
- colorPicker.SuperView.ColorScheme.Normal.Foreground,
- e.CurrentValue
- )
- };
- };
- var textView = new TextView
- {
- X = Pos.Center (),
- Y = 10,
- Title = "TextView",
- Text = "I have a 3 row top border.\nMy border inherits from the SuperView.\nI have 3 lines of text with room for 2.",
- AllowsTab = false,
- Width = 30,
- Height = 6 // TODO: Use Dim.Auto
- };
- textView.Border.Thickness = new (1, 3, 1, 1);
- var charMap = new CharMap
- {
- X = Pos.Center (),
- Y = Pos.Bottom (textView) + 1,
- Width = Dim.Auto (DimAutoStyle.Content, maximumContentDim: Dim.Func (() => view.GetContentSize ().Width)),
- Height = Dim.Auto (DimAutoStyle.Content, maximumContentDim: Dim.Percent (20)),
- };
- charMap.Accept += (s, e) =>
- MessageBox.Query (20, 7, "Hi", $"Am I a {view.GetType ().Name}?", "Yes", "No");
- var buttonAnchored = new Button
- {
- X = Pos.AnchorEnd (), Y = Pos.AnchorEnd (), Text = "Bottom Right"
- };
- view.Margin.Data = "Margin";
- view.Margin.Thickness = new (0);
- view.Border.Data = "Border";
- view.Border.Thickness = new (3);
- view.Padding.Data = "Padding";
- view.Add (buttonAnchored, textField, colorPicker, charMap, textView);
- var longLabel = new Label
- {
- Id = "label2",
- X = 0,
- Y = 30,
- Text =
- "This label is long. It should clip to the ContentArea if ClipContentOnly is set. This is a virtual scrolling demo. Use the arrow keys and/or mouse wheel to scroll the content."
- };
- longLabel.TextFormatter.WordWrap = true;
- view.Add (longLabel);
- List<object> options = new () { "Option 1", "Option 2", "Option 3" };
- Slider slider = new (options)
- {
- X = 0,
- Y = Pos.Bottom (textField) + 1,
- Orientation = Orientation.Vertical,
- Type = SliderType.Multiple,
- AllowEmpty = false,
- BorderStyle = LineStyle.Double,
- Title = "_Slider"
- };
- view.Add (slider);
- editor.Initialized += (s, e) => { editor.ViewToEdit = view; };
- app.Closed += (s, e) => View.Diagnostics = _diagnosticFlags;
- editor.AutoSelectViewToEdit = true;
- editor.AutoSelectSuperView = view;
- editor.AutoSelectAdornments = false;
- Application.Run (app);
- app.Dispose ();
- Application.Shutdown ();
- }
- }
|