mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2026-04-08 14:17:54 +00:00
Fixes
fix localeWithStrings globally (#30)
Fix badge on zoomed devices. closes #9
Hide channel bottom panel closes #27
Another attempt to fix badge on some Zoomed devices
Force System Share sheet tg://sg/debug
fixes for device badge
New Crowdin updates (#34)
* New translations sglocalizable.strings (Chinese Traditional)
* New translations sglocalizable.strings (Chinese Simplified)
* New translations sglocalizable.strings (Chinese Traditional)
Fix input panel hidden on selection (#31)
* added if check for selectionState != nil
* same order of subnodes
Revert "Fix input panel hidden on selection (#31)"
This reverts commit e8a8bb1496.
Fix input panel for channels Closes #37
Quickly share links with system's share menu
force tabbar when editing
increase height for correct animation
New translations sglocalizable.strings (Ukrainian) (#38)
Hide Post Story button
Fix 10.15.1
Fix archive option for long-tap
Enable in-app Safari
Disable some unsupported purchases
disableDeleteChatSwipeOption + refactor restart alert
Hide bot in suggestions list
Fix merge v11.0
Fix exceptions for safari webview controller
New Crowdin updates (#47)
* New translations sglocalizable.strings (Romanian)
* New translations sglocalizable.strings (French)
* New translations sglocalizable.strings (Spanish)
* New translations sglocalizable.strings (Afrikaans)
* New translations sglocalizable.strings (Arabic)
* New translations sglocalizable.strings (Catalan)
* New translations sglocalizable.strings (Czech)
* New translations sglocalizable.strings (Danish)
* New translations sglocalizable.strings (German)
* New translations sglocalizable.strings (Greek)
* New translations sglocalizable.strings (Finnish)
* New translations sglocalizable.strings (Hebrew)
* New translations sglocalizable.strings (Hungarian)
* New translations sglocalizable.strings (Italian)
* New translations sglocalizable.strings (Japanese)
* New translations sglocalizable.strings (Korean)
* New translations sglocalizable.strings (Dutch)
* New translations sglocalizable.strings (Norwegian)
* New translations sglocalizable.strings (Polish)
* New translations sglocalizable.strings (Portuguese)
* New translations sglocalizable.strings (Serbian (Cyrillic))
* New translations sglocalizable.strings (Swedish)
* New translations sglocalizable.strings (Turkish)
* New translations sglocalizable.strings (Vietnamese)
* New translations sglocalizable.strings (Indonesian)
* New translations sglocalizable.strings (Hindi)
* New translations sglocalizable.strings (Uzbek)
New Crowdin updates (#49)
* New translations sglocalizable.strings (Arabic)
* New translations sglocalizable.strings (Arabic)
New translations sglocalizable.strings (Russian) (#51)
Call confirmation
WIP Settings search
Settings Search
Localize placeholder
Update AccountUtils.swift
mark mutual contact
Align back context action to left
New Crowdin updates (#54)
* New translations sglocalizable.strings (Chinese Simplified)
* New translations sglocalizable.strings (Chinese Traditional)
* New translations sglocalizable.strings (Ukrainian)
Independent Playground app for simulator
New translations sglocalizable.strings (Ukrainian) (#55)
Playground UIKit base and controllers
Inject SwiftUI view with overflow to AsyncDisplayKit
Launch Playgound project on simulator
Create .swiftformat
Move Playground to example
Update .swiftformat
Init SwiftUIViewController
wip
New translations sglocalizable.strings (Chinese Traditional) (#57)
Xcode 16 fixes
Fix
New translations sglocalizable.strings (Italian) (#59)
New translations sglocalizable.strings (Chinese Simplified) (#63)
Force disable CallKit integration due to missing NSE Entitlement
Fix merge
Fix whole chat translator
Sweetpad config
Bump version
11.3.1 fixes
Mutual contact placement fix
Disable Video PIP swipe
Update versions.json
Fix PIP crash
346 lines
18 KiB
Swift
346 lines
18 KiB
Swift
// MARK: Swiftgram
|
|
import SGSimpleSettings
|
|
import TextFormat
|
|
import TranslateUI
|
|
import Foundation
|
|
import UIKit
|
|
import AsyncDisplayKit
|
|
import ContextUI
|
|
import Display
|
|
import SwiftSignalKit
|
|
import Postbox
|
|
import TelegramCore
|
|
import TelegramNotices
|
|
import ChatSendMessageActionUI
|
|
import AccountContext
|
|
import TopMessageReactions
|
|
import ReactionSelectionNode
|
|
import ChatControllerInteraction
|
|
import ChatSendAudioMessageContextPreview
|
|
|
|
extension ChatSendMessageEffect {
|
|
convenience init(_ effect: ChatSendMessageActionSheetController.SendParameters.Effect) {
|
|
self.init(id: effect.id)
|
|
}
|
|
}
|
|
|
|
func chatMessageDisplaySendMessageOptions(selfController: ChatControllerImpl, node: ASDisplayNode, gesture: ContextGesture) {
|
|
guard let peerId = selfController.chatLocation.peerId, let textInputView = selfController.chatDisplayNode.textInputView(), let layout = selfController.validLayout else {
|
|
return
|
|
}
|
|
let previousSupportedOrientations = selfController.supportedOrientations
|
|
if layout.size.width > layout.size.height {
|
|
selfController.supportedOrientations = ViewControllerSupportedOrientations(regularSize: .all, compactSize: .landscape)
|
|
} else {
|
|
selfController.supportedOrientations = ViewControllerSupportedOrientations(regularSize: .all, compactSize: .portrait)
|
|
}
|
|
|
|
let _ = ApplicationSpecificNotice.incrementChatMessageOptionsTip(accountManager: selfController.context.sharedContext.accountManager, count: 4).startStandalone()
|
|
|
|
var hasEntityKeyboard = false
|
|
if case .media = selfController.presentationInterfaceState.inputMode {
|
|
hasEntityKeyboard = true
|
|
}
|
|
|
|
let effectItems: Signal<[ReactionItem]?, NoError>
|
|
if peerId != selfController.context.account.peerId && peerId.namespace == Namespaces.Peer.CloudUser {
|
|
effectItems = effectMessageReactions(context: selfController.context)
|
|
|> map(Optional.init)
|
|
} else {
|
|
effectItems = .single(nil)
|
|
}
|
|
|
|
let availableMessageEffects = selfController.context.availableMessageEffects |> take(1)
|
|
let hasPremium = selfController.context.engine.data.get(TelegramEngine.EngineData.Item.Peer.Peer(id: selfController.context.account.peerId))
|
|
|> map { peer -> Bool in
|
|
guard case let .user(user) = peer else {
|
|
return false
|
|
}
|
|
return user.isPremium
|
|
}
|
|
|
|
let editMessages: Signal<[EngineMessage], NoError>
|
|
if let editMessage = selfController.presentationInterfaceState.interfaceState.editMessage {
|
|
editMessages = selfController.context.engine.data.get(
|
|
TelegramEngine.EngineData.Item.Messages.MessageGroup(id: editMessage.messageId)
|
|
)
|
|
} else {
|
|
editMessages = .single([])
|
|
}
|
|
|
|
var currentMessageEffect: ChatSendMessageActionSheetControllerSendParameters.Effect?
|
|
if selfController.presentationInterfaceState.interfaceState.editMessage == nil {
|
|
if let sendMessageEffect = selfController.presentationInterfaceState.interfaceState.sendMessageEffect {
|
|
currentMessageEffect = ChatSendMessageActionSheetControllerSendParameters.Effect(id: sendMessageEffect)
|
|
}
|
|
}
|
|
|
|
let _ = (combineLatest(
|
|
selfController.context.account.viewTracker.peerView(peerId) |> take(1),
|
|
effectItems,
|
|
availableMessageEffects,
|
|
hasPremium,
|
|
editMessages,
|
|
ChatSendMessageContextScreen.initialData(context: selfController.context, currentMessageEffectId: currentMessageEffect?.id)
|
|
)
|
|
|> deliverOnMainQueue).startStandalone(next: { [weak selfController] peerView, effectItems, availableMessageEffects, hasPremium, editMessages, initialData in
|
|
guard let selfController, let peer = peerViewMainPeer(peerView) else {
|
|
return
|
|
}
|
|
|
|
// MARK: Swiftgram
|
|
let outgoingMessageTranslateToLang = SGSimpleSettings.shared.outgoingLanguageTranslation[SGSimpleSettings.makeOutgoingLanguageTranslationKey(accountId: selfController.context.account.peerId.id._internalGetInt64Value(), peerId: peer.id.id._internalGetInt64Value())] ?? selfController.predictedChatLanguage
|
|
|
|
let sgTranslationContext: (outgoingMessageTranslateToLang: String?, translate: (() -> Void)?, changeTranslationLanguage: (() -> ())?) = (outgoingMessageTranslateToLang: outgoingMessageTranslateToLang, translate: { [weak selfController] in
|
|
guard let selfController else { return }
|
|
let textToTranslate = selfController.presentationInterfaceState.interfaceState.effectiveInputState.inputText.string
|
|
let textEntities = selfController.presentationInterfaceState.interfaceState.synchronizeableInputState?.entities ?? []
|
|
if let outgoingMessageTranslateToLang = outgoingMessageTranslateToLang {
|
|
let _ = (selfController.context.engine.messages.translate(text: textToTranslate, toLang: outgoingMessageTranslateToLang, entities: textEntities) |> deliverOnMainQueue).start(next: { [weak selfController] translatedTextAndEntities in
|
|
guard let selfController, let translatedTextAndEntities else { return }
|
|
let newInputText = chatInputStateStringWithAppliedEntities(translatedTextAndEntities.0, entities: translatedTextAndEntities.1)
|
|
let newTextInputState = ChatTextInputState(inputText: newInputText, selectionRange: 0 ..< newInputText.length)
|
|
selfController.updateChatPresentationInterfaceState(interactive: true, { state in
|
|
return state.updatedInterfaceState { interfaceState in
|
|
return interfaceState.withUpdatedEffectiveInputState(newTextInputState)
|
|
}
|
|
})
|
|
})
|
|
}
|
|
}, changeTranslationLanguage: { [weak selfController] in
|
|
guard let selfController else { return }
|
|
let controller = languageSelectionController(translateOutgoingMessage: true, context: selfController.context, forceTheme: selfController.presentationData.theme, fromLanguage: "", toLanguage: selfController.presentationInterfaceState.translationState?.fromLang ?? "", completion: { _, toLang in
|
|
guard let peerId = selfController.chatLocation.peerId else {
|
|
return
|
|
}
|
|
var langCode = toLang
|
|
if langCode == "nb" {
|
|
langCode = "no"
|
|
} else if langCode == "pt-br" {
|
|
langCode = "pt"
|
|
}
|
|
|
|
if !toLang.isEmpty {
|
|
SGSimpleSettings.shared.outgoingLanguageTranslation[SGSimpleSettings.makeOutgoingLanguageTranslationKey(accountId: selfController.context.account.peerId.id._internalGetInt64Value(), peerId: peerId.id._internalGetInt64Value())] = langCode
|
|
}
|
|
chatMessageDisplaySendMessageOptions(selfController: selfController, node: node, gesture: gesture)
|
|
})
|
|
controller.navigationPresentation = .modal
|
|
selfController.push(controller)
|
|
})
|
|
|
|
if let editMessage = selfController.presentationInterfaceState.interfaceState.editMessage {
|
|
if editMessages.isEmpty {
|
|
return
|
|
}
|
|
|
|
var mediaPreview: ChatSendMessageContextScreenMediaPreview?
|
|
if editMessages.contains(where: { message in
|
|
return message.media.contains(where: { media in
|
|
if media is TelegramMediaImage {
|
|
return true
|
|
} else if let file = media as? TelegramMediaFile, file.isVideo {
|
|
return true
|
|
} else if media is TelegramMediaPaidContent {
|
|
return true
|
|
}
|
|
return false
|
|
})
|
|
}) {
|
|
mediaPreview = ChatSendGroupMediaMessageContextPreview(
|
|
context: selfController.context,
|
|
presentationData: selfController.presentationData,
|
|
wallpaperBackgroundNode: selfController.chatDisplayNode.backgroundNode,
|
|
messages: editMessages
|
|
)
|
|
}
|
|
|
|
let mediaCaptionIsAbove: Bool
|
|
if let value = editMessage.mediaCaptionIsAbove {
|
|
mediaCaptionIsAbove = value
|
|
} else {
|
|
mediaCaptionIsAbove = editMessages.contains(where: {
|
|
$0.attributes.contains(where: {
|
|
$0 is InvertMediaMessageAttribute
|
|
})
|
|
})
|
|
}
|
|
|
|
let controller = makeChatSendMessageActionSheetController(
|
|
sgTranslationContext: sgTranslationContext,
|
|
initialData: initialData,
|
|
context: selfController.context,
|
|
updatedPresentationData: selfController.updatedPresentationData,
|
|
peerId: selfController.presentationInterfaceState.chatLocation.peerId,
|
|
params: .editMessage(SendMessageActionSheetControllerParams.EditMessage(
|
|
messages: editMessages,
|
|
mediaPreview: mediaPreview,
|
|
mediaCaptionIsAbove: (mediaCaptionIsAbove, { [weak selfController] updatedMediaCaptionIsAbove in
|
|
guard let selfController else {
|
|
return
|
|
}
|
|
selfController.updateChatPresentationInterfaceState(animated: false, interactive: false, { state in
|
|
return state.updatedInterfaceState { interfaceState in
|
|
guard var editMessage = interfaceState.editMessage else {
|
|
return interfaceState
|
|
}
|
|
editMessage.mediaCaptionIsAbove = updatedMediaCaptionIsAbove
|
|
return interfaceState.withUpdatedEditMessage(editMessage)
|
|
}
|
|
})
|
|
})
|
|
)),
|
|
hasEntityKeyboard: hasEntityKeyboard,
|
|
gesture: gesture,
|
|
sourceSendButton: node,
|
|
textInputView: textInputView,
|
|
emojiViewProvider: selfController.chatDisplayNode.textInputPanelNode?.emojiViewProvider,
|
|
wallpaperBackgroundNode: selfController.chatDisplayNode.backgroundNode,
|
|
completion: { [weak selfController] in
|
|
guard let selfController else {
|
|
return
|
|
}
|
|
selfController.supportedOrientations = previousSupportedOrientations
|
|
},
|
|
sendMessage: { [weak selfController] mode, parameters in
|
|
guard let selfController else {
|
|
return
|
|
}
|
|
selfController.interfaceInteraction?.editMessage()
|
|
},
|
|
schedule: { _ in
|
|
},
|
|
editPrice: { _ in
|
|
}, openPremiumPaywall: { [weak selfController] c in
|
|
guard let selfController else {
|
|
return
|
|
}
|
|
selfController.push(c)
|
|
},
|
|
reactionItems: nil,
|
|
availableMessageEffects: nil,
|
|
isPremium: hasPremium
|
|
)
|
|
selfController.sendMessageActionsController = controller
|
|
if layout.isNonExclusive {
|
|
selfController.present(controller, in: .window(.root))
|
|
} else {
|
|
selfController.presentInGlobalOverlay(controller, with: nil)
|
|
}
|
|
} else {
|
|
var sendWhenOnlineAvailable = false
|
|
if let presence = peerView.peerPresences[peer.id] as? TelegramUserPresence, case let .present(until) = presence.status {
|
|
let currentTime = Int32(CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970)
|
|
if currentTime > until {
|
|
sendWhenOnlineAvailable = true
|
|
}
|
|
}
|
|
if peer.id.namespace == Namespaces.Peer.CloudUser && peer.id.id._internalGetInt64Value() == 777000 {
|
|
sendWhenOnlineAvailable = false
|
|
}
|
|
|
|
if sendWhenOnlineAvailable {
|
|
let _ = ApplicationSpecificNotice.incrementSendWhenOnlineTip(accountManager: selfController.context.sharedContext.accountManager, count: 4).startStandalone()
|
|
}
|
|
|
|
var mediaPreview: ChatSendMessageContextScreenMediaPreview?
|
|
if let videoRecorderValue = selfController.videoRecorderValue {
|
|
mediaPreview = videoRecorderValue.makeSendMessageContextPreview()
|
|
}
|
|
if let mediaDraftState = selfController.presentationInterfaceState.interfaceState.mediaDraftState {
|
|
if case let .audio(audio) = mediaDraftState {
|
|
mediaPreview = ChatSendAudioMessageContextPreview(
|
|
context: selfController.context,
|
|
presentationData: selfController.presentationData,
|
|
wallpaperBackgroundNode: selfController.chatDisplayNode.backgroundNode,
|
|
waveform: audio.waveform
|
|
)
|
|
}
|
|
}
|
|
|
|
let controller = makeChatSendMessageActionSheetController(
|
|
sgTranslationContext: sgTranslationContext,
|
|
initialData: initialData,
|
|
context: selfController.context,
|
|
updatedPresentationData: selfController.updatedPresentationData,
|
|
peerId: selfController.presentationInterfaceState.chatLocation.peerId,
|
|
params: .sendMessage(SendMessageActionSheetControllerParams.SendMessage(
|
|
isScheduledMessages: false,
|
|
mediaPreview: mediaPreview,
|
|
mediaCaptionIsAbove: nil,
|
|
messageEffect: (currentMessageEffect, { [weak selfController] updatedEffect in
|
|
guard let selfController else {
|
|
return
|
|
}
|
|
selfController.updateChatPresentationInterfaceState(transition: .immediate, interactive: true, { presentationInterfaceState in
|
|
return presentationInterfaceState.updatedInterfaceState { interfaceState in
|
|
return interfaceState.withUpdatedSendMessageEffect(updatedEffect?.id)
|
|
}
|
|
})
|
|
}),
|
|
attachment: false,
|
|
canSendWhenOnline: sendWhenOnlineAvailable,
|
|
forwardMessageIds: selfController.presentationInterfaceState.interfaceState.forwardMessageIds ?? [],
|
|
canMakePaidContent: false,
|
|
currentPrice: nil,
|
|
hasTimers: false
|
|
)),
|
|
hasEntityKeyboard: hasEntityKeyboard,
|
|
gesture: gesture,
|
|
sourceSendButton: node,
|
|
textInputView: textInputView,
|
|
emojiViewProvider: selfController.chatDisplayNode.textInputPanelNode?.emojiViewProvider,
|
|
wallpaperBackgroundNode: selfController.chatDisplayNode.backgroundNode,
|
|
completion: { [weak selfController] in
|
|
guard let selfController else {
|
|
return
|
|
}
|
|
selfController.supportedOrientations = previousSupportedOrientations
|
|
},
|
|
sendMessage: { [weak selfController] mode, parameters in
|
|
guard let selfController else {
|
|
return
|
|
}
|
|
switch mode {
|
|
case .generic:
|
|
selfController.controllerInteraction?.sendCurrentMessage(false, parameters?.effect.flatMap(ChatSendMessageEffect.init))
|
|
case .silently:
|
|
selfController.controllerInteraction?.sendCurrentMessage(true, parameters?.effect.flatMap(ChatSendMessageEffect.init))
|
|
case .whenOnline:
|
|
selfController.chatDisplayNode.sendCurrentMessage(scheduleTime: scheduleWhenOnlineTimestamp, messageEffect: parameters?.effect.flatMap(ChatSendMessageEffect.init)) { [weak selfController] in
|
|
guard let selfController else {
|
|
return
|
|
}
|
|
selfController.updateChatPresentationInterfaceState(animated: true, interactive: false, saveInterfaceState: selfController.presentationInterfaceState.subject != .scheduledMessages, {
|
|
$0.updatedInterfaceState { $0.withUpdatedReplyMessageSubject(nil).withUpdatedSendMessageEffect(nil).withUpdatedForwardMessageIds(nil).withUpdatedForwardOptionsState(nil).withUpdatedComposeInputState(ChatTextInputState(inputText: NSAttributedString(string: ""))) }
|
|
})
|
|
selfController.openScheduledMessages()
|
|
}
|
|
}
|
|
},
|
|
schedule: { [weak selfController] params in
|
|
guard let selfController else {
|
|
return
|
|
}
|
|
selfController.controllerInteraction?.scheduleCurrentMessage(params)
|
|
}, editPrice: { _ in
|
|
}, openPremiumPaywall: { [weak selfController] c in
|
|
guard let selfController else {
|
|
return
|
|
}
|
|
selfController.push(c)
|
|
},
|
|
reactionItems: (!textInputView.text.isEmpty || mediaPreview != nil) ? effectItems : nil,
|
|
availableMessageEffects: availableMessageEffects,
|
|
isPremium: hasPremium
|
|
)
|
|
selfController.sendMessageActionsController = controller
|
|
if layout.isNonExclusive {
|
|
selfController.present(controller, in: .window(.root))
|
|
} else {
|
|
selfController.presentInGlobalOverlay(controller, with: nil)
|
|
}
|
|
}
|
|
})
|
|
}
|