Skip to content

Latest commit

 

History

History
370 lines (269 loc) · 24.3 KB

README.md

File metadata and controls

370 lines (269 loc) · 24.3 KB

Math Converter: A XAML Converter that does it all.

Installation:

MathConverter is available on Nuget. There are three packages:

Nuget Package UI Framework Target Frameworks
MathConverter WPF
  • .NET Framework 3.5+
  • .NET Core 3.0+
  • .NET 5.0 - 8.0
MathConverter.XamarinForms Xamarin.Forms
  • .NET Standard 1.0+
  • .NET Core 3.0+
  • .NET 5.0 - 8.0
  • Xamarin.iOS 10+
  • MonoAndroid 10+
  • UAP 10.0
  • Xamarin.Mac 2.0+
MathConverter.Maui .NET MAUI
  • .NET 7.0 - 8.0
  • Windows
  • MacCatalyst
  • iOS
  • Android

To install MathConverter, run the one of the following commands in the Package Manager Console:

PM> Install-Package MathConverter
PM> Install-Package MathConverter.XamarinForms
PM> Install-Package MathConverter.Maui

What is MathConverter?

MathConverter allows you to do Math in XAML.

MathConverter is a powerful Binding converter that allows you to specify how to perform conversions directly in XAML, without needing to define a new IValueConverter in C# for every single conversion.

Getting Started:

It's as easy as 1-2-3.

1) Install the Nuget package.

2) Add a MathConverter resource.

<Application.Resources>
    <math:MathConverter x:Key="Math" />
</Application.Resources>

The math namespace is defined as follows*:

xmlns:math="http://hexinnovation.com/math"

3) Do Math. Now, you can use MathConverter on any Binding. Specify a ConverterParameter to specify the rules of the conversion.

*Note: In some targets (e.g. .NET Standard 1.0), you might have to define the math namespace as xmlns:math="clr-namespace:HexInnovation;assembly=MathConverter.XamarinForms"

Example: Rounded Rectangle

Suppose we want to make a rounded rectangle. If we create a Border and bind bind its CornerRadius to its own ActualHeight, we end up with a flattened oval (Full XAML file):

<Border CornerRadius={Binding ActualHeight}" … />

If CornerRadius = ActualHeight, we get a flattened oval

We can use MathConverter to instead bind to ActualHeight / 2 (Full XAML file):

<Border CornerRadius="{Binding ActualHeight, ConverterParameter=x/2, Converter={StaticResource Math}}" … />

If CornerRadius = ActualHeight / 2, we get a rounded rectangle

The simple conversion of ActualHeight / 2 works well, as long as the rectangle is wider than it is tall. If we need to make a rounded rectangle of an arbitrary size, we need to use a MultiBinding to set the CornerRadius to the smaller of the ActualWidth and the ActualHeight divided by two (Full XAML file):

<Border.CornerRadius>
    <MultiBinding ConverterParameter="Min(x,y)/2" Converter="{StaticResource Math}">
        <Binding Path="ActualHeight" />
        <Binding Path="ActualWidth" />
    </MultiBinding>
</Border.CornerRadius>

Alternatively, we can use the math:Convert MarkupExtension, which is more elegant syntax for creating a MultiBinding. Under the covers, this works by actually creating a MultiBinding for us:

CornerRadius="{math:Convert 'Min(x,y)/2', x={Binding ActualHeight}, y={Binding ActualWidth}}"

If CornerRadius = ActualHeight / 2, we get a rounded rectangle

Note: Instead of using the Min function, we could also use the ternary operator: ConverterParameter = "(x > y ? y : x) / 2", but that's a little cumbersome to add to XAML.

Note: MathConverter can take any number of Bindings. The first binding's value can be accessed by x or [0], the second can be accessed by y or [1], and the third can be accessed by z or [2]. Any value beyond the third can be accessed only by its index: [3], [4], etc.

The math:Convert MarkupExtension is limited to ten variables: x, y, z, and Var3 through Var9. If you need more than ten variables, you're probably doing something wrong. But if you insist on using MathConverter with an obsene number of parameters, you'll have to use a MultiBinding.

math:Convert is a wrapper around MultiBinding, not Binding. If you're binding to only one variable, there's considerably less overhead if you simply use a Binding with a MathConverter.

Example: Different margins on different sides

You can specify multiple values for types like CornerRadius, Thickness, Size, Point, and Rect, just like in normal XAML. For example, we can specify different values for vertical/horizontal margins (Full XAML file):

<Rectangle Fill="Green"Margin="{Binding Source={StaticResource Margin}, ConverterParameter=0;x, Converter={StaticResource Math}}" />

Three rectangles with 20 pixels of margin between them. The middle rectangle has top and bottom margins of 20, but left and right margins of 0.

