public void Views_created_from_an_observable_can_be_updated_by_the_observable() { var observable = new TestObservable(); var view = ContentView.FromObservable(observable); var isViewUpdated = false; view.Updated += (s, e) => { isViewUpdated = true; }; var initialSize = view.Measure(_renderer, new Size(10, 10)); initialSize.Height.Should().Be(0); initialSize.Width.Should().Be(0); observable.UpdateViews("Four"); isViewUpdated.Should().BeTrue(); var updatedSize = view.Measure(_renderer, new Size(10, 10)); updatedSize.Height.Should().Be(1); updatedSize.Width.Should().Be(4); }
/// <summary> /// Demonstrates various rendering capabilities. /// </summary> /// <param name="invocationContext"></param> /// <param name="sample">Renders a specified sample</param> /// <param name="height">The height of the rendering area</param> /// <param name="width">The width of the rendering area</param> /// <param name="top">The top position of the render area</param> /// <param name="left">The left position of the render area</param> /// <param name="text">The text to render</param> /// <param name="overwrite">Overwrite the specified region. (If not, scroll.)</param> public static void Main( #pragma warning disable CS1573 // Parameter has no matching param tag in the XML comment (but other parameters do) InvocationContext invocationContext, SampleName sample = SampleName.Dir, int?height = null, int?width = null, int top = 0, int left = 0, string text = null, bool overwrite = true) #pragma warning restore CS1573 // Parameter has no matching param tag in the XML comment (but other parameters do) { var region = new Region(left, top, width ?? Console.WindowWidth, height ?? Console.WindowHeight, overwrite); if (overwrite && invocationContext.Console is ITerminal terminal) { terminal.Clear(); } var consoleRenderer = new ConsoleRenderer( invocationContext.Console, mode: invocationContext.BindingContext.OutputMode(), resetAfterRender: true); switch (sample) { case SampleName.Colors: { var screen = new ScreenView(renderer: consoleRenderer, invocationContext.Console); screen.Child = new ColorsView(text ?? "*"); screen.Render(region); } break; case SampleName.Dir: var directoryTableView = new DirectoryTableView(new DirectoryInfo(Directory.GetCurrentDirectory())); directoryTableView.Render(consoleRenderer, region); break; case SampleName.Moby: consoleRenderer.RenderToRegion( $"Call me {StyleSpan.BoldOn()}{StyleSpan.UnderlinedOn()}Ishmael{StyleSpan.UnderlinedOff()}{StyleSpan.BoldOff()}. Some years ago -- never mind how long precisely -- having little or no money in my purse, and nothing particular to interest me on shore, I thought I would sail about a little and see the watery part of the world. It is a way I have of driving off the spleen and regulating the circulation. Whenever I find myself growing grim about the mouth; whenever it is a damp, drizzly November in my soul; whenever I find myself involuntarily pausing before coffin warehouses, and bringing up the rear of every funeral I meet; and especially whenever my hypos get such an upper hand of me, that it requires a strong moral principle to prevent me from deliberately stepping into the street, and {ForegroundColorSpan.Rgb(60, 0, 0)}methodically{ForegroundColorSpan.Reset()} {ForegroundColorSpan.Rgb(90, 0, 0)}knocking{ForegroundColorSpan.Reset()} {ForegroundColorSpan.Rgb(120, 0, 0)}people's{ForegroundColorSpan.Reset()} {ForegroundColorSpan.Rgb(160, 0, 0)}hats{ForegroundColorSpan.Reset()} {ForegroundColorSpan.Rgb(220, 0, 0)}off{ForegroundColorSpan.Reset()} then, I account it high time to get to sea as soon as I can. This is my substitute for pistol and ball. With a philosophical flourish Cato throws himself upon his sword; I quietly take to the ship. There is nothing surprising in this. If they but knew it, almost all men in their degree, some time or other, cherish very nearly the same feelings towards the ocean with me.", region); break; case SampleName.Processes: { var view = new ProcessesView(Process.GetProcesses()); view.Render(consoleRenderer, region); } break; case SampleName.TableView: { var table = new TableView <Process> { Items = Process.GetProcesses().Where(x => !string.IsNullOrEmpty(x.MainWindowTitle)).OrderBy(p => p.ProcessName).ToList() }; table.AddColumn(process => $"{process.ProcessName} ", "Name"); table.AddColumn(process => ContentView.FromObservable(process.TrackCpuUsage(), x => $"{x.UsageTotal:P}"), "CPU", ColumnDefinition.Star(1)); var screen = new ScreenView(renderer: consoleRenderer, invocationContext.Console) { Child = table }; screen.Render(); } break; case SampleName.Clock: { var screen = new ScreenView(renderer: consoleRenderer, invocationContext.Console); var lastTime = DateTime.Now; var clockObservable = new BehaviorSubject <DateTime>(lastTime); var clockView = ContentView.FromObservable(clockObservable, x => $"{x:T}"); screen.Child = clockView; screen.Render(); while (!Console.KeyAvailable) { if (DateTime.Now - lastTime > TimeSpan.FromSeconds(1)) { lastTime = DateTime.Now; clockObservable.OnNext(lastTime); } } } break; case SampleName.GridLayout: { var screen = new ScreenView(renderer: consoleRenderer, invocationContext.Console); var content = new ContentView( "Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged. It was popularised in the 1960s with the release of Letraset sheets containing Lorem Ipsum passages, and more recently with desktop publishing software like Aldus PageMaker including versions of Lorem Ipsum for Kevin."); var smallContent = new ContentView("Kevin Bost"); var longContent = new ContentView("Hacking on System.CommandLine"); var gridView = new GridView(); gridView.SetColumns( ColumnDefinition.SizeToContent(), ColumnDefinition.Star(1), ColumnDefinition.Star(0.5) ); gridView.SetRows( RowDefinition.Star(0.5), RowDefinition.Star(0.5) ); gridView.SetChild(smallContent, 0, 0); gridView.SetChild(longContent, 0, 1); //gridView.SetChild(content, 0, 0); gridView.SetChild(content, 1, 1); gridView.SetChild(content, 2, 0); screen.Child = gridView; screen.Render(); } break; default: if (!string.IsNullOrWhiteSpace(text)) { consoleRenderer.RenderToRegion( text, region); } else { var screen = new ScreenView(renderer: consoleRenderer, invocationContext.Console); var stackLayout = new StackLayoutView(); var content1 = new ContentView("Hello World!"); var content2 = new ContentView( "Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged. It was popularised in the 1960s with the release of Letraset sheets containing Lorem Ipsum passages, and more recently with desktop publishing software like Aldus PageMaker including versions of Lorem Ipsum for Kevin."); stackLayout.Add(content2); stackLayout.Add(content1); stackLayout.Add(content2); screen.Child = stackLayout; screen.Render(new Region(0, 0, 50, Size.MaxValue)); //screen.Render(writer); } break; } if (!Console.IsOutputRedirected) { Console.ReadKey(); } }
static async Task <int> WatchAsync( StringLocalizer stringLocalizer, InvocationContext invocationContext, CancellationToken terminationToken, uint interval, bool plot, bool scan, byte scanRange, byte[] skip, bool hydra) { var console = new VirtualTerminal(invocationContext.Console, VirtualTerminalMode.TryEnable()); var reporter = new NvmeSmartAttributesReporter(stringLocalizer); var attributesProviderMap = new Dictionary <int, NvmeSmartAttributesProvider>(); try { try { if (!scan) { console.Out.Write(stringLocalizer["LOG_REGISTERING_PERFORMANCE_COUNTERS"]); reporter.CreateCategory(); } console.Out.Write(stringLocalizer["LOG_DETECTING_NVME_DRIVES"]); for (ushort driveNumber = 0; driveNumber < scanRange; ++driveNumber) { if (skip.Contains((byte)driveNumber)) { continue; } var attributesProvider = NvmeSmartAttributesProvider.Create((byte)driveNumber); if (attributesProvider == null) { continue; } if (attributesProvider.Query(out var _)) { attributesProviderMap.Add(driveNumber, attributesProvider); if (!scan) { reporter.AddInstance(attributesProvider.Moniker, attributesProvider.Model, attributesProvider.SerialNumber); } console.Out.Write($"\t{attributesProvider.Moniker.Replace("_", string.Empty)}: {attributesProvider.Model ?? stringLocalizer["NOT_AVAILABLE"]} (S/N: {attributesProvider.SerialNumber ?? stringLocalizer["NOT_AVAILABLE"]})\r\n"); } else { attributesProvider.Dispose(); } } if (attributesProviderMap.Count == 0) { console.Out.Write(stringLocalizer["LOG_CANNOT_DETECT_ANY_NVME_DRIVES"]); return(1); } if (scan) { return(0); } console.Out.Write(stringLocalizer["LOG_START_MONITORING"]); if (plot) { if (console is ITerminal terminal) { terminal.Clear(); terminal.SetCursorPosition(0, 0); terminal.HideCursor(); } var renderer = new ConsoleRenderer(console, invocationContext.BindingContext.OutputMode(), true); var table = new TableView <NvmeSmartAttributesReporterInstance>() { Items = reporter.Instances }; table.AddColumn(item => item.Moniker.Replace("_", string.Empty), new ContentView("#".Underline())); table.AddColumn(item => item.Model, new ContentView("Model".Underline())); table.AddColumn(item => item.SerialNumber, new ContentView("S/N".Underline())); table.AddColumn(item => ContentView.FromObservable(item.PercentageUsedEstimate, value => $"{value}%"), new ContentView("PUE".Underline())); table.AddColumn(item => ContentView.FromObservable(item.TemperatureDegCelsius, value => $"{value}°C"), new ContentView("CTemp".Underline())); table.AddColumn(item => ContentView.FromObservable(item.WarningCompositeTemperatureTime, value => $"{value} mins."), new ContentView("WCTemp".Underline())); table.AddColumn(item => ContentView.FromObservable(item.CriticalCompositeTemperatureTime, value => $"{value} mins."), new ContentView("CCTemp".Underline())); table.AddColumn(item => ContentView.FromObservable(item.HostReadCommands), new ContentView("HRC".Underline())); table.AddColumn(item => ContentView.FromObservable(item.HostWriteCommands), new ContentView("HWC".Underline())); table.AddColumn(item => ContentView.FromObservable(item.DataUnitsRead), new ContentView("DUR".Underline())); table.AddColumn(item => ContentView.FromObservable(item.DataUnitsWrite), new ContentView("DUW".Underline())); table.AddColumn(item => ContentView.FromObservable(item.MediaErrors), new ContentView("ME".Underline())); table.AddColumn(item => ContentView.FromObservable(item.NumberOfErrorInformationLogEntries), new ContentView("EE".Underline())); var screen = new ScreenView(renderer, console) { Child = table }; var region = new Region(0, 0, Console.WindowWidth, Console.WindowHeight, true); screen.Render(region); } while (!terminationToken.IsCancellationRequested) { foreach (var driveNumber in attributesProviderMap.Keys) { var attributesProvider = attributesProviderMap[driveNumber]; if (!attributesProvider.Query(out var attributes)) { console.Error.Write($"\t{attributesProvider.Moniker}: {stringLocalizer["LOG_FAILED_TO_QUERY_SMART"]}\n"); continue; } reporter.Next(attributesProvider.Moniker, attributes); } await Task.Delay((int)interval, terminationToken); } } finally { if (plot) { if (console is ITerminal terminal) { terminal.Clear(); terminal.SetCursorPosition(0, 0); terminal.ShowCursor(); } } console.Out.Write(stringLocalizer["LOG_QUITTING"]); if (!hydra) { reporter.DeleteCategory(); } reporter.Dispose(); foreach (var attributesProvider in attributesProviderMap.Values) { attributesProvider.Dispose(); } } } catch (TaskCanceledException) { } return(0); }