Fix styles as Styles (not ResourceDictionary); replace DataTriggers with width converters; add IsZeroConverter; remove unsupported properties; fix ancestor command bindings; bind SplitView props directly.
This commit is contained in:
		
					parent
					
						
							
								c624127bf9
							
						
					
				
			
			
				commit
				
					
						1667fc8b3d
					
				
			
		
					 9 changed files with 146 additions and 101 deletions
				
			
		|  | @ -1,10 +1,12 @@ | ||||||
| <ResourceDictionary xmlns="https://github.com/avaloniaui" | <Styles xmlns="https://github.com/avaloniaui" | ||||||
|         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> |         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> | ||||||
|  |   <Styles.Resources> | ||||||
|     <!-- Android overrides: larger, touch-first --> |     <!-- Android overrides: larger, touch-first --> | ||||||
|     <x:Double x:Key="FontSizeBase">16</x:Double> |     <x:Double x:Key="FontSizeBase">16</x:Double> | ||||||
|     <x:Double x:Key="FontSizeLarge">18</x:Double> |     <x:Double x:Key="FontSizeLarge">18</x:Double> | ||||||
|     <x:Double x:Key="IconSizeM">24</x:Double> |     <x:Double x:Key="IconSizeM">24</x:Double> | ||||||
|     <x:Double x:Key="LineHeightBase">1.35</x:Double> |     <x:Double x:Key="LineHeightBase">1.35</x:Double> | ||||||
|  |   </Styles.Resources> | ||||||
| 
 | 
 | ||||||
|   <!-- Thin overlay-like scrollbars --> |   <!-- Thin overlay-like scrollbars --> | ||||||
|   <Style Selector="ScrollBar:vertical"> |   <Style Selector="ScrollBar:vertical"> | ||||||
|  | @ -13,5 +15,4 @@ | ||||||
|   <Style Selector="ScrollBar:horizontal"> |   <Style Selector="ScrollBar:horizontal"> | ||||||
|     <Setter Property="Height" Value="8"/> |     <Setter Property="Height" Value="8"/> | ||||||
|   </Style> |   </Style> | ||||||
| </ResourceDictionary> | </Styles> | ||||||
| 
 |  | ||||||
|  |  | ||||||
|  | @ -1,9 +1,11 @@ | ||||||
| <ResourceDictionary xmlns="https://github.com/avaloniaui" | <Styles xmlns="https://github.com/avaloniaui" | ||||||
|         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> |         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> | ||||||
|  |   <Styles.Resources> | ||||||
|     <!-- Browser overrides: middle ground --> |     <!-- Browser overrides: middle ground --> | ||||||
|     <x:Double x:Key="FontSizeBase">15</x:Double> |     <x:Double x:Key="FontSizeBase">15</x:Double> | ||||||
|     <x:Double x:Key="IconSizeM">22</x:Double> |     <x:Double x:Key="IconSizeM">22</x:Double> | ||||||
|     <x:Double x:Key="LineHeightBase">1.3</x:Double> |     <x:Double x:Key="LineHeightBase">1.3</x:Double> | ||||||
|  |   </Styles.Resources> | ||||||
| 
 | 
 | ||||||
|   <!-- Slightly thinner scrollbars on web --> |   <!-- Slightly thinner scrollbars on web --> | ||||||
|   <Style Selector="ScrollBar:vertical"> |   <Style Selector="ScrollBar:vertical"> | ||||||
|  | @ -12,5 +14,4 @@ | ||||||
|   <Style Selector="ScrollBar:horizontal"> |   <Style Selector="ScrollBar:horizontal"> | ||||||
|     <Setter Property="Height" Value="8"/> |     <Setter Property="Height" Value="8"/> | ||||||
|   </Style> |   </Style> | ||||||
| </ResourceDictionary> | </Styles> | ||||||
| 
 |  | ||||||
