mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-06-16 05:55:20 +00:00
Voice Chat UI improvements
This commit is contained in:
parent
588f2be653
commit
9a5e3867d6
@ -57,6 +57,7 @@ swift_library(
|
|||||||
"//submodules/GridMessageSelectionNode:GridMessageSelectionNode",
|
"//submodules/GridMessageSelectionNode:GridMessageSelectionNode",
|
||||||
"//submodules/ChatListFilterSettingsHeaderItem:ChatListFilterSettingsHeaderItem",
|
"//submodules/ChatListFilterSettingsHeaderItem:ChatListFilterSettingsHeaderItem",
|
||||||
"//submodules/TelegramStringFormatting:TelegramStringFormatting",
|
"//submodules/TelegramStringFormatting:TelegramStringFormatting",
|
||||||
|
"//submodules/TelegramCallsUI:TelegramCallsUI",
|
||||||
],
|
],
|
||||||
visibility = [
|
visibility = [
|
||||||
"//visibility:public",
|
"//visibility:public",
|
||||||
|
@ -23,6 +23,7 @@ import AppBundle
|
|||||||
import LocalizedPeerData
|
import LocalizedPeerData
|
||||||
import TelegramIntents
|
import TelegramIntents
|
||||||
import TooltipUI
|
import TooltipUI
|
||||||
|
import TelegramCallsUI
|
||||||
|
|
||||||
private func fixListNodeScrolling(_ listNode: ListView, searchNode: NavigationBarSearchContentNode) -> Bool {
|
private func fixListNodeScrolling(_ listNode: ListView, searchNode: NavigationBarSearchContentNode) -> Bool {
|
||||||
if listNode.scroller.isDragging {
|
if listNode.scroller.isDragging {
|
||||||
@ -1727,6 +1728,20 @@ public class ChatListControllerImpl: TelegramBaseController, ChatListController
|
|||||||
|
|
||||||
(strongSelf.parent as? TabBarController)?.updateIsTabBarHidden(true, transition: .animated(duration: 0.4, curve: .spring))
|
(strongSelf.parent as? TabBarController)?.updateIsTabBarHidden(true, transition: .animated(duration: 0.4, curve: .spring))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
if let navigationController = self.navigationController as? NavigationController {
|
||||||
|
var voiceChatOverlayController: VoiceChatOverlayController?
|
||||||
|
for controller in navigationController.globalOverlayControllers {
|
||||||
|
if let controller = controller as? VoiceChatOverlayController {
|
||||||
|
voiceChatOverlayController = controller
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if let controller = voiceChatOverlayController {
|
||||||
|
controller.update(hidden: true, slide: true, animated: true)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1769,6 +1784,20 @@ public class ChatListControllerImpl: TelegramBaseController, ChatListController
|
|||||||
self.tabContainerNode.layer.animatePosition(from: CGPoint(x: 0.0, y: -64.0), to: CGPoint(), duration: 0.4, timingFunction: kCAMediaTimingFunctionSpring, additive: true)
|
self.tabContainerNode.layer.animatePosition(from: CGPoint(x: 0.0, y: -64.0), to: CGPoint(), duration: 0.4, timingFunction: kCAMediaTimingFunctionSpring, additive: true)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let navigationController = self.navigationController as? NavigationController {
|
||||||
|
var voiceChatOverlayController: VoiceChatOverlayController?
|
||||||
|
for controller in navigationController.globalOverlayControllers {
|
||||||
|
if let controller = controller as? VoiceChatOverlayController {
|
||||||
|
voiceChatOverlayController = controller
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if let controller = voiceChatOverlayController {
|
||||||
|
controller.update(hidden: false, slide: true, animated: true)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -26,21 +26,33 @@ private class CallStatusBarBackgroundNode: ASDisplayNode {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var speaking = false {
|
var connectingColor: UIColor = UIColor(rgb: 0xb6b6bb) {
|
||||||
|
didSet {
|
||||||
|
if self.connectingColor.rgb != oldValue.rgb {
|
||||||
|
self.updateGradientColors()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var speaking: Bool? = nil {
|
||||||
didSet {
|
didSet {
|
||||||
if self.speaking != oldValue {
|
if self.speaking != oldValue {
|
||||||
|
self.updateGradientColors()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func updateGradientColors() {
|
||||||
let initialColors = self.foregroundGradientLayer.colors
|
let initialColors = self.foregroundGradientLayer.colors
|
||||||
let targetColors: [CGColor]
|
let targetColors: [CGColor]
|
||||||
if speaking {
|
if let speaking = self.speaking {
|
||||||
targetColors = [green.cgColor, blue.cgColor]
|
targetColors = speaking ? [green.cgColor, blue.cgColor] : [blue.cgColor, lightBlue.cgColor]
|
||||||
} else {
|
} else {
|
||||||
targetColors = [blue.cgColor, lightBlue.cgColor]
|
targetColors = [connectingColor.cgColor, connectingColor.cgColor]
|
||||||
}
|
}
|
||||||
self.foregroundGradientLayer.colors = targetColors
|
self.foregroundGradientLayer.colors = targetColors
|
||||||
self.foregroundGradientLayer.animate(from: initialColors as AnyObject, to: targetColors as AnyObject, keyPath: "colors", timingFunction: CAMediaTimingFunctionName.linear.rawValue, duration: 0.3)
|
self.foregroundGradientLayer.animate(from: initialColors as AnyObject, to: targetColors as AnyObject, keyPath: "colors", timingFunction: CAMediaTimingFunctionName.linear.rawValue, duration: 0.3)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private let hierarchyTrackingNode: HierarchyTrackingNode
|
private let hierarchyTrackingNode: HierarchyTrackingNode
|
||||||
private var isCurrentlyInHierarchy = true
|
private var isCurrentlyInHierarchy = true
|
||||||
@ -145,7 +157,6 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private let backgroundNode: CallStatusBarBackgroundNode
|
private let backgroundNode: CallStatusBarBackgroundNode
|
||||||
private let microphoneNode: VoiceChatMicrophoneNode
|
|
||||||
private let titleNode: ImmediateTextNode
|
private let titleNode: ImmediateTextNode
|
||||||
private let subtitleNode: ImmediateAnimatedCountLabelNode
|
private let subtitleNode: ImmediateAnimatedCountLabelNode
|
||||||
|
|
||||||
@ -156,8 +167,9 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
private var currentSize: CGSize?
|
private var currentSize: CGSize?
|
||||||
private var currentContent: Content?
|
private var currentContent: Content?
|
||||||
|
|
||||||
private var strings: PresentationStrings?
|
private var presentationData: PresentationData?
|
||||||
private var nameDisplayOrder: PresentationPersonNameOrder = .firstLast
|
private let presentationDataDisposable = MetaDisposable()
|
||||||
|
|
||||||
private var currentPeer: Peer?
|
private var currentPeer: Peer?
|
||||||
private var currentCallTimer: SwiftSignalKit.Timer?
|
private var currentCallTimer: SwiftSignalKit.Timer?
|
||||||
private var currentCallState: PresentationCallState?
|
private var currentCallState: PresentationCallState?
|
||||||
@ -167,7 +179,6 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
|
|
||||||
public override init() {
|
public override init() {
|
||||||
self.backgroundNode = CallStatusBarBackgroundNode()
|
self.backgroundNode = CallStatusBarBackgroundNode()
|
||||||
self.microphoneNode = VoiceChatMicrophoneNode()
|
|
||||||
self.titleNode = ImmediateTextNode()
|
self.titleNode = ImmediateTextNode()
|
||||||
self.subtitleNode = ImmediateAnimatedCountLabelNode()
|
self.subtitleNode = ImmediateAnimatedCountLabelNode()
|
||||||
self.subtitleNode.reverseAnimationDirection = true
|
self.subtitleNode.reverseAnimationDirection = true
|
||||||
@ -180,6 +191,7 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
deinit {
|
deinit {
|
||||||
|
self.presentationDataDisposable.dispose()
|
||||||
self.audioLevelDisposable.dispose()
|
self.audioLevelDisposable.dispose()
|
||||||
self.stateDisposable.dispose()
|
self.stateDisposable.dispose()
|
||||||
self.currentCallTimer?.invalidate()
|
self.currentCallTimer?.invalidate()
|
||||||
@ -203,11 +215,10 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
let wasEmpty = (self.titleNode.attributedText?.string ?? "").isEmpty
|
let wasEmpty = (self.titleNode.attributedText?.string ?? "").isEmpty
|
||||||
|
|
||||||
if !self.didSetupData {
|
if !self.didSetupData {
|
||||||
|
self.didSetupData = true
|
||||||
switch content {
|
switch content {
|
||||||
case let .call(sharedContext, account, call):
|
case let .call(sharedContext, account, call):
|
||||||
let presentationData = sharedContext.currentPresentationData.with { $0 }
|
self.presentationData = sharedContext.currentPresentationData.with { $0 }
|
||||||
self.strings = presentationData.strings
|
|
||||||
self.nameDisplayOrder = presentationData.nameDisplayOrder
|
|
||||||
self.stateDisposable.set(
|
self.stateDisposable.set(
|
||||||
(combineLatest(
|
(combineLatest(
|
||||||
account.postbox.loadedPeerWithId(call.peerId),
|
account.postbox.loadedPeerWithId(call.peerId),
|
||||||
@ -223,9 +234,14 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
}
|
}
|
||||||
}))
|
}))
|
||||||
case let .groupCall(sharedContext, account, call):
|
case let .groupCall(sharedContext, account, call):
|
||||||
let presentationData = sharedContext.currentPresentationData.with { $0 }
|
self.presentationData = sharedContext.currentPresentationData.with { $0 }
|
||||||
self.strings = presentationData.strings
|
self.presentationDataDisposable.set((sharedContext.presentationData
|
||||||
self.nameDisplayOrder = presentationData.nameDisplayOrder
|
|> deliverOnMainQueue).start(next: { [weak self] presentationData in
|
||||||
|
if let strongSelf = self {
|
||||||
|
strongSelf.presentationData = presentationData
|
||||||
|
strongSelf.update()
|
||||||
|
}
|
||||||
|
}))
|
||||||
self.stateDisposable.set(
|
self.stateDisposable.set(
|
||||||
(combineLatest(
|
(combineLatest(
|
||||||
account.postbox.peerView(id: call.peerId),
|
account.postbox.peerView(id: call.peerId),
|
||||||
@ -238,7 +254,7 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
strongSelf.currentGroupCallState = state
|
strongSelf.currentGroupCallState = state
|
||||||
|
|
||||||
var isMuted = isMuted
|
var isMuted = isMuted
|
||||||
if let state = state, let muteState = state.callState.muteState, !muteState.canUnmute {
|
if let state = state, state.callState.muteState != nil {
|
||||||
isMuted = true
|
isMuted = true
|
||||||
}
|
}
|
||||||
strongSelf.currentIsMuted = isMuted
|
strongSelf.currentIsMuted = isMuted
|
||||||
@ -268,7 +284,6 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
strongSelf.backgroundNode.audioLevel = effectiveLevel
|
strongSelf.backgroundNode.audioLevel = effectiveLevel
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
self.didSetupData = true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var title: String = ""
|
var title: String = ""
|
||||||
@ -277,9 +292,9 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
let textColor = UIColor.white
|
let textColor = UIColor.white
|
||||||
var segments: [AnimatedCountLabelNode.Segment] = []
|
var segments: [AnimatedCountLabelNode.Segment] = []
|
||||||
|
|
||||||
if let strings = self.strings {
|
if let presentationData = self.presentationData {
|
||||||
if let currentPeer = self.currentPeer {
|
if let currentPeer = self.currentPeer {
|
||||||
title = currentPeer.displayTitle(strings: strings, displayOrder: self.nameDisplayOrder)
|
title = currentPeer.displayTitle(strings: presentationData.strings, displayOrder: presentationData.nameDisplayOrder)
|
||||||
}
|
}
|
||||||
var membersCount: Int32?
|
var membersCount: Int32?
|
||||||
if let groupCallState = self.currentGroupCallState {
|
if let groupCallState = self.currentGroupCallState {
|
||||||
@ -289,12 +304,12 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if let membersCount = membersCount {
|
if let membersCount = membersCount {
|
||||||
var membersPart = strings.VoiceChat_Status_Members(membersCount)
|
var membersPart = presentationData.strings.VoiceChat_Status_Members(membersCount)
|
||||||
if let startIndex = membersPart.firstIndex(of: "["), let endIndex = membersPart.firstIndex(of: "]") {
|
if let startIndex = membersPart.firstIndex(of: "["), let endIndex = membersPart.firstIndex(of: "]") {
|
||||||
membersPart.removeSubrange(startIndex ... endIndex)
|
membersPart.removeSubrange(startIndex ... endIndex)
|
||||||
}
|
}
|
||||||
|
|
||||||
let rawTextAndRanges = strings.VoiceChat_Status_MembersFormat("\(membersCount)", membersPart)
|
let rawTextAndRanges = presentationData.strings.VoiceChat_Status_MembersFormat("\(membersCount)", membersPart)
|
||||||
|
|
||||||
let (rawText, ranges) = rawTextAndRanges
|
let (rawText, ranges) = rawTextAndRanges
|
||||||
var textIndex = 0
|
var textIndex = 0
|
||||||
@ -326,6 +341,8 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
textIndex += 1
|
textIndex += 1
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
self.backgroundNode.connectingColor = presentationData.theme.chatList.unreadBadgeInactiveBackgroundColor
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.subtitleNode.segments != segments {
|
if self.subtitleNode.segments != segments {
|
||||||
@ -334,8 +351,6 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
|
|
||||||
self.titleNode.attributedText = NSAttributedString(string: title, font: Font.semibold(13.0), textColor: .white)
|
self.titleNode.attributedText = NSAttributedString(string: title, font: Font.semibold(13.0), textColor: .white)
|
||||||
|
|
||||||
let animationSize: CGFloat = 25.0
|
|
||||||
let iconSpacing: CGFloat = 0.0
|
|
||||||
let spacing: CGFloat = 5.0
|
let spacing: CGFloat = 5.0
|
||||||
let titleSize = self.titleNode.updateLayout(CGSize(width: 160.0, height: size.height))
|
let titleSize = self.titleNode.updateLayout(CGSize(width: 160.0, height: size.height))
|
||||||
let subtitleSize = self.subtitleNode.updateLayout(size: CGSize(width: 160.0, height: size.height), animated: true)
|
let subtitleSize = self.subtitleNode.updateLayout(size: CGSize(width: 160.0, height: size.height), animated: true)
|
||||||
@ -347,14 +362,10 @@ public class CallStatusBarNodeImpl: CallStatusBarNode {
|
|||||||
let verticalOrigin: CGFloat = size.height - contentHeight
|
let verticalOrigin: CGFloat = size.height - contentHeight
|
||||||
|
|
||||||
let transition: ContainedViewLayoutTransition = wasEmpty ? .immediate : .animated(duration: 0.2, curve: .easeInOut)
|
let transition: ContainedViewLayoutTransition = wasEmpty ? .immediate : .animated(duration: 0.2, curve: .easeInOut)
|
||||||
|
|
||||||
// transition.updateFrame(node: self.microphoneNode, frame: CGRect(origin: CGPoint(x: horizontalOrigin, y: verticalOrigin + floor((contentHeight - animationSize) / 2.0)), size: CGSize(width: animationSize, height: animationSize)))
|
|
||||||
// self.microphoneNode.update(state: VoiceChatMicrophoneNode.State(muted: self.currentIsMuted, color: UIColor.white), animated: true)
|
|
||||||
|
|
||||||
transition.updateFrame(node: self.titleNode, frame: CGRect(origin: CGPoint(x: horizontalOrigin, y: verticalOrigin + floor((contentHeight - titleSize.height) / 2.0)), size: titleSize))
|
transition.updateFrame(node: self.titleNode, frame: CGRect(origin: CGPoint(x: horizontalOrigin, y: verticalOrigin + floor((contentHeight - titleSize.height) / 2.0)), size: titleSize))
|
||||||
transition.updateFrame(node: self.subtitleNode, frame: CGRect(origin: CGPoint(x: horizontalOrigin + titleSize.width + spacing, y: verticalOrigin + floor((contentHeight - subtitleSize.height) / 2.0)), size: subtitleSize))
|
transition.updateFrame(node: self.subtitleNode, frame: CGRect(origin: CGPoint(x: horizontalOrigin + titleSize.width + spacing, y: verticalOrigin + floor((contentHeight - subtitleSize.height) / 2.0)), size: subtitleSize))
|
||||||
|
|
||||||
self.backgroundNode.speaking = self.currentIsConnected && !self.currentIsMuted
|
self.backgroundNode.speaking = self.currentIsConnected ? !self.currentIsMuted : nil
|
||||||
self.backgroundNode.frame = CGRect(origin: CGPoint(), size: CGSize(width: size.width, height: size.height + 18.0))
|
self.backgroundNode.frame = CGRect(origin: CGPoint(), size: CGSize(width: size.width, height: size.height + 18.0))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -43,12 +43,21 @@ final class VoiceChatActionButton: HighlightTrackingButtonNode {
|
|||||||
var outerColor: Signal<UIColor?, NoError> {
|
var outerColor: Signal<UIColor?, NoError> {
|
||||||
return outerColorPromise.get()
|
return outerColorPromise.get()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var connectingColor: UIColor = UIColor(rgb: 0xb6b6bb) {
|
||||||
|
didSet {
|
||||||
|
self.backgroundNode.connectingColor = self.connectingColor
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var activeDisposable = MetaDisposable()
|
var activeDisposable = MetaDisposable()
|
||||||
|
|
||||||
|
var isDisabled: Bool = false
|
||||||
|
|
||||||
var wasActiveWhenPressed = false
|
var wasActiveWhenPressed = false
|
||||||
var pressing: Bool = false {
|
var pressing: Bool = false {
|
||||||
didSet {
|
didSet {
|
||||||
guard let (_, _, state, _, _, _, _, snap) = self.currentParams else {
|
guard let (_, _, state, _, _, _, _, snap) = self.currentParams, !self.isDisabled else {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if self.pressing {
|
if self.pressing {
|
||||||
@ -93,7 +102,7 @@ final class VoiceChatActionButton: HighlightTrackingButtonNode {
|
|||||||
|
|
||||||
self.highligthedChanged = { [weak self] pressing in
|
self.highligthedChanged = { [weak self] pressing in
|
||||||
if let strongSelf = self {
|
if let strongSelf = self {
|
||||||
guard let (_, _, _, _, _, _, _, snap) = strongSelf.currentParams else {
|
guard let (_, _, _, _, _, _, _, snap) = strongSelf.currentParams, !strongSelf.isDisabled else {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if pressing {
|
if pressing {
|
||||||
@ -184,11 +193,13 @@ final class VoiceChatActionButton: HighlightTrackingButtonNode {
|
|||||||
transition.updateTransformScale(node: self.iconNode, scale: 0.5)
|
transition.updateTransformScale(node: self.iconNode, scale: 0.5)
|
||||||
transition.updateAlpha(node: self.titleLabel, alpha: 0.0)
|
transition.updateAlpha(node: self.titleLabel, alpha: 0.0)
|
||||||
transition.updateAlpha(node: self.subtitleLabel, alpha: 0.0)
|
transition.updateAlpha(node: self.subtitleLabel, alpha: 0.0)
|
||||||
|
transition.updateAlpha(layer: self.maskProgressLayer, alpha: 0.0)
|
||||||
} else {
|
} else {
|
||||||
transition.updateTransformScale(node: self.backgroundNode, scale: small ? 0.85 : 1.0)
|
transition.updateTransformScale(node: self.backgroundNode, scale: small ? 0.85 : 1.0)
|
||||||
transition.updateTransformScale(node: self.iconNode, scale: self.pressing ? 0.9 : 1.0)
|
transition.updateTransformScale(node: self.iconNode, scale: self.pressing ? 0.9 : 1.0)
|
||||||
transition.updateAlpha(node: self.titleLabel, alpha: 1.0)
|
transition.updateAlpha(node: self.titleLabel, alpha: 1.0)
|
||||||
transition.updateAlpha(node: self.subtitleLabel, alpha: 1.0)
|
transition.updateAlpha(node: self.subtitleLabel, alpha: 1.0)
|
||||||
|
transition.updateAlpha(layer: self.maskProgressLayer, alpha: 1.0)
|
||||||
}
|
}
|
||||||
|
|
||||||
let iconSize = CGSize(width: 90.0, height: 90.0)
|
let iconSize = CGSize(width: 90.0, height: 90.0)
|
||||||
@ -202,8 +213,9 @@ final class VoiceChatActionButton: HighlightTrackingButtonNode {
|
|||||||
if let previous = self.currentParams {
|
if let previous = self.currentParams {
|
||||||
self.currentParams = (previous.size, previous.buttonSize, previous.state, previous.dark, previous.small, previous.title, previous.subtitle, snap)
|
self.currentParams = (previous.size, previous.buttonSize, previous.state, previous.dark, previous.small, previous.title, previous.subtitle, snap)
|
||||||
|
|
||||||
|
self.backgroundNode.isSnap = snap
|
||||||
self.backgroundNode.glowHidden = snap
|
self.backgroundNode.glowHidden = snap
|
||||||
|
self.backgroundNode.updateColors()
|
||||||
self.applyParams(animated: animated)
|
self.applyParams(animated: animated)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -231,7 +243,7 @@ final class VoiceChatActionButton: HighlightTrackingButtonNode {
|
|||||||
case .connecting:
|
case .connecting:
|
||||||
backgroundState = .connecting
|
backgroundState = .connecting
|
||||||
}
|
}
|
||||||
self.backgroundNode.updateColor(dark: dark)
|
self.backgroundNode.isDark = dark
|
||||||
self.backgroundNode.update(state: backgroundState, animated: true)
|
self.backgroundNode.update(state: backgroundState, animated: true)
|
||||||
self.iconNode.update(state: VoiceChatMicrophoneNode.State(muted: iconMuted, color: iconColor), animated: true)
|
self.iconNode.update(state: VoiceChatMicrophoneNode.State(muted: iconMuted, color: iconColor), animated: true)
|
||||||
|
|
||||||
@ -819,6 +831,13 @@ private final class VoiceChatActionButtonBackgroundNode: ASDisplayNode {
|
|||||||
switch self.state {
|
switch self.state {
|
||||||
case .connecting:
|
case .connecting:
|
||||||
self.updatedActive?(false)
|
self.updatedActive?(false)
|
||||||
|
if let transition = self.transition {
|
||||||
|
self.updateGlowScale(nil)
|
||||||
|
if case .blob = transition {
|
||||||
|
playBlobsDisappearanceAnimation()
|
||||||
|
}
|
||||||
|
self.transition = nil
|
||||||
|
}
|
||||||
self.setupProgressAnimations()
|
self.setupProgressAnimations()
|
||||||
self.isActive = false
|
self.isActive = false
|
||||||
case let .blob(newActive):
|
case let .blob(newActive):
|
||||||
@ -858,10 +877,36 @@ private final class VoiceChatActionButtonBackgroundNode: ASDisplayNode {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func updateColor(dark: Bool) {
|
var isDark: Bool = false {
|
||||||
|
didSet {
|
||||||
|
if self.isDark != oldValue {
|
||||||
|
self.updateColors()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var isSnap: Bool = false {
|
||||||
|
didSet {
|
||||||
|
if self.isSnap != oldValue {
|
||||||
|
self.updateColors()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var connectingColor: UIColor = UIColor(rgb: 0xb6b6bb) {
|
||||||
|
didSet {
|
||||||
|
if self.connectingColor.rgb != oldValue.rgb {
|
||||||
|
self.updateColors()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fileprivate func updateColors() {
|
||||||
let previousColor: CGColor = self.backgroundCircleLayer.fillColor ?? greyColor.cgColor
|
let previousColor: CGColor = self.backgroundCircleLayer.fillColor ?? greyColor.cgColor
|
||||||
let targetColor: CGColor
|
let targetColor: CGColor
|
||||||
if dark {
|
if self.isSnap {
|
||||||
|
targetColor = self.connectingColor.cgColor
|
||||||
|
} else if self.isDark {
|
||||||
targetColor = secondaryGreyColor.cgColor
|
targetColor = secondaryGreyColor.cgColor
|
||||||
} else {
|
} else {
|
||||||
targetColor = greyColor.cgColor
|
targetColor = greyColor.cgColor
|
||||||
|
@ -376,6 +376,7 @@ public final class VoiceChatController: ViewController {
|
|||||||
private let context: AccountContext
|
private let context: AccountContext
|
||||||
private let call: PresentationGroupCall
|
private let call: PresentationGroupCall
|
||||||
private var presentationData: PresentationData
|
private var presentationData: PresentationData
|
||||||
|
private var presentationDataDisposable: Disposable?
|
||||||
private var darkTheme: PresentationTheme
|
private var darkTheme: PresentationTheme
|
||||||
|
|
||||||
private let dimNode: ASDisplayNode
|
private let dimNode: ASDisplayNode
|
||||||
@ -832,6 +833,14 @@ public final class VoiceChatController: ViewController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
self.presentationDataDisposable = (sharedContext.presentationData
|
||||||
|
|> deliverOnMainQueue).start(next: { [weak self] presentationData in
|
||||||
|
if let strongSelf = self {
|
||||||
|
strongSelf.presentationData = presentationData
|
||||||
|
strongSelf.actionButton.connectingColor = presentationData.theme.chatList.unreadBadgeInactiveBackgroundColor
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
self.memberStatesDisposable = (combineLatest(queue: .mainQueue(),
|
self.memberStatesDisposable = (combineLatest(queue: .mainQueue(),
|
||||||
self.call.state,
|
self.call.state,
|
||||||
self.call.members,
|
self.call.members,
|
||||||
@ -1092,6 +1101,7 @@ public final class VoiceChatController: ViewController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
deinit {
|
deinit {
|
||||||
|
self.presentationDataDisposable?.dispose()
|
||||||
self.peerViewDisposable?.dispose()
|
self.peerViewDisposable?.dispose()
|
||||||
self.leaveDisposable.dispose()
|
self.leaveDisposable.dispose()
|
||||||
self.isMutedDisposable?.dispose()
|
self.isMutedDisposable?.dispose()
|
||||||
@ -1184,6 +1194,9 @@ public final class VoiceChatController: ViewController {
|
|||||||
guard let callState = self.callState else {
|
guard let callState = self.callState else {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
if case .connecting = callState.networkState {
|
||||||
|
return
|
||||||
|
}
|
||||||
if let muteState = callState.muteState {
|
if let muteState = callState.muteState {
|
||||||
if !muteState.canUnmute {
|
if !muteState.canUnmute {
|
||||||
if case .ended = gestureRecognizer.state {
|
if case .ended = gestureRecognizer.state {
|
||||||
@ -1580,7 +1593,7 @@ public final class VoiceChatController: ViewController {
|
|||||||
actionButtonEnabled = false
|
actionButtonEnabled = false
|
||||||
}
|
}
|
||||||
|
|
||||||
self.actionButton.isUserInteractionEnabled = actionButtonEnabled
|
self.actionButton.isDisabled = !actionButtonEnabled
|
||||||
self.actionButton.update(size: centralButtonSize, buttonSize: CGSize(width: 144.0, height: 144.0), state: actionButtonState, title: actionButtonTitle, subtitle: actionButtonSubtitle, dark: self.isFullscreen, small: layout.size.width < 330.0, animated: true)
|
self.actionButton.update(size: centralButtonSize, buttonSize: CGSize(width: 144.0, height: 144.0), state: actionButtonState, title: actionButtonTitle, subtitle: actionButtonSubtitle, dark: self.isFullscreen, small: layout.size.width < 330.0, animated: true)
|
||||||
|
|
||||||
if self.actionButton.supernode === self.bottomPanelNode {
|
if self.actionButton.supernode === self.bottomPanelNode {
|
||||||
|
@ -159,7 +159,7 @@ public class VoiceChatParticipantItemNode: ItemListRevealOptionsItemNode {
|
|||||||
private let actionContainerNode: ASDisplayNode
|
private let actionContainerNode: ASDisplayNode
|
||||||
private var animationNode: VoiceChatMicrophoneNode?
|
private var animationNode: VoiceChatMicrophoneNode?
|
||||||
private var iconNode: ASImageNode?
|
private var iconNode: ASImageNode?
|
||||||
private var actionButtonNode: HighlightTrackingButtonNode
|
private var actionButtonNode: HighlightableButtonNode
|
||||||
|
|
||||||
private var audioLevelView: VoiceBlobView?
|
private var audioLevelView: VoiceBlobView?
|
||||||
private let audioLevelDisposable = MetaDisposable()
|
private let audioLevelDisposable = MetaDisposable()
|
||||||
@ -201,7 +201,7 @@ public class VoiceChatParticipantItemNode: ItemListRevealOptionsItemNode {
|
|||||||
self.statusNode.contentsScale = UIScreen.main.scale
|
self.statusNode.contentsScale = UIScreen.main.scale
|
||||||
|
|
||||||
self.actionContainerNode = ASDisplayNode()
|
self.actionContainerNode = ASDisplayNode()
|
||||||
self.actionButtonNode = HighlightTrackingButtonNode()
|
self.actionButtonNode = HighlightableButtonNode()
|
||||||
|
|
||||||
self.highlightedBackgroundNode = ASDisplayNode()
|
self.highlightedBackgroundNode = ASDisplayNode()
|
||||||
self.highlightedBackgroundNode.isLayerBacked = true
|
self.highlightedBackgroundNode.isLayerBacked = true
|
||||||
@ -613,7 +613,7 @@ public class VoiceChatParticipantItemNode: ItemListRevealOptionsItemNode {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
animationNode.update(state: VoiceChatMicrophoneNode.State(muted: muted, color: color), animated: true)
|
animationNode.update(state: VoiceChatMicrophoneNode.State(muted: muted, color: color), animated: true)
|
||||||
strongSelf.actionButtonNode.isUserInteractionEnabled = false
|
strongSelf.actionButtonNode.isUserInteractionEnabled = item.contextAction != nil
|
||||||
} else if let animationNode = strongSelf.animationNode {
|
} else if let animationNode = strongSelf.animationNode {
|
||||||
strongSelf.animationNode = nil
|
strongSelf.animationNode = nil
|
||||||
animationNode.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false)
|
animationNode.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false)
|
||||||
@ -643,7 +643,7 @@ public class VoiceChatParticipantItemNode: ItemListRevealOptionsItemNode {
|
|||||||
} else {
|
} else {
|
||||||
iconNode.image = generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/AddUser"), color: item.presentationData.theme.list.itemAccentColor)
|
iconNode.image = generateTintedImage(image: UIImage(bundleImageName: "Chat/Context Menu/AddUser"), color: item.presentationData.theme.list.itemAccentColor)
|
||||||
}
|
}
|
||||||
strongSelf.actionButtonNode.isUserInteractionEnabled = !invited
|
strongSelf.actionButtonNode.isUserInteractionEnabled = false
|
||||||
} else if let iconNode = strongSelf.iconNode {
|
} else if let iconNode = strongSelf.iconNode {
|
||||||
strongSelf.iconNode = nil
|
strongSelf.iconNode = nil
|
||||||
iconNode.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false)
|
iconNode.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, removeOnCompletion: false)
|
||||||
@ -734,8 +734,8 @@ public class VoiceChatParticipantItemNode: ItemListRevealOptionsItemNode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@objc private func actionButtonPressed() {
|
@objc private func actionButtonPressed() {
|
||||||
if let item = self.layoutParams?.0 {
|
if let item = self.layoutParams?.0, let contextAction = item.contextAction {
|
||||||
item.action?()
|
contextAction(self.contextSourceNode, nil)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1245,11 +1245,7 @@ final class SharedApplicationContext {
|
|||||||
|> map { loggedOutAccountPeerIds -> (AccountManager, Set<PeerId>) in
|
|> map { loggedOutAccountPeerIds -> (AccountManager, Set<PeerId>) in
|
||||||
return (sharedContext.sharedContext.accountManager, loggedOutAccountPeerIds)
|
return (sharedContext.sharedContext.accountManager, loggedOutAccountPeerIds)
|
||||||
}
|
}
|
||||||
}).start(next: { [weak self] accountManager, loggedOutAccountPeerIds in
|
}).start(next: { accountManager, loggedOutAccountPeerIds in
|
||||||
guard let strongSelf = self else {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
let _ = (updateIntentsSettingsInteractively(accountManager: accountManager) { current in
|
let _ = (updateIntentsSettingsInteractively(accountManager: accountManager) { current in
|
||||||
var updated = current
|
var updated = current
|
||||||
for peerId in loggedOutAccountPeerIds {
|
for peerId in loggedOutAccountPeerIds {
|
||||||
@ -1888,7 +1884,7 @@ final class SharedApplicationContext {
|
|||||||
|> take(1)
|
|> take(1)
|
||||||
|> deliverOnMainQueue).start(next: { sharedContext in
|
|> deliverOnMainQueue).start(next: { sharedContext in
|
||||||
let type = ApplicationShortcutItemType(rawValue: shortcutItem.type)
|
let type = ApplicationShortcutItemType(rawValue: shortcutItem.type)
|
||||||
var immediately = type == .account
|
let immediately = type == .account
|
||||||
let proceed: () -> Void = {
|
let proceed: () -> Void = {
|
||||||
let _ = (self.context.get()
|
let _ = (self.context.get()
|
||||||
|> take(1)
|
|> take(1)
|
||||||
|
@ -234,9 +234,11 @@ final class ChatMediaInputStickerGridItemNode: GridItemNode {
|
|||||||
if !animated {
|
if !animated {
|
||||||
placeholderNode.removeFromSupernode()
|
placeholderNode.removeFromSupernode()
|
||||||
} else {
|
} else {
|
||||||
|
placeholderNode.allowsGroupOpacity = true
|
||||||
placeholderNode.alpha = 0.0
|
placeholderNode.alpha = 0.0
|
||||||
placeholderNode.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, completion: { [weak placeholderNode] _ in
|
placeholderNode.layer.animateAlpha(from: 1.0, to: 0.0, duration: 0.2, completion: { [weak placeholderNode] _ in
|
||||||
placeholderNode?.removeFromSupernode()
|
placeholderNode?.removeFromSupernode()
|
||||||
|
placeholderNode?.allowsGroupOpacity = false
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user