question-mark
Stuck on an issue?

Lightrun Answers was designed to reduce the constant googling that comes with debugging 3rd party libraries. It collects links to all the places you might be looking at while hunting down a tough bug.

And, if you’re still stuck at the end, we’re happy to hop on a call to see how we can help out.

Plyer GPS OnLocationChanged issue

See original GitHub issue

Hi everyone,

I am using Plyer to retrieve the phone GPS position and battery level.

I managed to retrieve the battery level normally.

However, I have been struggling for hours to retrieve the GPS position, and I did not find any solution online.

I get the following error:

image

I inserted some logs into the python file “plyer.platforms.android.gps” and I confirm that the “onLocationChanged” method is never called. Something I don’t understand is that the java signature for this method is "‘(Landroid/location/Location;)V’ (see below) whereas the error stack says that the java signature at the source of the issue is “Ljava/util/List” (see above).

image

Note: the GPS lattitude and longitude are well captured, indeed the logcat show the following (I masked by actual position with XX):

Location[fused XX.780620,XX.302127 hAcc=20.0 et=+7d0h5m48s123ms alt=92.30000305175781 vAcc=1.1983389 vel=0.011362127 bear=32.304817 {Bundle[EMPTY_PARCEL]}]

So I guess the issue is related to plyer / pyjnius when processing those values.

Plyer version is 2.0.0 Jnius version is 1.3.0

I tried older versions of Plyer but the same issue occurs. I wanted to try a more recent version of Jnius (1.4.1) but when I specify pyjnius==1.4.1 in the buildozer.spec that leads to an error.

Thank you for you help.

Python file:

"""This module allows to print the GPS positions"""

from kivymd.app import MDApp
from kivy.utils import platform
from kivymd.uix.dialog import MDDialog
from kivymd.uix.boxlayout import MDBoxLayout
from kivy.clock import mainthread
from kivy.properties import NumericProperty, ObjectProperty

from plyer import gps, battery as battery_plyer
import plyer.platforms.android.gps

import plyer
print("plyer version", plyer.__version__)

import jnius
print("jnius version", jnius.__version__)

class BatteryDev:
    """Battery object for simulating android battery object"""
    def __init__(self):
        self.status = {
            "isCharging": True,
            "percentage": 58
        }


class Displayer(MDBoxLayout):

    lat = NumericProperty()
    lon = NumericProperty()
    battery = ObjectProperty()

    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)

    def display_gps_position(self):

        if platform == "android":
            from android.permissions import Permission, request_permissions

            def callback(permission, results):
                print("Permission results: ", results)
                if all(res for res in results):
                    print("Got all permissions")
                else:
                    print("Did not get permissions")

            request_permissions(
                [Permission.ACCESS_COARSE_LOCATION, Permission.ACCESS_FINE_LOCATION], callback
            )

        if platform == "android" or platform == "ios":

            self.ids.battery_percentage.text = f"battery level is: {battery_plyer.status['percentage']}"

            print("Configuring GPS...")
            gps.configure(
                on_location=self._on_location,  # every time the gps receive a new location, call this function
                on_status=self._on_auth_status,  # called whenever the gps status (on / off) changes
            )

            print("Starting GPS...")
            gps.start(
                minTime=1000, minDistance=0
            )  # allow GPS to check on location or on status every X ms
            print("GPS started")
        else:
            # battery simulation
            battery = BatteryDev()
            self.ids.battery_percentage.text = f"battery level is: {battery.status['percentage']}"

            print("self.battery_percentage", self.ids.battery_percentage.text)
            # GPS simulation
            self._on_location(lat=50.3, lon=6.2)

    @mainthread
    def _on_location(self, *args, **kwargs):
        """Update the GPS blinker position from GPS actual position"""

        print("on_location args, kwargs", args, kwargs)
        print("GPS position lat / lon: ", kwargs["lat"], kwargs["lon"])

        self.lat = kwargs["lat"]
        self.lon = kwargs["lon"]
        self.ids.gps_coordinates.text = f"lat is {self.lat}, lon is {self.lon}"

    @mainthread
    def _on_auth_status(self, general_status, status_message, *args, **kwargs):

        print("general_status, status_message, *args, **kwargs", general_status, status_message, args, kwargs)

        if general_status == "provider-enabled":
            print("General status: provider-enabled")
        else:  # someone turned off GPS
            self._open_gps_access_popup()  # popup saying "Hey you disabled GPS"

    def _open_gps_access_popup(self):

        dialog = MDDialog(
            title="GPS error",
            text="You need to enable the GPS access for the app to function properly",
        )
        dialog.size_hint = [0.8, 0.8]
        dialog.pos_hint = {"center_x": 0.5, "center_y": 0.5}
        dialog.open()