Note: To facilitate entering multiple values into XAML, MathConverter, commas and semicolons are equivalent. We can use either one as separators between the values. So the following margins are equivalent:

Example: No Parameter at all

The ConverterParameter is optional. When it is omitted, MathConverter will attempt to convert all of the binding values string-joined with a comma.

In this example, we create two different GridLength (margin) values: one by specifying Margin for all four sides, and the other by specifying Margin for horizontal margins, and SmallMargin for vertical margins (Full XAML file).

The first essentially converts as Margin="20". The second converts as Margin="20,10".

<Border BorderThickness="1" BorderBrush="Black" Grid.Row="2" Margin="{Binding Source={StaticResource Margin}, Converter={StaticResource Math}}">
    <Border BorderThickness="1" BorderBrush="Red">
        <Border.Margin>
            <MultiBinding Converter="{StaticResource Math}">
                <Binding Source="{StaticResource Margin}" />
                <Binding Source="{StaticResource SmallMargin}" />
            </MultiBinding>
        </Border.Margin>
    </Border>
</Border>

Alternatively, we could use the Convert MarkupExtension, which creates a MultiBinding for us:

<BorderMargin="{math:Convert x={Binding Margin}, y={Binding SmallMargin}}" />

We can convert one, two, or four values to GridLength without specifying a ConverterParameter.

Example: Boolean to Visibility

Suppose you want to show a Control based on a Boolean condition. You can simply bind the Visibility parameter and use the ternary operator to convert the boolean to a Visibility (Full XAML file):

<TextBox Visibility="{Binding IsChecked, ElementName=CheckBox, ConverterParameter='x ? `Visible` : `Collapsed`', Converter={StaticResource Math}}" />

When we toggle the CheckBox, the TextBox appears and disappears

There's a lot going on in this conversion, so let's take this one slowly.

The conversion parameter is x ? `Visible` : `Collapsed`. MathConverter allows us to input strings very similarly to C#. To more easily facilitate adding strings to XAML, we can use " (double quote), ' (single quote), or ` (grave) characters to start and end a string.

Suppose that x evaluates to true (CheckBox.IsChecked is true). Then, x ? `Visible` : `Collapsed` would evaluate to a System.String of "Visible". Since we're binding to a property of Visibility, MathConverter later converts this value to Visibility.Visible for us.

Note: You can backslash-escape characters such as \t and \n in strings, just like C#. Additionally, you can backslash-escape double quotes, single quotes, and grave characters.

Note: All strings must start and end with the same character. So a ConverterParameter of 'Hello, world" would throw an exception because ' and " do not match, whereas ConverterParameters of `Hello, world`, "Hello, world", and 'Hello, world' are equivalent.

Example: Interpolated Strings

Not only can we include arbitrary strings in the ConverterParameter, we can also use interpolated strings to format arbitrary strings (Full XAML file):

<TextBlock Text="{Binding NumClicks, ConverterParameter='$`You have clicked the button {x} time{(x == 1 ? `` : `s`)}.`', Converter={StaticResource Math}}" />

You have clicked the button x time(s), where x increments with each click

Interpolated strings work the same way as they do in C#. The same rules above apply: a string must start and end with the same character. For example, the following are all valid interpolated strings: $'Coordinates: ({x:N2},{y:N2}).', $"The weather outside is {x}.", $`Progress: {x:P} complete`, whereas the string $'Invalid" would throw an exception since ' and " do not match.

Note: Just like in C#, an interpolated string is just a wrapper around a call to the Format function (which uses string.Format), so the converter parameter $`Hello, {x}` is equivalent to Format('Hello, {0}', x).

Functions

We've already alluded to Min and Format. There are many more functions built into MathConverter, and you can always add your own functions (see the "Custom Functions" section). For now, we're just going to cover some of the functions built into MathConverter.

Functions are case-sensitive (They were not case sensitive in version 1.x).

