@blog.justoneplanet.info

日々勉強

androidでnotificationを使う

めも。備忘録。

■実装

以下のようにして実装する。

package info.justoneplanet.android.sample.notification;

import android.app.Activity;
import android.app.Notification;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.Intent;
import android.os.Bundle;

public class MainActivity extends Activity {
    public static final int NOTIFICATION_REQUEST_CODE = 100;
    
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        setNotification();
    }
    
    /**
     * 通知のセットアップ
     */
    private void setNotification() {
        NotificationManager notificationManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
        Notification notification = new Notification(
                R.drawable.ic_launcher,// 表示するアイコン
                getString(R.string.hello),// ステータスバーに表示するテキスト
                System.currentTimeMillis()// 表示する時間
        );
        
        // ペンディングできるIntentの入れ物
        PendingIntent intent = PendingIntent.getActivity(
                this,
                NOTIFICATION_REQUEST_CODE,
                new Intent(this, NotificationActivity.class),
                Intent.FLAG_ACTIVITY_NEW_TASK
        );
        
        // お知らせをクリックした時に起動するActivityを指定する
        notification.setLatestEventInfo(
                this,
                getString(R.string.app_name2),// 通知を開いた時のタイトル
                getString(R.string.hello2),// 通知を開いた時の本文
                intent
        );
        
        notification.vibrate = new long[]{0, 200, 100, 200, 100, 200, 100, 200, 100, 200, 100, 200};
        notificationManager.notify(R.string.app_name, notification);// 通知のID
    }
}

このように実装する事はまずないだろう。

Alarmと組み合わせる

MainActivity.java

PendingIntent.getBroadcastで生生したBroadcastReceicerをAlarmManagerに結びつける。

package info.justoneplanet.android.sample.notification;

import java.util.Date;

import android.app.Activity;
import android.app.AlarmManager;
import android.app.PendingIntent;
import android.content.Intent;
import android.os.Bundle;

public class MainActivity extends Activity {
    public static final int NOTIFICATION_BROADCAST_REQUEST_CODE = 100;
    
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        AlarmManager alarmManager = (AlarmManager) getSystemService(ALARM_SERVICE);
        alarmManager.setRepeating(
                AlarmManager.RTC_WAKEUP,
                new Date().getTime() + 1000,// 1秒後
                AlarmManager.INTERVAL_DAY,// 1日毎
                PendingIntent.getBroadcast(
                        this,
                        NOTIFICATION_BROADCAST_REQUEST_CODE,
                        new Intent(this, NotificationReceiver.class),
                        Intent.FLAG_ACTIVITY_NEW_TASK
                )
        );
    }
}
NotificationReceiver.java

broadcast receiver。contextが引き数で受け取れるのでActivityの時と同じ使い勝手でnotificationをセットアップする事ができる。

package info.justoneplanet.android.sample.notification;

import android.app.Notification;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;

public class NotificationReceiver extends BroadcastReceiver {
    @SuppressWarnings("unused")
    private static final String TAG = NotificationReceiver.class.getSimpleName();
    @SuppressWarnings("unused")
    private final NotificationReceiver self = this;
    public static final int NOTIFICATION_REQUEST_CODE = 100;

    @Override
    public void onReceive(Context context, Intent intent) {
        NotificationManager notificationManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
        Notification notification = new Notification(
                R.drawable.ic_launcher,// 表示するアイコン
                context.getString(R.string.hello),// ステータスバーに表示するテキスト
                System.currentTimeMillis()// 表示する時間
        );
        
        // ペンディングできるIntentの入れ物
        PendingIntent pendingIntent = PendingIntent.getActivity(
                context,
                NOTIFICATION_REQUEST_CODE,
                new Intent(context, NotificationActivity.class),
                0
        );
        
        // お知らせをクリックした時に起動するActivityを指定する
        notification.setLatestEventInfo(
                context,
                context.getString(R.string.app_name2),// 通知を開いた時のタイトル
                context.getString(R.string.hello2),// 通知を開いた時の本文
                pendingIntent
        );
        
        notification.vibrate = new long[]{0, 200, 100, 200, 100, 200, 100, 200, 100, 200, 100, 200};
        notificationManager.notify(R.string.app_name, notification);
     }
}
manifest

以下のように正しくreceiverを設定しないと上手く動作しない。

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

    <uses-sdk android:minSdkVersion="4" android:targetSdkVersion="14" />
    <uses-permission android:name="android.permission.VIBRATE" />
    
    <application
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name" >
        <receiver
            android:name=".NotificationReceiver"
            android:process=":remote" />
        <activity
            android:label="@string/app_name"
            android:name=".MainActivity" >
            <intent-filter >
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity
            android:label="@string/app_name"
            android:name=".NotificationActivity" >
        </activity>
    </application>
</manifest>

ServiceもBroadcastReceiverと組み合わせてnotificationを出せるし、C2DMもBroadcastReceiverなので同様である。

notificationの削除

起動されたActivity側で以下のようにして起動元別にキャンセル処理を挟む必要がある。

Intent intent = getIntent();
String origin = intent.getStringExtra("origin");
if (origin != null && origin.equals("notification")) {
    NotificationManager manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
    manager.cancel(R.string.app_name);
}

androidでサービスを使う

サービスには2種類の起動方法がある。

■サービスをスタート

ServiceActivity.java

package info.justoneplanet.android.sample.service;

import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
import android.widget.Toast;

