如何使用 Xamarin Forms 创建具有垂直粘性标题和水平粘性第一列的表格?

2023-12-12

在显示表格数据时,我认为在某些情况下,具有始终可见的标题行和始终可见的第一列确实可以提高表格的可读性和整体可用性,特别是当表格中有大量数据时。当表格必须支持水平和垂直滚动时,就会出现问题。在查看过去比赛的得分时,可以从 NBA 应用程序中找到此类表格的一个很好的示例。以下是来自 NBA Android 应用程序的示例图像:NBA 移动应用程序中的示例表

从图像中可以清楚地看到,标题行与实际表格数据水平对齐,第一列与表格数据垂直对齐。我不知道这是一个非自愿的还是自愿的决定,以防止使用相同的触摸动作水平和垂直滚动,但这是我不关心的一个小细节。

我不知道如何使用 Xamarin Forms 来实现这一点。我对闭源/付费解决方案不感兴趣,因为我想实际学习如何自己完成此任务。我确实意识到我很可能必须为 Android 和 IOS 使用自定义渲染器。我当前的想法是我有一个绝对布局,其中包含以下元素:

  • 第一个单元格(它是静止的,也是唯一静止的东西)
  • 水平滚动视图内标题行的其余部分
  • listview/stacklayout + 垂直滚动视图中的第一列
  • 列表视图+水平滚动视图/堆栈布局+水平和垂直滚动视图内的实际表格数据

通过此设置,我将捕获触摸事件并将其发送到其他列表视图/滚动视图,从而同步滚动。事实上,通过将表数据设置在与第一列相同的垂直滚动视图内,我可以轻松实现第一列和实际表数据的同步滚动。但我不知道如何将水平滚动同步到标题行,并且我确实相信这不能通过巧妙的组件结构来完成。到目前为止,我只在 Android 上进行了测试,我可以在滚动视图自定义渲染器的 OnTouchEvent 方法中捕获触摸事件,但我不知道如何将其从自定义渲染器发送到标题行滚动视图。

这是说明我的方法的 XAML 草案。

<AbsoluteLayout xmlns="http://xamarin.com/schemas/2014/forms"
             xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
             HorizontalOptions="FillAndExpand">
    <ScrollView
        Orientation="Horizontal"
        x:Name="HeaderScrollView"
        AbsoluteLayout.LayoutBounds="0,0,1,1"
        AbsoluteLayout.LayoutFlags="All">
        <Grid>
            <Grid.ColumnDefinitions>
                <ColumnDefinition Width="200" />
                <ColumnDefinition Width="100" />
                <ColumnDefinition Width="100" />
                <ColumnDefinition Width="100" />
                <ColumnDefinition Width="100" />
            </Grid.ColumnDefinitions>
            <Grid.RowDefinitions>
                <RowDefinition Height="50" />
            </Grid.RowDefinitions>
            <!-- Skip first column, leave it empty for stationary cell -->
            <Label Text="Column 1" Grid.Row="0" Grid.Column="1" />
            <Label Text="Column 2" Grid.Row="0" Grid.Column="2" />
            <Label Text="Column 3" Grid.Row="0" Grid.Column="3" />
            <Label Text="Column 4" Grid.Row="0" Grid.Column="4" />
        </Grid>
    </ScrollView>
    <ScrollView
        x:Name="FirstColumnScrollView"
        Orientation="Vertical"
        AbsoluteLayout.LayoutBounds="0,50,1,1"
        AbsoluteLayout.LayoutFlags="SizeProportional"
        BackgroundColor="Aqua">
        <Grid>
            <Grid.ColumnDefinitions>
                <ColumnDefinition Width="200" />
                <ColumnDefinition Width="*" />
            </Grid.ColumnDefinitions>
            <Grid.RowDefinitions>
                <RowDefinition Height="*" />
            </Grid.RowDefinitions>
            <StackLayout
                Grid.Column="0"
                Grid.Row="0"
                BindableLayout.ItemsSource="{Binding DataSource}">
                <BindableLayout.ItemTemplate>
                    <DataTemplate>
                        <Grid>
                            <Grid.RowDefinitions>
                                <RowDefinition Height="50" />
                            </Grid.RowDefinitions>
                            <Grid.ColumnDefinitions>
                                <ColumnDefinition Width="150" />
                            </Grid.ColumnDefinitions>
                            <Label Text="{Binding Column1}" Grid.Row="0" Grid.Column="0" />
                        </Grid>
                    </DataTemplate>
                </BindableLayout.ItemTemplate>
            </StackLayout>
            <ScrollView
                x:Name="TableDataScrollView"
                Grid.Column="1"
                Grid.Row="0"
                Orientation="Horizontal">
                <StackLayout
                    BindableLayout.ItemsSource="{Binding DataSource}">
                    <BindableLayout.ItemTemplate>
                        <DataTemplate>
                            <Grid>
                                <Grid.ColumnDefinitions>
                                    <ColumnDefinition Width="100" />
                                    <ColumnDefinition Width="100" />
                                    <ColumnDefinition Width="100" />
                                    <ColumnDefinition Width="100" />
                                </Grid.ColumnDefinitions>
                                <Grid.RowDefinitions>
                                    <RowDefinition Height="50" />
                                </Grid.RowDefinitions>
                                <Label Text="{Binding Column2}" Grid.Row="0" Grid.Column="0" />
                                <Label Text="{Binding Column3}" Grid.Row="0" Grid.Column="1" />
                                <Label Text="{Binding Column4}" Grid.Row="0" Grid.Column="2" />
                                <Label Text="{Binding Column5}" Grid.Row="0" Grid.Column="3" />
                            </Grid>
                        </DataTemplate>
                    </BindableLayout.ItemTemplate>
                </StackLayout>
            </ScrollView>
        </Grid>
    </ScrollView>
    <Label Text="First Column" BackgroundColor="White" AbsoluteLayout.LayoutBounds="0,0,200,50" />
