- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
一旦我们在 android 10 后台启动 Activity 受限中收到 FCM 推送通知消息。当我们在另一个应用程序中时,需要像 WhatsApp 和 Skype 通知来电这样的解决方案。
int NOTIFICATIONID = 1234;
// Uri sound = RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION);
Uri sound = Uri.parse("android.resource://" + getPackageName() + "/" + R.raw.capv_callingtone);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
NotificationManager notificationManager =
(NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
AudioAttributes audioAttributes = new AudioAttributes.Builder()
.setContentType(AudioAttributes.CONTENT_TYPE_SONIFICATION)
.setUsage(AudioAttributes.USAGE_NOTIFICATION)
.build();
String CHANNEL_ID = BuildConfig.APPLICATION_ID.concat("_notification_id");
String CHANNEL_NAME = BuildConfig.APPLICATION_ID.concat("_notification_name");
assert notificationManager != null;
NotificationChannel mChannel = notificationManager.getNotificationChannel(CHANNEL_ID);
if (mChannel == null) {
mChannel = new NotificationChannel(CHANNEL_ID, CHANNEL_NAME, NotificationManager.IMPORTANCE_HIGH);
mChannel.setSound(sound, audioAttributes);
notificationManager.createNotificationChannel(mChannel);
}
in.setClass(CapVFirebaseMessagingService.this, DashBoardActivity.class);
in.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
in.putExtra(NOTIFICATION_ID, NOTIFICATIONID);
Intent buttonIntent = new Intent(getBaseContext(), NotificationReceiver.class);
buttonIntent.putExtra(NOTIFICATION_ID, NOTIFICATIONID);
buttonIntent.putExtra(CapV.MESSAGE_TYPE,in.getSerializableExtra(CapV.MESSAGE_TYPE));
Log.d("Audiotask",""+in.getSerializableExtra(CapV.MESSAGE_TYPE));
PendingIntent dismissIntent = PendingIntent.getBroadcast(getBaseContext(), 0, buttonIntent, 0);
SharedPreferences localPrefs = getSharedPreferences(LOCAL_PREFERENCES,MODE_PRIVATE);
SharedPreferences.Editor editor = getSharedPreferences(LOCAL_PREFERENCES, MODE_PRIVATE).edit();
editor.putBoolean("Fragment_created",true).commit();
editor.putBoolean("Incoming_call",true).commit();
// The PendingIntent to launch activity.
PendingIntent activityPendingIntent = PendingIntent.getActivity(this, 0,
in, 0);
NotificationCompat.Builder builder = new NotificationCompat.Builder(this, CHANNEL_ID);
builder.setSmallIcon(R.drawable.logo)
.setContentTitle(("Incoming Call"))
.setContentText("Group")
.setDefaults(0)
.addAction(R.drawable.answer, getString(R.string.answer),
activityPendingIntent)
.addAction(R.drawable.reject, getString(R.string.reject),
dismissIntent)
.setPriority(NotificationCompat.PRIORITY_MAX)
.setCategory(NotificationCompat.CATEGORY_CALL)
.setVisibility(NotificationCompat.VISIBILITY_PUBLIC)
.setSound(sound)
.setOngoing(true);
android.app.Notification notification = builder.build();
notificationManager.notify(1234, notification);
private android.app.Notification getNotification(Intent in) {
in.putExtra(EXTRA_STARTED_FROM_NOTIFICATION, true);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0,in, PendingIntent.FLAG_UPDATE_CURRENT);
NotificationCompat.Builder builder = new NotificationCompat.Builder(this,"")
.setSmallIcon(R.drawable.logo)
.setAutoCancel(true)
.setPriority(NotificationCompat.PRIORITY_HIGH)
.setFullScreenIntent(pendingIntent, true);
// Set the Channel ID for Android O.
//builder.setChannelId("115"); // Channel ID
return builder.build();
}
最佳答案
我为此找到了解决方案。您需要为此创建一个前台服务。
这也将帮助您在锁定屏幕上显示振铃拨号器。在这段代码中,我添加了铃声和振动。
调用通知服务
public class HeadsUpNotificationService extends Service implements MediaPlayer.OnPreparedListener {
private String CHANNEL_ID = AppController.getInstance().getContext().getString(R.string.app_name)+"CallChannel";
private String CHANNEL_NAME = AppController.getInstance().getContext().getString(R.string.app_name)+"Call Channel";MediaPlayer mediaPlayer;
Vibrator mvibrator;
AudioManager audioManager;
AudioAttributes playbackAttributes;
private Handler handler;
AudioManager.OnAudioFocusChangeListener afChangeListener;
private boolean status = false;
private boolean vstatus = false;
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Bundle data = null;
String name="",callType="";
int NOTIFICATION_ID=120;try {
audioManager = (AudioManager)getSystemService(Context.AUDIO_SERVICE);
if (audioManager != null) {
switch (audioManager.getRingerMode()) {
case AudioManager.RINGER_MODE_NORMAL:
status = true;
break;
case AudioManager.RINGER_MODE_SILENT:
status = false;
break;
case AudioManager.RINGER_MODE_VIBRATE:
status = false;
vstatus=true;
Log.e("Service!!", "vibrate mode");
break;
}
}
if (status) {
Runnable delayedStopRunnable = new Runnable() {
@Override
public void run() {
releaseMediaPlayer();
}
};
afChangeListener = new AudioManager.OnAudioFocusChangeListener() {
public void onAudioFocusChange(int focusChange) {
if (focusChange == AudioManager.AUDIOFOCUS_LOSS) {
// Permanent loss of audio focus
// Pause playback immediately
//mediaController.getTransportControls().pause();
if (mediaPlayer!=null) {
if (mediaPlayer.isPlaying()) {
mediaPlayer.pause();
}
}
// Wait 30 seconds before stopping playback
handler.postDelayed(delayedStopRunnable,
TimeUnit.SECONDS.toMillis(30));
}
else if (focusChange == AudioManager.AUDIOFOCUS_LOSS_TRANSIENT) {
// Pause playback
} else if (focusChange == AudioManager.AUDIOFOCUS_LOSS_TRANSIENT_CAN_DUCK) {
// Lower the volume, keep playing
} else if (focusChange == AudioManager.AUDIOFOCUS_GAIN) {
// Your app has been granted audio focus again
// Raise volume to normal, restart playback if necessary
}
}
};
KeyguardManager keyguardManager = (KeyguardManager) getSystemService(Context.KEYGUARD_SERVICE);
mediaPlayer= MediaPlayer.create(this, Settings.System.DEFAULT_RINGTONE_URI);
mediaPlayer.setLooping(true);
//mediaPlayer.setAudioStreamType(AudioManager.STREAM_VOICE_CALL);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
handler = new Handler();
playbackAttributes = new AudioAttributes.Builder()
.setUsage(AudioAttributes.USAGE_ASSISTANCE_SONIFICATION)
.setContentType(AudioAttributes.CONTENT_TYPE_MUSIC)
.build();
AudioFocusRequest focusRequest = new AudioFocusRequest.Builder(AudioManager.AUDIOFOCUS_GAIN_TRANSIENT)
.setAudioAttributes(playbackAttributes)
.setAcceptsDelayedFocusGain(true)
.setOnAudioFocusChangeListener(afChangeListener, handler)
.build();
int res = audioManager.requestAudioFocus(focusRequest);
if (res == AudioManager.AUDIOFOCUS_REQUEST_GRANTED) {
if(!keyguardManager.isDeviceLocked()) {
mediaPlayer.start();
}
}
}else {
// Request audio focus for playback
int result = audioManager.requestAudioFocus(afChangeListener,
// Use the music stream.
AudioManager.STREAM_MUSIC,
// Request permanent focus.
AudioManager.AUDIOFOCUS_GAIN_TRANSIENT);
if (result == AudioManager.AUDIOFOCUS_REQUEST_GRANTED) {
if(!keyguardManager.isDeviceLocked()) {
// Start playback
mediaPlayer.start();
}
}
}
}
else if(vstatus){
mvibrator = (Vibrator) getSystemService(Context.VIBRATOR_SERVICE);
// Start without a delay
// Each element then alternates between vibrate, sleep, vibrate, sleep...
long[] pattern = {0, 250, 200, 250, 150, 150, 75,
150, 75, 150};
// The '-1' here means to vibrate once, as '-1' is out of bounds in the pattern array
mvibrator.vibrate(pattern,0);
Log.e("Service!!", "vibrate mode start");
}
} catch (Exception e) {
e.printStackTrace();
}
if (intent != null && intent.getExtras() != null) {
data = intent.getExtras();
name =data.getString("inititator");
if(AppController.getInstance().getCall_type().equalsIgnoreCase(ApplicationRef.Constants.AUDIO_CALL)){
callType ="Audio";
}
else {
callType ="Video";
}
}
try {
Intent receiveCallAction = new Intent(AppController.getInstance().getContext(), CallNotificationActionReceiver.class);
receiveCallAction.putExtra("ConstantApp.CALL_RESPONSE_ACTION_KEY", "ConstantApp.CALL_RECEIVE_ACTION");
receiveCallAction.putExtra("ACTION_TYPE", "RECEIVE_CALL");
receiveCallAction.putExtra("NOTIFICATION_ID",NOTIFICATION_ID);
receiveCallAction.setAction("RECEIVE_CALL");
Intent cancelCallAction = new Intent(AppController.getInstance().getContext(), CallNotificationActionReceiver.class);
cancelCallAction.putExtra("ConstantApp.CALL_RESPONSE_ACTION_KEY", "ConstantApp.CALL_CANCEL_ACTION");
cancelCallAction.putExtra("ACTION_TYPE", "CANCEL_CALL");
cancelCallAction.putExtra("NOTIFICATION_ID",NOTIFICATION_ID);
cancelCallAction.setAction("CANCEL_CALL");
Intent callDialogAction = new Intent(AppController.getInstance().getContext(), CallNotificationActionReceiver.class);
callDialogAction.putExtra("ACTION_TYPE", "DIALOG_CALL");
callDialogAction.putExtra("NOTIFICATION_ID",NOTIFICATION_ID);
callDialogAction.setAction("DIALOG_CALL");
PendingIntent receiveCallPendingIntent = PendingIntent.getBroadcast(AppController.getInstance().getContext(), 1200, receiveCallAction, PendingIntent.FLAG_UPDATE_CURRENT);
PendingIntent cancelCallPendingIntent = PendingIntent.getBroadcast(AppController.getInstance().getContext(), 1201, cancelCallAction, PendingIntent.FLAG_UPDATE_CURRENT);
PendingIntent callDialogPendingIntent = PendingIntent.getBroadcast(AppController.getInstance().getContext(), 1202, callDialogAction, PendingIntent.FLAG_UPDATE_CURRENT);
createChannel();
NotificationCompat.Builder notificationBuilder = null;
if (data != null) {
// Uri ringUri= Settings.System.DEFAULT_RINGTONE_URI;
notificationBuilder = new NotificationCompat.Builder(this, CHANNEL_ID)
.setContentTitle(name)
.setContentText("Incoming "+callType+" Call")
.setSmallIcon(R.drawable.ic_call_icon)
.setPriority(NotificationCompat.PRIORITY_MAX)
.setCategory(NotificationCompat.CATEGORY_CALL)
.addAction(R.drawable.ic_call_decline, getString(R.string.reject_call), cancelCallPendingIntent)
.addAction(R.drawable.ic_call_accept, getString(R.string.answer_call), receiveCallPendingIntent)
.setAutoCancel(true)
//.setSound(ringUri)
.setFullScreenIntent(callDialogPendingIntent, true);
}
Notification incomingCallNotification = null;
if (notificationBuilder != null) {
incomingCallNotification = notificationBuilder.build();
}
startForeground(NOTIFICATION_ID, incomingCallNotification);
} catch (Exception e) {
e.printStackTrace();
}
return START_STICKY;
}
@Override
public void onDestroy() {
super.onDestroy();// release your media player here audioManager.abandonAudioFocus(afChangeListener);
releaseMediaPlayer();
releaseVibration();
}
public void createChannel() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
try {
Uri ringUri= Settings.System.DEFAULT_RINGTONE_URI;
NotificationChannel channel = new NotificationChannel(CHANNEL_ID, CHANNEL_NAME, NotificationManager.IMPORTANCE_HIGH);
channel.setDescription("Call Notifications");
channel.setLockscreenVisibility(Notification.VISIBILITY_PUBLIC);
/* channel.setSound(ringUri,
new AudioAttributes.Builder().setContentType(AudioAttributes.CONTENT_TYPE_SONIFICATION)
.setLegacyStreamType(AudioManager.STREAM_RING)
.setUsage(AudioAttributes.USAGE_VOICE_COMMUNICATION).build());*/
Objects.requireNonNull(AppController.getInstance().getContext().getSystemService(NotificationManager.class)).createNotificationChannel(channel);
} catch (Exception e) {
e.printStackTrace();
}
}
}public void releaseVibration(){
try {
if(mvibrator!=null){
if (mvibrator.hasVibrator()) {
mvibrator.cancel();
}
mvibrator=null;
}
} catch (Exception e) {
e.printStackTrace();
}
}
private void releaseMediaPlayer() {
try {
if (mediaPlayer != null) {
if (mediaPlayer.isPlaying()) {
mediaPlayer.stop();
mediaPlayer.reset();
mediaPlayer.release();
}
mediaPlayer = null;
}
} catch (Exception e) {
e.printStackTrace();
}
}
@Override
public void onPrepared(MediaPlayer mediaPlayer) {
}}
服务接收者
public class CallNotificationActionReceiver extends BroadcastReceiver {
Context mContext;
@Override
public void onReceive(Context context, Intent intent) {
this.mContext=context;
if (intent != null && intent.getExtras() != null) {
String action ="";
action=intent.getStringExtra("ACTION_TYPE");
if (action != null&& !action.equalsIgnoreCase("")) {
performClickAction(context, action);
}
// Close the notification after the click action is performed.
Intent iclose = new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS);
context.sendBroadcast(iclose);
context.stopService(new Intent(context, CallNotificationService.class));
}
}
private void performClickAction(Context context, String action) {
if(action.equalsIgnoreCase("RECEIVE_CALL")) {
if (checkAppPermissions()) {
Intent intentCallReceive = new Intent(mContext, VideoCallActivity.class);
intentCallReceive.putExtra("Call", "incoming");
intentCallReceive.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP);
mContext.startActivity(intentCallReceive);
}
else{
Intent intent = new Intent(AppController.getInstance().getContext(), VideoCallRingingActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP);
intent.putExtra("CallFrom","call from push");
mContext.startActivity(intent);
}
}
else if(action.equalsIgnoreCase("DIALOG_CALL")){
// show ringing activity when phone is locked
Intent intent = new Intent(AppController.getInstance().getContext(), VideoCallRingingActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP);
mContext.startActivity(intent);
}
else {
context.stopService(new Intent(context, CallNotificationService.class));
Intent it = new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS);
context.sendBroadcast(it);
}
}
private Boolean checkAppPermissions() {
return hasReadPermissions() && hasWritePermissions() && hasCameraPermissions() && hasAudioPermissions();
}
private boolean hasAudioPermissions() {
return (ContextCompat.checkSelfPermission(AppController.getInstance().getContext(), Manifest.permission.RECORD_AUDIO) == PackageManager.PERMISSION_GRANTED);
}
private boolean hasReadPermissions() {
return (ContextCompat.checkSelfPermission(AppController.getInstance().getContext(), Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED);
}
private boolean hasWritePermissions() {
return (ContextCompat.checkSelfPermission(AppController.getInstance().getContext(), Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED);
}
private boolean hasCameraPermissions() {
return (ContextCompat.checkSelfPermission(AppController.getInstance().getContext(), Manifest.permission.CAMERA) == PackageManager.PERMISSION_GRANTED);
}
}
我们需要在应用程序选项卡内的 list 中设置这两个
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.USE_FULL_SCREEN_INTENT"/>
<uses-permission android:name="android.permission.VIBRATE" />
<!-- Incoming call -->
<service android:name=".CallNotificationService"/>
<receiver
android:name=".CallNotificationActionReceiver"
android:enabled="true">
<intent-filter android:priority="999">
<action android:name="ConstantApp.CALL_RECEIVE_ACTION" />
<action android:name="ConstantApp.CALL_CANCEL_ACTION"/>
</intent-filter>
</receiver>
要启动该服务,您需要从 Firebase 通知服务中调用它。
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
Intent serviceIntent = new Intent(getApplicationContext(), CallNotificationService.class);
Bundle mBundle = new Bundle();
mBundle.putString("inititator", name);
mBundle.putString("call_type",callType);
serviceIntent.putExtras(mBundle);
ContextCompat.startForegroundService(getApplicationContext(), serviceIntent);
}
停止通知
getApplicationContext().stopService(new Intent(AppController.getInstance().getContext(), HeadsUpNotificationService.class));
Intent istop = new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS);
getApplicationContext().sendBroadcast(istop);
关于当我们在另一个应用程序中时,Android 10 来电通知就像什么应用程序一样,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59861734/
我是 C 语言新手,我编写了这个 C 程序,让用户输入一年中的某一天,作为返回,程序将输出月份以及该月的哪一天。该程序运行良好,但我现在想简化该程序。我知道我需要一个循环,但我不知道如何去做。这是程序
我一直在努力找出我的代码有什么问题。这个想法是创建一个小的画图程序,并有红色、绿色、蓝色和清除按钮。我有我能想到的一切让它工作,但无法弄清楚代码有什么问题。程序打开,然后立即关闭。 import ja
我想安装screen,但是接下来我应该做什么? $ brew search screen imgur-screenshot screen
我有一个在服务器端工作的 UDP 套接字应用程序。为了测试服务器端,我编写了一个简单的 python 客户端程序,它发送消息“hello world how are you”。服务器随后应接收消息,将
我有一个 shell 脚本,它运行一个 Python 程序来预处理一些数据,然后运行一个 R 程序来执行一些长时间运行的任务。我正在学习使用 Docker 并且我一直在运行 FROM r-base:l
在 Linux 中。我有一个 c 程序,它读取一个 2048 字节的文本文件作为输入。我想从 Python 脚本启动 c 程序。我希望 Python 脚本将文本字符串作为参数传递给 c 程序,而不是将
对于一个类,我被要求编写一个 VHDL 程序,该程序接受两个整数输入 A 和 B,并用 A+B 替换 A,用 A-B 替换 B。我编写了以下程序和测试平台。它完成了实现和行为语法检查,但它不会模拟。尽
module Algorithm where import System.Random import Data.Maybe import Data.List type Atom = String ty
我想找到两个以上数字的最小公倍数 求给定N个数的最小公倍数的C++程序 最佳答案 int lcm(int a, int b) { return (a/gcd(a,b))*b; } 对于gcd,请查看
这个程序有错误。谁能解决这个问题? Error is :TempRecord already defines a member called 'this' with the same paramete
当我运行下面的程序时,我在 str1 和 str2 中得到了垃圾值。所以 #include #include #include using namespace std; int main() {
这是我的作业: 一对刚出生的兔子(一公一母)被放在田里。兔子在一个月大时可以交配,因此在第二个月的月底,每对兔子都会生出两对新兔子,然后死去。 注:在第0个月,有0对兔子。第 1 个月,有 1 对兔子
我编写了一个程序,通过对字母使用 switch 命令将十进制字符串转换为十六进制,但是如果我使用 char,该程序无法正常工作!没有 switch 我无法处理 9 以上的数字。我希望你能理解我,因为我
我是 C++ 新手(虽然我有一些 C 语言经验)和 MySQL,我正在尝试制作一个从 MySQL 读取数据库的程序,我一直在关注这个 tutorial但当我尝试“构建”解决方案时出现错误。 (我正在使
仍然是一个初学者,只是尝试使用 swift 中的一些基本函数。 有人能告诉我这段代码有什么问题吗? import UIKit var guessInt: Int var randomNum = arc
我正在用 C++11 编写一个函数,它采用 constant1 + constant2 形式的表达式并将它们折叠起来。 constant1 和 constant2 存储在 std::string 中,
我用 C++ 编写了这段代码,使用运算符重载对 2 个矩阵进行加法和乘法运算。当我执行代码时,它会在第 57 行和第 59 行产生错误,非法结构操作(两行都出现相同的错误)。请解释我的错误。提前致谢:
我是 C++ 的初学者,我想编写一个简单的程序来交换字符串中的两个字符。 例如;我们输入这个字符串:“EXAMPLE”,我们给它交换这两个字符:“E”和“A”,输出应该类似于“AXEMPLA”。 我在
我需要以下代码的帮助: 声明 3 个 double 类型变量,每个代表三角形的三个边中的一个。 提示用户为第一面输入一个值,然后 将用户的输入设置为您创建的代表三角形第一条边的变量。 将最后 2 个步
我是新来的,如果问题不好请见谅 任务:将给定矩阵旋转180度 输入: 1 4 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 输出: 16 15 14 13 12 11
我是一名优秀的程序员,十分优秀!