mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-06-16 05:55:20 +00:00
180 lines
7.4 KiB
Swift
180 lines
7.4 KiB
Swift
import Foundation
|
|
import Postbox
|
|
import TelegramApi
|
|
|
|
|
|
func updatePeerChatInclusionWithMinTimestamp(transaction: Transaction, id: PeerId, minTimestamp: Int32, forceRootGroupIfNotExists: Bool) {
|
|
let currentInclusion = transaction.getPeerChatListInclusion(id)
|
|
var updatedInclusion: PeerChatListInclusion?
|
|
switch currentInclusion {
|
|
case let .ifHasMessagesOrOneOf(groupId, pinningIndex, currentMinTimestamp):
|
|
let updatedMinTimestamp: Int32
|
|
if let currentMinTimestamp = currentMinTimestamp {
|
|
if minTimestamp > currentMinTimestamp {
|
|
updatedMinTimestamp = minTimestamp
|
|
} else {
|
|
updatedMinTimestamp = currentMinTimestamp
|
|
}
|
|
} else {
|
|
updatedMinTimestamp = minTimestamp
|
|
}
|
|
updatedInclusion = .ifHasMessagesOrOneOf(groupId: groupId, pinningIndex: pinningIndex, minTimestamp: updatedMinTimestamp)
|
|
default:
|
|
if forceRootGroupIfNotExists {
|
|
updatedInclusion = .ifHasMessagesOrOneOf(groupId: .root, pinningIndex: nil, minTimestamp: minTimestamp)
|
|
}
|
|
}
|
|
if let updatedInclusion = updatedInclusion {
|
|
transaction.updatePeerChatListInclusion(id, inclusion: updatedInclusion)
|
|
}
|
|
}
|
|
|
|
func minTimestampForPeerInclusion(_ peer: Peer) -> Int32? {
|
|
if let group = peer as? TelegramGroup {
|
|
return group.creationDate
|
|
} else if let channel = peer as? TelegramChannel {
|
|
return channel.creationDate
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
public func updatePeers(transaction: Transaction, peers: [Peer], update: (Peer?, Peer) -> Peer?) {
|
|
transaction.updatePeersInternal(peers, update: { previous, updated in
|
|
let peerId = updated.id
|
|
|
|
var updated = updated
|
|
|
|
if let previous = previous as? TelegramUser, let updatedUser = updated as? TelegramUser {
|
|
updated = TelegramUser.merge(lhs: previous, rhs: updatedUser)
|
|
} else if let previous = previous as? TelegramChannel, let updatedChannel = updated as? TelegramChannel {
|
|
updated = mergeChannel(lhs: previous, rhs: updatedChannel)
|
|
}
|
|
|
|
switch peerId.namespace {
|
|
case Namespaces.Peer.CloudUser:
|
|
break
|
|
case Namespaces.Peer.CloudGroup:
|
|
if let group = updated as? TelegramGroup {
|
|
if group.flags.contains(.deactivated) {
|
|
transaction.updatePeerChatListInclusion(peerId, inclusion: .notIncluded)
|
|
} else {
|
|
switch group.membership {
|
|
case .Member:
|
|
updatePeerChatInclusionWithMinTimestamp(transaction: transaction, id: peerId, minTimestamp: group.creationDate, forceRootGroupIfNotExists: false)
|
|
default:
|
|
transaction.updatePeerChatListInclusion(peerId, inclusion: .notIncluded)
|
|
}
|
|
}
|
|
} else {
|
|
assertionFailure()
|
|
}
|
|
case Namespaces.Peer.CloudChannel:
|
|
if let channel = updated as? TelegramChannel {
|
|
switch channel.participationStatus {
|
|
case .member:
|
|
updatePeerChatInclusionWithMinTimestamp(transaction: transaction, id: peerId, minTimestamp: channel.creationDate, forceRootGroupIfNotExists: false)
|
|
case .left:
|
|
transaction.updatePeerChatListInclusion(peerId, inclusion: .notIncluded)
|
|
case .kicked where channel.creationDate == 0:
|
|
transaction.updatePeerChatListInclusion(peerId, inclusion: .notIncluded)
|
|
default:
|
|
transaction.updatePeerChatListInclusion(peerId, inclusion: .notIncluded)
|
|
}
|
|
} else {
|
|
assertionFailure()
|
|
}
|
|
case Namespaces.Peer.SecretChat:
|
|
if let secretChat = updated as? TelegramSecretChat {
|
|
let isActive: Bool
|
|
switch secretChat.embeddedState {
|
|
case .active, .handshake:
|
|
isActive = true
|
|
case .terminated:
|
|
isActive = false
|
|
}
|
|
updatePeerChatInclusionWithMinTimestamp(transaction: transaction, id: peerId, minTimestamp: secretChat.creationDate, forceRootGroupIfNotExists: isActive)
|
|
} else {
|
|
assertionFailure()
|
|
}
|
|
default:
|
|
assertionFailure()
|
|
break
|
|
}
|
|
|
|
return update(previous, updated)
|
|
})
|
|
}
|
|
|
|
func updatePeerPresences(transaction: Transaction, accountPeerId: PeerId, peerPresences: [PeerId: PeerPresence]) {
|
|
var peerPresences = peerPresences
|
|
if peerPresences[accountPeerId] != nil {
|
|
peerPresences.removeValue(forKey: accountPeerId)
|
|
}
|
|
transaction.updatePeerPresencesInternal(presences: peerPresences, merge: { previous, updated in
|
|
if let previous = previous as? TelegramUserPresence, let updated = updated as? TelegramUserPresence, previous.lastActivity != updated.lastActivity {
|
|
return TelegramUserPresence(status: updated.status, lastActivity: max(previous.lastActivity, updated.lastActivity))
|
|
}
|
|
return updated
|
|
})
|
|
}
|
|
|
|
func updatePeerPresenceLastActivities(transaction: Transaction, accountPeerId: PeerId, activities: [PeerId: Int32]) {
|
|
var activities = activities
|
|
if activities[accountPeerId] != nil {
|
|
activities.removeValue(forKey: accountPeerId)
|
|
}
|
|
for (peerId, timestamp) in activities {
|
|
transaction.updatePeerPresenceInternal(peerId: peerId, update: { previous in
|
|
if let previous = previous as? TelegramUserPresence, previous.lastActivity < timestamp {
|
|
var updatedStatus = previous.status
|
|
switch updatedStatus {
|
|
case let .present(until):
|
|
if until < timestamp {
|
|
updatedStatus = .present(until: timestamp)
|
|
}
|
|
default:
|
|
break
|
|
}
|
|
return TelegramUserPresence(status: updatedStatus, lastActivity: timestamp)
|
|
}
|
|
return previous
|
|
})
|
|
}
|
|
}
|
|
|
|
func updateContacts(transaction: Transaction, apiUsers: [Api.User]) {
|
|
if apiUsers.isEmpty {
|
|
return
|
|
}
|
|
var contactIds = transaction.getContactPeerIds()
|
|
var updated = false
|
|
for user in apiUsers {
|
|
var isContact: Bool?
|
|
switch user {
|
|
case let .user(flags, _, _, _, _, _, _, _, _, _, _, _, _, _):
|
|
if (flags & (1 << 20)) == 0 {
|
|
isContact = (flags & (1 << 11)) != 0
|
|
}
|
|
case .userEmpty:
|
|
isContact = false
|
|
}
|
|
if let isContact = isContact {
|
|
if isContact {
|
|
if !contactIds.contains(user.peerId) {
|
|
contactIds.insert(user.peerId)
|
|
updated = true
|
|
}
|
|
} else {
|
|
if contactIds.contains(user.peerId) {
|
|
contactIds.remove(user.peerId)
|
|
updated = true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if updated {
|
|
transaction.replaceContactPeerIds(contactIds)
|
|
}
|
|
}
|