質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.47%

  • Python 3.x

    6962questions

    Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

  • Kivy

    18questions

kv言語で、Pythonコードからのデータの受け渡しができずに、悩んでいます。

解決済

回答 2

投稿

  • 評価
  • クリップ 0
  • VIEW 139

 前提・実現したいこと

テスト用に、kivyで簡単な画面を作っています。
これは、各classごとに開発という手法をとっています。

from kivy.app import App
from kivy.lang import Builder
from kivy.properties import ObjectProperty

from kivy.uix.boxlayout import BoxLayout


Builder.load_string('''     

<TestSCN>:
    tct_p: tct_v

    BoxLayout:
        OpeParts:
        ResultParts:
    Label:
        id: tct_v
        text: 'ZZZ'

<OpeParts>:
    Label:
        text: 'AAA'

<ResultParts>:
    rcAdd_p: rcAdd_v

    Label:
        id: rcAdd_v
        text: 'XXX'

''')


class OpeParts(BoxLayout):
    pass


class ResultParts(BoxLayout):
    rcAdd_p = ObjectProperty()

    def __init__(self,**kwargs):
        super(ResultParts, self).__init__(**kwargs)
        self.rcAdd_p.text = 'BBB'


class TestSCN(BoxLayout):
    tct_p = ObjectProperty()

    def __init__(self,**kwargs):
        super(TestSCN, self).__init__(**kwargs)
        self.tct_p.text = 'CCC'


class TestApp(App):
    def build(self):
        return TestSCN()


if __name__ == '__main__':
    TestApp().run()

さて、Pythonのコードからkvファイルを使って、値を表示させることを意図していますが、このようなエラーメッセージが出て、うまく動きません。

 Traceback (most recent call last):
   File "/home/arakawa/MyApp/01_filelist/kv19e2.py", line 60, in <module>
     TestApp().run()
   File "/usr/lib/python3/dist-packages/kivy/app.py", line 800, in run
     root = self.build()
   File "/home/arakawa/MyApp/01_filelist/kv19e2.py", line 56, in build
     return TestSCN()
   File "/home/arakawa/MyApp/01_filelist/kv19e2.py", line 50, in __init__
     super(TestSCN, self).__init__(**kwargs)
   File "/usr/lib/python3/dist-packages/kivy/uix/boxlayout.py", line 131, in __init__
     super(BoxLayout, self).__init__(**kwargs)
   File "/usr/lib/python3/dist-packages/kivy/uix/layout.py", line 76, in __init__
     super(Layout, self).__init__(**kwargs)
   File "/usr/lib/python3/dist-packages/kivy/uix/widget.py", line 348, in __init__
     Builder.apply(self, ignored_consts=self._kwargs_applied_init)
   File "/usr/lib/python3/dist-packages/kivy/lang/builder.py", line 469, in apply
     self._apply_rule(widget, rule, rule, ignored_consts=ignored_consts)
   File "/usr/lib/python3/dist-packages/kivy/lang/builder.py", line 585, in _apply_rule
     self._apply_rule(child, crule, rootrule)
   File "/usr/lib/python3/dist-packages/kivy/lang/builder.py", line 582, in _apply_rule
     child = cls(__no_builder=True)
   File "/home/arakawa/MyApp/01_filelist/kv19e2.py", line 43, in __init__
     self.rcAdd_p.text = 'BBB'
 AttributeError: 'NoneType' object has no attribute 'text'

プロセスは終了コード 1 で完了しました


ResultPartsの'BBB'を表示させたいのですが、うまくいきません。

    BoxLayout:
        OpeParts:
        ResultParts:


という書き方をやめればうまくいくと思いますが、先ほども言ったようにclassごとに開発をしたいと思っていますので、できればこの書き方を堅持したいと思っています。
この状況で、中央のLabelを'BBB'と表示させるには、どうしたらよいのでしょうか?

よろしくお願いいたします。

開発環境
Python 3.5.2 + kivy 1.10.1 + Ubuntu16.04

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 2

checkベストアンサー

+1

この原因はWidgetがKv言語上で他のWidgetの子になった場合そうでない場合の初期化のされ方の違いにあります。上記のコードだとTestSCN以外の全てのWidgetはKv言語上で別のWidgetの子として作られているので全て前者にあたります。前者だとどうなるのかというとKv言語で書かれた物が適用されるのが遅れます。今回の場合、前者であるResultsPartsのKv言語で書かれた物(rcAdd_p: rcAdd_v rcAdd_pプロパティのへの値の設定)が遅れてNoneになっている状態で.text = 'BBB'としたのでErrorになっています。