Functions include:

  • Now() returns System.DateTime.Now
  • UnsetValue() returns DependencyProperty.UnsetValue or BindableProperty.UnsetValue
  • Cos(x), Sin(x), Tan(x), Abs(x), Acos(x)/ArcCos(x), Asin(x)/ArcSin(x), Atan(x)/ArcTan(x), Ceil(x)/Ceiling(x), Floor(x), Sqrt(x), Log(x, y), Atan2(x, y)/ArcTan2(x, y), Round(x)/Round(x, y) all behave like their counterparts in System.Math. They return null if at least one argument is null.
  • Deg(x)/Degrees(x) returns x / pi * 180
  • Rad(x)/Radians(x) returns x / 180 * pi
  • ToLower(x)/LCase(x) returns $"{x}".ToLower()
  • ToUpper(x)/UCase(x) returns $"{x}".ToUpper()
  • TryParseDouble(x) will attempt to cast/convert x to double or cast/convert x to string and parse it to double. The function returns null if it fails to convert the input.
  • StartsWith(x, y) will return true or false if it can cast/convert x to string, based on if x starts with y or $"{y}". If x is not a string or $"{y}".Length is 0, the function returns null instead.
  • EndsWith(x, y) behaves the same way as StartsWith except it detects if x ends with y.
  • Contains(x, y) is a bit different. x can be an IEnumerable, in which case we check to see if it contains y, or if x is a string, the function checks if x contains $"{y}". If $"{y}".Length is zero (but notably, not if y == ""), then the function returns null instead.
  • IsNull(x, y)/IfNull(x, y) are equivalent to x ?? y
  • And(), Or(), and Nor() each accept an arbitrary number of functions. They use reflection to call the logical operators UnaryNot (Nor(x, y) evaluates as !Or(x, y)), BitwiseAnd, and BitwiseOr. This means that we can accept and return non-boolean values, provided that their types would compile with &&, ||, and ! in C#. We only evaluate as many parameters as we need to. For example, And(…) will evaulate parameters only until it encounters a false value, in which case it will return the false value.
  • Max(), Min(), and Avg()/Average() ignore values that can't be converted to double, and return null if no they do not encounter any numeric values.
  • Format() simply returns string.Format
  • Concat() simply returns string.Concat.
  • Join() simply returns string.Join.
  • GetType(x) simply returns x?.GetType()
  • ConvertType(x, y) will do whatever it can to convert x to type y. If y is not a Type or x cannot be converted, the function returns x instead. Because TypeConverters are inconsistent, we always use InvariantCulture when converting.
  • EnumEquals(x, y) will see if two enum values are equal. Example use cases: EnumEquals(x, `Visible`), EnumEquals(`Visible`, x). If two enum values are different types are compared, EnumEquals will return false, even if x.Equals(y) is true for the same inputs in C#.
  • Throw() will throw an exception when evaluated. The exception contains helpful information for debugging issues with a conversion.
  • TryCatch() takes two or more arguments, and returns immediately as soon as it finds an argument that does not throw an exception. If every argument throws an exception, TryCatch() will not catch the last exception.

Using these operators, you can do very powerful things. One such example (Full XAML file):

<TextBlock Text="{Binding Source={x:Type sys:TimeSpan}, ConverterParameter='$`Six hours from now, the time will be {Now() + ConvertType(`6:00:00`, x):h\':\'mm\':\'ss tt}`', Converter={StaticResource Math}}" />

We use ConvertType to convert "6:00:00" from string to TimeSpan, then add that TimeSpan to the current time, and format it with the format string "h:mm:ss tt".

Six hours from now, the time will be 12:35:09 AM

Custom Functions

MathConverter's built-in functions are implemented in CustomFunctions.cs. Those classes can be used as examples to follow to create your own custom functions. This allows you to effectively extend MathConverter to do whatever you want.

The main window of our demo app is a perfect example (Full XAML file).

We have a ListBox with Types added.

<ListBox>
    <ListBox.Items>
        <x:Type TypeName="demos:FlattenedOval" />
        <x:Type TypeName="demos:WideRoundedRectangle" />
        <x:Type TypeName="demos:TrueRoundedRectangle" />
        <!-- More Types -->
    </ListBox.Items>

    <!-- More stuff -->
</ListBox>

The ListBox uses a DataTemplate to show a TextBox for each item. We use the custom function GetWindowTitle() to convert the Types to a display value.

<TextBlock Text="{Binding ConverterParameter='GetWindowTitle(x)', Converter={StaticResource Math}}" />

The GetWindowTitle function is added to MathConverter as follows:

<Window.Resources>
    <math:MathConverter x:Key="Math">
        <!-- "GetWindowTitle" in the parameter will invoke the `GetWindowTitleFunction` function. -->
        <math:CustomFunctionDefinition Name="GetWindowTitle" Function="functions:GetWindowTitleFunction" />
    </math:MathConverter>
</Window.Resources>

