增加换肤功能

This commit is contained in:
启星
2025-08-14 10:07:49 +08:00
parent f6964c1e89
commit 4f9318d98e
8789 changed files with 978530 additions and 2 deletions

View File

@@ -0,0 +1,78 @@
//
// AudioAndVideoCalleeWaitingView.swift
// TUICallKit
//
// Created by vincepzhang on 2023/2/14.
//
import Foundation
import TUICallEngine
class AudioAndVideoCalleeWaitingView: UIView {
lazy var acceptBtn: BaseControlButton = {
weak var weakSelf = self
let acceptBtn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: "TUICallKit.answer") ?? "",
imageSize: kBtnSmallSize) { sender in
weakSelf?.acceptTouchEvent(sender: sender)
}
if let image = TUICallKitCommon.getBundleImage(name: "icon_dialing") {
acceptBtn.updateImage(image: image)
}
acceptBtn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return acceptBtn
}()
lazy var rejectBtn: BaseControlButton = {
weak var weakSelf = self
let rejectBtn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: "TUICallKit.decline") ?? "",
imageSize: kBtnSmallSize) { sender in
weakSelf?.rejectTouchEvent(sender: sender)
}
if let image = TUICallKitCommon.getBundleImage(name: "icon_hangup") {
rejectBtn.updateImage(image: image)
}
rejectBtn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return rejectBtn
}()
// MARK: UI Specification Processing
private var isViewReady: Bool = false
override func didMoveToWindow() {
super.didMoveToWindow()
if isViewReady { return }
constructViewHierarchy()
activateConstraints()
isViewReady = true
}
func constructViewHierarchy() {
addSubview(rejectBtn)
addSubview(acceptBtn)
}
func activateConstraints() {
rejectBtn.snp.makeConstraints { make in
make.centerX.equalTo(self).offset(TUICoreDefineConvert.getIsRTL() ? 80.scaleWidth() : -80.scaleWidth())
make.top.bottom.equalTo(self)
make.size.equalTo(kControlBtnSize)
}
acceptBtn.snp.makeConstraints { make in
make.centerX.equalTo(self).offset(TUICoreDefineConvert.getIsRTL() ? -80.scaleWidth() : 80.scaleWidth())
make.top.bottom.equalTo(self)
make.size.equalTo(kControlBtnSize)
}
}
// MARK: Event Action
func rejectTouchEvent(sender: UIButton) {
CallEngineManager.instance.reject()
}
func acceptTouchEvent(sender: UIButton) {
CallEngineManager.instance.accept()
}
}

View File