対策ですが

  • Clockを使ってKv言語適用後に行いたい処理を予約する
  • 無理やりResultParts後者にしてしまう
  • ResultsParts.on_rcAdd_p()を用意する

の3つが思いつきます。

 1. Clockを使ってKv言語適用後に行いたい処理を予約する

from kivy.clock import Clock

class ResultParts(BoxLayout):
    rcAdd_p = ObjectProperty()

    def __init__(self, **kwargs):
        super(ResultParts, self).__init__(**kwargs)
        Clock.schedule_once(self._after_kv_applied)

    def _after_kv_applied(self, dt):
        self.rcAdd_p.text = 'BBB'

式一つで済むのならlambdaでもいいと思います。

 2. 無理やりResultPartsを後者にしてしまう

これは主さんの望まない方法かもしれませんが一応。

    BoxLayout:
        OpeParts:
        ResultParts:

    BoxLayout:
        OpeParts:
        RelativeLayout:
            id: parent_of_results_parts

として、

class TestSCN(BoxLayout):
    tct_p = ObjectProperty()

    def __init__(self,**kwargs):
        super(TestSCN, self).__init__(**kwargs)
        self.tct_p.text = 'CCC'

class TestSCN(BoxLayout):
    tct_p = ObjectProperty()

    def __init__(self,**kwargs):
        super(TestSCN, self).__init__(**kwargs)
        self.tct_p.text = 'CCC'
        self.ids.parent_of_results_parts.add_widget(ResultsParts())

とします。この場合ResultsPartsはKv言語上で別のWidgetの子になってないので後者にあたり、ResultsParts.__init__()内でrcAdd_pにアクセスできるようになります。

 3 ResultsParts.on_rcAdd_p()を用意する

on_プロパティ名のMethodはプロパティが書き換えられた時に呼ばれるので、on_rcAdd_p()が呼ばれた時にはrcAdd_p: rcAdd_vが適用済みだと期待できます。

class ResultParts(BoxLayout):
    rcAdd_p = ObjectProperty()

    def __init__(self, **kwargs):
        super(ResultParts, self).__init__(**kwargs)

    def on_rcAdd_p(self, __, value):
        self.rcAdd_p.text = 'BBB'

でも今回の場合だと'BBB'で書き換えた後に'XXX'に上書きされてしまうので、さらにKv言語のtext: 'XXX'の部分を消す必要があります。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2018/09/07 13:12

    ありがとうございました。1番を採用して確認できました。

    当方、kivyでこういった表現ができるのか疑心暗鬼で、ずいぶん資料を調べたり、デバッガーをいじったりしていました。
    Clockを使う方法は知っていたのですが、こういうやり方は思いつきませんでした。

    本当にありがとうございました。

    キャンセル

  • 2018/09/07 14:04

    いえいえ、いつでも

    キャンセル

0

暫定的な回答です。
質問でのkv言語部分の

    BoxLayout:
        OpeParts:
        ResultParts:


表記はあきらめています。

from kivy.app import App
from kivy.lang import Builder
from kivy.properties import ObjectProperty

from kivy.uix.boxlayout import BoxLayout


Builder.load_string('''     

<TestSCN>:
    tct_p: tct_v
    rcAdd_p: rcAdd_v

    BoxLayout:
        Label:
            text: 'AAA'
        Label:
            id: rcAdd_v
            text: 'XXX'
    Label:
        id: tct_v
        text: 'ZZZ'

''')

class ResultParts(BoxLayout):

    def __init__(self,**kwargs):
        super(ResultParts, self).__init__(**kwargs)
        self.rpv = 'BBB'


class TestSCN(BoxLayout):
    tct_p = ObjectProperty()
    rcAdd_p = ObjectProperty()

    def __init__(self,**kwargs):
        super(TestSCN, self).__init__(**kwargs)
        self.tct_p.text = 'CCC'
        self.rcAdd_p.text = ResultParts().rpv


class TestApp(App):
    def build(self):
        return TestSCN()


if __name__ == '__main__':
    TestApp().run()


私の理解力では、1画面1クラスルールで書かないと、うまくいきません。

考えてくださった方、どうもありがとうございした。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 90.47%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る

  • Python 3.x

    6962questions

    Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

  • Kivy

    18questions