.kv 文件中的 MDDataTable

Posted

技术标签:

【中文标题】.kv 文件中的 MDDataTable【英文标题】:MDDataTable in .kv file 【发布时间】:2020-12-06 17:31:31 【问题描述】:

我在将 kivymd MDDataTable 组件作为子小部件添加到 .kv 文件中的屏幕时遇到了一些问题。不断收到KeyError: 'container'AttributeError: 'super' object has no attribute '__getattr__ 错误。我浏览了文档和多个站点,每个人似乎都在使用文档中示例的一些变体,它在构建方法中启动组件。

我想说的是,如果这行得通

class Example(MDApp):
    def build(self):
        screen = Screen()
        data_tables = MDDataTable(
            size_hint=(0.9, 0.6),
            column_data=[
                ('Template Id', dp(30)),
                ('Name', dp(30))
            ],
            row_data=[
                ('23lkjk33', 'Ayang Paul'),
                ('28ij399kk', 'Ringwa Justin')
            ]
        )
        screen.add_widget(data_tables)
        return screen
   

Example().run()

那为什么这不起作用

KV = '''
Screen:
    MDDataTable:
        size_hint: 0.9, 0.6
        pos_hint: "center_x": 0.5, "center_y": 0.5
        column_data: [('Template Id', dp(30)), ('Name', dp(30))]
        row_data: [('23lkjk33', 'Ayang Paul'), ('28ij399kk', 'Ringwa Justin')]
'''
class Example(MDApp):
    def build(self):
        return Builder.load_string(KV)
Example().run()

???

【问题讨论】:

请发minimal reproducible example。 @JohnAnderson 我已经编辑了问题 【参考方案1】:

问题是MDDataTable__init__() 方法引用了ids,但在kv 文件中使用时,ids 尚不可用。 MDDataTableModalView 的子类,因此它显然打算像 Popup 一样使用,而不是在 kv 文件中使用。

这是一个 hack,可以让它在 kv 文件中工作:

from kivymd.uix.datatables import MDDataTable, TableHeader, TableData, TablePagination
from kivymd.uix.dialog import BaseDialog


class MyDataTable(MDDataTable):
    def __init__(self, **kwargs):
        # skip the MDDataTable.__init__() and call its superclass __init__()
        super(BaseDialog, self).__init__(**kwargs)

        # schedule call to MDDataTable.__init__() contents after ids are populated
        Clock.schedule_once(partial(self.delayed_init, **kwargs))

    def delayed_init(self, dt, **kwargs):
        # this is copied from MDDataTable.__init__() with super() call deleted
        self.register_event_type("on_row_press")
        self.register_event_type("on_check_press")
        self.header = TableHeader(column_data=self.column_data, sort=self.sort)
        self.table_data = TableData(
            self.header,
            row_data=self.row_data,
            check=self.check,
            rows_num=self.rows_num,
            _parent=self,
        )
        self.pagination = TablePagination(table_data=self.table_data)
        self.table_data.pagination = self.pagination
        self.header.table_data = self.table_data
        self.table_data.fbind("scroll_x", self._scroll_with_header)
        self.ids.container.add_widget(self.header)
        self.ids.container.add_widget(self.table_data)
        if self.use_pagination:
            self.ids.container.add_widget(self.pagination)
        Clock.schedule_once(self.create_pagination_menu, 0.5)

所以在你的kv 中使用MyDataTable 代替MDDataTable。上面的代码延迟了 MDDataTable.__init__() 方法的执行,直到 ids 可用。如果MDDataTable 代码被更新,这个hack 可能不起作用。

【讨论】:

【参考方案2】:

看起来你有一些缩进错误。来自documentation:

缩进很重要,必须一致。间距必须 是第一个缩进行上使用的空格数的倍数。 鼓励使用空格:不建议混合使用制表符和空格。

试试这个:

<ModifyTeacherInfo@Screen>:
    name: "modifyTeacherInfo"
    MDDataTable:
        size_hint: 0.9, 0.6
        pos_hint: "center_x": 0.5, "center_y": 0.5
        column_data: [('Template Id', dp(30)), ('Name', dp(30))]
        row_data: [('23lkjk33', 'Ayang Paul'), ('28ij399kk', 'Ringwa Justin')]

【讨论】:

不是缩进。只是问题的代码部分写得不好

以上是关于.kv 文件中的 MDDataTable的主要内容,如果未能解决你的问题,请参考以下文章

将 kv 文件中的标签文本绑定到 python 文件中的方法

如何将函数绑定到kv文件中的按钮

如何使用 Python 代码从 kv 文件中的 TextField 获取数据?

如何使用 kv 文件刷新 Kivy 中的 GridLayout

python中的kivy小部件到kv文件,如何控制它

Python / Kivy:.kv 文件中的条件设计