class MainApp(MDApp):

    def on_start(self):
        self.root.display_gps_position()

    def on_pause(self):
        print("******Application paused******")
        gps.stop()
        return True

    def on_resume(self):
        print("******Application resumed******")
        gps.start(minTime=1000, minDistance=0)

if __name__ == "__main__":
    MainApp().run()

KV file:

Displayer:

<Displayer>:
    orientation: "vertical"
    MDLabel:
        id: battery_percentage
        text:
        pos_hint: {"x": 0.2, "y": 0.5}
    MDLabel:
        id: gps_coordinates
        text:
        pos_hint: {"x": 0.2, "y": 0.5}

My buildozer.spec file:

# (str) Source code where the main.py live
source.dir = .

# (list) Source files to include (let empty to include all the files)
source.include_exts = py,png,jpg,kv,atlas

# (list) List of inclusions using pattern matching
#source.include_patterns = assets/*,images/*.png

# (list) Source files to exclude (let empty to not exclude anything)
#source.exclude_exts = spec

# (list) List of directory to exclude (let empty to not exclude anything)
#source.exclude_dirs = tests, bin, venv

# (list) List of exclusions using pattern matching
#source.exclude_patterns = license,images/*/*.jpg

# (str) Application versioning (method 1)
version = 0.1

# (str) Application versioning (method 2)
# version.regex = __version__ = ['"](.*)['"]
# version.filename = %(source.dir)s/main.py

# (list) Application requirements
# comma separated e.g. requirements = sqlite3,kivy
requirements = python3, kivy, kivymd, pillow, plyer

# (str) Custom source folders for requirements
# Sets custom source for any requirements with recipes
# requirements.source.kivy = ../../kivy
# requirements.source.pyjnius = ~/Desktop/kivy_tutorials/gps_displayer/.buildozer/pyjnius

# (str) Presplash of the application
#presplash.filename = %(source.dir)s/data/presplash.png

# (str) Icon of the application
#icon.filename = %(source.dir)s/data/icon.png

# (str) Supported orientation (one of landscape, sensorLandscape, portrait or all)
orientation = portrait

# (list) List of service to declare
#services = NAME:ENTRYPOINT_TO_PY,NAME2:ENTRYPOINT2_TO_PY

#
# OSX Specific
#

#
# author = © Copyright Info

# change the major version of python used by the app
osx.python_version = 3

# Kivy version to use
osx.kivy_version = 1.9.1

#
# Android specific
#

# (bool) Indicate if the application should be fullscreen or not
fullscreen = 0

# (string) Presplash background color (for android toolchain)
# Supported formats are: #RRGGBB #AARRGGBB or one of the following names:
# red, blue, green, black, white, gray, cyan, magenta, yellow, lightgray,
# darkgray, grey, lightgrey, darkgrey, aqua, fuchsia, lime, maroon, navy,
# olive, purple, silver, teal.
#android.presplash_color = #FFFFFF

# (string) Presplash animation using Lottie format.
# see https://lottiefiles.com/ for examples and https://airbnb.design/lottie/
# for general documentation.
# Lottie files can be created using various tools, like Adobe After Effect or Synfig.
#android.presplash_lottie = "path/to/lottie/file.json"

# (str) Adaptive icon of the application (used if Android API level is 26+ at runtime)
#icon.adaptive_foreground.filename = %(source.dir)s/data/icon_fg.png
#icon.adaptive_background.filename = %(source.dir)s/data/icon_bg.png

