提交
This commit is contained in:
@@ -0,0 +1,83 @@
|
||||
//
|
||||
// FloatWindowViewController.swift
|
||||
// TUICallKit
|
||||
//
|
||||
// Created by vincepzhang on 2023/3/2.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import TUICore
|
||||
|
||||
protocol FloatingWindowViewDelegate: NSObject {
|
||||
func tapGestureAction(tapGesture: UITapGestureRecognizer)
|
||||
func panGestureAction(panGesture: UIPanGestureRecognizer)
|
||||
}
|
||||
|
||||
class FloatWindowViewController: UIViewController, FloatingWindowViewDelegate {
|
||||
|
||||
let callEventObserver = Observer()
|
||||
weak var delegate: FloatingWindowViewDelegate?
|
||||
|
||||
override func viewDidLoad() {
|
||||
super.viewDidLoad()
|
||||
|
||||
if TUICallState.instance.scene.value == .group {
|
||||
addFloatingWindowGroupView()
|
||||
} else if TUICallState.instance.scene.value == .single {
|
||||
addFloatingWindowSignalView()
|
||||
}
|
||||
|
||||
registerObserver()
|
||||
}
|
||||
|
||||
deinit {
|
||||
TUICallState.instance.event.removeObserver(callEventObserver)
|
||||
for view in view.subviews {
|
||||
view.removeFromSuperview()
|
||||
}
|
||||
}
|
||||
|
||||
func addFloatingWindowGroupView() {
|
||||
let floatView = FloatingWindowGroupView(frame: CGRect.zero)
|
||||
view.addSubview(floatView)
|
||||
floatView.snp.makeConstraints { make in
|
||||
make.center.equalTo(self.view)
|
||||
make.size.equalTo(self.view)
|
||||
}
|
||||
floatView.delegate = self
|
||||
}
|
||||
|
||||
func addFloatingWindowSignalView() {
|
||||
let floatView = FloatingWindowSignalView(frame: CGRect.zero)
|
||||
view.addSubview(floatView)
|
||||
floatView.snp.makeConstraints { make in
|
||||
make.center.equalTo(self.view)
|
||||
make.size.equalTo(self.view)
|
||||
}
|
||||
floatView.delegate = self
|
||||
}
|
||||
|
||||
// MARK: FloatingWindowViewDelegate
|
||||
func tapGestureAction(tapGesture: UITapGestureRecognizer) {
|
||||
if self.delegate != nil && ((self.delegate?.responds(to: Selector(("tapGestureAction")))) != nil) {
|
||||
self.delegate?.tapGestureAction(tapGesture: tapGesture)
|
||||
}
|
||||
}
|
||||
|
||||
func panGestureAction(panGesture: UIPanGestureRecognizer) {
|
||||
if self.delegate != nil && ((self.delegate?.responds(to: Selector(("panGestureAction")))) != nil) {
|
||||
self.delegate?.panGestureAction(panGesture: panGesture)
|
||||
}
|
||||
}
|
||||
|
||||
func registerObserver() {
|
||||
TUICallState.instance.event.addObserver(callEventObserver) { newValue, _ in
|
||||
if newValue.eventType == .ERROR {
|
||||
guard let errorCode = newValue.param[EVENT_KEY_CODE] as? Int32 else { return }
|
||||
guard let errorMessage = newValue.param[EVENT_KEY_MESSAGE] as? String else { return }
|
||||
TUITool.makeToast("error:\(errorCode):\(errorMessage)")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,85 @@
|
||||
//
|
||||
// FloatingWindowButton.swift
|
||||
// TUICallKit
|
||||
//
|
||||
// Created by vincepzhang on 2023/2/15.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
class FloatingWindowButton: UIView {
|
||||
|
||||
let mediaTypeObserver = Observer()
|
||||
|
||||
let floatButton: FloatingWindowCustomButton = {
|
||||
let floatButton = FloatingWindowCustomButton(type: .system)
|
||||
return floatButton
|
||||
}()
|
||||
|
||||
override init(frame: CGRect) {
|
||||
super.init(frame: frame)
|
||||
|
||||
updateImage()
|
||||
registerObserveState()
|
||||
}
|
||||
|
||||
required init?(coder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
deinit {
|
||||
TUICallState.instance.mediaType.removeObserver(mediaTypeObserver)
|
||||
}
|
||||
|
||||
// MARK: UI Specification Processing
|
||||
private var isViewReady: Bool = false
|
||||
override func didMoveToWindow() {
|
||||
super.didMoveToWindow()
|
||||
if isViewReady { return }
|
||||
constructViewHierarchy()
|
||||
activateConstraints()
|
||||
bindInteraction()
|
||||
isViewReady = true
|
||||
}
|
||||
|
||||
func constructViewHierarchy() {
|
||||
addSubview(floatButton)
|
||||
}
|
||||
|
||||
func activateConstraints() {
|
||||
floatButton.snp.makeConstraints { make in
|
||||
make.center.equalTo(self)
|
||||
make.width.height.equalTo(24)
|
||||
}
|
||||
}
|
||||
|
||||
func bindInteraction() {
|
||||
floatButton.addTarget(self, action: #selector(clickFloatButton(sender: )), for: .touchUpInside)
|
||||
}
|
||||
|
||||
// MARK: Action Event
|
||||
@objc func clickFloatButton(sender: UIButton) {
|
||||
WindowManager.instance.showFloatWindow()
|
||||
}
|
||||
|
||||
// MARK: Register TUICallState Observer && Update UI
|
||||
func registerObserveState() {
|
||||
TUICallState.instance.mediaType.addObserver(mediaTypeObserver) { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
self.updateImage()
|
||||
}
|
||||
}
|
||||
|
||||
func updateImage() {
|
||||
if let image = TUICallKitCommon.getBundleImage(name: "icon_min_window") {
|
||||
floatButton.setBackgroundImage(image, for: .normal)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class FloatingWindowCustomButton: UIButton {
|
||||
override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? {
|
||||
let expandedBounds = bounds.insetBy(dx: -6, dy: -6)
|
||||
return expandedBounds.contains(point) ? self : nil
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,96 @@
|
||||
//
|
||||
// FloatingWindowStatusView.swift
|
||||
// TUICallKit
|
||||
//
|
||||
// Created by noah on 2023/11/20.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
|
||||
class FloatingWindowGroupStatusView: UIView {
|
||||
|
||||
let isCameraOpenObserver = Observer()
|
||||
let isMicMuteObserver = Observer()
|
||||
|
||||
let videoImageView: UIImageView = {
|
||||
let imageView = UIImageView()
|
||||
let imageName = TUICallState.instance.isCameraOpen.value ? "icon_float_group_video_on" : "icon_float_group_video_off"
|
||||
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
|
||||
imageView.image = image
|
||||
}
|
||||
return imageView
|
||||
}()
|
||||
|
||||
let audioImageView: UIImageView = {
|
||||
let imageView = UIImageView()
|
||||
let imageName = TUICallState.instance.isMicMute.value ? "icon_float_group_audio_off" : "icon_float_group_audio_on"
|
||||
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
|
||||
imageView.image = image
|
||||
}
|
||||
return imageView
|
||||
}()
|
||||
|
||||
override init(frame: CGRect) {
|
||||
super.init(frame: frame)
|
||||
registerObserveState()
|
||||
constructViewHierarchy()
|
||||
activateConstraints()
|
||||
backgroundColor = TUICoreDefineConvert.getTUICallKitDynamicColor(colorKey: "callkit_float_window_group_status_view_bg_color",
|
||||
defaultHex: "#F9F6F4")
|
||||
}
|
||||
|
||||
required init?(coder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
deinit {
|
||||
TUICallState.instance.isCameraOpen.removeObserver(isCameraOpenObserver)
|
||||
TUICallState.instance.isMicMute.removeObserver(isMicMuteObserver)
|
||||
}
|
||||
|
||||
func constructViewHierarchy() {
|
||||
addSubview(videoImageView)
|
||||
addSubview(audioImageView)
|
||||
}
|
||||
|
||||
func activateConstraints() {
|
||||
videoImageView.snp.makeConstraints { make in
|
||||
make.centerY.equalTo(self)
|
||||
make.leading.equalTo(self).offset(14.scaleWidth())
|
||||
make.width.height.equalTo(16.scaleWidth())
|
||||
}
|
||||
audioImageView.snp.makeConstraints { make in
|
||||
make.centerY.equalTo(self)
|
||||
make.trailing.equalTo(self).offset(-14.scaleWidth())
|
||||
make.width.height.equalTo(16.scaleWidth())
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Register TUICallState Observer && Update UI
|
||||
func registerObserveState() {
|
||||
isCameraOpenChange()
|
||||
isMicMuteChange()
|
||||
}
|
||||
|
||||
func isCameraOpenChange() {
|
||||
TUICallState.instance.isCameraOpen.addObserver(isCameraOpenObserver, closure: { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
let imageName = newValue ? "icon_float_group_video_on" : "icon_float_group_video_off"
|
||||
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
|
||||
self.videoImageView.image = image
|
||||
}
|
||||
|
||||
})
|
||||
}
|
||||
|
||||
func isMicMuteChange() {
|
||||
TUICallState.instance.isMicMute.addObserver(isMicMuteObserver, closure: { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
let imageName = newValue ? "icon_float_group_audio_off" : "icon_float_group_audio_on"
|
||||
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
|
||||
self.audioImageView.image = image
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,384 @@
|
||||
//
|
||||
// FloatingWindowGroupView.swift
|
||||
// TUICallKit
|
||||
//
|
||||
// Created by noah on 2023/11/21.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
import TUICore
|
||||
|
||||
class FloatingWindowGroupView: UIView {
|
||||
|
||||
let viewModel = FloatingWindowViewModel()
|
||||
weak var delegate: FloatingWindowViewDelegate?
|
||||
|
||||
let selfCallStatusObserver = Observer()
|
||||
let remoteVideoAvailableObserver = Observer()
|
||||
let callTimeObserver = Observer()
|
||||
let currentSpeakUserObserver = Observer()
|
||||
let currentSpeakUserVideoAvailableObserver = Observer()
|
||||
|
||||
var localPreView: VideoView {
|
||||
if VideoFactory.instance.viewMap[viewModel.selfUser.value.id.value] == nil {
|
||||
let _ = VideoFactory.instance.createVideoView(userId: viewModel.selfUser.value.id.value, frame: CGRect.zero)
|
||||
}
|
||||
VideoFactory.instance.viewMap[viewModel.selfUser.value.id.value]?.videoView.isUserInteractionEnabled = false
|
||||
return VideoFactory.instance.viewMap[viewModel.selfUser.value.id.value]?.videoView ?? VideoView(frame: CGRect.zero)
|
||||
}
|
||||
|
||||
var remotePreView: VideoView {
|
||||
let remoteUser = viewModel.currentSpeakUser.value
|
||||
if VideoFactory.instance.viewMap[remoteUser.id.value] == nil {
|
||||
let _ = VideoFactory.instance.createVideoView(userId: remoteUser.id.value, frame: CGRect.zero)
|
||||
}
|
||||
VideoFactory.instance.viewMap[remoteUser.id.value]?.videoView.isUserInteractionEnabled = false
|
||||
return VideoFactory.instance.viewMap[remoteUser.id.value]?.videoView ?? VideoView(frame: CGRect.zero)
|
||||
}
|
||||
|
||||
let avatarImageView: UIImageView = {
|
||||
let avatarImageView = UIImageView()
|
||||
avatarImageView.contentMode = .scaleAspectFill
|
||||
avatarImageView.clipsToBounds = true
|
||||
avatarImageView.isUserInteractionEnabled = false
|
||||
return avatarImageView
|
||||
}()
|
||||
|
||||
let containerView: UIView = {
|
||||
let containerView = UIView()
|
||||
containerView.backgroundColor = TUICoreDefineConvert.getTUICallKitDynamicColor(colorKey: "callkit_float_window_bg_color",
|
||||
defaultHex: "#FFFFFF")
|
||||
containerView.layer.cornerRadius = 12.scaleWidth()
|
||||
containerView.layer.masksToBounds = true
|
||||
containerView.isUserInteractionEnabled = false
|
||||
return containerView
|
||||
}()
|
||||
|
||||
let audioContainerView: UIView = {
|
||||
let containerView = UIView()
|
||||
containerView.backgroundColor = TUICoreDefineConvert.getTUICallKitDynamicColor(colorKey: "callkit_float_window_bg_color",
|
||||
defaultHex: "#FFFFFF")
|
||||
return containerView
|
||||
}()
|
||||
|
||||
let shadowView: UIView = {
|
||||
let shadowView = UIView()
|
||||
shadowView.backgroundColor = TUICoreDefineConvert.getTUICallKitDynamicColor(colorKey: "callkit_float_window_bg_color",
|
||||
defaultHex: "#FFFFFF")
|
||||
shadowView.layer.shadowColor = UIColor.t_colorWithHexString(color: "353941").cgColor
|
||||
shadowView.layer.shadowOpacity = 0.4
|
||||
shadowView.layer.shadowRadius = 4.scaleWidth()
|
||||
shadowView.layer.shadowOffset = CGSize(width: 0, height: 0)
|
||||
shadowView.layer.cornerRadius = 12.scaleWidth()
|
||||
return shadowView
|
||||
}()
|
||||
|
||||
let imageView: UIImageView = {
|
||||
let imageView = UIImageView()
|
||||
if let image = TUICallKitCommon.getBundleImage(name: "icon_float_dialing") {
|
||||
imageView.image = image
|
||||
}
|
||||
imageView.isUserInteractionEnabled = false
|
||||
return imageView
|
||||
}()
|
||||
|
||||
let audioDescribeLabel: UILabel = {
|
||||
let describeLabel = UILabel()
|
||||
describeLabel.font = UIFont.systemFont(ofSize: 12.0)
|
||||
describeLabel.textColor = UIColor.t_colorWithHexString(color: "#12B969")
|
||||
describeLabel.textAlignment = .center
|
||||
describeLabel.isUserInteractionEnabled = false
|
||||
describeLabel.text = TUICallKitLocalize(key: "TUICallKit.FloatingWindow.waitAccept") ?? ""
|
||||
return describeLabel
|
||||
}()
|
||||
|
||||
lazy var timerLabel: UILabel = {
|
||||
let timerLabel = UILabel()
|
||||
timerLabel.font = UIFont.systemFont(ofSize: 12.0)
|
||||
timerLabel.textColor = UIColor.t_colorWithHexString(color: "#12B969")
|
||||
timerLabel.textAlignment = .center
|
||||
timerLabel.isUserInteractionEnabled = false
|
||||
timerLabel.text = viewModel.getCallTimeString()
|
||||
return timerLabel
|
||||
}()
|
||||
|
||||
let groupNameLabel: UILabel = {
|
||||
let nameLabel = UILabel()
|
||||
nameLabel.font = UIFont.systemFont(ofSize: 12.0)
|
||||
nameLabel.textColor = UIColor.t_colorWithHexString(color: "#FFFFFF")
|
||||
nameLabel.textAlignment = .center
|
||||
nameLabel.isUserInteractionEnabled = false
|
||||
return nameLabel
|
||||
}()
|
||||
|
||||
let groupStatusView: UIView = {
|
||||
let statusView = FloatingWindowGroupStatusView()
|
||||
return statusView
|
||||
}()
|
||||
|
||||
override init(frame: CGRect) {
|
||||
super.init(frame: frame)
|
||||
registerObserverState()
|
||||
constructViewHierarchy()
|
||||
activateConstraints()
|
||||
bindInteraction()
|
||||
updateUI()
|
||||
}
|
||||
|
||||
required init?(coder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
deinit {
|
||||
viewModel.selfCallStatus.removeObserver(selfCallStatusObserver)
|
||||
viewModel.selfCallStatus.removeObserver(currentSpeakUserObserver)
|
||||
viewModel.currentSpeakUser.value.videoAvailable.removeObserver(currentSpeakUserVideoAvailableObserver)
|
||||
viewModel.selfCallStatus.removeObserver(callTimeObserver)
|
||||
for index in 0..<viewModel.remoteUserList.value.count {
|
||||
guard index < viewModel.remoteUserList.value.count else {
|
||||
break
|
||||
}
|
||||
viewModel.remoteUserList.value[index].videoAvailable.removeObserver(remoteVideoAvailableObserver)
|
||||
}
|
||||
}
|
||||
|
||||
func constructViewHierarchy() {
|
||||
addSubview(shadowView)
|
||||
addSubview(containerView)
|
||||
containerView.addSubview(avatarImageView)
|
||||
containerView.addSubview(audioContainerView)
|
||||
audioContainerView.addSubview(imageView)
|
||||
audioContainerView.addSubview(audioDescribeLabel)
|
||||
audioContainerView.addSubview(timerLabel)
|
||||
containerView.addSubview(groupNameLabel)
|
||||
containerView.addSubview(groupStatusView)
|
||||
}
|
||||
|
||||
func activateConstraints() {
|
||||
shadowView.snp.makeConstraints { make in
|
||||
make.edges.equalTo(containerView)
|
||||
}
|
||||
containerView.snp.makeConstraints { make in
|
||||
make.center.equalToSuperview()
|
||||
make.leading.top.equalTo(8.scaleWidth())
|
||||
}
|
||||
audioContainerView.snp.makeConstraints { make in
|
||||
make.top.centerX.equalTo(containerView)
|
||||
make.width.height.equalTo(72.scaleWidth())
|
||||
}
|
||||
imageView.snp.makeConstraints { make in
|
||||
make.top.equalTo(audioContainerView).offset(8.scaleWidth())
|
||||
make.centerX.equalTo(audioContainerView)
|
||||
make.width.height.equalTo(36.scaleWidth())
|
||||
}
|
||||
audioDescribeLabel.snp.makeConstraints { make in
|
||||
make.centerX.width.equalTo(audioContainerView)
|
||||
make.top.equalTo(imageView.snp.bottom).offset(4.scaleWidth())
|
||||
make.height.equalTo(20.scaleWidth())
|
||||
}
|
||||
timerLabel.snp.makeConstraints { make in
|
||||
make.centerX.width.equalTo(audioContainerView)
|
||||
make.top.equalTo(imageView.snp.bottom).offset(5.scaleWidth())
|
||||
make.height.equalTo(20.scaleWidth())
|
||||
}
|
||||
avatarImageView.snp.makeConstraints { make in
|
||||
make.top.centerX.width.equalToSuperview()
|
||||
make.bottom.equalTo(groupStatusView.snp.top)
|
||||
}
|
||||
groupNameLabel.snp.makeConstraints { make in
|
||||
make.centerX.width.equalTo(containerView)
|
||||
make.bottom.equalTo(groupStatusView.snp.top)
|
||||
make.height.equalTo(20.scaleWidth())
|
||||
}
|
||||
groupStatusView.snp.makeConstraints { make in
|
||||
make.centerX.width.equalTo(containerView)
|
||||
make.bottom.equalTo(containerView)
|
||||
make.height.equalTo(20.scaleWidth())
|
||||
}
|
||||
}
|
||||
|
||||
func bindInteraction() {
|
||||
let tap = UITapGestureRecognizer(target: self, action: #selector(tapGestureAction(tapGesture: )))
|
||||
let pan = UIPanGestureRecognizer(target: self, action: #selector(panGestureAction(panGesture: )))
|
||||
self.addGestureRecognizer(tap)
|
||||
pan.require(toFail: tap)
|
||||
self.addGestureRecognizer(pan)
|
||||
}
|
||||
|
||||
func clearSubview() {
|
||||
for subview in subviews {
|
||||
subview.removeFromSuperview()
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Action Event
|
||||
@objc func tapGestureAction(tapGesture: UITapGestureRecognizer) {
|
||||
if self.delegate != nil && ((self.delegate?.responds(to: Selector(("tapGestureAction")))) != nil) {
|
||||
self.delegate?.tapGestureAction(tapGesture: tapGesture)
|
||||
}
|
||||
}
|
||||
|
||||
@objc func panGestureAction(panGesture: UIPanGestureRecognizer) {
|
||||
if self.delegate != nil && ((self.delegate?.responds(to: Selector(("panGestureAction")))) != nil) {
|
||||
self.delegate?.panGestureAction(panGesture: panGesture)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Register TUICallState Observer && Update UI
|
||||
func registerObserverState() {
|
||||
registerCallStatusObserver()
|
||||
registerCurrentSpeakUserObserver()
|
||||
registerCurrentSpeakUserVideoAvailableObserver()
|
||||
registerCallTimeObserver()
|
||||
registerUserVideoAvailableChange()
|
||||
}
|
||||
|
||||
func registerCallStatusObserver() {
|
||||
viewModel.selfCallStatus.addObserver(selfCallStatusObserver, closure: { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
self.updateUI()
|
||||
})
|
||||
}
|
||||
|
||||
func registerCallTimeObserver() {
|
||||
viewModel.callTime.addObserver(callTimeObserver, closure: { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
DispatchCallKitMainAsyncSafe {
|
||||
self.timerLabel.text = self.viewModel.getCallTimeString()
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func registerCurrentSpeakUserObserver() {
|
||||
viewModel.currentSpeakUser.addObserver(currentSpeakUserObserver, closure: { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
if self.viewModel.scene.value == .group {
|
||||
self.updateUI()
|
||||
self.groupNameLabel.text = newValue.nickname.value
|
||||
self.avatarImageView.sd_setImage(with: URL(string: newValue.avatar.value),
|
||||
placeholderImage: TUICallKitCommon.getBundleImage(name: "default_user_icon"))
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func registerCurrentSpeakUserVideoAvailableObserver() {
|
||||
viewModel.currentSpeakUser.value.videoAvailable.addObserver(currentSpeakUserVideoAvailableObserver, closure: { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
if self.viewModel.scene.value == .group {
|
||||
self.updateUI()
|
||||
self.groupNameLabel.text = self.viewModel.currentSpeakUser.value.nickname.value
|
||||
self.avatarImageView.sd_setImage(with: URL(string: self.viewModel.currentSpeakUser.value.avatar.value),
|
||||
placeholderImage: TUICallKitCommon.getBundleImage(name: "default_user_icon"))
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func registerUserVideoAvailableChange() {
|
||||
for index in 0..<viewModel.remoteUserList.value.count {
|
||||
guard index < viewModel.remoteUserList.value.count else {
|
||||
break
|
||||
}
|
||||
viewModel.remoteUserList.value[index].videoAvailable.addObserver(remoteVideoAvailableObserver, closure: { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
if (self.viewModel.currentSpeakUser.value.id.value == self.viewModel.remoteUserList.value[index].id.value) {
|
||||
self.updateUI()
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Update UI
|
||||
func updateUI() {
|
||||
cleanView()
|
||||
|
||||
if viewModel.selfCallStatus.value == .waiting {
|
||||
setGroupWaitingUI()
|
||||
} else if viewModel.selfCallStatus.value == .accept {
|
||||
setGroupAcceptUI()
|
||||
}
|
||||
}
|
||||
|
||||
func setGroupWaitingUI() {
|
||||
audioContainerView.isHidden = false
|
||||
imageView.isHidden = false
|
||||
audioDescribeLabel.isHidden = false
|
||||
groupStatusView.isHidden = false
|
||||
}
|
||||
|
||||
func setGroupAcceptUI() {
|
||||
if viewModel.currentSpeakUser.value.id.value == viewModel.selfUser.value.id.value {
|
||||
setGroupLocalUI()
|
||||
} else if viewModel.currentSpeakUser.value.id.value.count > 0 {
|
||||
setGroupRemoteUI()
|
||||
} else {
|
||||
setGroupDefineUI()
|
||||
}
|
||||
}
|
||||
|
||||
func setGroupDefineUI() {
|
||||
audioContainerView.isHidden = false
|
||||
imageView.isHidden = false
|
||||
timerLabel.isHidden = false
|
||||
groupStatusView.isHidden = false
|
||||
}
|
||||
|
||||
func setGroupLocalUI() {
|
||||
groupStatusView.isHidden = false
|
||||
groupNameLabel.isHidden = false
|
||||
|
||||
if TUICallState.instance.isCameraOpen.value {
|
||||
setGroupLocalVideoAvailableUI()
|
||||
} else {
|
||||
setGroupVideoNotAvailableUI()
|
||||
}
|
||||
|
||||
containerView.bringSubviewToFront(groupNameLabel)
|
||||
}
|
||||
|
||||
func setGroupRemoteUI() {
|
||||
groupStatusView.isHidden = false
|
||||
groupNameLabel.isHidden = false
|
||||
|
||||
if viewModel.currentSpeakUser.value.videoAvailable.value {
|
||||
setGroupRemoteVideoAvailableUI()
|
||||
} else {
|
||||
setGroupVideoNotAvailableUI()
|
||||
}
|
||||
|
||||
containerView.bringSubviewToFront(groupNameLabel)
|
||||
}
|
||||
|
||||
func setGroupLocalVideoAvailableUI() {
|
||||
avatarImageView.isHidden = true
|
||||
containerView.addSubview(localPreView)
|
||||
localPreView.snp.remakeConstraints { make in
|
||||
make.edges.equalTo(avatarImageView)
|
||||
}
|
||||
}
|
||||
|
||||
func setGroupRemoteVideoAvailableUI() {
|
||||
avatarImageView.isHidden = true
|
||||
containerView.addSubview(remotePreView)
|
||||
remotePreView.snp.makeConstraints { make in
|
||||
make.edges.equalTo(avatarImageView)
|
||||
}
|
||||
viewModel.startRemoteView(user: viewModel.currentSpeakUser.value, videoView: remotePreView)
|
||||
}
|
||||
|
||||
func setGroupVideoNotAvailableUI() {
|
||||
avatarImageView.isHidden = false
|
||||
containerView.bringSubviewToFront(avatarImageView)
|
||||
}
|
||||
|
||||
func cleanView() {
|
||||
avatarImageView.isHidden = true
|
||||
audioContainerView.isHidden = true
|
||||
imageView.isHidden = true
|
||||
audioDescribeLabel.isHidden = true
|
||||
timerLabel.isHidden = true
|
||||
groupStatusView.isHidden = true
|
||||
groupNameLabel.isHidden = true
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,336 @@
|
||||
//
|
||||
// FloatingWindow.swift
|
||||
// TUICallKit
|
||||
//
|
||||
// Created by vincepzhang on 2023/2/14.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import UIKit
|
||||
import TUICore
|
||||
|
||||
class FloatingWindowSignalView: UIView {
|
||||
|
||||
let viewModel = FloatingWindowViewModel()
|
||||
weak var delegate: FloatingWindowViewDelegate?
|
||||
|
||||
let selfCallStatusObserver = Observer()
|
||||
let remoteUserListObserver = Observer()
|
||||
let callTimeObserver = Observer()
|
||||
|
||||
let kFloatingWindowVideoViewRect = CGRect(x: 0,
|
||||
y: 0,
|
||||
width: kMicroVideoViewWidth - 16.scaleWidth(),
|
||||
height: kMicroVideoViewHeight - 16.scaleWidth())
|
||||
|
||||
var localPreView: VideoView {
|
||||
if VideoFactory.instance.viewMap[viewModel.selfUser.value.id.value] == nil {
|
||||
let _ = VideoFactory.instance.createVideoView(userId: viewModel.selfUser.value.id.value, frame: CGRect.zero)
|
||||
}
|
||||
VideoFactory.instance.viewMap[viewModel.selfUser.value.id.value]?.videoView.isUserInteractionEnabled = false
|
||||
return VideoFactory.instance.viewMap[viewModel.selfUser.value.id.value]?.videoView ?? VideoView(frame: CGRect.zero)
|
||||
}
|
||||
|
||||
var remotePreView: VideoView {
|
||||
guard let remoteUser = viewModel.remoteUserList.value.first else { return VideoView(frame: CGRect.zero) }
|
||||
if VideoFactory.instance.viewMap[remoteUser.id.value] == nil {
|
||||
let _ = VideoFactory.instance.createVideoView(userId: remoteUser.id.value, frame: CGRect.zero)
|
||||
}
|
||||
VideoFactory.instance.viewMap[remoteUser.id.value]?.videoView.isUserInteractionEnabled = false
|
||||
return VideoFactory.instance.viewMap[remoteUser.id.value]?.videoView ?? VideoView(frame: CGRect.zero)
|
||||
}
|
||||
|
||||
let avatarImageView: UIImageView = {
|
||||
let avatarImageView = UIImageView()
|
||||
avatarImageView.contentMode = .scaleAspectFill
|
||||
avatarImageView.clipsToBounds = true
|
||||
avatarImageView.layer.cornerRadius = 8.scaleWidth()
|
||||
avatarImageView.isUserInteractionEnabled = false
|
||||
return avatarImageView
|
||||
}()
|
||||
|
||||
let containerView: UIView = {
|
||||
let containerView = UIView()
|
||||
containerView.backgroundColor = TUICoreDefineConvert.getTUICallKitDynamicColor(colorKey: "callkit_float_window_bg_color",
|
||||
defaultHex: "#FFFFFF")
|
||||
containerView.layer.cornerRadius = 12.scaleWidth()
|
||||
containerView.layer.masksToBounds = true
|
||||
containerView.isUserInteractionEnabled = false
|
||||
return containerView
|
||||
}()
|
||||
|
||||
let audioContainerView: UIView = {
|
||||
let containerView = UIView()
|
||||
containerView.backgroundColor = TUICoreDefineConvert.getTUICallKitDynamicColor(colorKey: "callkit_float_window_bg_color",
|
||||
defaultHex: "#FFFFFF")
|
||||
return containerView
|
||||
}()
|
||||
|
||||
let shadowView: UIView = {
|
||||
let shadowView = UIView()
|
||||
shadowView.backgroundColor = TUICoreDefineConvert.getTUICallKitDynamicColor(colorKey: "callkit_float_window_bg_color",
|
||||
defaultHex: "#FFFFFF")
|
||||
shadowView.layer.shadowColor = UIColor.t_colorWithHexString(color: "353941").cgColor
|
||||
shadowView.layer.shadowOpacity = 0.4
|
||||
shadowView.layer.cornerRadius = 12.scaleWidth()
|
||||
shadowView.layer.shadowRadius = 4.scaleWidth()
|
||||
shadowView.layer.shadowOffset = CGSize(width: 0, height: 0)
|
||||
return shadowView
|
||||
}()
|
||||
|
||||
let imageView: UIImageView = {
|
||||
let imageView = UIImageView()
|
||||
if let image = TUICallKitCommon.getBundleImage(name: "icon_float_dialing") {
|
||||
imageView.image = image
|
||||
}
|
||||
imageView.isUserInteractionEnabled = false
|
||||
return imageView
|
||||
}()
|
||||
|
||||
let audioDescribeLabel: UILabel = {
|
||||
let describeLabel = UILabel()
|
||||
describeLabel.font = UIFont.systemFont(ofSize: 12.0)
|
||||
describeLabel.textColor = UIColor.t_colorWithHexString(color: "#12B969")
|
||||
describeLabel.textAlignment = .center
|
||||
describeLabel.isUserInteractionEnabled = false
|
||||
describeLabel.text = TUICallKitLocalize(key: "TUICallKit.FloatingWindow.waitAccept") ?? ""
|
||||
return describeLabel
|
||||
}()
|
||||
|
||||
let videoDescribeLabel: UILabel = {
|
||||
let describeLabel = UILabel()
|
||||
describeLabel.font = UIFont.systemFont(ofSize: 12.0)
|
||||
describeLabel.textColor = UIColor.t_colorWithHexString(color: "#FFFFFF")
|
||||
describeLabel.textAlignment = .center
|
||||
describeLabel.isUserInteractionEnabled = false
|
||||
describeLabel.text = TUICallKitLocalize(key: "TUICallKit.FloatingWindow.waitAccept") ?? ""
|
||||
return describeLabel
|
||||
}()
|
||||
|
||||
lazy var timerLabel: UILabel = {
|
||||
let timerLabel = UILabel()
|
||||
timerLabel.font = UIFont.systemFont(ofSize: 12.0)
|
||||
timerLabel.textColor = UIColor.t_colorWithHexString(color: "#12B969")
|
||||
timerLabel.textAlignment = .center
|
||||
timerLabel.isUserInteractionEnabled = false
|
||||
timerLabel.text = viewModel.getCallTimeString()
|
||||
return timerLabel
|
||||
}()
|
||||
|
||||
override init(frame: CGRect) {
|
||||
super.init(frame: frame)
|
||||
registerObserverState()
|
||||
constructViewHierarchy()
|
||||
activateConstraints()
|
||||
bindInteraction()
|
||||
updateUI()
|
||||
}
|
||||
|
||||
required init?(coder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
deinit {
|
||||
viewModel.selfCallStatus.removeObserver(selfCallStatusObserver)
|
||||
viewModel.selfCallStatus.removeObserver(callTimeObserver)
|
||||
}
|
||||
|
||||
func constructViewHierarchy() {
|
||||
addSubview(shadowView)
|
||||
addSubview(containerView)
|
||||
containerView.addSubview(avatarImageView)
|
||||
containerView.addSubview(audioContainerView)
|
||||
audioContainerView.addSubview(imageView)
|
||||
audioContainerView.addSubview(audioDescribeLabel)
|
||||
audioContainerView.addSubview(timerLabel)
|
||||
containerView.addSubview(videoDescribeLabel)
|
||||
}
|
||||
|
||||
func activateConstraints() {
|
||||
shadowView.snp.makeConstraints { make in
|
||||
make.edges.equalTo(containerView)
|
||||
}
|
||||
containerView.snp.makeConstraints { make in
|
||||
make.center.equalToSuperview()
|
||||
make.leading.top.equalTo(8.scaleWidth())
|
||||
}
|
||||
avatarImageView.snp.makeConstraints { make in
|
||||
make.center.equalToSuperview()
|
||||
make.width.height.equalTo(45.scaleWidth())
|
||||
}
|
||||
audioContainerView.snp.makeConstraints { make in
|
||||
make.top.centerX.equalTo(containerView)
|
||||
make.width.height.equalTo(72.scaleWidth())
|
||||
}
|
||||
imageView.snp.makeConstraints { make in
|
||||
make.top.equalTo(audioContainerView).offset(8.scaleWidth())
|
||||
make.centerX.equalTo(audioContainerView)
|
||||
make.width.height.equalTo(36.scaleWidth())
|
||||
}
|
||||
audioDescribeLabel.snp.makeConstraints { make in
|
||||
make.centerX.width.equalTo(audioContainerView)
|
||||
make.top.equalTo(imageView.snp.bottom).offset(4.scaleWidth())
|
||||
make.height.equalTo(20.scaleWidth())
|
||||
}
|
||||
timerLabel.snp.makeConstraints { make in
|
||||
make.centerX.width.equalTo(audioContainerView)
|
||||
make.top.equalTo(imageView.snp.bottom).offset(4.scaleWidth())
|
||||
make.height.equalTo(20.scaleWidth())
|
||||
}
|
||||
videoDescribeLabel.snp.makeConstraints { make in
|
||||
make.centerX.width.equalTo(containerView)
|
||||
make.bottom.equalTo(containerView).offset(-8.scaleWidth())
|
||||
make.height.equalTo(20.scaleWidth())
|
||||
}
|
||||
}
|
||||
|
||||
func bindInteraction() {
|
||||
let tap = UITapGestureRecognizer(target: self, action: #selector(tapGestureAction(tapGesture: )))
|
||||
let pan = UIPanGestureRecognizer(target: self, action: #selector(panGestureAction(panGesture: )))
|
||||
self.addGestureRecognizer(tap)
|
||||
pan.require(toFail: tap)
|
||||
self.addGestureRecognizer(pan)
|
||||
}
|
||||
|
||||
func clearSubview() {
|
||||
for subview in subviews {
|
||||
subview.removeFromSuperview()
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Action Event
|
||||
@objc func tapGestureAction(tapGesture: UITapGestureRecognizer) {
|
||||
if self.delegate != nil && ((self.delegate?.responds(to: Selector(("tapGestureAction")))) != nil) {
|
||||
self.delegate?.tapGestureAction(tapGesture: tapGesture)
|
||||
}
|
||||
}
|
||||
|
||||
@objc func panGestureAction(panGesture: UIPanGestureRecognizer) {
|
||||
if self.delegate != nil && ((self.delegate?.responds(to: Selector(("panGestureAction")))) != nil) {
|
||||
self.delegate?.panGestureAction(panGesture: panGesture)
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Register TUICallState Observer && Update UI
|
||||
func registerObserverState() {
|
||||
registerCallStatusObserver()
|
||||
registerCallTimeObserver()
|
||||
}
|
||||
|
||||
func registerCallStatusObserver() {
|
||||
viewModel.selfCallStatus.addObserver(selfCallStatusObserver, closure: { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
self.updateUI()
|
||||
})
|
||||
}
|
||||
|
||||
func registerCallTimeObserver() {
|
||||
viewModel.callTime.addObserver(callTimeObserver, closure: { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
DispatchCallKitMainAsyncSafe {
|
||||
self.timerLabel.text = self.viewModel.getCallTimeString()
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func userVideoAvailableChange() {
|
||||
if viewModel.remoteUserList.value.first != nil {
|
||||
viewModel.remoteUserList.value.first?.videoAvailable.addObserver(remoteUserListObserver, closure: { [weak self] newValue, _ in
|
||||
guard let self = self else { return }
|
||||
self.updateUI()
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: Update UI
|
||||
func updateUI() {
|
||||
cleanView()
|
||||
|
||||
if viewModel.mediaType.value == .audio {
|
||||
updateSingleAudioUI()
|
||||
} else if viewModel.mediaType.value == .video {
|
||||
updateSingleVideoUI()
|
||||
}
|
||||
}
|
||||
|
||||
func updateSingleAudioUI() {
|
||||
containerView.backgroundColor = UIColor.t_colorWithHexString(color: "FFFFFF")
|
||||
|
||||
if viewModel.selfCallStatus.value == .waiting {
|
||||
setSingleAudioWaitingUI()
|
||||
} else if viewModel.selfCallStatus.value == .accept {
|
||||
setSingleAudioAcceptUI()
|
||||
}
|
||||
}
|
||||
|
||||
func updateSingleVideoUI() {
|
||||
containerView.backgroundColor = UIColor.t_colorWithHexString(color: "303132")
|
||||
|
||||
if viewModel.selfCallStatus.value == .waiting {
|
||||
setSingleVideoWaitingUI()
|
||||
} else if viewModel.selfCallStatus.value == .accept {
|
||||
guard let remoteUser = viewModel.remoteUserList.value.first else { return }
|
||||
|
||||
if remoteUser.videoAvailable.value == true {
|
||||
setSingleVideoAcceptUI()
|
||||
} else {
|
||||
setSingleVideoAcceptWithUnavailableUI()
|
||||
}
|
||||
}
|
||||
|
||||
userVideoAvailableChange()
|
||||
}
|
||||
|
||||
func setSingleAudioWaitingUI() {
|
||||
audioContainerView.isHidden = false
|
||||
imageView.isHidden = false
|
||||
audioDescribeLabel.isHidden = false
|
||||
}
|
||||
|
||||
func setSingleAudioAcceptUI() {
|
||||
audioContainerView.isHidden = false
|
||||
imageView.isHidden = false
|
||||
timerLabel.isHidden = false
|
||||
}
|
||||
|
||||
func setSingleVideoWaitingUI() {
|
||||
localPreView.frame = kFloatingWindowVideoViewRect
|
||||
videoDescribeLabel.isHidden = false
|
||||
containerView.addSubview(localPreView)
|
||||
containerView.bringSubviewToFront(videoDescribeLabel)
|
||||
}
|
||||
|
||||
func setSingleVideoAcceptUI() {
|
||||
remotePreView.frame = kFloatingWindowVideoViewRect
|
||||
remotePreView.isHidden = false
|
||||
containerView.addSubview(remotePreView)
|
||||
guard let remoteUser = self.viewModel.remoteUserList.value.first else { return }
|
||||
viewModel.startRemoteView(user: remoteUser, videoView: remotePreView)
|
||||
}
|
||||
|
||||
func setSingleVideoAcceptWithUnavailableUI() {
|
||||
avatarImageView.isHidden = false
|
||||
remotePreView.isHidden = true
|
||||
containerView.bringSubviewToFront(avatarImageView)
|
||||
|
||||
guard let remoteUser = viewModel.remoteUserList.value.first else { return }
|
||||
let userIcon: UIImage? = TUICallKitCommon.getBundleImage(name: "default_user_icon")
|
||||
|
||||
if remoteUser.avatar.value == "" {
|
||||
guard let image = userIcon else { return }
|
||||
avatarImageView.image = image
|
||||
} else {
|
||||
avatarImageView.sd_setImage(with: URL(string: remoteUser.avatar.value), placeholderImage: userIcon)
|
||||
}
|
||||
}
|
||||
|
||||
func cleanView() {
|
||||
avatarImageView.isHidden = true
|
||||
audioContainerView.isHidden = true
|
||||
imageView.isHidden = true
|
||||
audioDescribeLabel.isHidden = true
|
||||
timerLabel.isHidden = true
|
||||
videoDescribeLabel.isHidden = true
|
||||
}
|
||||
|
||||
}
|
||||
Reference in New Issue
Block a user