AttributeError:“超级”对象没有属性“ __getattr__”,在Box和Kivy中将BoxLayout与多个kv文件一起使用时发生错误

时间:2019-03-15 23:27:55

标签: python kivy kivy-language

我非常清楚,这个问题已经被问过好几次了。但是尝试以下解决方案后:

我得出的结论是我需要解决我的特定问题。列出的解决方案似乎不适用于我的特定情况。

以下情况:

我目前正在尝试使用kivy为智能手机开发应用程序。由于我喜欢干净整洁的结构代码,因此将Kivy代码拆分为几个kv文件。 python代码应该主要具有逻辑,仅此而已。为了使其正常工作,我需要在不同的kv文件中引用不同对象的实例。为了弄清楚我的问题,我构建了一个相当简单的示例:

文件:try.py

from kivy.app import App
from kivy.uix.widget import Widget
from kivy.factory import Factory
from kivy.uix.label import Label
from kivy.lang import Builder

x= 1

class ComplexBox(Widget):
    def testit(self):
        self.ids.layout.add_widget(Label(text = "Requirement A met."))
    def addsome(self):
        global x
        self.ids.layout.add_widget(SomeWidget(id="XXX"+str(x)))
        x = x +1
    pass

class SomeWidget(Widget):
    def change(self):
        self.ids.REQB.text = "Requirement B met."
    pass

class RequirementC(Widget):
    def triggerC(self):
        self.ids.ERRORBUTTON.text = "Requirement C met"
    pass

class Attempt(App):
    def build(self):
        return presentation
    pass


presentation = Builder.load_file("attempt.kv")
Attempt().run()

文件:try.kv

#:kivy 1.0
#:include attemptsupp.kv
#:include attemptsuppC.kv

# root
<ComplexBox>:
    BoxLayout:
        id: layout
        size: root.size
        Button:
            id: ERRORBUTTON
            text: "add"
            on_press: root.addsome()
            on_release: root.testit()
BoxLayout:
    orientation: 'vertical'
    ComplexBox:
    RequirementC:

文件:trysupp.kv

#:kivy 1.0

# rules for the widget
<SomeWidget>:
    BoxLayout:
        pos: root.pos
        size: root.size
        orientation: "vertical"
        Label:
            id: REQB
            text: "hello"
        Button:
            text: "world"
            on_release: root.change()

文件:trysuppC.kv

#:kivy 1.0

<RequirementC>:
    Button:
        id: REQC
        text: "Press"
        on_release: root.triggerC()

picture of the Running Program - press the "Press"- button to get the error

以kivy版本1.10和Python版本3.7.2运行,程序首先可以完美启动。但是,当我按下ID为ERRORBUTTON的标有“ press”的按钮时,会出现此错误:

...--default --nodebug --client --host localhost --port 57777...\attempt.py "
[INFO   ] [Logger      ] Record log in...\.kivy\logs\kivy_19-03-15_31.txt
[INFO   ] [Kivy        ] v1.10.1
[INFO   ] [Python      ] v3.7.2 (tags/v3.7.2:9a3ffc0492, Dec 23 2018, 
...
[INFO   ] [Window      ] auto add sdl2 input provider
[INFO   ] [Window      ] virtual keyboard not allowed, single mode, not docked
[WARNING] [Lang        ] attemptsupp.kv has already been included!
[WARNING] [Lang        ] attemptsuppC.kv has already been included!
[INFO   ] [Base        ] Start application main loop
[INFO   ] [GL          ] NPOT texture support is available
[INFO   ] [Base        ] Leaving application in progress...
 Traceback (most recent call last):
   File "kivy\properties.pyx", line 838, in kivy.properties.ObservableDict.__getattr__
 KeyError: 'ERRORBUTTON'

 During handling of the above exception, another exception occurred:

 Traceback (most recent call last):
   File "...\ptvsd_launcher.py", line 45, in <module>
     main(ptvsdArgs)
   ...
   File "e:\Daten\Github_Projects\pc-clicker\attempt.py", line 35, in <module>
     Attempt().run()
   File "...\lib\site-packages\kivy\app.py", line 826, in run
     runTouchApp()
...
   File ...\lib\site-packages\kivy\lang\builder.py", line 64, in custom_callback
     exec(__kvlang__.co_value, idmap)
   File ...\attemptsuppC.kv", line 7, in <module>
     on_release: root.triggerC()
   File "...\attempt.py", line 25, in triggerC
     self.ids.ERRORBUTTON.text = "Requirement C met"
   File "kivy\properties.pyx", line 841, in kivy.properties.ObservableDict.__getattr__
 AttributeError: 'super' object has no attribute '__getattr__'

即使我缩短了错误消息,也应该清楚会发生什么情况。我在RequirementC类中引用的ERRORBUTTON ID在词典中找不到。现在我的问题:

我如何使其起作用?我缺少什么?

简而言之,我尝试了一些事情:

  • 我尝试将BoxLayouts包装在Screen中,并通过screenmanager访问它们。
  • 我尝试重新排列python代码中的顺序。 (例如,首先加载主要kv文件)
  • 我尝试使用Builder Factory并在那里注册不同的类。
  • 我尝试更改参考。 (例如self.ids。['ERRORBUTTON'] ...)

在我看来,这些尝试都没有奏效。

总结一下:

我如何才能使跨不同类的kivy引用正常工作,为什么ERRORBUTTON ID不在我正在研究的字典中?

2 个答案:

答案 0 :(得分:1)

问题是由常见错误引起的,id是相对于小部件的,例如,在您的情况下,让我们分析一下表达式:

self.ids.ERRORBUTTON

谁是自我? self是RequirementC的实例。

您拥有什么实例?,然后让我们看看在其中实现RequirementC的.kv:

<RequirementC>:
    Button:
        id: REQC
        text: "Press"
        on_release: root.triggerC()

如果您发现唯一有权访问REQC的ID,那么RequirementC的ID ERRORBUTTON不存在。

id ERRORBUTTON属于哪个类?那么,让我们回顾一下ERRORBUTTON的创建位置:

 # ...

<ComplexBox>:
    BoxLayout:
        id: layout
        size: root.size
        Button:
            id: ERRORBUTTON
            text: "add"
            on_press: root.addsome()
            on_release: root.testit()
 # ...

您会看到ERRORBUTTON是ComplexBox的ID。


使用上一部分中提到的内容,我们已经知道问题的原因。在给出解决方案之前,我们首先了解编程的基本原理:类是行为的抽象,它必须明确定义要暴露给外部的内容(因此,如果您检查任何库的文档,都不会记录所有方法。或所有类,因为这种想法是抽象类,也就是说,使用该库的人都不想知道它在内部如此精确地工作),因此设计时最好考虑一下类将拥有哪些方法。例如,假设我们创建了一个Person类,该类具有某些属性,例如大小或体重,如果您认为有必要揭露心脏或大脑的重量,该怎么办?好吧,不。您的情况也是如此。

解决方案是公开事件on_release,使其成为RequirementC类的一部分,除了将ERRORBUTTON公开为属性(在我的情况下,我不喜欢使用id,因为它们使代码难以读取),并且然后在具有共同作用域的地方进行连接。

*。py

# ...

class RequirementC(Widget):
    def __init__(self, **kwargs):
        self.register_event_type('on_release')
        super().__init__(**kwargs)

    def on_release(self):
        pass

# ...

attempt.kv

#:kivy 1.0
#:include attemptsupp.kv
#:include attemptsuppC.kv

# root
<ComplexBox>:
    error_button: ERRORBUTTON # <---
    BoxLayout:
        id: layout
        size: root.size
        Button:
            id: ERRORBUTTON
            text: "add"
            on_press: root.addsome()
            on_release: root.testit()
BoxLayout:
    orientation: 'vertical'
    ComplexBox:
        id: complex_box
    RequirementC:
        on_release: complex_box.error_button.text = "Requirement C met"

attemptsuppC.kv

#:kivy 1.0

<RequirementC>:
    Button:
        id: REQC
        text: "Press"
        on_release: root.dispatch('on_release')

答案 1 :(得分:0)

因此,经过一番研究,我找到了我想要的答案。对于那些可能也会在这里遇到我的问题的人来说,它来了:

class RequirementC(Widget):
def triggerC(self):
    presentation.children[1].ids.ERRORBUTTON.text = "Requirement C met"
pass

通过遍历演示文稿的子级,可以使用“ ids”方法。

但是对于那些现在想使用它的人,我建议遍历孩子以找到正确的ID,而不是给出“硬”引用(children [1])。