Swiftgram/submodules/TelegramCore/TelegramCore/SingleMessageView.swift
Peter 5c1613d104 Add 'submodules/TelegramCore/' from commit '9561227540acef69894e6546395ab223a6233600'
git-subtree-dir: submodules/TelegramCore
git-subtree-mainline: 971273e8f8f49a47f14b251d2f35e3445a61fc3f
git-subtree-split: 9561227540acef69894e6546395ab223a6233600
2019-06-11 18:59:08 +01:00

109 lines
4.7 KiB
Swift

import Foundation
#if os(macOS)
import PostboxMac
import SwiftSignalKitMac
import MtProtoKitMac
#else
import Postbox
import SwiftSignalKit
#if BUCK
import MtProtoKit
#else
import MtProtoKitDynamic
#endif
#endif
public func singleMessageView(account: Account, messageId: MessageId, loadIfNotExists: Bool) -> Signal<MessageView, NoError> {
return Signal { subscriber in
let loadedMessage = account.postbox.transaction { transaction -> Signal<Void, NoError> in
if transaction.getMessage(messageId) == nil, loadIfNotExists {
return fetchMessage(transaction: transaction, account: account, messageId: messageId)
} else {
return .complete()
}
} |> switchToLatest
let disposable = loadedMessage.start()
let viewDisposable = account.postbox.messageView(messageId).start(next: { view in
subscriber.putNext(view)
})
return ActionDisposable {
disposable.dispose()
viewDisposable.dispose()
}
}
}
private func fetchMessage(transaction: Transaction, account: Account, messageId: MessageId) -> Signal<Void, NoError> {
if let peer = transaction.getPeer(messageId.peerId) {
var signal: Signal<Api.messages.Messages, MTRpcError>?
if messageId.peerId.namespace == Namespaces.Peer.CloudUser || messageId.peerId.namespace == Namespaces.Peer.CloudGroup {
signal = account.network.request(Api.functions.messages.getMessages(id: [Api.InputMessage.inputMessageID(id: messageId.id)]))
} else if messageId.peerId.namespace == Namespaces.Peer.CloudChannel {
if let inputChannel = apiInputChannel(peer) {
signal = account.network.request(Api.functions.channels.getMessages(channel: inputChannel, id: [Api.InputMessage.inputMessageID(id: messageId.id)]))
}
}
if let signal = signal {
return signal
|> `catch` { _ -> Signal<Api.messages.Messages, NoError> in
return .single(.messages(messages: [], chats: [], users: []))
}
|> mapToSignal { result -> Signal<Void, NoError> in
return account.postbox.transaction { transaction -> Void in
let apiMessages: [Api.Message]
let apiChats: [Api.Chat]
let apiUsers: [Api.User]
switch result {
case let .messages(messages, chats, users):
apiMessages = messages
apiChats = chats
apiUsers = users
case let .messagesSlice(_, _, _, messages, chats, users):
apiMessages = messages
apiChats = chats
apiUsers = users
case let .channelMessages(_, _, _, messages, chats, users):
apiMessages = messages
apiChats = chats
apiUsers = users
case .messagesNotModified:
apiMessages = []
apiChats = []
apiUsers = []
}
var peers: [PeerId: Peer] = [:]
for user in apiUsers {
if let user = TelegramUser.merge(transaction.getPeer(user.peerId) as? TelegramUser, rhs: user) {
peers[user.id] = user
}
}
for chat in apiChats {
if let groupOrChannel = parseTelegramGroupOrChannel(chat: chat) {
peers[groupOrChannel.id] = groupOrChannel
}
}
updatePeers(transaction: transaction, peers: Array(peers.values), update: { _, updated in
return updated
})
for message in apiMessages {
if let message = StoreMessage(apiMessage: message) {
let _ = transaction.addMessages([message], location: .Random)
}
}
}
}
} else {
return .complete()
}
} else {
return .complete()
}
}