Android处理来自不同来源的同步权限

时间:2017-05-02 17:42:53

标签: android android-permissions

我正在尝试同时请求多个权限,来自多个来源。我的问题是只需要一个权限。

以下是一个例子:

我们有2个片段在MainActivity上同时加载,例如

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.example.myapplication.MainActivity"
    android:orientation="vertical">

    <fragment android:name="com.example.myapplication.fragments.FragmentOne"
        android:id="@+id/fragment_one"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

    <fragment android:name="com.example.myapplication.fragments.FragmentTwo"
        android:id="@+id/fragment_two"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

</LinearLayout>

我们有FragmentOne请求READ_CALL_LOG,例如

@Override
public void onStart() {
    super.onStart();
    Log.d(TAG, "onStart: ");
    requestPermissions(new String[]{Manifest.permission.READ_CALL_LOG}, 1);
}

我们还有FragmentTwo请求READ_CONTACTS,例如

@Override
public void onStart() {
    super.onStart();
    Log.d(TAG, "onStart: ");
    requestPermissions(new String[] { Manifest.permission.READ_CONTACTS }, 2);
}

这意味着一旦应用程序启动,就会创建2个片段,然后启动。当发生这种情况时,两个片段同时请求许可。我的问题是,只有第一个权限才会提示给用户。

现在要弄清楚我的问题,我在每个片段的onRequestPermissionsResult中添加了一些日志,例如

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    Log.d(TAG, "onRequestPermissionsResult: requestCode=" + requestCode + " permissions=" + permissions
            + " permissions.length=" + permissions.length + " grantResults.length=" + grantResults.length);
    for (String permission : permissions) {
        Log.d(TAG, "onRequestPermissionsResult: permission=" + permission);
    }
}

之后我发现了我的问题,但我没有解决方案。 当我查看日志时,一旦启动应用程序,我就会收到这些日志

2017-05-02 13:22:40:0356    + 677   05-02 13:22:38.650 27003 27003 D FragmentOne: onStart:
2017-05-02 13:22:40:0357    + 699   05-02 13:22:38.661 27003 27003 D FragmentTwo: onStart:
2017-05-02 13:22:40:0357    + 702   05-02 13:22:38.661 27003 27003 D FragmentTwo: onRequestPermissionsResult: requestCode=2 permissions=[Ljava.lang.String;@afbcb45 permissions.length=0 grantResults.length=0

同时,显示请求READ_CALL_LOG的对话框。 这里的问题似乎是它太早进入onRequestPermissionsResult。

一旦我接受对话框的READ_CALL_LOG权限,我就会得到2个新日志

2017-05-02 13:27:42:0017    + 2803  05-02 13:27:40.358 27003 27003 D FragmentOne: onRequestPermissionsResult: requestCode=1 permissions=[Ljava.lang.String;@5028ba8 permissions.length=1 grantResults.length=1
2017-05-02 13:27:42:0017    + 2804  05-02 13:27:40.358 27003 27003 D FragmentOne: onRequestPermissionsResult: permission=android.permission.READ_CALL_LOG

此时,不显示我期待的第二个权限对话框(对于READ_CONTACTS)。不会生成更多日志。

现在,我知道我可以从不同的片段获得所有权限,并通过传递相关的String数组同时提示它们。但这不是我想在这里完成的。我正在尝试让多个组件处理自己的权限。

我错过了什么吗?我做错了吗?

作为参考,我将在此处列出MainActivity和FragmentOne的完整代码。 FragmentTwo几乎相同(仅TAG和权限更改)。

MainActivity:

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
}

FragmentOne:

public class FragmentOne extends Fragment {

    private static final String TAG = "FragmentOne";

    public FragmentOne() {
        // Required empty public constructor
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        // Inflate the layout for this fragment
        return inflater.inflate(R.layout.fragment_fragment_one, container, false);
    }

    @Override
    public void onStart() {
        super.onStart();
        Log.d(TAG, "onStart: ");
        requestPermissions(new String[]{Manifest.permission.READ_CALL_LOG}, 1);
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        Log.d(TAG, "onRequestPermissionsResult: requestCode=" + requestCode + " permissions=" + permissions
                + " permissions.length=" + permissions.length + " grantResults.length=" + grantResults.length);
        for (String permission : permissions) {
            Log.d(TAG, "onRequestPermissionsResult: permission=" + permission);
        }
    }
}

1 个答案:

答案 0 :(得分:0)

尝试将权限请求放在活动中而不是片段中,并同时请求两者。 似乎问题在于,当您请求第二个权限时,第一个对话框将被关闭,因此被视为被拒绝。