我是新手。
我打算做的是获取当前在设备短信收件箱内的短信。 我希望这个数字出现在文本视图中。
我从其他帖子中了解到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>
然后在我的Widget 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小部件世界吗? 我应该从哪里开始? 你能给我一些指示/资源吗? 谢谢大家的耐心等待。
答案 0 :(得分:1)
所以我必须通过一个意图创建一个由小部件onUpdate()调用的活动SMSRead?
没有。您现有的代码很好(可能)用于获取消息和更新应用程序小部件。如果它适用于较旧的设备,并且如果您通过设置中的应用程序屏幕手动授予权限,则它适用于Android 6.0。那么这些东西就可以了。
您需要活动的地方是能够向用户请求READ_SMS
权限。
因此,在您的应用小部件代码中,您需要致电ContextCompat.checkSelfPermission()
以查看您是否拥有READ_SMS
访问权限。如果您确实有权限,请继续并更新您的应用小部件。如果不这样做,请提出Notification
,让用户知道您需要他们授予您权限。理想情况下,Notification
与您的活动相关联,然后您可以使用ActivityCompat.requestPermissions()
来请求READ_SMS
。
您还可以安排将该活动作为您的应用小部件配置活动,该活动会在您的应用小部件添加到用户的主屏幕时自动启动。这允许您立即请求许可。但是,您仍然需要在checkSelfPermission()
中致电AppWidgetProvider
,因为用户可以通过“设置”应用撤消权限。