public class ServiceActivity extends Activity {
    private Intent mIntent;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        // ブロードキャストレシーバの設定(サービスには直接関係しない)
        IntentFilter filter = new IntentFilter(SampleService.ACTION);
        registerReceiver(new ServiceReceiver(), filter);
    }
    
    @Override
    public void onResume() {
        super.onResume();
        
        // サービスの起動
        mIntent = new Intent(this, SampleService.class);
        startService(mIntent);
    }
    
    @Override
    public void onPause() {
        super.onPause();
        
        // サービスの停止
        stopService(mIntent);
    }
    
    /**
     * テスト用のブロードキャストレシーバ<br>
     * 実行されるとToastを表示する
     * @author justoneplanet
     */
    private class ServiceReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(getApplicationContext(), "hello", 200).show();
        }
    }
}

SampleService.java

package info.justoneplanet.android.sample.service;

import java.util.Timer;
import java.util.TimerTask;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.widget.Toast;

public class SampleService extends Service {
    public static final String ACTION = "SampleService.ACTION";
    private Timer mTimer;
    
    /**
     * サービスが一番初めに作られたときに実行される
     */
    @Override
    public void onCreate() {
        super.onCreate();
        mTimer = new Timer();
        Toast.makeText(getApplicationContext(), "onCreate", 1000).show();
    }
    
    /**
     * サービスがスタートするごとに実行される<br>
     * 但しこのメソッドは2.0以降は非推奨でonStartCommandを使うことが推奨されている
     */
    @Override
    public void onStart(Intent intent, int startId) {
        super.onStart(intent, startId);
        hello();
        Toast.makeText(getApplicationContext(), "onStart", 1000).show();
    }
    
    /**
     * サービスが無効になり取り除かれるときに実行される
     */
    @Override
    public void onDestroy() {
        super.onDestroy();
        mTimer.cancel();
        Toast.makeText(getApplicationContext(), "onDestroy", 1000).show();
    }
    
    /**
     * クライアントで登録したBroadcastReceiverに対してIntentを送る
     */
    public void hello() {
        mTimer.schedule(
                new TimerTask() {
                    @Override
                    public void run() {
                        sendBroadcast(new Intent(ACTION));
                    }
                },
                200,
                2000
        );
    }

    @Override
    public IBinder onBind(Intent intent) {
        Toast.makeText(getApplicationContext(), "onBind", 1000).show();
        return null;
    }
    
}

stopServiceを呼ばずにアプリを終了した場合は、バックグラウンドで動き続ける。

■サービスをバインド

ServiceActivity.java

package info.justoneplanet.android.sample.service;

import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.widget.Toast;

public class ServiceActivity extends Activity {
    private Intent mIntent;
    private SampleService mSampleService;
    
    /**
     * サービスをモニタリングするインターフェース<br>
     * 状態に応じて各メソッドがinvokeされる
     */
    private ServiceConnection conn = new ServiceConnection() {

        /**
         * サービスとの切断時に呼ばれる
         */
        @Override
        public void onServiceDisconnected(ComponentName name) {
            Toast.makeText(getApplicationContext(), "onServiceDisconnected", 1000).show();
            
        }
        
        /**
         * サービスとの接続時に呼ばれる
         */
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            Toast.makeText(getApplicationContext(), "onServiceConnected", 1000).show();
            mSampleService = ((SampleService.LocalBinder)service).getService();
            mSampleService.hello();
        }
    };

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        // ブロードキャストレシーバの設定(サービスには直接関係しない)
        IntentFilter filter = new IntentFilter(SampleService.ACTION);
        registerReceiver(new ServiceReceiver(), filter);
    }
    
    @Override
    public void onResume() {
        super.onResume();
        
        // サービスの起動
        mIntent = new Intent(this, SampleService.class);
        bindService(mIntent, conn, Context.BIND_AUTO_CREATE);
    }
    
    @Override
    public void onPause() {
        super.onPause();
        
        // サービスの停止
        unbindService(conn);
    }
    
    /**
     * テスト用のブロードキャストレシーバ<br>
     * 実行されるとToastを表示する
     * @author justoneplanet
     */
    private class ServiceReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(getApplicationContext(), "hello", 200).show();
        }
    }
}

SampleService.java

package info.justoneplanet.android.sample.service;

import java.util.Timer;
import java.util.TimerTask;

import android.app.Service;
import android.content.Intent;
import android.os.Binder;
import android.os.IBinder;
import android.widget.Toast;

public class SampleService extends Service {
    public static final String ACTION = "SampleService.ACTION";
    private Timer mTimer = new Timer();
    
    /**
     * ActivityでサービスのインターフェースにアクセスするためのBinder
     * @author justoneplanet
     */
    public class LocalBinder extends Binder {
        public SampleService getService() {
            return SampleService.this;
        }
    }
    private IBinder mIBinder = new LocalBinder();
    
    /**
     * バインドされた時に実行される
     */
    @Override
    public IBinder onBind(Intent intent) {
        Toast.makeText(getApplicationContext(), "onBind", 1000).show();
        return mIBinder;
    }
    
    /**
     * バインドが解除されtrueを返した後、再度バインドすると実行される
     */
    @Override
    public void onRebind(Intent intent) {
        Toast.makeText(getApplicationContext(), "onRebind", 1000).show();
    }
    
    /**
     * バインドが解除されたときに実行される
     */
    @Override
    public boolean onUnbind(Intent intent) {
        Toast.makeText(getApplicationContext(), "onUnbind", 1000).show();
        mTimer.cancel();
        return true;
    }
    
    /**
     * クライアントで登録したBroadcastReceiverに対してIntentを送る
     */
    public void hello() {
        mTimer.schedule(
                new TimerTask() {
                    @Override
                    public void run() {
                        sendBroadcast(new Intent(ACTION));
                    }
                },
                200,
                2000
        );
    }
}