-
Notifications
You must be signed in to change notification settings - Fork 481
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
issue #482: reorder options in auto help text #484
Changes from 9 commits
1811ce4
11beeca
f8d26bc
9a9d27a
113a21c
6fc506d
04a9bcf
db7c20a
a1ff100
727f10c
4dc8005
59db0cd
cff9fd1
11c3a6e
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -17,8 +17,71 @@ namespace CommandLine.Text | |
/// Provides means to format an help screen. | ||
/// You can assign it in place of a <see cref="System.String"/> instance. | ||
/// </summary> | ||
|
||
|
||
|
||
public struct ComparableOption | ||
{ | ||
public bool Required; | ||
public bool IsOption; | ||
public bool IsValue; | ||
public string LongName; | ||
public string ShortName; | ||
public int Index; | ||
} | ||
|
||
public class HelpText | ||
{ | ||
|
||
|
||
ComparableOption ToComparableOption(Specification spec, int index) | ||
{ | ||
OptionSpecification option = spec as OptionSpecification; | ||
ValueSpecification value = spec as ValueSpecification; | ||
bool required = option != null ? option.Required : false; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Simplify to:
|
||
|
||
return new ComparableOption() | ||
{ | ||
Required = required, | ||
IsOption = option != null, | ||
IsValue = value != null, | ||
LongName = option?.LongName, | ||
ShortName = option?.ShortName, | ||
Index = index | ||
}; | ||
} | ||
|
||
|
||
public Comparison<ComparableOption> OptionComparison = null; | ||
|
||
public static Comparison<ComparableOption> RequiredThenAlphaComparison = (ComparableOption attr1, ComparableOption attr2) => | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Remove redundant else and braces, and use StringComparison.Ordinal
|
||
{ | ||
if (attr1.IsOption && attr2.IsOption) | ||
{ | ||
if (attr1.Required && !attr2.Required) | ||
{ | ||
return -1; | ||
} | ||
else if (!attr1.Required && attr2.Required) | ||
{ | ||
return 1; | ||
} | ||
else | ||
{ | ||
int t = String.Compare(attr1.LongName, attr2.LongName, StringComparison.CurrentCulture); | ||
return t; | ||
} | ||
} | ||
else if (attr1.IsOption && attr2.IsValue) | ||
{ | ||
return -1; | ||
} | ||
else | ||
{ | ||
return 1; | ||
} | ||
}; | ||
|
||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. #endregion |
||
private const int BuilderCapacity = 128; | ||
private const int DefaultMaximumLength = 80; // default console width | ||
/// <summary> | ||
|
@@ -240,21 +303,24 @@ public SentenceBuilder SentenceBuilder | |
/// <param name='onExample'>A delegate used to customize <see cref="CommandLine.Text.Example"/> model used to render text block of usage examples.</param> | ||
/// <param name="verbsIndex">If true the output style is consistent with verb commands (no dashes), otherwise it outputs options.</param> | ||
/// <param name="maxDisplayWidth">The maximum width of the display.</param> | ||
/// <param name="comparison">a comparison lambda to order options in help text</param> | ||
/// <remarks>The parameter <paramref name="verbsIndex"/> is not ontly a metter of formatting, it controls whether to handle verbs or options.</remarks> | ||
public static HelpText AutoBuild<T>( | ||
ParserResult<T> parserResult, | ||
Func<HelpText, HelpText> onError, | ||
Func<Example, Example> onExample, | ||
bool verbsIndex = false, | ||
int maxDisplayWidth = DefaultMaximumLength) | ||
int maxDisplayWidth = DefaultMaximumLength, | ||
Comparison<ComparableOption> comparison = null) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. //remove this line, no more parameters are used There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. working on it. |
||
{ | ||
var auto = new HelpText | ||
{ | ||
Heading = HeadingInfo.Empty, | ||
Copyright = CopyrightInfo.Empty, | ||
AdditionalNewLineAfterOption = true, | ||
AddDashesToOption = !verbsIndex, | ||
MaximumDisplayWidth = maxDisplayWidth | ||
MaximumDisplayWidth = maxDisplayWidth, | ||
OptionComparison = comparison | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. //remove this line, no more parameters are used |
||
}; | ||
|
||
try | ||
|
@@ -736,14 +802,37 @@ private HelpText AddOptionsImpl( | |
int maximumLength) | ||
{ | ||
var maxLength = GetMaxLength(specifications); | ||
|
||
|
||
|
||
optionsHelp = new StringBuilder(BuilderCapacity); | ||
|
||
var remainingSpace = maximumLength - (maxLength + TotalOptionPadding); | ||
|
||
specifications.ForEach( | ||
option => | ||
AddOption(requiredWord, maxLength, option, remainingSpace)); | ||
if (OptionComparison != null) | ||
{ | ||
int i = -1; | ||
var comparables = specifications.ToList().Select(s => | ||
{ | ||
i++; | ||
return ToComparableOption(s, i); | ||
}).ToList(); | ||
comparables.Sort(OptionComparison); | ||
|
||
|
||
foreach (var comparable in comparables) | ||
{ | ||
Specification spec = specifications.ElementAt(comparable.Index); | ||
AddOption(requiredWord, maxLength, spec, remainingSpace); | ||
} | ||
} | ||
else | ||
{ | ||
specifications.ForEach( | ||
option => | ||
AddOption(requiredWord, maxLength, option, remainingSpace)); | ||
|
||
} | ||
|
||
return this; | ||
} | ||
|
@@ -953,5 +1042,3 @@ private static string FormatDefaultValue<T>(T value) | |
|
||
} | ||
} | ||
|
||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,48 @@ | ||
// Copyright 2005-2015 Giacomo Stelluti Scala & Contributors. All rights reserved. See License.md in the project root for license information. | ||
|
||
using System.Collections.Generic; | ||
using System.Runtime.CompilerServices; | ||
|
||
namespace CommandLine.Tests.Fakes | ||
{ | ||
|
||
[Verb("verb1")] | ||
class Options_HelpText_Ordering_Verb1 | ||
{ | ||
[Option('a', "alpha", Required = true)] | ||
public string alphaOption { get; set; } | ||
|
||
[Option('b', "alpha2", Required = true)] | ||
public string alphaTwoOption { get; set; } | ||
|
||
[Option('d', "charlie", Required = false)] | ||
public string deltaOption { get; set; } | ||
|
||
[Option('c', "bravo", Required = false)] | ||
public string charlieOption { get; set; } | ||
|
||
[Option('f', "foxtrot", Required = false)] | ||
public string foxOption { get; set; } | ||
|
||
[Option('e', "echo", Required = false)] | ||
public string echoOption { get; set; } | ||
|
||
[Value(0)] public string someExtraOption { get; set; } | ||
} | ||
|
||
[Verb("verb2")] | ||
class Options_HelpText_Ordering_Verb2 | ||
{ | ||
[Option('a', "alpha", Required = true)] | ||
public string alphaOption { get; set; } | ||
|
||
[Option('b', "alpha2", Required = true)] | ||
public string alphaTwoOption { get; set; } | ||
|
||
[Option('c', "bravo", Required = false)] | ||
public string charlieOption { get; set; } | ||
|
||
[Option('d', "charlie", Required = false)] | ||
public string deltaOption { get; set; } | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,200 @@ | ||
using System; | ||
using System.Collections.Generic; | ||
using System.Globalization; | ||
using CommandLine.Core; | ||
using System.Linq; | ||
using System.Reflection; | ||
using CommandLine.Infrastructure; | ||
using CommandLine.Tests.Fakes; | ||
using CommandLine.Text; | ||
using FluentAssertions; | ||
using Xunit; | ||
using System.Text; | ||
using Xunit.Sdk; | ||
|
||
namespace CommandLine.Tests.Unit | ||
{ | ||
public class Issue482Tests | ||
{ | ||
[Fact] | ||
public void AutoBuild_without_ordering() | ||
{ | ||
string expectedCompany = "Company"; | ||
|
||
|
||
var parser = Parser.Default; | ||
var parseResult = parser.ParseArguments<Options_HelpText_Ordering_Verb1, Options_HelpText_Ordering_Verb2>( | ||
new[] { "verb1", "--alpha", "alpaga", "--alpha2", "alala", "--charlie", "charlot" }) | ||
.WithNotParsed(errors => { throw new InvalidOperationException("Must be parsed."); }) | ||
.WithParsed(args => {; }); | ||
|
||
var message = HelpText.AutoBuild(parseResult, | ||
err => { throw new InvalidOperationException($"help text build failed. {err.ToString()}"); }, | ||
ex => | ||
{ | ||
return null; | ||
}); | ||
|
||
string helpMessage = message.ToString(); | ||
var helps = helpMessage.Split(new[] { '\r', '\n' }, StringSplitOptions.RemoveEmptyEntries).Skip(2).ToList<string>(); | ||
List<string> expected = new List<string>() | ||
{ | ||
" -a, --alpha Required.", | ||
" -b, --alpha2 Required.", | ||
" -d, --charlie", | ||
" -c, --bravo", | ||
"-f, --foxtrot", | ||
"-e, --echo", | ||
"--help Display this help screen.", | ||
"--version Display version information.", | ||
"value pos. 0" | ||
}; | ||
Assert.Equal(expected.Count, helps.Count); | ||
int i = 0; | ||
foreach (var expect in expected) | ||
{ | ||
Assert.Equal(expect.Trim(), helps[i].Trim()); | ||
i++; | ||
} | ||
|
||
; | ||
} | ||
|
||
[Fact] | ||
public void AutoBuild_with_ordering() | ||
{ | ||
string expectedCompany = "Company"; | ||
|
||
|
||
var parser = Parser.Default; | ||
var parseResult = parser.ParseArguments<Options_HelpText_Ordering_Verb1, Options_HelpText_Ordering_Verb2>( | ||
new[] { "verb1", "--alpha", "alpaga", "--alpha2", "alala", "--charlie", "charlot" }) | ||
.WithNotParsed(errors => { throw new InvalidOperationException("Must be parsed."); }) | ||
.WithParsed(args => {; }); | ||
|
||
Comparison<ComparableOption> comparison = HelpText.RequiredThenAlphaComparison; | ||
|
||
string message = HelpText.AutoBuild(parseResult, error => | ||
{ | ||
throw new InvalidOperationException($"help text build failed. {error.ToString()}"); | ||
}, | ||
ex => | ||
{ | ||
return null; | ||
}, | ||
false, | ||
80, | ||
comparison); | ||
|
||
|
||
string helpMessage = message.ToString(); | ||
var helps = helpMessage.Split(new[] { '\r', '\n' }, StringSplitOptions.RemoveEmptyEntries).Skip(2).ToList<string>(); | ||
List<string> expected = new List<string>() | ||
{ | ||
" -a, --alpha Required.", | ||
" -b, --alpha2 Required.", | ||
" -c, --bravo", | ||
" -d, --charlie", | ||
"-e, --echo", | ||
"-f, --foxtrot", | ||
"--help Display this help screen.", | ||
"--version Display version information.", | ||
"value pos. 0" | ||
}; | ||
Assert.Equal(expected.Count, helps.Count); | ||
int i = 0; | ||
foreach (var expect in expected) | ||
{ | ||
Assert.Equal(expect.Trim(), helps[i].Trim()); | ||
i++; | ||
} | ||
|
||
; | ||
} | ||
|
||
[Fact] | ||
public void AutoBuild_with_ordering_on_shortName() | ||
{ | ||
string expectedCompany = "Company"; | ||
|
||
|
||
var parser = Parser.Default; | ||
var parseResult = parser.ParseArguments<Options_HelpText_Ordering_Verb1, Options_HelpText_Ordering_Verb2>( | ||
new[] { "verb1", "--alpha", "alpaga", "--alpha2", "alala", "--charlie", "charlot" }) | ||
.WithNotParsed(errors => { throw new InvalidOperationException("Must be parsed."); }) | ||
.WithParsed(args => {; }); | ||
|
||
Comparison<ComparableOption> orderOnShortName = (ComparableOption attr1, ComparableOption attr2) => | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Remove redundant else and braces, and use StringComparison.Ordinal as done here |
||
{ | ||
if (attr1.IsOption && attr2.IsOption) | ||
{ | ||
if (attr1.Required && !attr2.Required) | ||
{ | ||
return -1; | ||
} | ||
else if (!attr1.Required && attr2.Required) | ||
{ | ||
return 1; | ||
} | ||
else | ||
{ | ||
if (string.IsNullOrEmpty(attr1.ShortName) && !string.IsNullOrEmpty(attr2.ShortName)) | ||
{ | ||
return 1; | ||
} | ||
else if (!string.IsNullOrEmpty(attr1.ShortName) && string.IsNullOrEmpty(attr2.ShortName)) | ||
{ | ||
return -1; | ||
} | ||
int t = String.Compare(attr1.ShortName, attr2.ShortName, StringComparison.CurrentCulture); | ||
return t; | ||
} | ||
} | ||
else if (attr1.IsOption && attr2.IsValue) | ||
{ | ||
return -1; | ||
} | ||
else | ||
{ | ||
return 1; | ||
} | ||
}; | ||
|
||
string message = HelpText.AutoBuild(parseResult, error => | ||
{ | ||
throw new InvalidOperationException($"help text build failed. {error.ToString()}"); | ||
}, | ||
ex => | ||
{ | ||
return null; | ||
}, | ||
false, | ||
80, | ||
orderOnShortName); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. //remove the line orderOnShortName
|
||
|
||
|
||
var helps = message.Split(new[] { '\r', '\n' }, StringSplitOptions.RemoveEmptyEntries).Skip(2).ToList<string>(); | ||
List<string> expected = new List<string>() | ||
{ | ||
" -a, --alpha Required.", | ||
" -b, --alpha2 Required.", | ||
" -c, --bravo", | ||
" -d, --charlie", | ||
"-e, --echo", | ||
"-f, --foxtrot", | ||
"--help Display this help screen.", | ||
"--version Display version information.", | ||
"value pos. 0" | ||
}; | ||
Assert.Equal(expected.Count, helps.Count); | ||
int i = 0; | ||
foreach (var expect in expected) | ||
{ | ||
Assert.Equal(expect.Trim(), helps[i].Trim()); | ||
i++; | ||
} | ||
} | ||
|
||
|
||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.