mirror of
				https://github.com/Tyrrrz/CliFx.git
				synced 2025-10-25 15:19:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			287 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			287 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Linq;
 | |
| using CliFx.Internal;
 | |
| using CliFx.Models;
 | |
| 
 | |
| namespace CliFx.Services
 | |
| {
 | |
|     /// <summary>
 | |
|     /// Default implementation of <see cref="IHelpTextRenderer"/>.
 | |
|     /// </summary>
 | |
|     public partial class HelpTextRenderer : IHelpTextRenderer
 | |
|     {
 | |
|         /// <inheritdoc />
 | |
|         public void RenderHelpText(IConsole console, HelpTextSource source)
 | |
|         {
 | |
|             console.GuardNotNull(nameof(console));
 | |
|             source.GuardNotNull(nameof(source));
 | |
| 
 | |
|             // Track position
 | |
|             var column = 0;
 | |
|             var row = 0;
 | |
| 
 | |
|             // Get built-in option schemas (help and version)
 | |
|             var builtInOptionSchemas = new List<CommandOptionSchema> {CommandOptionSchema.HelpOption};
 | |
|             if (source.TargetCommandSchema.IsDefault())
 | |
|                 builtInOptionSchemas.Add(CommandOptionSchema.VersionOption);
 | |
| 
 | |
|             // Get child command schemas
 | |
|             var childCommandSchemas = source.AvailableCommandSchemas
 | |
|                 .Where(c => source.AvailableCommandSchemas.FindParent(c.Name) == source.TargetCommandSchema)
 | |
|                 .ToArray();
 | |
| 
 | |
|             // Define helper functions
 | |
| 
 | |
|             bool IsEmpty() => column == 0 && row == 0;
 | |
| 
 | |
|             void Render(string text)
 | |
|             {
 | |
|                 console.Output.Write(text);
 | |
| 
 | |
|                 column += text.Length;
 | |
|             }
 | |
| 
 | |
|             void RenderNewLine()
 | |
|             {
 | |
|                 console.Output.WriteLine();
 | |
| 
 | |
|                 column = 0;
 | |
|                 row++;
 | |
|             }
 | |
| 
 | |
|             void RenderMargin(int lines = 1)
 | |
|             {
 | |
|                 if (!IsEmpty())
 | |
|                 {
 | |
|                     for (var i = 0; i < lines; i++)
 | |
|                         RenderNewLine();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             void RenderIndent(int spaces = 2)
 | |
|             {
 | |
|                 Render(' '.Repeat(spaces));
 | |
|             }
 | |
| 
 | |
|             void RenderColumnIndent(int spaces = 20, int margin = 2)
 | |
|             {
 | |
|                 if (column + margin >= spaces)
 | |
|                 {
 | |
|                     RenderNewLine();
 | |
|                     RenderIndent(spaces);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     RenderIndent(spaces - column);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             void RenderWithColor(string text, ConsoleColor foregroundColor)
 | |
|             {
 | |
|                 console.WithForegroundColor(foregroundColor, () => Render(text));
 | |
|             }
 | |
| 
 | |
|             void RenderWithColors(string text, ConsoleColor foregroundColor, ConsoleColor backgroundColor)
 | |
|             {
 | |
|                 console.WithColors(foregroundColor, backgroundColor, () => Render(text));
 | |
|             }
 | |
| 
 | |
|             void RenderHeader(string text)
 | |
|             {
 | |
|                 RenderWithColors(text, ConsoleColor.Black, ConsoleColor.DarkGray);
 | |
|                 RenderNewLine();
 | |
|             }
 | |
| 
 | |
|             void RenderApplicationInfo()
 | |
|             {
 | |
|                 if (!source.TargetCommandSchema.IsDefault())
 | |
|                     return;
 | |
| 
 | |
|                 // Title and version
 | |
|                 RenderWithColor(source.ApplicationMetadata.Title, ConsoleColor.Yellow);
 | |
|                 Render(" ");
 | |
|                 RenderWithColor(source.ApplicationMetadata.VersionText, ConsoleColor.Yellow);
 | |
|                 RenderNewLine();
 | |
| 
 | |
|                 // Description
 | |
|                 if (!source.ApplicationMetadata.Description.IsNullOrWhiteSpace())
 | |
|                 {
 | |
|                     Render(source.ApplicationMetadata.Description);
 | |
|                     RenderNewLine();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             void RenderDescription()
 | |
|             {
 | |
|                 if (source.TargetCommandSchema.Description.IsNullOrWhiteSpace())
 | |
|                     return;
 | |
| 
 | |
|                 // Margin
 | |
|                 RenderMargin();
 | |
| 
 | |
|                 // Header
 | |
|                 RenderHeader("Description");
 | |
| 
 | |
|                 // Description
 | |
|                 RenderIndent();
 | |
|                 Render(source.TargetCommandSchema.Description);
 | |
|                 RenderNewLine();
 | |
|             }
 | |
| 
 | |
|             void RenderUsage()
 | |
|             {
 | |
|                 // Margin
 | |
|                 RenderMargin();
 | |
| 
 | |
|                 // Header
 | |
|                 RenderHeader("Usage");
 | |
| 
 | |
|                 // Exe name
 | |
|                 RenderIndent();
 | |
|                 Render(source.ApplicationMetadata.ExecutableName);
 | |
| 
 | |
|                 // Command name
 | |
|                 if (!source.TargetCommandSchema.IsDefault())
 | |
|                 {
 | |
|                     Render(" ");
 | |
|                     RenderWithColor(source.TargetCommandSchema.Name, ConsoleColor.Cyan);
 | |
|                 }
 | |
| 
 | |
|                 // Child command
 | |
|                 if (childCommandSchemas.Any())
 | |
|                 {
 | |
|                     Render(" ");
 | |
|                     RenderWithColor("[command]", ConsoleColor.Cyan);
 | |
|                 }
 | |
| 
 | |
|                 // Options
 | |
|                 Render(" ");
 | |
|                 RenderWithColor("[options]", ConsoleColor.White);
 | |
|                 RenderNewLine();
 | |
|             }
 | |
| 
 | |
|             void RenderOptions()
 | |
|             {
 | |
|                 // Margin
 | |
|                 RenderMargin();
 | |
| 
 | |
|                 // Header
 | |
|                 RenderHeader("Options");
 | |
| 
 | |
|                 // Options
 | |
|                 foreach (var optionSchema in source.TargetCommandSchema.Options.Concat(builtInOptionSchemas))
 | |
|                 {
 | |
|                     // Is required
 | |
|                     if (optionSchema.IsRequired)
 | |
|                     {
 | |
|                         RenderWithColor("* ", ConsoleColor.Red);
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         RenderIndent();
 | |
|                     }
 | |
| 
 | |
|                     // Short name
 | |
|                     if (optionSchema.ShortName != null)
 | |
|                     {
 | |
|                         RenderWithColor($"-{optionSchema.ShortName}", ConsoleColor.White);
 | |
|                     }
 | |
| 
 | |
|                     // Delimiter
 | |
|                     if (!optionSchema.Name.IsNullOrWhiteSpace() && optionSchema.ShortName != null)
 | |
|                     {
 | |
|                         Render("|");
 | |
|                     }
 | |
| 
 | |
|                     // Name
 | |
|                     if (!optionSchema.Name.IsNullOrWhiteSpace())
 | |
|                     {
 | |
|                         RenderWithColor($"--{optionSchema.Name}", ConsoleColor.White);
 | |
|                     }
 | |
| 
 | |
|                     // Description
 | |
|                     if (!optionSchema.Description.IsNullOrWhiteSpace())
 | |
|                     {
 | |
|                         RenderColumnIndent();
 | |
|                         Render(optionSchema.Description);
 | |
|                     }
 | |
| 
 | |
|                     RenderNewLine();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             void RenderChildCommands()
 | |
|             {
 | |
|                 if (!childCommandSchemas.Any())
 | |
|                     return;
 | |
| 
 | |
|                 // Margin
 | |
|                 RenderMargin();
 | |
| 
 | |
|                 // Header
 | |
|                 RenderHeader("Commands");
 | |
| 
 | |
|                 // Child commands
 | |
|                 foreach (var childCommandSchema in childCommandSchemas)
 | |
|                 {
 | |
|                     var relativeCommandName = GetRelativeCommandName(childCommandSchema, source.TargetCommandSchema);
 | |
| 
 | |
|                     // Name
 | |
|                     RenderIndent();
 | |
|                     RenderWithColor(relativeCommandName, ConsoleColor.Cyan);
 | |
| 
 | |
|                     // Description
 | |
|                     if (!childCommandSchema.Description.IsNullOrWhiteSpace())
 | |
|                     {
 | |
|                         RenderColumnIndent();
 | |
|                         Render(childCommandSchema.Description);
 | |
|                     }
 | |
| 
 | |
|                     RenderNewLine();
 | |
|                 }
 | |
| 
 | |
|                 // Margin
 | |
|                 RenderMargin();
 | |
| 
 | |
|                 // Child command help tip
 | |
|                 Render("You can run `");
 | |
|                 Render(source.ApplicationMetadata.ExecutableName);
 | |
| 
 | |
|                 if (!source.TargetCommandSchema.IsDefault())
 | |
|                 {
 | |
|                     Render(" ");
 | |
|                     RenderWithColor(source.TargetCommandSchema.Name, ConsoleColor.Cyan);
 | |
|                 }
 | |
| 
 | |
|                 Render(" ");
 | |
|                 RenderWithColor("[command]", ConsoleColor.Cyan);
 | |
| 
 | |
|                 Render(" ");
 | |
|                 RenderWithColor("--help", ConsoleColor.White);
 | |
| 
 | |
|                 Render("` to show help on a specific command.");
 | |
| 
 | |
|                 RenderNewLine();
 | |
|             }
 | |
| 
 | |
|             // Reset color just in case
 | |
|             console.ResetColor();
 | |
| 
 | |
|             // Render everything
 | |
|             RenderApplicationInfo();
 | |
|             RenderDescription();
 | |
|             RenderUsage();
 | |
|             RenderOptions();
 | |
|             RenderChildCommands();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public partial class HelpTextRenderer
 | |
|     {
 | |
|         private static string GetRelativeCommandName(CommandSchema commandSchema, CommandSchema parentCommandSchema) =>
 | |
|             parentCommandSchema.Name.IsNullOrWhiteSpace()
 | |
|                 ? commandSchema.Name
 | |
|                 : commandSchema.Name.Substring(parentCommandSchema.Name.Length + 1);
 | |
|     }
 | |
| } |