你如何访问kivy中不同类的小部件

时间:2017-02-23 00:41:52

标签: python kivy kivy-language

如何在CurrentWeather中从Python访问位于AddLocationForm窗口小部件中的条件值?我想从AddLocationForm编辑条件值,而不是在CurrentWeather中进行修改。到目前为止,这是我的代码:

的Python:

from kivy.network.urlrequest import UrlRequest
import json
from kivy.core.window import Window
from kivy.uix.textinput import TextInput
from kivy.uix.listview import ListItemButton
from kivy.factory import Factory

Window.clearcolor = (0.45, 0.3, 1, 0.5)

class AddLocationForm(BoxLayout):
    search_input = ObjectProperty()
    search_results = ObjectProperty()
    currentWeather = ObjectProperty()

    def search_location(self):
        search_template = "http://api.openweathermap.org/data/2.5/weather?q={}&appid=f8020ca79e2de4c1dd5a125c3f22fa18"
        search_url = search_template.format(self.search_input.text)
        request = UrlRequest(search_url, self.found_location)

    def found_location(self, request, data):
        data = json.loads(data.decode()) if not isinstance(data, dict) else data
        weather = "%s" %(data["weather"][0]["description"])
        root.WeatherRoot.CurrentWeather.conditions = weather #Problem is here

class LocationButton(ListItemButton):
    pass

class WeatherApp(App):
    def build(self):
        global root
        root = self.root

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

的Kv:

#: import ListItemButton kivy.uix.listview.ListItemButton
#: import ListAdapter kivy.adapters.listadapter.ListAdapter
#: import main main

WeatherRoot:

<WeatherRoot>:
    AddLocationForm

<AddLocationForm>:
    orientation: "vertical"
    search_input: search_box 
    search_results: search_results_list
    BoxLayout:
        height: "40dp"
        size_hint_y: None
        TextInput:
            id: search_box 
            size_hint_x: 50
            multiline: False
        Button:
            text: "Search"
            size_hint_x: 25
            on_press: root.search_location()
            background_color: (1, 0.5, 0.656, 1)

<CurrentWeather@BoxLayout>:
    id: current_weather
    currentWeather: current_weather
    location: "" 
    conditions: ""
    temp: None
    temp_min: None
    temp_max: None

    orientation: "vertical"
    Label:
        text: root.location
        pos_hint: {'x': 0.435, "y": 1}
        font_size: '40dp'       
        size_hint_y: None
        size_hint_x: None
    BoxLayout:
        orientation: "horizontal"

        Label:
            id: curCon
            text: root.conditions

1 个答案:

答案 0 :(得分:1)

<强>树

WeatherRoot:           # root widget

<WeatherRoot>:         # root widget class
    AddLocationForm    # root widget's child

因此,您首先会使用app访问App的实例。 构建方法之后的实例有一个root property,它只是存储的根小部件的一个实例。这就是为什么global root在较大的代码中无用的原因,例如碰撞,与其他名称的混淆等,同时以一种漂亮的方式更容易访问。

要从python或kv访问以kv动态创建的类,您需要使用窗口小部件树(以获取已存在的实例):

app.root.children[0]...

Factory创建小工具的实例,例如如果您想在评估某些条件后创建新的。

但是!

你误解了动态类在kv中是如何工作的 - <Name@Widget>:只是一个声明,而不是一个实例。像模板一样。要实际使用它,您必须放在某处,例如:

蟒:

cw = Factory.CurrentWeather()
somewhere.add_widget(cw)

KV:

WeatherRoot:
    CurrentWeather:  # here

<WeatherRoot>:
    AddLocationForm:
        CurrentWeather:  # or here

如果您选择在python中添加它,您可以轻松地使用某个变量来存储实例,并且以后仍然可以通过窗口小部件树访问它。另一方面,当以kv格式添加时,您无权访问__init__,并且如果没有像这样的愚蠢黑客,您无法将实例存储到变量中:

#:import Factory kivy.factory.Factory

<Blob@Button>:

BoxLayout:
    Button:
        on_release: self.new = Factory.Blob(); self.parent.add_widget(self.new)

因此,要访问它,您必须首先创建它的实例并存储实例,或者通过小部件树。