-
-
Notifications
You must be signed in to change notification settings - Fork 685
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat(winforms): location support #3025
base: main
Are you sure you want to change the base?
Conversation
I think it would probably be worth waiting for #2999 to be merged before looking to merge this |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Noted about waiting until #2999 lands; but this looks pretty solid.
My only big question stems from the what the watcher API is being used. AIUI, this effectively turns on the watcher as soon as the user asks for permission, and essentially never turns it off. I can see how that works, but it slightly concerns me from a battery usage perspective (Ok, that's not a huge concern for desktop, but it's not nothing).
The solution/workaround depends a little on whether Start()
and Stop()
are blocking methods; but it seems like it should be possible to use onPermissionStatusChanged
, checks of the actual permission state, and possibly deferring call to set_result()
to inside the callback so that a Stop()
is called for every "transient" check, unless continuous tracking is turned on (in which case, Stop()
is called in stop_tracking()
. Essentially, there would always be handlers installed; but it would be a question of application state whether the Toga is notified of a position update.
Does that makes sense?
@@ -90,6 +90,8 @@ async def test_grant_background_permission(app, location_probe): | |||
|
|||
async def test_deny_background_permission(app, location_probe): | |||
"""A user can deny background permission to use location.""" | |||
skip_on_platforms("windows") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The approach we take for this kind of skip is to make the probe do the skip/xfail when an attempt is made to use the feature. In this case, we'd make request_background_location()
raise an XFAIL - we use skip to indicate a feature hasn't been implemented yet, and XFAIL to indicate "this won't ever be implemented because the platform fundamentally doesn't support this idea".
|
||
def request_permission(self, future: AsyncResult[bool]) -> None: | ||
self.watcher.Start(False) # TODO: where can we call stop? | ||
future.set_result(self.has_permission()) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is Start()
a blocking method? A method that never returns False? My concern here is whether has_permission()
is given a chance to return False on the basis of user input.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I believe this overload of Start is blocking yes, but the docs aren't super clear, and it doesn't seem possible to clear the flag on the python exe to test locally properly
That does somewhat make sense yes, aside from the fact that I think the issue with that would be a sacrifice of location quality, as it takes a while for the watcher to pin down the location, in my case, picking the Perth CBD (i assume via IP geocoding), before (I assume) using WIFI for a more accurate location. If we're okay with the current location result being very coarse unless the watcher is turned on fully, then happy to pursue that path |
I guess another alternative to improve the location quality, would be to await a decent quality location being returned (by tweaking the config on the watcher itself maybe) |
I can see how that could be a possible impediment to progress... 😅 So... why is it so lovingly documented if it doesn't exist? Is this a gap in what Python.net can expose? A .NET version thing?
There's no problem with the API taking time to return a response - that's why it's an async API. Would that mean wrapping any "transient" call with a Start()/Stop() pair is a viable solution? |
Well, it isn't documented? I think you misread "position" as "permission"?
I believe so yes? I'll look into it and test it out 😁 |
Actually, looks like we can use https://learn.microsoft.com/en-us/dotnet/api/system.device.location.geocoordinatewatcher.onpropertychanged?view=netframework-4.8.1#system-device-location-geocoordinatewatcher-onpropertychanged(system-string) to monitor the permission states |
I definitely got the method name wrong in the link I provided; but the link itself works, and the GeoPositionStatus enum looks like it refers to permissions - or at the very least, that it might be able to provide feedback for the "disabled" case. |
This PR adds support for location access on Windows via System.Device.Location.
There are still some rough edges here, as the only way to request location access is to try and access it, which isn't amazing.
Additionally there's no distinction between foreground and background access.
The code coverage isn't there all the way yet either - there are two missing lines, which I'm looking at fixing, but figured this was functional enough to get some reviews.
Also I've added some code to upload a raw coverage report, as the testbeds don't seem to have any mechanism in place to do that.
PR Checklist: