当前位置: 首页 > article >正文

WPF实现动态四宫格布局

需求描述

我们要设计一个界面,用户可以通过 CheckBox 控制哪些图表显示。图表的数量是动态的,最多可以选择显示四个图表。如果显示一个图表,它会占满整个区域;如果显示两个图表,它们会水平排列;显示三个图表时,上面两个水平排列,下面一个铺满宽度;如果显示四个图表,它们会均匀分布在四个格子中。

XAML 代码

首先,我们需要一个 Grid 布局,它将包含最多四个图表的占位区域。我们还需要一些 CheckBox 控制每个图表的显示。以下是 XAML 代码

<Window x:Class="DynamicGridLayout.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="动态四宫格布局" Height="450" Width="800">
    <Grid>
        <!-- 控制图表显示的 CheckBox -->
        <StackPanel Margin="10" HorizontalAlignment="Left" Orientation="Horizontal">
            <TextBlock Text="显示图表:" />
            <CheckBox x:Name="chbChart1" Margin="10,0,0,0" Checked="ChartShow_OnChecked" Unchecked="ChartShow_OnChecked" Content="图表 1" IsChecked="True"/>
            <CheckBox x:Name="chbChart2" Margin="10,0,0,0" Checked="ChartShow_OnChecked" Unchecked="ChartShow_OnChecked" Content="图表 2" IsChecked="True"/>
            <CheckBox x:Name="chbChart3" Margin="10,0,0,0" Checked="ChartShow_OnChecked" Unchecked="ChartShow_OnChecked" Content="图表 3" IsChecked="True"/>
            <CheckBox x:Name="chbChart4" Margin="10,0,0,0" Checked="ChartShow_OnChecked" Unchecked="ChartShow_OnChecked" Content="图表 4" IsChecked="True"/>
        </StackPanel>

        <!-- 动态布局的 Grid -->
        <Grid x:Name="MainGrid" Margin="10,60,10,10">
            <Border x:Name="chart1" Background="LightBlue" Visibility="Visible" BorderBrush="Black" BorderThickness="1" />
            <Border x:Name="chart2" Background="LightCoral" Visibility="Visible" BorderBrush="Black" BorderThickness="1" />
            <Border x:Name="chart3" Background="LightGreen" Visibility="Visible" BorderBrush="Black" BorderThickness="1" />
            <Border x:Name="chart4" Background="LightGoldenrodYellow" Visibility="Visible" BorderBrush="Black" BorderThickness="1" />
        </Grid>
    </Grid>
</Window>

C# 代码

接下来,我们在 C# 代码中根据用户的选择动态更新布局。关键是 ChartShow_OnChecked 方法,它根据当前选中的 CheckBox 控制图表的可见性,并动态调整 Grid 的行列定义。

