PDATE:以下のコードの問題を修正したので、GCM
の使用方法の基本的な作業例になります。
だから、Android GCM
をアプリに実装しようとしています。マニフェストに追加した関連部分は次のとおりです。
_<uses-sdk Android:minSdkVersion="8" Android:targetSdkVersion="20" />
<permission
Android:name=".permission.C2D_MESSAGE"
Android:protectionLevel="signature" />
<uses-permission Android:name="com.google.Android.c2dm.permission.RECEIVE" />
<uses-permission Android:name=".permission.C2D_MESSAGE" />
<uses-permission Android:name="Android.permission.WAKE_LOCK"/>
<uses-permission Android:name="Android.permission.GET_ACCOUNTS"/>
_
...
_<receiver
Android:name="com.google.Android.gcm.GCMBroadcastReceiver"
Android:permission="com.google.Android.c2dm.permission.SEND" >
<intent-filter>
<action Android:name="com.google.Android.c2dm.intent.RECEIVE" />
<action Android:name="com.google.Android.c2dm.intent.REGISTRATION" />
<category Android:name="com.badbob.app.gmctestapp" />
</intent-filter>
</receiver>
<service Android:name=".GCMIntentService" />
_
メインアクティビティのonCreateに次のコードを追加しました。
_ GCMRegistrar.checkDevice( this );
GCMRegistrar.checkManifest( this );
final String regId = GCMRegistrar.getRegistrationId( this );
if( regId.equals( "" ) ) {
GCMRegistrar.register( this, GCM_SENDER_ID );
}
else {
Log.v( LOG_TAG, "Already registered" );
}
_
また、次のようにGCMIntenetService
クラスを作成しました。
_public class GCMIntentService extends GCMBaseIntentService {
private static final String LOG_TAG = "GetAClue::GCMIntentService";
public GCMIntentService() {
super( GCM_SENDER_ID );
// TODO Auto-generated constructor stub
Log.i( LOG_TAG, "GCMIntentService constructor called" );
}
@Override
protected void onError( Context arg0, String errorId ) {
// TODO Auto-generated method stub
Log.i( LOG_TAG, "GCMIntentService onError called: " + errorId );
}
@Override
protected void onMessage( Context arg0, Intent intent ) {
// TODO Auto-generated method stub
Log.i( LOG_TAG, "GCMIntentService onMessage called" );
Log.i( LOG_TAG, "Message is: " + intent.getStringExtra( "message" ) );
}
@Override
protected void onRegistered( Context arg0, String registrationId ) {
// TODO Auto-generated method stub
Log.i( LOG_TAG, "GCMIntentService onRegistered called" );
Log.i( LOG_TAG, "Registration id is: " + registrationId );
}
@Override
protected void onUnregistered( Context arg0, String registrationId ) {
// TODO Auto-generated method stub
Log.i( LOG_TAG, "GCMIntentService onUnregistered called" );
Log.i( LOG_TAG, "Registration id is: " + registrationId );
}
}
_
これを実行するとLogCatで取得されます:
_07-11 11:28:46.340: V/GCMRegistrar(27435): Registering receiver
07-11 11:28:46.370: D/GCMRegistrar(27435): resetting backoff for com.badbob.app.getacluebeta
07-11 11:28:46.380: V/GCMRegistrar(27435): Registering app com.badbob.app.getacluebeta of senders 128205395388
_
私が他の投稿から収集したものから、LogCat
で登録IDを取得する必要がありますが、そうではありません。また、GCMIntentService
のonRegistered()
は呼び出されません。私は間違っていますか?
これは間違っています
protected GCMIntentService( String senderId ) {
super(senderId);}
ドキュメントに記載されているとおり。 GCMIntentServiceのPUBLIC、NO ARGUMENTコンストラクターを宣言する必要があります。そうしないと、バックグラウンドインテントサービスによってGCMIntentServiceを適切にインスタンス化できません。
public GCMIntentService(){
super(senderID);}
senderIDは、新しいGCMで変更されないため、ハードコードされた定数文字列にすることができます。正しいsenderIDを使用することも非常に重要です。 24時間で十分に長くなるため、上記の解決策が機能しない場合は、間違ったsenderIDを使用しています。他のすべてが素晴らしく見えます。
SenderIDは、Google APIアクセスページを閲覧しているときのWebブラウザーのURLにあります。 GCMをクリックすると、ブラウザのURLに12桁の数字が表示されます。これが正しいキーです。 APIキーではありません。これは、Appサーバー側で使用されます。
また、今までこの迷惑なバグがありました。問題は、別のパッケージ内でインテントサービスを宣言したことです... Androidマニフェストで名前を宣言し、classnotfound例外が配置されていなかった場合でも...エラーはまったく発生しませんでした見つかった...だから私はインテントサービスがルートパッケージにあることを確認しました。
[〜#〜] gcm [〜#〜]のチュートリアルを参照してください:-
コードに問題があると思います。
マニフェストファイルで独自のブロードキャストレシーバーの登録を行う必要があり、そのレシーバーは<service Android:name=".GCMIntentService" />
。
したがって、あなたは私が以下に書くようなことをしなければなりません。
受信者は次のように宣言する必要があります。
<receiver
Android:name="your.package.name.YourBroadCastReceiver"
Android:permission="com.google.Android.c2dm.permission.SEND" >
<intent-filter>
<action Android:name="com.google.Android.c2dm.intent.RECEIVE" />
<action Android:name="com.google.Android.c2dm.intent.REGISTRATION" />
</intent-filter>
サービスを開始する放送受信機。
public class YourBroadCastReceiver extends BroadcastReceiver {
@Override
public final void onReceive(Context context, Intent intent) {
GCMIntentService .runIntentInService(context, intent);
setResult(Activity.RESULT_OK, null, null);
}
}
公式のGCM documentation をご覧になるとよい例が見つかります。
そして...メインのGoogle APIコンソールページでGoogle Cloud Messagingサービスを有効にすることを忘れないでください。
役立つかどうか教えてください!
ここで、ゼロから開始してRegIDと通知を取得する方法の手順をいくつか書きました
完全なチュートリアルは、URLリンクの下にあります
登録ID(プッシュ通知用のデバイストークン)を取得するためのコードスニペット。
GCMのプロジェクトを構成する
プロジェクトでGCMを有効にするには、マニフェストファイルにいくつかの権限を追加する必要がありますAndroidManifest.xmlに移動し、以下のコードを追加します権限を追加
<uses-permission Android:name="Android.permission.INTERNET”/>
<uses-permission Android:name="Android.permission.GET_ACCOUNTS" />
<uses-permission Android:name="Android.permission.WAKE_LOCK" />
<uses-permission Android:name="Android.permission.VIBRATE" />
<uses-permission Android:name=“.permission.RECEIVE" />
<uses-permission Android:name=“<your_package_name_here>.permission.C2D_MESSAGE" />
<permission Android:name=“<your_package_name_here>.permission.C2D_MESSAGE"
Android:protectionLevel="signature" />
GCMブロードキャストレシーバー宣言を追加
アプリケーションタグにGCM Broadcast Receiver宣言を追加します
<application
<receiver
Android:name=".GcmBroadcastReceiver"
Android:permission="com.google.Android.c2dm.permission.SEND" ]]>
<intent-filter]]>
<action Android:name="com.google.Android.c2dm.intent.RECEIVE" />
<category Android:name="" />
</intent-filter]]>
</receiver]]>
<application/>
GCM Servie宣言を追加
<application
<service Android:name=".GcmIntentService" />
<application/>
Now Go to Launch/Splash Activity
定数とクラス変数を追加する
private final static int PLAY_SERVICES_RESOLUTION_REQUEST = 9000;
public static final String EXTRA_MESSAGE = "message";
public static final String PROPERTY_REG_ID = "registration_id";
private static final String PROPERTY_APP_VERSION = "appVersion";
private final static String TAG = "LaunchActivity";
protected String SENDER_ID = "Your_sender_id";
private GoogleCloudMessaging gcm =null;
private String regid = null;
private Context context= null;
OnCreateメソッドとOnResumeメソッドを更新する
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_launch);
context = getApplicationContext();
if (checkPlayServices())
{
gcm = GoogleCloudMessaging.getInstance(this);
regid = getRegistrationId(context);
if (regid.isEmpty())
{
registerInBackground();
}
else
{
Log.d(TAG, "No valid Google Play Services APK found.");
}
}
}
@Override protected void onResume()
{
super.onResume(); checkPlayServices();
}
# Implement GCM Required methods (Add below methods in LaunchActivity)
private boolean checkPlayServices() {
int resultCode = GooglePlayServicesUtil.isGooglePlayServicesAvailable(this);
if (resultCode != ConnectionResult.SUCCESS) {
if (GooglePlayServicesUtil.isUserRecoverableError(resultCode)) {
GooglePlayServicesUtil.getErrorDialog(resultCode, this,
PLAY_SERVICES_RESOLUTION_REQUEST).show();
} else {
Log.d(TAG, "This device is not supported - Google Play Services.");
finish();
}
return false;
}
return true;
}
private String getRegistrationId(Context context)
{
final SharedPreferences prefs = getGCMPreferences(context);
String registrationId = prefs.getString(PROPERTY_REG_ID, "");
if (registrationId.isEmpty()) {
Log.d(TAG, "Registration ID not found.");
return "";
}
int registeredVersion = prefs.getInt(PROPERTY_APP_VERSION, Integer.MIN_VALUE);
int currentVersion = getAppVersion(context);
if (registeredVersion != currentVersion) {
Log.d(TAG, "App version changed.");
return "";
}
return registrationId;
}
private SharedPreferences getGCMPreferences(Context context)
{
return getSharedPreferences(LaunchActivity.class.getSimpleName(),
Context.MODE_PRIVATE);
}
private static int getAppVersion(Context context)
{
try
{
PackageInfo packageInfo = context.getPackageManager()
.getPackageInfo(context.getPackageName(), 0);
return packageInfo.versionCode;
}
catch (NameNotFoundException e)
{
throw new RuntimeException("Could not get package name: " + e);
}
}
private void registerInBackground()
{ new AsyncTask() {
Override
protected Object doInBackground(Object... params)
{
String msg = "";
try
{
if (gcm == null)
{
gcm = GoogleCloudMessaging.getInstance(context);
}
regid = gcm.register(SENDER_ID); Log.d(TAG, "########################################");
Log.d(TAG, "Current Device's Registration ID is: "+msg);
}
catch (IOException ex)
{
msg = "Error :" + ex.getMessage();
}
return null;
} protected void onPostExecute(Object result)
{ //to do here };
}.execute(null, null, null);
}
注:REGISTRATION_KEYを保存してください。GCMにPNメッセージを送信することも重要です。プッシュ通知を送信します。
GCMブロードキャストレシーバークラスを追加する
マニフェストファイルで既に「GcmBroadcastReceiver.Java」を宣言しているので、このクラスを作成して、レシーバクラスコードをこのように更新します。
public class GcmBroadcastReceiver extends WakefulBroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent)
{ ComponentName comp = new ComponentName(context.getPackageName(),
GcmIntentService.class.getName()); startWakefulService(context, (intent.setComponent(comp)));
setResultCode(Activity.RESULT_OK);
Toast.makeText(context, “wow!! received new Push notification", Toast.LENGTH_LONG).show();
}
}
GCMサービスクラスを追加
マニフェストファイルで既に「GcmBroadcastReceiver.Java」を宣言しているので、このクラスを作成して、レシーバクラスコードをこのように更新します。
public class GcmIntentService extends IntentService
{ public static final int NOTIFICATION_ID = 1; private NotificationManager mNotificationManager; private final static String TAG = "GcmIntentService"; public GcmIntentService() {
super("GcmIntentService");
} @Override
protected void onHandleIntent(Intent intent) {
Bundle extras = intent.getExtras();
Log.d(TAG, "Notification Data Json :" + extras.getString("message"));
GoogleCloudMessaging gcm = GoogleCloudMessaging.getInstance(this);
String messageType = gcm.getMessageType(intent); if (!extras.isEmpty()) { if (GoogleCloudMessaging.MESSAGE_TYPE_SEND_ERROR
.equals(messageType)) {
sendNotification("Send error: " + extras.toString());
} else if (GoogleCloudMessaging.MESSAGE_TYPE_DELETED
.equals(messageType)) {
sendNotification("Deleted messages on server: "
+ extras.toString()); // If it's a regular GCM message, do some work.
} else if (GoogleCloudMessaging.MESSAGE_TYPE_MESSAGE
.equals(messageType)) {
// This loop represents the service doing some work.
for (int i = 0; i < 5; i++) {
Log.d(TAG," Working... " + (i + 1) + "/5 @ "
+ SystemClock.elapsedRealtime()); try {
Thread.sleep(5000);
} catch (InterruptedException e) {
}
}
Log.i(TAG, "Completed work @ " + SystemClock.elapsedRealtime());
sendNotification(extras.getString("message"));
}
} // Release the wake lock provided by the WakefulBroadcastReceiver.
GcmBroadcastReceiver.completeWakefulIntent(intent);
} // Put the message into a notification and post it.
// This is just one simple example of what you might choose to do with
// a GCM message.
private void sendNotification(String msg) { mNotificationManager = (NotificationManager) this
.getSystemService(Context.NOTIFICATION_SERVICE);
PendingIntent contentIntent = PendingIntent.getActivity(this, 0, new Intent(this, LaunchActivity.class), 0);
NotificationCompat.Builder mBuilder = new NotificationCompat.Builder( this)
.setSmallIcon(R.drawable.icon)
.setContentTitle("Ocutag Snap")
.setStyle(new NotificationCompat.BigTextStyle().bigText(msg))
.setContentText(msg)
.setDefaults(Notification.DEFAULT_SOUND | Notification.DEFAULT_VIBRATE);
mBuilder.setContentIntent(contentIntent); mNotificationManager.notify(NOTIFICATION_ID, mBuilder.build());
}
}
登録から応答が得られない場合、主な理由の1つは、マニフェストファイルが正しく構成されていないことです。特に、<permission>
および<intent filter>
正しく。
これが特定の問題に関連しているかどうかはわかりませんが、同じ問題が発生していました。私が知る限り、すべてが正しくセットアップされましたが、onRegistered()
ハンドラーは呼び出されませんでした。
GCMIntentServiceクラスの一部として、標準のonCreate()
、onStart()
およびonDestroy()
サービスメソッドがありました。それらを削除すると、期待どおりハンドラーが呼び出されました。
私の想定では、これらのメソッドをオーバーライドすることで、必要なコードの初期化をバイパスしていました。
GCMBaseIntentService.onRegistered() コールバックでアプリケーションサーバーに登録する必要もあります。そうしないと、アプリのメッセージを送信できません。
GoogleのGCMデモクライアント を実際にチェックアウトする必要があります。これは、すべての正しいコードの実用サンプルを提供するためです。また、githubに作業サンプルがあります: airbop-client 。 GCM対応のアプリに必要なアプリケーションサーバーとの相互作用があるため、もう少し完全な例を示します。 AirBop サーバー(これは作成に役立ちました)で動作し、最大1000台のデバイスが無料です。
ClouDeveloperの投稿を読んだ後、onCreate()
、onDestroy()
、およびその他のonXXX()コールバックメソッドを削除しました。また、私のGCMIntentService.onRegistered(Context, regId)
はregId
で呼び出されます。
ClouDeveloperに感謝します!