# (list) Permissions
android.permissions = INTERNET, ACCESS_COARSE_LOCATION, ACCESS_FINE_LOCATION

# (list) features (adds uses-feature -tags to manifest)
#android.features = android.hardware.usb.host

# (int) Target Android API, should be as high as possible.
#android.api = 27

# (int) Minimum API your APK will support.
#android.minapi = 21

# (int) Android SDK version to use
#android.sdk = 20

# (str) Android NDK version to use
#android.ndk = 19b

# (int) Android NDK API to use. This is the minimum API your app will support, it should usually match android.minapi.
#android.ndk_api = 21

# (bool) Use --private data storage (True) or --dir public storage (False)
#android.private_storage = True

# (str) Android NDK directory (if empty, it will be automatically downloaded.)
#android.ndk_path =

# (str) Android SDK directory (if empty, it will be automatically downloaded.)
#android.sdk_path =

# (str) ANT directory (if empty, it will be automatically downloaded.)
#android.ant_path =

# (bool) If True, then skip trying to update the Android sdk
# This can be useful to avoid excess Internet downloads or save time
# when an update is due and you just want to test/build your package
# android.skip_update = False

# (bool) If True, then automatically accept SDK license
# agreements. This is intended for automation only. If set to False,
# the default, you will be shown the license when first running
# buildozer.
# android.accept_sdk_license = False

# (str) Android entry point, default is ok for Kivy-based app
#android.entrypoint = org.kivy.android.PythonActivity

# (str) Full name including package path of the Java class that implements Android Activity
# use that parameter together with android.entrypoint to set custom Java class instead of PythonActivity
#android.activity_class_name = org.kivy.android.PythonActivity

# (str) Extra xml to write directly inside the <manifest> element of AndroidManifest.xml
# use that parameter to provide a filename from where to load your custom XML code
#android.extra_manifest_xml = ./src/android/extra_manifest.xml

# (str) Extra xml to write directly inside the <manifest><application> tag of AndroidManifest.xml
# use that parameter to provide a filename from where to load your custom XML arguments:
#android.extra_manifest_application_arguments = ./src/android/extra_manifest_application_arguments.xml

# (str) Full name including package path of the Java class that implements Python Service
# use that parameter to set custom Java class instead of PythonService
#android.service_class_name = org.kivy.android.PythonService

# (str) Android app theme, default is ok for Kivy-based app
# android.apptheme = "@android:style/Theme.NoTitleBar"

# (list) Pattern to whitelist for the whole project
#android.whitelist =

# (str) Path to a custom whitelist file
#android.whitelist_src =

# (str) Path to a custom blacklist file
#android.blacklist_src =

# (list) List of Java .jar files to add to the libs so that pyjnius can access
# their classes. Don't add jars that you do not need, since extra jars can slow
# down the build process. Allows wildcards matching, for example:
# OUYA-ODK/libs/*.jar
#android.add_jars = foo.jar,bar.jar,path/to/more/*.jar

# (list) List of Java files to add to the android project (can be java or a
# directory containing the files)
#android.add_src =

# (list) Android AAR archives to add
#android.add_aars =

# (list) Gradle dependencies to add
#android.gradle_dependencies =

# (bool) Enable AndroidX support. Enable when 'android.gradle_dependencies'
# contains an 'androidx' package, or any package from Kotlin source.
# android.enable_androidx requires android.api >= 28
#android.enable_androidx = False

# (list) add java compile options
# this can for example be necessary when importing certain java libraries using the 'android.gradle_dependencies' option
# see https://developer.android.com/studio/write/java8-support for further information
# android.add_compile_options = "sourceCompatibility = 1.8", "targetCompatibility = 1.8"

# (list) Gradle repositories to add {can be necessary for some android.gradle_dependencies}
# please enclose in double quotes 
# e.g. android.gradle_repositories = "maven { url 'https://kotlin.bintray.com/ktor' }"
#android.add_gradle_repositories =