@@ -0,0 +1,153 @@
//
// AudioCallerWaitingAndAcceptedView.swift
// TUICallKit
//
// Created by vincepzhang on 2023/2/14.
//
import Foundation
import UIKit
import TUICallEngine
class AudioCallerWaitingAndAcceptedView : UIView {
let isMicMuteObserver = Observer()
let audioDeviceObserver = Observer()
lazy var muteMicBtn: BaseControlButton = {
let titleKey = TUICallState.instance.isMicMute.value ? "TUICallKit.muted" : "TUICallKit.unmuted"
let muteAudioBtn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: titleKey) ?? "",
imageSize: kBtnSmallSize, buttonAction: { [weak self] sender in
guard let self = self else { return }
self.muteMicEvent(sender: sender)
})
let imageName = TUICallState.instance.isMicMute.value ? "icon_mute_on" : "icon_mute"
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
muteAudioBtn.updateImage(image: image)
}
muteAudioBtn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return muteAudioBtn
}()
lazy var hangupBtn: BaseControlButton = {
let hangupBtn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: "TUICallKit.hangup") ?? "",
imageSize: kBtnSmallSize, buttonAction: { [weak self] sender in
guard let self = self else { return }
self.hangupEvent(sender: sender)
})
if let image = TUICallKitCommon.getBundleImage(name: "icon_hangup") {
hangupBtn.updateImage(image: image)
}
hangupBtn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return hangupBtn
}()
lazy var changeSpeakerBtn: BaseControlButton = {
let titleKey = (TUICallState.instance.audioDevice.value == .speakerphone) ? "TUICallKit.speakerPhone" : "TUICallKit.earpiece"
let changeSpeakerBtn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: titleKey) ?? "",
imageSize: kBtnSmallSize, buttonAction: { [weak self] sender in
guard let self = self else { return }
self.changeSpeakerEvent(sender: sender)
})
let imageName = (TUICallState.instance.audioDevice.value == .speakerphone) ? "icon_handsfree_on" : "icon_handsfree"
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
changeSpeakerBtn.updateImage(image: image)
}
changeSpeakerBtn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return changeSpeakerBtn
}()
override init(frame: CGRect) {
super.init(frame: frame)
registerObserveState()
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
deinit {
TUICallState.instance.isMicMute.removeObserver(isMicMuteObserver)
TUICallState.instance.audioDevice.removeObserver(audioDeviceObserver)
}
// MARK: UI Specification Processing
private var isViewReady: Bool = false
override func didMoveToWindow() {
super.didMoveToWindow()
if isViewReady { return }
constructViewHierarchy()
activateConstraints()
isViewReady = true
}
func constructViewHierarchy() {
addSubview(muteMicBtn)
addSubview(hangupBtn)
addSubview(changeSpeakerBtn)
}
func activateConstraints() {
muteMicBtn.snp.makeConstraints { make in
make.centerX.equalTo(self).offset(TUICoreDefineConvert.getIsRTL() ? 110.scaleWidth() : -110.scaleWidth())
make.centerY.equalTo(hangupBtn)
make.size.equalTo(kControlBtnSize)
}
hangupBtn.snp.makeConstraints { make in
make.centerX.equalTo(self)
make.bottom.equalTo(self.snp.bottom)
make.size.equalTo(kControlBtnSize)
}
changeSpeakerBtn.snp.makeConstraints { make in
make.centerX.equalTo(self).offset(TUICoreDefineConvert.getIsRTL() ? -110.scaleWidth() : 110.scaleWidth())
make.centerY.equalTo(self.hangupBtn)
make.size.equalTo(kControlBtnSize)
}
}
// MARK: Action Event
func muteMicEvent(sender: UIButton) {
CallEngineManager.instance.muteMic()
}
func changeSpeakerEvent(sender: UIButton) {
CallEngineManager.instance.changeSpeaker()
}
func hangupEvent(sender: UIButton) {
CallEngineManager.instance.hangup()
}
// MARK: Register TUICallState Observer && Update UI
func registerObserveState() {
TUICallState.instance.isMicMute.addObserver(isMicMuteObserver, closure: { [weak self] newValue, _ in
guard let self = self else { return }
self.updateMuteAudioBtn(mute: newValue)
})
TUICallState.instance.audioDevice.addObserver(audioDeviceObserver, closure: { [weak self] newValue, _ in
guard let self = self else { return }
self.updateChangeSpeakerBtn(isSpeaker: newValue == .speakerphone)
})
}
func updateMuteAudioBtn(mute: Bool) {
muteMicBtn.updateTitle(title: TUICallKitLocalize(key: mute ? "TUICallKit.muted" : "TUICallKit.unmuted") ?? "")
if let image = TUICallKitCommon.getBundleImage(name: mute ? "icon_mute_on" : "icon_mute") {
muteMicBtn.updateImage(image: image)
}
}
func updateChangeSpeakerBtn(isSpeaker: Bool) {
changeSpeakerBtn.updateTitle(title: TUICallKitLocalize(key: isSpeaker ? "TUICallKit.speakerPhone" : "TUICallKit.earpiece") ?? "")
if let image = TUICallKitCommon.getBundleImage(name: isSpeaker ? "icon_handsfree_on" : "icon_handsfree") {
changeSpeakerBtn.updateImage(image: image)
}
}
}

View File

