В настоящее время я работаю над обработкой уведомлений pu sh с помощью Firebase Cloud Messaging. Все, что я реализовал, работает нормально, я получаю уведомления pu sh, как я хочу (относительно того, находится ли мое приложение на переднем плане, в фоне или убито) Но мой вопрос:
(собираюсь представить мою проблему в шаги для облегчения чтения)
- Мое приложение работает и находится на переднем плане.
- Я получаю уведомление pu sh, но не сразу его открываю
- Я убиваю приложение через диспетчер задач
- Я пытаюсь открыть приложение с уведомлением, но мое приложение разрывается и не открывается. (Я знаю, что это из-за объекта Intent, который я создал ранее, но я не нашел никакой документации, касающейся подобных ситуаций. Можно ли каким-либо образом манипулировать тем же уведомлением, которое уже отображается в трее, чтобы его намерение изменилось? )
Скелет моего приложения состоит из нескольких действий и множества фрагментов. Он начинается с действия запуска, которое вызывает мое основное действие, если каждый проверяемый параметр в порядке.
В моем уведомлении pu sh должен открываться фрагмент уведомления, который он выполняет, если Намерение относится к тому же состоянию приложения (фон, передний план или убит).
Вот мой класс Firebase
class MyFirebaseMessagingService : FirebaseMessagingService() {
val userPreferences = UserPreferences()
/**
* Called when message is received.
*
* @param remoteMessage Object representing the message received from Firebase Cloud Messaging.
*/
// [START receive_message]
override fun onMessageReceived(remoteMessage: RemoteMessage) {
// [START_EXCLUDE]
// There are two types of messages data messages and notification messages. Data messages are handled
// here in onMessageReceived whether the app is in the foreground or background. Data messages are the type
// traditionally used with GCM. Notification messages are only received here in onMessageReceived when the app
// is in the foreground. When the app is in the background an automatically generated notification is displayed.
// When the user taps on the notification they are returned to the app. Messages containing both notification
// and data payloads are treated as notification messages. The Firebase console always sends notification
// messages. For more see: https://firebase.google.com/docs/cloud-messaging/concept-options
// [END_EXCLUDE]
// TODO(developer): Handle FCM messages here.
// Not getting messages here? See why this may be:
// Check if message contains a data payload.
remoteMessage.data.isNotEmpty().let {
val messageTitle: String = remoteMessage.data["title"] ?: "Hotel"
val message: String = remoteMessage.data["message"] ?: "You have new notification"
val groupId: String = remoteMessage.data["group_id"]?: "0"
val id: String = remoteMessage.data["id"]?: "0"
if(!userPreferences.visibleFragment.isNullOrEmpty() && groupId=="3")//IF NOT NULL OR EMPTY MUST BE CHATFRAGMENT
{
//no message since its chat notification and the chatfragment is visible
}
else
{
if(groupId=="4")
{
MainActivity.isNotificationIconVisible.postValue(true)
userPreferences.isNotificationIconVisible=true
}
sendNotification(messageTitle, message, groupId, id)
}
}
// Check if message contains a notification payload.
remoteMessage.notification?.let {
}
// Also if you intend on generating your own notifications as a result of a received FCM
// message, here is where that should be initiated. See sendNotification method below.
}
// [END receive_message]
// [START on_new_token]
/**
* Called if InstanceID token is updated. This may occur if the security of
* the previous token had been compromised. Note that this is called when the InstanceID token
* is initially generated so this is where you would retrieve the token.
*/
override fun onNewToken(token: String) {
}
// [END on_new_token]
/**
* Handle time allotted to BroadcastReceivers.
*/
private fun handleNow() {
}
/**
* Persist token to third-party servers.
*
* Modify this method to associate the user's FCM InstanceID token with any server-side account
* maintained by your application.
*
* @param token The new token.
*/
private fun sendRegistrationToServer(token: String?) {
}
/**
* Create and show a simple notification containing the received FCM message.
*
* @param messageBody FCM message body received.
*/
private fun sendNotification(messageTitle: String, messageBody: String, groupId : String, id : String) {
val intent =
if (userPreferences.isAppDestroyed) {
Intent(this, MainActivity::class.java)
} else {
Intent(this, MainActivity::class.java)
.putExtra("id", id)
.putExtra("groupid", groupId)
.putExtra("appDestroyed", userPreferences.isAppDestroyed)
}
val pendingIntent =
if(userPreferences.isAppDestroyed)
{
TaskStackBuilder.create(this).run {
addNextIntentWithParentStack(intent)
editIntentAt(0).putExtra("id", id).putExtra("groupid", groupId).putExtra("appDestroyed", userPreferences.isAppDestroyed)
getPendingIntent(0, PendingIntent.FLAG_UPDATE_CURRENT)
}
}
else
{
PendingIntent.getActivity(this, 0 , intent, PendingIntent.FLAG_UPDATE_CURRENT)
}
val mode = applicationContext?.resources?.configuration?.uiMode?.and(Configuration.UI_MODE_NIGHT_MASK)
val channelId = "1"
val defaultSoundUri = RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION)
val notificationBuilder = NotificationCompat.Builder(this, channelId)
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
when (mode) {
Configuration.UI_MODE_NIGHT_YES -> {
notificationBuilder.setSmallIcon(R.drawable.ic_notification_icon_from_asset)
notificationBuilder.setColor(Color.TRANSPARENT)
}
Configuration.UI_MODE_NIGHT_NO -> {
notificationBuilder.setSmallIcon(R.drawable.ic_notification_icon_from_asset)
notificationBuilder.setColor(Color.GRAY)
}
else -> {
notificationBuilder.setSmallIcon(R.drawable.ic_notification_icon_from_asset)
notificationBuilder.setColor(Color.TRANSPARENT)
}
}
} else {
notificationBuilder.setSmallIcon(R.mipmap.ic_notification_icons)
}
.setContentTitle(messageTitle)
.setContentText(messageBody)
.setAutoCancel(true)
.setSound(defaultSoundUri)
.setContentIntent(pendingIntent)
val notificationManager = getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
// Since android Oreo notification channel is needed.
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
val channel = NotificationChannel(channelId,
"Channel human readable title",
NotificationManager.IMPORTANCE_DEFAULT)
notificationManager.createNotificationChannel(channel)
}
notificationManager.notify(1, notificationBuilder.build())
}
}
MainActivity onNewIntent fun (код транзакции фрагмента прокомментирован, что-то проверено)
override fun onNewIntent(intent: Intent?) {
super.onNewIntent(intent)
val data = intent?.extras
val groupId = data?.getString("groupid")
val id = data?.getString("id")
if (data != null) {
if (groupId == "4") {
val bundle = bundleOf("pushNotificationId" to id)
navController.navigate(R.id.notificationDetailsFragment, bundle)
/*val fragment = NotificationDetailsFragment()
fragment.arguments=bundle
val ft = supportFragmentManager.beginTransaction()
ft.replace(R.id.main_nav_fragment, fragment)
ft.commit()*/
}
else return
}
return
}
}
LauncherActivity
class LauncherActivity : AppCompatActivity(){
private val vm: LauncherViewModel by viewModel()
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
injectFeature()
// set observables
vm.isSplashScreenShown.observe(this, Observer {
finish()
when (it) {
true -> startActivity(Intent(this, MainActivity::class.java).putExtra("id",intent?.extras?.getString("id"))
.putExtra("groupid",intent?.extras?.getString("groupid")))
false -> startActivity(Intent(this, WelcomeActivity::class.java))
}
})
}
}
и часть моего манифеста, если необходимо
<activity
android:name=".presentation.launcher.LauncherActivity"
android:label="@string/app_name"
android:theme="@style/BaseAppTheme.NoActionBar.SplashScreenTheme"
android:screenOrientation="portrait">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name=".presentation.welcome.WelcomeActivity"
android:screenOrientation="portrait"
android:theme="@style/BaseAppTheme.NoActionBar.TransparentTheme"
android:windowSoftInputMode="adjustResize" />
<activity
android:name=".presentation.error.ErrorActivity"
android:screenOrientation="portrait"
android:theme="@style/BaseAppTheme.NoActionBar.TransparentTheme" />
<activity
android:name=".presentation.main.MainActivity"
android:theme="@style/BaseAppTheme.NoActionBar.TransparentTheme"
android:screenOrientation="portrait"
android:launchMode="singleTask"
android:parentActivityName=".presentation.launcher.LauncherActivity"/>
Я бы хотел, чтобы это было сделано так, как это делает Whatsapp (WhatsApp открывает свой чат, даже если повторите шаги, как я делал для своего приложения)
Большое спасибо, если нужно что-то еще, или я нужно переформатировать мой вопрос, пожалуйста, посоветуйте мне сделать это.