I have been trying to get my labels to auto-update themselves for quite a while now and I have read over a dozen StackOverflow questions but to no avail.
I have an global object that contains an integer value that I want to be displayed with a label inside of one of my widget classes.
The widget class looks like this:
class Battle(Widget):
def __init__(self, **kwargs):
super(Battle, self).__init__(**kwargs)
#Enemy Stats
self.enemyBar = BoxLayout(orientation="horizontal", size=(Window.width, Window.height/8), center_y = Window.height - self.height/2)
self.enemyBar.add_widget(Label(text=enemy.name))
#Enemy Health Label
health_label = Label(text=str(enemy.health))
self.enemyBar.add_widget(health_label)
self.add_widget(self.enemyBar)
def update_health(instance, value):
health_label.text = str(enemy.health) #<-- Error happens here
enemy.bind(health=update_health)
When the value of enemy.health
is changed in the program, I want my label to change as well. I do not want to use any of the kivy language because I prefer having just one main python file.
The enemy object is created with an entity class. Here is the entity code:
class entity(Widget):
#entity creation
health = NumericProperty()
def __init__(self, health):
self.health = health
When I run the code I press a button that calls a function that changes the enemy health and then I get an error:
global name ‘health_label’ is not defined
Somehow, when the update_health function is called, the program doesn’t see the health_label variable that was created in the init.
Advertisement
Answer
You need to use the bind method, something like the following
health_label = Label(text=enemy.health)
self.enemyBar.add_widget(health_label)
def update_health(instance, value):
health_label.text = str(enemy.health)
enemy.bind(health=update_health)
This is just a basic example off the top of my head, it can be made neater depending on the structure of your program. If enemy.health is a string, you can just do enemy.bind(health=health_label.setter('text'))
.
For this to work, the health attribute must be a kivy property:
class Enemy(Something):
health = StringProperty('')
In your code it seems that enemy is a global object. There is quite likely a better way to do that. Also, I recommend using kv language – there is not really any special value in putting your code in one file (indeed, it’s commonly considered bad practice once it gets non-trivial), and kv makes a lot of things easier where python just isn’t suited to certain tasks.