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
336 lines
17 KiB
Swift
336 lines
17 KiB
Swift
import TextFormat
|
|
import Foundation
|
|
import NaturalLanguage
|
|
import SwiftSignalKit
|
|
import TelegramCore
|
|
import AccountContext
|
|
import TelegramUIPreferences
|
|
|
|
public struct ChatTranslationState: Codable {
|
|
enum CodingKeys: String, CodingKey {
|
|
case baseLang
|
|
case fromLang
|
|
case timestamp
|
|
case toLang
|
|
case isEnabled
|
|
}
|
|
|
|
public let baseLang: String
|
|
public let fromLang: String
|
|
public let timestamp: Int32?
|
|
public let toLang: String?
|
|
public let isEnabled: Bool
|
|
|
|
public init(
|
|
baseLang: String,
|
|
fromLang: String,
|
|
timestamp: Int32?,
|
|
toLang: String?,
|
|
isEnabled: Bool
|
|
) {
|
|
self.baseLang = baseLang
|
|
self.fromLang = fromLang
|
|
self.timestamp = timestamp
|
|
self.toLang = toLang
|
|
self.isEnabled = isEnabled
|
|
}
|
|
|
|
public init(from decoder: Decoder) throws {
|
|
let container = try decoder.container(keyedBy: CodingKeys.self)
|
|
|
|
self.baseLang = try container.decode(String.self, forKey: .baseLang)
|
|
self.fromLang = try container.decode(String.self, forKey: .fromLang)
|
|
self.timestamp = try container.decodeIfPresent(Int32.self, forKey: .timestamp)
|
|
self.toLang = try container.decodeIfPresent(String.self, forKey: .toLang)
|
|
self.isEnabled = try container.decode(Bool.self, forKey: .isEnabled)
|
|
}
|
|
|
|
public func encode(to encoder: Encoder) throws {
|
|
var container = encoder.container(keyedBy: CodingKeys.self)
|
|
|
|
try container.encode(self.baseLang, forKey: .baseLang)
|
|
try container.encode(self.fromLang, forKey: .fromLang)
|
|
try container.encodeIfPresent(self.timestamp, forKey: .timestamp)
|
|
try container.encodeIfPresent(self.toLang, forKey: .toLang)
|
|
try container.encode(self.isEnabled, forKey: .isEnabled)
|
|
}
|
|
|
|
public func withFromLang(_ fromLang: String) -> ChatTranslationState {
|
|
return ChatTranslationState(
|
|
baseLang: self.baseLang,
|
|
fromLang: fromLang,
|
|
timestamp: self.timestamp,
|
|
toLang: self.toLang,
|
|
isEnabled: self.isEnabled
|
|
)
|
|
}
|
|
public func withToLang(_ toLang: String?) -> ChatTranslationState {
|
|
return ChatTranslationState(
|
|
baseLang: self.baseLang,
|
|
fromLang: self.fromLang,
|
|
timestamp: self.timestamp,
|
|
toLang: toLang,
|
|
isEnabled: self.isEnabled
|
|
)
|
|
}
|
|
|
|
public func withIsEnabled(_ isEnabled: Bool) -> ChatTranslationState {
|
|
return ChatTranslationState(
|
|
baseLang: self.baseLang,
|
|
fromLang: self.fromLang,
|
|
timestamp: self.timestamp,
|
|
toLang: self.toLang,
|
|
isEnabled: isEnabled
|
|
)
|
|
}
|
|
}
|
|
|
|
private func cachedChatTranslationState(engine: TelegramEngine, peerId: EnginePeer.Id) -> Signal<ChatTranslationState?, NoError> {
|
|
let key = EngineDataBuffer(length: 8)
|
|
key.setInt64(0, value: peerId.id._internalGetInt64Value())
|
|
|
|
return engine.data.subscribe(TelegramEngine.EngineData.Item.ItemCache.Item(collectionId: ApplicationSpecificItemCacheCollectionId.translationState, id: key))
|
|
|> map { entry -> ChatTranslationState? in
|
|
return entry?.get(ChatTranslationState.self)
|
|
}
|
|
}
|
|
|
|
private func updateChatTranslationState(engine: TelegramEngine, peerId: EnginePeer.Id, state: ChatTranslationState?) -> Signal<Never, NoError> {
|
|
let key = EngineDataBuffer(length: 8)
|
|
key.setInt64(0, value: peerId.id._internalGetInt64Value())
|
|
|
|
if let state {
|
|
return engine.itemCache.put(collectionId: ApplicationSpecificItemCacheCollectionId.translationState, id: key, item: state)
|
|
} else {
|
|
return engine.itemCache.remove(collectionId: ApplicationSpecificItemCacheCollectionId.translationState, id: key)
|
|
}
|
|
}
|
|
|
|
public func updateChatTranslationStateInteractively(engine: TelegramEngine, peerId: EnginePeer.Id, _ f: @escaping (ChatTranslationState?) -> ChatTranslationState?) -> Signal<Never, NoError> {
|
|
let key = EngineDataBuffer(length: 8)
|
|
key.setInt64(0, value: peerId.id._internalGetInt64Value())
|
|
|
|
return engine.data.get(TelegramEngine.EngineData.Item.ItemCache.Item(collectionId: ApplicationSpecificItemCacheCollectionId.translationState, id: key))
|
|
|> map { entry -> ChatTranslationState? in
|
|
return entry?.get(ChatTranslationState.self)
|
|
}
|
|
|> mapToSignal { current -> Signal<Never, NoError> in
|
|
if let current {
|
|
return updateChatTranslationState(engine: engine, peerId: peerId, state: f(current))
|
|
} else {
|
|
return .never()
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
@available(iOS 12.0, *)
|
|
private let languageRecognizer = NLLanguageRecognizer()
|
|
|
|
public func translateMessageIds(context: AccountContext, messageIds: [EngineMessage.Id], fromLang: String?, toLang: String, viaText: Bool = false, forQuickTranslate: Bool = false) -> Signal<Never, NoError> {
|
|
return context.account.postbox.transaction { transaction -> Signal<Never, NoError> in
|
|
var messageDictToTranslate: [EngineMessage.Id: String] = [:]
|
|
var messageIdsToTranslate: [EngineMessage.Id] = []
|
|
var messageIdsSet = Set<EngineMessage.Id>()
|
|
for messageId in messageIds {
|
|
if let message = transaction.getMessage(messageId) {
|
|
if let replyAttribute = message.attributes.first(where: { $0 is ReplyMessageAttribute }) as? ReplyMessageAttribute, let replyMessage = message.associatedMessages[replyAttribute.messageId] {
|
|
if !replyMessage.text.isEmpty {
|
|
if let translation = replyMessage.attributes.first(where: { $0 is TranslationMessageAttribute }) as? TranslationMessageAttribute, translation.toLang == toLang {
|
|
} else {
|
|
if !messageIdsSet.contains(replyMessage.id) {
|
|
messageIdsToTranslate.append(replyMessage.id)
|
|
messageIdsSet.insert(replyMessage.id)
|
|
messageDictToTranslate[replyMessage.id] = replyMessage.text
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// MARK: Swiftgram
|
|
guard forQuickTranslate || message.author?.id != context.account.peerId else {
|
|
continue
|
|
}
|
|
if let translation = message.attributes.first(where: { $0 is TranslationMessageAttribute }) as? TranslationMessageAttribute, translation.toLang == toLang {
|
|
continue
|
|
}
|
|
|
|
if !message.text.isEmpty {
|
|
if !messageIdsSet.contains(messageId) {
|
|
messageIdsToTranslate.append(messageId)
|
|
messageIdsSet.insert(messageId)
|
|
messageDictToTranslate[messageId] = message.text
|
|
}
|
|
// TODO(swiftgram): Translate polls
|
|
} else if let _ = message.media.first(where: { $0 is TelegramMediaPoll }), !viaText {
|
|
if !messageIdsSet.contains(messageId) {
|
|
messageIdsToTranslate.append(messageId)
|
|
messageIdsSet.insert(messageId)
|
|
}
|
|
}
|
|
} else {
|
|
if !messageIdsSet.contains(messageId) {
|
|
messageIdsToTranslate.append(messageId)
|
|
messageIdsSet.insert(messageId)
|
|
}
|
|
}
|
|
}
|
|
if viaText {
|
|
return context.engine.messages.translateMessagesViaText(messagesDict: messageDictToTranslate, fromLang: fromLang, toLang: toLang, generateEntitiesFunction: { text in
|
|
generateTextEntities(text, enabledTypes: .all)
|
|
}, enableLocalIfPossible: context.sharedContext.immediateExperimentalUISettings.enableLocalTranslation)
|
|
|> `catch` { _ -> Signal<Never, NoError> in
|
|
return .complete()
|
|
}
|
|
} else {
|
|
if forQuickTranslate && messageIdsToTranslate.isEmpty { return .complete() } // Otherwise Telegram's API will return .never()
|
|
return context.engine.messages.translateMessages(messageIds: messageIdsToTranslate, fromLang: fromLang, toLang: toLang, enableLocalIfPossible: context.sharedContext.immediateExperimentalUISettings.enableLocalTranslation)
|
|
|> `catch` { _ -> Signal<Never, NoError> in
|
|
return .complete()
|
|
}
|
|
}
|
|
} |> switchToLatest
|
|
}
|
|
|
|
public func chatTranslationState(context: AccountContext, peerId: EnginePeer.Id, forcePredict: Bool = false) -> Signal<ChatTranslationState?, NoError> {
|
|
if peerId.id == EnginePeer.Id.Id._internalFromInt64Value(777000) {
|
|
return .single(nil)
|
|
}
|
|
|
|
let loggingEnabled = context.sharedContext.immediateExperimentalUISettings.logLanguageRecognition
|
|
|
|
if #available(iOS 12.0, *) {
|
|
var baseLang = context.sharedContext.currentPresentationData.with { $0 }.strings.baseLanguageCode
|
|
let rawSuffix = "-raw"
|
|
if baseLang.hasSuffix(rawSuffix) {
|
|
baseLang = String(baseLang.dropLast(rawSuffix.count))
|
|
}
|
|
|
|
return context.sharedContext.accountManager.sharedData(keys: [ApplicationSpecificSharedDataKeys.translationSettings])
|
|
|> mapToSignal { sharedData in
|
|
let settings = sharedData.entries[ApplicationSpecificSharedDataKeys.translationSettings]?.get(TranslationSettings.self) ?? TranslationSettings.defaultSettings
|
|
if !settings.translateChats && !forcePredict {
|
|
return .single(nil)
|
|
}
|
|
|
|
var dontTranslateLanguages = Set<String>()
|
|
if let ignoredLanguages = settings.ignoredLanguages {
|
|
dontTranslateLanguages = Set(ignoredLanguages)
|
|
} else {
|
|
dontTranslateLanguages.insert(baseLang)
|
|
for language in systemLanguageCodes() {
|
|
dontTranslateLanguages.insert(language)
|
|
}
|
|
}
|
|
|
|
return cachedChatTranslationState(engine: context.engine, peerId: peerId)
|
|
|> mapToSignal { cached in
|
|
let currentTime = Int32(CFAbsoluteTimeGetCurrent() + kCFAbsoluteTimeIntervalSince1970)
|
|
if let cached, let timestamp = cached.timestamp, cached.baseLang == baseLang && currentTime - timestamp < 60 * 60 {
|
|
if !dontTranslateLanguages.contains(cached.fromLang) || forcePredict {
|
|
return .single(cached)
|
|
} else {
|
|
return .single(nil)
|
|
}
|
|
} else {
|
|
return .single(nil)
|
|
|> then(
|
|
context.account.viewTracker.aroundMessageHistoryViewForLocation(.peer(peerId: peerId, threadId: nil), index: .upperBound, anchorIndex: .upperBound, count: 32, fixedCombinedReadStates: nil)
|
|
|> filter { messageHistoryView -> Bool in
|
|
return messageHistoryView.0.entries.count > 1
|
|
}
|
|
|> take(1)
|
|
|> map { messageHistoryView, _, _ -> ChatTranslationState? in
|
|
let messages = messageHistoryView.entries.map(\.message)
|
|
|
|
if loggingEnabled {
|
|
Logger.shared.log("ChatTranslation", "Start language recognizing for \(peerId)")
|
|
}
|
|
var fromLangs: [String: Int] = [:]
|
|
var count = 0
|
|
for message in messages {
|
|
if message.effectivelyIncoming(context.account.peerId), message.text.count >= 10 {
|
|
var text = String(message.text.prefix(256))
|
|
if var entities = message.textEntitiesAttribute?.entities.filter({ entity in
|
|
switch entity.type {
|
|
case .Pre, .Code, .Url, .Email, .Mention, .Hashtag, .BotCommand:
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}) {
|
|
entities = entities.sorted(by: { $0.range.lowerBound > $1.range.lowerBound })
|
|
var ranges: [Range<String.Index>] = []
|
|
for entity in entities {
|
|
if entity.range.lowerBound > text.count || entity.range.upperBound > text.count {
|
|
continue
|
|
}
|
|
ranges.append(text.index(text.startIndex, offsetBy: entity.range.lowerBound) ..< text.index(text.startIndex, offsetBy: entity.range.upperBound))
|
|
}
|
|
for range in ranges {
|
|
if range.upperBound < text.endIndex {
|
|
text.removeSubrange(range)
|
|
}
|
|
}
|
|
}
|
|
|
|
if message.text.count < 10 {
|
|
continue
|
|
}
|
|
|
|
languageRecognizer.processString(text)
|
|
let hypotheses = languageRecognizer.languageHypotheses(withMaximum: 4)
|
|
languageRecognizer.reset()
|
|
|
|
let filteredLanguages = hypotheses.filter { supportedTranslationLanguages.contains(normalizeTranslationLanguage($0.key.rawValue)) }.sorted(by: { $0.value > $1.value })
|
|
if let language = filteredLanguages.first {
|
|
let fromLang = normalizeTranslationLanguage(language.key.rawValue)
|
|
if loggingEnabled && !["en", "ru"].contains(fromLang) && !dontTranslateLanguages.contains(fromLang) {
|
|
Logger.shared.log("ChatTranslation", "\(text)")
|
|
Logger.shared.log("ChatTranslation", "Recognized as: \(fromLang), other hypotheses: \(hypotheses.map { $0.key.rawValue }.joined(separator: ",")) ")
|
|
}
|
|
fromLangs[fromLang] = (fromLangs[fromLang] ?? 0) + message.text.count
|
|
count += 1
|
|
}
|
|
}
|
|
if count >= 16 {
|
|
break
|
|
}
|
|
}
|
|
|
|
var mostFrequent: (String, Int)?
|
|
for (lang, count) in fromLangs {
|
|
if let current = mostFrequent {
|
|
if count > current.1 {
|
|
mostFrequent = (lang, count)
|
|
}
|
|
} else {
|
|
mostFrequent = (lang, count)
|
|
}
|
|
}
|
|
let fromLang = mostFrequent?.0 ?? ""
|
|
if loggingEnabled {
|
|
Logger.shared.log("ChatTranslation", "Ended with: \(fromLang)")
|
|
}
|
|
let state = ChatTranslationState(
|
|
baseLang: baseLang,
|
|
fromLang: fromLang,
|
|
timestamp: currentTime,
|
|
toLang: cached?.toLang,
|
|
isEnabled: cached?.isEnabled ?? false
|
|
)
|
|
let _ = updateChatTranslationState(engine: context.engine, peerId: peerId, state: state).start()
|
|
if !dontTranslateLanguages.contains(fromLang) || forcePredict {
|
|
return state
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
)
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
return .single(nil)
|
|
}
|
|
}
|