public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
        UpdateChartLayout();
    }

    // 控制图表显示的事件
    private void ChartShow_OnChecked(object sender, RoutedEventArgs e)
    {
        // 确保至少有一个图表被选中
        bool isAnyChecked = chbChart1.IsChecked == true ||
                            chbChart2.IsChecked == true ||
                            chbChart3.IsChecked == true ||
                            chbChart4.IsChecked == true;

        // 如果没有任何图表选中,恢复当前选中的图表
        if (!isAnyChecked && sender is CheckBox currentCheckBox)
        {
            currentCheckBox.IsChecked = true;
            MessageBox.Show("至少需要选中一个图表!", "提示", MessageBoxButton.OK, MessageBoxImage.Warning);
            return;
        }

        // 控制对应图表的显示和隐藏
        chart1.Visibility = chbChart1.IsChecked == true ? Visibility.Visible : Visibility.Collapsed;
        chart2.Visibility = chbChart2.IsChecked == true ? Visibility.Visible : Visibility.Collapsed;
        chart3.Visibility = chbChart3.IsChecked == true ? Visibility.Visible : Visibility.Collapsed;
        chart4.Visibility = chbChart4.IsChecked == true ? Visibility.Visible : Visibility.Collapsed;

        // 更新布局
        UpdateChartLayout();
    }

    // 更新图表的布局
    private void UpdateChartLayout()
    {
        // 获取当前显示的图表
        var visibleCharts = new[] { chart1, chart2, chart3, chart4 }
            .Where(chart => chart.Visibility == Visibility.Visible)
            .ToList();

        // 清空现有的布局
        MainGrid.RowDefinitions.Clear();
        MainGrid.ColumnDefinitions.Clear();

        // 设置默认的布局
        foreach (var chart in visibleCharts)
        {
            Grid.SetRow(chart, 0);
            Grid.SetColumn(chart, 0);
            Grid.SetColumnSpan(chart, 1);
        }

        // 根据图表数量进行不同布局
        switch (visibleCharts.Count)
        {
            case 1:
                // 单个图表,铺满整个区域
                MainGrid.RowDefinitions.Add(new RowDefinition());
                MainGrid.ColumnDefinitions.Add(new ColumnDefinition());
                Grid.SetRow(visibleCharts[0], 0);
                Grid.SetColumn(visibleCharts[0], 0);
                break;

            case 2:
                // 两个图表,水平排列
                MainGrid.RowDefinitions.Add(new RowDefinition());
                MainGrid.ColumnDefinitions.Add(new ColumnDefinition());
                MainGrid.ColumnDefinitions.Add(new ColumnDefinition());
                Grid.SetRow(visibleCharts[0], 0);
                Grid.SetColumn(visibleCharts[0], 0);
                Grid.SetRow(visibleCharts[1], 0);
                Grid.SetColumn(visibleCharts[1], 1);
                break;

            case 3:
                // 三个图表,上方两个,下面一个铺满
                MainGrid.RowDefinitions.Add(new RowDefinition());
                MainGrid.RowDefinitions.Add(new RowDefinition());
                MainGrid.ColumnDefinitions.Add(new ColumnDefinition());
                MainGrid.ColumnDefinitions.Add(new ColumnDefinition());

                Grid.SetRow(visibleCharts[0], 0);
                Grid.SetColumn(visibleCharts[0], 0);
                Grid.SetRow(visibleCharts[1], 0);
                Grid.SetColumn(visibleCharts[1], 1);

                Grid.SetRow(visibleCharts[2], 1);
                Grid.SetColumn(visibleCharts[2], 0);
                Grid.SetColumnSpan(visibleCharts[2], 2); // 跨两列
                break;

            case 4:
                // 四个图表,标准两行两列
                MainGrid.RowDefinitions.Add(new RowDefinition());
                MainGrid.RowDefinitions.Add(new RowDefinition());
                MainGrid.ColumnDefinitions.Add(new ColumnDefinition());
                MainGrid.ColumnDefinitions.Add(new ColumnDefinition());

                for (int i = 0; i < visibleCharts.Count; i++)
                {
                    Grid.SetRow(visibleCharts[i], i / 2);
                    Grid.SetColumn(visibleCharts[i], i % 2);
                }
                break;
        }
    }
}


http://www.kler.cn/a/507955.html

相关文章:

  • PyTorch使用教程(2)-torch包
  • 大疆发布可折叠航拍无人机,仅重249g,支持 4800 万像素拍摄
  • SurfaceFlinger代码笔记
  • 云手机技术怎么实现的?
  • [Mac + Icarus Verilog + gtkwave] Mac运行Verilog及查看波形图
  • 生成订单号工具类
  • 灰度发布、金丝雀部署与蓝绿部署:软件发布的三把利剑
  • Redis | 第6章 事件与客户端《Redis设计与实现》
  • Ubuntu 部署Docker + Dify,遇到的坑, 最新亲测镜像
  • 如何在亚马逊云科技上大幅降低无服务器网页应用冷启动时间(上篇)
  • 在Mac m2系统下安装InSAR软件isce2
  • Python根据图片生成学生excel成绩表
  • [创业之路-254]:《华为数字化转型之道》-1-华为是一个由客户需求牵引、高度数字化、高度智能化、由无数个闭环流程组成的价值创造、评估、分配系统。
  • 学习微信小程序的下拉列表控件-picker
  • NC65增加按钮打开其他单据
  • DX12 快速教程(3) —— 画矩形
  • Java 数据结构 队列之双端队列 常用方法 示例代码 及其实现
  • vue+arcgis api for js实现地图经纬网格显示
  • 大模型WebUI:Gradio全解11——Chatbot:融合大模型的多模态聊天机器人(6)
  • Qt 使用共享内存的方式限制程序单一启动
  • 深入内核讲明白Android Binder【二】
  • 【JVM-8】使用 IBM Thread and Monitor Dump Analyzer for Java (TMDA) 分析线程转储
  • 深入了解卷积神经网络(CNN):图像处理与深度学习的革命性技术
  • 修复5.0.0r 64位版本浏览器和一些库找不到的问题
  • Flink (九):DataStream API (六) Process Function
  • 如何在 Google Cloud Shell 中使用 Visual Studio Code (VS Code)?