PendingIntent和RemoteViews

   本集内容:用于为AppWidget的控件绑定处理器,即点击Home上的Widget上的控件时发生相应的反应

PendingIntent----锦囊妙计

相关知识

PendingIntent继承自Object。因为该类为final,所以没有子类,无法被继承。

要想得到一个PendindIntent对象,需要使用方法类的静态方法 getActivity(Context, int, Intent, int)getActivities(Context, int, Intent[], int)getBroadcast(Context, int, Intent, int)getService(Context, int, Intent, int)

这几个静态方法分别对应着Intent的行为,跳转到Activity,Activities,Broadcast,Service。

这4个静态方法的参数都一样,其中第一个和第三个参数比较的重要,其次是第二个和第四个。第一个参数传入当前的context,第三个参数传入intent.

PendingIntent是一个特殊的Intent,主要区别是intent是立马执行,PendingIntent是待确定的Intent。PendingIntent的操作实际上是传入的intent的操作。

使用pendingIntent的目的主要是用于所包含的intent执行是否满足某些条件。

主要的地方:Notification,SmsManager,AlarmManager等

(扩展---pendingintent与intent的区别

intent 英文意思是意图,pending 表示即将发生或来临的事情。 
PendingIntent 这个类用于处理即将发生的事情,比如在通知Notification中用于跳转页面,但不是马上跳转。 

Intent 是及时启动,intent 随所在的activity 消失而消失。 

PendingIntent 可以看作是对intent的包装,通常通过getActivity,getBroadcast,getService来得到pendingintent的实例,当前activity并不能马上启动它所包含的intent,而是在外部执行 pendingintent 时,调用intent的。正由于pendingintent中保存有当前App的Context,使它赋予外部App一种能力,使得外部App可以如同当前App一样的执行pendingintent里的 Intent, 就算在执行时当前App已经不存在了,也能通过存在pendingintent里的Context照样执行Intent。另外还可以处理intent执行后的操作。PendingIntent常和alermanger 和notificationmanager一起使用。 

Intent一般是用作Activity、Service、BroadcastReceiver之间传递数据;而Pendingintent一般用在 Notification上,可以理解为延迟执行的intent,PendingIntent是对Intent一个包装。

PendingIntent pIntent = PendingIntent.getActivity(this, 0, new Intent(this, NotifyActivity.class), 0);



1、新建res/xml/app_widget_info.xml     描述一个App Widget元数据,比如App Widget的布局,更新频率,指定在home中显示样式的布局文件。这个应该在XML里定义。

<?xml version="1.0" encoding="UTF-8"?>
<appwidget-provider  xmlns:android="http://schemas.android.com/apk/res/android"
    android:minWidth="50dp"
    android:minHeight="50dp"
    android:updatePeriodMillis="86400000"
    android:initialLayout="@layout/appwidget_layout"
    android:resizeMode="horizontal|vertical"
    android:widgetCategory="home_screen|keyguard"
    android:previewImage="@drawable/girl1"
    
   />


2、创建res/layout/appwidget_layout.xml,指定在Home上显示的样式

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >
    
    <TextView 
        android:id="@+id/textId"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Mygirl"
        android:textSize="20sp"
        />
    
      <Button 
        android:id="@+id/buttonId"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_toRightOf="@id/textId"
        android:text="测试"
        />
    <ImageView
        android:id="@+id/iv_show"
        android:layout_width="wrap_content"  
        android:layout_height="wrap_content" 
        android:layout_below="@id/textId"
        android:layout_alignLeft="@id/textId"
        android:src="@drawable/girl2"
        android:layout_gravity="center"
      /> 
</RelativeLayout>

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >
    
<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="成功了"
     />

<Button 
    android:id="@+id/button2Id"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="跳转"
    />

</LinearLayout>


3、在AndroidManifest.xml中声明

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.s02_e06_pendingintent_remoteviews"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="17" />
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="com.example.s02_e06_pendingintent_remoteviews.MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <receiver android:name="ExampleAppWidgetProvider">
            <intent-filter >
                <action android:name="android.appwidget.action.APPWIDGET_UPDATE"/>
            </intent-filter>
            <meta-data android:name="android.appwidget.provider"
                   android:resource="@xml/app_widget_info"/>
        </receiver>
    </application>

</manifest>


4、实现AppWidgetProvider的类,在onUpdate方法中写代码,实现跳转到TargetActivity的功能

package com.example.s02_e06_pendingintent_remoteviews;
import android.app.PendingIntent;
import android.appwidget.AppWidgetManager;
import android.appwidget.AppWidgetProvider;
import android.content.Context;
import android.content.Intent;
import android.widget.RemoteViews;
public class ExampleAppWidgetProvider extends AppWidgetProvider {
   @Override
public void onUpdate(Context context, AppWidgetManager appWidgetManager,
		int[] appWidgetIds) {
   for(int i=0;i<appWidgetIds.length;i++){
	   System.out.println(appWidgetIds[i]);
	   System.out.println("点击反应");
	   //创建一个Intent对象
	   Intent intent=new Intent(context, TargetActivity.class);
	   //创建一个PendingIntent
	   PendingIntent pendingIntent=PendingIntent.getActivity(context, 0, intent, 0);
	   //
	   RemoteViews remoteViews=new RemoteViews(context.getPackageName(), R.layout.appwidget_layout);
	   
	   //为按钮绑定事件处理器
	   //参数1、用来指定被绑定处理器的控件的ID;2用于指定当事件发生时哪个pendingIntent将会被执行
	   remoteViews.setOnClickPendingIntent(R.id.buttonId, pendingIntent);
	   //更新AppWidget
	   //参数1、用于指定被更新的AppWidget的ID;2、要更新的remoteView对象
	   appWidgetManager.updateAppWidget(appWidgetIds[i],remoteViews);
   }
	super.onUpdate(context, appWidgetManager, appWidgetIds);
}
   
   @Override
	public void onDeleted(Context context, int[] appWidgetIds) {
		// TODO Auto-generated method stub
		super.onDeleted(context, appWidgetIds);
	}
   @Override
	public void onDisabled(Context context) {
		// TODO Auto-generated method stub
		super.onDisabled(context);
	}
   @Override
	public void onEnabled(Context context) {
		// TODO Auto-generated method stub
		super.onEnabled(context);
	}
   
   @Override
	public void onReceive(Context context, Intent intent) {
		// TODO Auto-generated method stub
		super.onReceive(context, intent);
	}
}


5、写个TargetActivity.class

package com.example.s02_e06_pendingintent_remoteviews;

import android.app.Activity;
import android.os.Bundle;


public class TargetActivity extends Activity {
     
   @Override
protected void onCreate(Bundle savedInstanceState) {
	// TODO Auto-generated method stub
	super.onCreate(savedInstanceState);
	setContentView(R.layout.target); 
}
}

R.layout.target

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >
    
<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="成功了"
     />
<Button 
    android:id="@+id/button2Id"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="跳转"
    />

</LinearLayout>


遗留问题:悲催的是,无法成功跳转到TargetActivity

测试

a、将步骤4中代码  Intent intent=new Intent(context, TargetActivity.class); TargetActivity改成MainActivity,点击按钮可以跳转到MainActivity,说明步骤4中的代码应该没有问题

b、怀疑是我的TargetActivity.java写的有问题,于是把MainActivity中的代码复制过去,----可是依然无法跳转

c、怀疑是我的target.xml布局写的有问题,于是做测试a的改动,还把MainActivity中的默认布局文件换成Target.xml,可以成功跳转并打开Target.xml


问题已解决!!!

Activity要在注册表了注册才能使用,在AndroidManifest的activity标签下再加一个activity标签,代码如下

<activity android:name="com.example.s02_e06_pendingintent_remoteviews.TargetActivity"></activity>


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值