mirror of
				https://github.com/pimoroni/grow-python
				synced 2025-10-25 15:19:23 +00:00 
			
		
		
		
	Popover help text
This commit is contained in:
		| @@ -141,14 +141,14 @@ class View: | |||||||
|         self._draw.rectangle((x, y, x2, y2), bgcolor) |         self._draw.rectangle((x, y, x2, y2), bgcolor) | ||||||
|         self._draw.text((x + margin, y + margin - 1), text, font=font, fill=textcolor) |         self._draw.text((x + margin, y + margin - 1), text, font=font, fill=textcolor) | ||||||
|  |  | ||||||
|     def draw_next_button(self, disabled=False): |     def overlay(self, text): | ||||||
|         if disabled: |         """Draw a slightly transparent overlay with some text.""" | ||||||
|             # Draw disabled "Next" button |         overlay = Image.new("RGBA", (DISPLAY_WIDTH, DISPLAY_HEIGHT), color=(0, 0, 0, 0)) | ||||||
|             self._draw.rectangle((0, 0, 19, 19), (138, 138, 138)) |         draw = ImageDraw.Draw(overlay) | ||||||
|             icon(self._image, icon_rightarrow, (0, 0), (150, 150, 150)) |         draw.rectangle((0, 20, DISPLAY_WIDTH, DISPLAY_HEIGHT), fill=(192, 225, 254, 240))  # Overlay backdrop | ||||||
|         else: |         draw.rectangle((0, 20, DISPLAY_WIDTH, 21), fill=COLOR_BLUE)  # Top border | ||||||
|             self._draw.rectangle((0, 0, 19, 19), COLOR_BLUE) |         text_in_rect(draw, text, font, (3, 20, DISPLAY_WIDTH - 3, DISPLAY_HEIGHT - 2), line_spacing=1) | ||||||
|             icon(self._image, icon_rightarrow, (0, 0), COLOR_WHITE) |         self._image.paste(Image.alpha_composite(self._image, overlay), (0, 0)) | ||||||
|  |  | ||||||
|  |  | ||||||
| class MainView(View): | class MainView(View): | ||||||
| @@ -157,6 +157,7 @@ class MainView(View): | |||||||
|         self._current_option = 0 |         self._current_option = 0 | ||||||
|         self._change_mode = False |         self._change_mode = False | ||||||
|         self._edit_mode = False |         self._edit_mode = False | ||||||
|  |         self._help_mode = False | ||||||
|  |  | ||||||
|         self.channels = channels |         self.channels = channels | ||||||
|         View.__init__(self, image) |         View.__init__(self, image) | ||||||
| @@ -238,8 +239,7 @@ class MainView(View): | |||||||
|             self.label("B", "Next", textcolor=COLOR_WHITE, bgcolor=COLOR_BLUE) |             self.label("B", "Next", textcolor=COLOR_WHITE, bgcolor=COLOR_BLUE) | ||||||
|             self.label("Y", "Change", textcolor=COLOR_WHITE, bgcolor=COLOR_YELLOW) |             self.label("Y", "Change", textcolor=COLOR_WHITE, bgcolor=COLOR_YELLOW) | ||||||
|  |  | ||||||
|         self._draw.rectangle((0, 20, DISPLAY_WIDTH, 40), fill=(200, 200, 200)) |         self.label("A", "?", textcolor=COLOR_WHITE, bgcolor=COLOR_BLUE) | ||||||
|         text_in_rect(self._draw, help, font, (3, 20, DISPLAY_WIDTH - 3, 38), line_spacing=1) |  | ||||||
|  |  | ||||||
|         self._draw.text( |         self._draw.text( | ||||||
|             (23, 3), |             (23, 3), | ||||||
| @@ -249,7 +249,8 @@ class MainView(View): | |||||||
|         ) |         ) | ||||||
|         self._draw.text((3, 43), f"{title} : {text}", font=font, fill=(0, 0, 0)) |         self._draw.text((3, 43), f"{title} : {text}", font=font, fill=(0, 0, 0)) | ||||||
|  |  | ||||||
|         self.draw_next_button(True) |         if self._help_mode: | ||||||
|  |             self.overlay(help) | ||||||
|  |  | ||||||
|     def render_overview(self): |     def render_overview(self): | ||||||
|         self._draw.rectangle((0, 0, DISPLAY_WIDTH, DISPLAY_HEIGHT), (255, 255, 255)) |         self._draw.rectangle((0, 0, DISPLAY_WIDTH, DISPLAY_HEIGHT), (255, 255, 255)) | ||||||
| @@ -269,9 +270,13 @@ class MainView(View): | |||||||
|         self.label("Y", "BL", textcolor=COLOR_WHITE, bgcolor=COLOR_GREEN) |         self.label("Y", "BL", textcolor=COLOR_WHITE, bgcolor=COLOR_GREEN) | ||||||
|  |  | ||||||
|     def button_a(self): |     def button_a(self): | ||||||
|  |         if self._edit_mode: | ||||||
|  |             self._help_mode = not self._help_mode | ||||||
|         return self._edit_mode |         return self._edit_mode | ||||||
|  |  | ||||||
|     def button_b(self): |     def button_b(self): | ||||||
|  |         if self._help_mode: | ||||||
|  |             return True | ||||||
|         if self._edit_mode: |         if self._edit_mode: | ||||||
|             if self._change_mode: |             if self._change_mode: | ||||||
|                 option = self._options[self._current_option] |                 option = self._options[self._current_option] | ||||||
| @@ -304,9 +309,12 @@ class MainView(View): | |||||||
|                 self._edit_mode = False |                 self._edit_mode = False | ||||||
|         else: |         else: | ||||||
|             self._edit_mode = True |             self._edit_mode = True | ||||||
|  |         self._help_mode = False | ||||||
|         return True |         return True | ||||||
|  |  | ||||||
|     def button_y(self): |     def button_y(self): | ||||||
|  |         if self._help_mode: | ||||||
|  |             return True | ||||||
|         if self._change_mode: |         if self._change_mode: | ||||||
|             option = self._options[self._current_option] |             option = self._options[self._current_option] | ||||||
|             prop = option["prop"] |             prop = option["prop"] | ||||||
| @@ -328,6 +336,10 @@ class MainView(View): | |||||||
|  |  | ||||||
|  |  | ||||||
| class ChannelView(View): | class ChannelView(View): | ||||||
|  |     def __init__(self, image, channel=None): | ||||||
|  |         self.channel = channel | ||||||
|  |         View.__init__(self, image) | ||||||
|  |  | ||||||
|     def draw_status(self, subtle=False): |     def draw_status(self, subtle=False): | ||||||
|         status = f"{self.channel.sensor.saturation * 100:.2f}% ({self.channel.sensor.moisture:.2f}Hz)" |         status = f"{self.channel.sensor.saturation * 100:.2f}% ({self.channel.sensor.moisture:.2f}Hz)" | ||||||
|  |  | ||||||
| @@ -342,10 +354,6 @@ class ChannelView(View): | |||||||
|  |  | ||||||
|  |  | ||||||
| class DetailView(ChannelView): | class DetailView(ChannelView): | ||||||
|     def __init__(self, image, channel=None): |  | ||||||
|         self.channel = channel |  | ||||||
|         View.__init__(self, image) |  | ||||||
|  |  | ||||||
|     def render(self): |     def render(self): | ||||||
|         width, height = self._image.size |         width, height = self._image.size | ||||||
|         self._draw.rectangle((0, 0, width, height), (255, 255, 255)) |         self._draw.rectangle((0, 0, width, height), (255, 255, 255)) | ||||||
| @@ -357,13 +365,10 @@ class DetailView(ChannelView): | |||||||
|             fill=(0, 0, 0), |             fill=(0, 0, 0), | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         graph_height = DISPLAY_HEIGHT - 40 |  | ||||||
|  |  | ||||||
|         self._draw.rectangle((0, 40, DISPLAY_WIDTH, 40 + graph_height), (10, 10, 10)) |  | ||||||
|         self.draw_status() |         self.draw_status() | ||||||
|  |  | ||||||
|         offset_x = 20 |         graph_height = DISPLAY_HEIGHT - 40 | ||||||
|         offset_y = 20 |         self._draw.rectangle((0, DISPLAY_HEIGHT - graph_height, DISPLAY_WIDTH, DISPLAY_HEIGHT), (10, 10, 10)) | ||||||
|  |  | ||||||
|         for x, value in enumerate(self.channel.sensor.history[:DISPLAY_WIDTH]): |         for x, value in enumerate(self.channel.sensor.history[:DISPLAY_WIDTH]): | ||||||
|             color = self.channel.indicator_color(value) |             color = self.channel.indicator_color(value) | ||||||
| @@ -396,7 +401,9 @@ class DetailView(ChannelView): | |||||||
|  |  | ||||||
|         icon(self._image, icon_snooze, (DISPLAY_WIDTH - 40, DISPLAY_HEIGHT - alarm_line - 10), (r, 0, 0)) |         icon(self._image, icon_snooze, (DISPLAY_WIDTH - 40, DISPLAY_HEIGHT - alarm_line - 10), (r, 0, 0)) | ||||||
|  |  | ||||||
|         self.draw_next_button() |         # Next button | ||||||
|  |         self._draw.rectangle((0, 0, 19, 19), COLOR_BLUE) | ||||||
|  |         icon(self._image, icon_rightarrow, (0, 0), COLOR_WHITE) | ||||||
|  |  | ||||||
|         # Edit |         # Edit | ||||||
|         self.label("X", "Edit", textcolor=(255, 255, 255), bgcolor=COLOR_RED) |         self.label("X", "Edit", textcolor=(255, 255, 255), bgcolor=COLOR_RED) | ||||||
| @@ -412,12 +419,14 @@ class EditView(ChannelView): | |||||||
|                 "min": 0, |                 "min": 0, | ||||||
|                 "max": 1.0, |                 "max": 1.0, | ||||||
|                 "format": lambda value: f"{value * 100:0.2f}%", |                 "format": lambda value: f"{value * 100:0.2f}%", | ||||||
|  |                 "help": "Saturation at which alarm is triggered", | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 "title": "Enabled", |                 "title": "Enabled", | ||||||
|                 "prop": "enabled", |                 "prop": "enabled", | ||||||
|                 "mode": "bool", |                 "mode": "bool", | ||||||
|                 "format": lambda value: "Yes" if value else "No" |                 "format": lambda value: "Yes" if value else "No", | ||||||
|  |                 "help": "Enable/disable this channel", | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 "title": "Wet Point", |                 "title": "Wet Point", | ||||||
| @@ -426,6 +435,7 @@ class EditView(ChannelView): | |||||||
|                 "min": 1, |                 "min": 1, | ||||||
|                 "max": 27, |                 "max": 27, | ||||||
|                 "format": lambda value: f"{value:0.2f}Hz", |                 "format": lambda value: f"{value:0.2f}Hz", | ||||||
|  |                 "help": "Frequency for fully saturated soil", | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|                 "title": "Dry Point", |                 "title": "Dry Point", | ||||||
| @@ -434,22 +444,20 @@ class EditView(ChannelView): | |||||||
|                 "min": 1, |                 "min": 1, | ||||||
|                 "max": 27, |                 "max": 27, | ||||||
|                 "format": lambda value: f"{value:0.2f}Hz", |                 "format": lambda value: f"{value:0.2f}Hz", | ||||||
|  |                 "help": "Frequency for fully dried soil", | ||||||
|             }, |             }, | ||||||
|         ] |         ] | ||||||
|         self._current_option = 0 |         self._current_option = 0 | ||||||
|         self._change_mode = False |         self._change_mode = False | ||||||
|         self.channel = channel |         self._help_mode = False | ||||||
|         View.__init__(self, image) |         ChannelView.__init__(self, image, channel) | ||||||
|  |  | ||||||
|     def render(self): |     def render(self): | ||||||
|         graph_height = DISPLAY_HEIGHT - 40 |         graph_height = DISPLAY_HEIGHT - 40 | ||||||
|  |  | ||||||
|         self._draw.rectangle((0, 0, DISPLAY_WIDTH, DISPLAY_HEIGHT), (255, 255, 255)) |         self._draw.rectangle((0, 0, DISPLAY_WIDTH, DISPLAY_HEIGHT), (255, 255, 255)) | ||||||
|  |  | ||||||
|         self._draw.text((23, 3), "{}".format(self.channel.title), font=font, fill=(0, 0, 0)) |         self._draw.text((23, 3), "{}".format(self.channel.title), font=font, fill=(0, 0, 0)) | ||||||
|  |  | ||||||
|         self.draw_status(True) |  | ||||||
|  |  | ||||||
|         option = self._options[self._current_option] |         option = self._options[self._current_option] | ||||||
|         title = option["title"] |         title = option["title"] | ||||||
|         prop = option["prop"] |         prop = option["prop"] | ||||||
| @@ -457,8 +465,9 @@ class EditView(ChannelView): | |||||||
|         text = option["format"](value) |         text = option["format"](value) | ||||||
|         mode = option.get("mode", "int") |         mode = option.get("mode", "int") | ||||||
|         self._draw.text((3, 43), f"{title} : {text}", font=font, fill=(0, 0, 0)) |         self._draw.text((3, 43), f"{title} : {text}", font=font, fill=(0, 0, 0)) | ||||||
|  |         help = option.get("help", "") | ||||||
|  |  | ||||||
|         self.draw_next_button(True) |         self.draw_status(True) | ||||||
|  |  | ||||||
|         if self._change_mode: |         if self._change_mode: | ||||||
|             self.label("Y", "Yes" if mode == "bool" else "++", textcolor=COLOR_WHITE, bgcolor=COLOR_YELLOW) |             self.label("Y", "Yes" if mode == "bool" else "++", textcolor=COLOR_WHITE, bgcolor=COLOR_YELLOW) | ||||||
| @@ -467,12 +476,20 @@ class EditView(ChannelView): | |||||||
|             self.label("Y", "Change", textcolor=COLOR_WHITE, bgcolor=COLOR_YELLOW) |             self.label("Y", "Change", textcolor=COLOR_WHITE, bgcolor=COLOR_YELLOW) | ||||||
|             self.label("B", "Next", textcolor=COLOR_WHITE, bgcolor=COLOR_BLUE) |             self.label("B", "Next", textcolor=COLOR_WHITE, bgcolor=COLOR_BLUE) | ||||||
|  |  | ||||||
|  |         self.label("A", "?", textcolor=COLOR_WHITE, bgcolor=COLOR_BLUE) | ||||||
|  |  | ||||||
|         self.label("X", "Done", textcolor=COLOR_WHITE, bgcolor=COLOR_RED) |         self.label("X", "Done", textcolor=COLOR_WHITE, bgcolor=COLOR_RED) | ||||||
|  |  | ||||||
|  |         if self._help_mode: | ||||||
|  |             self.overlay(help) | ||||||
|  |  | ||||||
|     def button_a(self): |     def button_a(self): | ||||||
|         pass |         self._help_mode = not self._help_mode | ||||||
|  |         return True | ||||||
|  |  | ||||||
|     def button_b(self): |     def button_b(self): | ||||||
|  |         if self._help_mode: | ||||||
|  |             return True | ||||||
|         if self._change_mode: |         if self._change_mode: | ||||||
|             option = self._options[self._current_option] |             option = self._options[self._current_option] | ||||||
|             prop = option["prop"] |             prop = option["prop"] | ||||||
| @@ -497,9 +514,12 @@ class EditView(ChannelView): | |||||||
|         if self._change_mode: |         if self._change_mode: | ||||||
|             self._change_mode = False |             self._change_mode = False | ||||||
|             return True |             return True | ||||||
|  |         self._help_mode = False | ||||||
|         return False |         return False | ||||||
|  |  | ||||||
|     def button_y(self): |     def button_y(self): | ||||||
|  |         if self._help_mode: | ||||||
|  |             return True | ||||||
|         if self._change_mode: |         if self._change_mode: | ||||||
|             option = self._options[self._current_option] |             option = self._options[self._current_option] | ||||||
|             prop = option["prop"] |             prop = option["prop"] | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user