247 improve the glance view (#249)
2s seems to be a good compromise between the status flickering and the update delay. When the menu is cached there's no flickering.
@ -1,4 +1,4 @@
|
|||||||
[Home](README.md) | [Switches](examples/Switches.md) | [Actions](examples/Actions.md) | [Templates](examples/Templates.md) | Battery Reporting | [Trouble Shooting](TroubleShooting.md) | [Version History](HISTORY.md)
|
[Home](README.md) | [Switches](examples/Switches.md) | [Actions](examples/Actions.md) | [Templates](examples/Templates.md) | [Glance](examples/Glance.md) | [Background Service](BackgroundService.md) | [Trouble Shooting](TroubleShooting.md) | [Version History](HISTORY.md)
|
||||||
|
|
||||||
# Background Service
|
# Background Service
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
[Home](README.md) | [Switches](examples/Switches.md) | [Actions](examples/Actions.md) | [Templates](examples/Templates.md) | [Background Service](BackgroundService.md) | [Trouble Shooting](TroubleShooting.md) | Version History
|
[Home](README.md) | [Switches](examples/Switches.md) | [Actions](examples/Actions.md) | [Templates](examples/Templates.md) | [Glance](examples/Glance.md) | [Background Service](BackgroundService.md) | [Trouble Shooting](TroubleShooting.md) | [Version History](HISTORY.md)
|
||||||
|
|
||||||
# Version History
|
# Version History
|
||||||
|
|
||||||
@ -34,7 +34,7 @@
|
|||||||
| 2.19 | A template to evaluate is now optionally allowed on both `group` and `toggle` menu items. The template to evaluate is non-optional on a `template` menu item. All updates are performed in a single HTTP GET request for efficiency. Bug fix for negative heading values. Vibration now (optionally) confirms toggle menu items being tapped. |
|
| 2.19 | A template to evaluate is now optionally allowed on both `group` and `toggle` menu items. The template to evaluate is non-optional on a `template` menu item. All updates are performed in a single HTTP GET request for efficiency. Bug fix for negative heading values. Vibration now (optionally) confirms toggle menu items being tapped. |
|
||||||
| 2.20 | Simplified the code base now that templates have been requested in all menu items. This means the `template` menu item became a superset of `tap`. Therefore the `tap` code has been has been upgraded to include `template` and the latter deprecated. JSON menu definitions continue to support `template` items by instantiating a `tap` menu item, but the schema marks them as deprecated and users should migrate their menu definitions now. Use the [web editor](https://house-of-abbey.github.io/GarminHomeAssistant/web/) for assistance with changes. |
|
| 2.20 | Simplified the code base now that templates have been requested in all menu items. This means the `template` menu item became a superset of `tap`. Therefore the `tap` code has been has been upgraded to include `template` and the latter deprecated. JSON menu definitions continue to support `template` items by instantiating a `tap` menu item, but the schema marks them as deprecated and users should migrate their menu definitions now. Use the [web editor](https://house-of-abbey.github.io/GarminHomeAssistant/web/) for assistance with changes. |
|
||||||
| 2.21 | Added 7 new devices (`edge1050`, `enduro3`, `fenix843mm`, `fenix847mm`, `fenix8solar47mm`, `fenix8solar51mm`, `fenixe`) and upgraded the SDK to 7.3.0. Fix for a bug on Edge devices introduced by v2.16 activity reporting improvements. |
|
| 2.21 | Added 7 new devices (`edge1050`, `enduro3`, `fenix843mm`, `fenix847mm`, `fenix8solar47mm`, `fenix8solar51mm`, `fenixe`) and upgraded the SDK to 7.3.0. Fix for a bug on Edge devices introduced by v2.16 activity reporting improvements. |
|
||||||
| 2.22 | Major feature release adding an optional PIN to menu items. This significant new feature has been provided by [moesterheld](https://github.com/moesterheld). Please do not rely on this application for security. Use at your own risk! |
|
| 2.22 | <img src="images/pin_view.png" width="200" title="PIN Entry View"/><br/>Major feature release adding an optional PIN to menu items. This significant new feature has been provided by [moesterheld](https://github.com/moesterheld). Please do not rely on this application for security. Use at your own risk! |
|
||||||
| 2.23 | Added "info" menu item for displaying information via a template without a tap or toggle. Essentially like the old 'template' type that was deprecated when all items were amended to display evaluated templates. That action removed the display only items too hastily. Added 5 new devices in the model range Instinct 3 and Instinct E. |
|
| 2.23 | Added "info" menu item for displaying information via a template without a tap or toggle. Essentially like the old 'template' type that was deprecated when all items were amended to display evaluated templates. That action removed the display only items too hastily. Added 5 new devices in the model range Instinct 3 and Instinct E. |
|
||||||
| 2.24 | Experiment to prevent new Webhook IDs being created unnecessarily. Reduced the latency for the first menu update. Added 4 new devices: approachs50, descentg2, descentmk1, and gpsmap66. |
|
| 2.24 | Experiment to prevent new Webhook IDs being created unnecessarily. Reduced the latency for the first menu update. Added 4 new devices: approachs50, descentg2, descentmk1, and gpsmap66. |
|
||||||
| 2.25 | 2 Bug fixes. First time startup issues caused by v2.24 change and a fix for pure numbers in returned templates. |
|
| 2.25 | 2 Bug fixes. First time startup issues caused by v2.24 change and a fix for pure numbers in returned templates. |
|
||||||
@ -42,3 +42,4 @@
|
|||||||
| 2.27 | Trivial bug fix for the glance view to prevent the "Unconfigured" result being erroneously displayed because the settings were not yet pulled from persistent storage. |
|
| 2.27 | Trivial bug fix for the glance view to prevent the "Unconfigured" result being erroneously displayed because the settings were not yet pulled from persistent storage. |
|
||||||
| 2.28 | Added support for Vivoactive 6 device which also required an SDK update to 8.1.0. |
|
| 2.28 | Added support for Vivoactive 6 device which also required an SDK update to 8.1.0. |
|
||||||
| 2.29 | Added support for three new devices, Forerunners 570 42mm & 47mm and 970. |
|
| 2.29 | Added support for three new devices, Forerunners 570 42mm & 47mm and 970. |
|
||||||
|
| 2.30 | <img src="images/Venu2_glance_default.png" width="200" title="Default Glance"/><br/>Extensive re-work of the [Glance](examples/Glance.md) view, including the ability to customise it with a user supplied template. |
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
Home | [Switches](examples/Switches.md) | [Actions](examples/Actions.md) | [Templates](examples/Templates.md) | [Background Service](BackgroundService.md) | [Trouble Shooting](TroubleShooting.md) | [Version History](HISTORY.md)
|
[Home](README.md) | [Switches](examples/Switches.md) | [Actions](examples/Actions.md) | [Templates](examples/Templates.md) | [Glance](examples/Glance.md) | [Background Service](BackgroundService.md) | [Trouble Shooting](TroubleShooting.md) | [Version History](HISTORY.md)
|
||||||
|
|
||||||
# GarminHomeAssistant
|
# GarminHomeAssistant
|
||||||
|
|
||||||
@ -24,7 +24,7 @@ As of version 2.0, there are now two installable versions. For older devices bef
|
|||||||
|
|
||||||
| Version | Explanation |
|
| Version | Explanation |
|
||||||
|------------------------|-------------|
|
|------------------------|-------------|
|
||||||
| Application (original) | For newer devices that allow glance views in their applications, e.g. Venu 2, the GarminHomeAssistant application can be started either from a glance or from the list of applications and activities. Head over to the [GarminHomeAssistant](https://apps.garmin.com/en-US/apps/61c91d28-ec5e-438d-9f83-39e9f45b199d) application page on the [Connect IQ application store](https://apps.garmin.com/en-US/) to download the application. The application can be started two different ways, either from the glance in the carousel, or as an application from the list of applications & activities. With the latter, it is worth marking the application as a favourite.<br/><img src="images/Venu2_app_start.png" width="200" title="Venu 2" style="margin:5px"/><img src="images/Vivoactive3_app_start.jpg" width="200" title="Venu 2" style="margin:5px"/><br/>If you place the application on your list of favourites, and rearrange it to appear near the top, then the item is just one button press away from the watch face. This second picture here shows the application menu on a Vivoactive 3 watch.<br/><img src="images/Venu2_glance_start.png" width="200" title="Venu 2" style="margin:5px"/><br/>On newer watches, you can also start the application from the glance carousel. The glance view here typically displays some trackable status, so ours provides some early indication of availability. Older watches will still allow you to start this application from the list of applications and activities. |
|
| Application (original) | For newer devices that allow glance views in their applications, e.g. Venu 2, the GarminHomeAssistant application can be started either from a glance or from the list of applications and activities. Head over to the [GarminHomeAssistant](https://apps.garmin.com/en-US/apps/61c91d28-ec5e-438d-9f83-39e9f45b199d) application page on the [Connect IQ application store](https://apps.garmin.com/en-US/) to download the application. The application can be started two different ways, either from the glance in the carousel, or as an application from the list of applications & activities. With the latter, it is worth marking the application as a favourite.<br/><img src="images/Venu2_app_start.png" width="200" title="Venu 2" style="margin:5px"/><img src="images/Vivoactive3_app_start.jpg" width="200" title="Vivoactive 3" style="margin:5px"/><br/>If you place the application on your list of favourites, and rearrange it to appear near the top, then the item is just one button press away from the watch face. This second picture here shows the application menu on a Vivoactive 3 watch.<br/><img src="images/Venu2_glance_default.png" width="200" title="Venu 2" style="margin:5px"/><br/>On newer watches, you can also start the application from the glance carousel. The glance view here typically displays some trackable status, so ours provides some early indication of availability. Older watches will still allow you to start this application from the list of applications and activities. |
|
||||||
| Widget | **"Maintenance only mode"** so no new features will be added to this version.<br>For older devices that use widgets, e.g. Venu (1) as opposed to applications with "glances", the GarminHomeAssistant application can instead be started from the widget carousel. This is a separate item in the Connect IQ AppStore and with this installation, the application will no longer appear in the list of applications and activities. Head over to the [GarminHomeAssistant](https://apps.garmin.com/en-US/apps/) widget page on the [Connect IQ application store](https://apps.garmin.com/en-US/) to download the widget.<br/><img src="images/Venu_Widget_sim.png" width="200" title="Venu 2" style="margin:5px"/><br/>Typically the widget view implements something similar to the glance view, e.g. status, and exists in a widget carousel to allow you to select an application to launch.<br>**Please note that memory in widgets is more limited than applications. This means a large menu definition can crash the widget without the code catching the error.**<br> This version was born out of the application version and from Ver 2.0 shared the same source code repository until Ver 2.8 when they were [separated](https://github.com/house-of-abbey/GarminHomeAssistantWidget) to allow the application version to take advantage of its increase memory availability. |
|
| Widget | **"Maintenance only mode"** so no new features will be added to this version.<br>For older devices that use widgets, e.g. Venu (1) as opposed to applications with "glances", the GarminHomeAssistant application can instead be started from the widget carousel. This is a separate item in the Connect IQ AppStore and with this installation, the application will no longer appear in the list of applications and activities. Head over to the [GarminHomeAssistant](https://apps.garmin.com/en-US/apps/) widget page on the [Connect IQ application store](https://apps.garmin.com/en-US/) to download the widget.<br/><img src="images/Venu_Widget_sim.png" width="200" title="Venu 2" style="margin:5px"/><br/>Typically the widget view implements something similar to the glance view, e.g. status, and exists in a widget carousel to allow you to select an application to launch.<br>**Please note that memory in widgets is more limited than applications. This means a large menu definition can crash the widget without the code catching the error.**<br> This version was born out of the application version and from Ver 2.0 shared the same source code repository until Ver 2.8 when they were [separated](https://github.com/house-of-abbey/GarminHomeAssistantWidget) to allow the application version to take advantage of its increase memory availability. |
|
||||||
|
|
||||||
### Features
|
### Features
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
[Home](README.md) | [Switches](examples/Switches.md) | [Actions](examples/Actions.md) | [Templates](examples/Templates.md) | [Background Service](BackgroundService.md) | Trouble Shooting | [Version History](HISTORY.md)
|
[Home](README.md) | [Switches](examples/Switches.md) | [Actions](examples/Actions.md) | [Templates](examples/Templates.md) | [Glance](examples/Glance.md) | [Background Service](BackgroundService.md) | [Trouble Shooting](TroubleShooting.md) | [Version History](HISTORY.md)
|
||||||
|
|
||||||
# Troubleshooting Guides
|
# Troubleshooting Guides
|
||||||
|
|
||||||
|
@ -2,14 +2,19 @@
|
|||||||
"$schema": "https://json-schema.org/draft/2020-12/schema",
|
"$schema": "https://json-schema.org/draft/2020-12/schema",
|
||||||
"type": "object",
|
"type": "object",
|
||||||
"properties": {
|
"properties": {
|
||||||
|
"$schema": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "The schema will prevent JSON file errors."
|
||||||
|
},
|
||||||
"title": {
|
"title": {
|
||||||
"type": "string"
|
"type": "string",
|
||||||
|
"description": "Top level menu title"
|
||||||
|
},
|
||||||
|
"glance": {
|
||||||
|
"$ref": "#/$defs/glance"
|
||||||
},
|
},
|
||||||
"items": {
|
"items": {
|
||||||
"$ref": "#/$defs/items"
|
"$ref": "#/$defs/items"
|
||||||
},
|
|
||||||
"$schema": {
|
|
||||||
"type": "string"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"required": ["title", "items"],
|
"required": ["title", "items"],
|
||||||
@ -29,8 +34,7 @@
|
|||||||
"const": "toggle"
|
"const": "toggle"
|
||||||
},
|
},
|
||||||
"content": {
|
"content": {
|
||||||
"$ref": "#/$defs/content",
|
"$ref": "#/$defs/content"
|
||||||
"description": "Optional in a toggle."
|
|
||||||
},
|
},
|
||||||
"tap_action": {
|
"tap_action": {
|
||||||
"type": "object",
|
"type": "object",
|
||||||
@ -135,8 +139,7 @@
|
|||||||
"const": "tap"
|
"const": "tap"
|
||||||
},
|
},
|
||||||
"content": {
|
"content": {
|
||||||
"$ref": "#/$defs/content",
|
"$ref": "#/$defs/content"
|
||||||
"description": "Optional in a tap."
|
|
||||||
},
|
},
|
||||||
"service": {
|
"service": {
|
||||||
"$ref": "#/$defs/entity",
|
"$ref": "#/$defs/entity",
|
||||||
@ -174,8 +177,7 @@
|
|||||||
"const": "group"
|
"const": "group"
|
||||||
},
|
},
|
||||||
"content": {
|
"content": {
|
||||||
"$ref": "#/$defs/content",
|
"$ref": "#/$defs/content"
|
||||||
"description": "Optional in a group."
|
|
||||||
},
|
},
|
||||||
"items": {
|
"items": {
|
||||||
"$ref": "#/$defs/items"
|
"$ref": "#/$defs/items"
|
||||||
@ -190,7 +192,6 @@
|
|||||||
},
|
},
|
||||||
"items": {
|
"items": {
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"maxItems": 16,
|
|
||||||
"items": {
|
"items": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
@ -248,7 +249,8 @@
|
|||||||
"required": ["service"]
|
"required": ["service"]
|
||||||
},
|
},
|
||||||
"content": {
|
"content": {
|
||||||
"title": "Jinja2 template defining the text to display.",
|
"title": "Home Assistant Template",
|
||||||
|
"description": "Jinja2 template defining the text to display. Must be included in an 'info'. Optional in a 'toggle', 'tap' and 'group'. Special characters may not render in the glance context.",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
"confirm": {
|
"confirm": {
|
||||||
@ -262,6 +264,35 @@
|
|||||||
"default": false,
|
"default": false,
|
||||||
"title": "PIN Confirmation",
|
"title": "PIN Confirmation",
|
||||||
"description": "Optional PIN confirmation of the action before execution as a precaution. Has precedence over 'confirm': true if both are set."
|
"description": "Optional PIN confirmation of the action before execution as a precaution. Has precedence over 'confirm': true if both are set."
|
||||||
|
},
|
||||||
|
"glance": {
|
||||||
|
"type": "object",
|
||||||
|
"title": "Glance customisation",
|
||||||
|
"oneOf": [
|
||||||
|
{
|
||||||
|
"properties": {
|
||||||
|
"type": {
|
||||||
|
"title": "Glance type",
|
||||||
|
"description": "One of 'info' or 'status'. 'info' renders the template specified in the 'content' field inside the glance view. 'status' reverts to the default glance view and ignores the 'content' field. This allows for disabling the template temporarily.",
|
||||||
|
"const": "info"
|
||||||
|
},
|
||||||
|
"content": {
|
||||||
|
"$ref": "#/$defs/content"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["type", "content"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"properties": {
|
||||||
|
"type": {
|
||||||
|
"title": "Glance type",
|
||||||
|
"description": "One of 'info' or 'status'.",
|
||||||
|
"const": "status"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["type"]
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
[Home](../README.md) | [Switches](Switches.md) | Actions | [Templates](Templates.md) | [Background Service](../BackgroundService.md) | [Trouble Shooting](../TroubleShooting.md) | [Version History](../HISTORY.md)
|
[Home](../README.md) | [Switches](Switches.md) | [Actions](Actions.md) | [Templates](Templates.md) | [Glance](Glance.md) | [Background Service](../BackgroundService.md) | [Trouble Shooting](../TroubleShooting.md) | [Version History](../HISTORY.md)
|
||||||
|
|
||||||
|
|
||||||
# Actions
|
# Actions
|
||||||
|
|
||||||
|
80
examples/Glance.md
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
[Home](../README.md) | [Switches](Switches.md) | [Actions](Actions.md) | [Templates](Templates.md) | [Glance](Glance.md) | [Background Service](../BackgroundService.md) | [Trouble Shooting](../TroubleShooting.md) | [Version History](../HISTORY.md)
|
||||||
|
|
||||||
|
# Glance
|
||||||
|
|
||||||
|
Since [version 2.30](../History.md), it is possible to ovverride the text displayed on the Glance view. This page explains how to customise the text.
|
||||||
|
|
||||||
|
|
||||||
|
## Default View
|
||||||
|
|
||||||
|
The default view has always been to display the status of the menu and API availability to indicate if there's a problem. This view has now been updated to be more colourful.
|
||||||
|
|
||||||
|
<img src="../images/Venu2_glance_default.png" width="200" title="Venu 2 Default Glance"/>
|
||||||
|
|
||||||
|
When either the API or the menu file is inaccessible, the fields will turn red.
|
||||||
|
|
||||||
|
|
||||||
|
## Customised View
|
||||||
|
|
||||||
|
In order to customise the Glance view you need to add a `glance` field to the top level of the JSON menu file as illustrated here:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"$schema": "https://raw.githubusercontent.com/house-of-abbey/GarminHomeAssistant/main/config.schema.json",
|
||||||
|
"title": "Home",
|
||||||
|
"glance": {
|
||||||
|
"type": "info",
|
||||||
|
"content": "Text: {% .. %}"
|
||||||
|
},
|
||||||
|
"items": [...]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
For example:
|
||||||
|
|
||||||
|
<img src="../images/Venu2_glance_custom.png" width="200" title="Venu 2 Customised Glance"/>
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"$schema": "https://raw.githubusercontent.com/house-of-abbey/GarminHomeAssistant/main/config.schema.json",
|
||||||
|
"glance": {
|
||||||
|
"type": "info",
|
||||||
|
"content": "Solar Battery: {{ states('sensor.battery_capacity_charge') }}%"
|
||||||
|
},
|
||||||
|
:
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
You may make this as complicated as you like! But you have limited space and only ASCII text characters. **It is best to turn on menu caching in order to speed up the display of the template**. The display is then nearly instantaneous.
|
||||||
|
|
||||||
|
The default view will persist showing until the errors are resolved. In order to extract the custom glance template both the menu and the API are required. So it is logical that the two tests must pass first. The exception here is if the menu is cached, in which case only the API needs to pass.
|
||||||
|
|
||||||
|
> [!IMPORTANT]
|
||||||
|
> Sadly what you cannot do is use special characters like: 🌞🔋⛅🪫. Whilst these do display in menu items, they do not seem to work on the Glance view. We really like them, so have tried but failed. Only ASCII text appears to be supported by the Garmin Connect IQ SDK's Glance View. This is not something we have any control over, please do not request this to be "fixed".
|
||||||
|
|
||||||
|
It is possible to revert to the default glance content without deleting the template by changing the `type` to `status`.
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"$schema": "https://raw.githubusercontent.com/house-of-abbey/GarminHomeAssistant/main/config.schema.json",
|
||||||
|
"title": "Home",
|
||||||
|
"glance": {
|
||||||
|
"type": "status",
|
||||||
|
"content": "Text: {% .. %}"
|
||||||
|
},
|
||||||
|
"items": [...]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
So the glance view object has a `type` field with two possible values: `info` and `status`. When the type is `status` the `content` field is not required.
|
||||||
|
|
||||||
|
|
||||||
|
## Displayed Errors
|
||||||
|
|
||||||
|
The following shows the default glance when the menu file is not available at the specified URL.
|
||||||
|
|
||||||
|
<img src="../images/Venu2_glance_no_menu.png" width="200" title="Venu 2 Glance showing errors"/>
|
||||||
|
|
||||||
|
Once the custom glance template has been retrieved and evaluated the display will change. Should the connectivity to your Home Assistant then be lost, e.g. you move out of range of your phone, the glance reflects this in the colour of the residual two rectangles. The top one remains an indicator for the API, and the bottom rectangle remains an indicator for the menu availability, reflecting the original placement in the default glance view that has now been replaced.
|
||||||
|
|
||||||
|
<img src="../images/Venu2_glance_no_bt.png" width="200" title="Venu 2 Glance showing lost connectivity"/>
|
@ -1,4 +1,5 @@
|
|||||||
[Home](../README.md) | Switches | [Actions](Actions.md) | [Templates](Templates.md) | [Background Service](../BackgroundService.md) | [Trouble Shooting](../TroubleShooting.md) | [Version History](../HISTORY.md)
|
[Home](../README.md) | [Switches](Switches.md) | [Actions](Actions.md) | [Templates](Templates.md) | [Glance](Glance.md) | [Background Service](../BackgroundService.md) | [Trouble Shooting](../TroubleShooting.md) | [Version History](../HISTORY.md)
|
||||||
|
|
||||||
|
|
||||||
# Switches
|
# Switches
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
[Home](../README.md) | [Switches](Switches.md) | [Actions](Actions.md) | Templates | [Background Service](../BackgroundService.md) | [Trouble Shooting](../TroubleShooting.md) | [Version History](../HISTORY.md)
|
[Home](../README.md) | [Switches](Switches.md) | [Actions](Actions.md) | [Templates](Templates.md) | [Glance](Glance.md) | [Background Service](../BackgroundService.md) | [Trouble Shooting](../TroubleShooting.md) | [Version History](../HISTORY.md)
|
||||||
|
|
||||||
# Templates
|
# Templates
|
||||||
|
|
||||||
|
@ -105,7 +105,6 @@ echo.
|
|||||||
--private-key %SRC%\..\developer_key ^
|
--private-key %SRC%\..\developer_key ^
|
||||||
--package-app ^
|
--package-app ^
|
||||||
--release
|
--release
|
||||||
rem --warn
|
|
||||||
|
|
||||||
echo.
|
echo.
|
||||||
echo Finished exporting HomeAssistant
|
echo Finished exporting HomeAssistant
|
||||||
|
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 23 KiB |
BIN
images/Venu2_glance_custom.png
Normal file
After Width: | Height: | Size: 8.3 KiB |
BIN
images/Venu2_glance_default.png
Normal file
After Width: | Height: | Size: 8.7 KiB |
BIN
images/Venu2_glance_no_bt.png
Normal file
After Width: | Height: | Size: 8.2 KiB |
BIN
images/Venu2_glance_no_menu.png
Normal file
After Width: | Height: | Size: 8.8 KiB |
Before Width: | Height: | Size: 9.1 KiB |
@ -25,7 +25,7 @@ class Globals {
|
|||||||
|
|
||||||
//! Time to let the existing HTTP responses get serviced after a
|
//! Time to let the existing HTTP responses get serviced after a
|
||||||
//! `Communications.NETWORK_RESPONSE_OUT_OF_MEMORY` response code.
|
//! `Communications.NETWORK_RESPONSE_OUT_OF_MEMORY` response code.
|
||||||
static const scApiBackoff = 1000; // ms
|
static const scApiBackoff = 2000; // ms
|
||||||
|
|
||||||
//! Needs to be long enough to enable a "double ESC" to quit the application from
|
//! Needs to be long enough to enable a "double ESC" to quit the application from
|
||||||
//! an ErrorView.
|
//! an ErrorView.
|
||||||
|
@ -24,18 +24,20 @@ using Toybox.Timer;
|
|||||||
//
|
//
|
||||||
(:glance, :background)
|
(:glance, :background)
|
||||||
class HomeAssistantApp extends Application.AppBase {
|
class HomeAssistantApp extends Application.AppBase {
|
||||||
private var mApiStatus as Lang.String or Null;
|
private var mApiStatus as Lang.String or Null;
|
||||||
private var mMenuStatus as Lang.String or Null;
|
private var mMenuStatus as Lang.String or Null;
|
||||||
private var mHaMenu as HomeAssistantView or Null;
|
private var mHaMenu as HomeAssistantView or Null;
|
||||||
private var mQuitTimer as QuitTimer or Null;
|
private var mGlanceTemplate as Lang.String or Null = null;
|
||||||
private var mGlanceTimer as Timer.Timer or Null;
|
private var mGlanceText as Lang.String or Null = null;
|
||||||
private var mUpdateTimer as Timer.Timer or Null;
|
private var mQuitTimer as QuitTimer or Null;
|
||||||
|
private var mGlanceTimer as Timer.Timer or Null;
|
||||||
|
private var mUpdateTimer as Timer.Timer or Null;
|
||||||
// Array initialised by onReturnFetchMenuConfig()
|
// Array initialised by onReturnFetchMenuConfig()
|
||||||
private var mItemsToUpdate as Lang.Array<HomeAssistantToggleMenuItem or HomeAssistantTapMenuItem or HomeAssistantGroupMenuItem> or Null;
|
private var mItemsToUpdate as Lang.Array<HomeAssistantToggleMenuItem or HomeAssistantTapMenuItem or HomeAssistantGroupMenuItem> or Null;
|
||||||
private var mIsGlance as Lang.Boolean = false;
|
private var mIsGlance as Lang.Boolean = false;
|
||||||
private var mIsApp as Lang.Boolean = false; // Or Widget
|
private var mIsApp as Lang.Boolean = false; // Or Widget
|
||||||
private var mUpdating as Lang.Boolean = false; // Don't start a second chain of updates
|
private var mUpdating as Lang.Boolean = false; // Don't start a second chain of updates
|
||||||
private var mTemplates as Lang.Dictionary = {};
|
private var mTemplates as Lang.Dictionary = {};
|
||||||
|
|
||||||
//! Class Constructor
|
//! Class Constructor
|
||||||
//
|
//
|
||||||
@ -203,7 +205,9 @@ class HomeAssistantApp extends Application.AppBase {
|
|||||||
mMenuStatus = WatchUi.loadResource($.Rez.Strings.Available) as Lang.String;
|
mMenuStatus = WatchUi.loadResource($.Rez.Strings.Available) as Lang.String;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!mIsGlance) {
|
if (mIsGlance) {
|
||||||
|
glanceTemplate(data);
|
||||||
|
} else {
|
||||||
if (data == null) {
|
if (data == null) {
|
||||||
ErrorView.show(WatchUi.loadResource($.Rez.Strings.NoJson) as Lang.String);
|
ErrorView.show(WatchUi.loadResource($.Rez.Strings.NoJson) as Lang.String);
|
||||||
} else {
|
} else {
|
||||||
@ -272,7 +276,9 @@ class HomeAssistantApp extends Application.AppBase {
|
|||||||
} else {
|
} else {
|
||||||
mMenuStatus = WatchUi.loadResource($.Rez.Strings.Cached) as Lang.String;
|
mMenuStatus = WatchUi.loadResource($.Rez.Strings.Cached) as Lang.String;
|
||||||
WatchUi.requestUpdate();
|
WatchUi.requestUpdate();
|
||||||
if (!mIsGlance) {
|
if (mIsGlance) {
|
||||||
|
glanceTemplate(menu);
|
||||||
|
} else {
|
||||||
buildMenu(menu);
|
buildMenu(menu);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
@ -303,6 +309,22 @@ class HomeAssistantApp extends Application.AppBase {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//! Extract the optional template to override the default glance view.
|
||||||
|
//
|
||||||
|
function glanceTemplate(menu as Lang.Dictionary) {
|
||||||
|
if (menu != null) {
|
||||||
|
if (menu.get("glance") != null) {
|
||||||
|
var glance = menu.get("glance") as Lang.Dictionary;
|
||||||
|
if (glance.get("type").equals("info")) {
|
||||||
|
mGlanceTemplate = glance.get("content") as Lang.String;
|
||||||
|
// System.println("HomeAssistantApp glanceTemplate() " + mGlanceTemplate);
|
||||||
|
} else { // if glance.get("type").equals("status")
|
||||||
|
mGlanceTemplate = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//! Callback function for each menu update GET request.
|
//! Callback function for each menu update GET request.
|
||||||
//!
|
//!
|
||||||
//! @param responseCode Response code.
|
//! @param responseCode Response code.
|
||||||
@ -416,10 +438,9 @@ class HomeAssistantApp extends Application.AppBase {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// https://developers.home-assistant.io/docs/api/native-app-integration/sending-data/#render-templates
|
// https://developers.home-assistant.io/docs/api/native-app-integration/sending-data/#render-templates
|
||||||
var url = Settings.getApiUrl() + "/webhook/" + Settings.getWebhookId();
|
|
||||||
// System.println("HomeAssistantApp updateMenuItems() URL=" + url + ", Template='" + mTemplate + "'");
|
// System.println("HomeAssistantApp updateMenuItems() URL=" + url + ", Template='" + mTemplate + "'");
|
||||||
Communications.makeWebRequest(
|
Communications.makeWebRequest(
|
||||||
url,
|
Settings.getApiUrl() + "/webhook/" + Settings.getWebhookId(),
|
||||||
{
|
{
|
||||||
"type" => "render_template",
|
"type" => "render_template",
|
||||||
"data" => mTemplates
|
"data" => mTemplates
|
||||||
@ -548,6 +569,100 @@ class HomeAssistantApp extends Application.AppBase {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//! Callback function after completing the GET request to render the glance template.
|
||||||
|
//!
|
||||||
|
//! @param responseCode Response code.
|
||||||
|
//! @param data Response data.
|
||||||
|
//
|
||||||
|
(:glance)
|
||||||
|
function onReturnFetchGlanceContent(
|
||||||
|
responseCode as Lang.Number,
|
||||||
|
data as Null or Lang.Dictionary or Lang.String
|
||||||
|
) as Void {
|
||||||
|
// System.println("HomeAssistantApp onReturnFetchGlanceContent() Response Code: " + responseCode);
|
||||||
|
// System.println("HomeAssistantApp onReturnFetchGlanceContent() Response Data: " + data);
|
||||||
|
|
||||||
|
switch (responseCode) {
|
||||||
|
case Communications.BLE_HOST_TIMEOUT:
|
||||||
|
case Communications.BLE_CONNECTION_UNAVAILABLE:
|
||||||
|
// System.println("HomeAssistantApp onReturnFetchGlanceContent() Response Code: BLE_HOST_TIMEOUT or BLE_CONNECTION_UNAVAILABLE, Bluetooth connection severed.");
|
||||||
|
if (!mIsGlance) {
|
||||||
|
ErrorView.show(WatchUi.loadResource($.Rez.Strings.NoPhone) as Lang.String);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Communications.BLE_QUEUE_FULL:
|
||||||
|
// System.println("HomeAssistantApp onReturnFetchGlanceContent() Response Code: BLE_QUEUE_FULL, API calls too rapid.");
|
||||||
|
if (!mIsGlance) {
|
||||||
|
ErrorView.show(WatchUi.loadResource($.Rez.Strings.ApiFlood) as Lang.String);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Communications.NETWORK_REQUEST_TIMED_OUT:
|
||||||
|
// System.println("HomeAssistantApp onReturnFetchGlanceContent() Response Code: NETWORK_REQUEST_TIMED_OUT, check Internet connection.");
|
||||||
|
if (!mIsGlance) {
|
||||||
|
ErrorView.show(WatchUi.loadResource($.Rez.Strings.NoResponse) as Lang.String);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Communications.INVALID_HTTP_BODY_IN_NETWORK_RESPONSE:
|
||||||
|
// System.println("HomeAssistantApp onReturnFetchGlanceContent() Response Code: INVALID_HTTP_BODY_IN_NETWORK_RESPONSE, check JSON is returned.");
|
||||||
|
if (!mIsGlance) {
|
||||||
|
ErrorView.show(WatchUi.loadResource($.Rez.Strings.NoJson) as Lang.String);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 404:
|
||||||
|
// System.println("HomeAssistantApp onReturnFetchGlanceContent() Response Code: 404, page not found. Check Configuration URL setting.");
|
||||||
|
if (!mIsGlance) {
|
||||||
|
ErrorView.show(WatchUi.loadResource($.Rez.Strings.ConfigUrlNotFound) as Lang.String);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 200:
|
||||||
|
if (data != null) {
|
||||||
|
mGlanceText = data.get("glanceTemplate");
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
// System.println("HomeAssistantApp onReturnFetchGlanceContent(): Unhandled HTTP response code = " + responseCode);
|
||||||
|
if (!mIsGlance) {
|
||||||
|
ErrorView.show(WatchUi.loadResource($.Rez.Strings.UnhandledHttpErr) as Lang.String + responseCode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
WatchUi.requestUpdate();
|
||||||
|
}
|
||||||
|
|
||||||
|
//! Construct the GET request to convert the optional glance template to text for display.
|
||||||
|
//
|
||||||
|
(:glance)
|
||||||
|
function fetchGlanceContent() as Void {
|
||||||
|
if (mGlanceTemplate != null) {
|
||||||
|
// https://developers.home-assistant.io/docs/api/native-app-integration/sending-data/#render-templates
|
||||||
|
Communications.makeWebRequest(
|
||||||
|
Settings.getApiUrl() + "/webhook/" + Settings.getWebhookId(),
|
||||||
|
{
|
||||||
|
"type" => "render_template",
|
||||||
|
"data" => {
|
||||||
|
"glanceTemplate" => {
|
||||||
|
"template" => mGlanceTemplate
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
:method => Communications.HTTP_REQUEST_METHOD_POST,
|
||||||
|
:headers => {
|
||||||
|
"Content-Type" => Communications.REQUEST_CONTENT_TYPE_JSON
|
||||||
|
},
|
||||||
|
:responseType => Communications.HTTP_RESPONSE_CONTENT_TYPE_JSON
|
||||||
|
},
|
||||||
|
method(:onReturnFetchGlanceContent)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//! Record the API status result.
|
//! Record the API status result.
|
||||||
//!
|
//!
|
||||||
//! @param s A string describing the API status
|
//! @param s A string describing the API status
|
||||||
@ -574,6 +689,16 @@ class HomeAssistantApp extends Application.AppBase {
|
|||||||
return mMenuStatus;
|
return mMenuStatus;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//! Return the optional glance text that overrides the default glance content. This
|
||||||
|
//! is derived from the glance template.
|
||||||
|
//!
|
||||||
|
//! @return A string derived from the glance template
|
||||||
|
//
|
||||||
|
(:glance)
|
||||||
|
function getGlanceText() as Lang.String or Null {
|
||||||
|
return mGlanceText;
|
||||||
|
}
|
||||||
|
|
||||||
//! Return the Menu construction status.
|
//! Return the Menu construction status.
|
||||||
//!
|
//!
|
||||||
//! @return A Boolean indicating if the menu is loaded into the application.
|
//! @return A Boolean indicating if the menu is loaded into the application.
|
||||||
@ -623,12 +748,21 @@ class HomeAssistantApp extends Application.AppBase {
|
|||||||
return [new HomeAssistantGlanceView(self)];
|
return [new HomeAssistantGlanceView(self)];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//! Return the glance theme.
|
||||||
|
//!
|
||||||
|
//! @return The glance colour
|
||||||
|
//
|
||||||
|
function getGlanceTheme() as Application.AppBase.GlanceTheme {
|
||||||
|
return Application.AppBase.GLANCE_THEME_LIGHT_BLUE;
|
||||||
|
}
|
||||||
|
|
||||||
//! Update the menu and API statuses. Required for the Glance update timer.
|
//! Update the menu and API statuses. Required for the Glance update timer.
|
||||||
//
|
//
|
||||||
function updateStatus() as Void {
|
function updateStatus() as Void {
|
||||||
mGlanceTimer = null;
|
mGlanceTimer = null;
|
||||||
fetchMenuConfig();
|
fetchMenuConfig();
|
||||||
fetchApiStatus();
|
fetchApiStatus();
|
||||||
|
fetchGlanceContent();
|
||||||
}
|
}
|
||||||
|
|
||||||
//! Code for when the application settings are updated.
|
//! Code for when the application settings are updated.
|
||||||
|
@ -21,15 +21,36 @@ using Toybox.Graphics;
|
|||||||
//
|
//
|
||||||
(:glance)
|
(:glance)
|
||||||
class HomeAssistantGlanceView extends WatchUi.GlanceView {
|
class HomeAssistantGlanceView extends WatchUi.GlanceView {
|
||||||
private static const scLeftMargin = 5; // in pixels
|
//! Margin left of the filled rectangle in pixels.
|
||||||
private static const scMidSep = 10; // Middle Separator "text:_text" in pixels
|
private static const scLeftRectMargin = 5;
|
||||||
private var mApp as HomeAssistantApp;
|
//! Filled rectangle width in pixels.
|
||||||
private var mTitle as WatchUi.Text or Null;
|
private static const scRectWidth = 20;
|
||||||
private var mApiText as WatchUi.Text or Null;
|
//! Margin right of the filled rectangle in pixels.
|
||||||
private var mApiStatus as WatchUi.Text or Null;
|
private static const scRightRectMargin = 5;
|
||||||
private var mMenuText as WatchUi.Text or Null;
|
//! Separator between the first column of text and the second in pixels.
|
||||||
private var mMenuStatus as WatchUi.Text or Null;
|
//! i.e. Middle Separator "text:_text"
|
||||||
private var mAntiAlias as Lang.Boolean = false;
|
private static const scMidSep = 10;
|
||||||
|
//! Margin on the right side of the glance in pixels.
|
||||||
|
private static const scRightGlanceMargin = 15;
|
||||||
|
//! Internal margin for the custom template between the border and the text in pixels.
|
||||||
|
private static const scIntCustMargin = 5;
|
||||||
|
//! Margin top and bottom of the rectangles in pixels.
|
||||||
|
private static const scVertMargin = 5;
|
||||||
|
//! Size of the rounded rectangle corners in pixels.
|
||||||
|
private static const scRectRadius = 5;
|
||||||
|
|
||||||
|
//! Dynamically scale the width of the first column of text based on the
|
||||||
|
//! language selection for the word "Menu".
|
||||||
|
private var mTextWidth as Lang.Number = 0;
|
||||||
|
// Re-usable text items for drawing
|
||||||
|
private var mApp as HomeAssistantApp;
|
||||||
|
private var mTitle as WatchUi.Text or Null;
|
||||||
|
private var mApiText as WatchUi.Text or Null;
|
||||||
|
private var mApiStatus as WatchUi.Text or Null;
|
||||||
|
private var mMenuText as WatchUi.Text or Null;
|
||||||
|
private var mMenuStatus as WatchUi.Text or Null;
|
||||||
|
private var mGlanceContent as WatchUi.TextArea or Null;
|
||||||
|
private var mAntiAlias as Lang.Boolean = false;
|
||||||
|
|
||||||
//! Class Constructor
|
//! Class Constructor
|
||||||
//
|
//
|
||||||
@ -46,15 +67,16 @@ class HomeAssistantGlanceView extends WatchUi.GlanceView {
|
|||||||
//! @param dc Device context
|
//! @param dc Device context
|
||||||
//
|
//
|
||||||
function onLayout(dc as Graphics.Dc) as Void {
|
function onLayout(dc as Graphics.Dc) as Void {
|
||||||
var h = dc.getHeight();
|
var h = dc.getHeight();
|
||||||
var tw = dc.getTextWidthInPixels(WatchUi.loadResource($.Rez.Strings.GlanceMenu) as Lang.String, Graphics.FONT_XTINY);
|
|
||||||
|
mTextWidth = dc.getTextWidthInPixels(WatchUi.loadResource($.Rez.Strings.GlanceMenu) as Lang.String + ":", Graphics.FONT_XTINY);
|
||||||
|
|
||||||
mTitle = new WatchUi.Text({
|
mTitle = new WatchUi.Text({
|
||||||
:text => WatchUi.loadResource($.Rez.Strings.AppName) as Lang.String,
|
:text => WatchUi.loadResource($.Rez.Strings.AppName) as Lang.String,
|
||||||
:color => Graphics.COLOR_WHITE,
|
:color => Graphics.COLOR_WHITE,
|
||||||
:font => Graphics.FONT_TINY,
|
:font => Graphics.FONT_TINY,
|
||||||
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
||||||
:locX => scLeftMargin,
|
:locX => scLeftRectMargin,
|
||||||
:locY => 1 * h / 6
|
:locY => 1 * h / 6
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -63,7 +85,7 @@ class HomeAssistantGlanceView extends WatchUi.GlanceView {
|
|||||||
:color => Graphics.COLOR_WHITE,
|
:color => Graphics.COLOR_WHITE,
|
||||||
:font => Graphics.FONT_XTINY,
|
:font => Graphics.FONT_XTINY,
|
||||||
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
||||||
:locX => scLeftMargin,
|
:locX => scLeftRectMargin + scRectWidth + scRightRectMargin,
|
||||||
:locY => 3 * h / 6
|
:locY => 3 * h / 6
|
||||||
});
|
});
|
||||||
mApiStatus = new WatchUi.Text({
|
mApiStatus = new WatchUi.Text({
|
||||||
@ -71,15 +93,16 @@ class HomeAssistantGlanceView extends WatchUi.GlanceView {
|
|||||||
:color => Graphics.COLOR_WHITE,
|
:color => Graphics.COLOR_WHITE,
|
||||||
:font => Graphics.FONT_XTINY,
|
:font => Graphics.FONT_XTINY,
|
||||||
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
||||||
:locX => scLeftMargin + scMidSep + tw,
|
:locX => scLeftRectMargin + scRectWidth + scRightRectMargin + scMidSep + mTextWidth,
|
||||||
:locY => 3 * h / 6
|
:locY => 3 * h / 6
|
||||||
});
|
});
|
||||||
|
|
||||||
mMenuText = new WatchUi.Text({
|
mMenuText = new WatchUi.Text({
|
||||||
:text => WatchUi.loadResource($.Rez.Strings.GlanceMenu) as Lang.String + ":",
|
:text => WatchUi.loadResource($.Rez.Strings.GlanceMenu) as Lang.String + ":",
|
||||||
:color => Graphics.COLOR_WHITE,
|
:color => Graphics.COLOR_WHITE,
|
||||||
:font => Graphics.FONT_XTINY,
|
:font => Graphics.FONT_XTINY,
|
||||||
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
||||||
:locX => scLeftMargin,
|
:locX => scLeftRectMargin + scRectWidth + scRightRectMargin,
|
||||||
:locY => 5 * h / 6
|
:locY => 5 * h / 6
|
||||||
});
|
});
|
||||||
mMenuStatus = new WatchUi.Text({
|
mMenuStatus = new WatchUi.Text({
|
||||||
@ -87,9 +110,20 @@ class HomeAssistantGlanceView extends WatchUi.GlanceView {
|
|||||||
:color => Graphics.COLOR_WHITE,
|
:color => Graphics.COLOR_WHITE,
|
||||||
:font => Graphics.FONT_XTINY,
|
:font => Graphics.FONT_XTINY,
|
||||||
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
||||||
:locX => scLeftMargin + scMidSep + tw,
|
:locX => scLeftRectMargin + scRectWidth + scRightRectMargin + scMidSep + mTextWidth,
|
||||||
:locY => 5 * h / 6
|
:locY => 5 * h / 6
|
||||||
});
|
});
|
||||||
|
|
||||||
|
mGlanceContent = new WatchUi.TextArea({
|
||||||
|
:text => "A longer piece of text to wrap.",
|
||||||
|
:color => Graphics.COLOR_WHITE,
|
||||||
|
:font => Graphics.FONT_XTINY,
|
||||||
|
:justification => Graphics.TEXT_JUSTIFY_LEFT | Graphics.TEXT_JUSTIFY_VCENTER,
|
||||||
|
:locX => scLeftRectMargin + scRectWidth + scRightRectMargin + scIntCustMargin,
|
||||||
|
:locY => (2 * h / 6) + scVertMargin,
|
||||||
|
:width => dc.getWidth() - scLeftRectMargin - scRectWidth - scRightRectMargin - (2 * scIntCustMargin) - scRightGlanceMargin,
|
||||||
|
:height => (4 * h / 6) - (2 * scVertMargin)
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
//! Update the view with the latest status text.
|
//! Update the view with the latest status text.
|
||||||
@ -97,8 +131,17 @@ class HomeAssistantGlanceView extends WatchUi.GlanceView {
|
|||||||
//! @param dc Device context
|
//! @param dc Device context
|
||||||
//
|
//
|
||||||
function onUpdate(dc as Graphics.Dc) as Void {
|
function onUpdate(dc as Graphics.Dc) as Void {
|
||||||
|
var h = dc.getHeight();
|
||||||
|
var w = dc.getWidth() - scLeftRectMargin - scRightGlanceMargin;
|
||||||
|
var apiStatus = mApp.getApiStatus();
|
||||||
|
var menuStatus = mApp.getMenuStatus();
|
||||||
|
var glanceText = mApp.getGlanceText();
|
||||||
|
var apiCol;
|
||||||
|
var menuCol;
|
||||||
|
// System.println("HomeAssistantGlanceView onUpdate() glanceText=" + glanceText);
|
||||||
|
|
||||||
GlanceView.onUpdate(dc);
|
GlanceView.onUpdate(dc);
|
||||||
if(mAntiAlias) {
|
if (mAntiAlias) {
|
||||||
dc.setAntiAlias(true);
|
dc.setAntiAlias(true);
|
||||||
}
|
}
|
||||||
dc.setColor(
|
dc.setColor(
|
||||||
@ -106,12 +149,60 @@ class HomeAssistantGlanceView extends WatchUi.GlanceView {
|
|||||||
Graphics.COLOR_TRANSPARENT
|
Graphics.COLOR_TRANSPARENT
|
||||||
);
|
);
|
||||||
dc.clear();
|
dc.clear();
|
||||||
|
mTitle.setColor(Graphics.COLOR_BLUE);
|
||||||
mTitle.draw(dc);
|
mTitle.draw(dc);
|
||||||
mApiText.draw(dc);
|
|
||||||
mApiStatus.setText(mApp.getApiStatus());
|
if (apiStatus.equals(WatchUi.loadResource($.Rez.Strings.Checking))) {
|
||||||
mApiStatus.draw(dc);
|
apiCol = Graphics.COLOR_YELLOW;
|
||||||
mMenuText.draw(dc);
|
} else if (apiStatus.equals(WatchUi.loadResource($.Rez.Strings.Available))) {
|
||||||
mMenuStatus.setText(mApp.getMenuStatus());
|
apiCol = Graphics.COLOR_GREEN;
|
||||||
mMenuStatus.draw(dc);
|
} else {
|
||||||
}
|
apiCol = Graphics.COLOR_RED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (menuStatus.equals(WatchUi.loadResource($.Rez.Strings.Checking))) {
|
||||||
|
menuCol = Graphics.COLOR_YELLOW;
|
||||||
|
} else if (menuStatus.equals(WatchUi.loadResource($.Rez.Strings.Available))) {
|
||||||
|
menuCol = Graphics.COLOR_GREEN;
|
||||||
|
} else if (menuStatus.equals(WatchUi.loadResource($.Rez.Strings.Cached))) {
|
||||||
|
menuCol = Graphics.COLOR_GREEN;
|
||||||
|
} else {
|
||||||
|
menuCol = Graphics.COLOR_RED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (glanceText == null) {
|
||||||
|
// Default Glance View
|
||||||
|
mApiText.draw(dc);
|
||||||
|
mApiStatus.setText(apiStatus);
|
||||||
|
mApiStatus.setColor(apiCol);
|
||||||
|
dc.setColor(apiCol, apiCol);
|
||||||
|
dc.drawRoundedRectangle(scLeftRectMargin, 2 * h / 6 + scVertMargin, w, 2 * h / 6 - (2 * scVertMargin), scRectRadius);
|
||||||
|
dc.fillRoundedRectangle(scLeftRectMargin, 2 * h / 6 + scVertMargin, scRectWidth, 2 * h / 6 - (2 * scVertMargin), scRectRadius);
|
||||||
|
mApiStatus.draw(dc);
|
||||||
|
|
||||||
|
mMenuText.draw(dc);
|
||||||
|
mMenuStatus.setText(menuStatus);
|
||||||
|
mMenuStatus.setColor(menuCol);
|
||||||
|
dc.setColor(menuCol, menuCol);
|
||||||
|
dc.drawRoundedRectangle(scLeftRectMargin, 4 * h / 6 + scVertMargin, w, 2 * h / 6 - (2 * scVertMargin), scRectRadius);
|
||||||
|
dc.fillRoundedRectangle(scLeftRectMargin, 4 * h / 6 + scVertMargin, scRectWidth, 2 * h / 6 - (2 * scVertMargin), scRectRadius);
|
||||||
|
mMenuStatus.draw(dc);
|
||||||
|
} else {
|
||||||
|
// Customised Glance View
|
||||||
|
dc.setColor(Graphics.COLOR_BLUE, Graphics.COLOR_BLUE);
|
||||||
|
dc.drawRoundedRectangle(
|
||||||
|
scLeftRectMargin + scRectWidth + scRightRectMargin,
|
||||||
|
2 * h / 6 + scVertMargin,
|
||||||
|
w - scRectWidth - scRightRectMargin,
|
||||||
|
4 * h / 6 - (2 * scVertMargin),
|
||||||
|
scRectRadius
|
||||||
|
);
|
||||||
|
dc.setColor(apiCol, apiCol);
|
||||||
|
dc.fillRoundedRectangle(scLeftRectMargin, 2 * h / 6 + scVertMargin, scRectWidth, 2 * h / 6 - (2 * scVertMargin), scRectRadius);
|
||||||
|
dc.setColor(menuCol, menuCol);
|
||||||
|
dc.fillRoundedRectangle(scLeftRectMargin, 4 * h / 6 + scVertMargin, scRectWidth, 2 * h / 6 - (2 * scVertMargin), scRectRadius);
|
||||||
|
mGlanceContent.setText(glanceText);
|
||||||
|
mGlanceContent.draw(dc);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -167,7 +167,7 @@ class WebLog {
|
|||||||
// }
|
// }
|
||||||
}
|
}
|
||||||
|
|
||||||
// Callback function to print the outcome of a clear() method. Typically used for debugging this class.
|
//! Callback function to print the outcome of a clear() method. Typically used for debugging this class.
|
||||||
//!
|
//!
|
||||||
//! @param responseCode Response code.
|
//! @param responseCode Response code.
|
||||||
//! @param data Response data.
|
//! @param data Response data.
|
||||||
|