|  |  | ||||||
|  | @ -1,9 +1,11 @@ | ||||||
| <ResourceDictionary xmlns="https://github.com/avaloniaui" | <Styles xmlns="https://github.com/avaloniaui" | ||||||
|         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> |         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> | ||||||
|  |   <Styles.Resources> | ||||||
|     <!-- Desktop overrides: slightly denser --> |     <!-- Desktop overrides: slightly denser --> | ||||||
|     <x:Double x:Key="FontSizeBase">13</x:Double> |     <x:Double x:Key="FontSizeBase">13</x:Double> | ||||||
|     <x:Double x:Key="LineHeightBase">1.2</x:Double> |     <x:Double x:Key="LineHeightBase">1.2</x:Double> | ||||||
|     <x:Double x:Key="IconSizeM">20</x:Double> |     <x:Double x:Key="IconSizeM">20</x:Double> | ||||||
|  |   </Styles.Resources> | ||||||
| 
 | 
 | ||||||
|   <!-- Scrollbars a bit thicker on desktop --> |   <!-- Scrollbars a bit thicker on desktop --> | ||||||
|   <Style Selector="ScrollBar:vertical"> |   <Style Selector="ScrollBar:vertical"> | ||||||
|  | @ -12,5 +14,4 @@ | ||||||
|   <Style Selector="ScrollBar:horizontal"> |   <Style Selector="ScrollBar:horizontal"> | ||||||
|     <Setter Property="Height" Value="12"/> |     <Setter Property="Height" Value="12"/> | ||||||
|   </Style> |   </Style> | ||||||
| </ResourceDictionary> | </Styles> | ||||||
| 
 |  | ||||||
|  |  | ||||||
							
								
								
									
										25
									
								
								src/AdvancedCalculator/Converters/IsZeroConverter.cs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								src/AdvancedCalculator/Converters/IsZeroConverter.cs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,25 @@ | ||||||
|  | using System; | ||||||
|  | using System.Globalization; | ||||||
|  | using Avalonia.Data.Converters; | ||||||
|  | 
 | ||||||
|  | namespace AdvancedCalculator.Converters; | ||||||
|  | 
 | ||||||
