2

私は長い間この質問に対する答えを探していましたが、誰も答えを持っていないようでした.

Google の最新の FusedLocationProviderAPI を使用して、位置情報の更新を購読しようとしています。私はオンラインでチュートリアルに従いましたが、requestLocationUpdatesによってトリガーされるintentserviceを使用して、時々自分の場所を取得することで機能します。

次に、オブジェクト「ユーザー」のカスタム クラスをインテント サービスに追加してみます。このカスタム クラスを使用すると、ロケーションをサーバーにアップロードして、バックグラウンドでロケーションを更新できます。

カスタム オブジェクトを追加する方法は、インテントにバンドルを追加することです。これは、IntentService をトリガーする pendingIntent に追加されます。残念ながら、インテントサービスはすぐに壊れてしまい、バンドルが追加されるとすぐに null を返します。

IntentService をトリガーする MainActivity のコードは次のとおりです。

@Override
public void onConnected(Bundle arg0) {

    // This is for creating the intent that is used for handler class
    Intent intent = new Intent(MainActivity.this, MyLocationHandler.class);
    Bundle b = new Bundle();
    b.putParcelable("user", user);
    //intent.putExtras(b);
    intent.putExtra("bundle",b);

    PendingIntent pendingIntent = PendingIntent.getService(getApplicationContext(), 0,
            intent, PendingIntent.FLAG_UPDATE_CURRENT);

    // Then the off screen periodic update
    PendingResult pendingResult = LocationServices.FusedLocationApi.requestLocationUpdates(mGoogleApiClient,
          mLocationRequest, pendingIntent);

    LocationServices.FusedLocationApi.requestLocationUpdates(mGoogleApiClient, mLocationRequest, this);

}

上記からわかるように、バンドル 'b' をインテントに追加します。これは pendingIntent に追加され、LocationHandler クラスをトリガーします。

public class MyLocationHandler extends IntentService {

User user;
private String TAG = this.getClass().getSimpleName();

public MyLocationHandler() {
    super("MyLocationHandler");
}


@Override
protected void onHandleIntent(Intent intent) {

     Bundle bundle = intent.getBundleExtra("bundle");
    //Bundle bundle = intent.getExtras();
    if (bundle == null) {
        Log.e(TAG, "bundle is null");
    } else {
        bundle.setClassLoader(User.class.getClassLoader());
    }
    user = (User) bundle.getParcelable("user");

    if (user == null) {
        Log.e(TAG, "user is null");
    }

    if (LocationResult.hasResult(intent)) {
        LocationResult locationResult = LocationResult.extractResult(intent);
        Location location = locationResult.getLastLocation();
        Log.i(TAG, Double.toString(location.getLatitude()) + ", " + Double.toString(location.getLongitude()));

        if (user != null) {
            user.updateLocation(location); // This is a method in the custom class that sends location to the server
        }
    } else
        Log.e(TAG, "Null object returned for location API");
}

バンドルを追加しようとすると、場所は LocationHandler クラスで null を返しますが、バンドルを削除すると、場所の更新が再開されます。

念のためマニフェストを添付しました:

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="com.google.android.providers.gsf.permission.READ_GSERVICES"/>
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="com.google.android.c2dm.permission.RECEIVE" />
<permission android:name="com.example.projecttesting.permission.C2D_MESSAGE"
    android:protectionLevel="signature" />
<uses-permission android:name="com.example.projecttesting.permission.C2D_MESSAGE" />

<uses-feature
    android:glEsVersion="0x00020000"
    android:required="true"/>

<application
    android:allowBackup="true"
    tools:replace="icon, label"
    android:icon="@drawable/ic_launcher"
    android:label="SamePage"
    android:theme="@style/Theme.AppCompat.Light.NoActionBar" >

    <meta-data android:name="com.facebook.sdk.ApplicationId" android:value="@string/facebook_app_id"/>
    <meta-data
        android:name="com.google.android.gms.version"
        android:value="@integer/google_play_services_version" />
    <meta-data
        android:name="com.google.android.geo.API_KEY"
        android:value="AIzaSyCDY8ulp1VGKwGdaRU19G4sfuXsymZGgoY" />


    <activity
        android:name=".LoginActivity" >
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />
            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>

    <activity
        android:name=".MainActivity"
        android:label="@string/app_name" >
    </activity>
    <activity
        android:name=".EventLocationInput"
        android:windowSoftInputMode="adjustPan" />
    <activity android:name=".EventPeopleInput" />
    <activity android:name=".EventCreation"/>
    <service android:enabled="true" android:name=".MyLocationHandler"/>

    <activity android:name="com.facebook.FacebookActivity"
        android:configChanges=
            "keyboard|keyboardHidden|screenLayout|screenSize|orientation"
        android:theme="@android:style/Theme.Translucent.NoTitleBar"
        android:label="@string/app_name" />
    <receiver
        android:name=".GcmBroadcastReceiver"
        android:permission="com.google.android.c2dm.permission.SEND" >
        <intent-filter>
            <action android:name="com.google.android.c2dm.intent.RECEIVE" />
            <category android:name="com.example.projecttesting" />
        </intent-filter>
    </receiver>
    <service android:name=".GcmMessageHandler" />
    <meta-data android:name="com.google.android.gms.version"
        android:value="@integer/google_play_services_version" />
</application>

</manifest>

Android FusedLocationProviderApi : 受信インテントには LocationResult または LocationAvailability がありません。

これは私を夢中にさせています。誰かがこれについて少し光を当てることができることを願っています。

ありがとう。

4

1 に答える 1

0

提案されたように、ここに私の答えを投稿します。私が抱えていた問題を正確に解決しているわけではありませんが、SharedPreferenceManager を使用して必要なものを別のアクティビティに保存し、それを Handler クラスで取得することで、なんとか機能させることができました。最もエレガントなソリューションではありませんが、それでも有効です。必要なもののサイズが小さく、頻繁に変更されない場合、これは回避策になる可能性があります。

于 2015-12-20T15:31:52.867 に答える