2021-02-03 11:01:29 +00:00
|
|
|
//
|
|
|
|
// Copyright 2020 New Vector Ltd
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
//
|
|
|
|
|
|
|
|
import UIKit
|
|
|
|
|
2021-03-12 01:05:10 +00:00
|
|
|
/// The number of milliseconds in one second.
|
|
|
|
private let MSEC_PER_SEC: TimeInterval = 1000
|
|
|
|
|
2021-02-03 11:01:29 +00:00
|
|
|
class RoomDirectCallStatusBubbleCell: RoomBaseCallBubbleCell {
|
|
|
|
|
2021-03-12 01:05:10 +00:00
|
|
|
private var callDurationString: String = ""
|
|
|
|
private var isVideoCall: Bool = false
|
|
|
|
private var isIncoming: Bool = false
|
|
|
|
private var callInviteEvent: MXEvent?
|
|
|
|
private var viewState: ViewState = .unknown {
|
|
|
|
didSet {
|
|
|
|
updateBottomContentView()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private enum ViewState {
|
|
|
|
case unknown
|
|
|
|
case ringing
|
|
|
|
case active
|
|
|
|
case declined
|
|
|
|
case missed
|
|
|
|
case ended
|
|
|
|
case failed
|
|
|
|
}
|
|
|
|
|
|
|
|
private static var callDurationFormatter: DateComponentsFormatter {
|
|
|
|
let formatter = DateComponentsFormatter()
|
|
|
|
formatter.zeroFormattingBehavior = .dropAll
|
|
|
|
formatter.allowedUnits = [.hour, .minute, .second]
|
|
|
|
formatter.unitsStyle = .abbreviated
|
|
|
|
return formatter
|
2021-02-03 11:01:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
override func update(theme: Theme) {
|
|
|
|
super.update(theme: theme)
|
2021-03-12 01:05:10 +00:00
|
|
|
if let themable = bottomContentView as? Themable {
|
|
|
|
themable.update(theme: theme)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func updateBottomContentView() {
|
|
|
|
bottomContentView = bottomView(for: viewState)
|
|
|
|
}
|
|
|
|
|
|
|
|
private var callTypeIcon: UIImage {
|
|
|
|
if isVideoCall {
|
|
|
|
return Asset.Images.callVideoIcon.image
|
|
|
|
} else {
|
|
|
|
return Asset.Images.voiceCallHangonIcon.image
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private var actionUserInfo: [AnyHashable: Any]? {
|
|
|
|
if let event = callInviteEvent {
|
|
|
|
return [kMXKRoomBubbleCellEventKey: event]
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
private func bottomView(for state: ViewState) -> UIView? {
|
|
|
|
switch state {
|
|
|
|
case .unknown:
|
|
|
|
return nil
|
|
|
|
case .ringing:
|
|
|
|
let view = HorizontalButtonsContainerView.loadFromNib()
|
|
|
|
|
|
|
|
view.firstButton.style = .negative
|
|
|
|
view.firstButton.setTitle(VectorL10n.eventFormatterCallDecline, for: .normal)
|
|
|
|
view.firstButton.setImage(Asset.Images.voiceCallHangupIcon.image, for: .normal)
|
|
|
|
view.firstButton.removeTarget(nil, action: nil, for: .touchUpInside)
|
|
|
|
view.firstButton.addTarget(self, action: #selector(declineCallAction(_:)), for: .touchUpInside)
|
|
|
|
|
|
|
|
view.secondButton.style = .positive
|
|
|
|
view.secondButton.setTitle(VectorL10n.eventFormatterCallAnswer, for: .normal)
|
|
|
|
view.secondButton.setImage(callTypeIcon, for: .normal)
|
|
|
|
view.secondButton.removeTarget(nil, action: nil, for: .touchUpInside)
|
|
|
|
view.secondButton.addTarget(self, action: #selector(answerCallAction(_:)), for: .touchUpInside)
|
|
|
|
|
|
|
|
return view
|
|
|
|
case .active:
|
|
|
|
return nil
|
|
|
|
case .declined:
|
|
|
|
let view = HorizontalButtonsContainerView.loadFromNib()
|
|
|
|
view.secondButton.isHidden = true
|
|
|
|
|
|
|
|
view.firstButton.style = .positive
|
|
|
|
view.firstButton.setTitle(VectorL10n.eventFormatterCallBack, for: .normal)
|
|
|
|
view.firstButton.setImage(callTypeIcon, for: .normal)
|
|
|
|
view.firstButton.removeTarget(nil, action: nil, for: .touchUpInside)
|
|
|
|
view.firstButton.addTarget(self, action: #selector(callBackAction(_:)), for: .touchUpInside)
|
|
|
|
|
|
|
|
return view
|
|
|
|
case .missed:
|
|
|
|
let view = HorizontalButtonsContainerView.loadFromNib()
|
|
|
|
view.secondButton.isHidden = true
|
|
|
|
|
|
|
|
view.firstButton.style = .positive
|
|
|
|
view.firstButton.setTitle(VectorL10n.eventFormatterCallBack, for: .normal)
|
|
|
|
view.firstButton.setImage(callTypeIcon, for: .normal)
|
|
|
|
view.firstButton.removeTarget(nil, action: nil, for: .touchUpInside)
|
|
|
|
view.firstButton.addTarget(self, action: #selector(callBackAction(_:)), for: .touchUpInside)
|
|
|
|
|
|
|
|
return view
|
|
|
|
case .ended:
|
|
|
|
return nil
|
|
|
|
case .failed:
|
|
|
|
let view = HorizontalButtonsContainerView.loadFromNib()
|
|
|
|
view.secondButton.isHidden = true
|
|
|
|
|
|
|
|
view.firstButton.style = .positive
|
|
|
|
view.firstButton.setTitle(VectorL10n.eventFormatterCallRetry, for: .normal)
|
|
|
|
view.firstButton.setImage(callTypeIcon, for: .normal)
|
|
|
|
view.firstButton.removeTarget(nil, action: nil, for: .touchUpInside)
|
|
|
|
view.firstButton.addTarget(self, action: #selector(callBackAction(_:)), for: .touchUpInside)
|
|
|
|
|
|
|
|
return view
|
|
|
|
}
|
2021-02-03 11:01:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private func configure(withCall call: MXCall) {
|
|
|
|
switch call.state {
|
|
|
|
case .connected,
|
|
|
|
.fledgling,
|
|
|
|
.waitLocalMedia,
|
|
|
|
.createOffer,
|
|
|
|
.inviteSent,
|
|
|
|
.createAnswer,
|
|
|
|
.connecting,
|
|
|
|
.onHold,
|
|
|
|
.remotelyOnHold:
|
2021-03-12 01:05:10 +00:00
|
|
|
viewState = .active
|
|
|
|
statusText = VectorL10n.eventFormatterCallYouCurrentlyIn
|
2021-02-03 11:01:29 +00:00
|
|
|
case .ringing:
|
|
|
|
if call.isIncoming {
|
2021-03-12 01:05:10 +00:00
|
|
|
viewState = .ringing
|
|
|
|
statusText = nil
|
2021-02-03 11:01:29 +00:00
|
|
|
} else {
|
2021-03-12 01:05:10 +00:00
|
|
|
viewState = .active
|
|
|
|
statusText = VectorL10n.eventFormatterCallYouCurrentlyIn
|
2021-02-03 11:01:29 +00:00
|
|
|
}
|
|
|
|
case .ended:
|
|
|
|
switch call.endReason {
|
|
|
|
case .unknown,
|
|
|
|
.hangup,
|
|
|
|
.hangupElsewhere,
|
|
|
|
.remoteHangup,
|
|
|
|
.answeredElseWhere:
|
2021-03-12 01:05:10 +00:00
|
|
|
viewState = .ended
|
|
|
|
statusText = VectorL10n.eventFormatterCallHasEnded(callDurationString)
|
|
|
|
case .missed:
|
|
|
|
if call.isIncoming {
|
|
|
|
viewState = .missed
|
|
|
|
statusText = VectorL10n.eventFormatterCallYouMissed
|
|
|
|
} else {
|
|
|
|
statusText = VectorL10n.eventFormatterCallHasEnded(callDurationString)
|
|
|
|
}
|
2021-02-03 11:01:29 +00:00
|
|
|
case .busy:
|
|
|
|
configureForRejectedCall(call: call)
|
|
|
|
@unknown default:
|
2021-03-12 01:05:10 +00:00
|
|
|
viewState = .ended
|
|
|
|
statusText = VectorL10n.eventFormatterCallHasEnded(callDurationString)
|
2021-02-03 11:01:29 +00:00
|
|
|
}
|
|
|
|
case .inviteExpired,
|
|
|
|
.answeredElseWhere:
|
2021-03-12 01:05:10 +00:00
|
|
|
viewState = .ended
|
|
|
|
statusText = VectorL10n.eventFormatterCallHasEnded(callDurationString)
|
2021-02-03 11:01:29 +00:00
|
|
|
@unknown default:
|
2021-03-12 01:05:10 +00:00
|
|
|
viewState = .ended
|
|
|
|
statusText = VectorL10n.eventFormatterCallHasEnded(callDurationString)
|
2021-02-03 11:01:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func configureForRejectedCall(withEvent event: MXEvent? = nil, call: MXCall? = nil, bubbleCellData: RoomBubbleCellData? = nil) {
|
|
|
|
|
|
|
|
let isMyReject: Bool
|
|
|
|
|
|
|
|
if let call = call, call.isIncoming {
|
|
|
|
isMyReject = true
|
|
|
|
} else if let event = event, let bubbleCellData = bubbleCellData, event.sender == bubbleCellData.mxSession.myUserId {
|
|
|
|
isMyReject = true
|
|
|
|
} else {
|
|
|
|
isMyReject = false
|
|
|
|
}
|
|
|
|
|
|
|
|
if isMyReject {
|
2021-03-12 01:05:10 +00:00
|
|
|
viewState = .declined
|
|
|
|
statusText = VectorL10n.eventFormatterCallYouDeclined
|
2021-02-03 11:01:29 +00:00
|
|
|
} else {
|
2021-03-12 01:05:10 +00:00
|
|
|
viewState = .ended
|
|
|
|
statusText = VectorL10n.eventFormatterCallHasEnded(callDurationString)
|
2021-02-03 11:01:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-12 01:05:10 +00:00
|
|
|
private func configureForHangupCall(withEvent event: MXEvent) {
|
|
|
|
guard let hangupEventContent = MXCallHangupEventContent(fromJSON: event.content) else {
|
|
|
|
viewState = .ended
|
|
|
|
statusText = VectorL10n.eventFormatterCallHasEnded(callDurationString)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
switch hangupEventContent.reasonType {
|
|
|
|
case .userHangup:
|
|
|
|
viewState = .ended
|
|
|
|
statusText = VectorL10n.eventFormatterCallHasEnded(callDurationString)
|
|
|
|
default:
|
|
|
|
viewState = .failed
|
|
|
|
statusText = VectorL10n.eventFormatterCallConnectionFailed
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func configureForUnansweredCall() {
|
|
|
|
if isIncoming {
|
|
|
|
// missed call
|
|
|
|
viewState = .missed
|
|
|
|
statusText = VectorL10n.eventFormatterCallYouMissed
|
|
|
|
} else {
|
|
|
|
// outgoing unanswered call
|
|
|
|
viewState = .ended
|
|
|
|
statusText = VectorL10n.eventFormatterCallHasEnded(callDurationString)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Actions
|
|
|
|
|
|
|
|
@objc
|
|
|
|
private func callBackAction(_ sender: CallTileActionButton) {
|
|
|
|
self.delegate?.cell(self,
|
|
|
|
didRecognizeAction: kMXKRoomBubbleCellCallBackButtonPressed,
|
|
|
|
userInfo: actionUserInfo)
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc
|
|
|
|
private func declineCallAction(_ sender: CallTileActionButton) {
|
|
|
|
self.delegate?.cell(self,
|
|
|
|
didRecognizeAction: kMXKRoomBubbleCellCallDeclineButtonPressed,
|
|
|
|
userInfo: actionUserInfo)
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc
|
|
|
|
private func answerCallAction(_ sender: CallTileActionButton) {
|
|
|
|
self.delegate?.cell(self,
|
|
|
|
didRecognizeAction: kMXKRoomBubbleCellCallAnswerButtonPressed,
|
|
|
|
userInfo: actionUserInfo)
|
|
|
|
}
|
|
|
|
|
2021-02-03 11:01:29 +00:00
|
|
|
// MARK: - MXKCellRendering
|
|
|
|
|
|
|
|
override func render(_ cellData: MXKCellData!) {
|
|
|
|
super.render(cellData)
|
|
|
|
|
2021-03-12 01:05:10 +00:00
|
|
|
viewState = .unknown
|
|
|
|
|
2021-02-03 11:01:29 +00:00
|
|
|
guard let bubbleCellData = cellData as? RoomBubbleCellData else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let events = bubbleCellData.allLinkedEvents()
|
|
|
|
|
2021-03-12 01:05:10 +00:00
|
|
|
guard let inviteEvent = events.first(where: { $0.eventType == .callInvite }) else {
|
2021-02-03 11:01:29 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-03-12 01:05:10 +00:00
|
|
|
guard let callInviteEventContent = MXCallInviteEventContent(fromJSON: inviteEvent.content) else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
self.isVideoCall = callInviteEventContent.isVideoCall()
|
|
|
|
self.callDurationString = readableCallDuration(from: events)
|
|
|
|
self.isIncoming = inviteEvent.sender != bubbleCellData.mxSession.myUserId
|
|
|
|
self.callInviteEvent = inviteEvent
|
|
|
|
|
|
|
|
let callId = callInviteEventContent.callId
|
2021-02-03 11:01:29 +00:00
|
|
|
guard let call = bubbleCellData.mxSession.callManager.call(withCallId: callId) else {
|
|
|
|
|
|
|
|
// check events include a reject event
|
|
|
|
if let rejectEvent = events.first(where: { $0.eventType == .callReject }) {
|
|
|
|
configureForRejectedCall(withEvent: rejectEvent, bubbleCellData: bubbleCellData)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-03-12 01:05:10 +00:00
|
|
|
// check events include an answer event
|
|
|
|
if !events.contains(where: { $0.eventType == .callAnswer }) {
|
|
|
|
configureForUnansweredCall()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// check events include a hangup event
|
|
|
|
if let hangupEvent = events.first(where: { $0.eventType == .callHangup }) {
|
|
|
|
configureForHangupCall(withEvent: hangupEvent)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// there is no reject or hangup event, we can just say this call has ended
|
|
|
|
viewState = .ended
|
|
|
|
statusText = VectorL10n.eventFormatterCallHasEnded(callDurationString)
|
2021-02-03 11:01:29 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
configure(withCall: call)
|
|
|
|
}
|
|
|
|
|
2021-03-12 01:05:10 +00:00
|
|
|
private func callDuration(from events: [MXEvent]) -> TimeInterval {
|
|
|
|
guard let startDate = events.first(where: { $0.eventType == .callAnswer })?.originServerTs else {
|
|
|
|
// never started
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
guard let endDate = events.first(where: { $0.eventType == .callHangup })?.originServerTs
|
|
|
|
?? events.first(where: { $0.eventType == .callReject })?.originServerTs else {
|
|
|
|
// not ended yet, compute the diff from now
|
|
|
|
return (NSTimeIntervalSince1970 - TimeInterval(startDate))/MSEC_PER_SEC
|
|
|
|
}
|
2021-02-03 11:01:29 +00:00
|
|
|
|
2021-03-12 01:05:10 +00:00
|
|
|
// ended, compute the diff between two dates
|
|
|
|
return TimeInterval(endDate - startDate)/MSEC_PER_SEC
|
2021-02-03 11:01:29 +00:00
|
|
|
}
|
|
|
|
|
2021-03-12 01:05:10 +00:00
|
|
|
private func readableCallDuration(from events: [MXEvent]) -> String {
|
|
|
|
let duration = callDuration(from: events)
|
|
|
|
|
|
|
|
if duration <= 0 {
|
|
|
|
return ""
|
2021-02-03 11:01:29 +00:00
|
|
|
}
|
2021-03-12 01:05:10 +00:00
|
|
|
|
|
|
|
return RoomDirectCallStatusBubbleCell.callDurationFormatter.string(from: duration) ?? ""
|
2021-02-03 11:01:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|