GetWindowTitleFunction is defined as follows (Full C# file):

public class GetWindowTitleFunction : OneArgFunction
{
    public override object Evaluate(CultureInfo cultureInfo, object argument)
    {
        return argument is Type t && t.IsAssignableTo(typeof(Window)) ? ((Window)Activator.CreateInstance(t)).Title : null;
    }
}

So, our GetWindowTitleFunction instantiates the Type and get the Title property of the resulting Window.

The GetWindowTitle function converts the Types to display names for us

In this example, GetWindowTitleFunction extends OneArgFunction, but all that matters is that we extend CustomFunction. It is recommended that you implement one of its predefined subclasses:

  • ZeroArgFunction
  • OneArgFunction
  • OneDoubleFunction
  • TwoArgFunction
  • ArbitraryArgFunction

Again, there are plenty of examples in CustomFunctions.cs.

Overriding Built-In Functions

Suppose you don't like how a function is implemented. You can always override the function with your own custom function.

As a concrete example, we can implement a CustomAverageFunction function. This is similar to MathConverter's built-in AverageFunction, except that it rounds each input, instead of simply taking the average.

<TextBlock x:Name="TextBlock" Text="{Binding ConverterParameter='`Average(1, 1.5) returns ` + Average(1, 1.5)', Converter={StaticResource Math}}" />

The Average function changes each time the RadioButton is changed

With a little bit of code-behind, we can remove and replace the Average function with our CustomAverageFunction:

private void RadioButton_Changed(object sender, RoutedEventArgs e)
{
    if (!(FindResource("Math") is HexInnovation.MathConverter math))
        return;

    if (UseStockFunction.IsChecked == true)
    {
        // Go back to the stock function.
        math.CustomFunctions.Clear();
        math.CustomFunctions.RegisterDefaultFunctions();
    }
    else
    {
        // Remove the default Average function and define our own.
        math.CustomFunctions.Remove("Average");
        math.CustomFunctions.Add(CustomFunctionDefinition.Create<MyCustomAverageFunction>("Average"));
    }

    // Tell the TextBlock to refresh its binding again.
    TextBlock?.GetBindingExpression(TextBlock.TextProperty).UpdateTarget();
}

Note: In this example, the built-in Average function is still available with the name Avg. Most functions are not defined with multiple names (see the "Functions" section).

Syntax

MathConverter's ConverterParameter syntax is very similar to C#, so you can generally expect it to behave just like C#. We follow the standard C# rules regarding operator ordering, except as noted below:

  • Since MathConverter is specifically designed to perform math calculations, the caret (^) operator does not perform the XOR operation. Rather, it is an exponent symbol. It uses System.Math.Pow to evaluate expressions, and its precedence is just above multiplicative operations (*, /, and %).
  • The multiplication operator can often be safely ommitted. A ConverterParameter value of xyz will evaluate to x*y*z. The parameter x2y will evaluate to x^2*y (or equivalently, xxy or x*x*y). Similarly, 2x3 is equivalent to 2*x^3 or 2*x*x*x. Note that x(2) is equivalent to x*(2), in the same way that x(y+z) is equivalent to x*(y+z). Note that 1/xy will evaluate to 1/x*y, not to 1/(x*y), as you might expect.
  • MathConverter doesn't support all of the operations that C# does. The following operators are examples of those not supported:
    • Assignment operators (=, +=, &&=, etc)
    • Logical operators (|, &, and ^ as XOR)
      • Note that || and && are supported operators.
    • switch and with expressions are not supported.
    • is and as (since Types are not supported)
    • Bitwise operations (<<, >>, ~) are not supported.
    • The unary operators ++ and -- are not supported, since they change the values of the inputs.
    • Primary operators (x.y, f(x), a[i], new, typeof, checked, unchecked, default, nameof, sizeof) are not supported.

MathConverter uses reflection to evaluate operator calls, so you can use custom types with custom operator implementations and MathConverter will use those operators while converting.

Numeric Types

Generally, MathConverter will favor using double values over other numeric types. When evaluating which operator to call, MathConverter will convert any operands to double, if possible, before calling the operator. If an input is of type char, it will convert to int then convert to double. Where a path to implicitly convert an operand to double exists, MathConverter will convert for you in order to apply an operator that takes numeric inputs.

Hence, supposing x = 1 (an integer), C# would evaluate that 1 + x/2 = 1, since (int)1 / 2 = 0. MathConverter will implicitly converter all variables to doubles. So, the expression 1 + x/2 is evaluated as 1.0 + (double)x/2.0, so MathConverter will return 1.5.

Parser

Each time a conversion must be made, MathConverter must parse and evaluate an expression. When it parses an expression, it reads through the string one character at a time, and returns a syntax tree. The parsing is done in the Parser class. The Parser returns an AbstractSyntaxTree for each comma-separated (or semicolon-separated) value. In an effort to improve efficiency, MathConverter uses a cache to save the AbstractSyntaxTrees for each string it evaluates. Therefore, if you have a lot of conversion strings, it is discouraged to use the same MathConverter instance across your entire application. It is a better idea to use a different MathConverter object for each UserControl, Page, or Window. You can turn off caching on a per-instance basis:

<math:MathConverter x:Key="nocache" UseCache="False" />

Breaking Changes From V1

There are a few breaking changes from version 1.

  • Function names are now case-sensitive.
  • e, pi, null, true, and false keywords are now required to be lower-case.
  • VisibleOrCollapsed and VisibleOrHidden functions were deprecated, and will be removed in a future release. You should change your conversions from VisibleOrCollapsed(x) to x ? `Visible` : `Collapsed`
  • There are several small differences in how/when types are converted. For example, we no longer convert from int to double unless it needs to be used as an operand in an operator such as +, *, etc.