如何在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
答案 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)
因此,要访问它,您必须首先创建它的实例并存储实例,或者通过小部件树。