mirror of
				https://github.com/spectreconsole/spectre.console.git
				synced 2025-10-25 15:19:23 +00:00 
			
		
		
		
	Add breakdown chart support
This also cleans up the bar chart code slightly and fixes some minor bugs that were detected in related code. Closes #244
This commit is contained in:
		
				
					committed by
					
						 Patrik Svensson
						Patrik Svensson
					
				
			
			
				
	
			
			
			
						parent
						
							58400fe74e
						
					
				
				
					commit
					b64e016e8c
				
			| @@ -1,21 +1,42 @@ | ||||
| using Spectre.Console; | ||||
| using Spectre.Console.Rendering; | ||||
|  | ||||
| namespace InfoExample | ||||
| namespace Charts | ||||
| { | ||||
|     public static class Program | ||||
|     { | ||||
|         public static void Main() | ||||
|         { | ||||
|             var chart = new BarChart() | ||||
|             // Render a bar chart | ||||
|             AnsiConsole.WriteLine(); | ||||
|             Render("Fruits per month", new BarChart() | ||||
|                 .Width(60) | ||||
|                 .Label("[green bold underline]Number of fruits[/]") | ||||
|                 .CenterLabel() | ||||
|                 .AddItem("Apple", 12, Color.Yellow) | ||||
|                 .AddItem("Orange", 54, Color.Green) | ||||
|                 .AddItem("Banana", 33, Color.Red); | ||||
|                 .AddItem("Banana", 33, Color.Red)); | ||||
|  | ||||
|             // Render a breakdown chart | ||||
|             AnsiConsole.WriteLine(); | ||||
|             AnsiConsole.Render(chart); | ||||
|             Render("Languages", new BreakdownChart() | ||||
|                 .FullSize() | ||||
|                 .Width(60) | ||||
|                 .ShowAsPercentages() | ||||
|                 .AddItem("SCSS", 37, Color.Red) | ||||
|                 .AddItem("HTML", 28.3, Color.Blue) | ||||
|                 .AddItem("C#", 22.6, Color.Green) | ||||
|                 .AddItem("JavaScript", 6, Color.Yellow) | ||||
|                 .AddItem("Ruby", 6, Color.LightGreen) | ||||
|                 .AddItem("Shell", 0.1, Color.Aqua)); | ||||
|         } | ||||
|  | ||||
|         private static void Render(string title, IRenderable chart) | ||||
|         { | ||||
|             AnsiConsole.Render( | ||||
|                 new Panel(chart) | ||||
|                     .Padding(1, 1) | ||||
|                     .Header(title)); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,3 +1 @@ | ||||
|   Greeting      | ||||
|   Hello World   | ||||
|                 | ||||
|  Hello World  | ||||
|   | ||||
| @@ -0,0 +1,4 @@ | ||||
|                           Number of fruits                   | ||||
|  Apple  0                                                    | ||||
| Orange  █████████████████████████████████████████████████ 54 | ||||
| Banana  ████████████████████████████ 33                      | ||||
| @@ -0,0 +1,4 @@ | ||||
| [38;5;9m████████████████████████[0m[38;5;12m█████████████████[0m[38;5;2m█████████████[0m[38;5;11m███[0m[38;5;119m███[0m | ||||
|                                                              | ||||
| [38;5;9m■[0m SCSS [38;5;8m37[0m    [38;5;12m■[0m HTML [38;5;8m28.3[0m     [38;5;2m■[0m C# [38;5;8m22.6[0m    [38;5;11m■[0m JavaScript [38;5;8m6[0m     | ||||
| [38;5;119m■[0m Ruby [38;5;8m6[0m     [38;5;14m■[0m Shell [38;5;8m0.1[0m                                     | ||||
| @@ -0,0 +1,3 @@ | ||||
| ████████████████████████████████████████████████████████████ | ||||
| ■ SCSS 37    ■ HTML 28,3     ■ C# 22,6    ■ JavaScript 6     | ||||
| ■ Ruby 6     ■ Shell 0,1                                     | ||||
| @@ -0,0 +1,2 @@ | ||||
| ████████████████████████████████████████████████████████████████████████████████ | ||||
| ■ SCSS 37   ■ HTML 28.3    ■ C# 22.6   ■ JavaScript 6   ■ Ruby 6   ■ Shell 0.1   | ||||
| @@ -0,0 +1,4 @@ | ||||
| ████████████████████████████████████████████████████████████ | ||||
|                                                              | ||||
| ■ SCSS 37    ■ HTML 28.3     ■ C# 22.6    ■ JavaScript 6     | ||||
| ■ Ruby 6     ■ Shell 0.1                                     | ||||
| @@ -0,0 +1,2 @@ | ||||
| ████████████████████████████████████████████████████████████ | ||||
| ■ SCSS    ■ HTML   ■ C#   ■ JavaScript    ■ Ruby   ■ Shell   | ||||
| @@ -0,0 +1 @@ | ||||
| ████████████████████████████████████████████████████████████ | ||||
| @@ -0,0 +1,3 @@ | ||||
| ████████████████████████████████████████████████████████████ | ||||
| ■ SCSS 37%    ■ HTML 28.3%   ■ C# 22.6%   ■ JavaScript 6%    | ||||
| ■ Ruby 6%     ■ Shell 0.1%                                   | ||||
| @@ -0,0 +1,3 @@ | ||||
| ████████████████████████████████████████████████████████████ | ||||
| ■ SCSS 37    ■ HTML 28.3     ■ C# 22.6    ■ JavaScript 6     | ||||
| ■ Ruby 6     ■ Shell 0.1                                     | ||||
| @@ -19,7 +19,7 @@ | ||||
|   <ItemGroup> | ||||
|     <PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.8.3" /> | ||||
|     <PackageReference Include="Shouldly" Version="4.0.3" /> | ||||
|     <PackageReference Include="Spectre.Verify.Extensions" Version="0.1.0" /> | ||||
|     <PackageReference Include="Spectre.Verify.Extensions" Version="0.3.0" /> | ||||
|     <PackageReference Include="Verify.Xunit" Version="9.0.0-beta.1" /> | ||||
|     <PackageReference Include="xunit" Version="2.4.1" /> | ||||
|     <PackageReference Include="xunit.runner.visualstudio" Version="2.4.3"> | ||||
|   | ||||
| @@ -28,5 +28,24 @@ namespace Spectre.Console.Tests.Unit | ||||
|             // Then | ||||
|             await Verifier.Verify(console.Output); | ||||
|         } | ||||
|  | ||||
|         [Fact] | ||||
|         [Expectation("Zero_Value")] | ||||
|         public async Task Should_Render_Correctly_2() | ||||
|         { | ||||
|             // Given | ||||
|             var console = new FakeConsole(width: 80); | ||||
|  | ||||
|             // When | ||||
|             console.Render(new BarChart() | ||||
|                 .Width(60) | ||||
|                 .Label("Number of fruits") | ||||
|                 .AddItem("Apple", 0) | ||||
|                 .AddItem("Orange", 54) | ||||
|                 .AddItem("Banana", 33)); | ||||
|  | ||||
|             // Then | ||||
|             await Verifier.Verify(console.Output); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										186
									
								
								src/Spectre.Console.Tests/Unit/BreakdownChartTests.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										186
									
								
								src/Spectre.Console.Tests/Unit/BreakdownChartTests.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,186 @@ | ||||
| using System.Threading.Tasks; | ||||
| using Spectre.Console.Testing; | ||||
| using Spectre.Verify.Extensions; | ||||
| using VerifyXunit; | ||||
| using Xunit; | ||||
|  | ||||
| namespace Spectre.Console.Tests.Unit | ||||
| { | ||||
|     [UsesVerify] | ||||
|     [ExpectationPath("Widgets/BreakdownChart")] | ||||
|     public sealed class BreakdownChartTests | ||||
|     { | ||||
|         [Fact] | ||||
|         [Expectation("Default")] | ||||
|         public async Task Should_Render_Correctly() | ||||
|         { | ||||
|             // Given | ||||
|             var console = new FakeConsole(width: 80); | ||||
|  | ||||
|             // When | ||||
|             console.Render(new BreakdownChart() | ||||
|                 .AddItem("SCSS", 37, Color.Red) | ||||
|                 .AddItem("HTML", 28.3, Color.Blue) | ||||
|                 .AddItem("C#", 22.6, Color.Green) | ||||
|                 .AddItem("JavaScript", 6, Color.Yellow) | ||||
|                 .AddItem("Ruby", 6, Color.LightGreen) | ||||
|                 .AddItem("Shell", 0.1, Color.Aqua)); | ||||
|  | ||||
|             // Then | ||||
|             await Verifier.Verify(console.Output); | ||||
|         } | ||||
|  | ||||
|         [Fact] | ||||
|         [Expectation("Width")] | ||||
|         public async Task Should_Render_With_Specific_Width() | ||||
|         { | ||||
|             // Given | ||||
|             var console = new FakeConsole(width: 80); | ||||
|  | ||||
|             // When | ||||
|             console.Render(new BreakdownChart() | ||||
|                 .Width(60) | ||||
|                 .AddItem("SCSS", 37, Color.Red) | ||||
|                 .AddItem("HTML", 28.3, Color.Blue) | ||||
|                 .AddItem("C#", 22.6, Color.Green) | ||||
|                 .AddItem("JavaScript", 6, Color.Yellow) | ||||
|                 .AddItem("Ruby", 6, Color.LightGreen) | ||||
|                 .AddItem("Shell", 0.1, Color.Aqua)); | ||||
|  | ||||
|             // Then | ||||
|             await Verifier.Verify(console.Output); | ||||
|         } | ||||
|  | ||||
|         [Fact] | ||||
|         [Expectation("ShowAsPercentages")] | ||||
|         public async Task Should_Render_Correctly_With_Specific_Width() | ||||
|         { | ||||
|             // Given | ||||
|             var console = new FakeConsole(width: 80); | ||||
|  | ||||
|             // When | ||||
|             console.Render(new BreakdownChart() | ||||
|                 .Width(60) | ||||
|                 .ShowAsPercentages() | ||||
|                 .AddItem("SCSS", 37, Color.Red) | ||||
|                 .AddItem("HTML", 28.3, Color.Blue) | ||||
|                 .AddItem("C#", 22.6, Color.Green) | ||||
|                 .AddItem("JavaScript", 6, Color.Yellow) | ||||
|                 .AddItem("Ruby", 6, Color.LightGreen) | ||||
|                 .AddItem("Shell", 0.1, Color.Aqua)); | ||||
|  | ||||
|             // Then | ||||
|             await Verifier.Verify(console.Output); | ||||
|         } | ||||
|  | ||||
|         [Fact] | ||||
|         [Expectation("HideTags")] | ||||
|         public async Task Should_Render_Correctly_Without_Tags() | ||||
|         { | ||||
|             // Given | ||||
|             var console = new FakeConsole(width: 80); | ||||
|  | ||||
|             // When | ||||
|             console.Render(new BreakdownChart() | ||||
|                 .Width(60) | ||||
|                 .HideTags() | ||||
|                 .AddItem("SCSS", 37, Color.Red) | ||||
|                 .AddItem("HTML", 28.3, Color.Blue) | ||||
|                 .AddItem("C#", 22.6, Color.Green) | ||||
|                 .AddItem("JavaScript", 6, Color.Yellow) | ||||
|                 .AddItem("Ruby", 6, Color.LightGreen) | ||||
|                 .AddItem("Shell", 0.1, Color.Aqua)); | ||||
|  | ||||
|             // Then | ||||
|             await Verifier.Verify(console.Output); | ||||
|         } | ||||
|  | ||||
|         [Fact] | ||||
|         [Expectation("HideTagValues")] | ||||
|         public async Task Should_Render_Correctly_Without_Tag_Values() | ||||
|         { | ||||
|             // Given | ||||
|             var console = new FakeConsole(width: 80); | ||||
|  | ||||
|             // When | ||||
|             console.Render(new BreakdownChart() | ||||
|                 .Width(60) | ||||
|                 .HideTagValues() | ||||
|                 .AddItem("SCSS", 37, Color.Red) | ||||
|                 .AddItem("HTML", 28.3, Color.Blue) | ||||
|                 .AddItem("C#", 22.6, Color.Green) | ||||
|                 .AddItem("JavaScript", 6, Color.Yellow) | ||||
|                 .AddItem("Ruby", 6, Color.LightGreen) | ||||
|                 .AddItem("Shell", 0.1, Color.Aqua)); | ||||
|  | ||||
|             // Then | ||||
|             await Verifier.Verify(console.Output); | ||||
|         } | ||||
|  | ||||
|         [Fact] | ||||
|         [Expectation("Culture")] | ||||
|         public async Task Should_Render_Correctly_With_Specific_Culture() | ||||
|         { | ||||
|             // Given | ||||
|             var console = new FakeConsole(width: 80); | ||||
|  | ||||
|             // When | ||||
|             console.Render(new BreakdownChart() | ||||
|                 .Width(60) | ||||
|                 .Culture("sv-SE") | ||||
|                 .AddItem("SCSS", 37, Color.Red) | ||||
|                 .AddItem("HTML", 28.3, Color.Blue) | ||||
|                 .AddItem("C#", 22.6, Color.Green) | ||||
|                 .AddItem("JavaScript", 6, Color.Yellow) | ||||
|                 .AddItem("Ruby", 6, Color.LightGreen) | ||||
|                 .AddItem("Shell", 0.1, Color.Aqua)); | ||||
|  | ||||
|             // Then | ||||
|             await Verifier.Verify(console.Output); | ||||
|         } | ||||
|  | ||||
|         [Fact] | ||||
|         [Expectation("FullSize")] | ||||
|         public async Task Should_Render_FullSize_Mode_Correctly() | ||||
|         { | ||||
|             // Given | ||||
|             var console = new FakeConsole(width: 80); | ||||
|  | ||||
|             // When | ||||
|             console.Render(new BreakdownChart() | ||||
|                 .Width(60) | ||||
|                 .FullSize() | ||||
|                 .AddItem("SCSS", 37, Color.Red) | ||||
|                 .AddItem("HTML", 28.3, Color.Blue) | ||||
|                 .AddItem("C#", 22.6, Color.Green) | ||||
|                 .AddItem("JavaScript", 6, Color.Yellow) | ||||
|                 .AddItem("Ruby", 6, Color.LightGreen) | ||||
|                 .AddItem("Shell", 0.1, Color.Aqua)); | ||||
|  | ||||
|             // Then | ||||
|             await Verifier.Verify(console.Output); | ||||
|         } | ||||
|  | ||||
|         [Fact] | ||||
|         [Expectation("Ansi")] | ||||
|         public async Task Should_Render_Correct_Ansi() | ||||
|         { | ||||
|             // Given | ||||
|             var console = new FakeAnsiConsole(ColorSystem.EightBit, width: 80); | ||||
|  | ||||
|             // When | ||||
|             console.Render(new BreakdownChart() | ||||
|                 .Width(60) | ||||
|                 .FullSize() | ||||
|                 .AddItem("SCSS", 37, Color.Red) | ||||
|                 .AddItem("HTML", 28.3, Color.Blue) | ||||
|                 .AddItem("C#", 22.6, Color.Green) | ||||
|                 .AddItem("JavaScript", 6, Color.Yellow) | ||||
|                 .AddItem("Ruby", 6, Color.LightGreen) | ||||
|                 .AddItem("Shell", 0.1, Color.Aqua)); | ||||
|  | ||||
|             // Then | ||||
|             await Verifier.Verify(console.Output); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -6,7 +6,7 @@ namespace Spectre.Console | ||||
|     /// <summary> | ||||
|     /// Contains extension methods for <see cref="BarChart"/>. | ||||
|     /// </summary> | ||||
|     public static class BarGraphExtensions | ||||
|     public static class BarChartExtensions | ||||
|     { | ||||
|         /// <summary> | ||||
|         /// Adds an item to the bar chart. | ||||
| @@ -42,10 +42,18 @@ namespace Spectre.Console | ||||
|                 throw new ArgumentNullException(nameof(chart)); | ||||
|             } | ||||
| 
 | ||||
|             chart.Data.Add(new BarChartItem( | ||||
|                     item.Label, | ||||
|                     item.Value, | ||||
|                     item.Color)); | ||||
|             if (item is BarChartItem barChartItem) | ||||
|             { | ||||
|                 chart.Data.Add(barChartItem); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 chart.Data.Add( | ||||
|                     new BarChartItem( | ||||
|                         item.Label, | ||||
|                         item.Value, | ||||
|                         item.Color)); | ||||
|             } | ||||
| 
 | ||||
|             return chart; | ||||
|         } | ||||
							
								
								
									
										266
									
								
								src/Spectre.Console/Extensions/BreakdownChartExtensions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										266
									
								
								src/Spectre.Console/Extensions/BreakdownChartExtensions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,266 @@ | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
|  | ||||
| namespace Spectre.Console | ||||
| { | ||||
|     /// <summary> | ||||
|     /// Contains extension methods for <see cref="BreakdownChart"/>. | ||||
|     /// </summary> | ||||
|     public static class BreakdownChartExtensions | ||||
|     { | ||||
|         /// <summary> | ||||
|         /// Adds an item to the breakdown chart. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <param name="label">The item label.</param> | ||||
|         /// <param name="value">The item value.</param> | ||||
|         /// <param name="color">The item color.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart AddItem(this BreakdownChart chart, string label, double value, Color color) | ||||
|         { | ||||
|             if (chart is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(chart)); | ||||
|             } | ||||
|  | ||||
|             chart.Data.Add(new BreakdownChartItem(label, value, color)); | ||||
|             return chart; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Adds an item to the breakdown chart. | ||||
|         /// </summary> | ||||
|         /// <typeparam name="T">A type that implements <see cref="IBreakdownChartItem"/>.</typeparam> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <param name="item">The item.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart AddItem<T>(this BreakdownChart chart, T item) | ||||
|             where T : IBreakdownChartItem | ||||
|         { | ||||
|             if (chart is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(chart)); | ||||
|             } | ||||
|  | ||||
|             if (item is BreakdownChartItem chartItem) | ||||
|             { | ||||
|                 chart.Data.Add(chartItem); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 chart.Data.Add( | ||||
|                     new BreakdownChartItem( | ||||
|                         item.Label, | ||||
|                         item.Value, | ||||
|                         item.Color)); | ||||
|             } | ||||
|  | ||||
|             return chart; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Adds multiple items to the breakdown chart. | ||||
|         /// </summary> | ||||
|         /// <typeparam name="T">A type that implements <see cref="IBreakdownChartItem"/>.</typeparam> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <param name="items">The items.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart AddItems<T>(this BreakdownChart chart, IEnumerable<T> items) | ||||
|             where T : IBreakdownChartItem | ||||
|         { | ||||
|             if (chart is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(chart)); | ||||
|             } | ||||
|  | ||||
|             if (items is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(items)); | ||||
|             } | ||||
|  | ||||
|             foreach (var item in items) | ||||
|             { | ||||
|                 AddItem(chart, item); | ||||
|             } | ||||
|  | ||||
|             return chart; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Adds multiple items to the breakdown chart. | ||||
|         /// </summary> | ||||
|         /// <typeparam name="T">A type that implements <see cref="IBarChartItem"/>.</typeparam> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <param name="items">The items.</param> | ||||
|         /// <param name="converter">The converter that converts instances of <c>T</c> to <see cref="IBreakdownChartItem"/>.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart AddItems<T>(this BreakdownChart chart, IEnumerable<T> items, Func<T, IBreakdownChartItem> converter) | ||||
|         { | ||||
|             if (chart is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(chart)); | ||||
|             } | ||||
|  | ||||
|             if (items is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(items)); | ||||
|             } | ||||
|  | ||||
|             if (converter is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(converter)); | ||||
|             } | ||||
|  | ||||
|             foreach (var item in items) | ||||
|             { | ||||
|                 chart.Data.Add(converter(item)); | ||||
|             } | ||||
|  | ||||
|             return chart; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Sets the width of the breakdown chart. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <param name="width">The breakdown chart width.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart Width(this BreakdownChart chart, int? width) | ||||
|         { | ||||
|             if (chart is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(chart)); | ||||
|             } | ||||
|  | ||||
|             chart.Width = width; | ||||
|             return chart; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// All values will be shown as percentages. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart ShowAsPercentages(this BreakdownChart chart) | ||||
|         { | ||||
|             if (chart is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(chart)); | ||||
|             } | ||||
|  | ||||
|             chart.ShowAsPercentages = true; | ||||
|             return chart; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Tags will be shown. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart ShowTags(this BreakdownChart chart) | ||||
|         { | ||||
|             return ShowTags(chart, true); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Tags will be not be shown. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart HideTags(this BreakdownChart chart) | ||||
|         { | ||||
|             return ShowTags(chart, false); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Sets whether or not tags will be shown. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <param name="show">Whether or not tags will be shown.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart ShowTags(this BreakdownChart chart, bool show) | ||||
|         { | ||||
|             if (chart is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(chart)); | ||||
|             } | ||||
|  | ||||
|             chart.ShowTags = show; | ||||
|             return chart; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Tag values will be shown. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart ShowTagValues(this BreakdownChart chart) | ||||
|         { | ||||
|             return ShowTagValues(chart, true); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Tag values will be not be shown. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart HideTagValues(this BreakdownChart chart) | ||||
|         { | ||||
|             return ShowTagValues(chart, false); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Sets whether or not tag values will be shown. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <param name="show">Whether or not tag values will be shown.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart ShowTagValues(this BreakdownChart chart, bool show) | ||||
|         { | ||||
|             if (chart is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(chart)); | ||||
|             } | ||||
|  | ||||
|             chart.ShowTagValues = show; | ||||
|             return chart; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Chart and tags is rendered in compact mode. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart Compact(this BreakdownChart chart) | ||||
|         { | ||||
|             return Compact(chart, true); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Chart and tags is rendered in full size mode. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart FullSize(this BreakdownChart chart) | ||||
|         { | ||||
|             return Compact(chart, false); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Sets whether or not the chart and tags should be rendered in compact mode. | ||||
|         /// </summary> | ||||
|         /// <param name="chart">The breakdown chart.</param> | ||||
|         /// <param name="compact">Whether or not the chart and tags should be rendered in compact mode.</param> | ||||
|         /// <returns>The same instance so that multiple calls can be chained.</returns> | ||||
|         public static BreakdownChart Compact(this BreakdownChart chart, bool compact) | ||||
|         { | ||||
|             if (chart is null) | ||||
|             { | ||||
|                 throw new ArgumentNullException(nameof(chart)); | ||||
|             } | ||||
|  | ||||
|             chart.Compact = compact; | ||||
|             return chart; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -10,7 +10,7 @@ namespace Spectre.Console | ||||
|             culture ??= CultureInfo.InvariantCulture; | ||||
|             return culture.DateTimeFormat | ||||
|                 .GetAbbreviatedDayName(day) | ||||
|                 .Capitalize(culture); | ||||
|                 .CapitalizeFirstLetter(culture); | ||||
|         } | ||||
|  | ||||
|         public static DayOfWeek GetNextWeekDay(this DayOfWeek day) | ||||
|   | ||||
| @@ -44,7 +44,7 @@ namespace Spectre.Console | ||||
|             return Cell.GetCellLength(context, text); | ||||
|         } | ||||
|  | ||||
|         internal static string Capitalize(this string? text, CultureInfo? culture = null) | ||||
|         internal static string CapitalizeFirstLetter(this string? text, CultureInfo? culture = null) | ||||
|         { | ||||
|             if (text == null) | ||||
|             { | ||||
|   | ||||
| @@ -10,6 +10,6 @@ namespace Spectre.Console | ||||
|         /// <summary> | ||||
|         /// Gets or sets the culture. | ||||
|         /// </summary> | ||||
|         CultureInfo Culture { get; set; } | ||||
|         CultureInfo? Culture { get; set; } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -40,7 +40,7 @@ namespace Spectre.Console | ||||
|             return result; | ||||
|         } | ||||
|  | ||||
|         public static List<int> Distribute(int total, List<int> ratios, List<int>? minimums = null) | ||||
|         public static List<int> Distribute(int total, IList<int> ratios, IList<int>? minimums = null) | ||||
|         { | ||||
|             if (minimums != null) | ||||
|             { | ||||
|   | ||||
| @@ -22,7 +22,7 @@ namespace Spectre.Console | ||||
|         private TableBorder _border; | ||||
|         private bool _useSafeBorder; | ||||
|         private Style? _borderStyle; | ||||
|         private CultureInfo _culture; | ||||
|         private CultureInfo? _culture; | ||||
|         private Style _highlightStyle; | ||||
|         private bool _showHeader; | ||||
|         private Style? _headerStyle; | ||||
| @@ -79,7 +79,7 @@ namespace Spectre.Console | ||||
|         /// <summary> | ||||
|         /// Gets or sets the calendar's <see cref="CultureInfo"/>. | ||||
|         /// </summary> | ||||
|         public CultureInfo Culture | ||||
|         public CultureInfo? Culture | ||||
|         { | ||||
|             get => _culture; | ||||
|             set => MarkAsDirty(() => _culture = value); | ||||
|   | ||||
| @@ -1,4 +1,6 @@ | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.Globalization; | ||||
| using System.Linq; | ||||
| using Spectre.Console.Rendering; | ||||
| 
 | ||||
| @@ -7,12 +9,12 @@ namespace Spectre.Console | ||||
|     /// <summary> | ||||
|     /// A renderable (horizontal) bar chart. | ||||
|     /// </summary> | ||||
|     public sealed class BarChart : Renderable | ||||
|     public sealed class BarChart : Renderable, IHasCulture | ||||
|     { | ||||
|         /// <summary> | ||||
|         /// Gets the bar chart data. | ||||
|         /// </summary> | ||||
|         public List<BarChartItem> Data { get; } | ||||
|         public List<IBarChartItem> Data { get; } | ||||
| 
 | ||||
|         /// <summary> | ||||
|         /// Gets or sets the width of the bar chart. | ||||
| @@ -35,24 +37,38 @@ namespace Spectre.Console | ||||
|         /// </summary> | ||||
|         public bool ShowValues { get; set; } = true; | ||||
| 
 | ||||
|         /// <summary> | ||||
|         /// Gets or sets the culture that's used to format values. | ||||
|         /// </summary> | ||||
|         /// <remarks>Defaults to invariant culture.</remarks> | ||||
|         public CultureInfo? Culture { get; set; } | ||||
| 
 | ||||
|         /// <summary> | ||||
|         /// Initializes a new instance of the <see cref="BarChart"/> class. | ||||
|         /// </summary> | ||||
|         public BarChart() | ||||
|         { | ||||
|             Data = new List<BarChartItem>(); | ||||
|             Data = new List<IBarChartItem>(); | ||||
|         } | ||||
| 
 | ||||
|         /// <inheritdoc/> | ||||
|         protected override Measurement Measure(RenderContext context, int maxWidth) | ||||
|         { | ||||
|             var width = Math.Min(Width ?? maxWidth, maxWidth); | ||||
|             return new Measurement(width, width); | ||||
|         } | ||||
| 
 | ||||
|         /// <inheritdoc/> | ||||
|         protected override IEnumerable<Segment> Render(RenderContext context, int maxWidth) | ||||
|         { | ||||
|             var width = Math.Min(Width ?? maxWidth, maxWidth); | ||||
|             var maxValue = Data.Max(item => item.Value); | ||||
| 
 | ||||
|             var grid = new Grid(); | ||||
|             grid.Collapse(); | ||||
|             grid.AddColumn(new GridColumn().PadRight(2).RightAligned()); | ||||
|             grid.AddColumn(new GridColumn().PadLeft(0)); | ||||
|             grid.Width = Width; | ||||
|             grid.Width = width; | ||||
| 
 | ||||
|             if (!string.IsNullOrWhiteSpace(Label)) | ||||
|             { | ||||
| @@ -73,10 +89,11 @@ namespace Spectre.Console | ||||
|                         UnicodeBar = '█', | ||||
|                         AsciiBar = '█', | ||||
|                         ShowValue = ShowValues, | ||||
|                         Culture = Culture, | ||||
|                     }); | ||||
|             } | ||||
| 
 | ||||
|             return ((IRenderable)grid).Render(context, maxWidth); | ||||
|             return ((IRenderable)grid).Render(context, width); | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										42
									
								
								src/Spectre.Console/Widgets/Charts/BreakdownBar.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								src/Spectre.Console/Widgets/Charts/BreakdownBar.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.Linq; | ||||
| using Spectre.Console.Rendering; | ||||
|  | ||||
| namespace Spectre.Console | ||||
| { | ||||
|     internal sealed class BreakdownBar : Renderable | ||||
|     { | ||||
|         private readonly List<IBreakdownChartItem> _data; | ||||
|  | ||||
|         public int? Width { get; set; } | ||||
|  | ||||
|         public BreakdownBar(List<IBreakdownChartItem> data) | ||||
|         { | ||||
|             _data = data ?? throw new ArgumentNullException(nameof(data)); | ||||
|         } | ||||
|  | ||||
|         protected override Measurement Measure(RenderContext context, int maxWidth) | ||||
|         { | ||||
|             var width = Math.Min(Width ?? maxWidth, maxWidth); | ||||
|             return new Measurement(width, width); | ||||
|         } | ||||
|  | ||||
|         protected override IEnumerable<Segment> Render(RenderContext context, int maxWidth) | ||||
|         { | ||||
|             var width = Math.Min(Width ?? maxWidth, maxWidth); | ||||
|  | ||||
|             // Chart | ||||
|             var maxValue = _data.Sum(i => i.Value); | ||||
|             var items = _data.ToArray(); | ||||
|             var bars = Ratio.Distribute(width, items.Select(i => Math.Max(0, (int)(width * (i.Value / maxValue)))).ToArray()); | ||||
|  | ||||
|             for (var index = 0; index < items.Length; index++) | ||||
|             { | ||||
|                 yield return new Segment(new string('█', bars[index]), new Style(items[index].Color)); | ||||
|             } | ||||
|  | ||||
|             yield return Segment.LineBreak; | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										102
									
								
								src/Spectre.Console/Widgets/Charts/BreakdownChart.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								src/Spectre.Console/Widgets/Charts/BreakdownChart.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.Globalization; | ||||
| using Spectre.Console.Rendering; | ||||
|  | ||||
| namespace Spectre.Console | ||||
| { | ||||
|     /// <summary> | ||||
|     /// A renderable breakdown chart. | ||||
|     /// </summary> | ||||
|     public sealed class BreakdownChart : Renderable, IHasCulture | ||||
|     { | ||||
|         /// <summary> | ||||
|         /// Gets the breakdown chart data. | ||||
|         /// </summary> | ||||
|         public List<IBreakdownChartItem> Data { get; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets or sets the width of the breakdown chart. | ||||
|         /// </summary> | ||||
|         public int? Width { get; set; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets or sets a value indicating whether or not | ||||
|         /// to show values as percentages or not. | ||||
|         /// </summary> | ||||
|         public bool ShowAsPercentages { get; set; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets or sets a value indicating whether or not to show tags. | ||||
|         /// </summary> | ||||
|         public bool ShowTags { get; set; } = true; | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets or sets a value indicating whether or not to show tag values. | ||||
|         /// </summary> | ||||
|         public bool ShowTagValues { get; set; } = true; | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets or sets a value indicating whether or not the | ||||
|         /// chart and tags should be rendered in compact mode. | ||||
|         /// </summary> | ||||
|         public bool Compact { get; set; } = true; | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets or sets the <see cref="CultureInfo"/> to use | ||||
|         /// when rendering values. | ||||
|         /// </summary> | ||||
|         /// <remarks>Defaults to invariant culture.</remarks> | ||||
|         public CultureInfo? Culture { get; set; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes a new instance of the <see cref="BreakdownChart"/> class. | ||||
|         /// </summary> | ||||
|         public BreakdownChart() | ||||
|         { | ||||
|             Data = new List<IBreakdownChartItem>(); | ||||
|             Culture = CultureInfo.InvariantCulture; | ||||
|         } | ||||
|  | ||||
|         /// <inheritdoc/> | ||||
|         protected override Measurement Measure(RenderContext context, int maxWidth) | ||||
|         { | ||||
|             var width = Math.Min(Width ?? maxWidth, maxWidth); | ||||
|             return new Measurement(width, width); | ||||
|         } | ||||
|  | ||||
|         /// <inheritdoc/> | ||||
|         protected override IEnumerable<Segment> Render(RenderContext context, int maxWidth) | ||||
|         { | ||||
|             var width = Math.Min(Width ?? maxWidth, maxWidth); | ||||
|  | ||||
|             var grid = new Grid().Width(width); | ||||
|             grid.AddColumn(new GridColumn().NoWrap()); | ||||
|  | ||||
|             // Bar | ||||
|             grid.AddRow(new BreakdownBar(Data) | ||||
|             { | ||||
|                 Width = width, | ||||
|             }); | ||||
|  | ||||
|             if (ShowTags) | ||||
|             { | ||||
|                 if (!Compact) | ||||
|                 { | ||||
|                     grid.AddEmptyRow(); | ||||
|                 } | ||||
|  | ||||
|                 // Tags | ||||
|                 grid.AddRow(new BreakdownTags(Data) | ||||
|                 { | ||||
|                     Width = width, | ||||
|                     Culture = Culture, | ||||
|                     ShowPercentages = ShowAsPercentages, | ||||
|                     ShowTagValues = ShowTagValues, | ||||
|                 }); | ||||
|             } | ||||
|  | ||||
|             return ((IRenderable)grid).Render(context, width); | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										38
									
								
								src/Spectre.Console/Widgets/Charts/BreakdownChartItem.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								src/Spectre.Console/Widgets/Charts/BreakdownChartItem.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| using System; | ||||
|  | ||||
| namespace Spectre.Console | ||||
| { | ||||
|     /// <summary> | ||||
|     /// An item that's shown in a breakdown chart. | ||||
|     /// </summary> | ||||
|     public sealed class BreakdownChartItem : IBreakdownChartItem | ||||
|     { | ||||
|         /// <summary> | ||||
|         /// Gets the item label. | ||||
|         /// </summary> | ||||
|         public string Label { get; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets the item value. | ||||
|         /// </summary> | ||||
|         public double Value { get; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets the item color. | ||||
|         /// </summary> | ||||
|         public Color Color { get; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes a new instance of the <see cref="BreakdownChartItem"/> class. | ||||
|         /// </summary> | ||||
|         /// <param name="label">The item label.</param> | ||||
|         /// <param name="value">The item value.</param> | ||||
|         /// <param name="color">The item color.</param> | ||||
|         public BreakdownChartItem(string label, double value, Color color) | ||||
|         { | ||||
|             Label = label ?? throw new ArgumentNullException(nameof(label)); | ||||
|             Value = value; | ||||
|             Color = color; | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										69
									
								
								src/Spectre.Console/Widgets/Charts/BreakdownTags.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								src/Spectre.Console/Widgets/Charts/BreakdownTags.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.Globalization; | ||||
| using Spectre.Console.Rendering; | ||||
|  | ||||
| namespace Spectre.Console | ||||
| { | ||||
|     internal sealed class BreakdownTags : Renderable | ||||
|     { | ||||
|         private readonly List<IBreakdownChartItem> _data; | ||||
|  | ||||
|         public int? Width { get; set; } | ||||
|         public CultureInfo? Culture { get; set; } | ||||
|         public bool ShowPercentages { get; set; } | ||||
|         public bool ShowTagValues { get; set; } = true; | ||||
|  | ||||
|         public BreakdownTags(List<IBreakdownChartItem> data) | ||||
|         { | ||||
|             _data = data ?? throw new ArgumentNullException(nameof(data)); | ||||
|         } | ||||
|  | ||||
|         protected override Measurement Measure(RenderContext context, int maxWidth) | ||||
|         { | ||||
|             var width = Math.Min(Width ?? maxWidth, maxWidth); | ||||
|             return new Measurement(width, width); | ||||
|         } | ||||
|  | ||||
|         protected override IEnumerable<Segment> Render(RenderContext context, int maxWidth) | ||||
|         { | ||||
|             var culture = Culture ?? CultureInfo.InvariantCulture; | ||||
|  | ||||
|             var panels = new List<Panel>(); | ||||
|             foreach (var item in _data) | ||||
|             { | ||||
|                 var panel = new Panel(GetTag(item, culture)); | ||||
|                 panel.Inline = true; | ||||
|                 panel.Padding = new Padding(0, 0); | ||||
|                 panel.NoBorder(); | ||||
|  | ||||
|                 panels.Add(panel); | ||||
|             } | ||||
|  | ||||
|             foreach (var segment in ((IRenderable)new Columns(panels).Padding(0, 0)).Render(context, maxWidth)) | ||||
|             { | ||||
|                 yield return segment; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         private string GetTag(IBreakdownChartItem item, CultureInfo culture) | ||||
|         { | ||||
|             return string.Format( | ||||
|                 culture, "[{0}]■[/] {1}", | ||||
|                 item.Color.ToMarkup() ?? "default", | ||||
|                 FormatValue(item, culture)).Trim(); | ||||
|         } | ||||
|  | ||||
|         private string FormatValue(IBreakdownChartItem item, CultureInfo culture) | ||||
|         { | ||||
|             if (ShowTagValues) | ||||
|             { | ||||
|                 return string.Format(culture, "{0} [grey]{1}{2}[/]", | ||||
|                     item.Label.EscapeMarkup(), item.Value, | ||||
|                     ShowPercentages ? "%" : string.Empty); | ||||
|             } | ||||
|  | ||||
|             return item.Label.EscapeMarkup(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										23
									
								
								src/Spectre.Console/Widgets/Charts/IBreakdownChartItem.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								src/Spectre.Console/Widgets/Charts/IBreakdownChartItem.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| namespace Spectre.Console | ||||
| { | ||||
|     /// <summary> | ||||
|     /// Represents a breakdown chart item. | ||||
|     /// </summary> | ||||
|     public interface IBreakdownChartItem | ||||
|     { | ||||
|         /// <summary> | ||||
|         /// Gets the item label. | ||||
|         /// </summary> | ||||
|         string Label { get; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets the item value. | ||||
|         /// </summary> | ||||
|         double Value { get; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets the item color. | ||||
|         /// </summary> | ||||
|         Color Color { get; } | ||||
|     } | ||||
| } | ||||
| @@ -40,6 +40,11 @@ namespace Spectre.Console | ||||
|         /// </summary> | ||||
|         public PanelHeader? Header { get; set; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets or sets a value indicating whether or not the panel is inlined. | ||||
|         /// </summary> | ||||
|         internal bool Inline { get; set; } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes a new instance of the <see cref="Panel"/> class. | ||||
|         /// </summary> | ||||
| @@ -71,29 +76,41 @@ namespace Spectre.Console | ||||
|         /// <inheritdoc/> | ||||
|         protected override IEnumerable<Segment> Render(RenderContext context, int maxWidth) | ||||
|         { | ||||
|             var edgeWidth = EdgeWidth; | ||||
|  | ||||
|             var border = BoxExtensions.GetSafeBorder(Border, (context.LegacyConsole || !context.Unicode) && UseSafeBorder); | ||||
|             var borderStyle = BorderStyle ?? Style.Plain; | ||||
|  | ||||
|             var showBorder = true; | ||||
|             if (border is NoBoxBorder) | ||||
|             { | ||||
|                 showBorder = false; | ||||
|                 edgeWidth = 0; | ||||
|             } | ||||
|  | ||||
|             var child = new Padder(_child, Padding); | ||||
|             var childWidth = maxWidth - EdgeWidth; | ||||
|             var childWidth = maxWidth - edgeWidth; | ||||
|  | ||||
|             if (!Expand) | ||||
|             { | ||||
|                 var measurement = ((IRenderable)child).Measure(context, maxWidth - EdgeWidth); | ||||
|                 var measurement = ((IRenderable)child).Measure(context, maxWidth - edgeWidth); | ||||
|                 childWidth = measurement.Max; | ||||
|             } | ||||
|  | ||||
|             var panelWidth = childWidth + EdgeWidth; | ||||
|             var panelWidth = childWidth + edgeWidth; | ||||
|             panelWidth = Math.Min(panelWidth, maxWidth); | ||||
|  | ||||
|             var result = new List<Segment>(); | ||||
|  | ||||
|             // Panel top | ||||
|             AddTopBorder(result, context, border, borderStyle, panelWidth); | ||||
|             if (showBorder) | ||||
|             { | ||||
|                 // Panel top | ||||
|                 AddTopBorder(result, context, border, borderStyle, panelWidth); | ||||
|             } | ||||
|  | ||||
|             // Split the child segments into lines. | ||||
|             var childSegments = ((IRenderable)child).Render(context, childWidth); | ||||
|             foreach (var line in Segment.SplitLines(context, childSegments, childWidth)) | ||||
|             foreach (var (_, _, last, line) in Segment.SplitLines(context, childSegments, childWidth).Enumerate()) | ||||
|             { | ||||
|                 if (line.Count == 1 && line[0].IsWhiteSpace) | ||||
|                 { | ||||
| @@ -102,7 +119,10 @@ namespace Spectre.Console | ||||
|                     continue; | ||||
|                 } | ||||
|  | ||||
|                 result.Add(new Segment(border.GetPart(BoxBorderPart.Left), borderStyle)); | ||||
|                 if (showBorder) | ||||
|                 { | ||||
|                     result.Add(new Segment(border.GetPart(BoxBorderPart.Left), borderStyle)); | ||||
|                 } | ||||
|  | ||||
|                 var content = new List<Segment>(); | ||||
|                 content.AddRange(line); | ||||
| @@ -117,20 +137,45 @@ namespace Spectre.Console | ||||
|  | ||||
|                 result.AddRange(content); | ||||
|  | ||||
|                 result.Add(new Segment(border.GetPart(BoxBorderPart.Right), borderStyle)); | ||||
|                 if (showBorder) | ||||
|                 { | ||||
|                     result.Add(new Segment(border.GetPart(BoxBorderPart.Right), borderStyle)); | ||||
|                 } | ||||
|  | ||||
|                 // Don't emit a line break if this is the last | ||||
|                 // line, we're not showing the border, and we're | ||||
|                 // not rendering this inline. | ||||
|                 var emitLinebreak = !(last && !showBorder && !Inline); | ||||
|                 if (!emitLinebreak) | ||||
|                 { | ||||
|                     continue; | ||||
|                 } | ||||
|  | ||||
|                 result.Add(Segment.LineBreak); | ||||
|             } | ||||
|  | ||||
|             // Panel bottom | ||||
|             result.Add(new Segment(border.GetPart(BoxBorderPart.BottomLeft), borderStyle)); | ||||
|             result.Add(new Segment(border.GetPart(BoxBorderPart.Bottom).Repeat(panelWidth - EdgeWidth), borderStyle)); | ||||
|             result.Add(new Segment(border.GetPart(BoxBorderPart.BottomRight), borderStyle)); | ||||
|             result.Add(Segment.LineBreak); | ||||
|             if (showBorder) | ||||
|             { | ||||
|                 result.Add(new Segment(border.GetPart(BoxBorderPart.BottomLeft), borderStyle)); | ||||
|                 result.Add(new Segment(border.GetPart(BoxBorderPart.Bottom).Repeat(panelWidth - EdgeWidth), borderStyle)); | ||||
|                 result.Add(new Segment(border.GetPart(BoxBorderPart.BottomRight), borderStyle)); | ||||
|             } | ||||
|  | ||||
|             // TODO: Need a better name for this? | ||||
|             // If we're rendering this as part of an inline parent renderable, | ||||
|             // such as columns, we should not emit the last line break. | ||||
|             if (!Inline) | ||||
|             { | ||||
|                 result.Add(Segment.LineBreak); | ||||
|             } | ||||
|  | ||||
|             return result; | ||||
|         } | ||||
|  | ||||
|         private void AddTopBorder(List<Segment> result, RenderContext context, BoxBorder border, Style borderStyle, int panelWidth) | ||||
|         private void AddTopBorder( | ||||
|             List<Segment> result, RenderContext context, BoxBorder border, | ||||
|             Style borderStyle, int panelWidth) | ||||
|         { | ||||
|             var rule = new Rule | ||||
|             { | ||||
|   | ||||
| @@ -5,7 +5,7 @@ using Spectre.Console.Rendering; | ||||
|  | ||||
| namespace Spectre.Console | ||||
| { | ||||
|     internal sealed class ProgressBar : Renderable | ||||
|     internal sealed class ProgressBar : Renderable, IHasCulture | ||||
|     { | ||||
|         public double Value { get; set; } | ||||
|         public double MaxValue { get; set; } = 100; | ||||
| @@ -15,6 +15,7 @@ namespace Spectre.Console | ||||
|         public char UnicodeBar { get; set; } = '━'; | ||||
|         public char AsciiBar { get; set; } = '-'; | ||||
|         public bool ShowValue { get; set; } | ||||
|         public CultureInfo? Culture { get; set; } | ||||
|  | ||||
|         public Style CompletedStyle { get; set; } = new Style(foreground: Color.Yellow); | ||||
|         public Style FinishedStyle { get; set; } = new Style(foreground: Color.Green); | ||||
| @@ -36,17 +37,26 @@ namespace Spectre.Console | ||||
|  | ||||
|             var bars = Math.Max(0, (int)(width * (completed / MaxValue))); | ||||
|  | ||||
|             var value = completed.ToString(CultureInfo.InvariantCulture); | ||||
|             var value = completed.ToString(Culture ?? CultureInfo.InvariantCulture); | ||||
|             if (ShowValue) | ||||
|             { | ||||
|                 bars = bars - value.Length - 1; | ||||
|                 bars = Math.Max(0, bars); | ||||
|             } | ||||
|  | ||||
|             yield return new Segment(new string(token, bars), style); | ||||
|  | ||||
|             if (ShowValue) | ||||
|             { | ||||
|                 yield return new Segment(" " + value, style); | ||||
|                 // TODO: Fix this at some point | ||||
|                 if (bars == 0) | ||||
|                 { | ||||
|                     yield return new Segment(value, style); | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     yield return new Segment(" " + value, style); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (bars < width) | ||||
|   | ||||
| @@ -163,7 +163,7 @@ namespace Spectre.Console | ||||
|                 return Array.Empty<Segment>(); | ||||
|             } | ||||
|  | ||||
|             var paragraph = new Markup(header.Text.Capitalize(), header.Style ?? defaultStyle) | ||||
|             var paragraph = new Markup(header.Text.CapitalizeFirstLetter(), header.Style ?? defaultStyle) | ||||
|                 .Alignment(Justify.Center) | ||||
|                 .Overflow(Overflow.Ellipsis); | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user