Swiftgram/submodules/TelegramCallsUI/Sources/PresentationCallToneData.swift
2019-06-16 00:51:10 +01:00

103 lines
3.1 KiB
Swift

import Foundation
import AVFoundation
private func loadToneData(name: String) -> Data? {
let outputSettings: [String: Any] = [
AVFormatIDKey: kAudioFormatLinearPCM as NSNumber,
AVSampleRateKey: 44100.0 as NSNumber,
AVLinearPCMBitDepthKey: 16 as NSNumber,
AVLinearPCMIsNonInterleaved: false as NSNumber,
AVLinearPCMIsFloatKey: false as NSNumber,
AVLinearPCMIsBigEndianKey: false as NSNumber,
AVNumberOfChannelsKey: 2 as NSNumber
]
let nsName: NSString = name as NSString
let baseName: String
let nameExtension: String
let pathExtension = nsName.pathExtension
if pathExtension.isEmpty {
baseName = name
nameExtension = "caf"
} else {
baseName = nsName.substring(with: NSRange(location: 0, length: (name.count - pathExtension.count - 1)))
nameExtension = pathExtension
}
guard let url = Bundle.main.url(forResource: baseName, withExtension: nameExtension) else {
return nil
}
let asset = AVURLAsset(url: url)
guard let assetReader = try? AVAssetReader(asset: asset) else {
return nil
}
let readerOutput = AVAssetReaderAudioMixOutput(audioTracks: asset.tracks, audioSettings: outputSettings)
if !assetReader.canAdd(readerOutput) {
return nil
}
assetReader.add(readerOutput)
if !assetReader.startReading() {
return nil
}
var data = Data()
while assetReader.status == .reading {
if let nextBuffer = readerOutput.copyNextSampleBuffer() {
var abl = AudioBufferList()
var blockBuffer: CMBlockBuffer? = nil
CMSampleBufferGetAudioBufferListWithRetainedBlockBuffer(nextBuffer, bufferListSizeNeededOut: nil, bufferListOut: &abl, bufferListSize: MemoryLayout<AudioBufferList>.size, blockBufferAllocator: nil, blockBufferMemoryAllocator: nil, flags: kCMSampleBufferFlag_AudioBufferList_Assure16ByteAlignment, blockBufferOut: &blockBuffer)
let size = Int(CMSampleBufferGetTotalSampleSize(nextBuffer))
if size != 0, let mData = abl.mBuffers.mData {
data.append(Data(bytes: mData, count: size))
}
} else {
break
}
}
return data
}
enum PresentationCallTone {
case ringing
case connecting
case busy
case failed
case ended
var loopCount: Int? {
switch self {
case .busy:
return 3
case .failed:
return 1
case .ended:
return 1
default:
return nil
}
}
}
func presentationCallToneData(_ tone: PresentationCallTone) -> Data? {
switch tone {
case .ringing:
return loadToneData(name: "voip_ringback.caf")
case .connecting:
return loadToneData(name: "voip_connecting.mp3")
case .busy:
return loadToneData(name: "voip_busy.caf")
case .failed:
return loadToneData(name: "voip_fail.caf")
case .ended:
return loadToneData(name: "voip_end.caf")
}
}