mirror of
https://github.com/gui-cs/Terminal.Gui.git
synced 2025-12-31 02:08:03 +01:00
421 lines
15 KiB
C#
421 lines
15 KiB
C#
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 ();
|
|
}
|
|
}
|