mirror of
https://gitlab.opencode.de/bwi/bundesmessenger/clients/bundesmessenger-ios.git
synced 2026-04-22 01:22:46 +02:00
b298dedc22
Merge commit 'f823ab9aae70e8d15ed7cc079210dd9bbbb6c8e1' into feature/foss_update_1_11_19 * commit 'f823ab9aae70e8d15ed7cc079210dd9bbbb6c8e1': finish version++ version++ comments update submodule remove obsolete tests removed unused code update submodule fix Libolm removal update license macro update license Prepare for new sprint # Conflicts: # Config/AppVersion.xcconfig # IDETemplateMacros.plist # LICENSE # README.md # Riot/Categories/MXSession+Riot.m # Riot/Managers/EncryptionKeyManager/EncryptionKeyManager.swift # Riot/Managers/KeyValueStorage/Extensions/Keychain.swift # Riot/Managers/KeyValueStorage/KeyValueStore.swift # Riot/Managers/KeyValueStorage/KeychainStore.swift # Riot/Managers/KeyValueStorage/MemoryStore.swift # Riot/Managers/PushNotification/PushNotificationService.m # Riot/Managers/Settings/RiotSettings.swift # Riot/Managers/Settings/Shared/RiotSharedSettings.swift # Riot/Modules/Analytics/AnalyticsUIElement.swift # Riot/Modules/Application/AppCoordinator.swift # Riot/Modules/Application/LegacyAppDelegate.h # Riot/Modules/Application/LegacyAppDelegate.m # Riot/Modules/Authentication/Legacy/AuthenticationViewController.h # Riot/Modules/Authentication/Legacy/AuthenticationViewController.m # Riot/Modules/Authentication/Legacy/Views/AuthInputsView.h # Riot/Modules/Authentication/Legacy/Views/AuthInputsView.m # Riot/Modules/Common/Recents/DataSources/RecentsDataSource.m # Riot/Modules/Common/Recents/RecentsViewController.m # Riot/Modules/Common/WebViewController/WebViewViewController.m # Riot/Modules/Contacts/Details/ContactDetailsViewController.m # Riot/Modules/Contacts/Views/ContactTableViewCell.m # Riot/Modules/Favorites/FavouritesViewController.h # Riot/Modules/Favorites/FavouritesViewController.m # Riot/Modules/GlobalSearch/UnifiedSearchViewController.m # Riot/Modules/People/PeopleViewController.h # Riot/Modules/People/PeopleViewController.m # Riot/Modules/Room/ContextualMenu/ReactionsMenu/ReactionsMenuViewModel.swift # Riot/Modules/Room/DataSources/RoomDataSource.m # Riot/Modules/Room/Files/RoomFilesViewController.m # Riot/Modules/Room/Members/Detail/RoomMemberDetailsViewController.m # Riot/Modules/Room/Members/RoomParticipantsViewController.m # Riot/Modules/Room/RoomViewController.m # Riot/Modules/Room/Settings/RoomSettingsViewController.m # Riot/Modules/Room/TimelineCells/RoomCreationIntro/RoomCreationIntroCell.swift # Riot/Modules/Room/TimelineCells/RoomCreationIntro/RoomCreationIntroCellContentView.swift # Riot/Modules/Room/TimelineCells/RoomCreationIntro/RoomCreationIntroViewData.swift # Riot/Modules/Room/TimelineCells/RoomTimelineCellIdentifier.h # Riot/Modules/Rooms/RoomsViewController.h # Riot/Modules/Rooms/ShowDirectory/Cells/Network/DirectoryNetworkTableHeaderFooterView.swift # Riot/Modules/Rooms/ShowDirectory/Cells/Room/DirectoryRoomTableViewCell.swift # Riot/Modules/Rooms/ShowDirectory/PublicRoomsDirectoryViewModel.swift # Riot/Modules/Secrets/Recover/RecoverWithKey/SecretsRecoveryWithKeyCoordinator.swift # Riot/Modules/Secrets/Recover/RecoverWithKey/SecretsRecoveryWithKeyViewController.swift # Riot/Modules/Secrets/Recover/RecoverWithPassphrase/SecretsRecoveryWithPassphraseCoordinator.swift # Riot/Modules/Secrets/Recover/RecoverWithPassphrase/SecretsRecoveryWithPassphraseViewController.swift # Riot/Modules/Secrets/Recover/SecretsRecoveryCoordinator.swift # Riot/Modules/SecureBackup/Setup/Intro/SecureBackupSetupIntroViewController.swift # Riot/Modules/SecureBackup/Setup/Intro/SecureBackupSetupIntroViewModel.swift # Riot/Modules/SecureBackup/Setup/Intro/SecureBackupSetupIntroViewModelType.swift # Riot/Modules/SetPinCode/PinCodePreferences.swift # Riot/Modules/SetPinCode/SetupBiometrics/BiometricsAuthenticationPresenter.swift # Riot/Modules/Settings/Security/ManageSession/ManageSessionViewController.m # Riot/Modules/Settings/Security/SecurityViewController.m # Riot/Modules/Settings/SettingsViewController.m # Riot/Modules/SplitView/SplitViewCoordinator.swift # Riot/Modules/SplitView/SplitViewCoordinatorType.swift # Riot/Modules/StartChat/StartChatViewController.m # Riot/Modules/TabBar/MasterTabBarController.h # Riot/Modules/TabBar/MasterTabBarController.m # Riot/Utils/EventFormatter.m # Riot/Utils/HTMLFormatter.swift # Riot/Utils/Tools.m # RiotNSE/NotificationService.swift
234 lines
7.6 KiB
Swift
234 lines
7.6 KiB
Swift
//
|
|
// Copyright 2021-2024 New Vector Ltd.
|
|
//
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
// Please see LICENSE in the repository root for full details.
|
|
//
|
|
|
|
import Foundation
|
|
import DSWaveformImage
|
|
|
|
enum VoiceMessagePlaybackControllerState {
|
|
case stopped
|
|
case playing
|
|
case paused
|
|
case error
|
|
}
|
|
|
|
class VoiceMessagePlaybackController: VoiceMessageAudioPlayerDelegate, VoiceMessagePlaybackViewDelegate {
|
|
|
|
private enum Constants {
|
|
static let elapsedTimeFormat = "m:ss"
|
|
}
|
|
|
|
private let mediaServiceProvider: VoiceMessageMediaServiceProvider
|
|
private let cacheManager: VoiceMessageAttachmentCacheManager
|
|
|
|
private var audioPlayer: VoiceMessageAudioPlayer?
|
|
private var displayLink: CADisplayLink!
|
|
private var samples: [Float] = []
|
|
private var duration: TimeInterval = 0
|
|
private var urlToLoad: URL?
|
|
private var loading: Bool = false
|
|
|
|
private var state: VoiceMessagePlaybackControllerState = .stopped {
|
|
didSet {
|
|
updateUI()
|
|
displayLink.isPaused = (state != .playing)
|
|
}
|
|
}
|
|
|
|
private static let timeFormatter: DateFormatter = {
|
|
let dateFormatter = DateFormatter()
|
|
dateFormatter.dateFormat = Constants.elapsedTimeFormat
|
|
return dateFormatter
|
|
}()
|
|
|
|
|
|
let playbackView: VoiceMessagePlaybackView
|
|
|
|
init(mediaServiceProvider: VoiceMessageMediaServiceProvider, cacheManager: VoiceMessageAttachmentCacheManager) {
|
|
self.mediaServiceProvider = mediaServiceProvider
|
|
self.cacheManager = cacheManager
|
|
|
|
playbackView = VoiceMessagePlaybackView.loadFromNib()
|
|
playbackView.delegate = self
|
|
|
|
displayLink = CADisplayLink(target: WeakTarget(self, selector: #selector(handleDisplayLinkTick)), selector: WeakTarget.triggerSelector)
|
|
displayLink.isPaused = true
|
|
displayLink.add(to: .current, forMode: .common)
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(updateTheme), name: .themeServiceDidChangeTheme, object: nil)
|
|
updateTheme()
|
|
updateUI()
|
|
|
|
if BWIBuildSettings.shared.bwiResignPlayingVoiceMessageInBackground {
|
|
NotificationCenter.default.addObserver(self, selector: #selector(appMovedToBackground), name: UIApplication.willResignActiveNotification, object: nil)
|
|
|
|
}
|
|
}
|
|
|
|
var attachment: MXKAttachment? {
|
|
didSet {
|
|
loadAttachmentData()
|
|
}
|
|
}
|
|
|
|
// MARK: - VoiceMessagePlaybackViewDelegate
|
|
|
|
func voiceMessagePlaybackViewDidRequestPlaybackToggle() {
|
|
guard let audioPlayer = audioPlayer else {
|
|
return
|
|
}
|
|
|
|
if audioPlayer.url != nil {
|
|
if audioPlayer.isPlaying {
|
|
audioPlayer.pause()
|
|
} else {
|
|
audioPlayer.play()
|
|
}
|
|
} else if let url = urlToLoad {
|
|
audioPlayer.loadContentFromURL(url, displayName: attachment?.originalFileName)
|
|
audioPlayer.play()
|
|
}
|
|
}
|
|
|
|
func voiceMessagePlaybackViewDidRequestSeek(to progress: CGFloat) {
|
|
guard let audioPlayer = audioPlayer else {
|
|
return
|
|
}
|
|
|
|
if audioPlayer.url == nil,
|
|
let url = urlToLoad {
|
|
audioPlayer.loadContentFromURL(url, displayName: attachment?.originalFileName)
|
|
}
|
|
|
|
audioPlayer.seekToTime(self.duration * Double(progress)) { [weak self] _ in
|
|
guard let self = self else { return }
|
|
self.updateUI()
|
|
}
|
|
}
|
|
|
|
func voiceMessagePlaybackViewDidChangeWidth() {
|
|
loadAttachmentData()
|
|
}
|
|
|
|
// MARK: - VoiceMessageAudioPlayerDelegate
|
|
|
|
func audioPlayerDidFinishLoading(_ audioPlayer: VoiceMessageAudioPlayer) {
|
|
updateUI()
|
|
}
|
|
|
|
func audioPlayerDidStartPlaying(_ audioPlayer: VoiceMessageAudioPlayer) {
|
|
state = .playing
|
|
}
|
|
|
|
func audioPlayerDidPausePlaying(_ audioPlayer: VoiceMessageAudioPlayer) {
|
|
state = .paused
|
|
}
|
|
|
|
func audioPlayerDidStopPlaying(_ audioPlayer: VoiceMessageAudioPlayer) {
|
|
state = .stopped
|
|
}
|
|
|
|
func audioPlayer(_ audioPlayer: VoiceMessageAudioPlayer, didFailWithError error: Error) {
|
|
state = .error
|
|
MXLog.error("Failed playing voice message", context: error)
|
|
}
|
|
|
|
func audioPlayerDidFinishPlaying(_ audioPlayer: VoiceMessageAudioPlayer) {
|
|
audioPlayer.seekToTime(0.0) { [weak self] _ in
|
|
guard let self = self else { return }
|
|
self.state = .stopped
|
|
// Reload its content if necessary, otherwise the seek won't work
|
|
self.audioPlayer?.reloadContentIfNeeded()
|
|
}
|
|
}
|
|
|
|
// MARK: - Private
|
|
|
|
@objc private func handleDisplayLinkTick() {
|
|
updateUI()
|
|
}
|
|
|
|
private func updateUI() {
|
|
var details = VoiceMessagePlaybackViewDetails()
|
|
|
|
details.playbackEnabled = (state != .error)
|
|
details.playing = (state == .playing)
|
|
details.samples = samples
|
|
// Show the current time if the player is paused, show duration when at 0.
|
|
let duration = self.duration
|
|
let currentTime = audioPlayer?.currentTime ?? 0
|
|
let displayTime = currentTime > 0 ? currentTime : duration
|
|
details.currentTime = VoiceMessagePlaybackController.timeFormatter.string(from: Date(timeIntervalSinceReferenceDate: displayTime))
|
|
details.progress = duration > 0 ? currentTime / duration : 0
|
|
details.loading = self.loading
|
|
playbackView.configureWithDetails(details)
|
|
}
|
|
|
|
private func loadAttachmentData() {
|
|
guard let attachment = attachment else {
|
|
return
|
|
}
|
|
|
|
self.state = .stopped
|
|
updateUI()
|
|
|
|
let requiredNumberOfSamples = playbackView.getRequiredNumberOfSamples()
|
|
|
|
cacheManager.loadAttachment(attachment, numberOfSamples: requiredNumberOfSamples) { [weak self] result in
|
|
guard let self = self else {
|
|
return
|
|
}
|
|
|
|
switch result {
|
|
case .success(let result):
|
|
guard result.eventIdentifier == attachment.eventId else {
|
|
return
|
|
}
|
|
|
|
// Avoid listening to old audio player delegates if the attachment for this playbackController/cell changes
|
|
self.audioPlayer?.deregisterDelegate(self)
|
|
|
|
self.audioPlayer = self.mediaServiceProvider.audioPlayerForIdentifier(result.eventIdentifier)
|
|
self.audioPlayer?.registerDelegate(self)
|
|
|
|
self.loading = false
|
|
self.urlToLoad = result.url
|
|
self.duration = result.duration
|
|
self.samples = result.samples
|
|
|
|
if let audioPlayer = self.audioPlayer {
|
|
if audioPlayer.isPlaying {
|
|
self.state = .playing
|
|
} else if audioPlayer.currentTime > 0 {
|
|
self.state = .paused
|
|
} else {
|
|
self.state = .stopped
|
|
}
|
|
}
|
|
case .failure:
|
|
self.state = .error
|
|
}
|
|
}
|
|
}
|
|
|
|
@objc private func updateTheme() {
|
|
playbackView.update(theme: ThemeService.shared().theme)
|
|
}
|
|
|
|
// bwi: resign voice playing in background
|
|
@objc func appMovedToBackground() {
|
|
guard let audioPlayer = audioPlayer else {
|
|
return
|
|
}
|
|
|
|
if audioPlayer.url != nil {
|
|
if audioPlayer.isPlaying {
|
|
audioPlayer.stop()
|
|
}
|
|
}
|
|
}
|
|
}
|