java - 当位置设置请求被拒绝时,设置 View 返回空指针?

标签 java android android-fragments google-location-services

我遇到这个问题,当用户拒绝在弹出窗口中打开他们的 Location 设置时,我的 Fragment 中的 Views 返回 null向上。我似乎无法引用 Fragment 中的任何 Views。我想这是因为 Location 请求 View 的弹出对话框取代了我的 Fragment View 但我不知道如何访问我的 Fragment 中的 Views。这是我的尝试:

private View fragmentView;
private View progressOverlay;
private View noPostsView;

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    if (fragmentView == null) {
        fragmentView = inflater.inflate(R.layout.posts_tab, container, false);
        setupActivity(savedInstanceState);
        progressOverlay = fragmentView.findViewById(R.id.progress_overlay);
        progressOverlay.setVisibility(View.VISIBLE);
        noPostsView = fragmentView.findViewById(R.id.no_posts_layout);
        setupLocation();
    }
    return fragmentView;
}

private void setupLocation() {
    Context context = getContext();
    if(locationService == null) {
        locationService = new LocationService(context, new LocationUpdateListener() {
            @Override
            public void canReceiveLocationUpdates() {

            }

            @Override
            public void cannotReceiveLocationUpdates() {
                createSettingsRequest();
                //well we know we cant receive updates so we have to create a settings request
            }

            //update location to our servers for tracking purpose
            @Override
            public void updateLocation(Location location) {
                if (location != null) {
                    //Populate Recycler View
                    databaseQuery.getPublicPosts(location, progressOverlay, fragmentView, getContext());
                }
            }

            @Override
            public void updateLocationName(String localityName, Location location) {
                locationService.stopLocationUpdates();
            }
        });
        locationService.startUpdates();
    }
}

private void createSettingsRequest() {
    LocationRequest locationRequest = LocationRequest.create();
    locationRequest.setPriority(LocationRequest.PRIORITY_HIGH_ACCURACY);
    locationRequest.setInterval(30 * 1000);
    locationRequest.setFastestInterval(5 * 1000);
    LocationSettingsRequest.Builder builder = new LocationSettingsRequest.Builder()
            .addLocationRequest(locationRequest);
    builder.setAlwaysShow(true); //this is the key ingredient

    PendingResult<LocationSettingsResult> result =
            LocationServices.SettingsApi.checkLocationSettings(locationService.mGoogleApiClient
                    , builder.build());
    result.setResultCallback(new ResultCallback<LocationSettingsResult>() {
        @Override
        public void onResult(LocationSettingsResult result) {
            final Status status = result.getStatus();
            final LocationSettingsStates state = result.getLocationSettingsStates();
            switch (status.getStatusCode()) {
                case LocationSettingsStatusCodes.SUCCESS:
                    // All location settings are satisfied. The client can initialize location
                    // requests here.
                    break;
                case LocationSettingsStatusCodes.RESOLUTION_REQUIRED:
                    // Location settings are not satisfied. But could be fixed by showing the user
                    // a dialog.
                    try {
                        // Show the dialog by calling startResolutionForResult(),
                        // and check the result in onActivityResult().
                        status.startResolutionForResult(getActivity(), REQUEST_LOCATION);
                    } catch (IntentSender.SendIntentException e) {
                        FirebaseCrash.report(e);
                    }
                    break;
                case LocationSettingsStatusCodes.SETTINGS_CHANGE_UNAVAILABLE:
                    // Location settings are not satisfied. However, we have no way to fix the
                    // settings so we won't show the dialog.
                    break;
            }
        }
    });
}

    @Override
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
        switch (requestCode) {
            // Check for the integer request code originally supplied to startResolutionForResult().
            case REQUEST_LOCATION:
                switch (resultCode) {
                    case Activity.RESULT_OK:
                        break;
                    case Activity.RESULT_CANCELED:
                        setupNoPostsView();
                        break;
                }
                break;
        }
    }

private void setupNoPostsView() {
    progressOverlay.setVisibility(View.GONE);
    noPostsView.setVisibility(View.VISIBLE);
    RecyclerView recyclerView = (RecyclerView) fragmentView.findViewById(R.id.rv_posts_feed);
    recyclerView.setVisibility(View.GONE);
    TextView noPosts = (TextView) noPostsView.findViewById(R.id.no_posts_text);
    noPosts.setText(R.string.permission_location_rationale);
}

我只是想调用 setupNoPostsView 如果用户点击 No 允许 Location Services 但无论出于何种原因,progressOverlay View 在我尝试设置 Visibility = View.GONE 时一直返回 null。关于如何解决这个问题的任何想法?

根据第一个答案的要求,在这里更新我的布局:

fragment .xml:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:id="@+id/posts_tab"
    tools:showIn="@layout/app_bar_news_feed" tools:context=".news_feed">

    <include layout="@layout/include_progress_overlay"/>

    <include layout="@layout/no_posts"/>

    <android.support.v7.widget.RecyclerView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/rv_posts_feed"
        android:visibility="gone"
        />

</FrameLayout>

no_posts.xml:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" android:layout_width="match_parent"
    android:id="@+id/no_posts_layout"
    android:visibility="gone"
    android:layout_height="match_parent">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:padding="16dp"
        android:textSize="16sp"
        android:layout_gravity="center"
        android:gravity="center"
        android:textAlignment="center"
        android:id="@+id/no_posts_text" />

</FrameLayout>

最佳答案

所以,方法

status.startResolutionForResult(getActivity(), REQUEST_LOCATION);

正在调用由 google play 服务提供的新 Activity ,该 Activity 显示用于打开位置设置的对话框。让我们将 A 称为您的 Activity ,将 B 称为 Google Play 服务提供的 Activity 。可能,当创建 B 时,由于内存压力或因为您更改了设备的方向,A 被销毁。当您在 B 上选择“拒绝”时,在 Android 重新创建 View 之前调用回调“onActivityResult”(这实际上是可能的)。所以......你必须以不同的方式管理这种情况,例如在你的 onAcvityResult 上设置一种标志。示例:

private Integer myResult = null;

@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
    if (REQUEST_LOCATION == requestCode) {
        View root = getView();
        if (root != null) { // Consider also isResumed()
            // you can operate on your views
            applyLogicForResult(resultCode);
        } else {
            myResult = resultCode;
        }
    }
}

@Override
public void onViewCreated(View view, @Nullable Bundle savedInstanceState) { // Consider also onResume()
    if (myResult != null) {
        applyLogicForResult(myResult);
        myResult = null;
    }
}

关于java - 当位置设置请求被拒绝时,设置 View 返回空指针?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39890449/

相关文章:

android - ansible android更新sdk给找不到sdkmanager.jar

android - 任务 ':app:compileDebugJavaWithJavac' 执行失败。错误 :(2055, 52) 错误 : ';' expected Error:(2055, 59) 错误:<标识符> 预期

android - 如何让本地服务在选项卡之间运行?

java - 如何使用java将日期和时间发送到SQL

java - 如何从我的 LinkedList 中获取数组 []

java - 无法解析为变量

java - 调用 getResources().openRawResource() 时出错

android - dropbox api - 我应该选择哪个保管箱图标?

android - onOptionsItemSelected 没有被调用

java - DefaultHttpClient GET 和 POST 命令 Java Android