mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			215 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			215 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			Swift
		
	
| import Accelerate
 | |
| import NVActivityIndicatorView
 | |
| 
 | |
| @objc(LKVoiceMessageView)
 | |
| final class VoiceMessageView : UIView {
 | |
|     private let voiceMessage: TSAttachment
 | |
|     private let isOutgoing: Bool
 | |
|     private var isLoading = false
 | |
|     private var isForcedAnimation = false
 | |
|     private var volumeSamples: [Float] = [] { didSet { updateShapeLayers() } }
 | |
|     @objc var progress: CGFloat = 0 { didSet { updateShapeLayers() } }
 | |
|     @objc var duration: Int = 0 { didSet { updateDurationLabel() } }
 | |
|     @objc var isPlaying = false { didSet { updateToggleImageView() } }
 | |
| 
 | |
|     // MARK: Components
 | |
|     private lazy var toggleImageView = UIImageView(image: #imageLiteral(resourceName: "Play"))
 | |
| 
 | |
|     private lazy var spinner = NVActivityIndicatorView(frame: CGRect.zero, type: .circleStrokeSpin, color: .black, padding: nil)
 | |
| 
 | |
|     private lazy var durationLabel: UILabel = {
 | |
|         let result = UILabel()
 | |
|         result.textColor = Colors.text
 | |
|         result.font = .systemFont(ofSize: Values.mediumFontSize)
 | |
|         return result
 | |
|     }()
 | |
| 
 | |
|     private lazy var backgroundShapeLayer: CAShapeLayer = {
 | |
|         let result = CAShapeLayer()
 | |
|         result.fillColor = Colors.text.cgColor
 | |
|         return result
 | |
|     }()
 | |
| 
 | |
|     private lazy var foregroundShapeLayer: CAShapeLayer = {
 | |
|         let result = CAShapeLayer()
 | |
|         result.fillColor = (isLightMode && isOutgoing) ? UIColor.white.cgColor : Colors.accent.cgColor
 | |
|         return result
 | |
|     }()
 | |
| 
 | |
|     // MARK: Settings
 | |
|     private let leadingInset: CGFloat = 0
 | |
|     private let sampleSpacing: CGFloat = 1
 | |
|     private let targetSampleCount = 48
 | |
|     private let toggleContainerSize: CGFloat = 32
 | |
|     private let vMargin: CGFloat = 0
 | |
| 
 | |
|     @objc public static let contentHeight: CGFloat = 40
 | |
| 
 | |
|     // MARK: Initialization
 | |
|     @objc(initWithVoiceMessage:isOutgoing:)
 | |
|     init(voiceMessage: TSAttachment, isOutgoing: Bool) {
 | |
|         self.voiceMessage = voiceMessage
 | |
|         self.isOutgoing = isOutgoing
 | |
|         super.init(frame: CGRect.zero)
 | |
|     }
 | |
| 
 | |
|     override init(frame: CGRect) {
 | |
|         preconditionFailure("Use init(voiceMessage:associatedWith:) instead.")
 | |
|     }
 | |
| 
 | |
|     required init?(coder: NSCoder) {
 | |
|         preconditionFailure("Use init(voiceMessage:associatedWith:) instead.")
 | |
|     }
 | |
| 
 | |
|     @objc func initialize() {
 | |
|         setUpViewHierarchy()
 | |
|         if voiceMessage.isDownloaded {
 | |
|             guard let url = (voiceMessage as? TSAttachmentStream)?.originalMediaURL else {
 | |
|                 return SNLog("Couldn't get URL for voice message.")
 | |
|             }
 | |
|             if let cachedVolumeSamples = Storage.shared.getVolumeSamples(for: voiceMessage.uniqueId!), cachedVolumeSamples.count == targetSampleCount {
 | |
|                 self.hideLoader()
 | |
|                 self.volumeSamples = cachedVolumeSamples
 | |
|             } else {
 | |
|                 let voiceMessageID = voiceMessage.uniqueId!
 | |
|                 AudioUtilities.getVolumeSamples(for: url, targetSampleCount: targetSampleCount).done(on: DispatchQueue.main) { [weak self] volumeSamples in
 | |
|                     guard let self = self else { return }
 | |
|                     self.hideLoader()
 | |
|                     self.isForcedAnimation = true
 | |
|                     self.volumeSamples = volumeSamples
 | |
|                     Storage.write { transaction in
 | |
|                         Storage.shared.setVolumeSamples(for: voiceMessageID, to: volumeSamples, using: transaction)
 | |
|                     }
 | |
|                 }.catch(on: DispatchQueue.main) { error in
 | |
|                     SNLog("Couldn't sample audio file due to error: \(error).")
 | |
|                 }
 | |
|             }
 | |
|         } else {
 | |
|             showLoader()
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private func setUpViewHierarchy() {
 | |
|         set(.width, to: 200)
 | |
|         set(.height, to: VoiceMessageView.contentHeight)
 | |
|         layer.insertSublayer(backgroundShapeLayer, at: 0)
 | |
|         layer.insertSublayer(foregroundShapeLayer, at: 1)
 | |
|         let toggleContainer = UIView()
 | |
|         toggleContainer.clipsToBounds = false
 | |
|         toggleContainer.addSubview(toggleImageView)
 | |
|         toggleImageView.set(.width, to: 12)
 | |
|         toggleImageView.set(.height, to: 12)
 | |
|         toggleImageView.center(in: toggleContainer)
 | |
|         toggleContainer.addSubview(spinner)
 | |
|         spinner.set(.width, to: 24)
 | |
|         spinner.set(.height, to: 24)
 | |
|         spinner.center(in: toggleContainer)
 | |
|         toggleContainer.set(.width, to: toggleContainerSize)
 | |
|         toggleContainer.set(.height, to: toggleContainerSize)
 | |
|         toggleContainer.layer.cornerRadius = toggleContainerSize / 2
 | |
|         toggleContainer.backgroundColor = UIColor.white
 | |
|         let glowRadius: CGFloat = isLightMode ? 1 : 2
 | |
|         let glowColor = isLightMode ? UIColor.black.withAlphaComponent(0.4) : UIColor.black
 | |
|         let glowConfiguration = UIView.CircularGlowConfiguration(size: toggleContainerSize, color: glowColor, radius: glowRadius)
 | |
|         toggleContainer.setCircularGlow(with: glowConfiguration)
 | |
|         addSubview(toggleContainer)
 | |
|         toggleContainer.center(.vertical, in: self)
 | |
|         toggleContainer.pin(.leading, to: .leading, of: self, withInset: leadingInset)
 | |
|         addSubview(durationLabel)
 | |
|         durationLabel.center(.vertical, in: self)
 | |
|         durationLabel.pin(.trailing, to: .trailing, of: self)
 | |
|     }
 | |
| 
 | |
|     // MARK: UI & Updating
 | |
|     private func showLoader() {
 | |
|         isLoading = true
 | |
|         toggleImageView.isHidden = true
 | |
|         spinner.startAnimating()
 | |
|         spinner.isHidden = false
 | |
|         Timer.scheduledTimer(withTimeInterval: 0.25, repeats: true) { [weak self] timer in
 | |
|             guard let self = self else { return timer.invalidate() }
 | |
|             if self.isLoading {
 | |
|                 self.updateFakeVolumeSamples()
 | |
|             } else {
 | |
|                 timer.invalidate()
 | |
|             }
 | |
|         }
 | |
|         updateFakeVolumeSamples()
 | |
|     }
 | |
| 
 | |
|     private func updateFakeVolumeSamples() {
 | |
|         let fakeVolumeSamples = (0..<targetSampleCount).map { _ in Float.random(in: 0...1) }
 | |
|         volumeSamples = fakeVolumeSamples
 | |
|     }
 | |
| 
 | |
|     private func hideLoader() {
 | |
|         isLoading = false
 | |
|         toggleImageView.isHidden = false
 | |
|         spinner.stopAnimating()
 | |
|         spinner.isHidden = true
 | |
|     }
 | |
| 
 | |
|     override func layoutSubviews() {
 | |
|         super.layoutSubviews()
 | |
|         updateShapeLayers()
 | |
|     }
 | |
| 
 | |
|     private func updateShapeLayers() {
 | |
|         clipsToBounds = false // Bit of a hack to do this here, but the containing stack view turns this off all the time
 | |
|         guard !volumeSamples.isEmpty else { return }
 | |
|         let sMin = CGFloat(volumeSamples.min()!)
 | |
|         let sMax = CGFloat(volumeSamples.max()!)
 | |
|         let w = width() - leadingInset - toggleContainerSize - durationLabel.width() - 2 * Values.smallSpacing
 | |
|         let h = height() - 2 * vMargin
 | |
|         let sW = (w - sampleSpacing * CGFloat(volumeSamples.count - 1)) / CGFloat(volumeSamples.count)
 | |
|         let backgroundPath = UIBezierPath()
 | |
|         let foregroundPath = UIBezierPath()
 | |
|         for (i, value) in volumeSamples.enumerated() {
 | |
|             let x = leadingInset + toggleContainerSize + Values.smallSpacing + CGFloat(i) * (sW + sampleSpacing)
 | |
|             let fraction = (CGFloat(value) - sMin) / (sMax - sMin)
 | |
|             let sH = max(8, h * fraction)
 | |
|             let y = vMargin + (h - sH) / 2
 | |
|             let subPath = UIBezierPath(roundedRect: CGRect(x: x, y: y, width: sW, height: sH), cornerRadius: sW / 2)
 | |
|             backgroundPath.append(subPath)
 | |
|             if progress > CGFloat(i) / CGFloat(volumeSamples.count) { foregroundPath.append(subPath) }
 | |
|         }
 | |
|         backgroundPath.close()
 | |
|         foregroundPath.close()
 | |
|         if isLoading || isForcedAnimation {
 | |
|             let animation = CABasicAnimation(keyPath: "path")
 | |
|             animation.duration = 0.25
 | |
|             animation.toValue = backgroundPath
 | |
|             animation.timingFunction = CAMediaTimingFunction(name: CAMediaTimingFunctionName.easeInEaseOut)
 | |
|             backgroundShapeLayer.add(animation, forKey: "path")
 | |
|             backgroundShapeLayer.path = backgroundPath.cgPath
 | |
|         } else {
 | |
|             backgroundShapeLayer.path = backgroundPath.cgPath
 | |
|         }
 | |
|         foregroundShapeLayer.path = foregroundPath.cgPath
 | |
|         isForcedAnimation = false
 | |
|     }
 | |
| 
 | |
|     private func updateDurationLabel() {
 | |
|         durationLabel.text = OWSFormat.formatDurationSeconds(duration)
 | |
|         updateShapeLayers()
 | |
|     }
 | |
| 
 | |
|     private func updateToggleImageView() {
 | |
|         toggleImageView.image = isPlaying ? #imageLiteral(resourceName: "Pause") : #imageLiteral(resourceName: "Play")
 | |
|     }
 | |
| 
 | |
|     // MARK: Interaction
 | |
|     @objc(getCurrentTime:)
 | |
|     func getCurrentTime(for panGestureRecognizer: UIPanGestureRecognizer) -> TimeInterval {
 | |
|         guard voiceMessage.isDownloaded else { return 0 }
 | |
|         let locationInSelf = panGestureRecognizer.location(in: self)
 | |
|         let waveformFrameOrigin = CGPoint(x: leadingInset + toggleContainerSize + Values.smallSpacing, y: vMargin)
 | |
|         let waveformFrameSize = CGSize(width: width() - leadingInset - toggleContainerSize - durationLabel.width() - 2 * Values.smallSpacing,
 | |
|             height: height() - 2 * vMargin)
 | |
|         let waveformFrame = CGRect(origin: waveformFrameOrigin, size: waveformFrameSize)
 | |
|         guard waveformFrame.contains(locationInSelf) else { return 0 }
 | |
|         let fraction = (locationInSelf.x - waveformFrame.minX) / (waveformFrame.maxX - waveformFrame.minX)
 | |
|         return Double(fraction) * Double(duration)
 | |
|     }
 | |
| }
 |