|  | public class IsZeroConverter : IValueConverter | ||||||
|  | { | ||||||
|  |     public static readonly IsZeroConverter Instance = new(); | ||||||
|  | 
 | ||||||
|  |     public object? Convert(object? value, Type targetType, object? parameter, CultureInfo? culture) | ||||||
|  |     { | ||||||
|  |         if (value is int i) | ||||||
|  |             return i == 0; | ||||||
|  |         if (value is long l) | ||||||
|  |             return l == 0L; | ||||||
|  |         if (value is double d) | ||||||
|  |             return Math.Abs(d) < double.Epsilon; | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public object? ConvertBack(object? value, Type targetType, object? parameter, CultureInfo? culture) | ||||||
|  |         => throw new NotSupportedException(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @ -0,0 +1,29 @@ | ||||||
|  | using System; | ||||||
|  | using System.Globalization; | ||||||
|  | using Avalonia.Data.Converters; | ||||||
|  | 
 | ||||||
|  | namespace AdvancedCalculator.Converters; | ||||||
|  | 
 | ||||||
|  | public class WidthToPaneOpenConverter : IValueConverter | ||||||
|  | { | ||||||
|  |     public static readonly WidthToPaneOpenConverter Instance = new(); | ||||||
|  | 
 | ||||||
|  |     public object? Convert(object? value, Type targetType, object? parameter, CultureInfo? culture) | ||||||
|  |     { | ||||||
|  |         if (value is double width) | ||||||
|  |         { | ||||||
|  |             double threshold = 640; | ||||||
|  |             if (parameter is double p) | ||||||
|  |                 threshold = p; | ||||||
|  |             else if (parameter is string s && double.TryParse(s, NumberStyles.Number, CultureInfo.InvariantCulture, out var parsed)) | ||||||
|  |                 threshold = parsed; | ||||||
|  | 
 | ||||||
|  |             return width >= threshold; | ||||||
|  |         } | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public object? ConvertBack(object? value, Type targetType, object? parameter, CultureInfo? culture) | ||||||
|  |         => throw new NotSupportedException(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @ -0,0 +1,30 @@ | ||||||
|  | using System; | ||||||
|  | using System.Globalization; | ||||||
|  | using Avalonia.Controls; | ||||||
|  | using Avalonia.Data.Converters; | ||||||
|  | 
 | ||||||
|  | namespace AdvancedCalculator.Converters; | ||||||
|  | 
 | ||||||
|  | public class WidthToSplitViewModeConverter : IValueConverter | ||||||
|  | { | ||||||
|  |     public static readonly WidthToSplitViewModeConverter Instance = new(); | ||||||
|  | 
 | ||||||
|  |     public object? Convert(object? value, Type targetType, object? parameter, CultureInfo? culture) | ||||||
|  |     { | ||||||
|  |         if (value is double width) | ||||||
|  |         { | ||||||
|  |             double threshold = 640; | ||||||
|  |             if (parameter is double p) | ||||||
|  |                 threshold = p; | ||||||
|  |             else if (parameter is string s && double.TryParse(s, NumberStyles.Number, CultureInfo.InvariantCulture, out var parsed)) | ||||||
|  |                 threshold = parsed; | ||||||
|  | 
 | ||||||
|  |             return width < threshold ? SplitViewDisplayMode.Overlay : SplitViewDisplayMode.Inline; | ||||||
|  |         } | ||||||
|  |         return SplitViewDisplayMode.Inline; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public object? ConvertBack(object? value, Type targetType, object? parameter, CultureInfo? culture) | ||||||
|  |         => throw new NotSupportedException(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @ -1,4 +1,4 @@ | ||||||
| <ResourceDictionary xmlns="https://github.com/avaloniaui" | <Styles xmlns="https://github.com/avaloniaui" | ||||||
|         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> |         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> | ||||||
|   <!-- Shared scrolling defaults --> |   <!-- Shared scrolling defaults --> | ||||||
|   <Style Selector="ListBox"> |   <Style Selector="ListBox"> | ||||||
|  | @ -6,10 +6,6 @@ | ||||||
|     <Setter Property="ScrollViewer.HorizontalScrollBarVisibility" Value="Disabled"/> |     <Setter Property="ScrollViewer.HorizontalScrollBarVisibility" Value="Disabled"/> | ||||||
|   </Style> |   </Style> | ||||||
| 
 | 
 | ||||||
|   <Style Selector="ScrollViewer"> |  | ||||||
|     <Setter Property="CanScrollHorizontally" Value="False"/> |  | ||||||
|   </Style> |  | ||||||
| 
 |  | ||||||
|   <!-- Thin-ish scrollbars by default; platform overrides will adjust --> |   <!-- Thin-ish scrollbars by default; platform overrides will adjust --> | ||||||
|   <Style Selector="ScrollBar:vertical"> |   <Style Selector="ScrollBar:vertical"> | ||||||
|     <Setter Property="Width" Value="10"/> |     <Setter Property="Width" Value="10"/> | ||||||
|  | @ -17,5 +13,4 @@ | ||||||
|   <Style Selector="ScrollBar:horizontal"> |   <Style Selector="ScrollBar:horizontal"> | ||||||
|     <Setter Property="Height" Value="10"/> |     <Setter Property="Height" Value="10"/> | ||||||
|   </Style> |   </Style> | ||||||
| </ResourceDictionary> | </Styles> | ||||||
| 
 |  | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| <ResourceDictionary xmlns="https://github.com/avaloniaui" | <Styles xmlns="https://github.com/avaloniaui" | ||||||
|         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> |         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> | ||||||
|  |   <Styles.Resources> | ||||||
|     <!-- Shared typography resources --> |     <!-- Shared typography resources --> | ||||||
|     <x:Double x:Key="FontSizeSmall">12</x:Double> |     <x:Double x:Key="FontSizeSmall">12</x:Double> | ||||||
|     <x:Double x:Key="FontSizeBase">14</x:Double> |     <x:Double x:Key="FontSizeBase">14</x:Double> | ||||||
|  | @ -13,6 +14,7 @@ | ||||||
|     <x:Double x:Key="LineHeightTight">1.1</x:Double> |     <x:Double x:Key="LineHeightTight">1.1</x:Double> | ||||||
|     <x:Double x:Key="LineHeightBase">1.25</x:Double> |     <x:Double x:Key="LineHeightBase">1.25</x:Double> | ||||||
|     <x:Double x:Key="LineHeightRelaxed">1.4</x:Double> |     <x:Double x:Key="LineHeightRelaxed">1.4</x:Double> | ||||||
|  |   </Styles.Resources> | ||||||
| 
 | 
 | ||||||
|   <!-- Defaults for text --> |   <!-- Defaults for text --> | ||||||
|   <Style Selector="TextBlock"> |   <Style Selector="TextBlock"> | ||||||
|  | @ -26,5 +28,4 @@ | ||||||
|     <Setter Property="TextElement.FontSize" Value="{DynamicResource FontSizeSmall}"/> |     <Setter Property="TextElement.FontSize" Value="{DynamicResource FontSizeSmall}"/> | ||||||
|     <Setter Property="Opacity" Value="0.85"/> |     <Setter Property="Opacity" Value="0.85"/> | ||||||
|   </Style> |   </Style> | ||||||
| </ResourceDictionary> | </Styles> | ||||||
| 
 |  | ||||||
|  |  | ||||||
|  | @ -11,34 +11,16 @@ | ||||||
|   <UserControl.Resources> |   <UserControl.Resources> | ||||||
|     <conv:BoolToGridLengthConverter x:Key="BoolToGridLengthConverter" /> |     <conv:BoolToGridLengthConverter x:Key="BoolToGridLengthConverter" /> | ||||||
|     <conv:WidthToBooleanConverter x:Key="WidthToBooleanConverter" /> |     <conv:WidthToBooleanConverter x:Key="WidthToBooleanConverter" /> | ||||||
|  |     <conv:IsZeroConverter x:Key="IsZeroConverter" /> | ||||||
|  |     <conv:WidthToSplitViewModeConverter x:Key="WidthToSplitViewModeConverter" /> | ||||||
|  |     <conv:WidthToPaneOpenConverter x:Key="WidthToPaneOpenConverter" /> | ||||||
|   </UserControl.Resources> |   </UserControl.Resources> | ||||||
|   <!-- Replace columns with a responsive SplitView --> |   <!-- Replace columns with a responsive SplitView --> | ||||||
|   <SplitView x:Name="RootSplit" |   <SplitView x:Name="RootSplit" | ||||||
|              OpenPaneLength="320" |              OpenPaneLength="320" | ||||||
|              CompactPaneLength="0"> |              CompactPaneLength="0" | ||||||
|     <!-- Responsive behavior: set DisplayMode and IsPaneOpen based on width --> |              DisplayMode="{Binding $parent[Window].Bounds.Width, Converter={StaticResource WidthToSplitViewModeConverter}, ConverterParameter=640}" | ||||||
|     <SplitView.Styles> |              IsPaneOpen="{Binding $parent[Window].Bounds.Width, Converter={StaticResource WidthToPaneOpenConverter}, ConverterParameter=640}"> | ||||||
|       <Style Selector="SplitView#RootSplit"> |  | ||||||
|         <Style.Triggers> |  | ||||||
|           <!-- Narrow: Overlay mode, pane closed by default --> |  | ||||||
|           <DataTrigger Value="True" |  | ||||||
|                        Binding="{Binding $parent[Window].Bounds.Width, |  | ||||||
|                                          Converter={StaticResource WidthToBooleanConverter}, |  | ||||||
|                                          ConverterParameter=640}"> |  | ||||||
|             <Setter Property="DisplayMode" Value="Overlay" /> |  | ||||||
|             <Setter Property="IsPaneOpen" Value="False" /> |  | ||||||
|           </DataTrigger> |  | ||||||
|           <!-- Wide: Inline mode, pane open --> |  | ||||||
|           <DataTrigger Value="False" |  | ||||||
|                        Binding="{Binding $parent[Window].Bounds.Width, |  | ||||||
|                                          Converter={StaticResource WidthToBooleanConverter}, |  | ||||||
|                                          ConverterParameter=640}"> |  | ||||||
|             <Setter Property="DisplayMode" Value="Inline" /> |  | ||||||
|             <Setter Property="IsPaneOpen" Value="True" /> |  | ||||||
|           </DataTrigger> |  | ||||||
|         </Style.Triggers> |  | ||||||
|       </Style> |  | ||||||
|     </SplitView.Styles> |  | ||||||
| 
 | 
 | ||||||
|     <!-- Left pane: Variables + Functions --> |     <!-- Left pane: Variables + Functions --> | ||||||
|     <SplitView.Pane> |     <SplitView.Pane> | ||||||
|  | @ -50,11 +32,11 @@ | ||||||
| 
 | 
 | ||||||
|         <!-- Variables list with empty-state hint overlay --> |         <!-- Variables list with empty-state hint overlay --> | ||||||
|         <Grid Grid.Row="0"> |         <Grid Grid.Row="0"> | ||||||
|           <ListBox ItemsSource="{Binding Variables}" SelectedIndex="-1" HorizontalContentAlignment="Stretch" |           <ListBox ItemsSource="{Binding Variables}" SelectedIndex="-1" | ||||||
|                    AutomationProperties.Name="Variables list"> |                    AutomationProperties.Name="Variables list"> | ||||||
|             <ListBox.ItemTemplate> |             <ListBox.ItemTemplate> | ||||||
|               <DataTemplate x:DataType="m:VariableItem"> |               <DataTemplate x:DataType="m:VariableItem"> | ||||||
|                 <Button Command="{Binding DataContext.InsertVariableCommand, RelativeSource={RelativeSource AncestorType=UserControl}}" |                 <Button Command="{Binding $parent[UserControl].DataContext.InsertVariableCommand}" | ||||||
|                         CommandParameter="{Binding VariableName}" |                         CommandParameter="{Binding VariableName}" | ||||||
|                         Background="Transparent" BorderThickness="0" Padding="8" MinHeight="44" |                         Background="Transparent" BorderThickness="0" Padding="8" MinHeight="44" | ||||||
|                         AutomationProperties.Name="Insert variable"> |                         AutomationProperties.Name="Insert variable"> | ||||||
|  | @ -72,29 +54,19 @@ | ||||||
|             </ListBox.ItemTemplate> |             </ListBox.ItemTemplate> | ||||||
|           </ListBox> |           </ListBox> | ||||||
|           <TextBlock Text="No variables yet. Define with: x = 5" |           <TextBlock Text="No variables yet. Define with: x = 5" | ||||||
|                      IsVisible="False" |                      IsVisible="{Binding Variables.Count, Converter={StaticResource IsZeroConverter}}" | ||||||
|                      IsHitTestVisible="False" |                      IsHitTestVisible="False" | ||||||
|                      Opacity="0.6" |                      Opacity="0.6" | ||||||
|                      HorizontalAlignment="Center" VerticalAlignment="Center"> |                      HorizontalAlignment="Center" VerticalAlignment="Center" /> | ||||||
|             <TextBlock.Styles> |  | ||||||
|               <Style Selector="TextBlock"> |  | ||||||
|                 <Style.Triggers> |  | ||||||
|                   <DataTrigger Binding="{Binding Variables.Count}" Value="0"> |  | ||||||
|                     <Setter Property="IsVisible" Value="True" /> |  | ||||||
|                   </DataTrigger> |  | ||||||
|                 </Style.Triggers> |  | ||||||
|               </Style> |  | ||||||
|             </TextBlock.Styles> |  | ||||||
|           </TextBlock> |  | ||||||
|         </Grid> |         </Grid> | ||||||
| 
 | 
 | ||||||
|         <!-- Function definitions --> |         <!-- Function definitions --> | ||||||
|         <ListBox Grid.Row="1" BorderThickness="0" HorizontalContentAlignment="Stretch" |         <ListBox Grid.Row="1" BorderThickness="0" | ||||||
|                  ItemsSource="{x:Static m:FunctionDefinitionItem.DefinedFunctions}" |                  ItemsSource="{x:Static m:FunctionDefinitionItem.DefinedFunctions}" | ||||||
|                  AutomationProperties.Name="Functions list"> |                  AutomationProperties.Name="Functions list"> | ||||||
|           <ListBox.ItemTemplate> |           <ListBox.ItemTemplate> | ||||||
|             <DataTemplate x:DataType="m:FunctionDefinitionItem"> |             <DataTemplate x:DataType="m:FunctionDefinitionItem"> | ||||||
|               <Button Command="{Binding DataContext.InsertFunctionCommand, RelativeSource={RelativeSource AncestorType=UserControl}}" |               <Button Command="{Binding $parent[UserControl].DataContext.InsertFunctionCommand}" | ||||||
|                       CommandParameter="{Binding FunctionName}" |                       CommandParameter="{Binding FunctionName}" | ||||||
|                       Background="Transparent" BorderThickness="0" Padding="8" MinHeight="44" |                       Background="Transparent" BorderThickness="0" Padding="8" MinHeight="44" | ||||||
|                       AutomationProperties.Name="Insert function"> |                       AutomationProperties.Name="Insert function"> | ||||||
|  | @ -143,24 +115,14 @@ | ||||||
|             </ListBox.ItemTemplate> |             </ListBox.ItemTemplate> | ||||||
|           </ListBox> |           </ListBox> | ||||||
|           <TextBlock Text="No history yet. Enter an expression and Evaluate." |           <TextBlock Text="No history yet. Enter an expression and Evaluate." | ||||||
|                      IsVisible="False" |                      IsVisible="{Binding History.Count, Converter={StaticResource IsZeroConverter}}" | ||||||
|                      IsHitTestVisible="False" |                      IsHitTestVisible="False" | ||||||
|                      Opacity="0.6" |                      Opacity="0.6" | ||||||
|                      HorizontalAlignment="Center" VerticalAlignment="Center"> |                      HorizontalAlignment="Center" VerticalAlignment="Center" /> | ||||||
|             <TextBlock.Styles> |  | ||||||
|               <Style Selector="TextBlock"> |  | ||||||
|                 <Style.Triggers> |  | ||||||
|                   <DataTrigger Binding="{Binding History.Count}" Value="0"> |  | ||||||
|                     <Setter Property="IsVisible" Value="True" /> |  | ||||||
|                   </DataTrigger> |  | ||||||
|                 </Style.Triggers> |  | ||||||
|               </Style> |  | ||||||
|             </TextBlock.Styles> |  | ||||||
|           </TextBlock> |  | ||||||
|         </Grid> |         </Grid> | ||||||
| 
 | 
 | ||||||
|         <!-- Input Row --> |         <!-- Input Row --> | ||||||
|         <Grid Grid.Row="1" ColumnDefinitions="Auto,Auto,*,Auto" Margin="4" ColumnSpacing="6"> |         <Grid Grid.Row="1" ColumnDefinitions="Auto,Auto,*,Auto" Margin="4"> | ||||||
|           <!-- Pane toggle: visible on narrow only --> |           <!-- Pane toggle: visible on narrow only --> | ||||||
|           <ToggleButton Margin="0,0,6,0" |           <ToggleButton Margin="0,0,6,0" | ||||||
|                         IsVisible="{Binding $parent[Window].Bounds.Width, |                         IsVisible="{Binding $parent[Window].Bounds.Width, | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue