0

我是android新手。Android Widget - Read_SMS权限被拒绝API 23

我故意做的是让当前在设备的短信收件箱里面的短信NR。 我希望此编号出现在文本视图内。

我从其他职位的权限机制在API 23 改变知道,但是在所有这些文章中,我们还有一个活动。 在我的情况下,我没有一个活动,我试图获得一个AppWidgetProvider的onUpdate方法内的短数。

在我的清单文件我增加了READ_SMS的权限。

AndroidManifest.xml中

<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
    package="com.globa8track.smsgtw">  
    **<uses-permission android:name="android.permission.READ_SMS"/>** 
    <application 
     android:allowBackup="true" 
     android:icon="@mipmap/ic_launcher" 
     android:label="@string/app_name" 
     android:supportsRtl="true" 
     android:theme="@style/AppTheme"> 
     <receiver android:name=".smsGtw"> 
      <intent-filter> 
       <action android:name="android.appwidget.action.APPWIDGET_UPDATE" /> 
      </intent-filter> 

      <meta-data 
       android:name="android.appwidget.provider" 
       android:resource="@xml/sms_gtw_info" /> 
     </receiver>  
    </application> 
</manifest> 

然后在我的窗口小部件的Java类 smsGtw.java

package com.globa8track.smsgtw; 

import android.Manifest; 
import android.app.DownloadManager; 
import android.appwidget.AppWidgetManager; 
import android.appwidget.AppWidgetProvider; 
import android.content.ContentResolver; 
import android.content.Context; 
import android.content.pm.PackageManager; 
import android.database.Cursor; 
import android.net.Uri; 
import android.support.v4.app.ActivityCompat; 
import android.support.v4.content.ContextCompat; 
import android.widget.RemoteViews; 

import java.util.ArrayList; 

    public class smsGtw extends AppWidgetProvider { 

     void updateAppWidget(Context context, AppWidgetManager appWidgetManager, 
          int appWidgetId) { 

      ArrayList<String> inboxSms; 
      int nrSmsInInbox = 0; 


      inboxSms = fetchInbox(context); 
      nrSmsInInbox = inboxSms.size(); 



      CharSequence widgetText = "Gtw Total Sms Received: " + nrSmsInInbox; 

      // Construct the RemoteViews object 
      RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.sms_gtw); 
      views.setTextViewText(R.id.appwidget_text, widgetText); 

      // Instruct the widget manager to update the widget 
      appWidgetManager.updateAppWidget(appWidgetId, views); 
     } 

     @Override 
     public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) { 
      // There may be multiple widgets active, so update all of them 
      for (int appWidgetId : appWidgetIds) { 
       updateAppWidget(context, appWidgetManager, appWidgetId); 
      } 
     } 

     @Override 
     public void onEnabled(Context context) { 
      // Enter relevant functionality for when the first widget is created 
     } 

     @Override 
     public void onDisabled(Context context) { 
      // Enter relevant functionality for when the last widget is disabled 
     } 


     public static ArrayList<String> fetchInbox(Context context){ 

      ArrayList<String> sms=new ArrayList<String>(); 

      Uri uriSms=Uri.parse("content://sms/inbox"); 

      ContentResolver cr = context.getContentResolver(); 
      Cursor cursor = cr.query(uriSms, new String[]{"_id", "address", "date", "body"}, null, null, null); 
      cursor.moveToFirst(); 

      while(cursor.moveToNext()){ 

       String address=cursor.getString(1); 
       String body=cursor.getString(3); 

       sms.add("Address=>"+address+"\n Sms=>"+body); 
      } 

      return sms; 
     } 


    } 

当我运行Android Studio中模拟器此代码( Nexus 5 API 23)我得到 以下错误:

Process: com.globa8track.smsgtw, PID: 5397 java.lang.RuntimeException: Unable to start receiver com.globa8track.smsgtw.smsGtw: java.lang.SecurityException: Permission Denial: reading com.android.providers.telephony.SmsProvider uri content://sms/inbox from pid=5397, uid=10057 requires android.permission.READ_SMS, or grantUriPermission() 

从帖子:Why does Android ignore READ_SMS permission?

我理解,这需要先询问权限,然后检查该权限。正如@TomášNavara所说的。但在他的情况下,有一个活动。

我也必须有活动吗?我不希望用户交互,也不需要用户界面来检查存储在收件箱中的短信数量。

另一件事,你可以帮我在Android窗口小部件开始的世界? 我应该从哪里开始? 你能提供一些指针/资源吗? 谢谢大家的耐心等待。

+1

“我还必须有活动吗?” - 是的。 – CommonsWare

+0

@ @CommonsWare。所以我必须通过意图创建一个由小部件onUpdate()调用的活动SMSRead?我如何获得打印在小部件文本视图上的结果? – ThelmaJay

回答

1

So I have to create an activity SMSRead that is called by the widgets onUpdate() through an intent?

不是。您现有的代码很好(大概)用于获取消息和更新应用程序窗口小部件。如果它适用于较旧的设备,并且如果它在Android 6.0上工作,如果您通过设置中的应用程序屏幕手动授予权限,那么这些内容就没有问题。

,你需要一个活动是能够从用户请求READ_SMS许可。

因此,在您的应用程序小部件代码中,您需要致电ContextCompat.checkSelfPermission()以查看是否有READ_SMS访问权限。如果您有权访问,请继续并更新您的应用小部件。如果您不这样做,请提出Notification以让用户知道您需要他们授予您许可。理想情况下,Notification将与您的活动相关联,然后您可以使用ActivityCompat.requestPermissions()请求READ_SMS

你也能安排这种活动是你的应用程序窗口小部件配置活动,当您的应用程序widget被添加到用户的主屏幕,其会自动启动。这使您可以立即请求权限。但是,您仍然需要拨打AppWidgetProvider中的checkSelfPermission(),因为用户可以通过设置应用撤消权限。

+0

谢谢@CommonsWare。我添加了一个主要活动到我的小部件并添加了'checkSelfPermission'和'requestPermissions'。我现在可以在我的小部件上获得收到的短信数量。这[链接](https://guides.codepath.com/android/Understanding-App-Permissions)帮助我了解了如何申请许可。 – ThelmaJay