提问人:Dmitrii Volkov 提问时间:11/17/2023 最后编辑:Dmitrii Volkov 更新时间:11/18/2023 访问量:52
需要在指定时间显示通知,android kotlin
Need to show notification at specified time, android kotlin
问:
创建的类通知:BroadcastReceiver():
const val notificationID = 0
const val channelID = "com.example.avnotification.MyUIRoom"
const val titleExtra = "titleExtra"
const val messageExtra = "messageExtra"
class Notification : BroadcastReceiver() {
override fun onReceive(context: Context, intent: Intent) {
Log.d("MyLog", "Notification: onReceive ")
val activityToOpen = intent.getSerializableExtra("activity") as Class<*>
//val notificationManager = context.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
//val pendingIntent: PendingIntent = PendingIntent.getActivity(context, 0, Intent(context, activityToOpen), PendingIntent.FLAG_IMMUTABLE)
val pendingIntent: PendingIntent = PendingIntent.getActivity(context, 0, Intent(context, activityToOpen), 0)
val notification = NotificationCompat.Builder(context, channelID)
.setSmallIcon(R.drawable.ic_launcher_foreground)
.setContentTitle(intent.getStringExtra(titleExtra))
.setContentText(intent.getStringExtra(messageExtra))
.setPriority(NotificationCompat.PRIORITY_HIGH)
.setContentIntent(pendingIntent)
.setAutoCancel(true)
.build()
// if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
// val channel = NotificationChannel(
// channelID,
// "Notif Channel",
// NotificationManager.IMPORTANCE_DEFAULT
// )
// notificationManager.createNotificationChannel(channel)
// }
val manager = context.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
manager.notify(notificationID, notification)
}
}
在此代码片段中,我设置了日期和时间,检查权限并尝试查看通知,但它没有出现:
@RequiresApi(Build.VERSION_CODES.O)
class PanelEditTask : BottomSheetDialogFragment(), View.OnClickListener{
private var binding: PanelEditTaskBinding? = null
private var nameFormParent: String? = ""
private var taskAction: String? = ""
private var complet: String? = null
private var idTask: Int? = null
private var email: String? = "test"
private var typeTask: String? = null
private var taskRepository: TaskRepository? = null
private var taskViewModel: TaskViewModel? = null
private var taskFactory: TaskFactory? = null
private var firstStart: Boolean = true
private var currentDate = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
LocalDate.now()
} else {
TODO("VERSION.SDK_INT < O")
}
private val dateFormatter: DateTimeFormatter
init {
dateFormatter = DateTimeFormatterBuilder()
.appendValue(ChronoField.YEAR, 4)
.appendLiteral('-')
.appendValue(ChronoField.MONTH_OF_YEAR, 2)
.appendLiteral('-')
.appendValue(ChronoField.DAY_OF_MONTH, 2)
.toFormatter()
}
override fun onCreateView(
inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?
): View? {
// Inflate the layout for this fragment
binding = PanelEditTaskBinding.inflate(inflater, container, false)
taskAction = arguments?.getString("taskAction").toString()
nameFormParent = arguments?.getString("nameForm").toString()
binding?.DateStartTask?.setText(arguments?.getString("dateStart").toString())
complet = arguments?.getString("completed").toString()
if (complet == "true"){
binding?.checkBoxCompleted?.isChecked = true
binding?.DateEndTask?.setText(arguments?.getString("dateEnd").toString())
}
else {
binding?.DateEndTask?.setText(getString(R.string.enter_end_start))
binding?.checkBoxCompleted?.isChecked = false}
if(taskAction == "Edit") {
idTask = arguments?.getString("idTask")?.toInt()
binding?.editNameTask?.setText(arguments?.getString("nameTask").toString())
email = arguments?.getString("email")?.toString()
typeTask = arguments?.getString("typeTask")?.toString()
binding?.editInfoTask?.setText(arguments?.getString("infoTask").toString())
}
else{
val sharedPreferences = requireContext().getSharedPreferences("MyPrefs", Context.MODE_PRIVATE)
val savedValue = sharedPreferences.getString("email", "default value")
if(savedValue != "default value") {
email = savedValue
}
}
val taskDao = Database.getInstance((context as FragmentActivity).application).taskDao
taskRepository = TaskRepository(taskDao)
taskFactory = TaskFactory(taskRepository!!)
taskViewModel = ViewModelProvider(this, taskFactory!!).get(TaskViewModel::class.java)
spinerProcessing()
// createNotification()
binding?.finishEdit?.setOnClickListener(this)
binding?.checkBoxCompleted?.setOnCheckedChangeListener{ buttonView, isChecked ->
if(isChecked){
binding?.DateEndTask?.setText(currentDate.toString())
}
else {
binding?.DateEndTask?.setText(getString(R.string.enter_end_start))}
}
binding?.DateEndTask?.setOnClickListener{
showDatePickerDialog("End")
}
binding?.DateStartTask?.setOnClickListener{
showDatePickerDialog("Start")
}
createNotification()
binding?.greateNotif?.setOnClickListener{scheduleNotification()}
return binding?.root
}
private fun createNotification() {
val name = "Notif Channel"
val desc = "A description of the Channel"
val importance = NotificationManager.IMPORTANCE_HIGH
val channel = NotificationChannel(channelID, name, importance)
channel.description = desc
// Set notification sound
channel.setSound(RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION), null)
channel.enableVibration(true)
channel.enableLights(true)
val notificationManager = requireContext().getSystemService(NOTIFICATION_SERVICE) as NotificationManager
notificationManager?.createNotificationChannel(channel)
}
@RequiresApi(Build.VERSION_CODES.O)
private fun showDatePickerDialog(type:String) {
val calendar = Calendar.getInstance()
val year = calendar.get(Calendar.YEAR)
val month = calendar.get(Calendar.MONTH)
val day = calendar.get(Calendar.DAY_OF_MONTH)
var dateTextView: AppCompatTextView? = null
if(type == "End"){
dateTextView = binding?.DateEndTask as AppCompatTextView
}
else if(type == "Start"){
dateTextView = binding?.DateStartTask as AppCompatTextView
}
if(dateTextView != null) {
val datePickerDialog = DatePickerDialog(
context as FragmentActivity,
{ _, selectedYear, selectedMonth, selectedDay ->
val selectedDate = LocalDate.of(selectedYear, selectedMonth + 1, selectedDay)
val formattedDate = selectedDate.format(dateFormatter)
dateTextView.setText(formattedDate)
},
year,
month,
day
)
datePickerDialog.show()
}
// { _, selectedYear, selectedMonth, selectedDay ->
// val selectedDate = "$selectedYear-${selectedMonth + 1}-$selectedDay"
// dateTextView.setText(selectedDate)
// },
}
fun spinerProcessing(){
val taskTypeSpinner = binding?.taskTypeSpinner
val taskTypes = resources.getStringArray(R.array.type)
val adapter = ArrayAdapter(context as FragmentActivity, android.R.layout.simple_spinner_item, taskTypes)
adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item)
taskTypeSpinner?.adapter = adapter
if (firstStart && typeTask != null){
val position = adapter.getPosition(typeTask)
taskTypeSpinner?.setSelection(position)
firstStart = false
}
taskTypeSpinner?.onItemSelectedListener = object : AdapterView.OnItemSelectedListener{
override fun onItemSelected(
parent: AdapterView<*>?,
view: View?,
position: Int,
id: Long
) {
typeTask = parent?.getItemAtPosition(position).toString()
}
override fun onNothingSelected(parent: AdapterView<*>?) {
TODO("Not yet implemented")
}
}
}
override fun onClick(view: View) {
if(binding?.checkBoxCompleted?.isChecked == true){
complet = "true"
}
else{
complet = ""
}
if(binding?.editNameTask?.text?.toString()!!.isNotEmpty() and
email.toString().isNotEmpty() and typeTask.toString().isNotEmpty() and
binding?.editInfoTask?.text?.toString()!!.isNotEmpty() and
binding?.DateStartTask?.text?.toString()!!.isNotEmpty() and
binding?.DateEndTask?.text?.toString()!!.isNotEmpty()) {
if (nameFormParent == "TaskAll" && taskAction != "Edit") {
taskViewModel?.startInsert(binding?.editNameTask?.text?.toString()!!,
email.toString(), typeTask.toString(),binding?.editInfoTask?.text?.toString()!!,binding?.DateStartTask?.text?.toString()!!,
binding?.DateEndTask?.text?.toString()!!, complet.toString())
dismiss()
(context as FragmentActivity).supportFragmentManager.beginTransaction().replace(R.id.content, TaskAll()).commit()
}
else if(nameFormParent == "TaskAll" && taskAction == "Edit"){
if(idTask != null) {
taskViewModel?.startUpdateTask(
idTask?.toInt()!!,
binding?.editNameTask?.text?.toString()!!,
email.toString(),
typeTask.toString(),
binding?.editInfoTask?.text?.toString()!!,
binding?.DateStartTask?.text?.toString()!!,
binding?.DateEndTask?.text?.toString()!!,
complet.toString()
)
}
dismiss()
(context as FragmentActivity).supportFragmentManager.beginTransaction().replace(R.id.content, TaskAll()).commit()
}
else if(nameFormParent == "TaskForType" && taskAction == "Edit"){
if(idTask != null) {
taskViewModel?.startUpdateTask(
idTask?.toInt()!!,
binding?.editNameTask?.text?.toString()!!,
email.toString(),
typeTask.toString(),
binding?.editInfoTask?.text?.toString()!!,
binding?.DateStartTask?.text?.toString()!!,
binding?.DateEndTask?.text?.toString()!!,
complet.toString()
)
}
dismiss()
(context as FragmentActivity).supportFragmentManager.beginTransaction().replace(R.id.content, TaskForType()).commit()
}
else if (nameFormParent == "TaskForType" && taskAction != "Edit") {
taskViewModel?.startInsert(binding?.editNameTask?.text?.toString()!!,
email.toString(), typeTask.toString(),binding?.editInfoTask?.text?.toString()!!,binding?.DateStartTask?.text?.toString()!!,
binding?.DateEndTask?.text?.toString()!!, complet.toString())
dismiss()
(context as FragmentActivity).supportFragmentManager.beginTransaction().replace(R.id.content, TaskForType()).commit()
}
// scheduleNotification()
}
else
Toast.makeText(context, getString(R.string.error), Toast.LENGTH_SHORT).show()
}
@RequiresApi(Build.VERSION_CODES.M)
private fun scheduleNotification() {
Log.d("MyLog", "scheduleNotification: ")
val intent = Intent(requireContext(), Notification::class.java)
val title = binding?.editNameTask?.text.toString()
val message = getString(R.string.close_task)
intent.putExtra(titleExtra, title)
intent.putExtra(messageExtra, message)
intent.putExtra("activity", MainActivity::class.java)
val pendingIntent = PendingIntent.getBroadcast(
requireContext(),
notificationID,
intent,
PendingIntent.FLAG_IMMUTABLE or PendingIntent.FLAG_UPDATE_CURRENT
)
val alarmManager = requireContext().getSystemService(Context.ALARM_SERVICE) as AlarmManager
val time = getTime()
val result_SCHEDULE = requireContext().checkCallingOrSelfPermission("android.permission.SCHEDULE_EXACT_ALARM")
Log.d("MyLog", "test SCHEDULE_EXACT_ALARM, premission $result_SCHEDULE ")
val result_POST = requireContext().checkCallingOrSelfPermission("android.permission.POST_NOTIFICATIONS")
Log.d("MyLog", "test result_POST, premission $result_POST ")
//alarmManager.set(AlarmManager.RTC_WAKEUP, time, pendingIntent)
alarmManager.setExactAndAllowWhileIdle(
AlarmManager.RTC_WAKEUP,
time,
pendingIntent
)
showAlert(time, title, message)
}
private fun showAlert(time: Long, title: String, message: String) {
val date = Date(time)
val dateFormat = android.text.format.DateFormat.getLongDateFormat(requireContext())
val timeFormat = android.text.format.DateFormat.getTimeFormat(requireContext())
AlertDialog.Builder(context as FragmentActivity)
.setTitle("Notification Scheuled")
.setMessage(
"Title: " + title +
"\nMessage: " + message +
"\nAt: " + dateFormat.format(date) + " " + timeFormat.format(date)
)
.setPositiveButton("Okay"){_,_ ->}
.show()
}
private fun getTime(): Long {
val minute = binding?.timePicker?.minute
val hour = binding?.timePicker?.hour
val day = binding?.datePicker?.dayOfMonth
val month = binding?.datePicker?.month
val year = binding?.datePicker?.year
val calendar = Calendar.getInstance()
calendar.set(year!!, month!!,day!!,hour!!,minute!!)
return calendar.timeInMillis
}
}
Android清单:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
package="com.example.myuiroom">
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"/>
<uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="android.permission.POST_NOTIFICATIONS"/>
<application
android:allowBackup="true"
android:dataExtractionRules="@xml/data_extraction_rules"
android:fullBackupContent="@xml/backup_rules"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.MyUIRoom"
tools:targetApi="31"
android:usesCleartextTraffic="true">
<receiver
android:name=".notices.Notification"/>
<activity
android:name=".MainActivity"
android:screenOrientation="portrait"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
我已经在手机中设置了此应用程序的权限,请帮助,问题可能出在哪里?
我所做的一切: 我简化了代码,只需在按下按钮时调用通知,并删除了传递的活动
private fun testNotification() {
Log.d("MyLog", "scheduleNotification: ")
val intent = Intent(requireContext(), Notification::class.java)
val title = binding?.editNameTask?.text.toString()
val message = getString(R.string.close_task)
intent.putExtra(titleExtra, title)
intent.putExtra(messageExtra, message)
requireContext().sendBroadcast(intent)
}
我在 BroadcastReceiver 本身中写了我的片段的调用,但仍然在日志中没有出现一个条目
line Log.d("MyLog", "Notification: onReceive ").
const val notificationID = 0
const val channelID = "com.example.avnotification.MyUIRoom"
const val titleExtra = "titleExtra"
const val messageExtra = "messageExtra"
class Notification : BroadcastReceiver() {
override fun onReceive(context: Context, intent: Intent) {
Log.d("MyLog", "Notification: onReceive ")
val openFragmentIntent = Intent(context, MainActivity::class.java).apply {
putExtra("openFragment", "TaskForType")
}
val pendingIntent = PendingIntent.getActivity(
context,
0,
openFragmentIntent,
PendingIntent.FLAG_UPDATE_CURRENT or PendingIntent.FLAG_IMMUTABLE
)
val notification = NotificationCompat.Builder(context, channelID)
.setSmallIcon(R.drawable.ic_launcher_foreground)
.setContentTitle(intent.getStringExtra(titleExtra))
.setContentText(intent.getStringExtra(messageExtra))
.setPriority(NotificationCompat.PRIORITY_HIGH)
.setContentIntent(pendingIntent)
.setAutoCancel(true)
.build()
val manager = context.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
manager.notify(notificationID, notification)
}
}
答:
0赞
theJango
11/17/2023
#1
很难用这么多代码来回答,也没有明确的解释什么是有效的,什么是无效的。让我们试一试。
代码的几个问题: 您无法通过 Intent 将 Activity 作为一个整体传递
intent.putExtra("activity", MainActivity::class.java)
并检索它。
val activityToOpen = intent.getSerializableExtra("activity") as Class<*>
我假设名称为 MainActivity 的类是 Android 合法活动。
另一个是在注册广播接收器时,您还需要添加 intent-action,否则您的广播接收器将无法工作。
<receiver android:name=".BroadcastReceiver" android:exported="false">
<intent-filter>
<action android:name="YOUR_OWN_INTENT_ACTION" />
</intent-filter>
</receiver>
您的通知代码对我来说似乎很好,请确保您的广播接收器工作正常,如果工作正常,则应显示通知。
评论
0赞
Dmitrii Volkov
11/17/2023
感谢您的评论,是的,问题显然是我的 BroadcastReceiver 无法正常工作,我简化了代码,删除了不正确的活动传输:私人乐趣 testNotification() { Log.d(“MyLog”, “scheduleNotification: ”) val intent = Intent(requireContext(), Notification::class.java) val title = binding?.editNameTask?。text.toString() val message = getString(R.string.close_task) intent.putExtra(titleExtra, title) intent.putExtra(messageExtra, message) requireContext().sendBroadcast(intent) }
0赞
Dmitrii Volkov
11/17/2023
我会在问题中添加对我所做的事情的描述
0赞
theJango
11/17/2023
@DmitriiVolkov 我用清单文件中关于广播接收器的声明示例更新了答案。您也需要这样做才能使广播正常工作。
0赞
Dmitrii Volkov
11/17/2023
这就是我添加清单文件的方式:<receiver android:name=“.notices。通知“android:enabled=”true“ android:exported=”false“ > <intent-filter> <action android:name=”com.example.myuiroom.MY_ACTION“/> </intent-filter> </receiver>
0赞
Dmitrii Volkov
11/17/2023
在创建通知时,但仍然没有结果。Log.d(“MyLog”, “scheduleNotification: ”) val intent = Intent(context, Notification::class.java) intent.setAction(“com.example.myuiroom.MY_ACTION”) val title = binding?.editNameTask?。text.toString() val message = getString(R.string.close_task) intent.putExtra(titleExtra, title) intent.putExtra(messageExtra, message)
0赞
Dmitrii Volkov
11/18/2023
#2
找到了答案。代码中的一切都很好,我之前为 BroadcastReceiver 创建了一个 Notification 类。在标准库中,有相同的类,而不是我的类,我导入了它“导入android.app.Notification”,而不是我的类,我导入了它“import com.example.myuiroom.Notification”。一切都立即奏效了。
评论