@@ -0,0 +1,388 @@
//
// GroupCallerAndCalleeAcceptedView.swift
// TUICallKit
//
// Created by noah on 2023/11/8.
//
import Foundation
import SnapKit
protocol GroupCallerAndCalleeAcceptedViewDelegate: AnyObject {
func showAnimation()
func restoreExpansion()
func handleTransform(animationScale: CGFloat)
}
let groupFunctionAnimationDuration = 0.3
let groupFunctionBaseControlBtnHeight = 60.scaleWidth() + 5.scaleHeight() + 20
let groupFunctionBottomHeight = Bottom_SafeHeight > 1 ? Bottom_SafeHeight : 8
let groupFunctionViewHeight = 22 + groupFunctionBaseControlBtnHeight + 20.scaleHeight() + 60.scaleWidth() + groupFunctionBottomHeight
let groupSmallFunctionViewHeight = 22 + 60.scaleWidth() + groupFunctionBottomHeight
class GroupCallerAndCalleeAcceptedView: UIView {
weak var delegate: GroupCallerAndCalleeAcceptedViewDelegate?
let isCameraOpenObserver = Observer()
let showLargeViewUserIdObserver = Observer()
let isMicMuteObserver = Observer()
let audioDeviceObserver = Observer()
var isShowLittleContainerView = false
var panGestureBeganY = 0.0
lazy var containerView: UIView = {
let containerView = UIView(frame: CGRect.zero)
containerView.backgroundColor = UIColor.t_colorWithHexString(color: "#4F586B")
containerView.alpha = 0.5
containerView.isUserInteractionEnabled = true
let panGesture = UIPanGestureRecognizer(target: self, action: #selector(handlePanGesture(_:)))
containerView.addGestureRecognizer(panGesture)
return containerView
}()
lazy var muteMicBtn: BaseControlButton = {
let titleKey = TUICallState.instance.isMicMute.value ? "TUICallKit.muted" : "TUICallKit.unmuted"
let btn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: titleKey) ?? "",
imageSize: kBtnSmallSize) { [weak self] sender in
guard let self = self else { return }
self.muteMicEvent(sender: sender)
}
let imageName = TUICallState.instance.isMicMute.value ? "icon_mute_on" : "icon_mute"
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
btn.updateImage(image: image)
}
btn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return btn
}()
lazy var closeCameraBtn: BaseControlButton = {
let titleKey = TUICallState.instance.isCameraOpen.value ? "TUICallKit.cameraOn" : "TUICallKit.cameraOff"
let btn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: titleKey) ?? "",
imageSize: kBtnSmallSize) { [weak self] sender in
guard let self = self else { return }
self.closeCameraTouchEvent(sender: sender)
}
if let image = TUICallKitCommon.getBundleImage(name: TUICallState.instance.isCameraOpen.value ? "icon_camera_on" : "icon_camera_off") {
btn.updateImage(image: image)
}
btn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return btn
}()
lazy var changeSpeakerBtn: BaseControlButton = {
let titleKey = (TUICallState.instance.audioDevice.value == .speakerphone) ? "TUICallKit.speakerPhone" : "TUICallKit.earpiece"
let btn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: titleKey) ?? "",
imageSize: kBtnSmallSize) { [weak self] sender in
guard let self = self else { return }
self.changeSpeakerEvent(sender: sender)
}
let imageName = (TUICallState.instance.audioDevice.value == .speakerphone) ? "icon_handsfree_on" : "icon_handsfree"
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
btn.updateImage(image: image)
}
btn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return btn
}()
lazy var hangupBtn: BaseControlButton = {
let btn = BaseControlButton.create(frame: CGRect.zero,
title: "",
imageSize: kBtnSmallSize) { [weak self] sender in
guard let self = self else { return }
self.hangupTouchEvent(sender: sender)
}
if let image = TUICallKitCommon.getBundleImage(name: "icon_hangup") {
btn.updateImage(image: image)
}
return btn
}()
lazy var matchBtn: UIButton = {
let btn = UIButton(type: .system)
btn.addTarget(self,action:#selector(matchTouchEvent(sender:)), for: .touchUpInside)
btn.setBackgroundImage(TUICallKitCommon.getBundleImage(name: "icon_match"), for: .normal)
return btn
}()
override init(frame: CGRect) {
super.init(frame: frame)
registerObserveState()
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
deinit {
TUICallState.instance.isCameraOpen.removeObserver(isCameraOpenObserver)
TUICallState.instance.showLargeViewUserId.removeObserver(showLargeViewUserIdObserver)
}
// MARK: UI Specification Processing
private var isViewReady: Bool = false
override func didMoveToWindow() {
super.didMoveToWindow()
if isViewReady { return }
constructViewHierarchy()
activateConstraints()
setContainerViewCorner()
let isHidden: Bool = (TUICallState.instance.showLargeViewUserId.value.count <= 1)
containerView.isHidden = isHidden
matchBtn.isHidden = isHidden
isViewReady = true
}
func constructViewHierarchy() {
addSubview(containerView)
addSubview(muteMicBtn)
addSubview(changeSpeakerBtn)
addSubview(closeCameraBtn)
addSubview(hangupBtn)
addSubview(matchBtn)
}
func activateConstraints() {
let top = 22.0
containerView.frame = CGRect(origin: CGPoint(x: 0, y: 0), size: CGSize(width: frame.size.width, height: groupFunctionViewHeight))
muteMicBtn.snp.makeConstraints { make in
make.centerX.equalTo(self).offset(TUICoreDefineConvert.getIsRTL() ? 110.scaleWidth() : -110.scaleWidth())
make.centerY.equalTo(changeSpeakerBtn)
make.width.height.equalTo(60.scaleWidth())
}
changeSpeakerBtn.snp.makeConstraints { make in
make.top.equalTo(self).offset(top)
make.centerX.equalTo(self)
make.width.height.equalTo(60.scaleWidth())
}
closeCameraBtn.snp.makeConstraints { make in
make.centerX.equalTo(self).offset(TUICoreDefineConvert.getIsRTL() ? -110.scaleWidth() : 110.scaleWidth())
make.centerY.equalTo(self.changeSpeakerBtn)
make.width.height.equalTo(60.scaleWidth())
}
hangupBtn.snp.makeConstraints { make in
make.top.equalTo(changeSpeakerBtn.snp.bottom).offset(5.scaleHeight() + 20 + 20.scaleHeight())
make.centerX.equalTo(self)
make.width.height.equalTo(60.scaleWidth())
}
matchBtn.snp.makeConstraints { make in
make.centerY.equalTo(hangupBtn)
make.leading.width.height.equalTo(30.scaleWidth())
}
}
// MARK: Register TUICallState Observer && Update UI
func registerObserveState() {
TUICallState.instance.isCameraOpen.addObserver(isCameraOpenObserver, closure: { [weak self] newValue, _ in
guard let self = self else { return }
self.updateCloseCameraBtn(open: newValue)
})
TUICallState.instance.showLargeViewUserId.addObserver(showLargeViewUserIdObserver) { [weak self] newValue, _ in
guard let self = self else { return }
if newValue.count > 1 {
self.showAnimation()
self.containerView.isHidden = false
self.matchBtn.isHidden = false
} else {
self.restoreExpansion()
}
}
TUICallState.instance.isMicMute.addObserver(isMicMuteObserver, closure: { [weak self] newValue, _ in
guard let self = self else { return }
self.updateMuteAudioBtn(mute: newValue)
})
TUICallState.instance.audioDevice.addObserver(audioDeviceObserver, closure: { [weak self] newValue, _ in
guard let self = self else { return }
self.updateChangeSpeakerBtn(isSpeaker: newValue == .speakerphone)
})
}
// MARK: Action Event
func muteMicEvent(sender: UIButton) {
CallEngineManager.instance.muteMic()
updateMuteAudioBtn(mute: TUICallState.instance.isMicMute.value == true)
}
func closeCameraTouchEvent(sender: UIButton) {
updateCloseCameraBtn(open: TUICallState.instance.isCameraOpen.value != true)
if TUICallState.instance.isCameraOpen.value == true {
CallEngineManager.instance.closeCamera()
} else {
guard let videoViewEntity = VideoFactory.instance.viewMap[TUICallState.instance.selfUser.value.id.value] else { return }
CallEngineManager.instance.openCamera(videoView: videoViewEntity.videoView)
}
}
func changeSpeakerEvent(sender: UIButton) {
CallEngineManager.instance.changeSpeaker()
updateChangeSpeakerBtn(isSpeaker: TUICallState.instance.audioDevice.value == .speakerphone)
}
@objc func hangupTouchEvent(sender: UIButton) {
CallEngineManager.instance.hangup()
}
@objc func matchTouchEvent(sender: UIButton) {
if isShowLittleContainerView {
restoreExpansion()
} else {
showAnimation()
}
}
@objc func handlePanGesture(_ gesture: UIPanGestureRecognizer) {
let translation = gesture.translation(in: containerView)
let scale = ((translation.y - panGestureBeganY) / 105)
if gesture.state == .began {
panGestureBeganY = translation.y
} else if gesture.state == .changed {
if scale > 0 && !isShowLittleContainerView {
// Swipe down
handleTransform(animationScale: (scale > 1) ? 1 : scale)
} else if scale < 0 && isShowLittleContainerView {
// Swipe up
handleTransform(animationScale: (scale < -1) ? 0 : (1 + scale))
}
} else if gesture.state == .ended {
if scale > 0 {
// Swipe down
if (scale > 0.5) {
showAnimation()
} else {
restoreExpansion()
}
} else {
// Swipe up
if (scale < -0.5) {
restoreExpansion()
} else {
showAnimation()
}
}
}
}
func showAnimation() {
isShowLittleContainerView = true
delegate?.showAnimation()
UIView.animate(withDuration: groupFunctionAnimationDuration) {
self.handleTransform(animationScale: 1)
}
let rotationAnimation = CABasicAnimation(keyPath: "transform.rotation.z")
rotationAnimation.toValue = NSNumber(value: Double.pi)
rotationAnimation.duration = groupFunctionAnimationDuration
rotationAnimation.fillMode = .forwards
rotationAnimation.isRemovedOnCompletion = false
self.matchBtn.layer.add(rotationAnimation, forKey: "rotationAnimation")
}
func restoreExpansion() {
isShowLittleContainerView = false
delegate?.restoreExpansion()
UIView.animate(withDuration: groupFunctionAnimationDuration, animations: {
self.muteMicBtn.titleLabel.alpha = 1
self.changeSpeakerBtn.titleLabel.alpha = 1
self.closeCameraBtn.titleLabel.alpha = 1
self.muteMicBtn.titleLabel.transform = CGAffineTransform.identity
self.changeSpeakerBtn.titleLabel.transform = CGAffineTransform.identity
self.closeCameraBtn.titleLabel.transform = CGAffineTransform.identity
self.muteMicBtn.button.transform = CGAffineTransform.identity
self.changeSpeakerBtn.button.transform = CGAffineTransform.identity
self.closeCameraBtn.button.transform = CGAffineTransform.identity
self.hangupBtn.button.transform = CGAffineTransform.identity
self.muteMicBtn.transform = CGAffineTransform.identity
self.changeSpeakerBtn.transform = CGAffineTransform.identity
self.closeCameraBtn.transform = CGAffineTransform.identity
self.hangupBtn.transform = CGAffineTransform.identity
self.matchBtn.transform = CGAffineTransform.identity
})
let rotationAnimation = CABasicAnimation(keyPath: "transform.rotation.z")
rotationAnimation.toValue = NSNumber(value: 0.0)
rotationAnimation.duration = groupFunctionAnimationDuration
rotationAnimation.fillMode = .forwards
rotationAnimation.isRemovedOnCompletion = true
self.matchBtn.layer.add(rotationAnimation, forKey: "rotationAnimation")
}
func handleTransform(animationScale: CGFloat) {
delegate?.handleTransform(animationScale: animationScale)
let alpha: CGFloat = 1 - 1 * animationScale
let scale: CGFloat = 1 - 1 / 3 * animationScale
let muteMicX = ((TUICoreDefineConvert.getIsRTL() ? -28 : 28) * animationScale).scaleWidth()
let changeSpeakerX = ((TUICoreDefineConvert.getIsRTL() ? 2 : -2) * animationScale).scaleWidth()
let closeCameraX = ((TUICoreDefineConvert.getIsRTL() ? 28 : -28) * animationScale).scaleWidth()
let hangupX = ((TUICoreDefineConvert.getIsRTL() ? -138 : 138) * animationScale).scaleWidth()
let hangupTranslationY = -(groupFunctionBaseControlBtnHeight + 20.scaleHeight()) * animationScale
let titleLabelTranslationY = -12.scaleWidth() * animationScale
self.muteMicBtn.titleLabel.alpha = alpha
self.changeSpeakerBtn.titleLabel.alpha = alpha
self.closeCameraBtn.titleLabel.alpha = alpha
self.muteMicBtn.titleLabel.transform = CGAffineTransform(translationX: 0, y: titleLabelTranslationY)
self.changeSpeakerBtn.titleLabel.transform = CGAffineTransform(translationX: 0, y: titleLabelTranslationY)
self.closeCameraBtn.titleLabel.transform = CGAffineTransform(translationX: 0, y: titleLabelTranslationY)
self.muteMicBtn.button.transform = CGAffineTransform(scaleX: scale, y: scale)
self.changeSpeakerBtn.button.transform = CGAffineTransform(scaleX: scale, y: scale)
self.closeCameraBtn.button.transform = CGAffineTransform(scaleX: scale, y: scale)
self.hangupBtn.button.transform = CGAffineTransform(scaleX: scale, y: scale)
self.muteMicBtn.transform = CGAffineTransform(translationX: muteMicX, y: 0)
self.changeSpeakerBtn.transform = CGAffineTransform(translationX: changeSpeakerX, y: 0)
self.closeCameraBtn.transform = CGAffineTransform(translationX: closeCameraX, y: 0)
self.hangupBtn.transform = CGAffineTransform(translationX: hangupX, y: hangupTranslationY)
self.matchBtn.transform = CGAffineTransform(translationX: 0, y: hangupTranslationY)
}
// MARK: Update UI
func updateMuteAudioBtn(mute: Bool) {
muteMicBtn.updateTitle(title: TUICallKitLocalize(key: mute ? "TUICallKit.muted" : "TUICallKit.unmuted") ?? "")
if let image = TUICallKitCommon.getBundleImage(name: mute ? "icon_mute_on" : "icon_mute") {
muteMicBtn.updateImage(image: image)
}
}
func updateChangeSpeakerBtn(isSpeaker: Bool) {
changeSpeakerBtn.updateTitle(title: TUICallKitLocalize(key: isSpeaker ? "TUICallKit.speakerPhone" : "TUICallKit.earpiece") ?? "")
if let image = TUICallKitCommon.getBundleImage(name: isSpeaker ? "icon_handsfree_on" : "icon_handsfree") {
changeSpeakerBtn.updateImage(image: image)
}
}
func updateCloseCameraBtn(open: Bool) {
closeCameraBtn.updateTitle(title: TUICallKitLocalize(key: open ? "TUICallKit.cameraOn" : "TUICallKit.cameraOff") ?? "")
if let image = TUICallKitCommon.getBundleImage(name: open ? "icon_camera_on" : "icon_camera_off") {
closeCameraBtn.updateImage(image: image)
}
}
func setContainerViewCorner() {
let maskLayer = CAShapeLayer()
let path = UIBezierPath(roundedRect: containerView.bounds,
byRoundingCorners: [.topLeft, .topRight],
cornerRadii: CGSize(width: 20, height: 20))
maskLayer.path = path.cgPath
containerView.layer.mask = maskLayer
}
}

View File

@@ -0,0 +1,233 @@
//
// VideoCallerAndCalleeAcceptedView.swift
// TUICallKit
//
// Created by vincepzhang on 2023/2/14.
//
import Foundation
import SnapKit
class VideoCallerAndCalleeAcceptedView: UIView {
let isCameraOpenObserver = Observer()
let isMicMuteObserver = Observer()
let audioDeviceObserver = Observer()
lazy var muteMicBtn: BaseControlButton = {
let titleKey = TUICallState.instance.isMicMute.value ? "TUICallKit.muted" : "TUICallKit.unmuted"
let btn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: titleKey) ?? "",
imageSize: kBtnSmallSize) { [weak self] sender in
guard let self = self else { return }
self.muteMicEvent(sender: sender)
}
let imageName = TUICallState.instance.isMicMute.value ? "icon_mute_on" : "icon_mute"
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
btn.updateImage(image: image)
}
btn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return btn
}()
lazy var closeCameraBtn: BaseControlButton = {
let titleKey = TUICallState.instance.isCameraOpen.value ? "TUICallKit.cameraOn" : "TUICallKit.cameraOff"
let btn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: titleKey) ?? "",
imageSize: kBtnSmallSize) { [weak self] sender in
guard let self = self else { return }
self.closeCameraTouchEvent(sender: sender)
}
let imageName = TUICallState.instance.isCameraOpen.value ? "icon_camera_on" : "icon_camera_off"
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
btn.updateImage(image: image)
}
btn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return btn
}()
lazy var changeSpeakerBtn: BaseControlButton = {
let titleKey = (TUICallState.instance.audioDevice.value == .speakerphone) ? "TUICallKit.speakerPhone" : "TUICallKit.earpiece"
let btn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: titleKey) ?? "",
imageSize: kBtnSmallSize) { [weak self] sender in
guard let self = self else { return }
self.changeSpeakerEvent(sender: sender)
}
let imageName = (TUICallState.instance.audioDevice.value == .speakerphone) ? "icon_handsfree_on" : "icon_handsfree"
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
btn.updateImage(image: image)
}
btn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return btn
}()
lazy var hangupBtn: UIButton = {
let btn = UIButton(type: .system)
btn.addTarget(self,action:#selector(hangupTouchEvent(sender:)), for: .touchUpInside)
btn.setBackgroundImage(TUICallKitCommon.getBundleImage(name: "icon_hangup"), for: .normal)
return btn
}()
lazy var switchCameraBtn: UIButton = {
let btn = UIButton(type: .system)
if let image = TUICallKitCommon.getBundleImage(name: "switch_camera") {
btn.setBackgroundImage(image, for: .normal)
}
btn.addTarget(self, action: #selector(switchCameraTouchEvent(sender: )), for: .touchUpInside)
return btn
}()
lazy var virtualBackgroundButton: UIButton = {
let btn = UIButton(type: .system)
if let image = TUICallKitCommon.getBundleImage(name: "virtual_background") {
btn.setBackgroundImage(image, for: .normal)
}
btn.addTarget(self, action: #selector(virtualBackgroundTouchEvent(sender: )), for: .touchUpInside)
return btn
}()
override init(frame: CGRect) {
super.init(frame: frame)
registerObserveState()
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
// MARK: UI Specification Processing
private var isViewReady: Bool = false
override func didMoveToWindow() {
super.didMoveToWindow()
if isViewReady { return }
constructViewHierarchy()
activateConstraints()
isViewReady = true
}
func constructViewHierarchy() {
addSubview(muteMicBtn)
addSubview(changeSpeakerBtn)
addSubview(closeCameraBtn)
addSubview(hangupBtn)
addSubview(switchCameraBtn)
if TUICallState.instance.showVirtualBackgroundButton {
addSubview(virtualBackgroundButton)
}
}
func activateConstraints() {
muteMicBtn.snp.makeConstraints { make in
make.centerX.equalTo(self).offset(TUICoreDefineConvert.getIsRTL() ? 110.scaleWidth() : -110.scaleWidth())
make.centerY.equalTo(changeSpeakerBtn)
make.size.equalTo(kControlBtnSize)
}
changeSpeakerBtn.snp.makeConstraints { make in
make.centerX.equalTo(self)
make.bottom.equalTo(hangupBtn.snp.top).offset(-20.scaleHeight())
make.size.equalTo(kControlBtnSize)
}
closeCameraBtn.snp.makeConstraints { make in
make.centerX.equalTo(self).offset(TUICoreDefineConvert.getIsRTL() ? -110.scaleWidth() : 110.scaleWidth())
make.centerY.equalTo(self.changeSpeakerBtn)
make.size.equalTo(kControlBtnSize)
}
hangupBtn.snp.makeConstraints { make in
make.centerX.equalTo(self)
make.bottom.equalTo(snp.bottom)
make.width.height.equalTo(60.scaleWidth())
}
switchCameraBtn.snp.makeConstraints { make in
make.centerY.equalTo(hangupBtn)
make.leading.equalTo(hangupBtn.snp.trailing).offset(40.scaleWidth())
make.size.equalTo(CGSize(width: 28.scaleWidth(), height: 28.scaleWidth()))
}
if TUICallState.instance.showVirtualBackgroundButton {
virtualBackgroundButton.snp.makeConstraints { make in
make.centerY.equalTo(hangupBtn)
make.trailing.equalTo(hangupBtn.snp.leading).offset(-40.scaleWidth())
make.size.equalTo(CGSize(width: 28.scaleWidth(), height: 28.scaleWidth()))
}
}
}
// MARK: Action Event
func muteMicEvent(sender: UIButton) {
CallEngineManager.instance.muteMic()
updateMuteAudioBtn(mute: TUICallState.instance.isMicMute.value == true)
}
func closeCameraTouchEvent(sender: UIButton) {
updateCloseCameraBtn(open: TUICallState.instance.isCameraOpen.value != true)
if TUICallState.instance.isCameraOpen.value == true {
CallEngineManager.instance.closeCamera()
virtualBackgroundButton.isHidden = true
switchCameraBtn.isHidden = true
} else {
guard let videoViewEntity = VideoFactory.instance.viewMap[TUICallState.instance.selfUser.value.id.value] else { return }
CallEngineManager.instance.openCamera(videoView: videoViewEntity.videoView)
virtualBackgroundButton.isHidden = false
switchCameraBtn.isHidden = false
}
}
func changeSpeakerEvent(sender: UIButton) {
CallEngineManager.instance.changeSpeaker()
updateChangeSpeakerBtn(isSpeaker: TUICallState.instance.audioDevice.value == .speakerphone)
}
@objc func hangupTouchEvent(sender: UIButton) {
CallEngineManager.instance.hangup()
}
@objc func switchCameraTouchEvent(sender: UIButton) {
CallEngineManager.instance.switchCamera()
}
@objc func virtualBackgroundTouchEvent(sender: UIButton) {
CallEngineManager.instance.setBlurBackground()
}
// MARK: Update UI
func updateMuteAudioBtn(mute: Bool) {
muteMicBtn.updateTitle(title: TUICallKitLocalize(key: mute ? "TUICallKit.muted" : "TUICallKit.unmuted") ?? "")
if let image = TUICallKitCommon.getBundleImage(name: mute ? "icon_mute_on" : "icon_mute") {
muteMicBtn.updateImage(image: image)
}
}
func updateChangeSpeakerBtn(isSpeaker: Bool) {
changeSpeakerBtn.updateTitle(title: TUICallKitLocalize(key: isSpeaker ? "TUICallKit.speakerPhone" : "TUICallKit.earpiece") ?? "")
if let image = TUICallKitCommon.getBundleImage(name: isSpeaker ? "icon_handsfree_on" : "icon_handsfree") {
changeSpeakerBtn.updateImage(image: image)
}
}
func updateCloseCameraBtn(open: Bool) {
closeCameraBtn.updateTitle(title: TUICallKitLocalize(key: open ? "TUICallKit.cameraOn" : "TUICallKit.cameraOff") ?? "")
if let image = TUICallKitCommon.getBundleImage(name: open ? "icon_camera_on" : "icon_camera_off") {
closeCameraBtn.updateImage(image: image)
}
}
func registerObserveState() {
TUICallState.instance.isCameraOpen.addObserver(isCameraOpenObserver) { [weak self] newValue, _ in
guard let self = self else { return }
self.updateCloseCameraBtn(open: TUICallState.instance.isCameraOpen.value)
}
TUICallState.instance.isMicMute.addObserver(isMicMuteObserver, closure: { [weak self] newValue, _ in
guard let self = self else { return }
self.updateMuteAudioBtn(mute: newValue)
})
TUICallState.instance.audioDevice.addObserver(audioDeviceObserver, closure: { [weak self] newValue, _ in
guard let self = self else { return }
self.updateChangeSpeakerBtn(isSpeaker: newValue == .speakerphone)
})
}
}

View File

@@ -0,0 +1,187 @@
//
// VideoCallerWaitingView.swift
// TUICallKit
//
// Created by vincepzhang on 2023/2/14.
//
import Foundation
class VideoCallerWaitingView: UIView {
let enableBlurBackgroundObserver = Observer()
lazy var switchCameraBtn: BaseControlButton = {
let btn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: "TUICallKit.switchCamera") ?? "",
imageSize: kBtnLargeSize) { [weak self] sender in
guard let self = self else { return }
self.switchCameraTouchEvent(sender: sender)
}
if let image = TUICallKitCommon.getBundleImage(name: "icon_big_switch_camera") {
btn.updateImage(image: image)
}
btn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return btn
}()
lazy var virtualBackgroundButton: BaseControlButton = {
let btn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: "TUICallKit.blurBackground") ?? "",
imageSize: kBtnLargeSize) { [weak self] sender in
guard let self = self else { return }
self.virtualBackgroundTouchEvent(sender: sender)
}
let imageName = TUICallState.instance.enableBlurBackground.value ? "icon_big_virtual_background_on" : "icon_big_virtual_background_off"
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
btn.updateImage(image: image)
}
btn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return btn
}()
lazy var closeCameraBtn: BaseControlButton = {
let titleKey = TUICallState.instance.isCameraOpen.value ? "TUICallKit.cameraOn" : "TUICallKit.cameraOff"
let btn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: titleKey) ?? "",
imageSize: kBtnSmallSize) { [weak self] sender in
guard let self = self else { return }
self.closeCameraTouchEvent(sender: sender)
}
let imageName = TUICallState.instance.isCameraOpen.value ? "icon_camera_on" : "icon_camera_off"
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
btn.updateImage(image: image)
}
btn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return btn
}()
lazy var hangupBtn: BaseControlButton = {
weak var weakSelf = self
let btn = BaseControlButton.create(frame: CGRect.zero,
title: TUICallKitLocalize(key: "TUICallKit.hangup") ?? "",
imageSize: kBtnLargeSize) { [weak self] sender in
guard let self = self else { return }
self.hangupTouchEvent(sender: sender)
}
if let image = TUICallKitCommon.getBundleImage(name: "icon_hangup") {
btn.updateImage(image: image)
}
btn.updateTitleColor(titleColor: UIColor.t_colorWithHexString(color: "#D5E0F2"))
return btn
}()
override init(frame: CGRect) {
super.init(frame: frame)
registerObserveState()
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
deinit {
if TUICallState.instance.showVirtualBackgroundButton {
TUICallState.instance.selfUser.removeObserver(enableBlurBackgroundObserver)
}
}
// MARK: UI Specification Processing
private var isViewReady: Bool = false
override func didMoveToWindow() {
super.didMoveToWindow()
if isViewReady { return }
constructViewHierarchy()
activateConstraints()
isViewReady = true
}
func constructViewHierarchy() {
addSubview(switchCameraBtn)
addSubview(closeCameraBtn)
addSubview(hangupBtn)
if TUICallState.instance.showVirtualBackgroundButton {
addSubview(virtualBackgroundButton)
}
}
func activateConstraints() {
if TUICallState.instance.showVirtualBackgroundButton {
virtualBackgroundButton.snp.makeConstraints { make in
make.top.centerX.equalTo(self)
make.bottom.equalTo(hangupBtn.snp.top).offset(-8.scaleWidth())
make.size.equalTo(kControlBtnSize)
}
}
switchCameraBtn.snp.makeConstraints { make in
make.centerY.equalTo(TUICallState.instance.showVirtualBackgroundButton ? virtualBackgroundButton : hangupBtn)
make.centerX.equalTo(self).offset(TUICoreDefineConvert.getIsRTL() ? 110.scaleWidth() : -110.scaleWidth())
make.size.equalTo(kControlBtnSize)
}
hangupBtn.snp.makeConstraints { make in
if !TUICallState.instance.showVirtualBackgroundButton {
make.top.equalTo(self)
}
make.centerX.bottom.equalTo(self)
make.size.equalTo(kControlBtnSize)
}
closeCameraBtn.snp.makeConstraints { make in
make.centerY.equalTo(TUICallState.instance.showVirtualBackgroundButton ? virtualBackgroundButton : hangupBtn)
make.centerX.equalTo(self).offset(TUICoreDefineConvert.getIsRTL() ? -110.scaleWidth() : 110.scaleWidth())
make.size.equalTo(kControlBtnSize)
}
}
// MARK: Action Event
func closeCameraTouchEvent(sender: UIButton) {
updateCloseCameraBtn(open: TUICallState.instance.isCameraOpen.value != true)
if TUICallState.instance.isCameraOpen.value == true {
CallEngineManager.instance.closeCamera()
virtualBackgroundButton.button.isEnabled = false
switchCameraBtn.button.isEnabled = false
} else {
guard let videoViewEntity = VideoFactory.instance.viewMap[TUICallState.instance.selfUser.value.id.value] else { return }
CallEngineManager.instance.openCamera(videoView: videoViewEntity.videoView)
virtualBackgroundButton.button.isEnabled = true
switchCameraBtn.button.isEnabled = true
}
}
func updateCloseCameraBtn(open: Bool) {
closeCameraBtn.updateTitle(title: TUICallKitLocalize(key: open ? "TUICallKit.cameraOn" : "TUICallKit.cameraOff") ?? "")
if let image = TUICallKitCommon.getBundleImage(name: open ? "icon_camera_on" : "icon_camera_off") {
closeCameraBtn.updateImage(image: image)
}
}
func hangupTouchEvent(sender: UIButton) {
CallEngineManager.instance.hangup()
}
func switchCameraTouchEvent(sender: UIButton ) {
CallEngineManager.instance.switchCamera()
}
func virtualBackgroundTouchEvent(sender: UIButton) {
CallEngineManager.instance.setBlurBackground()
}
// MARK: Register TUICallState Observer && Update UI
func registerObserveState() {
if TUICallState.instance.showVirtualBackgroundButton {
TUICallState.instance.enableBlurBackground.addObserver(enableBlurBackgroundObserver, closure: { [weak self] newValue, _ in
guard let self = self else { return }
self.updateVirtualBackgroundButton()
})
}
}
func updateVirtualBackgroundButton() {
let imageName = TUICallState.instance.enableBlurBackground.value ? "icon_big_virtual_background_on" : "icon_big_virtual_background_off"
if let image = TUICallKitCommon.getBundleImage(name: imageName) {
virtualBackgroundButton.updateImage(image: image)
}
}
}