- Update StorageView.xaml: DataGrid top, GridSplitter, chart panel bottom - Add PieChart and CartesianChart with MultiDataTrigger visibility - Add radio buttons for donut/bar chart toggle in left panel - Create BytesLabelConverter for chart tooltip formatting - Add stor.chart.* localization keys in EN and FR resx files Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
200 lines
13 KiB
XML
200 lines
13 KiB
XML
<UserControl x:Class="SharepointToolbox.Views.Tabs.StorageView"
|
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
|
xmlns:loc="clr-namespace:SharepointToolbox.Localization"
|
|
xmlns:conv="clr-namespace:SharepointToolbox.Views.Converters"
|
|
xmlns:lvc="clr-namespace:LiveChartsCore.SkiaSharpView.WPF;assembly=LiveChartsCore.SkiaSharpView.WPF">
|
|
<DockPanel LastChildFill="True">
|
|
<!-- Options panel -->
|
|
<ScrollViewer DockPanel.Dock="Left" Width="240" VerticalScrollBarVisibility="Auto"
|
|
Margin="8,8,4,8">
|
|
<StackPanel>
|
|
<!-- Site URL -->
|
|
<Label Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[lbl.site.url]}" />
|
|
<TextBox Text="{Binding SiteUrl, UpdateSourceTrigger=PropertyChanged}"
|
|
IsEnabled="{Binding IsRunning, Converter={StaticResource InverseBoolConverter}}"
|
|
Height="26" Margin="0,0,0,8"
|
|
ToolTip="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[ph.site.url]}" />
|
|
|
|
<!-- Scan options group -->
|
|
<GroupBox Header="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[grp.scan.opts]}"
|
|
Margin="0,0,0,8">
|
|
<StackPanel Margin="4">
|
|
<CheckBox Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[chk.per.lib]}"
|
|
IsChecked="{Binding PerLibrary}" Margin="0,2" />
|
|
<CheckBox Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[chk.subsites]}"
|
|
IsChecked="{Binding IncludeSubsites}" Margin="0,2" />
|
|
<StackPanel Orientation="Horizontal" Margin="0,4,0,0">
|
|
<Label Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[lbl.folder.depth]}"
|
|
VerticalAlignment="Center" Padding="0,0,4,0" />
|
|
<TextBox Text="{Binding FolderDepth, UpdateSourceTrigger=PropertyChanged}"
|
|
Width="40" Height="22" VerticalAlignment="Center"
|
|
IsEnabled="{Binding IsMaxDepth, Converter={StaticResource InverseBoolConverter}}" />
|
|
</StackPanel>
|
|
<CheckBox Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[chk.max.depth]}"
|
|
IsChecked="{Binding IsMaxDepth}" Margin="0,2" />
|
|
<TextBlock Text="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.note]}"
|
|
TextWrapping="Wrap" FontSize="11" Foreground="#888"
|
|
Margin="0,6,0,0" />
|
|
</StackPanel>
|
|
</GroupBox>
|
|
|
|
<!-- Action buttons -->
|
|
<Button Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[btn.gen.storage]}"
|
|
Command="{Binding RunCommand}"
|
|
Height="28" Margin="0,0,0,4" />
|
|
<Button Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[btn.cancel]}"
|
|
Command="{Binding CancelCommand}"
|
|
Height="28" Margin="0,0,0,8" />
|
|
|
|
<!-- Export group -->
|
|
<GroupBox Header="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[grp.export.fmt]}"
|
|
Margin="0,0,0,8">
|
|
<StackPanel Margin="4">
|
|
<Button Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.rad.csv]}"
|
|
Command="{Binding ExportCsvCommand}"
|
|
Height="26" Margin="0,2" />
|
|
<Button Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.rad.html]}"
|
|
Command="{Binding ExportHtmlCommand}"
|
|
Height="26" Margin="0,2" />
|
|
</StackPanel>
|
|
</GroupBox>
|
|
|
|
<!-- Chart view toggle -->
|
|
<GroupBox Header="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.chart.toggle]}"
|
|
Margin="0,0,0,8">
|
|
<StackPanel Margin="4">
|
|
<RadioButton Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.chart.donut]}"
|
|
IsChecked="{Binding IsDonutChart}" Margin="0,2" />
|
|
<RadioButton Content="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.chart.bar]}"
|
|
IsChecked="{Binding IsDonutChart, Converter={StaticResource InverseBoolConverter}}"
|
|
Margin="0,2" />
|
|
</StackPanel>
|
|
</GroupBox>
|
|
|
|
<!-- Status -->
|
|
<TextBlock Text="{Binding StatusMessage}" TextWrapping="Wrap"
|
|
FontSize="11" Foreground="#555" Margin="0,4" />
|
|
</StackPanel>
|
|
</ScrollViewer>
|
|
|
|
<!-- Right content area: DataGrid on top, Chart on bottom -->
|
|
<Grid Margin="4,8,8,8">
|
|
<Grid.RowDefinitions>
|
|
<RowDefinition Height="*" MinHeight="150" />
|
|
<RowDefinition Height="Auto" />
|
|
<RowDefinition Height="300" MinHeight="200" />
|
|
</Grid.RowDefinitions>
|
|
|
|
<!-- Results DataGrid -->
|
|
<DataGrid x:Name="ResultsGrid"
|
|
Grid.Row="0"
|
|
ItemsSource="{Binding Results}"
|
|
IsReadOnly="True"
|
|
AutoGenerateColumns="False"
|
|
VirtualizingPanel.IsVirtualizing="True"
|
|
VirtualizingPanel.VirtualizationMode="Recycling">
|
|
<DataGrid.Columns>
|
|
<DataGridTemplateColumn Header="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.col.library]}"
|
|
Width="*" MinWidth="160">
|
|
<DataGridTemplateColumn.CellTemplate>
|
|
<DataTemplate>
|
|
<TextBlock Text="{Binding Name}"
|
|
Margin="{Binding IndentLevel, Converter={StaticResource IndentConverter}}"
|
|
VerticalAlignment="Center" />
|
|
</DataTemplate>
|
|
</DataGridTemplateColumn.CellTemplate>
|
|
</DataGridTemplateColumn>
|
|
<DataGridTextColumn Header="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.col.site]}"
|
|
Binding="{Binding SiteTitle}" Width="140" />
|
|
<DataGridTextColumn Header="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.col.files]}"
|
|
Binding="{Binding TotalFileCount, StringFormat=N0}"
|
|
Width="70" ElementStyle="{StaticResource RightAlignStyle}" />
|
|
<DataGridTextColumn Header="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.col.size]}"
|
|
Binding="{Binding TotalSizeBytes, Converter={StaticResource BytesConverter}}"
|
|
Width="100" ElementStyle="{StaticResource RightAlignStyle}" />
|
|
<DataGridTextColumn Header="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.col.versions]}"
|
|
Binding="{Binding VersionSizeBytes, Converter={StaticResource BytesConverter}}"
|
|
Width="110" ElementStyle="{StaticResource RightAlignStyle}" />
|
|
<DataGridTextColumn Header="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.col.lastmod]}"
|
|
Binding="{Binding LastModified, StringFormat=yyyy-MM-dd}"
|
|
Width="110" />
|
|
</DataGrid.Columns>
|
|
</DataGrid>
|
|
|
|
<!-- Splitter between DataGrid and Chart -->
|
|
<GridSplitter Grid.Row="1" Height="5" HorizontalAlignment="Stretch"
|
|
Background="#DDD" ResizeDirection="Rows" />
|
|
|
|
<!-- Chart panel -->
|
|
<Border Grid.Row="2" BorderBrush="#DDD" BorderThickness="1" CornerRadius="4"
|
|
Padding="8" Background="White">
|
|
<Grid>
|
|
<!-- Chart title -->
|
|
<TextBlock Text="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.chart.title]}"
|
|
FontWeight="SemiBold" FontSize="14" VerticalAlignment="Top"
|
|
HorizontalAlignment="Left" Margin="4,0,0,0" />
|
|
|
|
<!-- No data placeholder -->
|
|
<TextBlock HorizontalAlignment="Center" VerticalAlignment="Center"
|
|
Foreground="#888" FontSize="12"
|
|
Text="{Binding Source={x:Static loc:TranslationSource.Instance}, Path=[stor.chart.nodata]}">
|
|
<TextBlock.Style>
|
|
<Style TargetType="TextBlock">
|
|
<Setter Property="Visibility" Value="Collapsed" />
|
|
<Style.Triggers>
|
|
<DataTrigger Binding="{Binding HasChartData}" Value="False">
|
|
<Setter Property="Visibility" Value="Visible" />
|
|
</DataTrigger>
|
|
</Style.Triggers>
|
|
</Style>
|
|
</TextBlock.Style>
|
|
</TextBlock>
|
|
|
|
<!-- Pie/Donut chart wrapper (visible when IsDonutChart=true AND HasChartData=true) -->
|
|
<Grid Margin="4,24,4,4">
|
|
<Grid.Style>
|
|
<Style TargetType="Grid">
|
|
<Setter Property="Visibility" Value="Collapsed" />
|
|
<Style.Triggers>
|
|
<MultiDataTrigger>
|
|
<MultiDataTrigger.Conditions>
|
|
<Condition Binding="{Binding IsDonutChart}" Value="True" />
|
|
<Condition Binding="{Binding HasChartData}" Value="True" />
|
|
</MultiDataTrigger.Conditions>
|
|
<Setter Property="Visibility" Value="Visible" />
|
|
</MultiDataTrigger>
|
|
</Style.Triggers>
|
|
</Style>
|
|
</Grid.Style>
|
|
<lvc:PieChart Series="{Binding PieChartSeries}"
|
|
LegendPosition="Right" />
|
|
</Grid>
|
|
|
|
<!-- Bar chart wrapper (visible when IsDonutChart=false AND HasChartData=true) -->
|
|
<Grid Margin="4,24,4,4">
|
|
<Grid.Style>
|
|
<Style TargetType="Grid">
|
|
<Setter Property="Visibility" Value="Collapsed" />
|
|
<Style.Triggers>
|
|
<MultiDataTrigger>
|
|
<MultiDataTrigger.Conditions>
|
|
<Condition Binding="{Binding IsDonutChart}" Value="False" />
|
|
<Condition Binding="{Binding HasChartData}" Value="True" />
|
|
</MultiDataTrigger.Conditions>
|
|
<Setter Property="Visibility" Value="Visible" />
|
|
</MultiDataTrigger>
|
|
</Style.Triggers>
|
|
</Style>
|
|
</Grid.Style>
|
|
<lvc:CartesianChart Series="{Binding BarChartSeries}"
|
|
XAxes="{Binding BarXAxes}"
|
|
YAxes="{Binding BarYAxes}"
|
|
LegendPosition="Hidden" />
|
|
</Grid>
|
|
</Grid>
|
|
</Border>
|
|
</Grid>
|
|
</DockPanel>
|
|
</UserControl>
|