当来电和app在前台时,我正在做一个VOIP应用程序。我用下面的代码播放铃声
self.ringTimer = [[TimerService sharedInstance] createIntervalTimerWithInterval:2500 queue:dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0) andBlock:^{
AudioServicesPlayAlertSound(kSystemSoundID_Vibrate);
}];
NSString *ringingPath = [[NSBundle mainBundle] pathForResource:@"incoming_sound" ofType:@"mp3"];
AVAudioSession *audioSession = [AVAudioSession sharedInstance];
[audioSession setCategory:AVAudioSessionCategoryAmbient withOptions:AVAudioSessionCategoryOptionAllowBluetooth | AVAudioSessionCategoryOptionMixWithOthers error:nil];
// for normal phone receiver route it to speaker
if ([[audioSession audioRoute] isEqualToString:kAudioSessionDevice_Phone]) {
[audioSession overrideOutputAudioPort:AVAudioSessionPortOverrideSpeaker error:nil];
}
NSURL *soundUrl = [NSURL fileURLWithPath:ringingPath];
NSError *error;
self.audioPlayer = [[AVAudioPlayer alloc] initWithContentsOfURL:soundUrl error:&error];
self.audioPlayer.numberOfLoops = -1;
[self.audioPlayer play];
以上代码效果很好,在播放声音时也会尊重静音开关按钮。
但是,如果我最小化应用程序,设备将停止振动并播放传入的声音。如果我将AVAudioSession
类别设置为AVAudioSessionCategoryPlayback
,则可以解决这个问题,但是当静音开关打开时,此类别不会使声音静音。
我看到其他应用
(在应用程序中接收传入)他们会显示一个很长的通知以保持设备振动。如果我驳回该通知,它将立即停止。
(在应用关闭时接收来电)如果我关闭通知,设备会再振动2-3次并显示长时间通知(与在应用中接收传入时相同)
(在应用程序中接收传入)我不确定skype,但是对于viber最小化应用程序时它会立即停止振动。
(在应用关闭时收到来电)我认为他们的做法和Facebook一样(不确定)
在最小化应用时,如何在没有Line
通知的情况下让设备振动?
如果我设法找到一种方法并让它像Line
那样工作,我是否需要在应用程序不在前台时使用Voip push来实现该方法? (正常推送通知会起作用吗?)
答案 0 :(得分:2)
您必须使用pushkit(无声推送通知)。
适用于背景或杀死状态。
Skype,whatsapp,viber等基于VOIP的应用程序的工作原理如下。
当您获得有效负载时,您必须安排本地通知。您可以在本地通知中设置声音文件。声音文件最长播放30秒。如果您想要重复声音文件,直到最终用户不接收或拒绝呼叫,您也可以使用标记管理和重新安排本地通知。
使用以下结构来完成您的任务。
使用此simplepush.php文件
<?php
// Put your device token here (without spaces):
$deviceToken = '1234567890123456789';
//
// Put your private key's passphrase here:
$passphrase = 'ProjectName';
// Put your alert message here:
$message = 'My first push notification!';
$ctx = stream_context_create();
stream_context_set_option($ctx, 'ssl', 'local_cert', 'PemFileName.pem');
stream_context_set_option($ctx, 'ssl', 'passphrase', $passphrase);
// Open a connection to the APNS server
$fp = stream_socket_client(
// 'ssl://gateway.push.apple.com:2195', $err,
'ssl://gateway.sandbox.push.apple.com:2195', $err,
$errstr, 60, STREAM_CLIENT_CONNECT|STREAM_CLIENT_PERSISTENT, $ctx);
if (!$fp)
exit("Failed to connect: $err $errstr" . PHP_EOL);
echo 'Connected to APNS' . PHP_EOL;
// Create the payload body
$body['aps'] = array(
'content-available'=> 1,
'alert' => $message,
'sound' => 'default',
'badge' => 0,
);
// Encode the payload as JSON
$payload = json_encode($body);
// Build the binary notification
$msg = chr(0) . pack('n', 32) . pack('H*', $deviceToken) . pack('n', strlen($payload)) . $payload;
// Send it to the server
$result = fwrite($fp, $msg, strlen($msg));
if (!$result)
echo 'Message not delivered' . PHP_EOL;
else
echo 'Message successfully delivered' . PHP_EOL;
// Close the connection to the server
fclose($fp);
使用以下命令创建pem文件并在上面的代码中使用它
$ openssl x509 -in aps_development.cer -inform der -out PushCert.pem
# Convert .p12 to .pem. Enter your pass pharse which is the same pwd that you have given while creating the .p12 certificate. PEM pass phrase also same as .p12 cert.
$ openssl pkcs12 -nocerts -out PushKey1.pem -in pushkey.p12
Enter Import Password:
MAC verified OK
Enter PEM pass phrase:
Verifying - Enter PEM pass phrase:
# To remove passpharse for the key to access globally. This only solved my stream_socket_client() & certificate capath warnings.
$ openssl rsa -in PushKey1.pem -out PushKey1_Rmv.pem
Enter pass phrase for PushChatKey1.pem:
writing RSA key
# To join the two .pem file into one file:
$ cat PushCert.pem PushKey1_Rmv.pem > ApnsDev.pem
之后转到simplepush.php位置和fire命令 - &gt; php simplepush.php
通过这种方式,您可以测试推送套件通知设置架构。
https://zeropush.com/guide/guide-to-pushkit-and-voip
https://www.raywenderlich.com/123862/push-notifications-tutorial
import UIKit
import PushKit
class AppDelegate: UIResponder, UIApplicationDelegate,PKPushRegistryDelegate{
func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
let types: UIRemoteNotificationType = [.Alert, .Badge, .Sound]
application.registerForRemoteNotificationTypes(types)
self. PushKitRegistration()
return true
}
//MARK: - PushKitRegistration
func PushKitRegistration()
{
let mainQueue = dispatch_get_main_queue()
// Create a push registry object
if #available(iOS 8.0, *) {
let voipRegistry: PKPushRegistry = PKPushRegistry(queue: mainQueue)
// Set the registry's delegate to self
voipRegistry.delegate = self
// Set the push type to VoIP
voipRegistry.desiredPushTypes = [PKPushTypeVoIP]
} else {
// Fallback on earlier versions
}
}
@available(iOS 8.0, *)
func pushRegistry(registry: PKPushRegistry!, didUpdatePushCredentials credentials: PKPushCredentials!, forType type: String!) {
// Register VoIP push token (a property of PKPushCredentials) with server
let hexString : String = UnsafeBufferPointer<UInt8>(start: UnsafePointer(credentials.token.bytes),
count: credentials.token.length).map { String(format: "%02x", $0) }.joinWithSeparator("")
print(hexString)
}
@available(iOS 8.0, *)
func pushRegistry(registry: PKPushRegistry!, didReceiveIncomingPushWithPayload payload: PKPushPayload!, forType type: String!) {
// Process the received push
// From here you have to schedule your local notification
}
}
并行设置Socket架构。然后Socket用于广播数据,例如哪些用户在线/离线,你需要知道没有调用API然后你可以在服务器上使用套接字,套接字架构将在线/离线用户的数据发送到设备而无需从设备请求。< / p>
如果我可以帮助你,请告诉我。
有一个快乐的编码。
答案 1 :(得分:0)
经过一些测试后,我找到了Line
这样做的方式。
接收应用中的来电
他们使用
[[UIApplication sharedApplication] beginBackgroundTaskWithName:expirationHandler:]
即使您已经将应用程序最小化,也要保持应用程序振动。此方法将使应用程序保持运行,并在到期前保持180秒的持续时间。
关闭应用时接听来电
这需要使用PushKit
作为@Hasya提及使应用程序清醒,并调用与上述方案相同的方法(保持设备振动)
[[UIApplication sharedApplication] beginBackgroundTaskWithName:expirationHandler:]
这次它将在40而不是180秒到期。此外,您无法通过此AVAudioPlayer
类别的Ambient
播放传入的声音。
所以你需要附加一个声音(需要一个扩展版本,因为没有重复功能)到UILocalNotification
,我已经读过一个你可以播放声音长达30秒的声音。
使用时遇到的一些问题PushKit
如果您的旧系统使用APNS
,则需要在服务器端维护令牌(APNS和VOIP)。因为您无法使用VOIP证书发送APNS令牌。需要使用正确的证书类型发送令牌。