Setup
Installation
Prerequisite
- Add Firebase to your Android project. official docs
- Get your FCM Sender ID & FCM Server API Key.
- Login/Register on CleverPush
- Your CleverPush Channel ID, available in
Channels
>App Push
>Implementation
>Channel ID
in the CleverPush Developer console.
SDK Setup
You can find the newest sdk version number here Android SDK
- Add the dependency to your app-level
app/build.gradle
file
dependencies {
// ...
implementation 'com.cleverpush:cleverpush:1.34.14'
}
- In the
onCreate
method of your Main activity or Application, callCleverPush.getInstance(this).init(...)
with your CleverPush Channel ID.
import com.cleverpush.CleverPush;
public class MainActivity extends Activity {
public void onCreate(Bundle savedInstanceState) {
CleverPush.getInstance(this).init("CLEVERPUSH_CHANNEL_ID");
}
}
import com.cleverpush.CleverPush;
class MainActivity:Activity() {
fun onCreate(savedInstanceState:Bundle) {
CleverPush.getInstance(this).init("CLEVERPUSH_CHANNEL_ID")
}
}
If you didnt already do this while adding firebase to your project or you have some error logs then do/check following points
Put your google-services.json from firebase into the project e.g. in the app folder
To your project gradle add following dependency
buildscript {
dependencies {
classpath 'com.google.gms:google-services:4.3.3'
}
}
- And add this at the end of your app gradle file
apply plugin: 'com.google.gms.google-services'
googleServices { disableVersionCheck = true }
Please note that autoRegister
is turned to true
in the above example. It means that the CleverPush SDK will automatically try to subscribe the user on the first launch of the app. If you call unsubscribe()
the SDK will not automatically try to subscribe again.
Setup HMS (Huawei Mobile Services)
These additional steps are recommended if you want to support the newest Huawei devices. Please note that you need at least CleverPush Android SDK version 1.1.0.
Important: we highly recommend to set up different build flavors/variants for Huawei and Firebase, because Google will not allow any apps in the Play Store with the Huawei SDK. You can then upload the Huawei build to the Huawei AppGallery and the Firebase build to the Google Play Store.
Create a free Huawei Developer Account and get it verified: https://developer.huawei.com/consumer/en/console
Create your app inside Huwei AppGallery Connect: https://developer.huawei.com/consumer/en/doc/distribution/app/agc-create_app
Enable the PushKit API: https://developer.huawei.com/consumer/en/doc/distribution/app/agc-enable_service
Add the following to your project level build.gradle:
buildscript { repositories { // ... maven { url 'http://developer.huawei.com/repo/' } } dependencies { // ... classpath 'com.huawei.agconnect:agcp:1.2.1.301' } } allprojects { repositories { // ... maven { url 'http://developer.huawei.com/repo/' } } }
Add the following to your app level build.gradle:
dependencies { // ... implementation 'com.huawei.hms:push:4.0.3.301' } // at the bottom apply plugin: "com.huawei.agconnect"
If your App has a minSdkVersion lower than 17, please set your minSdkVersion to at least 17.
Copy the agconnect-services.json file from the Huawei Developer Console inside your project's app directory.
- Enter the App ID + App Secret in the CleverPush Channel settings.
Proguard
-keep class com.cleverpush.** { *; }
-keep interface com.cleverpush.** { *; }
-keep class com.firebase.** { *; }
-keep class com.google.firebase.** { *; }
-keep class com.google.gson.internal.LinkedTreeMap { *; }
Custom sounds
Android supports mp3
, wav
and ogg
audio files.
- Add the sound file(s) to the
res/raw
directory. - When sending a notification you can enter the filename (with extension) in the field "Sound" in the advanced settings.
- If you send notifications via the API you can use the parameter "soundFilename".
Badge icon
You can place your custom badge icon with the correct sizes in this folders, then the SDK will automatically use it:
/drawable-[SIZE]/cleverpush_notification_icon.png
Notification open behaviour
You can place this meta data option in your AndroidManifest.xml file to prevent our SDK from starting your launcher activity when a notification is opened:
<meta-data android:name="com.cleverpush.notification_open_activity_disabled" android:value="true" />
This can be used in combination with the NotificationOpenedCallbackListener so that you can decide when to finish the NotificationOpenedActivity.
public void notificationOpenedCallback(NotificationOpenedResult result, Activity notificationOpenedActivity) {
System.out.println("Opened CleverPush Notification: " + result.getNotification().getTitle());
// Add your custom code here and afterwards call:
notificationOpenedActivity.finish();
}
fun notificationOpenedCallback(result: NotificationOpenedResult, notificationOpenedActivity: Activity) {
println("Opened CleverPush Notification: ${result.notification.title}")
// Add your custom code here and afterwards call:
notificationOpenedActivity.finish()
}
Badge Counts
Disable automatic clearing of badge count when opening a notification. Enabled by default.
CleverPush.getInstance(this).setAutoClearBadge(false);
CleverPush.getInstance(this).setAutoClearBadge(false)
Enable automatic incrementation of badge count. Disabled by default.
CleverPush.getInstance(this).setIncrementBadge(false);
CleverPush.getInstance(this).setIncrementBadge(false)
Notification permission
By default, the SDK automatically unsubscribes users who have revoked their notification permission in the Android settings. Sometimes it still makes sense to subscribe those users (e.g. for silent notifications). You can disable this behaviour with this method call (before init).
CleverPush.getInstance(this).setIgnoreDisabledNotificationPermission(true);
CleverPush.getInstance(this).setIgnoreDisabledNotificationPermission(true)
Using CleverPush together with other FirebaseMessagingServices
As it is only possible to have one FirebaseMessagingService registered at the same time, we recommend the following code. The CleverPush FCM Service will only process CleverPush notifications and ignore all other messages.
Create the FirebaseMessagingServiceProxy
class:
import androidx.annotation.NonNull;
import com.cleverpush.service.CleverPushFcmListenerService;
import com.google.firebase.messaging.FirebaseMessagingService;
import com.google.firebase.messaging.RemoteMessage;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.List;
public class FirebaseMessagingServiceProxy extends FirebaseMessagingService {
private final List<FirebaseMessagingService> messagingServices = new ArrayList<>(2);
public FirebaseMessagingServiceProxy() {
messagingServices.add(new CleverPushFcmListenerService());
// Add any other FirebaseMessagingServices here
messagingServices.add(new CustomFirebaseMessagingService());
}
@Override
public void onNewToken(@NonNull String token) {
delegate(service -> {
injectContext(service);
service.onNewToken(token);
});
}
@Override
public void onMessageReceived(@NonNull RemoteMessage remoteMessage) {
delegate(service -> {
injectContext(service);
service.onMessageReceived(remoteMessage);
});
}
private void delegate(CPAction<FirebaseMessagingService> action) {
for (FirebaseMessagingService service : messagingServices) {
action.run(service);
}
}
private void injectContext(FirebaseMessagingService service) {
setField(service, "mBase", this);
}
private boolean setField(Object targetObject, String fieldName, Object fieldValue) {
Field field;
try {
field = targetObject.getClass().getDeclaredField(fieldName);
} catch (NoSuchFieldException e) {
field = null;
}
Class superClass = targetObject.getClass().getSuperclass();
while (field == null && superClass != null) {
try {
field = superClass.getDeclaredField(fieldName);
} catch (NoSuchFieldException e) {
superClass = superClass.getSuperclass();
}
}
if (field == null) {
return false;
}
field.setAccessible(true);
try {
field.set(targetObject, fieldValue);
return true;
} catch (IllegalAccessException e) {
return false;
}
}
interface CPAction<T> {
void run(T t);
}
}
import androidx.annotation.NonNull
import com.cleverpush.service.CleverPushFcmListenerService
import com.google.firebase.messaging.FirebaseMessagingService
import com.google.firebase.messaging.RemoteMessage
import java.lang.reflect.Field
class FirebaseMessagingServiceProxy : FirebaseMessagingService() {
private val messagingServices = listOf<CleverPushFcmListenerService>(
CleverPushFcmListenerService(),
// Add any other FirebaseMessagingServices here
CustomFirebaseMessagingService()
)
override fun onNewToken(@NonNull token: String) {
delegate { service ->
injectContext(service)
service.onNewToken(token)
}
}
override fun onMessageReceived(@NonNull remoteMessage: RemoteMessage) {
delegate { service ->
injectContext(service)
service.onMessageReceived(remoteMessage)
}
}
private fun delegate(action: (FirebaseMessagingService) -> Unit) {
messagingServices.forEach(action)
}
private fun injectContext(service: FirebaseMessagingService) {
setField(service, "mBase", this)
}
private fun setField(targetObject: Any, fieldName: String, fieldValue: Any): Boolean {
var field: Field? = null
try {
field = targetObject.javaClass.getDeclaredField(fieldName)
} catch (e: NoSuchFieldException) {
var superClass: Class<*>? = targetObject.javaClass.superclass
while (field == null && superClass != null) {
try {
field = superClass.getDeclaredField(fieldName)
} catch (e: NoSuchFieldException) {
superClass = superClass.superclass
}
}
}
if (field == null) {
return false
}
field.isAccessible = true
try {
field.set(targetObject, fieldValue)
return true
} catch (e: IllegalAccessException) {
return false
}
}
}
Add this to your AndroidManifest.xml inside the <application>
tag:
<service
android:name=".FirebaseMessagingServiceProxy"
android:exported="false">
<intent-filter>
<action android:name="com.google.firebase.MESSAGING_EVENT" />
</intent-filter>
</service>
If you're only looking to customize the incoming CleverPush notifications, please look at the "Notification Service Extension" in the left menu.