# (list) packaging options to add 
# see https://google.github.io/android-gradle-dsl/current/com.android.build.gradle.internal.dsl.PackagingOptions.html
# can be necessary to solve conflicts in gradle_dependencies
# please enclose in double quotes 
# e.g. android.add_packaging_options = "exclude 'META-INF/common.kotlin_module'", "exclude 'META-INF/*.kotlin_module'"
#android.add_packaging_options =

# (list) Java classes to add as activities to the manifest.
#android.add_activities = com.example.ExampleActivity

# (str) OUYA Console category. Should be one of GAME or APP
# If you leave this blank, OUYA support will not be enabled
#android.ouya.category = GAME

# (str) Filename of OUYA Console icon. It must be a 732x412 png image.
#android.ouya.icon.filename = %(source.dir)s/data/ouya_icon.png

# (str) XML file to include as an intent filters in <activity> tag
#android.manifest.intent_filters =

# (str) launchMode to set for the main activity
#android.manifest.launch_mode = standard

# (list) Android additional libraries to copy into libs/armeabi
#android.add_libs_armeabi = libs/android/*.so
#android.add_libs_armeabi_v7a = libs/android-v7/*.so
#android.add_libs_arm64_v8a = libs/android-v8/*.so
#android.add_libs_x86 = libs/android-x86/*.so
#android.add_libs_mips = libs/android-mips/*.so

# (bool) Indicate whether the screen should stay on
# Don't forget to add the WAKE_LOCK permission if you set this to True
#android.wakelock = False

# (list) Android application meta-data to set (key=value format)
#android.meta_data =

# (list) Android library project to add (will be added in the
# project.properties automatically.)
#android.library_references =

# (list) Android shared libraries which will be added to AndroidManifest.xml using <uses-library> tag
#android.uses_library =

# (str) Android logcat filters to use
#android.logcat_filters = *:S python:D

# (bool) Android logcat only display log for activity's pid
#android.logcat_pid_only = False

# (str) Android additional adb arguments
#android.adb_args = -H host.docker.internal

# (bool) Copy library instead of making a libpymodules.so
#android.copy_libs = 1

# (str) The Android arch to build for, choices: armeabi-v7a, arm64-v8a, x86, x86_64
android.arch = armeabi-v7a

# (int) overrides automatic versionCode computation (used in build.gradle)
# this is not the same as app version and should only be edited if you know what you're doing
# android.numeric_version = 1

# (bool) enables Android auto backup feature (Android API >=23)
android.allow_backup = True

# (str) XML file for custom backup rules (see official auto backup documentation)
# android.backup_rules =

# (str) If you need to insert variables into your AndroidManifest.xml file,
# you can do so with the manifestPlaceholders property.
# This property takes a map of key-value pairs. (via a string)
# Usage example : android.manifest_placeholders = [myCustomUrl:\"org.kivy.customurl\"]
# android.manifest_placeholders = [:]

# (bool) disables the compilation of py to pyc/pyo files when packaging
# android.no-compile-pyo = True

#
# Python for android (p4a) specific
#

# (str) python-for-android URL to use for checkout
#p4a.url =

# (str) python-for-android fork to use in case if p4a.url is not specified, defaults to upstream (kivy)
#p4a.fork = kivy

# (str) python-for-android branch to use, defaults to master
#p4a.branch = master

# (str) python-for-android specific commit to use, defaults to HEAD, must be within p4a.branch
#p4a.commit = HEAD

# (str) python-for-android git clone directory (if empty, it will be automatically cloned from github)
#p4a.source_dir =

# (str) The directory in which python-for-android should look for your own build recipes (if any)
#p4a.local_recipes =

# (str) Filename to the hook for p4a
#p4a.hook =

# (str) Bootstrap to use for android builds
# p4a.bootstrap = sdl2

# (int) port number to specify an explicit --port= p4a argument (eg for bootstrap flask)
#p4a.port =

# Control passing the --use-setup-py vs --ignore-setup-py to p4a
# "in the future" --use-setup-py is going to be the default behaviour in p4a, right now it is not
# Setting this to false will pass --ignore-setup-py, true will pass --use-setup-py
# NOTE: this is general setuptools integration, having pyproject.toml is enough, no need to generate
# setup.py if you're using Poetry, but you need to add "toml" to source.include_exts.
#p4a.setup_py = false


