2023-12-15 22:59:33 +04:00

411 lines
20 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
}
}
func shouldKeepUserStoriesInFeed(peerId: PeerId, isContactOrMember: Bool) -> Bool {
if peerId.namespace == Namespaces.Peer.CloudUser && (peerId.id._internalGetInt64Value() == 777000 || peerId.id._internalGetInt64Value() == 333000) {
return true
}
return isContactOrMember
}
func updatePeers(transaction: Transaction, accountPeerId: PeerId, peers: AccumulatedPeers) {
var parsedPeers: [Peer] = []
for (_, user) in peers.users {
if let telegramUser = TelegramUser.merge(transaction.getPeer(user.peerId) as? TelegramUser, rhs: user) {
parsedPeers.append(telegramUser)
switch user {
case let .user(flags, flags2, _, _, _, _, _, _, _, _, _, _, _, _, _, _, storiesMaxId, _, _):
let isMin = (flags & (1 << 20)) != 0
let storiesUnavailable = (flags2 & (1 << 4)) != 0
if let storiesMaxId = storiesMaxId {
transaction.setStoryItemsInexactMaxId(peerId: user.peerId, id: storiesMaxId)
} else if !isMin && storiesUnavailable {
transaction.clearStoryItemsInexactMaxId(peerId: user.peerId)
}
if !isMin {
let isContact = (flags & (1 << 11)) != 0
_internal_updatePeerIsContact(transaction: transaction, user: telegramUser, isContact: isContact)
}
case .userEmpty:
break
}
}
}
for (_, chat) in peers.chats {
switch chat {
case let .channel(flags, flags2, _, _, _, _, _, _, _, _, _, _, _, _, storiesMaxId, _, _, _, _):
let isMin = (flags & (1 << 12)) != 0
let storiesUnavailable = (flags2 & (1 << 3)) != 0
if let storiesMaxId = storiesMaxId {
transaction.setStoryItemsInexactMaxId(peerId: chat.peerId, id: storiesMaxId)
} else if !isMin && storiesUnavailable {
transaction.clearStoryItemsInexactMaxId(peerId: chat.peerId)
}
default:
break
}
}
for (_, peer) in peers.peers {
parsedPeers.append(peer)
}
updatePeersCustom(transaction: transaction, peers: parsedPeers, update: { _, updated in updated })
updatePeerPresences(transaction: transaction, accountPeerId: accountPeerId, peerPresences: peers.users)
}
func _internal_updatePeerIsContact(transaction: Transaction, user: TelegramUser, isContact: Bool) {
let previousValue = shouldKeepUserStoriesInFeed(peerId: user.id, isContactOrMember: transaction.isPeerContact(peerId: user.id))
let updatedValue = shouldKeepUserStoriesInFeed(peerId: user.id, isContactOrMember: isContact)
if previousValue != updatedValue, let storiesHidden = user.storiesHidden {
if updatedValue {
if storiesHidden {
if transaction.storySubscriptionsContains(key: .filtered, peerId: user.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .filtered)
peerIds.removeAll(where: { $0 == user.id })
transaction.replaceAllStorySubscriptions(key: .filtered, state: state, peerIds: peerIds)
}
if !transaction.storySubscriptionsContains(key: .hidden, peerId: user.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .hidden)
if !peerIds.contains(user.id) {
peerIds.append(user.id)
transaction.replaceAllStorySubscriptions(key: .hidden, state: state, peerIds: peerIds)
}
}
} else {
if transaction.storySubscriptionsContains(key: .hidden, peerId: user.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .hidden)
peerIds.removeAll(where: { $0 == user.id })
transaction.replaceAllStorySubscriptions(key: .hidden, state: state, peerIds: peerIds)
}
if !transaction.storySubscriptionsContains(key: .filtered, peerId: user.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .filtered)
if !peerIds.contains(user.id) {
peerIds.append(user.id)
transaction.replaceAllStorySubscriptions(key: .filtered, state: state, peerIds: peerIds)
}
}
}
} else {
if transaction.storySubscriptionsContains(key: .filtered, peerId: user.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .filtered)
peerIds.removeAll(where: { $0 == user.id })
transaction.replaceAllStorySubscriptions(key: .filtered, state: state, peerIds: peerIds)
}
if transaction.storySubscriptionsContains(key: .hidden, peerId: user.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .hidden)
peerIds.removeAll(where: { $0 == user.id })
transaction.replaceAllStorySubscriptions(key: .hidden, state: state, peerIds: peerIds)
}
}
}
}
private func _internal_updateChannelMembership(transaction: Transaction, channel: TelegramChannel, isMember: Bool, justJoined: Bool) {
if isMember, let storiesHidden = channel.storiesHidden {
if storiesHidden {
if transaction.storySubscriptionsContains(key: .filtered, peerId: channel.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .filtered)
peerIds.removeAll(where: { $0 == channel.id })
transaction.replaceAllStorySubscriptions(key: .filtered, state: state, peerIds: peerIds)
}
if !transaction.storySubscriptionsContains(key: .hidden, peerId: channel.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .hidden)
if !peerIds.contains(channel.id) {
peerIds.append(channel.id)
transaction.replaceAllStorySubscriptions(key: .hidden, state: state, peerIds: peerIds)
}
}
} else {
if transaction.storySubscriptionsContains(key: .hidden, peerId: channel.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .hidden)
peerIds.removeAll(where: { $0 == channel.id })
transaction.replaceAllStorySubscriptions(key: .hidden, state: state, peerIds: peerIds)
}
if !transaction.storySubscriptionsContains(key: .filtered, peerId: channel.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .filtered)
if !peerIds.contains(channel.id) {
peerIds.append(channel.id)
transaction.replaceAllStorySubscriptions(key: .filtered, state: state, peerIds: peerIds)
}
}
}
if justJoined {
_internal_addSynchronizePeerStoriesOperation(peerId: channel.id, transaction: transaction)
}
} else {
if transaction.storySubscriptionsContains(key: .filtered, peerId: channel.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .filtered)
peerIds.removeAll(where: { $0 == channel.id })
transaction.replaceAllStorySubscriptions(key: .filtered, state: state, peerIds: peerIds)
}
if transaction.storySubscriptionsContains(key: .hidden, peerId: channel.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .hidden)
peerIds.removeAll(where: { $0 == channel.id })
transaction.replaceAllStorySubscriptions(key: .hidden, state: state, peerIds: peerIds)
}
}
}
public func updatePeersCustom(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)
}
if let updatedChannel = updated as? TelegramChannel {
var wasMember = false
var wasHidden: Bool?
if let previous = previous as? TelegramChannel {
wasMember = previous.participationStatus == .member
wasHidden = previous.storiesHidden
updated = mergeChannel(lhs: previous, rhs: updatedChannel)
}
if let updated = updated as? TelegramChannel {
let isMember = updated.participationStatus == .member
if isMember != wasMember || updated.storiesHidden != wasHidden {
_internal_updateChannelMembership(transaction: transaction, channel: updated, isMember: isMember, justJoined: previous == nil || wasHidden == nil)
}
}
}
switch peerId.namespace {
case Namespaces.Peer.CloudUser:
if let updated = updated as? TelegramUser, let previous = previous as? TelegramUser {
if let storiesHidden = updated.storiesHidden, storiesHidden != previous.storiesHidden {
if storiesHidden {
if transaction.storySubscriptionsContains(key: .filtered, peerId: updated.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .filtered)
peerIds.removeAll(where: { $0 == updated.id })
transaction.replaceAllStorySubscriptions(key: .filtered, state: state, peerIds: peerIds)
if !transaction.storySubscriptionsContains(key: .hidden, peerId: updated.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .hidden)
if !peerIds.contains(updated.id) {
peerIds.append(updated.id)
transaction.replaceAllStorySubscriptions(key: .hidden, state: state, peerIds: peerIds)
}
}
}
} else {
if transaction.storySubscriptionsContains(key: .hidden, peerId: updated.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .hidden)
peerIds.removeAll(where: { $0 == updated.id })
transaction.replaceAllStorySubscriptions(key: .hidden, state: state, peerIds: peerIds)
if !transaction.storySubscriptionsContains(key: .filtered, peerId: updated.id) {
var (state, peerIds) = transaction.getAllStorySubscriptions(key: .filtered)
if !peerIds.contains(updated.id) {
peerIds.append(updated.id)
transaction.replaceAllStorySubscriptions(key: .filtered, state: state, peerIds: peerIds)
}
}
}
}
}
}
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 {
if case .personal = channel.accessHash {
switch channel.participationStatus {
case .member:
updatePeerChatInclusionWithMinTimestamp(transaction: transaction, id: peerId, minTimestamp: channel.creationDate, forceRootGroupIfNotExists: true)
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: Api.User]) {
var parsedPresences: [PeerId: PeerPresence] = [:]
for (peerId, user) in peerPresences {
guard let presence = TelegramUserPresence(apiUser: user) else {
continue
}
switch presence.status {
case .present:
parsedPresences[peerId] = presence
default:
switch user {
case let .user(flags, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _, _):
let isMin = (flags & (1 << 20)) != 0
if isMin, let _ = transaction.getPeerPresence(peerId: peerId) {
} else {
parsedPresences[peerId] = presence
}
default:
break
}
}
}
parsedPresences.removeValue(forKey: accountPeerId)
transaction.updatePeerPresencesInternal(presences: parsedPresences, 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 updatePeerPresencesClean(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)
}
}