Skip to content

Commit d7bcf92

Browse files
committed
* Removed dynamic invocation in StringFormatConverter
* Created unit tests for StringFormatConverter
1 parent b1228d7 commit d7bcf92

File tree

5 files changed

+215
-20
lines changed

5 files changed

+215
-20
lines changed

.gitignore

+3
Original file line numberDiff line numberDiff line change
@@ -37,3 +37,6 @@ Example.xlsx
3737
packages/
3838
.vs/
3939
*.nupkg
40+
41+
# Jetbrain Rider Cache
42+
.idea/

src/ValueConverters/StringFormatConverter.cs

+7-20
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,6 @@ namespace WPFLocalizeExtension.ValueConverters
1111
#region Usings
1212
using System;
1313
using System.Globalization;
14-
using System.Reflection;
1514
using System.Windows;
1615
using System.Windows.Data;
1716
#endregion
@@ -21,28 +20,10 @@ namespace WPFLocalizeExtension.ValueConverters
2120
/// </summary>
2221
public class StringFormatConverter : TypeValueConverterBase, IMultiValueConverter
2322
{
24-
private static MethodInfo miFormat = null;
25-
2623
#region IMultiValueConverter
2724
/// <inheritdoc/>
2825
public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
2926
{
30-
if (miFormat == null)
31-
{
32-
try
33-
{
34-
// try to load SmartFormat Assembly
35-
var asSmartFormat = Assembly.Load("SmartFormat");
36-
var tt = asSmartFormat.GetType("SmartFormat.Smart");
37-
miFormat = tt.GetMethod("Format", BindingFlags.Static | BindingFlags.Public, null, new Type[] { typeof(string), typeof(object) }, null);
38-
}
39-
catch
40-
{
41-
// fallback just take String.Format
42-
miFormat = typeof(string).GetMethod("Format", BindingFlags.Static | BindingFlags.Public, null, new Type[] { typeof(string), typeof(object) }, null);
43-
}
44-
}
45-
4627
if (!targetType.IsAssignableFrom(typeof(string)))
4728
throw new Exception("TargetType is not supported strings");
4829

@@ -55,7 +36,13 @@ public object Convert(object[] values, Type targetType, object parameter, Cultur
5536
if (values.Length > 1 && values[1] == DependencyProperty.UnsetValue)
5637
return null;
5738

58-
return (string)miFormat.Invoke(null, values);
39+
var format = values[0].ToString();
40+
if (values.Length == 1)
41+
return format;
42+
43+
var args = new object[values.Length - 1];
44+
Array.Copy(values, 1, args, 0, args.Length);
45+
return string.Format(format, args);
5946
}
6047

6148
/// <inheritdoc/>

tests/Tests.sln

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
2+
Microsoft Visual Studio Solution File, Format Version 12.00
3+
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "WPFLocalizeExtension", "..\src\WPFLocalizeExtension.csproj", "{4A1CC23E-BA4A-4AC8-89CF-E29E0547EAEC}"
4+
EndProject
5+
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "WPFLocalizeExtension.UnitTests", "WPFLocalizeExtension.UnitTests\WPFLocalizeExtension.UnitTests.csproj", "{93C35807-8585-43E3-BA53-15A7B811828E}"
6+
EndProject
7+
Global
8+
GlobalSection(SolutionConfigurationPlatforms) = preSolution
9+
Debug|Any CPU = Debug|Any CPU
10+
Release|Any CPU = Release|Any CPU
11+
EndGlobalSection
12+
GlobalSection(ProjectConfigurationPlatforms) = postSolution
13+
{4A1CC23E-BA4A-4AC8-89CF-E29E0547EAEC}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
14+
{4A1CC23E-BA4A-4AC8-89CF-E29E0547EAEC}.Debug|Any CPU.Build.0 = Debug|Any CPU
15+
{4A1CC23E-BA4A-4AC8-89CF-E29E0547EAEC}.Release|Any CPU.ActiveCfg = Release|Any CPU
16+
{4A1CC23E-BA4A-4AC8-89CF-E29E0547EAEC}.Release|Any CPU.Build.0 = Release|Any CPU
17+
{93C35807-8585-43E3-BA53-15A7B811828E}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
18+
{93C35807-8585-43E3-BA53-15A7B811828E}.Debug|Any CPU.Build.0 = Debug|Any CPU
19+
{93C35807-8585-43E3-BA53-15A7B811828E}.Release|Any CPU.ActiveCfg = Release|Any CPU
20+
{93C35807-8585-43E3-BA53-15A7B811828E}.Release|Any CPU.Build.0 = Release|Any CPU
21+
EndGlobalSection
22+
EndGlobal
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,162 @@
1+
namespace WPFLocalizeExtension.UnitTests.ValueConvertersTests
2+
{
3+
#region Usings
4+
using System;
5+
using System.Collections.Generic;
6+
using System.Globalization;
7+
using System.Text;
8+
using System.Windows;
9+
using Xunit;
10+
using WPFLocalizeExtension.ValueConverters;
11+
#endregion
12+
13+
/// <summary>
14+
/// Tests for converter <see cref="StringFormatConverter" />.
15+
/// </summary>
16+
public class StringFormatConverterTests
17+
{
18+
private const string CONVERTED_VALUE = "hello world";
19+
private readonly object[] _values = new object[] { "{0} {1}", "hello", "world" };
20+
21+
/// <summary>
22+
/// Test data for <see cref="Convert_SpecifiedValues_ValueConverted" />.
23+
/// </summary>
24+
public static IReadOnlyList<object[]> ExpectedStringAndValuesData =>
25+
new List<object[]>
26+
{
27+
new object[] { "hello world", "{0} {1}", "hello", "world" },
28+
new object[] { "12345", "{0}{1}{2}{3}{4}", 1, 2, 3, 4, 5 },
29+
new object[] { "hello world", "hello world" },
30+
new object[] { "01.01.1970", "{0:dd.MM.yyyy}", new DateTime(1970, 1, 1, 10, 0, 0) },
31+
new object[] { "hello world", new StringBuilder("{0} {1}"), "hello", "world" },
32+
};
33+
34+
/// <summary>
35+
/// Check that converter is supported <see cref="object" /> and <see cref="string" /> as target type.
36+
/// </summary>
37+
[Theory]
38+
[InlineData(typeof(object))]
39+
[InlineData(typeof(string))]
40+
public void Convert_SupportedTargetType_ValueConverted(Type targetType)
41+
{
42+
// ARRANGE.
43+
var converter = new StringFormatConverter();
44+
45+
// ACT.
46+
var convertedValue = converter.Convert(_values, targetType, null, CultureInfo.InvariantCulture);
47+
48+
// ASSERT.
49+
Assert.Equal(CONVERTED_VALUE, convertedValue);
50+
}
51+
52+
/// <summary>
53+
/// Check that converter is not supported target types others except <see cref="object" /> and <see cref="string" />.
54+
/// </summary>
55+
[Theory]
56+
[InlineData(typeof(int))]
57+
[InlineData(typeof(DateTime))]
58+
[InlineData(typeof(StringBuilder))]
59+
public void Convert_UnsupportableTargetTypes_ExceptionThrown(Type targetType)
60+
{
61+
// ARRANGE.
62+
var converter = new StringFormatConverter();
63+
64+
// ACT + ASSERT.
65+
var exception = Assert.Throws<Exception>(() => converter.Convert(_values, targetType, null, CultureInfo.InvariantCulture));
66+
Assert.Equal("TargetType is not supported strings", exception.Message);
67+
}
68+
69+
/// <summary>
70+
/// Check that exception is thrown if passed null as values parameter.
71+
/// </summary>
72+
[Fact]
73+
public void Convert_ValuesIsNull_ExceptionThrown()
74+
{
75+
// ARRANGE.
76+
var converter = new StringFormatConverter();
77+
78+
// ACT + ASSERT.
79+
var exception = Assert.Throws<Exception>(() => converter.Convert(null, typeof(string), null, CultureInfo.InvariantCulture));
80+
Assert.Equal("Not enough parameters", exception.Message);
81+
}
82+
83+
/// <summary>
84+
/// Check that exception is thrown if passed empty array as values parameter.
85+
/// </summary>
86+
[Fact]
87+
public void Convert_ValuesIsEmpty_ExceptionThrown()
88+
{
89+
// ARRANGE.
90+
var converter = new StringFormatConverter();
91+
92+
// ACT + ASSERT.
93+
var exception = Assert.Throws<Exception>(() => converter.Convert(Array.Empty<object>(), typeof(string), null, CultureInfo.InvariantCulture));
94+
Assert.Equal("Not enough parameters", exception.Message);
95+
}
96+
97+
/// <summary>
98+
/// Check that returns null if passed null format string.
99+
/// </summary>
100+
[Fact]
101+
public void Convert_FormatStringIsNull_ReturnsNull()
102+
{
103+
// ARRANGE.
104+
var converter = new StringFormatConverter();
105+
106+
// ACT.
107+
var convertedValue = converter.Convert(new object[] { null, "hello", "world" }, typeof(string), null, CultureInfo.InvariantCulture);
108+
109+
// ASSERT.
110+
Assert.Null(convertedValue);
111+
}
112+
113+
/// <summary>
114+
/// Check that returns null if passed UnsetValue as second value.
115+
/// </summary>
116+
[Fact]
117+
public void Convert_SecondValueIsUnsetValue_ReturnsNull()
118+
{
119+
// ARRANGE.
120+
var converter = new StringFormatConverter();
121+
122+
// ACT.
123+
var convertedValue = converter.Convert(new[] { CONVERTED_VALUE, DependencyProperty.UnsetValue }, typeof(string), null, CultureInfo.InvariantCulture);
124+
125+
// ASSERT.
126+
Assert.Null(convertedValue);
127+
}
128+
129+
/// <summary>
130+
/// Check different combinations of input values.
131+
/// </summary>
132+
[Theory]
133+
[MemberData(nameof(ExpectedStringAndValuesData))]
134+
public void Convert_SpecifiedValues_ValueConverted(string expectedConvertedValue, params object[] values)
135+
{
136+
// ARRANGE.
137+
var converter = new StringFormatConverter();
138+
139+
// ACT.
140+
var convertedValue = converter.Convert(values, typeof(string), null, CultureInfo.InvariantCulture);
141+
142+
// ASSERT.
143+
Assert.Equal(expectedConvertedValue, convertedValue);
144+
}
145+
146+
/// <summary>
147+
/// Check that ConvertBack just return null value without throw exceptions.
148+
/// </summary>
149+
[Fact]
150+
public void ConvertBack_AnyValue_ReturnsNull()
151+
{
152+
// ARRANGE.
153+
var converter = new StringFormatConverter();
154+
155+
// ACT.
156+
var originalValues = converter.ConvertBack(CONVERTED_VALUE, new []{ typeof(string) }, null, CultureInfo.InvariantCulture);
157+
158+
// ASSERT.
159+
Assert.Null(originalValues);
160+
}
161+
}
162+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
<Project Sdk="Microsoft.NET.Sdk">
2+
3+
<PropertyGroup>
4+
5+
<IsPackable>false</IsPackable>
6+
7+
<TargetFramework>netcoreapp3.1</TargetFramework>
8+
</PropertyGroup>
9+
10+
<ItemGroup>
11+
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.5.0" />
12+
<PackageReference Include="xunit" Version="2.4.0" />
13+
<PackageReference Include="xunit.runner.visualstudio" Version="2.4.0" />
14+
<PackageReference Include="coverlet.collector" Version="1.2.0" />
15+
</ItemGroup>
16+
17+
<ItemGroup>
18+
<ProjectReference Include="..\..\src\WPFLocalizeExtension.csproj" />
19+
</ItemGroup>
20+
21+
</Project>

0 commit comments

Comments
 (0)