#
# iOS specific
#

# (str) Path to a custom kivy-ios folder
#ios.kivy_ios_dir = ../kivy-ios
# Alternately, specify the URL and branch of a git checkout:
ios.kivy_ios_url = https://github.com/kivy/kivy-ios
ios.kivy_ios_branch = master

# Another platform dependency: ios-deploy
# Uncomment to use a custom checkout
#ios.ios_deploy_dir = ../ios_deploy
# Or specify URL and branch
ios.ios_deploy_url = https://github.com/phonegap/ios-deploy
ios.ios_deploy_branch = 1.10.0

# (bool) Whether or not to sign the code
ios.codesign.allowed = false

# (str) Name of the certificate to use for signing the debug version
# Get a list of available identities: buildozer ios list_identities
#ios.codesign.debug = "iPhone Developer: <lastname> <firstname> (<hexstring>)"

# (str) The development team to use for signing the debug version
#ios.codesign.development_team.debug = <hexstring>

# (str) Name of the certificate to use for signing the release version
#ios.codesign.release = %(ios.codesign.debug)s

# (str) The development team to use for signing the release version
#ios.codesign.development_team.release = <hexstring>

# (str) URL pointing to .ipa file to be installed
# This option should be defined along with `display_image_url` and `full_size_image_url` options.
#ios.manifest.app_url =

# (str) URL pointing to an icon (57x57px) to be displayed during download
# This option should be defined along with `app_url` and `full_size_image_url` options.
#ios.manifest.display_image_url =

# (str) URL pointing to a large icon (512x512px) to be used by iTunes
# This option should be defined along with `app_url` and `display_image_url` options.
#ios.manifest.full_size_image_url =


[buildozer]

# (int) Log level (0 = error only, 1 = info, 2 = debug (with command output))
log_level = 2

# (int) Display warning if buildozer is run as root (0 = False, 1 = True)
warn_on_root = 1

# (str) Path to build artifact storage, absolute or relative to spec file
# build_dir = ./.buildozer

# (str) Path to build output (i.e. .apk, .ipa) storage
# bin_dir = ./bin

Issue Analytics

  • State:open
  • Created 2 years ago
  • Reactions:2
  • Comments:7

github_iconTop GitHub Comments

2reactions
robmaster2016commented, May 25, 2022

The GPS no longer works for me since Android 12.

1reaction
eb-sailcommented, Feb 9, 2022

I’m having the same problem. Seems to be an issue with Android 12. Everything was working until my phone got the update.

Read more comments on GitHub >

github_iconTop Results From Across the Web

Plyer GPS doesn't work on Android 12 - Google Groups
Hi, I developed a navigation app with kivy using plyer.gps. Everything worked fine until my phone updated to Android 12. The app doesn't...
Read more >
Issues with Plyers GPS module on Android 12 device
I've tried modifying gps.py with changes from a pull-request, tried requesting both permissions in a single runtime request (not sure if I did ......
Read more >
plyer - PyPI
Plyer is a platform-independent api to use features commonly found on various platforms, ... GPS Issue after in iOS after last changes in...
Read more >
[Android] Location app crashes due to error 'jstring has wrong ...
The issue is occurring for me on 5.5.0f3 and is blocking an app update I need ... public void onLocationChanged(AndroidJavaObject location)
Read more >
Location services on Android - Xamarin | Microsoft Learn
It uses less power than the GPS Provider, but returns location data ... user can resolve the issue var errorString = GoogleApiAvailability.
Read more >

github_iconTop Related Medium Post

No results found

github_iconTop Related StackOverflow Question

No results found

github_iconTroubleshoot Live Code

Lightrun enables developers to add logs, metrics and snapshots to live code - no restarts or redeploys required.
Start Free

github_iconTop Related Reddit Thread

No results found

github_iconTop Related Hackernoon Post

No results found

github_iconTop Related Tweet

No results found

github_iconTop Related Dev.to Post

No results found

github_iconTop Related Hashnode Post

No results found