</AbsoluteLayout>

正如您所看到的,问题在于 HeaderScrollView 和 TableDataScrollView 之间的水平滚动事件不共享,我不知道如何以最好的方式或根本不知道如何实现这一点。

我非常感谢所有的帮助和反馈!


您正在寻找的是具有冻结行和冻结列功能的 DataGrid 组件。有一些第三方组件可以满足您的要求。

Syncfusion、Telerik 和 Infragistics DataGrids 具有您正在寻找的功能。请参阅以下链接。

  • https://www.syncfusion.com/xamarin-ui-controls/xamarin-datagrid
  • https://www.telerik.com/xamarin-ui/datagrid
  • https://www.infragistics.com/products/xamarin/grids-and-lists/data-grid

也有很少的开源 DataGrid 可用。但不确定它们是否具有行和列固定功能。检查以下链接。

  • https://github.com/akgulebubekir/Xamarin.Forms.DataGrid
  • https://www.nuget.org/packages/Xamarin.Forms.DataGrid/
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

如何使用 Xamarin Forms 创建具有垂直粘性标题和水平粘性第一列的表格? 的相关文章

随机推荐

  • android - 如何拥有可点击和可检查的文本视图

    我想要以下内容 一个文本视图 单击时更改其背景 保持该背景直到再次单击 这一切都归结为 可检查 状态 但我无法弄清楚这到底是如何工作的 这是我用于背景的 xml
  • 使用 FtpLib 获取文件夹大小

    我正在使用 FtpLib Python 从 ftp 服务器下载包含文件的文件夹 但我想在开始下载之前知道文件夹的大小 在我找到的文档中FTP size filename 但这仅适用于文件而不适用于文件夹 我可以循环遍历文件夹中的所有文件 但
  • Android在Canvas上获取Bitmap Rect(左、上、右、下)

    我在画布上绘制位图 然后在其上进行一些缩放 非常简单 只需使用 canvas scale int int hub hub 然后 缩放完成后 我需要获取位图的坐标相对于视口 有没有什么方便的方法可以做到这一点 而无需自己计算初始位置是什么 然
  • 通过链接将一张纸上的列转置为另一张纸上的行

    我在工作表 A 中的 A1 A10 中有一列数据 在工作表 B 中 我有 A1 J1 中的数据范围 我需要将工作表 A 中的列中的数据转置到工作表 B 中的数据范围 我可以 选择性粘贴 并粘贴值 但我需要工作表 A 中的信息自动更新工作表
  • JAXB 提供商列表

    AFAIK JAXB 只是一个规范 JSR222 Java 1 6 有一个它的实现 但它也被称为 JAXB 这让我感到困惑 因为 JPA 也是一种 JSR 规范 但 JPA 提供者有 EclipseLink Hibernate 等名称 那么
  • Hibernate Search:为自定义 FieldBridge 配置 Facet

    在这个例子中DateSplitBridge java动态字段被添加到索引文档中 public class DateSplitBridge implements FieldBridge public void set String name
  • 检查哪个摄像头是前置摄像头还是后置摄像头 Android

    我知道我可以设置一个boolean flag while opening front Camera 如果 flag 为 true 则表示前置摄像头已打开 但是有没有办法使用 Android API 知道哪个相机现在处于打开状态 正面或背面
  • SQLAlchemy 循环一对一关系

    我正在尝试与 SQLAlchemy 建立循环的一对一关系 不确定正确的术语是什么 如下所示 class Parent Base tablename parents id db Column Integer primary key True
  • 当绑定发生在另一个函数中时,Sqlite _only_ 找不到行

    所以我自己写了一个小包装函数来为我准备一个声明 sqlite3 stmt Gladiateur run query unfinalized string query vector
  • WPF,使用 XAML 中的 XPath 和 XmlDataProvider 根据 ComboBox 中选定的值选择节点

    这个问题与我的上一个问题但更具体 假设我有两个组合框 一个填充了产品名称 另一个为空 选择产品后 我希望第二个组合框填充与该产品相关的数据 我有如下 XML
  • Microsoft Office OneNote C++ API?

    我正在考虑通过 C 编程修改 Microsoft Office OneNote 内容 具体在使用快速归档对话框界面 但是那里提供的所有示例都是针对C 的 我想知道C 的API是否可用 如果有的话我可以从哪里开始学习它们 我只是想使用该对话框
  • Angular 5测试组件静态方法

    我正在测试的 Component 类上有一个静态方法 我的问题是如何在我的规范测试文件中访问该方法 到目前为止 我可以通过以下方式访问组件实例 let fixture TestBed createComponent MyComponent
  • 如何执行递归搜索?

    我有一个任务类 它可以有相同类型的子任务 public class Task public DateTime Start get set public DateTime Finish get set public List
  • 我对套接字的 fwrite 不会刷新,直到套接字关闭。如何改变?

    我对套接字的 fwrite 不会刷新 直到套接字关闭 如何改变 我希望它在每次写入后刷新 我试过了 1 冲洗 2 刷新 3 ob implicit flush 真 这些都不起作用 我仍然必须退出 php 让我的套接字接收数据 包括一些示例代
  • Rails:如何使用范围在数组数组中查找元素

    我有一个数组数组 例如 2 3 3 1 6 1 每个子数组的第一个元素是用户 ID 第二个元素是用户为活动预订的座位数 我想让每个用户通过在数组中查找他的 ID 来查看他的预订 假设我有两个模型 用户和事件 在用户控制器中 我想使用类似的范
  • 唯一标识调试器中的引用类型

    我有 C 背景 如果这是非 C 的思维方式 我很抱歉 但我只需要知道 在 C 中 如果我有两个指针 并且我想知道它们是否指向同一对象 我可以查看内存 监视窗口并查看它们的值 看看它们是否指向相同的内存空间 在 C 中 我还没有找到类似的东西
  • Silverlight 中的对象深复制

    我试图创建对象的副本银光5其中 IFormatters 和 IcCloanble 等接口不支持 我的对象是这样的 注意这些对象是在反序列化xml时获得的 我尝试像这样复制 XmlRoot ElementName component publ
  • 如何更改每个选项卡的颜色?

    我有一个表单 上面有四个选项卡 我希望每个选项卡具有不同的颜色 我在互联网上找到的唯一内容是如何更改所选选项卡的颜色 而其余选项卡保持原始颜色 我还没有找到任何东西可以为每个选项卡赋予自己的颜色 我目前拥有的代码是 Private Sub
  • 通过谷歌脚本更改表单调色板

    我需要帮助 我正在使用 Google 应用程序脚本来创建 Google 表单 如何更改表单的颜色 调色板 我试图在这里找到https developers google com apps script reference forms 但什么
  • 如何使用 Xamarin Forms 创建具有垂直粘性标题和水平粘性第一列的表格?

    在显示表格数据时 我认为在某些情况下 具有始终可见的标题行和始终可见的第一列确实可以提高表格的可读性和整体可用性 特别是当表格中有大量数据时 当表格必须支持水平和垂直滚动时 就会出现问题 在查看过去比赛的得分时 可以从 NBA 应用程序中找